Uh oh!
There was an error while loading. Please reload this page.
- Notifications
You must be signed in to change notification settings - Fork 34.4k
assert: adds rejects() and doesNotReject()#18023
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
Uh oh!
There was an error while loading. Please reload this page.
Conversation
feugy commented Jan 7, 2018 • edited by tniessen
Loading Uh oh!
There was an error while loading. Please reload this page.
edited by tniessen
Uh oh!
There was an error while loading. Please reload this page.
lib/assert.js Outdated
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.
Nit: s/conveniency/convenience/?
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 😅
lib/assert.js Outdated
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.
Maybe name assert.doesNotThrow as a top level function and reference it directly here to avoid being messed up by monkey-patched assert.doesNotThrow
lib/assert.js Outdated
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.
Maybe name assert.throws as a top level function and reference it directly here to avoid being messed up by monkey-patched assert.throws
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.
👌
lib/assert.js Outdated
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.
Since other assigned APIs do not return promises here:
require('assert').promises.fail().then(()=>{});// does not workThis behavior looks somewhat strange to me..
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.
To me as well. I would rather have a dedicated assert.rejects function. Reject is used for promises, so this is a good fit out of my perspective.
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.
That's an interesting point.
My intention was to keep the existing behaviour for all assert functions except throws() and doesNotThrow(), which are the only two able to run (potentially async) code.
I wouldn't like to force people writing:
constassert=require('assert').promisesawaitassert(something)But we can definitely add new functions like reject() that would be async equivalent of existing one.
Are there any other function you would like to cover?
joyeecheungJan 10, 2018 • edited
Loading Uh oh!
There was an error while loading. Please reload this page.
edited
Uh oh!
There was an error while loading. Please reload this page.
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.
If we are not talking about redesigning a new assert API here, providing a promisified version of the existing assert APIs is good enough to me. Something like
for(constkeyofObject.keys(assert)){constsyncFunc=assert[key];promises[key]=asyncfunction(...args){returnsyncFunc(...args)};}// override throws later, or in the blockThere 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.
hum, but it's actually what I'd like to avoid.
There's no point making assert.ok() or assert.fail() asynchronous. It will make the API more cumbersome to use with no benefit.
It make sense for assert.throws() and assert.doesNotThrow() to turn this code (that we used quite a lot in our projects):
it('should report database option errors',async()=>{try{awaittransferData({},'');thrownewError('it should have failed');}catch(err){assert(err.message.includes('"host" is required'));}});into
it('should report database option errors',async()=>assert.throws(async()=>transferData({},''),/^Error:"host"isrequired/));I don't want to redesign the whole API, but rather wants to support the above.
test/parallel/test-assert.js Outdated
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.
Why not?
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.
In this PR, you have currently (all synchronous):
require('assert').equalrequire('assert').strict.equalrequire('assert').promises.equalrequire('assert').promises.strict.equal
I wasn't sure it worth providing require('assert').strict.promises. Do you think it worth it?
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.
In general strict and regular assert should have exactly the same API.
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.
Copy that
lib/assert.js Outdated
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.
To me as well. I would rather have a dedicated assert.rejects function. Reject is used for promises, so this is a good fit out of my perspective.
BridgeAR commented Jan 11, 2018 • edited
Loading Uh oh!
There was an error while loading. Please reload this page.
edited
Uh oh!
There was an error while loading. Please reload this page.
Because there seems to be a bit of confusion in the discussion: I personally am strongly against adding a new The only thing that should be added out of my perspective are two new functions They behave similar to the current |
feugy commented Jan 11, 2018
@BridgeAR, by bringing But what you suggested above makes more sense, so I'm 100% with you, and implement it if @joyeecheung is happy with it. |
joyeecheung commented Jan 11, 2018 • edited
Loading Uh oh!
There was an error while loading. Please reload this page.
edited
Uh oh!
There was an error while loading. Please reload this page.
@feugy If adding asynchronous versions of the synchronous APIs doesn't make sense, then maybe we don't even need to add them to |
apapirovski left a comment • edited
Loading Uh oh!
There was an error while loading. Please reload this page.
edited
Uh oh!
There was an error while loading. Please reload this page.
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 is a good start but I'm also in favour of using rejects and doesNotReject instead of a separate promises namespace.
(Worth noting that this would also be in line with the fact that promises emit unhandledRejection rather than uncaughtException.)
lib/assert.js Outdated
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.
Can this be moved down closer to trySyncBlock which is its first usage? Or otherwise, move up those two functions here.
I would actually even suggest just inlining this function since it's only used twice. The savings are minimal and now the error thrown will have an extra function in the trace. The latter can be worked around (captureStackTrace) but I don't think it's worth it.
test/parallel/test-assert-async.js Outdated
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.
We don't need the common.platformTimeout here. Just matters that it's async.
lib/assert.js Outdated
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.
nits:
s/doNotThrows()/doesNotThrow()
s/truely/truly
s/other/others
lib/assert.js Outdated
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.
s/or let it bubble/or rethrow.
lib/assert.js Outdated
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.
catch (err) or catch (e) is more in line with other usage in the code base.
447adf8 to c0ba6adComparefeugy commented Jan 18, 2018
Hello gents! What do you think about the latest update? |
jasnell left a comment
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.
LGTM if CI is green
jasnell commented Jan 18, 2018
quick fyi... we're not all "gents" :-) |
feugy commented Jan 18, 2018
Sorry, it's bad habit 😓 |
BridgeAR left a comment
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.
Overall it is already looking very promising to me! Just a few small comments :-)
lib/assert.js Outdated
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 is actually a slightly different behavior as before.
Before assert.throws(() =>{}, 'foo', undefined) would also result in an error. That is not the case anymore with this change. As this is completely independent from the actual change here, I would rather not have this included, especially as I think this should throw.
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.
You're absolutely right!
To keep this code in an internal function, and not break the existing behaviour, I've used rest params and splat so arguments.length would still make sense.
lib/assert.js Outdated
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 comment is actually not correct anymore as it is only a no-op in case actual is the sentinel.
I personally feel like it is not necessary to have these comments here because the code is easy to understand but others might disagree.
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.
👌
test/parallel/test-assert-async.js Outdated
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.
Super tiny nit:
Destructuring would be nice as in const{promisify } = require('util')
test/parallel/test-assert-async.js Outdated
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 am not totally happy with this test as it does not test that the code does not fail sync.
So I would write something like:
assert.doesNotThrow(()=>{assert.rejects(()=>assert.fail(),common.expectsError({code: 'ERR_ASSERTION',type: assert.AssertionError,message: 'Failed',operator: undefined,actual: undefined,expected: undefined}))});The same applies to the test below.
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.
👌
test/parallel/test-assert-async.js Outdated
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.
If I am not mistaken, we do not know if these really work because the function itself might theoretically execute sync.
So a better test out of my perspective would be:
letpromise;letthrew=false;try{promise=assert.doesNotReject(async()=>{awaitwait(10);assert.fail();});}catch(err){threw=true;}assert.strictEqual(threw,false);assert.rejects(()=>promise,common.expectsError({code: 'ERR_ASSERTION',type: assert.AssertionError,message: 'Got unwanted exception ... '...}));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.
👌
test/parallel/test-assert-async.js Outdated
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.
Hm, I would say it tests the functionality of assert.rejects() and assert.doesNotReject() but not that it "ensures" async support for those as they should always be async.
I would describe it as e.g. "Tests assert.rejects() and assert.doesNotReject() by checking their expected output and by verifying that they do not work sync."
lib/assert.js Outdated
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 am a bit surprised that this for the stackStartFn shall work. this should normally refer to either assert.ok or assert.strict.
Can you please write a test for this as well if none exist? :-)
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 added a test, and you were absolutely right.
Providing explicit values keeps the existing stack-traces:
6cdbbe6 to 6779293Compare
BridgeAR left a comment
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 code change itself looks very good now! Just the tests might need a bit more polishing :-)
lib/assert.js Outdated
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.
Super super tiny nit: you can actually access the own function without assert. It would just save some characters.
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 also keep you away from monkey-patching. I would prefer referencing it directly here.
test/parallel/test-assert-async.js Outdated
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.
Nit: It would be really nice if the error message is changed to Missing expected rejection.
test/parallel/test-assert-async.js Outdated
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 two tests above are already tested for in test-assert.js. This adds the check for the error stack, so it indeed tests something new, but in that case I would rather replace the ones in test-assert.js with these tests instead of "duplicating" them.
test/parallel/test-assert-async.js Outdated
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 tests the same as https://github.com/nodejs/node/pull/18023/files#diff-cdf49dc677a8ea6f00de9f95639355e8R44.
What was not yet tested for is if the functions throws sync.
test/parallel/test-assert-async.js Outdated
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.
Using a async function as wrapper will actually hide if the function is executed sync or async. So I would just use the pattern as above without the async wrapper.
BridgeAR commented Jan 23, 2018
@feugy thanks a lot for being so patient! |
Leko left a comment
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.
LGTM
lib/assert.js Outdated
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 also keep you away from monkey-patching. I would prefer referencing it directly here.
joyeecheung commented Jan 23, 2018
LGTM with @BridgeAR 's comments addressed |
6779293 to 4f2a349Comparedoc/api/assert.md Outdated
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.
Maybe the next lines (copied from assert.doesNotThrow() doc) may be skipped in favour of a reference?
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 agree. I would actually even go so far and remove everything from here on. A text like:
Besides the async nature to await the completion it behaves identical to `assert.doesNotThrow()`.Keep an example with async await and also maybe add a example using promises.
The same can be applied to the reject function.
lib/assert.js Outdated
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.
operator & message will now contain appropriate values
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 is the unwritten rule to not use multi line template strings in Node.js. This does look somewhat elegant but it is probably best to just have something like: Missing expected ${fnType}${details}.
test/parallel/test-assert.js Outdated
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 wasn't confident to replace common.expectsError() with such custom assertions, but couldn't find a better way to reuse the existing test, as suggested.
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 would actually just keep this as is and just add a single line to one of the existing tests that checks for assert.ok(!err.stack.includes('at throws').
doc/api/assert.md Outdated
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 exceeds 80 characters.
lib/assert.js Outdated
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.
Nit (non-blocking): I personally prefer the following to save lines:
assert.throws=functionthrows(block, ...args){expectsError(throws,getActual(block), ...args);};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.
Good point. We could check for the function name instead but that could theoretically be manipulated.
So just keep it as is. Thanks for the heads up.
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.
Hm, while looking at the code again: we actually already use the function name and do not strictly test if it it a "valid" function. So we could indeed just switch to name checking only.
doc/api/assert.md Outdated
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 agree. I would actually even go so far and remove everything from here on. A text like:
Besides the async nature to await the completion it behaves identical to `assert.doesNotThrow()`.Keep an example with async await and also maybe add a example using promises.
The same can be applied to the reject function.
lib/assert.js Outdated
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 is the unwritten rule to not use multi line template strings in Node.js. This does look somewhat elegant but it is probably best to just have something like: Missing expected ${fnType}${details}.
test/parallel/test-assert-async.js Outdated
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 am relatively certain that this is going to trigger a lint error. The err should probably be in parentheses. The same applies to the test below.
test/parallel/test-assert-async.js Outdated
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.
After thinking about this again, the try catch and the threw is not necessary here at all. If the promise throws the test would fail right away. threw is only necessary for tests that check for threw === true. So the tests can be much smaller (I know I originally suggested this pattern).
test/parallel/test-assert-async.js Outdated
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 wrapper is not needed as the test would fail right away if assert.rejects would fail.
test/parallel/test-assert-async.js Outdated
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 wrapper is not needed due to the reason mentioned above.
test/parallel/test-assert.js Outdated
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 would actually just keep this as is and just add a single line to one of the existing tests that checks for assert.ok(!err.stack.includes('at throws').
test/parallel/test-assert.js Outdated
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.
Hm, I am not certain if that is actually true? I thought that frame would actually be excluded? Do the tests pass?
4f2a349 to 556ecdcCompareBridgeAR commented Feb 10, 2018
apapirovski commented Mar 2, 2018
@BridgeAR It seems to me like rejections have slightly different semantics than |
BridgeAR commented Mar 2, 2018
@apapirovski I agree that it is not exactly the same but the reason for that is that we have another issue that has to be solved as well. See #15126. |
apapirovski commented Mar 2, 2018 • edited
Loading Uh oh!
There was an error while loading. Please reload this page.
edited
Uh oh!
There was an error while loading. Please reload this page.
@BridgeAR Yeah, I'm aware and have dug around in that PR, actually, but that code will take a while to land given the performance implications. I'm not really certain we have a great path forward that doesn't impact performance. Maybe I'm wrong... (I have some thoughts on how to do it without all of the C++ boundary crossing but only theoretical atm.) |
apapirovski commented Mar 5, 2018 • edited
Loading Uh oh!
There was an error while loading. Please reload this page.
edited
Uh oh!
There was an error while loading. Please reload this page.
@BridgeAR given that this is only semver-minor whereas any changes to uncaught Promise handling would be semver-major, do you think it would be ok to land as is? Especially in the light of the recent rejection of the |
apapirovski commented Mar 5, 2018
BridgeAR commented Mar 5, 2018
@apapirovski yes, I am now ok with landing it as is. I will open another PR to add a comment to it similar to @feugy thanks a lot for your work and for being patient :) |
feugy commented Mar 6, 2018
It's all fine! |
Implement asynchronous equivalent for assert.throws() and assert.doesNotThrow().
feugy commented Mar 10, 2018
And done! What's next? |
Implement asynchronous equivalent for assert.throws() and assert.doesNotThrow(). PR-URL: nodejs#18023 Reviewed-By: James M Snell <[email protected]> Reviewed-By: Shingo Inoue <[email protected]> Reviewed-By: Joyee Cheung <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
BridgeAR commented Mar 11, 2018
Landed in 599337f 🎉 I removed the |
MylesBorins commented Mar 20, 2018
Should this be backported to |
not-an-aardvark commented Apr 4, 2018
If this gets backported, it should be backported at the same time as #19650 (possibly in the same commit). |
Implement asynchronous equivalent for assert.throws() and assert.doesNotThrow(). PR-URL: nodejs#18023 Reviewed-By: James M Snell <[email protected]> Reviewed-By: Shingo Inoue <[email protected]> Reviewed-By: Joyee Cheung <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
Implement asynchronous equivalent for assert.throws() and assert.doesNotThrow(). PR-URL: nodejs#18023 Reviewed-By: James M Snell <[email protected]> Reviewed-By: Shingo Inoue <[email protected]> Reviewed-By: Joyee Cheung <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
Implement asynchronous equivalent for assert.throws() and assert.doesNotThrow(). PR-URL: nodejs#18023 Reviewed-By: James M Snell <[email protected]> Reviewed-By: Shingo Inoue <[email protected]> Reviewed-By: Joyee Cheung <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
Implement asynchronous equivalent for assert.throws() and assert.doesNotThrow(). Backport-PR-URL: #24019 PR-URL: #18023 Reviewed-By: James M Snell <[email protected]> Reviewed-By: Shingo Inoue <[email protected]> Reviewed-By: Joyee Cheung <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
Implement async equivalent of
assert.throws()andassert.doesNotThrow().This PR is a follow-up of #17843, but takes into account James' suggestion to bring this as experimental feature, without change the existing API.
Checklist
make -j4 test(UNIX), orvcbuild test(Windows) passesAffected core subsystem(s)
assert