Skip to content
This repository has been archived by the owner on Sep 2, 2023. It is now read-only.
This repository has been archived by the owner on Sep 2, 2023. It is now read-only.

Flags functionality options #300

Closed
Closed
@GeoffreyBooth

Description

I thought it might be useful to have a post laying out in neutral terms how the various ESM flag proposals work and the implications of each, to help in discussion for choosing the best approach. If anything in this initial post is incorrect or incomplete or biased, I will edit as appropriate.

There are three options discussed so far for how what was previously called --type should behave. I’ll refer to the options by proposed new names based on how the flag would behave. They all take either module or commonjs as the single accepted and required argument.

  1. --input-type would set the type of --eval, --print and STDIN input—and that’s it. If the initial entry point is a file, an error is thrown.

  2. --entry-type would set the type of the initial entry point to the program, whether that be a file or one of the non-file input types (--eval etc.); but setting of the type of that initial entry point implies nothing about any other files.

  3. --package-type would set or override the "type" field of the package.json file that controls the parsing of the initial entry point (or of the virtual package.json at the root of the volume, if there are no package.json files up the path from the entry point). It would also apply to the non-file input types. Like the "type" field, it would not override any package scopes beyond the one containing the initial entry point.

This isn’t meant to be a discussion of names. I don’t feel that it’s a good use of GitHub issue threads to bikeshed what we should name things. Once we decide on which functionality we want, we can separately determine the best name for it. Please consider the names below to be placeholders.

Pros and cons of each option

--input-type

Pros:

  • Provides a way to use ESM in --eval, --print and STDIN, where otherwise ESM syntax would be impossible to enable.

Cons:

  • Users would probably expect this to apply to files as well. It’s not obvious why it wouldn’t.

--entry-type

Pros:

  • Provides a way to use ESM in “loose” .js or extensionless files, that live outside of any project/package and don’t have a parent package.json.

  • Explicitly applies to the file or string being referenced in the node command, so is straightforward in that regard.

Cons:

  • So far, all users who have read about this have assumed that setting the type of the entry also opts in to that type for all files imported by that entry point. As in, if you set the entry point to be ESM, import statements of .js files should treat those .js files also as ESM. This user expectation is likely to only become stronger as ESM in browsers becomes more widely used, as this is how <script type="module"> behaves in browsers.

  • It is inconsistent for entry.js and dep.js to be side by side, where entry.js imports dep.js, and node --entry-type=module entry.js loads entry.js as ESM but then dep.js is treated as CommonJS. This is the only case where files with the same extension in the same folder are treated differently by Node.

  • There’s no use case for overriding the initial entry point of a project while relying on file extensions or package.json to define the type of all other files in the project.

--package-type

Pros:

  • Behaves as users expect the flag to behave, by setting the type for an entire project.

  • By referencing package.json "type" in its name, this should be easier for users to understand as they should grasp that it behaves the same way as package.json "type" does.

  • Allows “loose” .js or extensionless files to import other ESM .js files, so “shell script” .js files don’t need to be limited to a single file.

  • Without this, we can’t have --package-type=auto, as it wouldn’t make sense to have type detection for an entry point only. The use case for auto is a project that lacks an explicit "type" field (and uses .js), and it’s implausible to imagine a project with a .js entry point where all other files are .mjs (or in a subfolder under a package.json with a "type" field).

Cons:

  • Applies to more than just the file or string passed to node, so users would need to be aware that it’s the equivalent to package.json "type".

Both:

  • The only use case for needing this flag for a project is when a project is already using ESM in .js files but without "type": "module"; but most such projects expect Babel or the like to transpile them, and may not be compatible with Node without changes. (For example, they may require refactoring to enable explicit extensions; though --es-module-specifier-resolution=node might be sufficient for most such projects to run without changes.) If we build --package-type=auto, regardless of its effectiveness for Babel ESM projects auto would work great for a CommonJS project without a package.json "type" field.

  • Allows opting into ESM mode by default system-wide via NODE_OPTIONS=--package-type=module. After setting such an option in a user’s environment, CommonJS projects would need to either have a "type": "commonjs" in their package.json or be run via --package-type=commonjs. Allowing changing Node to be ESM by default system-wide would be seen as a pro by some and as a con by others; it’s a pro for those who want to leave CommonJS behind and don’t plan on adopting .mjs; and as a con for those who don’t want to encourage people to expect ESM by default and publish projects and packages that assume so. (The latter concern could presumably be addressed somewhat by npm publish checking for import/export syntax in packages about to be published, and erroring if "type": "module" is not present.)

Other considerations

It is not clear to me how these flags would interact with loaders, test runners, stubs/mocks and the like. I’m not sure if any flags are better or worse than others with regard to such things. On the one hand --input-type or --entry-type would seem to be simpler for such add-ons to handle, as they only apply to one string or file; yet if the add-ons need to know how to handle package.json "type", it might be simpler for them to support that and --package-type (which should behave identically) rather than needing to special-case the entry point.

See also

Activity

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions