• Stars
    star
    129
  • Rank 278,253 (Top 6 %)
  • Language
  • License
    MIT License
  • Created almost 6 years ago
  • Updated over 3 years ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Proposal for Bare Module Specifier Resolution in node.js

Bare Module Specifier Resolution in node.js

Contributors: Guy Bedford, Geoffrey Booth, Jan Krems, Saleh Abdel Motaal

Motivating Examples

  • A package (react-dom) has a dedicated entrypoint react-dom/server for code that isn't compatible with a browser environment.
  • A package (angular) exposes multiple independent APIs, modeled via import paths like angular/common/http.
  • A package (lodash) allows to import individual functions, e.g. lodash/map.
  • A package is exclusively exposing an ESM interface.
  • A package is exclusively exposing a CJS interface.
  • A package is exposing both an ESM and a CJS interface.
  • A project wants to mix both ESM and CJS code, with CJS running as part of the ESM module graph.
  • A package wants to expose multiple entrypoints as its public API without leaking internal directory structure.
  • A package wants to reference an internally aliased subpath, without exposing it publicly.
  • A package wants to polyfill a builtin module and handle this through a package fallback mechanism like import maps.

High Level Considerations

  • The baseline behavior of relative imports should match a browser's with a simple file server. This implies that ./x will only ever import exactly the sibling file "x" without appending paths or extensions. "x" is never resolved to x.mjs or x/index.mjs (or the .js equivalents).
  • The primary compatibility boundary are bare specifiers. Relative and absolute imports can follow simpler rules.
  • Resolution should not depend on file extensions, allowing ESM syntax in .js files.
  • The directory structure of a module should be treated as private implementation detail.
  • Validations should apply similarly to import maps in supporting forwards-compatibility with possible future features.

package.json Interfaces

We propose two fields in package.json to specify entrypoints and internal aliasing of bare specifiers - "exports" and "imports".

For both fields the final names of "exports" and "imports" are still TBD, and these names should be considered placeholders.

These interfaces will only be respected for bare specifiers, e.g. import _ from 'lodash' where the specifier 'lodash' doesn’t start with a . or /.

Package exports and imports can be supported fully independently, and in both CommonJS and ES modules.

1. Exports Field

Example

Here’s a complete package.json example, for a hypothetical module named @momentjs/moment:

{
  "name": "@momentjs/moment",
  "version": "0.0.0",
  "type": "module",
  "main": "./dist/index.js",
  "exports": {
    "./": "./src/util/",
    "./timezones/": "./data/timezones/",
    "./timezones/utc": "./data/timezones/utc/index.mjs",
    "./core-polyfill": ["std:core-module", "./core-polyfill.js"]
  }
}

Within the "exports" object, the key string after the '.' is concatenated on the end of the name field, e.g. import utc from '@momentjs/moment/timezones/utc' is formed from '@momentjs/moment' + '/timezones/utc'. Note that this is string manipulation, not a file path: "./timezones/utc" is allowed, but just "timezones/utc" is not.

Keys that end in slashes can map to folder roots, following the pattern in the browser import maps proposal: "./timezones/": "./data/timezones/" would allow import pdt from "@momentjs/moment/timezones/pdt.mjs" to import ./data/timezones/pdt.mjs.

  • Using "./" maps the root, so "./": "./src/util/" would allow import tick from "@momentjs/moment/tick.mjs" to import ./src/util/tick.mjs.

  • Mapping a key of "./" to a value of "./" exposes all files in the package, where "./": "./" would allow import privateHelpers from "@momentjs/moment/private-helpers.mjs" to import ./private-helpers.mjs.

  • When mapping to a folder root, both the left and right sides must end in slashes: "./": "./dist/", not ".": "./dist".

  • Unlike in CommonJS, there is no automatic searching for index.js or index.mjs or for file extensions. This matches the behavior of the import maps proposal.

The value of an export, e.g. "./src/moment.mjs", must begin with . to signify a relative path (e.g. "./src" is okay, but "/src" or "src" are not). This is to reserve potential future use for "exports" to export things referenced via specifiers that aren’t relatively-resolved files, such as other packages or other protocols.

There is the potential for collisions in the exports, such as "./timezones/" and "./timezones/utc" in the example above (e.g. if there’s a file named utc in the ./data/timezones folder). Rough outline of a possible resolution algorithm:

  1. Find the package matching the base specifier, e.g. @momentjs/moment or request.
  2. Load its exports map.
  3. If there is an exact match for the requested specifier, return the resolution.
  4. Otherwise, find the longest matching path prefix. If there is a path prefix, return the resolution by applying the prefix.
  5. Return an error - no mapping found.

In the future, the algorithm might be adjusted to align with work done in the import maps proposal.

Validations and Fallbacks

The following validations are performed for an exports start to resolve:

  • The exports target must be a string and start with "./" (URLs and absolute paths are not currently supported but may be supported in future).
  • The exports target cannot backtrack below the package base.
  • Exports targets cannot map into a nested node_modules path.

For directory resolutions the following validations also apply:

  • Directory exports targets must end in a trailing "/".
  • Directory exports targets may not backtrack above the package base.
  • Directory exports subpaths may not backtrack above the target folder.

Whenever there is a validation failure, any exports match must throw a Module Not Found error, and any validation failure context can be included in the error message.

Fallback arrays allow validation failures to continue checking the next item in the fallback array providing forwards compatiblitiy for new features in future based on extending these validation rules to new cases.

Usage

For a consumer, the above @momentjs/moment and request packages can be used as follows, assuming the user’s project is in /app with /app/package.json and /app/node_modules:

import request from 'request';
// Loads file:///app/node_modules/request/request.mjs

import request from './node_modules/request/request.mjs';
// Loads file:///app/node_modules/request/request.mjs

import request from 'file:///app/node_modules/request/request.mjs';
// Loads file:///app/node_modules/request/request.mjs

import utc from '@momentjs/moment/timezones/utc';
// Loads file:///app/node_modules/@momentjs/moment/timezones/utc/index.mjs

The following don’t work - please note that error messages and codes are TBD:

import request from 'request/';
// Error: trailing slash not mapped

import request from 'request/request.mjs';
// Error: no such mapping

import moment from '@momentjs/moment/';
// Error: trailing slash not allowed (cannot import folders, only files)

import request from 'file:///app/node_modules/request';
// Error: folders cannot be imported, package.json only considered for bare imports

import request from 'file:///app/node_modules/request/';
// Error: folders cannot be imported, package.json only considered for bare imports

import utc from '@momentjs/moment/timezones/utc/'; // Note trailing slash
// Error: folders cannot be imported (there is no index.* magic)

2. Conditional Mapping

Conditional mapping is an extension of "exports" that allows defining different mappings between different environments, for example having a different module path between Node.js and the browser.

Conditional mappings are defined as objects in the target slot for "exports".

The object has keys which are condition names, and values which correspond to the mapping target.

In Node.js, the following condition names are matched in priority order:

  1. "require": Indicates we are resolving from a CommonJS importer.
  2. "node": Indicates we are in a Node.js environment.
  3. "default": The generic fallback.

Note: Using a "require" condition opens up the dual specifier hazard in Node.js where a package can have different instances between CJS and ESM importers. There is an argument that this condition is an opt-in behaviour to the hazard which is less risky than the main concerns of the hazard which were non-intentional cases. It is still not clear if this condition will get consensus, and it may still be removed.

The first key of the above in a conditional object will be matched. If the target fails the next matching condition is matched. Because the target is itself a target, nesting is naturally supported for more complex condition compositions.

Other resolvers are free to define their own conditions to match. Eg it is expected that users will use a "browser" condition name for browser mappings.

Example

Taking the previous moment example we can provide browser / Node.js mappings for some modules with the following:

{
  "name": "@momentjs/moment",
  "version": "0.0.0",
  "type": "module",
  "main": "./dist/index.js",
  "exports": {
    ".": {
      "node": "./dist/index.js",
      "browser": "./dist/index-browser.js"
    },
    "./": {
      "node": "./src/util/",
      "browser": "./src/util-browser/"
    },
    "./timezones/": "./data/timezones/",
    "./timezones/utc": "./data/timezones/utc/index.mjs",
    "./core-polyfill": {
      "node": ["std:core-module", "./core-polyfill.js"],
      "browser": "./core-polyfill-browser.js"
    }
  }
}

Note we are able to share some exports with both Node.js and the browser and others split between the environments.

Dual package example

For an example of a package that wants to support legacy Node.js, require() and import (noting that this is an opt-in to the instancing hazard, and pending consensus), we can use the "require" condition which will always beat the "node" condition as it is a more specific condition:

{
  "type": "module"
  "main": "./index-legacy.cjs",
  "exports": {
    ".": {
      "require": "./index-legacy.cjs",
      "default": "./index.js"
    },
    "./features/": {
      "require": "./features-cjs/",
      "default": "./features/"
    }
  }
}

Combined dual package browser example

To show how conditions handle combined scenarios, here is another example of a package that supports require(), import() and a separate browser entry for both require() and import():

{
  "type": "module",
  "main": "./index.cjs",
  "exports": {
    ".": {
      "browser": {
        "require": "./index-browser.cjs",
        "default": "./index-browser.js"
      },
      "default": {
        "require": "./index.cjs",
        "default": "./index.js"
      }
    }
  }
}

In Node.js the "browser" condition is skipped, hitting the "require" or "default" path depending on if resolution is from CommonJS or an ES module importer.

For browser tools, they can match the appropriate browser index.

Similarly, the above could apply to any exports as well.

Configuring conditions

It can be possible for Node.js to accept a new flag for setting custom conditions and their priorities.

For example via a --env top-level flag and option to Node.js as a comma-separated list of custom condition names to apply before in the matching priority order.

This would support eg:

{
  "exports": {
    ".": {
      "production": "./index-production.js",
      "react-native": "./index-react-native.js",
      "default": "./index-dev.js"
    }
  }
}

where node --env=production,react-native would match the "production" condition first followed by the "react-native" condition and then the standard remaining conditions in priority order.

This way, just like the existing userland process.env.NODE_ENV=production convention, Node.js doesn't need to explicitly natively support the "production" condition but it can exist by convention on its own.

In addition this customization allows environments like Electron and React Native that build on top of Node.js to supplement the condition matching priority system with their own environment conditions.

3. Imports Field

Imports provide the ability to remap bare specifiers within packages before they hit the node_modules resolution process.

The current proposal prefixes all imports with # to provide a clear signal that it's a symbolic specifier and also to prevent packages that use imports from working in any environment (runtime, bundler) that isn't aware of imports.

Whether this restriction is maintained in the final proposal, or what exact symbol is used for # is still TBD.

Example

For the same example package as provided for "exports", consider if we wanted to make the timezones implementation something that is only referenced internally by code within @momentjs/moment, instead of exposing it to external importers.

{
  "name": "@momentjs/moment",
  "version": "0.0.0",
  "type": "module",
  "main": "./dist/index.js",
  "imports": {
    "#timezones/": "./data/timezones/",
    "#timezones/utc": "./data/timezones/utc/index.mjs",
    "#external-feature": "external-pkg/feature",
    "#moment/": "./"
  }
}

As with package exports, mappings are mapped relative to the package base, and keys that end in slashes can map to folder roots.

The resolution algorithms remain the same except "imports" provide the added feature that they can also map into third-party packages that would be looked up in node_modules, including to subpaths that would be in turn resolved through "exports". There is no risk of circular resolution here, since "exports" themselves only ever resolve to direct internal paths and can't in turn map to aliases.

The "imports" that apply within a given file are determined based on looking up the package boundary of that file.

Usage

For the author of @momentjs/moment, they can use these aliases with the following code in any file in @momentjs/moment/*.js, provided it matches the package boundary of the package.json file:

import utc from '#timezones/utc';
// Loads file:///app/node_modules/@momentjs/moment/data/timezones/utc/index.mjs

import utc from './data/timezones/utc/index.mjs';
// Loads file:///app/node_modules/@momentjs/moment/data/timezones/utc/index.mjs

import utc from '#timezones/utc/index.mjs';
// Loads file:///app/node_modules/@momentjs/moment/data/timezones/utc/index.mjs

import utc from '#moment/data/timezones/utc/index.mjs';
// Loads file:///app/node_modules/@momentjs/moment/data/timezones/utc/index.mjs

The following don’t work - please note that error messages and codes are TBD:

import utc from '#timezones/utc/';
// Error: trailing slash not mapped

import unknown from '#unknown';
// Error: no such import alias

import timezones from '#timezones/';
// Error: trailing slash not allowed (cannot import folders, only files)

import utc from '#moment';
// Error: no mapping provided (folder mappings require subpaths)

Prior Art

More Repositories

1

babel-plugin-transform-decorators-stage-2-initial

Babel plugin for transforming the initial stage 2 draft of ES decorators
JavaScript
24
star
2

babel-tap

tap with babel and promise support
JavaScript
8
star
3

import

Hack to expose V8's module API in node
C++
5
star
4

loader

Highly Experimental Loading of ESM From Userland
JavaScript
4
star
5

embracejs

Embracing JS with MJS
JavaScript
3
star
6

zoidberg

Go
2
star
7

nhpp

Node Hypertext PreProcessor
JavaScript
2
star
8

begin-apollo-graphql-api

Begin app
JavaScript
2
star
9

singleton-issue

JavaScript
2
star
10

jkrems.github.io

HTML
1
star
11

server-side-ecma-script

Almost a node.js fork
JavaScript
1
star
12

testium2-demo

A small demo for testium2
JavaScript
1
star
13

spidernode

Like nodejs but with spidermonkey and ES6
JavaScript
1
star
14

test

Experimental test runner using V8 APIs
JavaScript
1
star
15

srv-gofer

Experimenting with gofer and SRV record support
JavaScript
1
star
16

debugging

JavaScript
1
star
17

react-timeline-list

A list of events on a timeline
JavaScript
1
star
18

jpk

an npm client
JavaScript
1
star
19

testium2

Trying another testium
CoffeeScript
1
star
20

expect-match

Evaluating a throwing version of String.prototype.match
JavaScript
1
star
21

pig

Like grunt but cute
JavaScript
1
star
22

resolve-deep

Resolve all promises in a structure
JavaScript
1
star
23

loading

A super sketchy bootstrapping "benchmark"
JavaScript
1
star
24

footnote

Annotation utilities for JavaScript
JavaScript
1
star
25

sdacbogp

Slightly Different Automated CHANGELOG Based On Github PRs
1
star
26

host-pattern

See: https://github.com/groupon/host-pattern
JavaScript
1
star
27

progressive-fetching

Discussions around efficient code organization and loading in web apps
TypeScript
1
star
28

zoidtypes

Types for Zoidberg
JavaScript
1
star
29

pinky-test

Promise-first test runner
JavaScript
1
star
30

gofer-hub

A replacement for gofer/hub
CoffeeScript
1
star
31

now-chunk-list

JavaScript
1
star
32

zoidberg.js

Zoidberg playground
JavaScript
1
star
33

maybe-modules

Checking what kind of module-y things work
HTML
1
star
34

zbc

The elegance of C++ with the raw speed of V8
JavaScript
1
star
35

hmr

Experiments with HMR and modules
JavaScript
1
star
36

webpack-dynamic-import

Show dynamic import bug
JavaScript
1
star