Skip to content

Commit 42ca435

Browse files
Fix ability to get multiple failures (ianstormtaylor#394) (ianstormtaylor#440)
* Test for multiple failures (with minimal changes to tests) * Do not attempt to iterate on `Generator`s more than once, fixes ianstormtaylor#394 * Refine public types: StructError.failures & StructContext.check * Update documentation regarding StructError.failures * Clean up tests: Have all fixtures export `failures` instead of `error` Co-authored-by: Ian Storm Taylor <[email protected]>
1 parent 537b929 commit 42ca435

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

56 files changed

+488
-342
lines changed

docs/reference.md

+8-8
Original file line numberDiff line numberDiff line change
@@ -546,17 +546,17 @@ The error class that Superstruct uses for its validation errors. This is exposed
546546
547547
Each error thrown includes the following properties:
548548
549-
| **Property** | **Type** | **Example** | **Description** |
550-
| ------------ | ------------------------- | ----------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
551-
| `branch` | `Array<any>` | `[{...}, false]` | An array of the values being validated at every layer. The first element in the array is the root value, and the last element is the current value that failed. This allows you to inspect the entire validation tree. |
552-
| `path` | `Array<string \| number>` | `['address', 'street']` | The path to the invalid value relative to the root value. |
553-
| `value` | `any` | `false` | The invalid value. |
554-
| `type` | `string` | `'string'` | The expected type of the invalid value. |
555-
| `failures` | `Array<StructFailure>` | `[{...}]` | All the validation failures that were encountered. The error object always represents the first failure, but you can write more complex logic involving other failures if you need to. |
549+
| **Property** | **Type** | **Example** | **Description** |
550+
| ------------ | ---------------------------- | ----------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
551+
| `branch` | `Array<any>` | `[{...}, false]` | An array of the values being validated at every layer. The first element in the array is the root value, and the last element is the current value that failed. This allows you to inspect the entire validation tree. |
552+
| `path` | `Array<string \| number>` | `['address', 'street']` | The path to the invalid value relative to the root value. |
553+
| `value` | `any` | `false` | The invalid value. |
554+
| `type` | `string` | `'string'` | The expected type of the invalid value. |
555+
| `failures` | `() => Array<StructFailure>` | | A function that returns all the validation failures that were encountered. The error object always represents the first failure, but you can write more complex logic involving other failures if you need to. |
556556
557557
### Multiple Errors
558558
559-
The error thrown by Superstruct is always the first validation failure that was encountered, because this makes for convenient and simple logic in the majority of cases. However, the `failures` property is available with a list of all of the validation failures that occurred in case you want to add support for multiple error handling.
559+
The error thrown by Superstruct is always the first validation failure that was encountered, because this makes for convenient and simple logic in the majority of cases. However, the `failures` method will return a list of all of the validation failures that occurred in case you want to add support for multiple error handling.
560560
561561
## Utilities
562562

src/struct.ts

+18-12
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { toFailures } from './utils'
1+
import { toFailures, iteratorShift } from './utils'
22

33
/**
44
* `Struct` objects encapsulate the schema for a specific data type (with
@@ -47,18 +47,24 @@ export class StructError extends TypeError {
4747
type: string
4848
path: Array<number | string>
4949
branch: Array<any>
50-
failures: () => Iterable<StructFailure>;
50+
failures: () => Array<StructFailure>;
5151
[key: string]: any
5252

53-
constructor(failure: StructFailure, iterable: Iterable<StructFailure>) {
53+
constructor(
54+
failure: StructFailure,
55+
moreFailures: IterableIterator<StructFailure>
56+
) {
5457
const { path, value, type, branch, ...rest } = failure
5558
const message = `Expected a value of type \`${type}\`${
5659
path.length ? ` for \`${path.join('.')}\`` : ''
5760
} but received \`${JSON.stringify(value)}\`.`
5861

59-
function* failures(): Iterable<StructFailure> {
60-
yield failure
61-
yield* iterable
62+
let failuresResult: Array<StructFailure> | undefined
63+
function failures(): Array<StructFailure> {
64+
if (!failuresResult) {
65+
failuresResult = [failure, ...moreFailures]
66+
}
67+
return failuresResult
6268
}
6369

6470
super(message)
@@ -89,7 +95,7 @@ export type StructContext = {
8995
struct: Struct<any> | Struct<never>,
9096
parent?: any,
9197
key?: string | number
92-
) => Iterable<StructFailure>
98+
) => IterableIterator<StructFailure>
9399
}
94100

95101
/**
@@ -163,11 +169,11 @@ export function validate<T>(
163169
value = struct.coercer(value)
164170
}
165171

166-
const iterable = check(value, struct)
167-
const [failure] = iterable
172+
const failures = check(value, struct)
173+
const failure = iteratorShift(failures)
168174

169175
if (failure) {
170-
const error = new StructError(failure, iterable)
176+
const error = new StructError(failure, failures)
171177
return [error, undefined]
172178
} else {
173179
return [undefined, value as T]
@@ -183,7 +189,7 @@ function* check<T>(
183189
struct: Struct<T>,
184190
path: any[] = [],
185191
branch: any[] = []
186-
): Iterable<StructFailure> {
192+
): IterableIterator<StructFailure> {
187193
const { type } = struct
188194
const ctx: StructContext = {
189195
value,
@@ -201,7 +207,7 @@ function* check<T>(
201207
}
202208

203209
const failures = toFailures(struct.validator(value, ctx), ctx)
204-
const [failure] = failures
210+
const failure = iteratorShift(failures)
205211

206212
if (failure) {
207213
yield failure

src/utils.ts

+15-5
Original file line numberDiff line numberDiff line change
@@ -7,15 +7,25 @@ export type StructTuple<T> = { [K in keyof T]: Struct<T[K]> }
77
* Convert a validation result to an iterable of failures.
88
*/
99

10-
export function toFailures(
10+
export function* toFailures(
1111
result: StructResult,
1212
context: StructContext
13-
): Iterable<StructFailure> {
13+
): IterableIterator<StructFailure> {
1414
if (result === true) {
15-
return []
15+
// yield nothing
1616
} else if (result === false) {
17-
return [context.fail()]
17+
yield context.fail()
1818
} else {
19-
return result
19+
yield* result
2020
}
2121
}
22+
23+
/**
24+
* Shifts (removes and returns) the first value from the `input` iterator.
25+
* Like `Array.prototype.shift()` but for an `Iterator`.
26+
*/
27+
28+
export function iteratorShift<T>(input: Iterator<T>): T | undefined {
29+
const { done, value } = input.next()
30+
return done ? undefined : value
31+
}

test/fixtures/array/invalid-element-property.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = array(object({ id: string() }))
44

55
export const data = [{ id: '1' }, { id: false }, { id: '3' }]
66

7-
export const error = {
8-
value: false,
9-
type: 'string',
10-
path: [1, 'id'],
11-
branch: [data, data[1], data[1].id],
12-
}
7+
export const failures = [
8+
{
9+
value: false,
10+
type: 'string',
11+
path: [1, 'id'],
12+
branch: [data, data[1], data[1].id],
13+
},
14+
]

test/fixtures/array/invalid-element.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = array(number())
44

55
export const data = [1, 'invalid', 3]
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'number',
10-
path: [1],
11-
branch: [data, data[1]],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'number',
11+
path: [1],
12+
branch: [data, data[1]],
13+
},
14+
]

test/fixtures/array/invalid-opaque.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = array()
44

55
export const data = 'invalid'
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'Array<unknown>',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'Array<unknown>',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/array/invalid.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = array(number())
44

55
export const data = 'invalid'
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'Array<number>',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'Array<number>',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/boolean/invalid.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = boolean()
44

55
export const data = 'invalid'
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'boolean',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'boolean',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/date/invalid-date.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = date()
44

55
export const data = new Date('invalid')
66

7-
export const error = {
8-
value: data,
9-
type: 'Date',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: data,
10+
type: 'Date',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/date/invalid.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = date()
44

55
export const data = 'invalid'
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'Date',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'Date',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/dynamic/invalid-reference.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -29,9 +29,11 @@ export const data = {
2929
price: 'Only $19.99!',
3030
}
3131

32-
export const error = {
33-
value: 'Only $19.99!',
34-
type: 'number',
35-
path: ['price'],
36-
branch: [data, data.price],
37-
}
32+
export const failures = [
33+
{
34+
value: 'Only $19.99!',
35+
type: 'number',
36+
path: ['price'],
37+
branch: [data, data.price],
38+
},
39+
]

test/fixtures/dynamic/invalid.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = dynamic(() => string())
44

55
export const data = 3
66

7-
export const error = {
8-
value: 3,
9-
type: 'string',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 3,
10+
type: 'string',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/empty/invalid-array.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = empty(array())
44

55
export const data = ['invalid']
66

7-
export const error = {
8-
value: ['invalid'],
9-
type: 'Array<unknown> & Empty',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: ['invalid'],
10+
type: 'Array<unknown> & Empty',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/empty/invalid-string.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = empty(string())
44

55
export const data = 'invalid'
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'string & Empty',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'string & Empty',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/enums/invalid-numbers.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = enums([1, 2])
44

55
export const data = 'invalid'
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'Enum<1,2>',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'Enum<1,2>',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/enums/invalid-strings.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = enums(['one', 'two'])
44

55
export const data = 'invalid'
66

7-
export const error = {
8-
value: 'invalid',
9-
type: 'Enum<"one","two">',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: 'invalid',
10+
type: 'Enum<"one","two">',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

test/fixtures/function/invalid.ts

+8-6
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@ export const Struct = func()
44

55
export const data = false
66

7-
export const error = {
8-
value: false,
9-
type: 'Function',
10-
path: [],
11-
branch: [data],
12-
}
7+
export const failures = [
8+
{
9+
value: false,
10+
type: 'Function',
11+
path: [],
12+
branch: [data],
13+
},
14+
]

0 commit comments

Comments
 (0)