@based/schema 2.0.0 → 2.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,48 +0,0 @@
1
- import { BasedSchema } from '../types'
2
- import { ArgsClass } from './args'
3
- import { AsyncOperation, Opts } from './types'
4
-
5
- export const walk = async <T>(
6
- schema: BasedSchema,
7
- opts: Opts<T>,
8
- value: any,
9
- asyncOperationHandler?: AsyncOperation<T>
10
- ): Promise<T> => {
11
- if (!('collect' in opts)) {
12
- opts.collect = () => {}
13
- }
14
-
15
- if (!('error' in opts)) {
16
- opts.error = () => {}
17
- }
18
-
19
- if (asyncOperationHandler) {
20
- opts = {
21
- ...opts,
22
- asyncOperationHandler,
23
- }
24
- }
25
-
26
- const argsOpts = await opts.init(value, schema, opts.error)
27
-
28
- if (!argsOpts) {
29
- return <T>{}
30
- }
31
-
32
- if (!argsOpts.value) {
33
- argsOpts.value = value
34
- }
35
-
36
- const args = new ArgsClass(argsOpts)
37
- args.root = args
38
- args._opts = opts
39
- args._schema = schema
40
-
41
- await args.parse()
42
-
43
- return args.target
44
- }
45
-
46
- export { ArgsClass }
47
-
48
- export * from './types'
@@ -1,233 +0,0 @@
1
- import { deepEqual } from '@saulx/utils'
2
- import {
3
- BasedSchemaField,
4
- BasedSchemaFieldObject,
5
- BasedSchemaFieldRecord,
6
- BasedSchemaFields,
7
- } from '../types'
8
- import { ArgsClass } from './args'
9
- import { ArgsOpts, FieldParser, KeyParser, Stopped } from './types'
10
-
11
- export type ParseResult<T> = ArgsClass<T> | void
12
-
13
- // TODO needs cleaning
14
- function createOrUseArgs<T>(
15
- from: ArgsClass<T>,
16
- newArgs: ArgsClass<T> | ArgsOpts<T> | void
17
- ): ParseResult<T> {
18
- if (!newArgs) {
19
- return
20
- }
21
- if (newArgs instanceof ArgsClass) {
22
- return newArgs
23
- }
24
-
25
- const x = from.create(newArgs)
26
-
27
- // x.collectedCommands = from.collectedCommands
28
- // x.fromBackTrack = from.fromBackTrack
29
-
30
- return x
31
- }
32
-
33
- async function parseKey<T>(
34
- from: ArgsClass<T>,
35
- key: string | number,
36
- parser: KeyParser<T>
37
- ): Promise<ParseResult<T>> {
38
- const keyArgs = new ArgsClass(
39
- {
40
- key,
41
- value: from.value[key],
42
- fieldSchema: from.fieldSchema,
43
- },
44
- from
45
- )
46
-
47
- // if same
48
-
49
- const newArgs = createOrUseArgs(keyArgs, await parser(keyArgs))
50
-
51
- if (newArgs) {
52
- return newArgs.parse()
53
- }
54
- }
55
-
56
- function createFieldArgs<T>(
57
- from: ArgsClass<T>,
58
- key: string | number,
59
- fieldSchema: BasedSchemaField
60
- ): ArgsClass<T> {
61
- return new ArgsClass(
62
- {
63
- key,
64
- value: from.value[key],
65
- // @ts-ignore needs key
66
- fieldSchema,
67
- },
68
- from
69
- )
70
- }
71
-
72
- function getFieldParser<T>(
73
- args: ArgsClass<T>
74
- ): void | FieldParser<keyof BasedSchemaFields> {
75
- const fieldParser =
76
- 'enum' in args.fieldSchema
77
- ? args.root._opts.parsers.fields.enum
78
- : args.root._opts.parsers.fields[args.fieldSchema.type]
79
- return fieldParser
80
- }
81
-
82
- export async function parse<T>(
83
- args: ArgsClass<T>
84
- ): Promise<ArgsClass<T> | void> {
85
- const opts = args.root._opts
86
-
87
- if (args.parseTopLevel) {
88
- const parser = opts.parsers.any
89
- if (parser) {
90
- const nArgs = await parser(args)
91
-
92
- if (nArgs) {
93
- // @ts-ignore
94
- return parse(createOrUseArgs(args, nArgs))
95
- }
96
- }
97
- }
98
-
99
- if (typeof args.value === 'object' && args.value !== null) {
100
- const keyQ: Promise<ParseResult<T>>[] = []
101
- const keysHandled: Set<string | number> = new Set()
102
- let allKeysHandled = false
103
-
104
- for (const key in opts.parsers.keys) {
105
- if (key in args.value) {
106
- keysHandled.add(key)
107
- keyQ.push(parseKey(args, key, opts.parsers.keys[key]))
108
- }
109
- }
110
- await Promise.all(keyQ)
111
-
112
- if (typeof args.value !== 'object' || args.value === null) {
113
- return
114
- }
115
-
116
- // schema
117
- if (args.stopped === undefined) {
118
- const fieldQ: Promise<ParseResult<T>>[] = []
119
- if (args.typeSchema && !args.fieldSchema) {
120
- for (const key in args.typeSchema.fields) {
121
- const fieldSchema = args.typeSchema.fields[key]
122
- if (key in args.value) {
123
- keysHandled.add(key)
124
- fieldQ.push(createFieldArgs(args, key, fieldSchema).parse())
125
- }
126
- }
127
- } else if (args.fieldSchema && !args.stopped) {
128
- if (args.fieldSchema.type === 'object') {
129
- // @ts-ignore should detect from line above
130
- const objFieldSchema: BasedSchemaFieldObject = args.fieldSchema
131
- for (const key in objFieldSchema.properties) {
132
- const fieldSchema = objFieldSchema.properties[key]
133
- if (key in args.value) {
134
- keysHandled.add(key)
135
- fieldQ.push(createFieldArgs(args, key, fieldSchema).parse())
136
- }
137
- }
138
- } else if (args.fieldSchema.type === 'record') {
139
- // @ts-ignore should detect from line above
140
- const objFieldSchema: BasedSchemaFieldRecord = args.fieldSchema
141
- for (const key in args.value) {
142
- const fieldSchema = objFieldSchema.values
143
- keysHandled.add(key)
144
- fieldQ.push(createFieldArgs(args, key, fieldSchema).parse())
145
- }
146
- } else if (args.fieldSchema) {
147
- const fieldParser = getFieldParser(args)
148
- if (fieldParser) {
149
- const newArgs = createOrUseArgs(args, await fieldParser(args))
150
- if (newArgs) {
151
- return newArgs.parse()
152
- }
153
- }
154
- }
155
- }
156
- await Promise.all(fieldQ)
157
- if (
158
- args.fieldSchema &&
159
- fieldQ.length > 0 &&
160
- (args.fieldSchema.type === 'object' ||
161
- args.fieldSchema.type === 'record')
162
- ) {
163
- const fieldParser = getFieldParser(args)
164
- if (fieldParser) {
165
- fieldParser(args)
166
- }
167
- }
168
- }
169
-
170
- if (typeof args.value !== 'object' || args.value === null) {
171
- return
172
- }
173
-
174
- // any
175
- if (args.stopped !== Stopped.stopAll) {
176
- const parser = opts.parsers.any || opts.parsers.catch
177
- if (parser) {
178
- const q: Promise<ParseResult<T>>[] = []
179
- if (Array.isArray(args.value)) {
180
- for (let i = 0; i < args.value.length; i++) {
181
- if ((!opts.parsers.any && keysHandled.has(i)) || allKeysHandled) {
182
- continue
183
- }
184
- q.push(parseKey(args, i, parser))
185
- }
186
- } else {
187
- for (const key in args.value) {
188
- if ((!opts.parsers.any && keysHandled.has(key)) || allKeysHandled) {
189
- continue
190
- }
191
- q.push(parseKey(args, key, parser))
192
- }
193
- }
194
- await Promise.all(q)
195
- }
196
- }
197
-
198
- if (
199
- opts.backtrack &&
200
- !args.skipCollection &&
201
- (args.fromBackTrack.length || args.collectedCommands.length)
202
- ) {
203
- const backtracked = opts.backtrack(
204
- args,
205
- args.fromBackTrack ?? [],
206
- args.collectedCommands ?? []
207
- )
208
- if (backtracked) {
209
- const target = args.getBackTrackTarget()
210
- if (!target.fromBackTrack) {
211
- target.fromBackTrack = []
212
- }
213
- target.fromBackTrack.push(backtracked)
214
- }
215
- }
216
- } else {
217
- // more
218
- if (args.fieldSchema) {
219
- const fieldParser = getFieldParser(args)
220
- if (fieldParser) {
221
- const newArgs = createOrUseArgs(args, await fieldParser(args))
222
- if (newArgs) {
223
- return newArgs.parse()
224
- }
225
- } else {
226
- console.warn('fieldSchema type not implemented yet!', args.fieldSchema)
227
- const anyParser = opts.parsers.any || opts.parsers.catch
228
- anyParser(args)
229
- }
230
- } else {
231
- }
232
- }
233
- }
@@ -1,81 +0,0 @@
1
- import { ParseError } from '../error'
2
- import {
3
- BasedSchemaType,
4
- BasedSchemaFields,
5
- BasedSchemaField,
6
- BasedSchema,
7
- } from '../types'
8
- import { ArgsClass } from './args'
9
-
10
- export type Path = (string | number)[]
11
-
12
- export type ErrorHandler<T> = (
13
- code: ParseError,
14
- args: ArgsClass<T> | ArgsOpts<T>
15
- ) => void
16
-
17
- export type Collect<T> = (args: ArgsClass<T>) => any
18
-
19
- export type FieldParser<K extends keyof BasedSchemaFields, T = any> = (
20
- args: ArgsClass<T, K>
21
- ) => Promise<ArgsClass<T> | ArgsOpts<T> | void>
22
-
23
- export type KeyParser<T = any> = (
24
- args: ArgsClass<T, keyof BasedSchemaFields>
25
- ) => Promise<ArgsOpts<T> | ArgsClass<T> | void>
26
-
27
- export type FieldParsers<T = any> = {
28
- [Key in keyof BasedSchemaFields]: FieldParser<Key, T>
29
- }
30
-
31
- export type AsyncOperation<T> = (
32
- args: ArgsClass<T>,
33
- type?: string
34
- ) => Promise<any>
35
-
36
- export type Opts<T> = {
37
- init: (
38
- value: any,
39
- schema: BasedSchema,
40
- error: ErrorHandler<T>
41
- ) => Promise<ArgsOpts<T>>
42
- parsers: {
43
- fields: Partial<{
44
- [Key in keyof BasedSchemaFields]: FieldParser<Key, T>
45
- }>
46
- keys: { [key: string]: KeyParser<T> }
47
- any?: KeyParser<T>
48
- catch?: KeyParser<T>
49
- }
50
- collect?: (args: ArgsClass<T>) => any
51
- error?: ErrorHandler<T>
52
- backtrack?: (
53
- args: ArgsClass<T>,
54
- fromBackTrack: any[],
55
- collectedCommands: any[]
56
- ) => any
57
- asyncOperationHandler?: AsyncOperation<T>
58
- }
59
-
60
- export enum Stopped {
61
- onlyStopFieldParser,
62
- stopAll,
63
- }
64
-
65
- export type ArgsOpts<
66
- T,
67
- K extends keyof BasedSchemaFields = keyof BasedSchemaFields
68
- > = {
69
- parseTopLevel?: boolean
70
- target?: T
71
- key?: string | number
72
- path?: Path
73
- value?: any
74
- prev?: ArgsClass<T, K>
75
- fieldSchema?: BasedSchemaField
76
- typeSchema?: BasedSchemaType
77
- skipCollection?: boolean
78
- collect?: (args: ArgsClass<T, K>, value?: any) => any
79
- }
80
-
81
- // Add asyncOperations // requiresAsyncValidation -> asyncOperation: () => {}