Skip to content

Commit 100230e

Browse files
hi-ogawasheremet-va
andauthoredDec 10, 2024··
fix(expect)!: check more properties for error equality (#5876)
Co-authored-by: Vladimir <sleuths.slews0s@icloud.com>
1 parent 2fb585a commit 100230e

File tree

8 files changed

+525
-139
lines changed

8 files changed

+525
-139
lines changed
 

‎docs/api/expect.md

+21-3
Original file line numberDiff line numberDiff line change
@@ -431,7 +431,17 @@ test('stocks are not the same', () => {
431431
```
432432

433433
:::warning
434-
A _deep equality_ will not be performed for `Error` objects. Only the `message` property of an Error is considered for equality. To customize equality to check properties other than `message`, use [`expect.addEqualityTesters`](#expect-addequalitytesters). To test if something was thrown, use [`toThrowError`](#tothrowerror) assertion.
434+
For `Error` objects, non-enumerable properties such as `name`, `message`, `cause` and `AggregateError.errors` are also compared. For `Error.cause`, the comparison is done asymmetrically:
435+
436+
```ts
437+
// success
438+
expect(new Error('hi', { cause: 'x' })).toEqual(new Error('hi'))
439+
440+
// fail
441+
expect(new Error('hi')).toEqual(new Error('hi', { cause: 'x' }))
442+
```
443+
444+
To test if something was thrown, use [`toThrowError`](#tothrowerror) assertion.
435445
:::
436446

437447
## toStrictEqual
@@ -649,8 +659,9 @@ test('the number of elements must match exactly', () => {
649659

650660
You can provide an optional argument to test that a specific error is thrown:
651661

652-
- regular expression: error message matches the pattern
653-
- string: error message includes the substring
662+
- `RegExp`: error message matches the pattern
663+
- `string`: error message includes the substring
664+
- `Error`, `AsymmetricMatcher`: compare with a received object similar to `toEqual(received)`
654665

655666
:::tip
656667
You must wrap the code in a function, otherwise the error will not be caught, and test will fail.
@@ -678,6 +689,13 @@ test('throws on pineapples', () => {
678689
expect(() => getFruitStock('pineapples')).toThrowError(
679690
/^Pineapples are not in stock$/,
680691
)
692+
693+
expect(() => getFruitStock('pineapples')).toThrowError(
694+
new Error('Pineapples are not in stock'),
695+
)
696+
expect(() => getFruitStock('pineapples')).toThrowError(expect.objectContaining({
697+
message: 'Pineapples are not in stock',
698+
}))
681699
})
682700
```
683701

‎packages/expect/src/jest-expect.ts

+9-5
Original file line numberDiff line numberDiff line change
@@ -794,12 +794,16 @@ export const JestChaiExpect: ChaiPlugin = (chai, utils) => {
794794
}
795795

796796
if (expected instanceof Error) {
797+
const equal = jestEquals(thrown, expected, [
798+
...customTesters,
799+
iterableEquality,
800+
])
797801
return this.assert(
798-
thrown && expected.message === thrown.message,
799-
`expected error to have message: ${expected.message}`,
800-
`expected error not to have message: ${expected.message}`,
801-
expected.message,
802-
thrown && thrown.message,
802+
equal,
803+
'expected a thrown error to be #{exp}',
804+
'expected a thrown error not to be #{exp}',
805+
expected,
806+
thrown,
803807
)
804808
}
805809

‎packages/expect/src/jest-utils.ts

+41-4
Original file line numberDiff line numberDiff line change
@@ -114,10 +114,6 @@ function eq(
114114
}
115115
}
116116

117-
if (a instanceof Error && b instanceof Error) {
118-
return a.message === b.message
119-
}
120-
121117
if (typeof URL === 'function' && a instanceof URL && b instanceof URL) {
122118
return a.href === b.href
123119
}
@@ -196,6 +192,16 @@ function eq(
196192
return false
197193
}
198194

195+
if (a instanceof Error && b instanceof Error) {
196+
try {
197+
return isErrorEqual(a, b, aStack, bStack, customTesters, hasKey)
198+
}
199+
finally {
200+
aStack.pop()
201+
bStack.pop()
202+
}
203+
}
204+
199205
// Deep compare objects.
200206
const aKeys = keys(a, hasKey)
201207
let key
@@ -225,6 +231,37 @@ function eq(
225231
return result
226232
}
227233

234+
function isErrorEqual(
235+
a: Error,
236+
b: Error,
237+
aStack: Array<unknown>,
238+
bStack: Array<unknown>,
239+
customTesters: Array<Tester>,
240+
hasKey: any,
241+
) {
242+
// https://nodejs.org/docs/latest-v22.x/api/assert.html#comparison-details
243+
// - [[Prototype]] of objects are compared using the === operator.
244+
// - Only enumerable "own" properties are considered.
245+
// - Error names, messages, causes, and errors are always compared, even if these are not enumerable properties. errors is also compared.
246+
247+
let result = (
248+
Object.getPrototypeOf(a) === Object.getPrototypeOf(b)
249+
&& a.name === b.name
250+
&& a.message === b.message
251+
)
252+
// check Error.cause asymmetrically
253+
if (typeof b.cause !== 'undefined') {
254+
result &&= eq(a.cause, b.cause, aStack, bStack, customTesters, hasKey)
255+
}
256+
// AggregateError.errors
257+
if (a instanceof AggregateError && b instanceof AggregateError) {
258+
result &&= eq(a.errors, b.errors, aStack, bStack, customTesters, hasKey)
259+
}
260+
// spread to compare enumerable properties
261+
result &&= eq({ ...a }, { ...b }, aStack, bStack, customTesters, hasKey)
262+
return result
263+
}
264+
228265
function keys(obj: object, hasKey: (obj: object, key: string) => boolean) {
229266
const keys = []
230267

‎packages/pretty-format/src/index.ts

+31
Original file line numberDiff line numberDiff line change
@@ -296,6 +296,35 @@ function printComplexValue(
296296
)}}`
297297
}
298298

299+
const ErrorPlugin: NewPlugin = {
300+
test: val => val && val instanceof Error,
301+
serialize(val: Error, config, indentation, depth, refs, printer) {
302+
if (refs.includes(val)) {
303+
return '[Circular]'
304+
}
305+
refs = [...refs, val]
306+
const hitMaxDepth = ++depth > config.maxDepth
307+
const { message, cause, ...rest } = val
308+
const entries = {
309+
message,
310+
...typeof cause !== 'undefined' ? { cause } : {},
311+
...val instanceof AggregateError ? { errors: val.errors } : {},
312+
...rest,
313+
}
314+
const name = val.name !== 'Error' ? val.name : getConstructorName(val as any)
315+
return hitMaxDepth
316+
? `[${name}]`
317+
: `${name} {${printIteratorEntries(
318+
Object.entries(entries).values(),
319+
config,
320+
indentation,
321+
depth,
322+
refs,
323+
printer,
324+
)}}`
325+
},
326+
}
327+
299328
function isNewPlugin(plugin: Plugin): plugin is NewPlugin {
300329
return (plugin as NewPlugin).serialize != null
301330
}
@@ -535,11 +564,13 @@ export const plugins: {
535564
Immutable: NewPlugin
536565
ReactElement: NewPlugin
537566
ReactTestComponent: NewPlugin
567+
Error: NewPlugin
538568
} = {
539569
AsymmetricMatcher,
540570
DOMCollection,
541571
DOMElement,
542572
Immutable,
543573
ReactElement,
544574
ReactTestComponent,
575+
Error: ErrorPlugin,
545576
}

‎packages/utils/src/diff/index.ts

+14
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,7 @@ const PLUGINS = [
5050
DOMCollection,
5151
Immutable,
5252
AsymmetricMatcher,
53+
prettyFormatPlugins.Error,
5354
]
5455
const FORMAT_OPTIONS = {
5556
plugins: PLUGINS,
@@ -298,6 +299,19 @@ export function replaceAsymmetricMatcher(
298299
replacedActual: any
299300
replacedExpected: any
300301
} {
302+
// handle asymmetric Error.cause diff
303+
if (
304+
actual instanceof Error
305+
&& expected instanceof Error
306+
&& typeof actual.cause !== 'undefined'
307+
&& typeof expected.cause === 'undefined'
308+
) {
309+
delete actual.cause
310+
return {
311+
replacedActual: actual,
312+
replacedExpected: expected,
313+
}
314+
}
301315
if (!isReplaceable(actual, expected)) {
302316
return { replacedActual: actual, replacedExpected: expected }
303317
}

‎test/core/test/__snapshots__/jest-expect.test.ts.snap

+237-3
Original file line numberDiff line numberDiff line change
@@ -279,7 +279,9 @@ exports[`asymmetric matcher error 21`] = `
279279
StringContaining "ll"
280280
281281
+ Received:
282-
[Error: hello]",
282+
Error {
283+
"message": "hello",
284+
}",
283285
"expected": "StringContaining "ll"",
284286
"message": "expected error to match asymmetric matcher",
285287
}
@@ -292,7 +294,9 @@ exports[`asymmetric matcher error 22`] = `
292294
stringContainingCustom<ll>
293295
294296
+ Received:
295-
[Error: hello]",
297+
Error {
298+
"message": "hello",
299+
}",
296300
"expected": "stringContainingCustom<ll>",
297301
"message": "expected error to match asymmetric matcher",
298302
}
@@ -305,7 +309,9 @@ exports[`asymmetric matcher error 23`] = `
305309
[Function MyError1]
306310
307311
+ Received:
308-
[Error: hello]",
312+
MyError2 {
313+
"message": "hello",
314+
}",
309315
"expected": "[Function MyError1]",
310316
"message": "expected error to be instance of MyError1",
311317
}
@@ -392,6 +398,234 @@ null
392398
}
393399
`;
394400
401+
exports[`error equality 1`] = `
402+
{
403+
"actual": "[Error: hi]",
404+
"diff": "- Expected
405+
+ Received
406+
407+
MyError {
408+
"message": "hi",
409+
- "custom": "b",
410+
+ "custom": "a",
411+
}",
412+
"expected": "[Error: hi]",
413+
"message": "expected Error: hi { custom: 'a' } to deeply equal Error: hi { custom: 'b' }",
414+
}
415+
`;
416+
417+
exports[`error equality 2`] = `
418+
{
419+
"actual": "[Error: hi]",
420+
"diff": "- Expected
421+
+ Received
422+
423+
MyError {
424+
"message": "hi",
425+
- "custom": "b",
426+
+ "custom": "a",
427+
}",
428+
"expected": "[Error: hi]",
429+
"message": "expected a thrown error to be Error: hi { custom: 'b' }",
430+
}
431+
`;
432+
433+
exports[`error equality 3`] = `
434+
{
435+
"actual": "[Error: hi]",
436+
"diff": "- Expected
437+
+ Received
438+
439+
MyError {
440+
- "message": "hello",
441+
+ "message": "hi",
442+
"custom": "a",
443+
}",
444+
"expected": "[Error: hello]",
445+
"message": "expected Error: hi { custom: 'a' } to deeply equal Error: hello { custom: 'a' }",
446+
}
447+
`;
448+
449+
exports[`error equality 4`] = `
450+
{
451+
"actual": "[Error: hello]",
452+
"diff": "- Expected
453+
+ Received
454+
455+
- YourError {
456+
+ MyError {
457+
"message": "hello",
458+
"custom": "a",
459+
}",
460+
"expected": "[Error: hello]",
461+
"message": "expected Error: hello { custom: 'a' } to deeply equal Error: hello { custom: 'a' }",
462+
}
463+
`;
464+
465+
exports[`error equality 5`] = `
466+
{
467+
"actual": "[Error: hello]",
468+
"diff": "- Expected
469+
+ Received
470+
471+
Error {
472+
"message": "hello",
473+
- "cause": "y",
474+
+ "cause": "x",
475+
}",
476+
"expected": "[Error: hello]",
477+
"message": "expected Error: hello to deeply equal Error: hello",
478+
}
479+
`;
480+
481+
exports[`error equality 6`] = `
482+
{
483+
"actual": "[Error: hello]",
484+
"diff": "- Expected
485+
+ Received
486+
487+
Error {
488+
"message": "hello",
489+
- "cause": "y",
490+
}",
491+
"expected": "[Error: hello]",
492+
"message": "expected Error: hello to deeply equal Error: hello",
493+
}
494+
`;
495+
496+
exports[`error equality 7`] = `
497+
{
498+
"actual": "[Error: hello]",
499+
"diff": "- Expected
500+
+ Received
501+
502+
Error {
503+
- "message": "world",
504+
+ "message": "hello",
505+
}",
506+
"expected": "[Error: world]",
507+
"message": "expected Error: hello to deeply equal Error: world",
508+
}
509+
`;
510+
511+
exports[`error equality 8`] = `
512+
{
513+
"actual": "[Error: hello]",
514+
"diff": "- Expected
515+
+ Received
516+
517+
- {
518+
- "something": "else",
519+
+ Error {
520+
+ "message": "hello",
521+
+ "cause": "x",
522+
}",
523+
"expected": "Object {
524+
"something": "else",
525+
}",
526+
"message": "expected Error: hello to deeply equal { something: 'else' }",
527+
}
528+
`;
529+
530+
exports[`error equality 9`] = `
531+
{
532+
"actual": "[AggregateError: outer]",
533+
"diff": "- Expected
534+
+ Received
535+
536+
AggregateError {
537+
"message": "outer",
538+
"cause": "x",
539+
"errors": [
540+
Error {
541+
"message": "inner",
542+
- "cause": "y",
543+
+ "cause": "x",
544+
},
545+
],
546+
}",
547+
"expected": "[AggregateError: outer]",
548+
"message": "expected AggregateError: outer { …(1) } to deeply equal AggregateError: outer { …(1) }",
549+
}
550+
`;
551+
552+
exports[`error equality 10`] = `
553+
{
554+
"actual": "[Error: hello]",
555+
"diff": "- Expected
556+
+ Received
557+
558+
Error {
559+
- "message": "world",
560+
+ "message": "hello",
561+
"cause": [Circular],
562+
}",
563+
"expected": "[Error: world]",
564+
"message": "expected Error: hello to deeply equal Error: world",
565+
}
566+
`;
567+
568+
exports[`error equality 11`] = `
569+
{
570+
"actual": "[Error: hello]",
571+
"diff": "- Expected
572+
+ Received
573+
574+
- ObjectContaining {
575+
- "cause": "y",
576+
+ Error {
577+
"message": "hello",
578+
+ "cause": "x",
579+
}",
580+
"expected": "ObjectContaining {
581+
"cause": "y",
582+
"message": "hello",
583+
}",
584+
"message": "expected Error: hello to deeply equal ObjectContaining{}",
585+
}
586+
`;
587+
588+
exports[`error equality 12`] = `
589+
{
590+
"actual": "[Error: hello]",
591+
"diff": "- Expected
592+
+ Received
593+
594+
- ObjectContaining {
595+
+ Error {
596+
+ "message": "hello",
597+
"cause": "x",
598+
- "message": "world",
599+
}",
600+
"expected": "ObjectContaining {
601+
"cause": "x",
602+
"message": "world",
603+
}",
604+
"message": "expected Error: hello to deeply equal ObjectContaining{}",
605+
}
606+
`;
607+
608+
exports[`error equality 13`] = `
609+
{
610+
"actual": "[Error: hello]",
611+
"diff": "- Expected
612+
+ Received
613+
614+
- ObjectContaining {
615+
- "cause": "y",
616+
- "message": "world",
617+
+ Error {
618+
+ "message": "hello",
619+
+ "cause": "x",
620+
}",
621+
"expected": "ObjectContaining {
622+
"cause": "y",
623+
"message": "world",
624+
}",
625+
"message": "expected Error: hello to deeply equal ObjectContaining{}",
626+
}
627+
`;
628+
395629
exports[`toHaveBeenNthCalledWith error 1`] = `
396630
{
397631
"actual": "Array [

‎test/core/test/expect.test.ts

+1-121
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import type { Tester } from '@vitest/expect'
2-
import nodeAssert from 'node:assert'
32
import { getCurrentTest } from '@vitest/runner'
4-
import { assert, describe, expect, expectTypeOf, test, vi } from 'vitest'
3+
import { describe, expect, expectTypeOf, test, vi } from 'vitest'
54

65
describe('expect.soft', () => {
76
test('types', () => {
@@ -281,125 +280,6 @@ describe('recursive custom equality tester', () => {
281280
})
282281
})
283282

284-
describe('Error equality', () => {
285-
class MyError extends Error {
286-
constructor(message: string, public custom: string) {
287-
super(message)
288-
}
289-
}
290-
291-
class YourError extends Error {
292-
constructor(message: string, public custom: string) {
293-
super(message)
294-
}
295-
}
296-
297-
test('basic', () => {
298-
//
299-
// default behavior
300-
//
301-
302-
{
303-
// different custom property
304-
const e1 = new MyError('hi', 'a')
305-
const e2 = new MyError('hi', 'b')
306-
expect(e1).toEqual(e2)
307-
expect(e1).toStrictEqual(e2)
308-
assert.deepEqual(e1, e2)
309-
nodeAssert.notDeepStrictEqual(e1, e2)
310-
}
311-
312-
{
313-
// different message
314-
const e1 = new MyError('hi', 'a')
315-
const e2 = new MyError('hello', 'a')
316-
expect(e1).not.toEqual(e2)
317-
expect(e1).not.toStrictEqual(e2)
318-
assert.notDeepEqual(e1, e2)
319-
nodeAssert.notDeepStrictEqual(e1, e2)
320-
}
321-
322-
{
323-
// different class
324-
const e1 = new MyError('hello', 'a')
325-
const e2 = new YourError('hello', 'a')
326-
expect(e1).toEqual(e2)
327-
expect(e1).not.toStrictEqual(e2) // toStrictEqual checks constructor already
328-
assert.deepEqual(e1, e2)
329-
nodeAssert.notDeepStrictEqual(e1, e2)
330-
}
331-
332-
{
333-
// same
334-
const e1 = new MyError('hi', 'a')
335-
const e2 = new MyError('hi', 'a')
336-
expect(e1).toEqual(e2)
337-
expect(e1).toStrictEqual(e2)
338-
assert.deepEqual(e1, e2)
339-
nodeAssert.deepStrictEqual(e1, e2)
340-
}
341-
342-
//
343-
// stricter behavior with custom equality tester
344-
//
345-
346-
expect.addEqualityTesters(
347-
[
348-
function tester(a, b, customTesters) {
349-
const aOk = a instanceof Error
350-
const bOk = b instanceof Error
351-
if (aOk && bOk) {
352-
// cf. assert.deepStrictEqual https://nodejs.org/api/assert.html#comparison-details_1
353-
// > [[Prototype]] of objects are compared using the === operator.
354-
// > Only enumerable "own" properties are considered.
355-
// > Error names and messages are always compared, even if these are not enumerable properties.
356-
return (
357-
Object.getPrototypeOf(a) === Object.getPrototypeOf(b)
358-
&& a.name === b.name
359-
&& a.message === b.message
360-
&& this.equals({ ...a }, { ...b }, customTesters)
361-
)
362-
}
363-
return aOk !== bOk ? false : undefined
364-
},
365-
],
366-
)
367-
368-
{
369-
// different custom property
370-
const e1 = new MyError('hi', 'a')
371-
const e2 = new MyError('hi', 'b')
372-
expect(e1).not.toEqual(e2) // changed
373-
expect(e1).not.toStrictEqual(e2) // changed
374-
assert.deepEqual(e1, e2) // chai assert is still same
375-
}
376-
377-
{
378-
// different message
379-
const e1 = new MyError('hi', 'a')
380-
const e2 = new MyError('hello', 'a')
381-
expect(e1).not.toEqual(e2)
382-
expect(e1).not.toStrictEqual(e2)
383-
}
384-
385-
{
386-
// different class
387-
const e1 = new MyError('hello', 'a')
388-
const e2 = new YourError('hello', 'a')
389-
expect(e1).not.toEqual(e2) // changed
390-
expect(e1).not.toStrictEqual(e2)
391-
}
392-
393-
{
394-
// same
395-
const e1 = new MyError('hi', 'a')
396-
const e2 = new MyError('hi', 'a')
397-
expect(e1).toEqual(e2)
398-
expect(e1).toStrictEqual(e2)
399-
}
400-
})
401-
})
402-
403283
describe('iterator', () => {
404284
test('returns true when given iterator within equal objects', () => {
405285
const a = {

‎test/core/test/jest-expect.test.ts

+171-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/* eslint-disable no-sparse-arrays */
2-
import { AssertionError } from 'node:assert'
2+
import nodeAssert, { AssertionError } from 'node:assert'
33
import { stripVTControlCharacters } from 'node:util'
44
import { generateToBeMessage } from '@vitest/expect'
55
import { processError } from '@vitest/utils/error'
@@ -1114,15 +1114,15 @@ describe('async expect', () => {
11141114
expect.unreachable()
11151115
}
11161116
catch (error) {
1117-
expect(error).toEqual(new Error('promise resolved "+0" instead of rejecting'))
1117+
expect(error).toMatchObject({ message: 'promise resolved "+0" instead of rejecting' })
11181118
}
11191119

11201120
try {
11211121
await expect({ then: (_: any, reject: any) => reject(0) }).resolves.toBe(0)
11221122
expect.unreachable()
11231123
}
11241124
catch (error) {
1125-
expect(error).toEqual(new Error('promise rejected "+0" instead of resolving'))
1125+
expect(error).toMatchObject({ message: 'promise rejected "+0" instead of resolving' })
11261126
}
11271127
})
11281128
})
@@ -1608,6 +1608,174 @@ it('asymmetric matcher error', () => {
16081608
}).toThrow(MyError1))
16091609
})
16101610

1611+
it('error equality', () => {
1612+
class MyError extends Error {
1613+
constructor(message: string, public custom: string) {
1614+
super(message)
1615+
}
1616+
}
1617+
1618+
class YourError extends Error {
1619+
constructor(message: string, public custom: string) {
1620+
super(message)
1621+
}
1622+
}
1623+
1624+
{
1625+
// different custom property
1626+
const e1 = new MyError('hi', 'a')
1627+
const e2 = new MyError('hi', 'b')
1628+
snapshotError(() => expect(e1).toEqual(e2))
1629+
expect(e1).not.toEqual(e2)
1630+
expect(e1).not.toStrictEqual(e2)
1631+
assert.deepEqual(e1, e2)
1632+
nodeAssert.notDeepStrictEqual(e1, e2)
1633+
1634+
// toThrowError also compare errors similar to toEqual
1635+
snapshotError(() =>
1636+
expect(() => {
1637+
throw e1
1638+
}).toThrowError(e2),
1639+
)
1640+
}
1641+
1642+
{
1643+
// different message
1644+
const e1 = new MyError('hi', 'a')
1645+
const e2 = new MyError('hello', 'a')
1646+
snapshotError(() => expect(e1).toEqual(e2))
1647+
expect(e1).not.toEqual(e2)
1648+
expect(e1).not.toStrictEqual(e2)
1649+
assert.notDeepEqual(e1, e2)
1650+
nodeAssert.notDeepStrictEqual(e1, e2)
1651+
}
1652+
1653+
{
1654+
// different class
1655+
const e1 = new MyError('hello', 'a')
1656+
const e2 = new YourError('hello', 'a')
1657+
snapshotError(() => expect(e1).toEqual(e2))
1658+
expect(e1).not.toEqual(e2)
1659+
expect(e1).not.toStrictEqual(e2) // toStrictEqual checks constructor already
1660+
assert.deepEqual(e1, e2)
1661+
nodeAssert.notDeepStrictEqual(e1, e2)
1662+
}
1663+
1664+
{
1665+
// same
1666+
const e1 = new MyError('hi', 'a')
1667+
const e2 = new MyError('hi', 'a')
1668+
expect(e1).toEqual(e2)
1669+
expect(e1).toStrictEqual(e2)
1670+
assert.deepEqual(e1, e2)
1671+
nodeAssert.deepStrictEqual(e1, e2)
1672+
1673+
expect(() => {
1674+
throw e1
1675+
}).toThrowError(e2)
1676+
}
1677+
1678+
{
1679+
// same
1680+
const e1 = new MyError('hi', 'a')
1681+
const e2 = new MyError('hi', 'a')
1682+
expect(e1).toEqual(e2)
1683+
expect(e1).toStrictEqual(e2)
1684+
assert.deepEqual(e1, e2)
1685+
nodeAssert.deepStrictEqual(e1, e2)
1686+
}
1687+
1688+
{
1689+
// different cause
1690+
const e1 = new Error('hello', { cause: 'x' })
1691+
const e2 = new Error('hello', { cause: 'y' })
1692+
snapshotError(() => expect(e1).toEqual(e2))
1693+
expect(e1).not.toEqual(e2)
1694+
}
1695+
1696+
{
1697+
// different cause (asymmetric fail)
1698+
const e1 = new Error('hello')
1699+
const e2 = new Error('hello', { cause: 'y' })
1700+
snapshotError(() => expect(e1).toEqual(e2))
1701+
expect(e1).not.toEqual(e2)
1702+
}
1703+
1704+
{
1705+
// different cause (asymmetric pass)
1706+
const e1 = new Error('hello', { cause: 'x' })
1707+
const e2 = new Error('hello')
1708+
expect(e1).toEqual(e2)
1709+
}
1710+
1711+
{
1712+
// different cause (fail by other props)
1713+
const e1 = new Error('hello', { cause: 'x' })
1714+
const e2 = new Error('world')
1715+
snapshotError(() => expect(e1).toEqual(e2))
1716+
}
1717+
1718+
{
1719+
// different cause
1720+
const e1 = new Error('hello', { cause: 'x' })
1721+
const e2 = { something: 'else' }
1722+
snapshotError(() => expect(e1).toEqual(e2))
1723+
}
1724+
1725+
{
1726+
// AggregateError (pass)
1727+
const e1 = new AggregateError([new Error('inner')], 'outer', { cause: 'x' })
1728+
const e2 = new AggregateError([new Error('inner')], 'outer', { cause: 'x' })
1729+
expect(e1).toEqual(e2)
1730+
}
1731+
1732+
{
1733+
// AggregateError (fail)
1734+
const e1 = new AggregateError([new Error('inner', { cause: 'x' })], 'outer', { cause: 'x' })
1735+
const e2 = new AggregateError([new Error('inner', { cause: 'y' })], 'outer', { cause: 'x' })
1736+
snapshotError(() => expect(e1).toEqual(e2))
1737+
}
1738+
1739+
{
1740+
// cyclic (pass)
1741+
const e1 = new Error('hi')
1742+
e1.cause = e1
1743+
const e2 = new Error('hi')
1744+
e2.cause = e2
1745+
expect(e1).toEqual(e2)
1746+
}
1747+
1748+
{
1749+
// cyclic (fail)
1750+
const e1 = new Error('hello')
1751+
e1.cause = e1
1752+
const e2 = new Error('world')
1753+
e2.cause = e2
1754+
snapshotError(() => expect(e1).toEqual(e2))
1755+
}
1756+
1757+
{
1758+
// asymmetric matcher
1759+
const e1 = new Error('hello', { cause: 'x' })
1760+
expect(e1).toEqual(expect.objectContaining({
1761+
message: 'hello',
1762+
cause: 'x',
1763+
}))
1764+
snapshotError(() => expect(e1).toEqual(expect.objectContaining({
1765+
message: 'hello',
1766+
cause: 'y',
1767+
})))
1768+
snapshotError(() => expect(e1).toEqual(expect.objectContaining({
1769+
message: 'world',
1770+
cause: 'x',
1771+
})))
1772+
snapshotError(() => expect(e1).toEqual(expect.objectContaining({
1773+
message: 'world',
1774+
cause: 'y',
1775+
})))
1776+
}
1777+
})
1778+
16111779
it('toHaveBeenNthCalledWith error', () => {
16121780
const fn = vi.fn()
16131781
fn('World')

0 commit comments

Comments
 (0)
Please sign in to comment.