-
-
Notifications
You must be signed in to change notification settings - Fork 6.3k
Commit
- Loading branch information
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
import cjs from 'cjs-package' | ||
import { esm, cjsFromEsm } from 'esm-package' | ||
|
||
export default { cjs, esm, cjsFromEsm } |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
module.exports = 'cjs-dev' |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
module.exports = 'cjs-prod' |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"name": "cjs-package", | ||
"version": "0.0.0", | ||
"exports": { | ||
".": { | ||
"development": "./index.dev.js", | ||
"default": "./index.js" | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
import cjs from 'cjs-package' | ||
|
||
export const esm = 'esm-dev' | ||
export const cjsFromEsm = cjs |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
import cjs from 'cjs-package' | ||
|
||
export const esm = 'esm-prod' | ||
export const cjsFromEsm = cjs |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"name": "esm-package", | ||
"version": "0.0.0", | ||
"exports": { | ||
".": { | ||
"development": "./index.dev.mjs", | ||
"default": "./index.mjs" | ||
} | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
{ | ||
"name": "test-ssr-esm", | ||
"private": true, | ||
"version": "0.0.0", | ||
"license": "MIT", | ||
"dependencies": { | ||
"cjs-package": "link:./cjs-package", | ||
"esm-package": "link:./esm-package", | ||
"vite": "link:../../vite" | ||
}, | ||
"scripts": { | ||
"serve": "node server.mjs" | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
import path from 'path' | ||
import { fileURLToPath } from 'url' | ||
|
||
import { install } from 'source-map-support' | ||
install() | ||
|
||
const __dirname = path.dirname(fileURLToPath(import.meta.url)) | ||
|
||
const vite = await import('vite') | ||
const server = await vite.createServer({ | ||
root: path.join(__dirname, 'app'), | ||
// mode: 'production', | ||
ssr: { | ||
external: ['cjs-package', 'esm-package'] | ||
}, | ||
server: { | ||
middlewareMode: 'ssr' | ||
} | ||
}) | ||
|
||
const entryModule = await server.ssrLoadModule('/entry-server.jsx') | ||
console.log(entryModule.default) |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
/** | ||
* @type {import('vite').UserConfig} | ||
*/ | ||
module.exports = { | ||
build: { | ||
minify: false | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,5 @@ | ||
import vm from 'vm' | ||
import fs from 'fs' | ||
import path from 'path' | ||
import { Module } from 'module' | ||
import { ViteDevServer } from '..' | ||
|
@@ -95,9 +97,7 @@ async function instantiateModule( | |
extensions: ['.js', '.mjs', '.ts', '.jsx', '.tsx', '.json'], | ||
isBuild: true, | ||
isProduction, | ||
// Disable "module" condition. | ||
isRequire: true, | ||
mainFields: ['main'], | ||
mainFields: ['main', 'module'], | ||
root | ||
} | ||
|
||
|
@@ -108,7 +108,7 @@ async function instantiateModule( | |
// account for multiple pending deps and duplicate imports. | ||
const pendingDeps: string[] = [] | ||
|
||
const ssrImport = async (dep: string) => { | ||
async function ssrImport(dep: string) { | ||
if (dep[0] !== '.' && dep[0] !== '/') { | ||
return nodeRequire(dep, mod.file, resolveOptions) | ||
} | ||
|
@@ -128,7 +128,7 @@ async function instantiateModule( | |
return moduleGraph.urlToModuleMap.get(dep)?.ssrModule | ||
} | ||
|
||
const ssrDynamicImport = (dep: string) => { | ||
function ssrDynamicImport(dep: string) { | ||
// #3087 dynamic import vars is ignored at rewrite import path, | ||
// so here need process relative path | ||
if (dep[0] === '.') { | ||
|
@@ -152,26 +152,25 @@ async function instantiateModule( | |
} | ||
|
||
const ssrImportMeta = { url } | ||
const ssrArguments = { | ||
global: context.global, | ||
[ssrModuleExportsKey]: ssrModule, | ||
[ssrImportMetaKey]: ssrImportMeta, | ||
[ssrImportKey]: ssrImport, | ||
[ssrDynamicImportKey]: ssrDynamicImport, | ||
[ssrExportAllKey]: ssrExportAll | ||
} | ||
|
||
try { | ||
// eslint-disable-next-line @typescript-eslint/no-empty-function | ||
const AsyncFunction = async function () {}.constructor as typeof Function | ||
const initModule = new AsyncFunction( | ||
`global`, | ||
ssrModuleExportsKey, | ||
ssrImportMetaKey, | ||
ssrImportKey, | ||
ssrDynamicImportKey, | ||
ssrExportAllKey, | ||
result.code + `\n//# sourceURL=${mod.url}` | ||
) | ||
await initModule( | ||
context.global, | ||
ssrModule, | ||
ssrImportMeta, | ||
ssrImport, | ||
ssrDynamicImport, | ||
ssrExportAll | ||
) | ||
const ssrModuleImpl = `(0,async function(${Object.keys(ssrArguments)}){\n${ | ||
result.code | ||
}\n})` | ||
const ssrModuleInit = vm.runInThisContext(ssrModuleImpl, { | ||
This comment has been minimized.
Sorry, something went wrong. |
||
filename: mod.file || mod.url, | ||
columnOffset: 1, | ||
displayErrors: false | ||
}) | ||
await ssrModuleInit(...Object.values(ssrArguments)) | ||
} catch (e) { | ||
const stacktrace = ssrRewriteStacktrace(e.stack, moduleGraph) | ||
rebindErrorStacktrace(e, stacktrace) | ||
|
@@ -188,17 +187,24 @@ async function instantiateModule( | |
return Object.freeze(ssrModule) | ||
} | ||
|
||
function nodeRequire( | ||
async function nodeRequire( | ||
id: string, | ||
importer: string | null, | ||
resolveOptions: InternalResolveOptions | ||
) { | ||
const loadModule = Module.createRequire(importer || resolveOptions.root + '/') | ||
let resolvedId: string | undefined | ||
|
||
// Hook into `require` so that `resolveOptions` are respected. | ||
// Note: ESM-only dependencies don't use this hook at all. | ||
This comment has been minimized.
Sorry, something went wrong.
aleclarson
Author
Member
|
||
const unhookNodeResolve = hookNodeResolve( | ||
(nodeResolve) => (id, parent, isMain, options) => { | ||
if (id[0] === '.' || Module.builtinModules.includes(id)) { | ||
return nodeResolve(id, parent, isMain, options) | ||
} | ||
// No parent exists when an ESM package imports a CJS package. | ||
if (!parent) { | ||
return id | ||
} | ||
const resolved = tryNodeResolve(id, parent.id, resolveOptions, false) | ||
if (!resolved) { | ||
throw Error(`Cannot find module '${id}' imported from '${parent.id}'`) | ||
|
@@ -207,19 +213,14 @@ function nodeRequire( | |
} | ||
) | ||
|
||
let mod: any | ||
try { | ||
mod = loadModule(id) | ||
// Resolve the import manually, to avoid the ESM resolver. | ||
resolvedId = fs.realpathSync.native( | ||
Module.createRequire(importer || resolveOptions.root + '/').resolve(id) | ||
) | ||
// TypeScript transforms dynamic `import` so we must use eval. | ||
return await eval(`import("${resolvedId}")`) | ||
This comment has been minimized.
Sorry, something went wrong.
aleclarson
Author
Member
|
||
} finally { | ||
unhookNodeResolve() | ||
} | ||
|
||
// rollup-style default import interop for cjs | ||
const defaultExport = mod.__esModule ? mod.default : mod | ||
return new Proxy(mod, { | ||
get(mod, prop) { | ||
if (prop === 'default') return defaultExport | ||
return mod[prop] | ||
} | ||
}) | ||
} |
Using
vm
here is not strictly required. We could keep doing theAsyncFunction
hack (see the left side of the diff). But ESM will be supported by thevm
library in the future (seevm.Module
, currently behind a CLI flag). Also, avoidingnew AsyncFunction
is a good idea, sincevm
has better support for breakpoints (though, we'll need to merge #3928 for that).