Description
π Search Terms
require(esm), module.exports export, node 22.12
β Viability Checklist
- This wouldn't be a breaking change in existing TypeScript/JavaScript code
- This wouldn't change the runtime behavior of existing JavaScript code
- This could be implemented without emitting different JS based on the types of the expressions
- This isn't a runtime feature (e.g. library functionality, non-ECMAScript syntax with JavaScript output, new syntax sugar for JS, etc.)
- This isn't a request to add a new utility type: https://github.com/microsoft/TypeScript/wiki/No-New-Utility-Types
- This feature would agree with the rest of our Design Goals: https://github.com/Microsoft/TypeScript/wiki/TypeScript-Design-Goals
β Suggestion
Node 20.19.0, 22.12.0, 23.0.0 added requiring ESM files (known as require(esm)
). This feature includes module.exports
interop export name feature that allows users to customize what the module.exports
value would be when a module is require
d.
(nodejs/node#53848)
Currently TypeScript does not use the module.exports
export type when requiring an ESM file.
I'd like TypeScript to use the module.exports
export type for the types when that ESM file is require
d, aligning the type with the actual runtime object.
π Motivating Example
I made a reproduction: https://github.com/sapphi-red-repros/typescript-require-esm-module-exports-export
You can see how the types and the actual runtime values are different by the following steps
- Run
pnpm i
- (Run
pnpm build
) (the built file is already included in the repo) - Run
node index.cjs
- See types in
index.cts
The concrete differences are:
testPkg['module.exports'
isundefined
at runtime, but the type is{ foo: string; default: { bar: string; }; bar: string; }
testPkg
containsbar: string
at runtime, but the type does not include that
nodejs/TSC#1622 (comment) describes the motivation of this module.exports
export interop feature.
π» Use Cases
-
What do you want to use this for?
To migrate a package that had CJS files in past, to ESM-only while keeping the interface as-is without generating a separate type definition file. -
What shortcomings exist with current approaches?
It requires two type definition types, one forrequire
and one forimport
, so two builds are needed. The package.json would also be convoluted.main
field cannot be used as the type definition file is different forrequire
andimport
. Theexports
field would need to be like:
{
"exports": {
".": {
"types": { "require": "./index.d.cts" },
"default": "./index.js"
}
}
}
- What workarounds are you using in the meantime?
I haven't considered it yet. Probably I'll use theexports
field workaround described above.