effect-app 4.0.0-beta.21 → 4.0.0-beta.211
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.
- package/CHANGELOG.md +954 -0
- package/dist/Array.d.ts +1 -1
- package/dist/Chunk.d.ts +1 -1
- package/dist/Chunk.d.ts.map +1 -1
- package/dist/Config/SecretURL.d.ts +1 -1
- package/dist/Config/SecretURL.d.ts.map +1 -1
- package/dist/Config/SecretURL.js +2 -2
- package/dist/Config/internal/configSecretURL.d.ts +1 -1
- package/dist/Config/internal/configSecretURL.d.ts.map +1 -1
- package/dist/Config.d.ts +7 -0
- package/dist/Config.d.ts.map +1 -0
- package/dist/Config.js +6 -0
- package/dist/ConfigProvider.d.ts +39 -0
- package/dist/ConfigProvider.d.ts.map +1 -0
- package/dist/ConfigProvider.js +42 -0
- package/dist/Context.d.ts +40 -0
- package/dist/Context.d.ts.map +1 -0
- package/dist/Context.js +67 -0
- package/dist/Effect.d.ts +9 -10
- package/dist/Effect.d.ts.map +1 -1
- package/dist/Effect.js +3 -6
- package/dist/Function.d.ts +1 -1
- package/dist/Function.d.ts.map +1 -1
- package/dist/Inputify.type.d.ts +1 -1
- package/dist/Layer.d.ts +7 -6
- package/dist/Layer.d.ts.map +1 -1
- package/dist/Layer.js +1 -1
- package/dist/NonEmptySet.d.ts +1 -1
- package/dist/NonEmptySet.d.ts.map +1 -1
- package/dist/Option.d.ts +1 -1
- package/dist/Option.d.ts.map +1 -1
- package/dist/Pure.d.ts +5 -5
- package/dist/Pure.d.ts.map +1 -1
- package/dist/Pure.js +13 -13
- package/dist/Schema/Class.d.ts +66 -20
- package/dist/Schema/Class.d.ts.map +1 -1
- package/dist/Schema/Class.js +189 -22
- package/dist/Schema/FastCheck.d.ts +1 -1
- package/dist/Schema/FastCheck.d.ts.map +1 -1
- package/dist/Schema/Methods.d.ts +1 -1
- package/dist/Schema/SchemaParser.d.ts +5 -0
- package/dist/Schema/SchemaParser.d.ts.map +1 -0
- package/dist/Schema/SchemaParser.js +6 -0
- package/dist/Schema/SpecialJsonSchema.d.ts +33 -0
- package/dist/Schema/SpecialJsonSchema.d.ts.map +1 -0
- package/dist/Schema/SpecialJsonSchema.js +122 -0
- package/dist/Schema/SpecialOpenApi.d.ts +32 -0
- package/dist/Schema/SpecialOpenApi.d.ts.map +1 -0
- package/dist/Schema/SpecialOpenApi.js +123 -0
- package/dist/Schema/brand.d.ts +4 -2
- package/dist/Schema/brand.d.ts.map +1 -1
- package/dist/Schema/brand.js +1 -1
- package/dist/Schema/email.d.ts +1 -1
- package/dist/Schema/email.d.ts.map +1 -1
- package/dist/Schema/email.js +7 -4
- package/dist/Schema/ext.d.ts +117 -45
- package/dist/Schema/ext.d.ts.map +1 -1
- package/dist/Schema/ext.js +131 -42
- package/dist/Schema/moreStrings.d.ts +37 -25
- package/dist/Schema/moreStrings.d.ts.map +1 -1
- package/dist/Schema/moreStrings.js +15 -16
- package/dist/Schema/numbers.d.ts +15 -15
- package/dist/Schema/numbers.d.ts.map +1 -1
- package/dist/Schema/numbers.js +10 -12
- package/dist/Schema/phoneNumber.d.ts +1 -1
- package/dist/Schema/phoneNumber.d.ts.map +1 -1
- package/dist/Schema/phoneNumber.js +6 -3
- package/dist/Schema/schema.d.ts +1 -1
- package/dist/Schema/strings.d.ts +5 -5
- package/dist/Schema/strings.d.ts.map +1 -1
- package/dist/Schema/strings.js +1 -5
- package/dist/Schema.d.ts +147 -15
- package/dist/Schema.d.ts.map +1 -1
- package/dist/Schema.js +131 -16
- package/dist/Set.d.ts +1 -1
- package/dist/Set.d.ts.map +1 -1
- package/dist/TypeTest.d.ts +1 -1
- package/dist/Types.d.ts +1 -1
- package/dist/Widen.type.d.ts +1 -1
- package/dist/_ext/Array.d.ts +1 -1
- package/dist/_ext/Array.d.ts.map +1 -1
- package/dist/_ext/date.d.ts +1 -1
- package/dist/_ext/misc.d.ts +1 -1
- package/dist/_ext/ord.ext.d.ts +1 -1
- package/dist/_ext/ord.ext.d.ts.map +1 -1
- package/dist/builtin.d.ts +1 -1
- package/dist/builtin.d.ts.map +1 -1
- package/dist/client/InvalidationKeys.d.ts +29 -0
- package/dist/client/InvalidationKeys.d.ts.map +1 -0
- package/dist/client/InvalidationKeys.js +33 -0
- package/dist/client/apiClientFactory.d.ts +20 -32
- package/dist/client/apiClientFactory.d.ts.map +1 -1
- package/dist/client/apiClientFactory.js +95 -32
- package/dist/client/clientFor.d.ts +51 -17
- package/dist/client/clientFor.d.ts.map +1 -1
- package/dist/client/clientFor.js +9 -1
- package/dist/client/errors.d.ts +49 -25
- package/dist/client/errors.d.ts.map +1 -1
- package/dist/client/errors.js +43 -17
- package/dist/client/makeClient.d.ts +481 -33
- package/dist/client/makeClient.d.ts.map +1 -1
- package/dist/client/makeClient.js +66 -24
- package/dist/client.d.ts +2 -1
- package/dist/client.d.ts.map +1 -1
- package/dist/client.js +2 -1
- package/dist/faker.d.ts +1 -1
- package/dist/faker.d.ts.map +1 -1
- package/dist/http/Request.d.ts +2 -2
- package/dist/http/Request.d.ts.map +1 -1
- package/dist/http/internal/lib.d.ts +1 -1
- package/dist/http.d.ts +1 -1
- package/dist/ids.d.ts +12 -12
- package/dist/ids.d.ts.map +1 -1
- package/dist/ids.js +3 -2
- package/dist/index.d.ts +5 -8
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +6 -8
- package/dist/logger.d.ts +1 -1
- package/dist/middleware.d.ts +14 -8
- package/dist/middleware.d.ts.map +1 -1
- package/dist/middleware.js +14 -8
- package/dist/rpc/Invalidation.d.ts +402 -0
- package/dist/rpc/Invalidation.d.ts.map +1 -0
- package/dist/rpc/Invalidation.js +150 -0
- package/dist/rpc/MiddlewareMaker.d.ts +5 -4
- package/dist/rpc/MiddlewareMaker.d.ts.map +1 -1
- package/dist/rpc/MiddlewareMaker.js +57 -37
- package/dist/rpc/RpcContextMap.d.ts +3 -3
- package/dist/rpc/RpcContextMap.d.ts.map +1 -1
- package/dist/rpc/RpcContextMap.js +4 -4
- package/dist/rpc/RpcMiddleware.d.ts +5 -4
- package/dist/rpc/RpcMiddleware.d.ts.map +1 -1
- package/dist/rpc/RpcMiddleware.js +1 -1
- package/dist/rpc.d.ts +2 -2
- package/dist/rpc.d.ts.map +1 -1
- package/dist/rpc.js +2 -2
- package/dist/transform.d.ts +1 -1
- package/dist/transform.d.ts.map +1 -1
- package/dist/transform.js +3 -3
- package/dist/utils/effectify.d.ts +1 -1
- package/dist/utils/extend.d.ts +1 -1
- package/dist/utils/extend.d.ts.map +1 -1
- package/dist/utils/gen.d.ts +2 -2
- package/dist/utils/gen.d.ts.map +1 -1
- package/dist/utils/logLevel.d.ts +2 -2
- package/dist/utils/logLevel.d.ts.map +1 -1
- package/dist/utils/logger.d.ts +3 -3
- package/dist/utils/logger.d.ts.map +1 -1
- package/dist/utils/logger.js +3 -3
- package/dist/utils.d.ts +31 -38
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +12 -25
- package/dist/validation/validators.d.ts +1 -1
- package/dist/validation/validators.d.ts.map +1 -1
- package/dist/validation.d.ts +1 -1
- package/dist/validation.d.ts.map +1 -1
- package/package.json +46 -24
- package/src/Config/SecretURL.ts +2 -1
- package/src/Config.ts +14 -0
- package/src/ConfigProvider.ts +48 -0
- package/src/{ServiceMap.ts → Context.ts} +52 -59
- package/src/Effect.ts +12 -14
- package/src/Layer.ts +6 -5
- package/src/Pure.ts +17 -18
- package/src/Schema/Class.ts +268 -62
- package/src/Schema/SchemaParser.ts +12 -0
- package/src/Schema/SpecialJsonSchema.ts +137 -0
- package/src/Schema/SpecialOpenApi.ts +130 -0
- package/src/Schema/brand.ts +21 -1
- package/src/Schema/email.ts +7 -2
- package/src/Schema/ext.ts +204 -72
- package/src/Schema/moreStrings.ts +40 -37
- package/src/Schema/numbers.ts +14 -16
- package/src/Schema/phoneNumber.ts +5 -1
- package/src/Schema/strings.ts +4 -8
- package/src/Schema.ts +314 -20
- package/src/client/InvalidationKeys.ts +50 -0
- package/src/client/apiClientFactory.ts +223 -129
- package/src/client/clientFor.ts +95 -29
- package/src/client/errors.ts +52 -26
- package/src/client/makeClient.ts +572 -71
- package/src/client.ts +1 -0
- package/src/ids.ts +3 -2
- package/src/index.ts +5 -10
- package/src/middleware.ts +13 -9
- package/src/rpc/Invalidation.ts +226 -0
- package/src/rpc/MiddlewareMaker.ts +65 -60
- package/src/rpc/README.md +2 -2
- package/src/rpc/RpcContextMap.ts +6 -5
- package/src/rpc/RpcMiddleware.ts +5 -4
- package/src/rpc.ts +1 -1
- package/src/transform.ts +2 -2
- package/src/utils/gen.ts +1 -1
- package/src/utils/logger.ts +2 -2
- package/src/utils.ts +50 -132
- package/test/dist/rpc.test.d.ts.map +1 -1
- package/test/dist/secretURL.test.d.ts.map +1 -0
- package/test/dist/special.test.d.ts.map +1 -0
- package/test/dist/stream-error.types.d.ts +2 -0
- package/test/dist/stream-error.types.d.ts.map +1 -0
- package/test/dist/stream-error.types.js +27 -0
- package/test/rpc.test.ts +45 -6
- package/test/schema.test.ts +581 -7
- package/test/secretURL.test.ts +157 -0
- package/test/special.test.ts +1023 -0
- package/test/utils.test.ts +6 -6
- package/tsconfig.base.json +3 -4
- package/tsconfig.json +0 -1
- package/tsconfig.json.bak +2 -2
- package/tsconfig.src.json +29 -29
- package/tsconfig.test.json +2 -2
- package/dist/Operations.d.ts +0 -123
- package/dist/Operations.d.ts.map +0 -1
- package/dist/Operations.js +0 -29
- package/dist/ServiceMap.d.ts +0 -44
- package/dist/ServiceMap.d.ts.map +0 -1
- package/dist/ServiceMap.js +0 -91
- package/eslint.config.mjs +0 -26
- package/src/Operations.ts +0 -55
package/src/Schema/Class.ts
CHANGED
|
@@ -1,101 +1,307 @@
|
|
|
1
1
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
2
|
-
import {
|
|
3
|
-
import type { Struct } from "effect/Schema"
|
|
2
|
+
import { type Cause, Effect, Option, Schema, SchemaAST, SchemaIssue } from "effect"
|
|
4
3
|
import * as S from "effect/Schema"
|
|
4
|
+
import { copyOrigin } from "../utils.js"
|
|
5
|
+
import { concurrencyUnbounded } from "./ext.js"
|
|
6
|
+
import * as SchemaParser from "./SchemaParser.js"
|
|
5
7
|
|
|
6
8
|
type ClassAnnotations<Self> = S.Annotations.Declaration<Self, readonly [any]>
|
|
7
9
|
|
|
8
|
-
export interface EnhancedClass<Self, SchemaS extends S.Top & { readonly fields: Struct.Fields }, Inherited>
|
|
9
|
-
extends S.Class<Self, SchemaS, Inherited
|
|
10
|
+
export interface EnhancedClass<Self, SchemaS extends S.Top & { readonly fields: S.Struct.Fields }, Inherited>
|
|
11
|
+
extends S.Class<Self, SchemaS, Inherited>
|
|
10
12
|
{
|
|
13
|
+
/**
|
|
14
|
+
* See `copyOrigin` docs in `utils.ts` for return-type design details.
|
|
15
|
+
*/
|
|
16
|
+
readonly copy: ReturnType<typeof copyOrigin<new(_: any) => Self>>
|
|
11
17
|
}
|
|
12
18
|
type MissingSelfGeneric<Usage extends string, Params extends string = ""> =
|
|
13
19
|
`Missing \`Self\` generic - use \`class Self extends ${Usage}<Self>()(${Params}{ ... })\``
|
|
14
20
|
|
|
15
|
-
|
|
16
|
-
// include: <NewProps extends S.Struct.Fields>(
|
|
17
|
-
// fnc: (fields: Fields) => NewProps
|
|
18
|
-
// ) => NewProps
|
|
19
|
-
pick: <P extends keyof Fields>(...keys: readonly P[]) => Pick<Fields, P>
|
|
20
|
-
omit: <P extends keyof Fields>(...keys: readonly P[]) => Omit<Fields, P>
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
type HasFields<Fields extends Struct.Fields> = {
|
|
21
|
+
type HasFields<Fields extends S.Struct.Fields> = {
|
|
24
22
|
readonly fields: Fields
|
|
25
23
|
} | {
|
|
26
24
|
readonly from: HasFields<Fields>
|
|
27
25
|
}
|
|
28
26
|
|
|
29
|
-
|
|
27
|
+
type ClassOptions = {
|
|
28
|
+
readonly strict?: boolean
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export declare const ExtendedSchemaNoEncoded: unique symbol
|
|
32
|
+
|
|
33
|
+
export type ExtendedSchemaNoEncoded = typeof ExtendedSchemaNoEncoded
|
|
34
|
+
|
|
35
|
+
type WithEncoded<SchemaS extends S.Top, Encoded> = Omit<SchemaS, "Encoded"> & { readonly Encoded: Encoded }
|
|
36
|
+
|
|
37
|
+
type ExtendedSchema<SchemaS extends S.Top, Encoded> = [Encoded] extends [ExtendedSchemaNoEncoded] ? SchemaS
|
|
38
|
+
: WithEncoded<SchemaS, Encoded>
|
|
39
|
+
|
|
40
|
+
export type Class<Self, S extends S.Top & { readonly fields: S.Struct.Fields }, Inherited> = EnhancedClass<
|
|
41
|
+
Self,
|
|
42
|
+
S,
|
|
43
|
+
Inherited
|
|
44
|
+
>
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Build a modified Declaration that accepts struct-matching values during
|
|
48
|
+
* encoding, given the original Declaration and the class's fields.
|
|
49
|
+
*/
|
|
50
|
+
function makeRelaxedDeclaration(
|
|
51
|
+
ast: SchemaAST.Declaration,
|
|
52
|
+
fields: Schema.Struct.Fields,
|
|
53
|
+
cls: any
|
|
54
|
+
): SchemaAST.Declaration {
|
|
55
|
+
const parseOptions = ast.annotations?.["parseOptions"] as SchemaAST.ParseOptions | undefined
|
|
56
|
+
const structSchema = Schema.Struct(fields)
|
|
57
|
+
const annotatedStruct = parseOptions ? S.toType(structSchema).annotate({ parseOptions }) : S.toType(structSchema)
|
|
58
|
+
const decodeStruct = SchemaParser.decodeUnknownEffect(annotatedStruct)
|
|
59
|
+
|
|
60
|
+
return new SchemaAST.Declaration(
|
|
61
|
+
ast.typeParameters,
|
|
62
|
+
() => (input: unknown, self: SchemaAST.Declaration, options: SchemaAST.ParseOptions) => {
|
|
63
|
+
if (input instanceof cls) {
|
|
64
|
+
return Effect.succeed(input)
|
|
65
|
+
}
|
|
66
|
+
if (input !== null && typeof input === "object") {
|
|
67
|
+
return decodeStruct(input, options)
|
|
68
|
+
}
|
|
69
|
+
return Effect.fail(new SchemaIssue.InvalidType(self, Option.some(input)))
|
|
70
|
+
},
|
|
71
|
+
ast.annotations,
|
|
72
|
+
ast.checks,
|
|
73
|
+
ast.encoding,
|
|
74
|
+
ast.context
|
|
75
|
+
)
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// ---------------------------------------------------------------------------
|
|
79
|
+
// Class — like Schema.Class but with relaxed encoding
|
|
80
|
+
// ---------------------------------------------------------------------------
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Like `Schema.Class`, but the resulting class accepts plain objects matching
|
|
84
|
+
* the struct schema during encoding — not only `instanceof` or type-id
|
|
85
|
+
* checks.
|
|
86
|
+
*
|
|
87
|
+
* @example
|
|
88
|
+
* ```ts
|
|
89
|
+
* import { Schema } from "effect"
|
|
90
|
+
* import { Class } from "./Class.js"
|
|
91
|
+
*
|
|
92
|
+
* class A extends Class<A>("A")({ a: Schema.String }) {}
|
|
93
|
+
*
|
|
94
|
+
* // Construction works as normal:
|
|
95
|
+
* new A({ a: "hello" })
|
|
96
|
+
*
|
|
97
|
+
* // Encoding accepts plain objects:
|
|
98
|
+
* Schema.encodeUnknownSync(A)({ a: "hello" }) // { a: "hello" }
|
|
99
|
+
* ```
|
|
100
|
+
*/
|
|
101
|
+
export const Class: <Self = never, Encoded = ExtendedSchemaNoEncoded>(
|
|
102
|
+
identifier: string
|
|
103
|
+
) => <Fields extends S.Struct.Fields>(
|
|
30
104
|
fieldsOr: Fields | HasFields<Fields>,
|
|
31
|
-
annotations?: ClassAnnotations<Self
|
|
105
|
+
annotations?: ClassAnnotations<Self>,
|
|
106
|
+
options?: ClassOptions
|
|
32
107
|
) => [Self] extends [never] ? MissingSelfGeneric<"Class">
|
|
33
108
|
: EnhancedClass<
|
|
34
109
|
Self,
|
|
35
|
-
S.Struct<Fields>,
|
|
110
|
+
ExtendedSchema<S.Struct<Fields>, Encoded>,
|
|
36
111
|
{}
|
|
37
|
-
> = (identifier) => (fields, annotations) => {
|
|
38
|
-
const
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
112
|
+
> = (identifier) => (fields, annotations, options) => {
|
|
113
|
+
const relaxed = options?.strict === false
|
|
114
|
+
// Build the original Schema.Class
|
|
115
|
+
const Base = (S.Class as any)(identifier)(fields, { ...concurrencyUnbounded, ...annotations })
|
|
116
|
+
// Get the original ast getter from the base class
|
|
117
|
+
const originalAstDescriptor = Object.getOwnPropertyDescriptor(Base, "ast")!
|
|
118
|
+
|
|
119
|
+
// Cache per-class to avoid recomputing
|
|
120
|
+
const astCache = new WeakMap<any, SchemaAST.Declaration>()
|
|
121
|
+
const copyCache = new WeakMap<any, ReturnType<typeof copyOrigin>>()
|
|
122
|
+
|
|
123
|
+
return class extends Base {
|
|
124
|
+
static get copy() {
|
|
125
|
+
let cached = copyCache.get(this)
|
|
126
|
+
if (cached === undefined) {
|
|
127
|
+
cached = copyOrigin(this)
|
|
128
|
+
copyCache.set(this, cached)
|
|
129
|
+
}
|
|
130
|
+
return cached
|
|
131
|
+
}
|
|
132
|
+
static get ast(): SchemaAST.Declaration {
|
|
133
|
+
let cached = astCache.get(this)
|
|
134
|
+
if (cached !== undefined) return cached
|
|
135
|
+
// Call the original getter with `this` bound to the actual user class,
|
|
136
|
+
// so getClassSchema(this) creates a schema that uses `new this(...)`.
|
|
137
|
+
const originalAst = originalAstDescriptor.get!.call(this) as SchemaAST.Declaration
|
|
138
|
+
cached = relaxed ? makeRelaxedDeclaration(originalAst, Base.fields, this) : originalAst
|
|
139
|
+
astCache.set(this, cached)
|
|
140
|
+
return cached
|
|
141
|
+
}
|
|
142
|
+
static mapFields(f: any, options?: any) {
|
|
143
|
+
return Base.mapFields(f, options).annotate(concurrencyUnbounded)
|
|
42
144
|
}
|
|
43
|
-
// static readonly include = include(fields)
|
|
44
|
-
static readonly pick = (...selection: any[]) => pipe(this["fields"], Struct2.pick(selection))
|
|
45
|
-
static readonly omit = (...selection: any[]) => pipe(this["fields"], Struct2.omit(selection))
|
|
46
145
|
} as any
|
|
47
146
|
}
|
|
48
147
|
|
|
49
|
-
|
|
148
|
+
// ---------------------------------------------------------------------------
|
|
149
|
+
// TaggedClass — like Schema.TaggedClass but with relaxed encoding
|
|
150
|
+
// ---------------------------------------------------------------------------
|
|
151
|
+
|
|
152
|
+
/**
|
|
153
|
+
* Like `Schema.TaggedClass`, but the resulting class accepts plain objects
|
|
154
|
+
* matching the struct schema during encoding.
|
|
155
|
+
*
|
|
156
|
+
* @example
|
|
157
|
+
* ```ts
|
|
158
|
+
* import { Schema } from "effect"
|
|
159
|
+
* import { TaggedClass } from "./Class.js"
|
|
160
|
+
*
|
|
161
|
+
* class Circle extends TaggedClass<Circle>()("Circle", {
|
|
162
|
+
* radius: Schema.Number
|
|
163
|
+
* }) {}
|
|
164
|
+
*
|
|
165
|
+
* Schema.encodeUnknownSync(Circle)({ _tag: "Circle", radius: 5 })
|
|
166
|
+
* ```
|
|
167
|
+
*/
|
|
168
|
+
export const TaggedClass: <Self = never, Encoded = ExtendedSchemaNoEncoded>(
|
|
169
|
+
identifier?: string
|
|
170
|
+
) => <Tag extends string, Fields extends S.Struct.Fields>(
|
|
50
171
|
tag: Tag,
|
|
51
172
|
fieldsOr: Fields | HasFields<Fields>,
|
|
52
|
-
annotations?: ClassAnnotations<Self
|
|
53
|
-
|
|
173
|
+
annotations?: ClassAnnotations<Self>,
|
|
174
|
+
options?: ClassOptions
|
|
175
|
+
) => [Self] extends [never] ? MissingSelfGeneric<"TaggedClass">
|
|
54
176
|
: EnhancedClass<
|
|
55
177
|
Self,
|
|
56
|
-
S.Struct<{ readonly _tag: S.tag<Tag> } & Fields>,
|
|
178
|
+
ExtendedSchema<S.Struct<{ readonly _tag: S.tag<Tag> } & Fields>, Encoded>,
|
|
57
179
|
{}
|
|
58
|
-
> = (identifier) => (tag, fields, annotations) => {
|
|
59
|
-
const
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
180
|
+
> = (identifier) => (tag, fields, annotations, options) => {
|
|
181
|
+
const relaxed = options?.strict === false
|
|
182
|
+
const Base = (S.TaggedClass as any)(identifier)(tag, fields, { ...concurrencyUnbounded, ...annotations })
|
|
183
|
+
const originalAstDescriptor = Object.getOwnPropertyDescriptor(Base, "ast")!
|
|
184
|
+
const astCache = new WeakMap<any, SchemaAST.Declaration>()
|
|
185
|
+
const copyCache = new WeakMap<any, ReturnType<typeof copyOrigin>>()
|
|
186
|
+
|
|
187
|
+
return class extends Base {
|
|
188
|
+
static get copy() {
|
|
189
|
+
let cached = copyCache.get(this)
|
|
190
|
+
if (cached === undefined) {
|
|
191
|
+
cached = copyOrigin(this)
|
|
192
|
+
copyCache.set(this, cached)
|
|
193
|
+
}
|
|
194
|
+
return cached
|
|
195
|
+
}
|
|
196
|
+
static get ast(): SchemaAST.Declaration {
|
|
197
|
+
let cached = astCache.get(this)
|
|
198
|
+
if (cached !== undefined) return cached
|
|
199
|
+
const originalAst = originalAstDescriptor.get!.call(this) as SchemaAST.Declaration
|
|
200
|
+
cached = relaxed ? makeRelaxedDeclaration(originalAst, Base.fields, this) : originalAst
|
|
201
|
+
astCache.set(this, cached)
|
|
202
|
+
return cached
|
|
203
|
+
}
|
|
204
|
+
static mapFields(f: any, options?: any) {
|
|
205
|
+
return Base.mapFields(f, options).annotate(concurrencyUnbounded)
|
|
63
206
|
}
|
|
64
|
-
// static readonly include = include(fields)
|
|
65
|
-
static readonly pick = (...selection: any[]) => pipe(this["fields"], Struct2.pick(selection))
|
|
66
|
-
static readonly omit = (...selection: any[]) => pipe(this["fields"], Struct2.omit(selection))
|
|
67
207
|
} as any
|
|
68
208
|
}
|
|
69
209
|
|
|
70
|
-
|
|
210
|
+
// ---------------------------------------------------------------------------
|
|
211
|
+
// ErrorClass — like Schema.ErrorClass but with relaxed encoding
|
|
212
|
+
// ---------------------------------------------------------------------------
|
|
213
|
+
|
|
214
|
+
export const ErrorClass: <Self = never, Encoded = ExtendedSchemaNoEncoded, Brand = {}>(
|
|
215
|
+
identifier: string
|
|
216
|
+
) => <Fields extends S.Struct.Fields>(
|
|
71
217
|
fieldsOr: Fields | HasFields<Fields>,
|
|
72
|
-
annotations?: ClassAnnotations<Self
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
{
|
|
87
|
-
|
|
88
|
-
|
|
218
|
+
annotations?: ClassAnnotations<Self>,
|
|
219
|
+
options?: ClassOptions
|
|
220
|
+
) => [Self] extends [never] ? MissingSelfGeneric<"ErrorClass">
|
|
221
|
+
: EnhancedClass<
|
|
222
|
+
Self,
|
|
223
|
+
ExtendedSchema<S.Struct<Fields>, Encoded>,
|
|
224
|
+
Cause.YieldableError & Brand
|
|
225
|
+
> = (identifier) => (fields, annotations, options) => {
|
|
226
|
+
const relaxed = options?.strict === false
|
|
227
|
+
const Base = (S.ErrorClass as any)(identifier)(fields, { ...concurrencyUnbounded, ...annotations })
|
|
228
|
+
const originalAstDescriptor = Object.getOwnPropertyDescriptor(Base, "ast")!
|
|
229
|
+
const astCache = new WeakMap<any, SchemaAST.Declaration>()
|
|
230
|
+
const copyCache = new WeakMap<any, ReturnType<typeof copyOrigin>>()
|
|
231
|
+
|
|
232
|
+
return class extends Base {
|
|
233
|
+
static get copy() {
|
|
234
|
+
let cached = copyCache.get(this)
|
|
235
|
+
if (cached === undefined) {
|
|
236
|
+
cached = copyOrigin(this)
|
|
237
|
+
copyCache.set(this, cached)
|
|
238
|
+
}
|
|
239
|
+
return cached
|
|
240
|
+
}
|
|
241
|
+
static get ast(): SchemaAST.Declaration {
|
|
242
|
+
let cached = astCache.get(this)
|
|
243
|
+
if (cached !== undefined) return cached
|
|
244
|
+
const originalAst = originalAstDescriptor.get!.call(this) as SchemaAST.Declaration
|
|
245
|
+
cached = relaxed ? makeRelaxedDeclaration(originalAst, Base.fields, this) : originalAst
|
|
246
|
+
astCache.set(this, cached)
|
|
247
|
+
return cached
|
|
248
|
+
}
|
|
249
|
+
static mapFields(f: any, options?: any) {
|
|
250
|
+
return Base.mapFields(f, options).annotate(concurrencyUnbounded)
|
|
251
|
+
}
|
|
252
|
+
} as any
|
|
253
|
+
}
|
|
89
254
|
|
|
90
|
-
|
|
255
|
+
// ---------------------------------------------------------------------------
|
|
256
|
+
// TaggedErrorClass — like Schema.TaggedErrorClass but with relaxed encoding
|
|
257
|
+
// ---------------------------------------------------------------------------
|
|
258
|
+
|
|
259
|
+
export const TaggedErrorClass: <Self = never, Encoded = ExtendedSchemaNoEncoded, Brand = {}>(
|
|
91
260
|
identifier?: string
|
|
92
261
|
) => <Tag extends string, Fields extends S.Struct.Fields>(
|
|
93
262
|
tag: Tag,
|
|
94
263
|
fieldsOr: Fields | HasFields<Fields>,
|
|
95
|
-
annotations?: ClassAnnotations<Self
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
264
|
+
annotations?: ClassAnnotations<Self>,
|
|
265
|
+
options?: ClassOptions
|
|
266
|
+
) => [Self] extends [never] ? MissingSelfGeneric<"TaggedErrorClass">
|
|
267
|
+
: EnhancedClass<
|
|
268
|
+
Self,
|
|
269
|
+
ExtendedSchema<S.Struct<{ readonly _tag: S.tag<Tag> } & Fields>, Encoded>,
|
|
270
|
+
Cause.YieldableError & Brand
|
|
271
|
+
> = (identifier) => (tag, fields, annotations, options) => {
|
|
272
|
+
const relaxed = options?.strict === false
|
|
273
|
+
const Base = (S.TaggedErrorClass as any)(identifier)(tag, fields, { ...concurrencyUnbounded, ...annotations })
|
|
274
|
+
const originalAstDescriptor = Object.getOwnPropertyDescriptor(Base, "ast")!
|
|
275
|
+
const astCache = new WeakMap<any, SchemaAST.Declaration>()
|
|
276
|
+
const copyCache = new WeakMap<any, ReturnType<typeof copyOrigin>>()
|
|
277
|
+
|
|
278
|
+
return class extends Base {
|
|
279
|
+
static get copy() {
|
|
280
|
+
let cached = copyCache.get(this)
|
|
281
|
+
if (cached === undefined) {
|
|
282
|
+
cached = copyOrigin(this)
|
|
283
|
+
copyCache.set(this, cached)
|
|
284
|
+
}
|
|
285
|
+
return cached
|
|
286
|
+
}
|
|
287
|
+
static get ast(): SchemaAST.Declaration {
|
|
288
|
+
let cached = astCache.get(this)
|
|
289
|
+
if (cached !== undefined) return cached
|
|
290
|
+
const originalAst = originalAstDescriptor.get!.call(this) as SchemaAST.Declaration
|
|
291
|
+
cached = relaxed ? makeRelaxedDeclaration(originalAst, Base.fields, this) : originalAst
|
|
292
|
+
astCache.set(this, cached)
|
|
293
|
+
return cached
|
|
294
|
+
}
|
|
295
|
+
static mapFields(f: any, options?: any) {
|
|
296
|
+
return Base.mapFields(f, options).annotate(concurrencyUnbounded)
|
|
297
|
+
}
|
|
298
|
+
} as any
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
export interface Opaque<Self, Encoded, SchemaS extends S.Top, Brand>
|
|
302
|
+
extends S.Opaque<Self, ExtendedSchema<SchemaS, Encoded>, Brand>
|
|
303
|
+
{}
|
|
304
|
+
|
|
305
|
+
export const Opaque: <Self, Encoded = ExtendedSchemaNoEncoded, Brand = {}>() => <S extends S.Top>(
|
|
306
|
+
schema: S
|
|
307
|
+
) => Opaque<Self, Encoded, S, Brand> & Omit<S, keyof S.Top> = S.Opaque as any
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import * as SchemaParser from "effect/SchemaParser"
|
|
2
|
+
import { withDefaultParseOptions } from "./ext.js"
|
|
3
|
+
|
|
4
|
+
export * from "effect/SchemaParser"
|
|
5
|
+
|
|
6
|
+
export const decodeEffectConcurrently: typeof SchemaParser.decodeEffect = withDefaultParseOptions(
|
|
7
|
+
SchemaParser.decodeEffect
|
|
8
|
+
)
|
|
9
|
+
|
|
10
|
+
export const decodeUnknownEffectConcurrently: typeof SchemaParser.decodeUnknownEffect = withDefaultParseOptions(
|
|
11
|
+
SchemaParser.decodeUnknownEffect
|
|
12
|
+
)
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SpecialJsonSchema — A variant of Schema.toJsonSchemaDocument that
|
|
3
|
+
* post-processes the output (e.g. flattens simple allOf).
|
|
4
|
+
*/
|
|
5
|
+
import { type JsonSchema, type Schema, SchemaRepresentation } from "effect"
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Converts a schema to a JSON Schema Document (draft-2020-12), with
|
|
9
|
+
* post-processing that flattens simple allOf entries.
|
|
10
|
+
*/
|
|
11
|
+
export function specialJsonSchemaDocument(
|
|
12
|
+
schema: Schema.Top,
|
|
13
|
+
options?: Schema.ToJsonSchemaOptions
|
|
14
|
+
): JsonSchema.Document<"draft-2020-12"> {
|
|
15
|
+
const doc = SchemaRepresentation.fromAST(schema.ast)
|
|
16
|
+
const jd = SchemaRepresentation.toJsonSchemaDocument(doc, options)
|
|
17
|
+
const processedDefs: JsonSchema.Definitions = {}
|
|
18
|
+
for (const [key, def] of Object.entries(jd.definitions)) {
|
|
19
|
+
processedDefs[key] = postProcessJsonSchema(def)
|
|
20
|
+
}
|
|
21
|
+
return {
|
|
22
|
+
dialect: "draft-2020-12",
|
|
23
|
+
schema: postProcessJsonSchema(jd.schema),
|
|
24
|
+
definitions: processedDefs
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Flattens `allOf` entries into the parent when the parent already has a
|
|
30
|
+
* `type` and every `allOf` entry is a plain constraint object (no `$ref`,
|
|
31
|
+
* no `type`). Merged properties from `allOf` entries win on conflict.
|
|
32
|
+
*/
|
|
33
|
+
export function flattenSimpleAllOf(obj: unknown): unknown {
|
|
34
|
+
if (obj === null || typeof obj !== "object") return obj
|
|
35
|
+
|
|
36
|
+
if (globalThis.Array.isArray(obj)) {
|
|
37
|
+
return obj.map(flattenSimpleAllOf)
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const record = obj as Record<string, unknown>
|
|
41
|
+
const result: Record<string, unknown> = {}
|
|
42
|
+
for (const [key, value] of Object.entries(record)) {
|
|
43
|
+
result[key] = flattenSimpleAllOf(value)
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
if (result["type"] && globalThis.Array.isArray(result["allOf"])) {
|
|
47
|
+
const allOf = result["allOf"] as Array<Record<string, unknown>>
|
|
48
|
+
const canFlatten = allOf.every((entry) =>
|
|
49
|
+
typeof entry === "object" && entry !== null && !("$ref" in entry) && !("type" in entry)
|
|
50
|
+
)
|
|
51
|
+
if (canFlatten) {
|
|
52
|
+
const { allOf: _, ...rest } = result
|
|
53
|
+
const merged: Record<string, unknown> = { ...rest }
|
|
54
|
+
for (const entry of allOf) {
|
|
55
|
+
Object.assign(merged, entry)
|
|
56
|
+
}
|
|
57
|
+
return merged
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
return result
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Recursively removes `additionalProperties: false` from JSON Schema objects.
|
|
66
|
+
* Only removes when the value is exactly `false` -- other values are left intact.
|
|
67
|
+
*/
|
|
68
|
+
export function removeAdditionalPropertiesFalse(obj: unknown): unknown {
|
|
69
|
+
if (obj === null || typeof obj !== "object") return obj
|
|
70
|
+
|
|
71
|
+
if (globalThis.Array.isArray(obj)) {
|
|
72
|
+
return obj.map(removeAdditionalPropertiesFalse)
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
const record = obj as Record<string, unknown>
|
|
76
|
+
const result: Record<string, unknown> = {}
|
|
77
|
+
for (const [key, value] of Object.entries(record)) {
|
|
78
|
+
if (key === "additionalProperties" && value === false) continue
|
|
79
|
+
result[key] = removeAdditionalPropertiesFalse(value)
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
return result
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
/**
|
|
86
|
+
* Flattens nested `anyOf` entries: if an anyOf entry is itself just `{ anyOf: [...] }`
|
|
87
|
+
* with no other keys, its children are inlined. If only one item remains, the anyOf
|
|
88
|
+
* wrapper is removed entirely.
|
|
89
|
+
*/
|
|
90
|
+
export function flattenNestedAnyOf(obj: unknown): unknown {
|
|
91
|
+
if (obj === null || typeof obj !== "object") return obj
|
|
92
|
+
if (globalThis.Array.isArray(obj)) return obj.map(flattenNestedAnyOf)
|
|
93
|
+
|
|
94
|
+
const record = obj as Record<string, unknown>
|
|
95
|
+
const result: Record<string, unknown> = {}
|
|
96
|
+
for (const [key, value] of Object.entries(record)) {
|
|
97
|
+
result[key] = flattenNestedAnyOf(value)
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
if (globalThis.Array.isArray(result["anyOf"])) {
|
|
101
|
+
const anyOf = result["anyOf"] as Array<unknown>
|
|
102
|
+
const flattened: Array<unknown> = []
|
|
103
|
+
for (const entry of anyOf) {
|
|
104
|
+
if (
|
|
105
|
+
typeof entry === "object"
|
|
106
|
+
&& entry !== null
|
|
107
|
+
&& !globalThis.Array.isArray(entry)
|
|
108
|
+
&& "anyOf" in entry
|
|
109
|
+
&& Object.keys(entry).length === 1
|
|
110
|
+
&& globalThis.Array.isArray((entry as Record<string, unknown>)["anyOf"])
|
|
111
|
+
) {
|
|
112
|
+
flattened.push(...(entry as Record<string, unknown>)["anyOf"] as Array<unknown>)
|
|
113
|
+
} else {
|
|
114
|
+
flattened.push(entry)
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
if (flattened.length === 1) {
|
|
118
|
+
const { anyOf: _, ...rest } = result
|
|
119
|
+
const single = flattened[0]
|
|
120
|
+
if (typeof single === "object" && single !== null && !globalThis.Array.isArray(single)) {
|
|
121
|
+
return { ...rest, ...single }
|
|
122
|
+
}
|
|
123
|
+
return single
|
|
124
|
+
}
|
|
125
|
+
result["anyOf"] = flattened
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
return result
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Applies JSON Schema post-processing: flattens simple allOf,
|
|
133
|
+
* flattens nested anyOf, then strips additionalProperties: false.
|
|
134
|
+
*/
|
|
135
|
+
export function postProcessJsonSchema(obj: JsonSchema.JsonSchema): JsonSchema.JsonSchema {
|
|
136
|
+
return removeAdditionalPropertiesFalse(flattenNestedAnyOf(flattenSimpleAllOf(obj))) as JsonSchema.JsonSchema
|
|
137
|
+
}
|
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SpecialOpenApi — Deduplicates `components/schemas` entries in an OpenAPI spec
|
|
3
|
+
* and applies JSON Schema post-processing (null removal, allOf flattening).
|
|
4
|
+
*
|
|
5
|
+
* When `OpenApi.fromApi` generates the spec, different AST nodes sharing the
|
|
6
|
+
* same identifier can produce duplicate entries (e.g. "X" and "X1") in
|
|
7
|
+
* `components.schemas`. This module provides a transform function that
|
|
8
|
+
* collapses those duplicates, rewrites all `$ref` pointers accordingly,
|
|
9
|
+
* and post-processes schemas for better codegen compatibility.
|
|
10
|
+
*
|
|
11
|
+
* Usage with the OpenApi `Transform` annotation:
|
|
12
|
+
*
|
|
13
|
+
* ```ts
|
|
14
|
+
* import { OpenApi } from "effect/unstable"
|
|
15
|
+
* import { deduplicateOpenApiSchemas } from "./SpecialOpenApi.js"
|
|
16
|
+
*
|
|
17
|
+
* const api = HttpApi.make("myApi")
|
|
18
|
+
* .pipe(HttpApi.annotateContext(OpenApi.annotations({ transform: deduplicateOpenApiSchemas })))
|
|
19
|
+
* ```
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
import { postProcessJsonSchema } from "./SpecialJsonSchema.js"
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Deduplicates `components.schemas` entries in an OpenAPI spec.
|
|
26
|
+
*
|
|
27
|
+
* Entries sharing the same base identifier (e.g. "X" and "X1") whose JSON
|
|
28
|
+
* representations are identical are collapsed into a single canonical entry,
|
|
29
|
+
* and all `$ref` pointers throughout the spec are rewritten to point to
|
|
30
|
+
* the canonical key.
|
|
31
|
+
*
|
|
32
|
+
* Designed to be used as the `transform` option in `OpenApi.annotations`.
|
|
33
|
+
*/
|
|
34
|
+
export function deduplicateOpenApiSchemas(
|
|
35
|
+
spec: Record<string, any>
|
|
36
|
+
): Record<string, any> {
|
|
37
|
+
const components = spec["components"] as Record<string, any> | undefined
|
|
38
|
+
if (!components) return spec
|
|
39
|
+
const schemas = components["schemas"] as Record<string, any> | undefined
|
|
40
|
+
if (!schemas) return spec
|
|
41
|
+
|
|
42
|
+
const keys = Object.keys(schemas)
|
|
43
|
+
if (keys.length === 0) return spec
|
|
44
|
+
|
|
45
|
+
// Group keys by base identifier (strip trailing digits)
|
|
46
|
+
const groups = new Map<string, Array<{ key: string; fingerprint: string }>>()
|
|
47
|
+
for (const key of keys) {
|
|
48
|
+
const base = getBaseIdentifier(key)
|
|
49
|
+
const fingerprint = JSON.stringify(schemas[key])
|
|
50
|
+
const group = groups.get(base)
|
|
51
|
+
if (group === undefined) {
|
|
52
|
+
groups.set(base, [{ key, fingerprint }])
|
|
53
|
+
} else {
|
|
54
|
+
group.push({ key, fingerprint })
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// Build remapping from duplicate keys to canonical keys
|
|
59
|
+
const remapping = new Map<string, string>()
|
|
60
|
+
for (const [, group] of groups) {
|
|
61
|
+
if (group.length <= 1) continue
|
|
62
|
+
const seen = new Map<string, string>() // fingerprint -> canonical key
|
|
63
|
+
for (const entry of group) {
|
|
64
|
+
const canonical = seen.get(entry.fingerprint)
|
|
65
|
+
if (canonical !== undefined) {
|
|
66
|
+
remapping.set(entry.key, canonical)
|
|
67
|
+
} else {
|
|
68
|
+
seen.set(entry.fingerprint, entry.key)
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
if (remapping.size === 0) return postProcessJsonSchema(spec)
|
|
74
|
+
|
|
75
|
+
// Build new schemas object without duplicates
|
|
76
|
+
const newSchemas: Record<string, any> = {}
|
|
77
|
+
for (const key of keys) {
|
|
78
|
+
if (!remapping.has(key)) {
|
|
79
|
+
newSchemas[key] = schemas[key]
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// Deep clone the spec, replace schemas, and rewrite all $ref pointers
|
|
84
|
+
const newSpec = structuredClone(spec)
|
|
85
|
+
newSpec["components"]["schemas"] = newSchemas
|
|
86
|
+
rewriteRefs(newSpec, remapping)
|
|
87
|
+
|
|
88
|
+
return postProcessJsonSchema(newSpec)
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
/**
|
|
92
|
+
* Extracts the base identifier from a schema key by stripping trailing
|
|
93
|
+
* digits appended by the gen() function.
|
|
94
|
+
* E.g. "X1" -> "X", "X" -> "X", "MyType2" -> "MyType"
|
|
95
|
+
*/
|
|
96
|
+
function getBaseIdentifier(key: string): string {
|
|
97
|
+
const match = key.match(/^(.+?)(\d+)$/)
|
|
98
|
+
return match ? match[1]! : key
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
/**
|
|
102
|
+
* Recursively rewrites `$ref` values in a JSON object tree.
|
|
103
|
+
* Mutates the object in-place (caller should pass a deep clone).
|
|
104
|
+
*/
|
|
105
|
+
function rewriteRefs(obj: any, remapping: Map<string, string>): void {
|
|
106
|
+
if (obj === null || typeof obj !== "object") return
|
|
107
|
+
|
|
108
|
+
if (Array.isArray(obj)) {
|
|
109
|
+
for (const item of obj) {
|
|
110
|
+
rewriteRefs(item, remapping)
|
|
111
|
+
}
|
|
112
|
+
return
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
if (typeof obj.$ref === "string") {
|
|
116
|
+
// OpenAPI refs look like "#/components/schemas/X1"
|
|
117
|
+
const prefix = "#/components/schemas/"
|
|
118
|
+
if (obj.$ref.startsWith(prefix)) {
|
|
119
|
+
const refKey = obj.$ref.slice(prefix.length)
|
|
120
|
+
const canonical = remapping.get(refKey)
|
|
121
|
+
if (canonical !== undefined) {
|
|
122
|
+
obj.$ref = prefix + canonical
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
for (const value of Object.values(obj)) {
|
|
128
|
+
rewriteRefs(value, remapping)
|
|
129
|
+
}
|
|
130
|
+
}
|
package/src/Schema/brand.ts
CHANGED
|
@@ -35,11 +35,31 @@ type BrandAnnotations<C extends B.Brand<any>> =
|
|
|
35
35
|
: {}
|
|
36
36
|
)
|
|
37
37
|
|
|
38
|
+
export interface BrandedSchema<S extends S.Top, C extends B.Brand<any>> extends
|
|
39
|
+
S.Bottom<
|
|
40
|
+
C,
|
|
41
|
+
S["Encoded"],
|
|
42
|
+
S["DecodingServices"],
|
|
43
|
+
S["EncodingServices"],
|
|
44
|
+
S["ast"],
|
|
45
|
+
BrandedSchema<S, C>,
|
|
46
|
+
S["~type.make.in"],
|
|
47
|
+
S["Iso"],
|
|
48
|
+
S["~type.parameters"],
|
|
49
|
+
C,
|
|
50
|
+
S["~type.mutability"],
|
|
51
|
+
S["~type.optionality"],
|
|
52
|
+
S["~type.constructor.default"],
|
|
53
|
+
S["~encoded.mutability"],
|
|
54
|
+
S["~encoded.optionality"]
|
|
55
|
+
>
|
|
56
|
+
{}
|
|
57
|
+
|
|
38
58
|
export const fromBrand = <C extends B.Brand<any>>(
|
|
39
59
|
constructor: Constructor<C>,
|
|
40
60
|
options?: BrandAnnotations<C>
|
|
41
61
|
) =>
|
|
42
|
-
<Self extends S.Top>(self: Self):
|
|
62
|
+
<Self extends S.Top>(self: Self): BrandedSchema<Self, C> => {
|
|
43
63
|
const branded = S.fromBrand(options?.identifier ?? "Brand", constructor as any)(self as any)
|
|
44
64
|
return options ? (branded as any).pipe(S.annotate(options)) : branded as any
|
|
45
65
|
}
|