-
Notifications
You must be signed in to change notification settings - Fork 30.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
module: improve error message for top-level await in CommonJS #55874
base: main
Are you sure you want to change the base?
Conversation
Codecov ReportAll modified and coverable lines are covered by tests ✅
Additional details and impacted files@@ Coverage Diff @@
## main #55874 +/- ##
==========================================
+ Coverage 88.00% 88.53% +0.53%
==========================================
Files 656 657 +1
Lines 189000 189866 +866
Branches 35995 36453 +458
==========================================
+ Hits 166320 168092 +1772
+ Misses 15840 14981 -859
+ Partials 6840 6793 -47
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thank you for tackling this! I’d love to see this issue addressed.
src/node_contextify.cc
Outdated
for (const auto& error_message : throws_only_in_cjs_error_messages) { | ||
if (message_view.find(error_message) != std::string_view::npos) { | ||
isolate->ThrowException(v8::Exception::SyntaxError( | ||
String::NewFromUtf8( | ||
isolate, | ||
"Top-level await is not supported in CommonJS. " | ||
"Consider using ESM or wrap await in an async function.") | ||
.ToLocalChecked())); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It seems like this message is being printed for any of the “throws only in CJS” error messages? I would think that we would want to print this only when the user wrote a top-level await
in a file that didn’t parse successfully as ESM.
Also in general our messages don’t use “ESM” as that’s not a widely known term. I think we usually write “module syntax” or “import/export syntax”.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
thank you very much for the improvement suggestions, I wonder if you could comment on the broken test/es-module/test-esm-detect-ambiguous.mjs tests, I am trying to improve them
in addition, I wonder if it would be better to include my newly added test/parallel/test-commonjs-top-level-await.js test in test-esm-detect-ambiguous.mjs
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What test failure do you see?
in addition, I wonder if it would be better to include my newly added test/parallel/test-commonjs-top-level-await.js test in test-esm-detect-ambiguous.mjs
Yes, my preference would be to keep all the syntax detection tests together. If you could follow the pattern of the other related tests that would be great.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I saw this
nodejs/node/actions/runs/11863231944/job/33064268346?pr=55874
I don't see the failure in this output. When you run the tests locally, what fails?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
when I run the tests locally:
./node test/es-module/test-esm-detect-ambiguous.mjs
I see that 7 tests are exploding, I think the code we do error handling is crushing the expectations here
failing tests:
✖ does not warn when there are no package.json
✖ permits declaration of CommonJS module variables
✖ throws on undefined require
when top-level await
triggers ESM parsing
✖ still throws on await
in an ordinary sync function
✖ permits top-level await
above import/export syntax
✖ reports unfinished top-level await
✖ permits top-level await
I'm trying to fix this place.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@GeoffreyBooth would you have any suggestions for these test failures?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think those failures are probably legit and the implementation needs to be adjusted so that they pass.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think those failures are probably legit and the implementation needs to be adjusted so that they pass.
thank you, as you said I will try to fix the tests as legitimate
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I made an update for this place, many thanks for the suggestions @GeoffreyBooth
3dccce9
to
76f87e4
Compare
76f87e4
to
c9531ef
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for taking another pass at this. I mention this in the particular notes, but in general:
-
Please don't change lines based on personal preference or readability. We value the ability for
git blame
to point directly to the commit/PR that added or meaningfully changed a particular line, and that ability is diluted when unrelated PRs rewrite lines of code. -
Very few, if any, tests should change as a result of your PR: only tests specifically related to top-level
await
in CommonJS. Any other tests that fail as a result of your changes mean that there's a bug in the implementation that needs to be addressed. My guess is that the bug is that the new code isn't checking that the error message being thrown is specifically about top-levelawait
. If there aren't currently tests for top-levelawait
in CommonJS and your new error message, one or more should be added.
This is close; another pass and I think you may get it there. Thanks for your effort!
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
Co-authored-by: Geoffrey Booth <[email protected]>
thank you very much for the review, I will be sending a new commit following what you said |
Greetings, when I did as you suggested to cover only the top level error, I saw that all tests passed except 1 test |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, this is much better. What’s the one test that is failing?
|
||
match( | ||
stderr, | ||
/await is only valid in async functions and the top level bodies of modules/ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The assertion should include some of the new text you’re adding, such as “To use top-level await,” to verify that your new error message is what’s getting printed.
Co-authored-by: Geoffrey Booth <[email protected]>
I saw wrong, all the tests passed my local. 🙏 |
There’s a test failing in here, in the relevant file: https://github.com/nodejs/node/actions/runs/12133213207/job/33828380981?pr=55874 @cjihrig why isn’t the CI output showing me which test is failing within that file? |
No clue. But the output is incomplete. There is no summary at the end either. It looks like maybe the Python runner killed the Node process or otherwise truncated its output. |
@GeoffreyBooth just FYI - IIRC, you asked me a similar question in Slack last year. There was output truncated only in GitHub Actions. Then you asked me to find a test run where truncation was happening without the use of |
Yes, last August: https://openjs-foundation.slack.com/archives/C019Y2T6STH/p1691860008875109. I’m lucky if I can remember last month, much less last year. Sorry to repeat. I also opened #49120 but that got closed. |
yes, ı view my local enviorement this test is fail test at test/es-module/test-esm-detect-ambiguous.mjs:367:5
✖ does not warn when there are no package.json (63.625667ms)
AssertionError [ERR_ASSERTION]: Expected values to be strictly equal:
+ actual - expected
+ "(node:18411) [MODULE_TYPELESS_PACKAGE_JSON] Warning: Module type of file:///Users/mert/Desktop/openSource/node/test/fixtures/es-modules/loose.js is not specified and it doesn't parse as CommonJS.\n" +
+ 'Reparsing as ES module because module syntax was detected. This incurs a performance overhead.\n' +
+ 'To eliminate this warning, add "type": "module" to /Users/mert/package.json.\n' +
+ '(Use `node --trace-warnings ...` to show where the warning was created)\n'
- ''
at TestContext.<anonymous> (file:///Users/mert/Desktop/openSource/node/test/es-module/test-esm-detect-ambiguous.mjs:372:7)
at process.processTicksAndRejections (node:internal/process/task_queues:105:5)
at async Test.run (node:internal/test_runner/test:932:9)
at async Promise.all (index 3)
at async Suite.run (node:internal/test_runner/test:1310:7)
at async Promise.all (index 6)
at async Suite.run (node:internal/test_runner/test:1310:7)
at async startSubtestAfterBootstrap (node:internal/test_runner/harness:297:3) {
generatedMessage: true,
code: 'ERR_ASSERTION',
actual: '(node:18411) [MODULE_TYPELESS_PACKAGE_JSON] Warning: Module type of file:///Users/mert/Desktop/openSource/node/test/fixtures/es-modules/loose.js is not specified and it doesn\'t parse as CommonJS.\nReparsing as ES module because module syntax was detected. This incurs a performance overhead.\nTo eliminate this warning, add "type": "module" to /Users/mert/package.json.\n(Use `node --trace-warnings ...` to show where the warning was created)\n',
expected: '',
operator: 'strictEqual'
} |
greetings I tried to make the error message more descriptive, in addition I tried to throw it when there is a file that cannot be successfully parsed as ESM |
hello, I would be very grateful if you could look here when you are available, thank you very much @GeoffreyBooth |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@mertcanaltin Thank you for your persistence. I think the approach of “a test is failing, so I’ll update the test” isn’t serving us well here; most of these updated tests shouldn’t need updating to achieve the goals of this PR, and the fact that they’re failing is an indicator that there are bugs in the implementation.
match( | ||
stderr, | ||
/SyntaxError: Top-level await is not supported in CommonJS modules\. To use top-level await, add "type": "module" to your package\.json or rename the file to use the \.mjs extension\. Alternatively, wrap the await expression in an async function\./ | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This test is titled “throws on undefined require
when top-level await
triggers ESM parsing”. That’s not happening anymore per your assertion here.
I think erroring about the top-level await
instead of the undefined require
is a better approach, but then the test title needs updating.
match(stderr, /SyntaxError: await is only valid in async function/); | ||
match( | ||
stderr, | ||
/Top-level await is not supported in CommonJS modules/ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This test is titled “still throws on await
in an ordinary sync function”. Based on the change you made to the assertion, your PR breaks this test.
strictEqual(code, 0); | ||
match( | ||
stderr, | ||
/Top-level await is not supported in CommonJS modules\. To use top-level await, add "type": "module" to your package\.json or rename the file to use the \.mjs extension/ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This test is titled “permits top-level await
above import/export syntax”. Based on the change you made to the assertion, your PR breaks this test.
strictEqual(stderr, ''); | ||
match( | ||
stderr, | ||
/SyntaxError: Top-level await is not supported in CommonJS modules\. To use top-level await, add "type": "module" to your package\.json or rename the file to use the \.mjs extension/ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This test is titled “reports unfinished top-level await
”. The code here should be parsed and run as ESM, and it should error with exit code 13 like the original test. This isn’t a test that should need updating as part of the goals of this PR.
strictEqual(code, 0); | ||
match( | ||
stderr, | ||
/Top-level await is not supported in CommonJS modules\. To use top-level await, add "type": "module" to your package\.json or rename the file to use the \.mjs extension/ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This test is titled “permits top-level await
”. This PR shouldn’t have the effect of prohibiting top-level await
in ambiguous files that get detected to be ESM. This isn’t a test that should need updating as part of the goals of this PR.
many thanks for your answers I will improve this place |
Added a specific error message for using top-level await in CommonJS modules. The error now suggests using ESM or wrapping await in an async function for clarity.
Fixes: #55776