Node executor including ESM module resolution for buildable libraries within Nx workspaces.
Nx allows you to easily add structure to your workspace.
For JavaScript/TypeScript projects this includes a clean structure, DX and tooling for developing multiple applications and libraries:
- Build. When compiling code that references
buildable libraries
within the workspace,
@nx/js:tsc
automatically generates temporarytsconfig
's that link the code being compiled and the libraries referenced (imported/required). - Execute. When running code during development (within the workspace prior
to deployment/release/publication), the
@nx/js:node
executor includes support forrequire
ing CommonJS code from libraries referenced from other buildable libraries within the workspace.
Warning
However, this currently works for CommonJS applications
("type": "commonjs"
in package.json
) but not for
ESM applications ("type": "module"
).
Errors look something like this:
Error: Cannot find package '@my-scope/my-lib' imported from /Users/daniel/projects/test/dist/apps/test-app/src/lib/test-app.js
Nx will probably address this in the future - it's discussed at least here, and here - but for today this plugin seeks to provide a close to drop-in replacememt.
The nx-node-esm-plugin
plugin includes a sample application and a preset
to enable you to try it easily:
View demo screencast.
# Create Nx workspace with preset sample app
npx create-nx-workspace test --preset=@harves/nx-node-esm-plugin
# Take a look at the sampleapp
cd test
# Run the standard Node executor (fails)
npx nx run test-app:run-js-node
# Run using the nx-node-esm-plugin executor
npx nx run test-app:run-node-esm-plugin
# Add plugin
npx nx add @harves/nx-node-esm-plugin
# Use code generator to add the sample library + application
npx nx g @harves/nx-node-esm-plugin:sample-app test
# View sample app project details
npx nx show project test-app --web
# Run the standard Node executor (fails)
npx nx run test-app:run-js-node
# Run using the nx-node-esm-plugin executor
npx nx run test-app:run-node-esm-plugin
A quickstart guide to using the node executor in this plugin:
From this project.json
:
"targets": {
"serve": {
"executor": "@nx/js:node",
"dependsOn": ["build"],
"options": {
"buildTarget": "build",
"watch": false
}
}
}
To this:
"targets": {
"serve": {
"executor": "@harves/nx-node-esm-plugin:node",
"dependsOn": ["build"],
"options": {
"buildTarget": "build"
}
}
}
(*) The nx-node-esm-plugin:node
executor in this plugin does not support file watch mode.
The resolver and loader function provided with this plugin includes support for module resolution for ESM and require support for CommonJS (equivalent to the existing Nx Node executor).
This plugin requires at least Node v18.19.0, when the
module.register()
API used to customise module resolution for ESM was added.
This plugin resolves Node ESM import specifiers for buildable Nx libraries by mapping the library specifiers to the library's built output paths.
It also includes support for configurable overrides.
All other specifiers are deferred to the parent resolver.
The resolution algorithm works as follows (based on the Node specification):
-
Assumes that the mapped Nx library output paths contain a valid
package.json
. -
Attempts to resolve the library imports for mapped libraries using the Node's ESM Module resolution algorithm, with the assumption that all Nx library specifiers are "bare specifiers".
-
Attempts to resolve using the
package.json
exports
property using theresolve-pkg-maps
package. -
Falls back to attempting resolution using the
package.json
main
property using Node's legacy CommonJS resolution algorithm.
The legacy CommonJS main resolution is as follows:
- let M = pkg_url + (json main field)
- TRY(M, M.js, M.json, M.node)
- TRY(M/index.js, M/index.json, M/index.node)
- TRY(pkg_url/index.js, pkg_url/index.json, pkg_url/index.node)
- NOT_FOUND
References:
- Node's ESM Module Resolution algorithm
- Node's legacy CommonJS main resolution code (here and here)
- The resolve-pkg-maps package
This plugin resolves Node CommonJS require's for buildable Nx libraries by mapping the library require request to the library's built output paths.
It also includes support for configurable overrides.
All other request values are deferred to the parent loader.
The @harves/nx-node-esm-plugin:node
executor in this plugin functions
very similarly to the @nx/js:node
that is provided by Nx.
The basic operation of the executor:
-
Utilise the provided
buildTarget
and create a task graph (**) from which the build dependencies may be computed. -
For these dependencies, assemble mappings from the library specifiers (e.g.
@my-scope/my-lib
) to the build output directory (e.g.dist/libs/my-lib
). -
Invoke the selected Node file with module resolution and require loaders customised to utilise these mappings.
Use the --verbose
flag or NX_VERBOSE_LOGGING
environment variable when
running Nx to see logs of the mappings and the module resolution / loader activity.
(**) The task graph algorithm is based on a more recent Nx changes
enabled by the NX_BUILDABLE_LIBRARIES_TASK_GRAPH
environment variable
(not required to be set for this plugin) - see
here,
here, and
here.
The full executor schema may be found in libs/nx-node-esm-plugin/src/executors/node/schema.json.
The available options
are as follows:
string
The target to run to build you the app.object
Additional options to pass into the build target.string; enum ("packageJson", "fromBuildTarget", "specified")
Mode specifying how the Node file to run is determined; either from the package.json of the build target, inferred from the build target or specified explicitly.string
Optional specification of the Node file to run (if present this setting overridesfileToRunMode
).
array [string]
Arguments passed to the Node script.object
Optional ESM module resolution overrides; key is the specifier, value if the full absolute path of the file to load. e.g.{
"moduleResolutionOverrides": {
"@my-scope/my-lib": "/Users/daniel/project-name/libs/some-dir"
}
}
boolean (default false)
Use colors when showing output of command.string
Current working directory of the commands. If it's not specified the commands will run in the workspace root, if a relative path is specified the commands will run in that path relative to the workspace root and if it's an absolute path the commands will run in that path.object
Environment variables that will be made available to the commands. This property has priority over the.env
files.
{
"env": {
"SOME_ENV_VAR": "true"
}
}
string
You may specify a custom .env file path.array [string]
Additional arguments added toargs
and passed to the Node script. Allows command line args to be passed to the script by the executor.
Learn about the latest changes.
Read about contributing to this project. Please report issues on GitHub.
✨ This workspace has been generated by Nx, Smart Monorepos · Fast CI. ✨
Rather than recreating from scratch, much of the code in this plugin is taken from Nx. Links to the relevant Code in the Nx GitHub repo are included.