From f4615916470af0aa61d6a4615730db6b8af9cbe2 Mon Sep 17 00:00:00 2001 From: Daniel Roe Date: Thu, 4 Nov 2021 23:24:06 +0000 Subject: [PATCH] docs: add initial guide to node + es modules (#1694) Co-authored-by: Pooya Parsa --- docs/content/1.getting-started/3.bridge.md | 40 +-- docs/content/1.getting-started/6.migration.md | 40 +-- docs/content/2.concepts/4.esm.md | 245 ++++++++++++++++++ 3 files changed, 249 insertions(+), 76 deletions(-) create mode 100644 docs/content/2.concepts/4.esm.md diff --git a/docs/content/1.getting-started/3.bridge.md b/docs/content/1.getting-started/3.bridge.md index 92ffc9a7ed..979331f70c 100644 --- a/docs/content/1.getting-started/3.bridge.md +++ b/docs/content/1.getting-started/3.bridge.md @@ -135,45 +135,9 @@ If you are using TypeScript, you can edit your `tsconfig.json` to benefit from a If you were previously using `@vue/composition-api` or `@nuxtjs/composition-api`, please read the [composition api migration guide](/getting-started/bridge-composition-api). -## Avoid CommonJS syntax +### Migrate from CommonJS to ESM -Nuxt 3 natively supports TypeScript and ECMAScript Modules. - -### Update the imports - -::code-group - -```js [Before] -const lib = require('lib') -``` - -```js [After] -import lib from 'lib' -// or using code-splitting -const lib = await import('lib').then(e => e.default || e) -``` - -:: - -### Update the exports - -::code-group - -```js [Before] -module.exports = ... -``` - -```js [After] -export default ... -// or using named export -export const hello = ... -``` - -:: - -### Avoid using specific CJS syntax - -Avoid the usage of `__dirname` and `__filename` as much as possible. +Nuxt 3 natively supports TypeScript and ECMAScript Modules. Please check [Native ES Modules](/concepts/esm) for more info and upgrading. ## Remove incompatible and obsolete modules diff --git a/docs/content/1.getting-started/6.migration.md b/docs/content/1.getting-started/6.migration.md index bef8ebc50b..dd36402dd0 100644 --- a/docs/content/1.getting-started/6.migration.md +++ b/docs/content/1.getting-started/6.migration.md @@ -71,45 +71,9 @@ Migrating to `@nuxt/bridge` is the first and most important step for supporting If you have a fixture or example in your module, add `@nuxt/bridge` package to its config (see [example](/getting-started/bridge#update-nuxtconfig)) -### Avoid CommonJS syntax +### Migrate from CommonJS to ESM -Nuxt 3 natively supports TypeScript and ECMAScript Modules. - -#### Update the imports - -::code-group - -```js [Before] -const lib = require('lib') -``` - -```js [After] -import lib from 'lib' -// or using code-splitting -const lib = await import('lib').then(e => e.default || e) -``` - -:: - -#### Update the exports - -::code-group - -```js [Before] -module.exports = ... -``` - -```js [After] -export default ... -// or using named export -export const hello = ... -``` - -:: - -#### Avoid using specific CJS syntax - -Avoid the usage of `__dirname` and `__filename` as much as possible. +Nuxt 3 natively supports TypeScript and ECMAScript Modules. Please check [Native ES Modules](/concepts/esm) for more info and upgrading. ### Ensure plugins default export diff --git a/docs/content/2.concepts/4.esm.md b/docs/content/2.concepts/4.esm.md new file mode 100644 index 0000000000..185428ee8b --- /dev/null +++ b/docs/content/2.concepts/4.esm.md @@ -0,0 +1,245 @@ +# Native ES Modules + +Nuxt 3 (and Bridge) uses Native ES Modules. + +This guide helps explain what ES Modules are and how to make a Nuxt app (or upstream library) compatible with ESM. + +## Background + +### CommonJS Modules + +CommonJS (CJS) is a format introduced by Node.js that allows sharing functionality between isolated JavaScript modules ([read more](https://nodejs.org/api/modules.html)). +You might be already familiar with this syntax: + +```js +const a = require('./a') + +module.exports.a = a +``` + +Bundlers like webpack and Rollup support this syntax and allow you to use modules written in CommonJS in the browser. + +### ESM Syntax + +Most of the time, when people talk about ESM vs CJS, they are talking about a different syntax for writing [modules](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Modules). + +```js +import a from './a' + +export { a } +``` + +Before ECMAScript Modules (ESM) became a standard (it took more than 10 years!), tooling like +[webpack](https://webpack.js.org/guides/ecma-script-modules/) and even languages like TypeScript started supporting so-called **ESM syntax**. +However, there are some key differences with actual spec; here's [a helpful explainer](https://hacks.mozilla.org/2018/03/es-modules-a-cartoon-deep-dive/). + +### What is 'Native' ESM? + +You may have been writing your app using ESM syntax for a long time. After all, it's natively supported by the browser, and in Nuxt 2 we simply compiled all the code you wrote to the appropriate format (CJS for server, ESM for browser). + +When using modules you'd install into your package, things were a little different. A sample library might expose both CJS and ESM versions, and let us pick which one we wanted: + +```json +{ + "name": "sample-library", + "main": "dist/sample-library.cjs.js", + "module": "dist/sample-library.esm.js" +} +``` + +So in Nuxt 2, the bundler (webpack) would simply pull in the CJS file ('main') for the server build, and use the ESM file ('module') for the client build. + +However, in recent Node.js LTS releases, it is now possible to [use native ESM module](https://nodejs.org/api/esm.html) within Node.js. That means that Node.js itself can process JavaScript using ESM syntax, although it doesn't do it by default. The two most common ways to enable ESM syntax are: + +- set `type: 'module'` within your `package.json` and keep using `.js` extension +- use the `.mjs` file extensions (recommended) + +This is what we do for Nuxt Nitro; we output a `.output/server/index.mjs` file. That tells Node.js to treat this file as a native ES module. + +### What are valid imports in a Node.js context? + +When you `import` a module rather than `require` it, Node.js resolves it differently. For example, when you import `sample-library`, Node.js will look not for the `main` but for the `exports` or `module` entry in that library's `package.json`. + +This is also true of dynamic imports, like `const b = await import('sample-library')`. + +Node supports the following kinds of imports (see [docs](https://nodejs.org/api/packages.html#determining-module-system)): + +1. files ending in `.mjs` - these are expected to use ESM syntax +1. files ending in `.cjs` - these are expected to use CJS syntax +1. files ending in `.js` - these are expected to use CJS syntax unless their `package.json` has `type: 'module'` + +### What kinds of problems can there be? + +For a long time module authors have been producing ESM-syntax builds but using conventions like `.esm.js` or `.es.js`, which they have added to the `module` field in their `package.json`. This hasn't been a problem until now because they have only been used by bundlers like webpack, which don't especially care about the file extension. + +However, if you try to import a package with an `.esm.js` file in a Node.js ESM context, it just won't work, and you'll get an error like: + +```bash +(node:22145) Warning: To load an ES module, set "type": "module" in the package.json or use the .mjs extension. +/path/to/index.js:1 + +export default {} +^^^^^^ + +SyntaxError: Unexpected token 'export' + at wrapSafe (internal/modules/cjs/loader.js:1001:16) + at Module._compile (internal/modules/cjs/loader.js:1049:27) + at Object.Module._extensions..js (internal/modules/cjs/loader.js:1114:10) + .... + at async Object.loadESM (internal/process/esm_loader.js:68:5) +``` + +You might also get this error if you have a named import from an ESM-syntax build that Node.js thinks is CJS: + +```bash +file:///path/to/index.mjs:5 +import { named } from 'sample-library' + ^^^^^ +SyntaxError: Named export 'named' not found. The requested module 'sample-library' is a CommonJS module, which may not support all module.exports as named exports. + +CommonJS modules can always be imported via the default export, for example using: + +import pkg from 'sample-library'; +const { named } = pkg; + + at ModuleJob._instantiate (internal/modules/esm/module_job.js:120:21) + at async ModuleJob.run (internal/modules/esm/module_job.js:165:5) + at async Loader.import (internal/modules/esm/loader.js:177:24) + at async Object.loadESM (internal/process/esm_loader.js:68:5) +``` + +## Troubleshooting ESM issues + +If you encounter these errors, the issue is almost certainly with the upstream library. They need to [fix their library](#library-author-guide) to support being imported by Node. + +### Transpiling libraries + +In the mean-time, you can tell Nuxt not to try to import these libraries by adding them to `build.transpile`: + +```js +import { defineNuxtConfig } from 'nuxt3' + +export default defineNuxtConfig({ + build: { + transpile: ['sample-library'] + } +}) +``` + +You may find that you _also_ need to add other packages that are being imported by these libraries. + +### Aliasing libraries + +In some cases, you may also need to manually alias the library to the CJS version, for example: + +```js +import { defineNuxtConfig } from 'nuxt3' + +export default defineNuxtConfig({ + alias: { + 'sample-library': ['sample-library/dist/sample-library.cjs.js'] + } +}) +``` + +## Library author guide + +The good news is that it's relatively simple to fix issues of ESM compatibility. There are really two main options: + +1. **You can rename your ESM files to end with `.mjs`.** + + _This is the recommended and simplest approach._ You may have to sort out issues with your library's dependencies, and possibly with your build system, but in most cases this should fix the problem for you. It's also recommended to rename your CJS files to end with `.cjs`, for the greatest explicitness. + +1. **You can opt to make your entire library ESM-only**. + + This would mean setting `type: 'module'` in your `package.json` and ensuring that your built library uses ESM syntax. However, you may face issues with your dependencies - and this approach means your library can _only_ be consumed in an ESM context. + +### Migration + +The initial step from CJS to ESM is updating any usage of `require` to use `import` instead: + +::code-group + +```js [Before] +module.exports = ... + +exports.hello = ... +``` + +```js [After] +export default ... + +export const hello = ... +``` + +:: + +::code-group + +```js [Before] +const myLib = require('my-lib') +``` + +```js [After] +import myLib from 'my-lib' +// or +const myLib = await import('my-lib').then(lib => lib.default || lib) +``` + +:: + +In ESM Modules, unlike CJS, `require`, `require.resolve`, `__filename` and `__dirname` globals are not available +and should be replaced with `import()` and `import.meta.filename`. + +You can use [createCommonJS](https://github.com/unjs/mlly#createcommonjs) from `unjs/mlly` to create a CJS compatible context in ESM (or use an inline shim): +::code-group + +```js [mlly] +import { createCommonJS } from 'mlly' + +const { __dirname, __filename, require } = createCommonJS(import.meta.url) +``` + +```js [manual] +import { fileURLToPath } from 'url' +import { dirname } from 'path' +import { createRequire } from 'module' + +const __filename = fileURLToPath(import.meta.url) +const __dirname = dirname(__filename) +const require = createRequire(import.meta.url) +``` + +:: + +::code-group + +```js [Before] +const someFile = require.resolve('./lib/foo.js') +``` + +```js [After] +import { resolvePath } from 'mlly' + +const someFile = await resolvePath('my-lib', { url: import.meta.url }) +``` + +:: + +### Best practices + +- Prefer named exports rather than default export. This helps reduce CJS conflicts. + +- Avoid depending on Node.js built-ins and CommonJS or Node.js-only dependencies as much as possible to make your library usable in Browsers and Edge Workers without needing Nitro polyfills. + +- Use new `exports` field with conditional exports. ([read more](https://nodejs.org/api/packages.html#conditional-exports)). + +```json +{ + "exports": { + ".": { + "import": "./dist/mymodule.mjs" + } + } +} +```