Skip to content
This repository was archived by the owner on Jul 31, 2018. It is now read-only.

Commit 86e0241

Browse files
jdaltontrevnorris
authored andcommittedJul 6, 2016
Replace section 5.1 with unambiguous JS grammer
PR-URL: #33 Reviewed-By: CTC
1 parent 5dae5a5 commit 86e0241

File tree

1 file changed

+78
-134
lines changed

1 file changed

+78
-134
lines changed
 

‎002-es6-modules.md

+78-134
Original file line numberDiff line numberDiff line change
@@ -140,8 +140,8 @@ following steps are taken:
140140

141141
#### 3.2.2. `DelegatedModuleNamespaceObjectCreate(module, O)`
142142

143-
The abstract operation DelegatedModuleNamespaceObjectCreate with arguments `O`
144-
is used to create a DelegatedModuleNamespaceObject. It performs the following
143+
The abstract operation `DelegatedModuleNamespaceObjectCreate` with arguments `O`
144+
is used to create a `DelegatedModuleNamespaceObject`. It performs the following
145145
steps:
146146

147147
1. Assert: `module` is a Module Record.
@@ -172,7 +172,7 @@ steps:
172172

173173
### 4.1. NodeModuleEvaluationJob(source, mode)
174174

175-
The abstractjob operation NodeModuleEvaluationJob with parameters `source` and
175+
The abstractjob operation `NodeModuleEvaluationJob` with parameters `source` and
176176
`mode`. The results of this should be placed in the cache that
177177
`HostResolveImportedModule` uses.
178178

@@ -193,110 +193,79 @@ The abstractjob operation NodeModuleEvaluationJob with parameters `source` and
193193

194194
**NOTE:** This still guarantees:
195195

196-
* that ES module dependencies are all executed prior to the module itself
196+
* ES module dependencies are all executed prior to the module itself
197197
* CJS modules have a full shape prior to being handed to ES modules
198-
* allows CJS modules to imperatively start the loading of other modules,
199-
including ES modules
198+
* CJS modules can imperatively start loading other modules, including ES modules
200199

201200
## 5. Semantics
202201

203202
### 5.1. Determining if source is an ES Module
204203

205-
A new file type will be recognised, `.mjs` as ES modules. `.mjs` files will be
206-
treated as having different loading semantics that are compatible with the
207-
existing CJS system, just like `.node`, `.json`, or usage of
208-
`require.extension` (even though deprecated) are compatible. This file type
209-
will be registered with IANA as an official file type, see [TC39 issue]
210-
(https://github.com/tc39/ecma262/issues/322).
211-
212-
The `.mjs` file extension will have a higher loading priority than `.js` for
213-
`require`. This means that, once the Node resolution algorithm reaches file
214-
expansion, the path for `path + '.mjs'` would be attempted prior to `path +
215-
'.js'` when performing `require(path)`.
216-
217-
#### 5.1.1 Reason for decision
218-
219-
The choice of `.mjs` was made due to a number of factors.
220-
221-
* `.jsm`
222-
* conflict with Firefox, which includes escalated
223-
[privileges over the `file://` protocol] that can access
224-
[sensistive information][4]. This could affect things like
225-
[test runners providing browser test viewers]
226-
* [decent usage on npm]
227-
(https://gist.github.com/ChALkeR/9e1fb15d0a0e18589e6dadc34b80875d)
228-
* `.es`
229-
* lacks conflicts with other major software
230-
* removes the JS moniker/signifier in many projects such as Node.js,
231-
Cylon.js, Three.js, DynJS, JSX, ...
232-
* removes JavaScript -> JS acronym association for learning
233-
* is an existing TLD, could be place for squatting / confusion.
234-
* [small usage on npm](https://gist.github.com/ChALkeR/261550d903ec9867bbab)
235-
* `.m.js`
236-
* potential conflict with existing software targeting wrong goal
237-
* allows `*.js` style globbing to work still
238-
* toolchain problems for asset pipelines/Node/editors that only check after
239-
last `.`
240-
* [small usage on npm](https://gist.github.com/ChALkeR/c10642f2531b1be36e5d)
241-
* `.mjs`
242-
* lacks conflicts with other major software, conflicts with
243-
[metascript](https://github.com/metascript/metascript) but that was last
244-
published in 2015
245-
* [small usage on npm]
246-
(https://gist.github.com/bmeck/07a5beb6541c884acbe908df7b28df3f)
247-
248-
249-
#### 5.1.1.1 Inter package loading using file extension breakage.
250-
251-
There is knowledge of breakage for code that *upgrades* inner package
252-
dependencies such as `require('foo/bar.js')`. As `bar.js` may move to
253-
`bar.mjs`. Since `bar.js` is not the listed entry point this was considered
254-
okay. If foo was providing this file explicitly like `lodash` has this can be
255-
mitigated easily by using as proxy module should `foo` choose to provide one:
256-
257-
```js
258-
Object.defineProperty(module, 'exports', {
259-
get() {
260-
return require(__filename.replace(/\.js$/,'.mjs'))
261-
},
262-
configurable:false
263-
});
264-
Object.freeze(module);
265-
```
266-
267-
It is recommended going forward that developers not rely on the file extensions
268-
in packages they do not control.
269-
270-
### 5.1.2. Ecosystem Concerns
271-
272-
Concerns of ecosystem damage when using a new file extension were considered as
273-
well. Firewall rules and server scripts using `*.js` as the detection mechanism
274-
for JavaScript would be affected by our change, largely just affecting
275-
browsers. However, new file extensions and mimes continue to be introduced and
276-
supported by such things. If a front-end is unable to make such a change, using
277-
a rewrite rule from `.mjs` to `.js` should sufficiently mitigate this.
278-
279-
There were proposals of using `package.json` as an out of band configuration as
280-
well.
281-
282-
* removes one-off file execution for quick scripting like using `.save` in the
283-
repl then running.
284-
* causes editors/asset pipelines to have knowledge of this. most work solely on
285-
file extension and it would be prohibitive to change.
286-
* HTTP/2 PUSH solutions would also need this and would be affected
287-
* no direction for complete removal of CJS. A file extension leads to a world
288-
without `.js` and only `.mjs` files. This would be a permanent field in
289-
`package.json`
290-
* per file mode requirements mean using globs or large lists
291-
* discussion of allowing only one mode per package was a non-starter for
292-
migration path issues
293-
* `package.json` is not required in `node_modules/` and raw files can live in
294-
`node_modules/`
295-
* e.g. `node_modules/foo/index.js`
296-
* e.g. `node_modules/foo.js`
297-
* complex abstraction to handle symlinks used by tools like
298-
[link-local](https://github.com/timoxley/linklocal)
299-
* e.g. `node_modules/foo -> ../app/components/foo.js`
204+
Require that Module source text has at least one `import` or `export` declaration.
205+
A module with only an `import` declaration and no `export` declaration is valid.
206+
Modules, that do not export anything, should specify an `export {}` to make
207+
intentions clear and avoid accidental parse errors while removing `import`
208+
declarations. The `export {}` is **not** new syntax and does **not** export an
209+
empty object. It is simply the standard way to specify exporting nothing.
210+
211+
A package opts-in to the Module goal by specifying `"module"` as the parse goal
212+
field *(name not final)* in its `package.json`. Package dependencies are not
213+
affected by the opt-in and may be a mix of CJS and ES module packages. If a parse
214+
goal is not specified, then attempt to parse source text as the preferred goal
215+
*(Script for now since most modules are CJS)*. If there is a parse error that
216+
may allow another goal to parse, then parse as the other goal, and so on. After
217+
this, the goal is known unambiguously and the environment can safely perform
218+
initialization without the possibility of the source text being run in the wrong
219+
goal.
220+
221+
<em>Note: While the ES2015 specification
222+
[does not forbid](http://www.ecma-international.org/ecma-262/6.0/#sec-forbidden-extensions)
223+
this extension, Node wants to avoid acting as a rogue agent. Node has a TC39
224+
representative, [@bmeck](https://github.com/bmeck), to champion this proposal.
225+
A specification change or at least an official endorsement of this Node proposal
226+
would be welcomed. If a resolution is not possible, this proposal will fallback
227+
to the previous [`.mjs` file extension proposal](https://github.com/nodejs/node-eps/blob/5dae5a537c2d56fbaf23aaf2ae9da15e74474021/002-es6-modules.md#51-determining-if-source-is-an-es-module).</em>
228+
229+
#### 5.1.1 Goal Detection
230+
231+
##### Parse (source, goal, throws)
232+
233+
The abstract operation to parse source text as a given goal.
234+
235+
1. Bootstrap `source` for `goal`.
236+
2. Parse `source` as `goal`.
237+
3. If success, return `true`.
238+
4. If `throws`, throw exception.
239+
5. Return `false`.
240+
241+
##### Operation
242+
243+
1. If a package parse goal is specified, then
244+
1. Let `goal` be the resolved parse goal.
245+
2. Call `Parse(source, goal, true)` and return.
246+
247+
2. Else fallback to multiple parse.
248+
1. If `Parse(Source, Script, false)` is `true`, then
249+
1. Return.
250+
2. Else
251+
1. Call `Parse(Source, Module, true)`.
252+
253+
*Note: A host can choose either goal to parse first and may change their order
254+
over time or as new parse goals are introduced. Feel free to swap the order of
255+
Script and Module.*
256+
257+
#### 5.1.2 Implementation
258+
259+
To improve performance, host environments may want to specify a goal to parse
260+
first. This can be done in several ways:<br>
261+
cache on disk, a command line flag, a manifest file, HTTP header, file extension, etc.
262+
263+
#### 5.1.3 Tooling Concerns
264+
265+
Some tools, outside of Node, may not have access to a JS parser *(Bash programs,
266+
some asset pipelines, etc.)*. These tools generally operate on files as opaque
267+
blobs / plain text files and can use the techniques, listed under
268+
[Implementation](#512-implementation), to get parse goal information.
300269

301270
### 5.2. ES Import Path Resolution
302271

@@ -308,55 +277,43 @@ In summary:
308277

309278
```javascript
310279
// looks at
311-
// ./foo.mjs
312280
// ./foo.js
313281
// ./foo/package.json
314-
// ./foo/index.mjs
315282
// ./foo/index.js
316283
// etc.
317284
import './foo';
318285
```
319286

320287
```javascript
321288
// looks at
322-
// /bar.mjs
323289
// /bar.js
324290
// /bar/package.json
325-
// /bar/index.mjs
326291
// /bar/index.js
327292
// etc.
328293
import '/bar';
329294
```
330295

331296
```javascript
332297
// looks at:
333-
// ./node_modules/baz.mjs
334298
// ./node_modules/baz.js
335299
// ./node_modules/baz/package.json
336-
// ./node_modules/baz/index.mjs
337300
// ./node_modules/baz/index.js
338301
// and parent node_modules:
339-
// ../node_modules/baz.mjs
340302
// ../node_modules/baz.js
341303
// ../node_modules/baz/package.json
342-
// ../node_modules/baz/index.mjs
343304
// ../node_modules/baz/index.js
344305
// etc.
345306
import 'baz';
346307
```
347308

348309
```javascript
349310
// looks at:
350-
// ./node_modules/abc/123.mjs
351311
// ./node_modules/abc/123.js
352312
// ./node_modules/abc/123/package.json
353-
// ./node_modules/abc/123/index.mjs
354313
// ./node_modules/abc/123/index.js
355314
// and parent node_modules:
356-
// ../node_modules/abc/123.mjs
357315
// ../node_modules/abc/123.js
358316
// ../node_modules/abc/123/package.json
359-
// ../node_modules/abc/123/index.mjs
360317
// ../node_modules/abc/123/index.js
361318
// etc.
362319
import 'abc/123';
@@ -408,19 +365,6 @@ In the case that an `import` statement is unable to find a module, Node should
408365
make a **best effort** to see if `require` would have found the module and
409366
print out where it was found, if `NODE_PATH` was used, if `HOME` was used, etc.
410367

411-
#### 5.2.3. Shipping both ES and CJS
412-
413-
When a `package.json` main is encountered, file extension searches are used to
414-
provide a means to ship both ES and CJS variants of packages. If we have two
415-
entry points `index.mjs` and `index.js` setting `"main":"./index"` in
416-
`package.json` will make Node pick up either, depending on what is supported.
417-
418-
##### 5.2.3.1. Excluding main
419-
420-
Since `main` in `package.json` is entirely optional even inside of npm
421-
packages, some people may prefer to exclude main entirely in the case of using
422-
`./index` as that is still in the Node module search algorithm.
423-
424368
### 5.3. `this` in ES modules
425369

426370
ES modules will have a `this` value set to `undefined`. This
@@ -456,7 +400,7 @@ module.exports = {
456400
You will grab `module.exports` when performing an ES import.
457401

458402
```javascript
459-
// es.mjs
403+
// es.js
460404

461405
// grabs the namespace
462406
import * as baz from './cjs.js';
@@ -486,7 +430,7 @@ module.exports = null;
486430
You will grab `module.exports` when performing an ES import.
487431

488432
```javascript
489-
// es.mjs
433+
// es.js
490434
import foo from './cjs.js';
491435
// foo = null;
492436

@@ -508,7 +452,7 @@ module.exports = function two() {
508452
You will grab `module.exports` when performing an ES import.
509453

510454
```javascript
511-
// es.mjs
455+
// es.js
512456
import foo from './cjs.js';
513457
foo(); // 2
514458

@@ -530,7 +474,7 @@ module.exports = Promise.resolve(3);
530474
You will grab `module.exports` when performing an ES import.
531475

532476
```javascript
533-
// es.mjs
477+
// es.js
534478
import foo from './cjs.js';
535479
foo.then(console.log); // outputs 3
536480

@@ -551,7 +495,7 @@ the property named `default`.
551495
Given:
552496

553497
```javascript
554-
// es.mjs
498+
// es.js
555499
let foo = {bar:'my-default'};
556500
// note:
557501
// this is a value
@@ -577,7 +521,7 @@ console.log(es_namespace.default);
577521
Given:
578522

579523
```javascript
580-
// es.mjs
524+
// es.js
581525
export let foo = {bar:'my-default'};
582526
export {foo as bar};
583527
export function f() {};
@@ -691,7 +635,7 @@ require('./es');
691635
```
692636

693637
```javascript
694-
// es.mjs
638+
// es.js
695639
import * as ns from './cjs.js';
696640
// ns = ?
697641
import cjs from './cjs.js';
@@ -712,7 +656,7 @@ similar to how you cannot affect a generator while it is running.
712656
This would change the ES module behavior to:
713657

714658
```javascript
715-
// es.mjs
659+
// es.js
716660
import * as ns from './cjs.js';
717661
// throw new EvalError('./cjs is not an ES module and has not finished evaluation');
718662
```

0 commit comments

Comments
 (0)
This repository has been archived.