Skip to content

Commit 7ed5aad

Browse files
marco-ippolitoruyadorno
authored andcommitted
module: add module.stripTypeScriptTypes
PR-URL: #55282 Backport-PR-URL: #56208 Fixes: #54300 Reviewed-By: Yagiz Nizipli <[email protected]> Reviewed-By: Joyee Cheung <[email protected]> Reviewed-By: Chemi Atlow <[email protected]> Reviewed-By: Paolo Insogna <[email protected]> Reviewed-By: Chengzhong Wu <[email protected]> Reviewed-By: James M Snell <[email protected]> Reviewed-By: Richard Lau <[email protected]>
1 parent 420d599 commit 7ed5aad

10 files changed

+358
-89
lines changed

doc/api/module.md

+101
Original file line numberDiff line numberDiff line change
@@ -352,6 +352,105 @@ changes:
352352
Register a module that exports [hooks][] that customize Node.js module
353353
resolution and loading behavior. See [Customization hooks][].
354354
355+
## `module.stripTypeScriptTypes(code[, options])`
356+
357+
<!-- YAML
358+
added: REPLACEME
359+
-->
360+
361+
> Stability: 1.0 - Early development
362+
363+
* `code` {string} The code to strip type annotations from.
364+
* `options` {Object}
365+
* `mode` {string} **Default:** `'strip'`. Possible values are:
366+
* `'strip'` Only strip type annotations without performing the transformation of TypeScript features.
367+
* `'transform'` Strip type annotations and transform TypeScript features to JavaScript.
368+
* `sourceMap` {boolean} **Default:** `false`. Only when `mode` is `'transform'`, if `true`, a source map
369+
will be generated for the transformed code.
370+
* `sourceUrl` {string} Specifies the source url used in the source map.
371+
* Returns: {string} The code with type annotations stripped.
372+
`module.stripTypeScriptTypes()` removes type annotations from TypeScript code. It
373+
can be used to strip type annotations from TypeScript code before running it
374+
with `vm.runInContext()` or `vm.compileFunction()`.
375+
By default, it will throw an error if the code contains TypeScript features
376+
that require transformation such as `Enums`,
377+
see [type-stripping][] for more information.
378+
When mode is `'transform'`, it also transforms TypeScript features to JavaScript,
379+
see [transform TypeScript features][] for more information.
380+
When mode is `'strip'`, source maps are not generated, because locations are preserved.
381+
If `sourceMap` is provided, when mode is `'strip'`, an error will be thrown.
382+
383+
_WARNING_: The output of this function should not be considered stable across Node.js versions,
384+
due to changes in the TypeScript parser.
385+
386+
```mjs
387+
import { stripTypeScriptTypes } from 'node:module';
388+
const code = 'const a: number = 1;';
389+
const strippedCode = stripTypeScriptTypes(code);
390+
console.log(strippedCode);
391+
// Prints: const a = 1;
392+
```
393+
394+
```cjs
395+
const { stripTypeScriptTypes } = require('node:module');
396+
const code = 'const a: number = 1;';
397+
const strippedCode = stripTypeScriptTypes(code);
398+
console.log(strippedCode);
399+
// Prints: const a = 1;
400+
```
401+
402+
If `sourceUrl` is provided, it will be used appended as a comment at the end of the output:
403+
404+
```mjs
405+
import { stripTypeScriptTypes } from 'node:module';
406+
const code = 'const a: number = 1;';
407+
const strippedCode = stripTypeScriptTypes(code, { mode: 'strip', sourceUrl: 'source.ts' });
408+
console.log(strippedCode);
409+
// Prints: const a = 1\n\n//# sourceURL=source.ts;
410+
```
411+
412+
```cjs
413+
const { stripTypeScriptTypes } = require('node:module');
414+
const code = 'const a: number = 1;';
415+
const strippedCode = stripTypeScriptTypes(code, { mode: 'strip', sourceUrl: 'source.ts' });
416+
console.log(strippedCode);
417+
// Prints: const a = 1\n\n//# sourceURL=source.ts;
418+
```
419+
420+
When `mode` is `'transform'`, the code is transformed to JavaScript:
421+
422+
```mjs
423+
import { stripTypeScriptTypes } from 'node:module';
424+
const code = `
425+
namespace MathUtil {
426+
export const add = (a: number, b: number) => a + b;
427+
}`;
428+
const strippedCode = stripTypeScriptTypes(code, { mode: 'transform', sourceMap: true });
429+
console.log(strippedCode);
430+
// Prints:
431+
// var MathUtil;
432+
// (function(MathUtil) {
433+
// MathUtil.add = (a, b)=>a + b;
434+
// })(MathUtil || (MathUtil = {}));
435+
// # sourceMappingURL=data:application/json;base64, ...
436+
```
437+
438+
```cjs
439+
const { stripTypeScriptTypes } = require('node:module');
440+
const code = `
441+
namespace MathUtil {
442+
export const add = (a: number, b: number) => a + b;
443+
}`;
444+
const strippedCode = stripTypeScriptTypes(code, { mode: 'transform', sourceMap: true });
445+
console.log(strippedCode);
446+
// Prints:
447+
// var MathUtil;
448+
// (function(MathUtil) {
449+
// MathUtil.add = (a, b)=>a + b;
450+
// })(MathUtil || (MathUtil = {}));
451+
// # sourceMappingURL=data:application/json;base64, ...
452+
```
453+
355454
### `module.syncBuiltinESMExports()`
356455
357456
<!-- YAML
@@ -1333,3 +1432,5 @@ returned object contains the following keys:
13331432
[realm]: https://tc39.es/ecma262/#realm
13341433
[source map include directives]: https://sourcemaps.info/spec.html#h.lmz475t4mvbx
13351434
[transferrable objects]: worker_threads.md#portpostmessagevalue-transferlist
1435+
[transform TypeScript features]: typescript.md#typescript-features
1436+
[type-stripping]: typescript.md#type-stripping

lib/internal/main/eval_string.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,8 @@ const {
1414
markBootstrapComplete,
1515
} = require('internal/process/pre_execution');
1616
const { evalModuleEntryPoint, evalScript } = require('internal/process/execution');
17-
const { addBuiltinLibsToObject, stripTypeScriptTypes } = require('internal/modules/helpers');
18-
17+
const { addBuiltinLibsToObject } = require('internal/modules/helpers');
18+
const { stripTypeScriptModuleTypes } = require('internal/modules/typescript');
1919
const { getOptionValue } = require('internal/options');
2020

2121
prepareMainThreadExecution();
@@ -24,7 +24,7 @@ markBootstrapComplete();
2424

2525
const code = getOptionValue('--eval');
2626
const source = getOptionValue('--experimental-strip-types') ?
27-
stripTypeScriptTypes(code) :
27+
stripTypeScriptModuleTypes(code) :
2828
code;
2929

3030
const print = getOptionValue('--print');

lib/internal/modules/cjs/loader.js

+5-5
Original file line numberDiff line numberDiff line change
@@ -153,8 +153,8 @@ const {
153153
setHasStartedUserCJSExecution,
154154
stripBOM,
155155
toRealPath,
156-
stripTypeScriptTypes,
157156
} = require('internal/modules/helpers');
157+
const { stripTypeScriptModuleTypes } = require('internal/modules/typescript');
158158
const packageJsonReader = require('internal/modules/package_json_reader');
159159
const { getOptionValue, getEmbedderOptions } = require('internal/options');
160160
const shouldReportRequiredModules = getLazy(() => process.env.WATCH_REPORT_DEPENDENCIES);
@@ -1347,7 +1347,7 @@ let emittedRequireModuleWarning = false;
13471347
function loadESMFromCJS(mod, filename) {
13481348
let source = getMaybeCachedSource(mod, filename);
13491349
if (getOptionValue('--experimental-strip-types') && path.extname(filename) === '.mts') {
1350-
source = stripTypeScriptTypes(source, filename);
1350+
source = stripTypeScriptModuleTypes(source, filename);
13511351
}
13521352
const cascadedLoader = require('internal/modules/esm/loader').getOrInitializeCascadedLoader();
13531353
const isMain = mod[kIsMainSymbol];
@@ -1584,7 +1584,7 @@ function getMaybeCachedSource(mod, filename) {
15841584

15851585
function loadCTS(module, filename) {
15861586
const source = getMaybeCachedSource(module, filename);
1587-
const code = stripTypeScriptTypes(source, filename);
1587+
const code = stripTypeScriptModuleTypes(source, filename);
15881588
module._compile(code, filename, 'commonjs');
15891589
}
15901590

@@ -1596,7 +1596,7 @@ function loadCTS(module, filename) {
15961596
function loadTS(module, filename) {
15971597
// If already analyzed the source, then it will be cached.
15981598
const source = getMaybeCachedSource(module, filename);
1599-
const content = stripTypeScriptTypes(source, filename);
1599+
const content = stripTypeScriptModuleTypes(source, filename);
16001600
let format;
16011601
const pkg = packageJsonReader.getNearestParentPackageJSON(filename);
16021602
// Function require shouldn't be used in ES modules.
@@ -1616,7 +1616,7 @@ function loadTS(module, filename) {
16161616
if (Module._cache[parentPath]) {
16171617
let parentSource;
16181618
try {
1619-
parentSource = stripTypeScriptTypes(fs.readFileSync(parentPath, 'utf8'), parentPath);
1619+
parentSource = stripTypeScriptModuleTypes(fs.readFileSync(parentPath, 'utf8'), parentPath);
16201620
} catch {
16211621
// Continue regardless of error.
16221622
}

lib/internal/modules/esm/get_format.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -164,9 +164,10 @@ function getFileProtocolModuleFormat(url, context = { __proto__: null }, ignoreE
164164
// Since experimental-strip-types depends on detect-module, we always return null
165165
// if source is undefined.
166166
if (!source) { return null; }
167-
const { stripTypeScriptTypes, stringify } = require('internal/modules/helpers');
167+
const { stringify } = require('internal/modules/helpers');
168+
const { stripTypeScriptModuleTypes } = require('internal/modules/typescript');
168169
const stringifiedSource = stringify(source);
169-
const parsedSource = stripTypeScriptTypes(stringifiedSource, fileURLToPath(url));
170+
const parsedSource = stripTypeScriptModuleTypes(stringifiedSource, fileURLToPath(url));
170171
const detectedFormat = detectModuleFormat(parsedSource, url);
171172
const format = `${detectedFormat}-typescript`;
172173
if (format === 'module-typescript' && foundPackageJson) {

lib/internal/modules/esm/translators.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -30,10 +30,10 @@ const {
3030
assertBufferSource,
3131
loadBuiltinModule,
3232
stringify,
33-
stripTypeScriptTypes,
3433
stripBOM,
3534
urlToFilename,
3635
} = require('internal/modules/helpers');
36+
const { stripTypeScriptModuleTypes } = require('internal/modules/typescript');
3737
const {
3838
kIsCachedByESMLoader,
3939
Module: CJSModule,
@@ -244,7 +244,7 @@ translators.set('require-commonjs', (url, source, isMain) => {
244244
translators.set('require-commonjs-typescript', (url, source, isMain) => {
245245
emitExperimentalWarning('Type Stripping');
246246
assert(cjsParse);
247-
const code = stripTypeScriptTypes(stringify(source), url);
247+
const code = stripTypeScriptModuleTypes(stringify(source), url);
248248
return createCJSModuleWrap(url, code);
249249
});
250250

@@ -459,7 +459,7 @@ translators.set('wasm', async function(url, source) {
459459
translators.set('commonjs-typescript', function(url, source) {
460460
emitExperimentalWarning('Type Stripping');
461461
assertBufferSource(source, true, 'load');
462-
const code = stripTypeScriptTypes(stringify(source), url);
462+
const code = stripTypeScriptModuleTypes(stringify(source), url);
463463
debug(`Translating TypeScript ${url}`);
464464
return FunctionPrototypeCall(translators.get('commonjs'), this, url, code, false);
465465
});
@@ -468,7 +468,7 @@ translators.set('commonjs-typescript', function(url, source) {
468468
translators.set('module-typescript', function(url, source) {
469469
emitExperimentalWarning('Type Stripping');
470470
assertBufferSource(source, true, 'load');
471-
const code = stripTypeScriptTypes(stringify(source), url);
471+
const code = stripTypeScriptModuleTypes(stringify(source), url);
472472
debug(`Translating TypeScript ${url}`);
473473
return FunctionPrototypeCall(translators.get('module'), this, url, code, false);
474474
});

lib/internal/modules/helpers.js

+1-74
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,6 @@ const {
1515
const {
1616
ERR_INVALID_ARG_TYPE,
1717
ERR_INVALID_RETURN_PROPERTY_VALUE,
18-
ERR_INVALID_TYPESCRIPT_SYNTAX,
19-
ERR_UNSUPPORTED_NODE_MODULES_TYPE_STRIPPING,
2018
} = require('internal/errors').codes;
2119
const { BuiltinModule } = require('internal/bootstrap/realm');
2220

@@ -27,9 +25,8 @@ const path = require('path');
2725
const { pathToFileURL, fileURLToPath } = require('internal/url');
2826
const assert = require('internal/assert');
2927

30-
const { Buffer } = require('buffer');
3128
const { getOptionValue } = require('internal/options');
32-
const { assertTypeScript, setOwnProperty, getLazy, isUnderNodeModules } = require('internal/util');
29+
const { setOwnProperty, getLazy } = require('internal/util');
3330
const { inspect } = require('internal/util/inspect');
3431

3532
const lazyTmpdir = getLazy(() => require('os').tmpdir());
@@ -314,75 +311,6 @@ function getBuiltinModule(id) {
314311
return normalizedId ? require(normalizedId) : undefined;
315312
}
316313

317-
/**
318-
* The TypeScript parsing mode, either 'strip-only' or 'transform'.
319-
* @type {string}
320-
*/
321-
const getTypeScriptParsingMode = getLazy(() =>
322-
(getOptionValue('--experimental-transform-types') ? 'transform' : 'strip-only'),
323-
);
324-
325-
/**
326-
* Load the TypeScript parser.
327-
* and returns an object with a `code` property.
328-
* @returns {Function} The TypeScript parser function.
329-
*/
330-
const loadTypeScriptParser = getLazy(() => {
331-
assertTypeScript();
332-
const amaro = require('internal/deps/amaro/dist/index');
333-
return amaro.transformSync;
334-
});
335-
336-
/**
337-
*
338-
* @param {string} source the source code
339-
* @param {object} options the options to pass to the parser
340-
* @returns {TransformOutput} an object with a `code` property.
341-
*/
342-
function parseTypeScript(source, options) {
343-
const parse = loadTypeScriptParser();
344-
try {
345-
return parse(source, options);
346-
} catch (error) {
347-
throw new ERR_INVALID_TYPESCRIPT_SYNTAX(error);
348-
}
349-
}
350-
351-
/**
352-
* @typedef {object} TransformOutput
353-
* @property {string} code The compiled code.
354-
* @property {string} [map] The source maps (optional).
355-
*
356-
* Performs type-stripping to TypeScript source code.
357-
* @param {string} source TypeScript code to parse.
358-
* @param {string} filename The filename of the source code.
359-
* @returns {TransformOutput} The stripped TypeScript code.
360-
*/
361-
function stripTypeScriptTypes(source, filename) {
362-
if (isUnderNodeModules(filename)) {
363-
throw new ERR_UNSUPPORTED_NODE_MODULES_TYPE_STRIPPING(filename);
364-
}
365-
assert(typeof source === 'string');
366-
const options = {
367-
__proto__: null,
368-
mode: getTypeScriptParsingMode(),
369-
sourceMap: getOptionValue('--enable-source-maps'),
370-
filename,
371-
};
372-
const { code, map } = parseTypeScript(source, options);
373-
if (map) {
374-
// TODO(@marco-ippolito) When Buffer.transcode supports utf8 to
375-
// base64 transformation, we should change this line.
376-
const base64SourceMap = Buffer.from(map).toString('base64');
377-
return `${code}\n\n//# sourceMappingURL=data:application/json;base64,${base64SourceMap}`;
378-
}
379-
// Source map is not necessary in strip-only mode. However, to map the source
380-
// file in debuggers to the original TypeScript source, add a sourceURL magic
381-
// comment to hint that it is a generated source.
382-
return `${code}\n\n//# sourceURL=${filename}`;
383-
}
384-
385-
386314
/**
387315
* Enable on-disk compiled cache for all user modules being complied in the current Node.js instance
388316
* after this method is called.
@@ -485,7 +413,6 @@ module.exports = {
485413
loadBuiltinModule,
486414
makeRequireFunction,
487415
normalizeReferrerURL,
488-
stripTypeScriptTypes,
489416
stringify,
490417
stripBOM,
491418
toRealPath,

0 commit comments

Comments
 (0)