Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 848108a

Browse files
committedDec 8, 2023
feat: add cjsInterop support without splitting flag
1 parent 8c26e63 commit 848108a

File tree

4 files changed

+135
-6
lines changed

4 files changed

+135
-6
lines changed
 

‎package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@
2020
"url": "https://github.com/egoist/tsup.git"
2121
},
2222
"scripts": {
23-
"dev": "npm run build-fast -- --watch",
23+
"dev": "npm run build-fast -- --sourcemap --watch",
2424
"build": "tsup src/cli-*.ts src/index.ts src/rollup.ts --clean --splitting",
2525
"prepublishOnly": "npm run build",
2626
"test": "npm run build && npm run test-only",

‎src/plugin.ts

+4-1
Original file line numberDiff line numberDiff line change
@@ -125,7 +125,10 @@ export class PluginContainer {
125125
.map((file): ChunkInfo | AssetInfo => {
126126
if (isJS(file.path) || isCSS(file.path)) {
127127
const relativePath = path.relative(process.cwd(), file.path)
128-
const meta = metafile?.outputs[relativePath]
128+
const meta =
129+
metafile?.outputs[relativePath] ||
130+
// esbuild is using "/" as a separator in Windows as well
131+
metafile?.outputs[relativePath.replaceAll(path.sep, path.posix.sep)]
129132
return {
130133
type: 'chunk',
131134
path: file.path,

‎src/plugins/cjs-interop.ts

+73-4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,9 @@
1+
import type { ExportDefaultExpression, ModuleDeclaration } from '@swc/core'
2+
import type { Visitor } from '@swc/core/Visitor'
3+
import fs from 'fs/promises'
4+
import { PrettyError } from '../errors'
15
import { Plugin } from '../plugin'
6+
import { localRequire } from '../utils'
27

38
export const cjsInterop = (): Plugin => {
49
return {
@@ -10,17 +15,81 @@ export const cjsInterop = (): Plugin => {
1015
this.format !== 'cjs' ||
1116
info.type !== 'chunk' ||
1217
!/\.(js|cjs)$/.test(info.path) ||
13-
!info.entryPoint ||
14-
info.exports?.length !== 1 ||
15-
info.exports[0] !== 'default'
18+
!info.entryPoint
1619
) {
1720
return
1821
}
1922

23+
if (this.splitting) {
24+
// there is exports metadata when cjs+splitting is set
25+
if (info.exports?.length !== 1 || info.exports[0] !== 'default') return
26+
} else {
27+
const swc: typeof import('@swc/core') = localRequire('@swc/core')
28+
const { Visitor }: typeof import('@swc/core/Visitor') =
29+
localRequire('@swc/core/Visitor')
30+
if (!swc || !Visitor) {
31+
throw new PrettyError(
32+
`@swc/core is required for cjsInterop when splitting is not enabled. Please install it with \`npm install @swc/core -D\``
33+
)
34+
}
35+
36+
let entrySource: string | undefined
37+
try {
38+
entrySource = await fs.readFile(info.entryPoint!, {
39+
encoding: 'utf8',
40+
})
41+
} catch {}
42+
if (!entrySource) return
43+
44+
const ast = await swc.parse(entrySource, {
45+
syntax: 'typescript',
46+
decorators: true,
47+
tsx: true,
48+
})
49+
const visitor = createExportVisitor(Visitor)
50+
visitor.visitProgram(ast)
51+
52+
if (
53+
!visitor.hasExportDefaultExpression ||
54+
visitor.hasNonDefaultExportDeclaration
55+
)
56+
return
57+
}
58+
2059
return {
21-
code: code + '\nmodule.exports = exports.default;\n',
60+
code: code + '\nmodule.exports=module.exports.default;\n',
2261
map: info.map,
2362
}
2463
},
2564
}
2665
}
66+
67+
function createExportVisitor(VisitorCtor: typeof Visitor) {
68+
class ExportVisitor extends VisitorCtor {
69+
hasNonDefaultExportDeclaration = false
70+
hasExportDefaultExpression = false
71+
constructor() {
72+
super()
73+
type ExtractDeclName<T> = T extends `visit${infer N}` ? N : never
74+
const nonDefaultExportDecls: ExtractDeclName<keyof Visitor>[] = [
75+
'ExportDeclaration', // export const a = {}
76+
'ExportNamedDeclaration', // export {}, export * as a from './a'
77+
'ExportAllDeclaration', // export * from './a'
78+
]
79+
80+
nonDefaultExportDecls.forEach((decl) => {
81+
this[`visit${decl}`] = (n: any) => {
82+
this.hasNonDefaultExportDeclaration = true
83+
return n
84+
}
85+
})
86+
}
87+
visitExportDefaultExpression(
88+
n: ExportDefaultExpression
89+
): ModuleDeclaration {
90+
this.hasExportDefaultExpression = true
91+
return n
92+
}
93+
}
94+
return new ExportVisitor()
95+
}

‎test/index.test.ts

+57
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import fs from 'fs-extra'
55
import glob from 'globby'
66
import waitForExpect from 'wait-for-expect'
77
import { fileURLToPath } from 'url'
8+
import { runInNewContext } from 'vm'
89
import { debouncePromise, slash } from '../src/utils'
910

1011
const __dirname = path.dirname(fileURLToPath(import.meta.url))
@@ -1715,3 +1716,59 @@ test('.d.ts files should be cleaned when --clean and --experimental-dts are prov
17151716
expect(result3.outFiles).not.toContain('bar.d.ts')
17161717
expect(result3.outFiles).not.toContain('bar.js')
17171718
})
1719+
1720+
test('cjsInterop', async () => {
1721+
async function runCjsInteropTest(
1722+
name: string,
1723+
files: Record<string, string>
1724+
) {
1725+
const { output } = await run(`${getTestName()}-${name}`, files, {
1726+
flags: ['--format', 'cjs', '--cjsInterop'],
1727+
})
1728+
const exp = {}
1729+
const mod = { exports: exp }
1730+
runInNewContext(output, { module: mod, exports: exp })
1731+
return mod.exports
1732+
}
1733+
1734+
await expect(
1735+
runCjsInteropTest('simple', {
1736+
'input.ts': `export default { hello: 'world' }`,
1737+
})
1738+
).resolves.toEqual({ hello: 'world' })
1739+
1740+
await expect(
1741+
runCjsInteropTest('non-default', {
1742+
'input.ts': `export const a = { hello: 'world' }`,
1743+
})
1744+
).resolves.toEqual(expect.objectContaining({ a: { hello: 'world' } }))
1745+
1746+
await expect(
1747+
runCjsInteropTest('multiple-export', {
1748+
'input.ts': `
1749+
export const a = 1
1750+
export default { hello: 'world' }
1751+
`,
1752+
})
1753+
).resolves.toEqual(
1754+
expect.objectContaining({ a: 1, default: { hello: 'world' } })
1755+
)
1756+
1757+
await expect(
1758+
runCjsInteropTest('multiple-files', {
1759+
'input.ts': `
1760+
export * as a from './a'
1761+
export default { hello: 'world' }
1762+
`,
1763+
'a.ts': 'export const a = 1',
1764+
})
1765+
).resolves.toEqual(
1766+
expect.objectContaining({ a: { a: 1 }, default: { hello: 'world' } })
1767+
)
1768+
1769+
await expect(
1770+
runCjsInteropTest('no-export', {
1771+
'input.ts': `console.log()`,
1772+
})
1773+
).resolves.toEqual({})
1774+
})

0 commit comments

Comments
 (0)
Please sign in to comment.