Skip to content

Commit 1614e8e

Browse files
marco-ippolitoaduh95
authored andcommitted
module: add ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX
PR-URL: #56610 Backport-PR-URL: #56912 Reviewed-By: James M Snell <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]> Reviewed-By: Geoffrey Booth <[email protected]> Reviewed-By: Ethan Arrowood <[email protected]> Reviewed-By: Chengzhong Wu <[email protected]>
1 parent c658a8a commit 1614e8e

File tree

8 files changed

+96
-38
lines changed

8 files changed

+96
-38
lines changed

doc/api/cli.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -1423,7 +1423,8 @@ Node.js will try to detect the syntax with the following steps:
14231423
1. Run the input as CommonJS.
14241424
2. If step 1 fails, run the input as an ES module.
14251425
3. If step 2 fails with a SyntaxError, strip the types.
1426-
4. If step 3 fails with an error code [`ERR_INVALID_TYPESCRIPT_SYNTAX`][],
1426+
4. If step 3 fails with an error code [`ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX`][]
1427+
or [`ERR_INVALID_TYPESCRIPT_SYNTAX`][],
14271428
throw the error from step 2, including the TypeScript error in the message,
14281429
else run as CommonJS.
14291430
5. If step 4 fails, run the input as an ES module.
@@ -3727,6 +3728,7 @@ node --stack-trace-limit=12 -p -e "Error.stackTraceLimit" # prints 12
37273728
[`Buffer`]: buffer.md#class-buffer
37283729
[`CRYPTO_secure_malloc_init`]: https://www.openssl.org/docs/man3.0/man3/CRYPTO_secure_malloc_init.html
37293730
[`ERR_INVALID_TYPESCRIPT_SYNTAX`]: errors.md#err_invalid_typescript_syntax
3731+
[`ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX`]: errors.md#err_unsupported_typescript_syntax
37303732
[`NODE_OPTIONS`]: #node_optionsoptions
37313733
[`NO_COLOR`]: https://no-color.org
37323734
[`SlowBuffer`]: buffer.md#class-slowbuffer

doc/api/errors.md

+17-3
Original file line numberDiff line numberDiff line change
@@ -2106,11 +2106,13 @@ does not consist of exactly two elements.
21062106

21072107
<!-- YAML
21082108
added: v22.10.0
2109+
changes:
2110+
- version: REPLACEME
2111+
pr-url: https://github.com/nodejs/node/pull/56610
2112+
description: This error is no longer thrown on valid yet unsupported syntax.
21092113
-->
21102114

2111-
The provided TypeScript syntax is not valid or unsupported.
2112-
This could happen when using TypeScript syntax that requires
2113-
transformation with [type-stripping][].
2115+
The provided TypeScript syntax is not valid.
21142116

21152117
<a id="ERR_INVALID_URI"></a>
21162118

@@ -3103,6 +3105,18 @@ try {
31033105
}
31043106
```
31053107

3108+
<a id="ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX"></a>
3109+
3110+
### `ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX`
3111+
3112+
<!-- YAML
3113+
added: REPLACEME
3114+
-->
3115+
3116+
The provided TypeScript syntax is unsupported.
3117+
This could happen when using TypeScript syntax that requires
3118+
transformation with [type-stripping][].
3119+
31063120
<a id="ERR_USE_AFTER_CLOSE"></a>
31073121

31083122
### `ERR_USE_AFTER_CLOSE`

lib/internal/errors.js

+1
Original file line numberDiff line numberDiff line change
@@ -1841,6 +1841,7 @@ E('ERR_UNSUPPORTED_NODE_MODULES_TYPE_STRIPPING',
18411841
E('ERR_UNSUPPORTED_RESOLVE_REQUEST',
18421842
'Failed to resolve module specifier "%s" from "%s": Invalid relative URL or base scheme is not hierarchical.',
18431843
TypeError);
1844+
E('ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX', '%s', SyntaxError);
18441845
E('ERR_USE_AFTER_CLOSE', '%s was closed', Error);
18451846

18461847
// This should probably be a `TypeError`.

lib/internal/modules/typescript.js

+16-1
Original file line numberDiff line numberDiff line change
@@ -12,8 +12,10 @@ const { assertTypeScript,
1212
isUnderNodeModules,
1313
kEmptyObject } = require('internal/util');
1414
const {
15+
ERR_INTERNAL_ASSERTION,
1516
ERR_INVALID_TYPESCRIPT_SYNTAX,
1617
ERR_UNSUPPORTED_NODE_MODULES_TYPE_STRIPPING,
18+
ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX,
1719
} = require('internal/errors').codes;
1820
const { getOptionValue } = require('internal/options');
1921
const assert = require('internal/assert');
@@ -49,7 +51,20 @@ function parseTypeScript(source, options) {
4951
try {
5052
return parse(source, options);
5153
} catch (error) {
52-
throw new ERR_INVALID_TYPESCRIPT_SYNTAX(error.message);
54+
/**
55+
* Amaro v0.3.0 (from SWC v1.10.7) throws an object with `message` and `code` properties.
56+
* It allows us to distinguish between invalid syntax and unsupported syntax.
57+
*/
58+
switch (error.code) {
59+
case 'UnsupportedSyntax':
60+
throw new ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX(error.message);
61+
case 'InvalidSyntax':
62+
throw new ERR_INVALID_TYPESCRIPT_SYNTAX(error.message);
63+
default:
64+
// SWC will throw strings when something goes wrong.
65+
// Check if has the `message` property or treat it as a string.
66+
throw new ERR_INTERNAL_ASSERTION(error.message ?? error);
67+
}
5368
}
5469
}
5570

lib/internal/process/execution.js

+15-25
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ const { getOptionValue } = require('internal/options');
3535
const {
3636
makeContextifyScript, runScriptInThisContext,
3737
} = require('internal/vm');
38-
const { emitExperimentalWarning, isError } = require('internal/util');
38+
const { emitExperimentalWarning } = require('internal/util');
3939
// shouldAbortOnUncaughtToggle is a typed array for faster
4040
// communication with JS.
4141
const { shouldAbortOnUncaughtToggle } = internalBinding('util');
@@ -254,10 +254,6 @@ function evalTypeScript(name, source, breakFirstLine, print, shouldLoadESM = fal
254254
try {
255255
compiledScript = compileScript(name, source, baseUrl);
256256
} catch (originalError) {
257-
// If it's not a SyntaxError, rethrow it.
258-
if (!isError(originalError) || originalError.name !== 'SyntaxError') {
259-
throw originalError;
260-
}
261257
try {
262258
sourceToRun = stripTypeScriptModuleTypes(source, name, false);
263259
// Retry the CJS/ESM syntax detection after stripping the types.
@@ -270,15 +266,14 @@ function evalTypeScript(name, source, breakFirstLine, print, shouldLoadESM = fal
270266
// Emit the experimental warning after the code was successfully evaluated.
271267
emitExperimentalWarning('Type Stripping');
272268
} catch (tsError) {
273-
// If its not an error, or it's not an invalid typescript syntax error, rethrow it.
274-
if (!isError(tsError) || tsError?.code !== 'ERR_INVALID_TYPESCRIPT_SYNTAX') {
275-
throw tsError;
269+
// If it's invalid or unsupported TypeScript syntax, rethrow the original error
270+
// with the TypeScript error message added to the stack.
271+
if (tsError.code === 'ERR_INVALID_TYPESCRIPT_SYNTAX' || tsError.code === 'ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX') {
272+
originalError.stack = decorateCJSErrorWithTSMessage(originalError.stack, tsError.message);
273+
throw originalError;
276274
}
277275

278-
try {
279-
originalError.stack = decorateCJSErrorWithTSMessage(originalError.stack, tsError.message);
280-
} catch { /* Ignore potential errors coming from `stack` getter/setter */ }
281-
throw originalError;
276+
throw tsError;
282277
}
283278
}
284279

@@ -322,28 +317,23 @@ function evalTypeScriptModuleEntryPoint(source, print) {
322317
// Compile the module to check for syntax errors.
323318
moduleWrap = loader.createModuleWrap(source, url);
324319
} catch (originalError) {
325-
// If it's not a SyntaxError, rethrow it.
326-
if (!isError(originalError) || originalError.name !== 'SyntaxError') {
327-
throw originalError;
328-
}
329-
let strippedSource;
330320
try {
331-
strippedSource = stripTypeScriptModuleTypes(source, url, false);
321+
const strippedSource = stripTypeScriptModuleTypes(source, url, false);
332322
// If the moduleWrap was successfully created, execute the module job.
333323
// outside the try-catch block to avoid catching runtime errors.
334324
moduleWrap = loader.createModuleWrap(strippedSource, url);
335325
// Emit the experimental warning after the code was successfully compiled.
336326
emitExperimentalWarning('Type Stripping');
337327
} catch (tsError) {
338-
// If its not an error, or it's not an invalid typescript syntax error, rethrow it.
339-
if (!isError(tsError) || tsError?.code !== 'ERR_INVALID_TYPESCRIPT_SYNTAX') {
340-
throw tsError;
341-
}
342-
try {
328+
// If it's invalid or unsupported TypeScript syntax, rethrow the original error
329+
// with the TypeScript error message added to the stack.
330+
if (tsError.code === 'ERR_INVALID_TYPESCRIPT_SYNTAX' ||
331+
tsError.code === 'ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX') {
343332
originalError.stack = `${tsError.message}\n\n${originalError.stack}`;
344-
} catch { /* Ignore potential errors coming from `stack` getter/setter */ }
333+
throw originalError;
334+
}
345335

346-
throw originalError;
336+
throw tsError;
347337
}
348338
}
349339
// If the moduleWrap was successfully created either with by just compiling

test/es-module/test-typescript-eval.mjs

+30-8
Original file line numberDiff line numberDiff line change
@@ -110,36 +110,36 @@ test('expect fail eval TypeScript ESM syntax with input-type commonjs-typescript
110110
strictEqual(result.code, 1);
111111
});
112112

113-
test('check syntax error is thrown when passing invalid syntax', async () => {
113+
test('check syntax error is thrown when passing unsupported syntax', async () => {
114114
const result = await spawnPromisified(process.execPath, [
115115
'--experimental-strip-types',
116116
'--eval',
117117
'enum Foo { A, B, C }']);
118118
strictEqual(result.stdout, '');
119119
match(result.stderr, /SyntaxError/);
120-
doesNotMatch(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
120+
doesNotMatch(result.stderr, /ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX/);
121121
strictEqual(result.code, 1);
122122
});
123123

124-
test('check syntax error is thrown when passing invalid syntax with --input-type=module-typescript', async () => {
124+
test('check syntax error is thrown when passing unsupported syntax with --input-type=module-typescript', async () => {
125125
const result = await spawnPromisified(process.execPath, [
126126
'--experimental-strip-types',
127127
'--input-type=module-typescript',
128128
'--eval',
129129
'enum Foo { A, B, C }']);
130130
strictEqual(result.stdout, '');
131-
match(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
131+
match(result.stderr, /ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX/);
132132
strictEqual(result.code, 1);
133133
});
134134

135-
test('check syntax error is thrown when passing invalid syntax with --input-type=commonjs-typescript', async () => {
135+
test('check syntax error is thrown when passing unsupported syntax with --input-type=commonjs-typescript', async () => {
136136
const result = await spawnPromisified(process.execPath, [
137137
'--experimental-strip-types',
138138
'--input-type=commonjs-typescript',
139139
'--eval',
140140
'enum Foo { A, B, C }']);
141141
strictEqual(result.stdout, '');
142-
match(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
142+
match(result.stderr, /ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX/);
143143
strictEqual(result.code, 1);
144144
});
145145

@@ -152,7 +152,7 @@ test('should not parse TypeScript with --type-module=commonjs', async () => {
152152

153153
strictEqual(result.stdout, '');
154154
match(result.stderr, /SyntaxError/);
155-
doesNotMatch(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
155+
doesNotMatch(result.stderr, /ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX/);
156156
strictEqual(result.code, 1);
157157
});
158158

@@ -165,7 +165,7 @@ test('should not parse TypeScript with --type-module=module', async () => {
165165

166166
strictEqual(result.stdout, '');
167167
match(result.stderr, /SyntaxError/);
168-
doesNotMatch(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
168+
doesNotMatch(result.stderr, /ERR_UNSUPPORTED_TYPESCRIPT_SYNTAX/);
169169
strictEqual(result.code, 1);
170170
});
171171

@@ -241,3 +241,25 @@ test('typescript CJS code is throwing a syntax error at runtime', async () => {
241241
strictEqual(result.stdout, '');
242242
strictEqual(result.code, 1);
243243
});
244+
245+
test('check syntax error is thrown when passing invalid syntax with --input-type=commonjs-typescript', async () => {
246+
const result = await spawnPromisified(process.execPath, [
247+
'--experimental-strip-types',
248+
'--input-type=commonjs-typescript',
249+
'--eval',
250+
'function foo(){ await Promise.resolve(1); }']);
251+
strictEqual(result.stdout, '');
252+
match(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
253+
strictEqual(result.code, 1);
254+
});
255+
256+
test('check syntax error is thrown when passing invalid syntax with --input-type=module-typescript', async () => {
257+
const result = await spawnPromisified(process.execPath, [
258+
'--experimental-strip-types',
259+
'--input-type=module-typescript',
260+
'--eval',
261+
'function foo(){ await Promise.resolve(1); }']);
262+
strictEqual(result.stdout, '');
263+
match(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
264+
strictEqual(result.code, 1);
265+
});

test/es-module/test-typescript.mjs

+11
Original file line numberDiff line numberDiff line change
@@ -435,3 +435,14 @@ test('execute a TypeScript loader and a .js file', async () => {
435435
match(result.stdout, /Hello, TypeScript!/);
436436
strictEqual(result.code, 0);
437437
});
438+
439+
test('execute invalid TypeScript syntax', async () => {
440+
const result = await spawnPromisified(process.execPath, [
441+
'--experimental-strip-types',
442+
fixtures.path('typescript/ts/test-invalid-syntax.ts'),
443+
]);
444+
445+
match(result.stderr, /ERR_INVALID_TYPESCRIPT_SYNTAX/);
446+
strictEqual(result.stdout, '');
447+
strictEqual(result.code, 1);
448+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
function foo(): string {
2+
await Promise.resolve(1);
3+
}

0 commit comments

Comments
 (0)