effect-app 4.0.0-beta.26 → 4.0.0-beta.261
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 +1194 -0
- package/dist/Array.d.ts +3 -2
- package/dist/Array.d.ts.map +1 -1
- package/dist/Array.js +4 -4
- package/dist/Chunk.d.ts +1 -1
- package/dist/Chunk.d.ts.map +1 -1
- package/dist/Config/SecretURL.d.ts +4 -2
- package/dist/Config/SecretURL.d.ts.map +1 -1
- package/dist/Config/SecretURL.js +3 -6
- package/dist/Config/internal/configSecretURL.d.ts +1 -1
- package/dist/Config/internal/configSecretURL.d.ts.map +1 -1
- package/dist/Config/internal/configSecretURL.js +2 -2
- 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 +42 -0
- package/dist/Context.d.ts.map +1 -0
- package/dist/Context.js +67 -0
- package/dist/DataDependencies.d.ts +158 -0
- package/dist/DataDependencies.d.ts.map +1 -0
- package/dist/DataDependencies.js +39 -0
- package/dist/Effect.d.ts +13 -12
- package/dist/Effect.d.ts.map +1 -1
- package/dist/Effect.js +5 -8
- package/dist/Emailer.d.ts +51 -0
- package/dist/Emailer.d.ts.map +1 -0
- package/dist/Emailer.js +7 -0
- 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 +11 -7
- package/dist/Layer.d.ts.map +1 -1
- package/dist/Layer.js +3 -2
- package/dist/Model/Repository/Registry.d.ts +21 -0
- package/dist/Model/Repository/Registry.d.ts.map +1 -0
- package/dist/Model/Repository/Registry.js +18 -0
- package/dist/Model/Repository/ext.d.ts +60 -0
- package/dist/Model/Repository/ext.d.ts.map +1 -0
- package/dist/Model/Repository/ext.js +122 -0
- package/dist/Model/Repository/internal/internal.d.ts +63 -0
- package/dist/Model/Repository/internal/internal.d.ts.map +1 -0
- package/dist/Model/Repository/internal/internal.js +430 -0
- package/dist/Model/Repository/legacy.d.ts +21 -0
- package/dist/Model/Repository/legacy.d.ts.map +1 -0
- package/dist/Model/Repository/legacy.js +2 -0
- package/dist/Model/Repository/makeRepo.d.ts +54 -0
- package/dist/Model/Repository/makeRepo.d.ts.map +1 -0
- package/dist/Model/Repository/makeRepo.js +27 -0
- package/dist/Model/Repository/service.d.ts +121 -0
- package/dist/Model/Repository/service.d.ts.map +1 -0
- package/dist/Model/Repository/service.js +2 -0
- package/dist/Model/Repository/validation.d.ts +58 -0
- package/dist/Model/Repository/validation.d.ts.map +1 -0
- package/dist/Model/Repository/validation.js +32 -0
- package/dist/Model/Repository.d.ts +7 -0
- package/dist/Model/Repository.d.ts.map +1 -0
- package/dist/Model/Repository.js +7 -0
- package/dist/Model/dsl.d.ts +33 -0
- package/dist/Model/dsl.d.ts.map +1 -0
- package/dist/Model/dsl.js +43 -0
- package/dist/Model/filter/filterApi.d.ts +30 -0
- package/dist/Model/filter/filterApi.d.ts.map +1 -0
- package/dist/Model/filter/filterApi.js +2 -0
- package/dist/Model/filter/types/errors.d.ts +29 -0
- package/dist/Model/filter/types/errors.d.ts.map +1 -0
- package/dist/Model/filter/types/errors.js +2 -0
- package/dist/Model/filter/types/fields.d.ts +15 -0
- package/dist/Model/filter/types/fields.d.ts.map +1 -0
- package/dist/Model/filter/types/fields.js +2 -0
- package/dist/Model/filter/types/path/common.d.ts +316 -0
- package/dist/Model/filter/types/path/common.d.ts.map +1 -0
- package/dist/Model/filter/types/path/common.js +2 -0
- package/dist/Model/filter/types/path/eager.d.ts +95 -0
- package/dist/Model/filter/types/path/eager.d.ts.map +1 -0
- package/dist/Model/filter/types/path/eager.js +31 -0
- package/dist/Model/filter/types/path/index.d.ts +4 -0
- package/dist/Model/filter/types/path/index.d.ts.map +1 -0
- package/dist/Model/filter/types/path/index.js +3 -0
- package/dist/Model/filter/types/utils.d.ts +79 -0
- package/dist/Model/filter/types/utils.d.ts.map +1 -0
- package/dist/Model/filter/types/utils.js +2 -0
- package/dist/Model/filter/types/validator.d.ts +30 -0
- package/dist/Model/filter/types/validator.d.ts.map +1 -0
- package/dist/Model/filter/types/validator.js +2 -0
- package/dist/Model/filter/types.d.ts +5 -0
- package/dist/Model/filter/types.d.ts.map +1 -0
- package/dist/Model/filter/types.js +7 -0
- package/dist/Model/query/dsl.d.ts +493 -0
- package/dist/Model/query/dsl.d.ts.map +1 -0
- package/dist/Model/query/dsl.js +376 -0
- package/dist/Model/query/new-kid-interpreter.d.ts +136 -0
- package/dist/Model/query/new-kid-interpreter.d.ts.map +1 -0
- package/dist/Model/query/new-kid-interpreter.js +336 -0
- package/dist/Model/query.d.ts +15 -0
- package/dist/Model/query.d.ts.map +1 -0
- package/dist/Model/query.js +3 -0
- package/dist/Model.d.ts +5 -0
- package/dist/Model.d.ts.map +1 -0
- package/dist/Model.js +5 -0
- package/dist/NonEmptySet.d.ts +4 -2
- package/dist/NonEmptySet.d.ts.map +1 -1
- package/dist/NonEmptySet.js +2 -2
- package/dist/Option.d.ts +2 -1
- package/dist/Option.d.ts.map +1 -1
- package/dist/Option.js +3 -1
- package/dist/Pure.d.ts +8 -6
- package/dist/Pure.d.ts.map +1 -1
- package/dist/Pure.js +17 -14
- package/dist/QueueMaker.d.ts +13 -0
- package/dist/QueueMaker.d.ts.map +1 -0
- package/dist/QueueMaker.js +4 -0
- package/dist/RequestContext.d.ts +91 -0
- package/dist/RequestContext.d.ts.map +1 -0
- package/dist/RequestContext.js +49 -0
- package/dist/Schema/Class.d.ts +66 -20
- package/dist/Schema/Class.d.ts.map +1 -1
- package/dist/Schema/Class.js +192 -27
- 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 +34 -0
- package/dist/Schema/SpecialJsonSchema.d.ts.map +1 -0
- package/dist/Schema/SpecialJsonSchema.js +118 -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 +5 -3
- package/dist/Schema/brand.d.ts.map +1 -1
- package/dist/Schema/brand.js +3 -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 +339 -56
- package/dist/Schema/ext.d.ts.map +1 -1
- package/dist/Schema/ext.js +358 -53
- package/dist/Schema/moreStrings.d.ts +83 -37
- package/dist/Schema/moreStrings.d.ts.map +1 -1
- package/dist/Schema/moreStrings.js +49 -42
- package/dist/Schema/numbers.d.ts +35 -22
- package/dist/Schema/numbers.d.ts.map +1 -1
- package/dist/Schema/numbers.js +55 -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 +19 -5
- package/dist/Schema/strings.d.ts.map +1 -1
- package/dist/Schema/strings.js +1 -5
- package/dist/Schema.d.ts +214 -8
- package/dist/Schema.d.ts.map +1 -1
- package/dist/Schema.js +190 -11
- package/dist/Set.d.ts +5 -2
- package/dist/Set.d.ts.map +1 -1
- package/dist/Set.js +3 -2
- package/dist/Store.d.ts +170 -0
- package/dist/Store.d.ts.map +1 -0
- package/dist/Store.js +121 -0
- 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 +2 -2
- package/dist/_ext/Array.d.ts.map +1 -1
- package/dist/_ext/Array.js +4 -2
- package/dist/_ext/date.d.ts +1 -1
- package/dist/_ext/misc.d.ts +5 -2
- package/dist/_ext/misc.d.ts.map +1 -1
- package/dist/_ext/misc.js +4 -2
- package/dist/_ext/ord.ext.d.ts +3 -2
- package/dist/_ext/ord.ext.d.ts.map +1 -1
- package/dist/_ext/ord.ext.js +2 -2
- 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 +104 -34
- package/dist/client/clientFor.d.ts +53 -19
- 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 +495 -33
- package/dist/client/makeClient.d.ts.map +1 -1
- package/dist/client/makeClient.js +66 -24
- package/dist/client.d.ts +6 -5
- 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/Request.js +2 -2
- package/dist/http/internal/lib.d.ts +1 -1
- package/dist/http.d.ts +1 -1
- package/dist/ids.d.ts +43 -15
- package/dist/ids.d.ts.map +1 -1
- package/dist/ids.js +30 -5
- package/dist/index.d.ts +7 -8
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +8 -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 +420 -0
- package/dist/rpc/Invalidation.d.ts.map +1 -0
- package/dist/rpc/Invalidation.js +168 -0
- package/dist/rpc/MiddlewareMaker.d.ts +12 -8
- package/dist/rpc/MiddlewareMaker.d.ts.map +1 -1
- package/dist/rpc/MiddlewareMaker.js +59 -38
- package/dist/rpc/RpcContextMap.d.ts +4 -4
- package/dist/rpc/RpcContextMap.d.ts.map +1 -1
- package/dist/rpc/RpcContextMap.js +4 -4
- package/dist/rpc/RpcMiddleware.d.ts +15 -11
- 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/runtime.d.ts +19 -0
- package/dist/runtime.d.ts.map +1 -0
- package/dist/runtime.js +40 -0
- package/dist/setupRequest.d.ts +19 -0
- package/dist/setupRequest.d.ts.map +1 -0
- package/dist/setupRequest.js +69 -0
- package/dist/toast.d.ts +51 -0
- package/dist/toast.d.ts.map +1 -0
- package/dist/toast.js +34 -0
- package/dist/transform.d.ts +2 -2
- package/dist/transform.d.ts.map +1 -1
- package/dist/transform.js +4 -5
- package/dist/utils/effectify.d.ts +2 -2
- package/dist/utils/effectify.d.ts.map +1 -1
- package/dist/utils/effectify.js +2 -2
- package/dist/utils/extend.d.ts +1 -1
- package/dist/utils/extend.d.ts.map +1 -1
- package/dist/utils/gen.d.ts +5 -5
- package/dist/utils/gen.d.ts.map +1 -1
- package/dist/utils/logLevel.d.ts +3 -3
- package/dist/utils/logLevel.d.ts.map +1 -1
- package/dist/utils/logger.d.ts +5 -4
- package/dist/utils/logger.d.ts.map +1 -1
- package/dist/utils/logger.js +4 -4
- package/dist/utils.d.ts +40 -45
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +19 -27
- 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/dist/withToast.d.ts +30 -0
- package/dist/withToast.d.ts.map +1 -0
- package/dist/withToast.js +64 -0
- package/package.json +20 -245
- package/src/Array.ts +3 -3
- package/src/Config/SecretURL.ts +5 -2
- package/src/Config/internal/configSecretURL.ts +1 -1
- package/src/Config.ts +14 -0
- package/src/ConfigProvider.ts +48 -0
- package/src/{ServiceMap.ts → Context.ts} +56 -63
- package/src/Effect.ts +12 -14
- package/src/Emailer.ts +51 -0
- package/src/Layer.ts +10 -6
- package/src/Model/Repository/Registry.ts +34 -0
- package/src/Model/Repository/ext.ts +375 -0
- package/src/Model/Repository/internal/internal.ts +741 -0
- package/src/Model/Repository/legacy.ts +29 -0
- package/src/Model/Repository/makeRepo.ts +145 -0
- package/src/Model/Repository/service.ts +666 -0
- package/src/Model/Repository/validation.ts +31 -0
- package/src/Model/Repository.ts +6 -0
- package/src/Model/dsl.ts +129 -0
- package/src/Model/filter/filterApi.ts +60 -0
- package/src/Model/filter/types/errors.ts +47 -0
- package/src/Model/filter/types/fields.ts +50 -0
- package/src/Model/filter/types/path/common.ts +404 -0
- package/src/Model/filter/types/path/eager.ts +297 -0
- package/src/Model/filter/types/path/index.ts +4 -0
- package/src/Model/filter/types/utils.ts +128 -0
- package/src/Model/filter/types/validator.ts +46 -0
- package/src/Model/filter/types.ts +6 -0
- package/src/Model/query/dsl.ts +2694 -0
- package/src/Model/query/new-kid-interpreter.ts +484 -0
- package/src/Model/query.ts +13 -0
- package/src/Model.ts +4 -0
- package/src/NonEmptySet.ts +3 -1
- package/src/Option.ts +2 -0
- package/src/Pure.ts +21 -19
- package/src/QueueMaker.ts +19 -0
- package/src/RequestContext.ts +62 -0
- package/src/Schema/Class.ts +274 -68
- package/src/Schema/SchemaParser.ts +12 -0
- package/src/Schema/SpecialJsonSchema.ts +139 -0
- package/src/Schema/SpecialOpenApi.ts +130 -0
- package/src/Schema/brand.ts +22 -2
- package/src/Schema/email.ts +7 -2
- package/src/Schema/ext.ts +443 -88
- package/src/Schema/moreStrings.ts +146 -67
- package/src/Schema/numbers.ts +96 -27
- package/src/Schema/phoneNumber.ts +5 -1
- package/src/Schema/strings.ts +22 -13
- package/src/Schema.ts +374 -10
- package/src/Set.ts +5 -1
- package/src/Store.ts +277 -0
- package/src/_ext/Array.ts +3 -1
- package/src/_ext/misc.ts +4 -1
- package/src/_ext/ord.ext.ts +2 -1
- package/src/client/InvalidationKeys.ts +50 -0
- package/src/client/apiClientFactory.ts +230 -131
- package/src/client/clientFor.ts +102 -31
- package/src/client/errors.ts +52 -26
- package/src/client/makeClient.ts +592 -71
- package/src/client.ts +5 -4
- package/src/http/Request.ts +1 -1
- package/src/ids.ts +32 -5
- package/src/index.ts +7 -10
- package/src/middleware.ts +13 -9
- package/src/rpc/Invalidation.ts +261 -0
- package/src/rpc/MiddlewareMaker.ts +83 -75
- package/src/rpc/README.md +2 -2
- package/src/rpc/RpcContextMap.ts +6 -5
- package/src/rpc/RpcMiddleware.ts +18 -12
- package/src/rpc.ts +1 -1
- package/src/runtime.ts +56 -0
- package/src/setupRequest.ts +134 -0
- package/src/toast.ts +54 -0
- package/src/transform.ts +3 -3
- package/src/utils/effectify.ts +1 -1
- package/src/utils/gen.ts +8 -8
- package/src/utils/logLevel.ts +1 -1
- package/src/utils/logger.ts +4 -3
- package/src/utils.ts +62 -139
- package/src/withToast.ts +133 -0
- package/test/dist/rpc-dynamic-middleware.test.d.ts.map +1 -0
- 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/moreStrings.test.ts +1 -1
- package/test/rpc.test.ts +46 -6
- package/test/schema.test.ts +459 -30
- package/test/secretURL.test.ts +160 -0
- package/test/special.test.ts +1026 -0
- package/test/utils.test.ts +7 -7
- package/tsconfig.base.json +6 -5
- package/tsconfig.json +2 -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/ext.ts
CHANGED
|
@@ -1,8 +1,47 @@
|
|
|
1
1
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
2
2
|
/* eslint-disable @typescript-eslint/no-unsafe-return */
|
|
3
|
-
|
|
3
|
+
/**
|
|
4
|
+
* # `withConstructorDefault` policy
|
|
5
|
+
*
|
|
6
|
+
* The `withConstructorDefault` properties exported throughout this module
|
|
7
|
+
* (and from `numbers.ts`, `moreStrings.ts`, `ids.ts`) attach a default value
|
|
8
|
+
* that is **only** applied during construction — i.e. when the field is
|
|
9
|
+
* omitted from the input to a Schema constructor / `.make(...)` call.
|
|
10
|
+
*
|
|
11
|
+
* They are **NOT** applied during `decode` (JSON, database rows, RPC payloads,
|
|
12
|
+
* etc.). Decoding a payload with a missing field will still fail with a parse
|
|
13
|
+
* error, exactly as if the default were not present.
|
|
14
|
+
*
|
|
15
|
+
* Concretely this means `withConstructorDefault` MUST NOT be relied on as a
|
|
16
|
+
* just-in-time migration mechanism for database fields. If a stored record is
|
|
17
|
+
* missing a newly added field, the constructor default will not fill it in on
|
|
18
|
+
* read — decoding will fail.
|
|
19
|
+
*
|
|
20
|
+
* ## Don't reach for `withDecodingDefault*` either
|
|
21
|
+
*
|
|
22
|
+
* The sibling `withDecodingDefaultType` (and `withDecodingDefault`) extensions
|
|
23
|
+
* exist, but they are discouraged for migrating persisted data. A missing
|
|
24
|
+
* field in a stored record is just as likely to be data corruption as it is
|
|
25
|
+
* an old-shape document; silently substituting a default hides the problem
|
|
26
|
+
* and can poison downstream aggregates.
|
|
27
|
+
*
|
|
28
|
+
* Prefer an **explicit, preferably versioned** migration of database data
|
|
29
|
+
* (a schema-version field, a one-shot backfill, or a transform on read that
|
|
30
|
+
* is gated on an explicit version marker) over shoving missing fields under
|
|
31
|
+
* the rug with a decode-time default.
|
|
32
|
+
*/
|
|
33
|
+
import * as Config from "effect/Config"
|
|
34
|
+
import * as Effect from "effect/Effect"
|
|
35
|
+
import { pipe } from "effect/Function"
|
|
36
|
+
import * as Function from "effect/Function"
|
|
37
|
+
import * as Option from "effect/Option"
|
|
4
38
|
import * as S from "effect/Schema"
|
|
39
|
+
import { isDateValid } from "effect/Schema"
|
|
40
|
+
import type * as SchemaAST from "effect/SchemaAST"
|
|
41
|
+
import * as SchemaIssue from "effect/SchemaIssue"
|
|
42
|
+
import * as SchemaTransformation from "effect/SchemaTransformation"
|
|
5
43
|
import { type NonEmptyReadonlyArray } from "../Array.js"
|
|
44
|
+
import * as Context from "../Context.js"
|
|
6
45
|
import { extendM, typedKeysOf } from "../utils.js"
|
|
7
46
|
import { type AST } from "./schema.js"
|
|
8
47
|
|
|
@@ -13,131 +52,429 @@ type ProvidedCodec<Self extends S.Top, R> = S.Codec<
|
|
|
13
52
|
Exclude<Self["EncodingServices"], R>
|
|
14
53
|
>
|
|
15
54
|
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
type Narrowed = Self & S.WithoutConstructorDefault
|
|
25
|
-
return S.withConstructorDefault<Narrowed>(
|
|
26
|
-
() => Option.some(makeDefault() as Narrowed["~type.make.in"])
|
|
27
|
-
)(self as Narrowed)
|
|
55
|
+
const concurrencySetting = Effect.runSync(
|
|
56
|
+
Config
|
|
57
|
+
.literal("unbounded", "SCHEMA_CONCURRENCY")
|
|
58
|
+
.pipe(Config.orElse(() => Config.number("SCHEMA_CONCURRENCY")), Config.option)
|
|
59
|
+
)
|
|
60
|
+
|
|
61
|
+
export const DefaultParseOptions: SchemaAST.ParseOptions = {
|
|
62
|
+
concurrency: Option.getOrElse(concurrencySetting, () => "unbounded" as const)
|
|
28
63
|
}
|
|
29
64
|
|
|
65
|
+
/**
|
|
66
|
+
* Parse-options annotation used on schema constructors for decode paths where callers
|
|
67
|
+
* cannot currently pass parse options (notably some RPC / HttpApi integration paths).
|
|
68
|
+
*
|
|
69
|
+
* Keep this annotation in place so those framework-managed decodes still run with
|
|
70
|
+
* unbounded concurrency by default.
|
|
71
|
+
*/
|
|
72
|
+
export const concurrencyUnbounded = { parseOptions: DefaultParseOptions } as const
|
|
73
|
+
|
|
74
|
+
type DecodeLike = (schema: any) => (input: any, options?: SchemaAST.ParseOptions) => any
|
|
75
|
+
|
|
76
|
+
export const withDefaultParseOptions = <Decode extends DecodeLike>(
|
|
77
|
+
decode: Decode,
|
|
78
|
+
defaultParseOptions: SchemaAST.ParseOptions = DefaultParseOptions
|
|
79
|
+
): Decode =>
|
|
80
|
+
((schema: any) => {
|
|
81
|
+
const run = decode(schema)
|
|
82
|
+
return (input: any, options?: SchemaAST.ParseOptions) => run(input, { ...defaultParseOptions, ...options })
|
|
83
|
+
}) as Decode
|
|
84
|
+
|
|
30
85
|
// TODO: v4 migration - Date is no longer by default encoded to string.
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
)
|
|
86
|
+
|
|
87
|
+
const DateString = S.String.annotate({
|
|
88
|
+
identifier: "DateOrInvalid",
|
|
89
|
+
description: "an ISO 8601 date string that will be decoded as a Date (may be invalid)",
|
|
90
|
+
format: "date-time"
|
|
91
|
+
})
|
|
37
92
|
|
|
38
93
|
/**
|
|
39
|
-
*
|
|
94
|
+
* Schema type for {@link DateFromString}.
|
|
95
|
+
*
|
|
96
|
+
* @category Schemas
|
|
97
|
+
* @since 4.0.0
|
|
40
98
|
*/
|
|
41
|
-
export
|
|
42
|
-
withDefault: DateFromString.pipe(withDefaultConstructor(() => new global.Date()))
|
|
43
|
-
})
|
|
99
|
+
export interface DateFromString extends S.decodeTo<S.Date, S.String> {}
|
|
44
100
|
|
|
45
101
|
/**
|
|
46
|
-
*
|
|
102
|
+
* A transformation schema that parses an ISO 8601 string into a `Date`.
|
|
103
|
+
*
|
|
104
|
+
* Decoding:
|
|
105
|
+
* - A `string` is decoded as a `Date`.
|
|
106
|
+
*
|
|
107
|
+
* Encoding:
|
|
108
|
+
* - A `Date` is encoded as a `string`.
|
|
109
|
+
*
|
|
110
|
+
* @since 4.0.0
|
|
47
111
|
*/
|
|
112
|
+
export const DateFromString: DateFromString = DateString.pipe(S.decodeTo(S.Date, SchemaTransformation.dateFromString))
|
|
113
|
+
|
|
114
|
+
/** Like the default Schema `Date` but from String, with default helpers. */
|
|
115
|
+
export const Date = extendM(DateFromString, (s) => ({
|
|
116
|
+
/**
|
|
117
|
+
* Construction-only default `new Date()`. Applied only when the field is
|
|
118
|
+
* omitted from `.make(...)` input. NOT applied during decode — cannot be
|
|
119
|
+
* used to JIT-migrate database fields. See file-level note.
|
|
120
|
+
*/
|
|
121
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.sync(() => new global.Date()))),
|
|
122
|
+
/**
|
|
123
|
+
* Decode-time default `new Date()`. **Discouraged for persisted data:** a
|
|
124
|
+
* missing field may be data corruption, not an old-shape document; silently
|
|
125
|
+
* substituting `new Date()` hides the problem. Prefer an explicit,
|
|
126
|
+
* preferably versioned migration over a decode-time fallback. See
|
|
127
|
+
* file-level note.
|
|
128
|
+
*/
|
|
129
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.sync(() => new global.Date())))
|
|
130
|
+
}))
|
|
131
|
+
|
|
132
|
+
const DateValidString = S.String.annotate({
|
|
133
|
+
identifier: "Date",
|
|
134
|
+
description: "a valid ISO 8601 date string that will be decoded as a Date",
|
|
135
|
+
format: "date-time"
|
|
136
|
+
})
|
|
137
|
+
|
|
138
|
+
const DateValidFromString = DateValidString
|
|
139
|
+
.pipe(
|
|
140
|
+
S.decodeTo(S.Date, SchemaTransformation.dateFromString)
|
|
141
|
+
)
|
|
142
|
+
.check(isDateValid())
|
|
143
|
+
|
|
144
|
+
/** Like the default Schema `DateValid` but from String, with default helpers. */
|
|
145
|
+
export const DateValid = extendM(DateValidFromString, (s) => ({
|
|
146
|
+
/**
|
|
147
|
+
* Construction-only default `new Date()`. Applied only when the field is
|
|
148
|
+
* omitted from `.make(...)` input. NOT applied during decode — cannot be
|
|
149
|
+
* used to JIT-migrate database fields. See file-level note.
|
|
150
|
+
*/
|
|
151
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.sync(() => new global.Date()))),
|
|
152
|
+
/**
|
|
153
|
+
* Decode-time default `new Date()`. **Discouraged for persisted data:** a
|
|
154
|
+
* missing field may be data corruption, not an old-shape document; silently
|
|
155
|
+
* substituting `new Date()` hides the problem. Prefer an explicit,
|
|
156
|
+
* preferably versioned migration over a decode-time fallback. See
|
|
157
|
+
* file-level note.
|
|
158
|
+
*/
|
|
159
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.sync(() => new global.Date())))
|
|
160
|
+
}))
|
|
161
|
+
|
|
162
|
+
/** Like the default Schema `Boolean` but with default helpers. */
|
|
48
163
|
export const Boolean = Object.assign(S.Boolean, {
|
|
49
|
-
|
|
164
|
+
/**
|
|
165
|
+
* Construction-only default `false`. Applied only when the field is
|
|
166
|
+
* omitted from `.make(...)` input. NOT applied during decode — cannot be
|
|
167
|
+
* used to JIT-migrate database fields. See file-level note.
|
|
168
|
+
*/
|
|
169
|
+
withConstructorDefault: S.Boolean.pipe(S.withConstructorDefault(Effect.succeed(false))),
|
|
170
|
+
/**
|
|
171
|
+
* Decode-time default `false`. **Discouraged for persisted data:** a
|
|
172
|
+
* missing field may be data corruption, not an old-shape document; silently
|
|
173
|
+
* substituting `false` hides the problem. Prefer an explicit, preferably
|
|
174
|
+
* versioned migration over a decode-time fallback. See file-level note.
|
|
175
|
+
*/
|
|
176
|
+
withDecodingDefaultType: S.Boolean.pipe(S.withDecodingDefaultType(Effect.succeed(false)))
|
|
50
177
|
})
|
|
51
178
|
|
|
52
179
|
/**
|
|
53
|
-
*
|
|
180
|
+
* You probably want to use `Finite` instead of this. Like the default Schema
|
|
181
|
+
* `Number` but with default helpers.
|
|
54
182
|
*/
|
|
55
|
-
export const Number = Object.assign(S.Number, {
|
|
183
|
+
export const Number = Object.assign(S.Number, {
|
|
184
|
+
/**
|
|
185
|
+
* Construction-only default `0`. Applied only when the field is omitted
|
|
186
|
+
* from `.make(...)` input. NOT applied during decode — cannot be used to
|
|
187
|
+
* JIT-migrate database fields. See file-level note.
|
|
188
|
+
*/
|
|
189
|
+
withConstructorDefault: S.Number.pipe(S.withConstructorDefault(Effect.succeed(0))),
|
|
190
|
+
/**
|
|
191
|
+
* Decode-time default `0`. **Discouraged for persisted data:** a missing
|
|
192
|
+
* field may be data corruption, not an old-shape document; silently
|
|
193
|
+
* substituting `0` hides the problem. Prefer an explicit, preferably
|
|
194
|
+
* versioned migration over a decode-time fallback. See file-level note.
|
|
195
|
+
*/
|
|
196
|
+
withDecodingDefaultType: S.Number.pipe(S.withDecodingDefaultType(Effect.succeed(0)))
|
|
197
|
+
})
|
|
56
198
|
|
|
57
|
-
/**
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
199
|
+
/** Like the default Schema `Finite` but with default helpers. */
|
|
200
|
+
export const Finite = Object.assign(S.Finite, {
|
|
201
|
+
/**
|
|
202
|
+
* Construction-only default `0`. Applied only when the field is omitted
|
|
203
|
+
* from `.make(...)` input. NOT applied during decode — cannot be used to
|
|
204
|
+
* JIT-migrate database fields. See file-level note.
|
|
205
|
+
*/
|
|
206
|
+
withConstructorDefault: S.Finite.pipe(S.withConstructorDefault(Effect.succeed(0))),
|
|
207
|
+
/**
|
|
208
|
+
* Decode-time default `0`. **Discouraged for persisted data:** a missing
|
|
209
|
+
* field may be data corruption, not an old-shape document; silently
|
|
210
|
+
* substituting `0` hides the problem. Prefer an explicit, preferably
|
|
211
|
+
* versioned migration over a decode-time fallback. See file-level note.
|
|
212
|
+
*/
|
|
213
|
+
withDecodingDefaultType: S.Finite.pipe(S.withDecodingDefaultType(Effect.succeed(0)))
|
|
214
|
+
})
|
|
215
|
+
|
|
216
|
+
/** Like the default Schema `Literals` but with default helpers. Default value is `literals[0]`. */
|
|
217
|
+
export const Literals = <const Literals extends NonEmptyReadonlyArray<AST.LiteralValue>>(literals: Literals) =>
|
|
61
218
|
pipe(
|
|
62
219
|
S.Literals(literals),
|
|
63
220
|
(s) =>
|
|
64
221
|
Object.assign(s, {
|
|
222
|
+
/** Override the default literal value used by `withConstructorDefault` / `withDecodingDefaultType`. */
|
|
65
223
|
changeDefault: <A extends Literals[number]>(a: A) => {
|
|
66
224
|
return Object.assign(S.Literals(literals), {
|
|
67
225
|
Default: a,
|
|
68
|
-
|
|
226
|
+
/**
|
|
227
|
+
* Construction-only default. Applied only when the field is
|
|
228
|
+
* omitted from `.make(...)` input. NOT applied during decode —
|
|
229
|
+
* cannot be used to JIT-migrate database fields. See file-level
|
|
230
|
+
* note.
|
|
231
|
+
*/
|
|
232
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.succeed(a))),
|
|
233
|
+
/**
|
|
234
|
+
* Decode-time default. **Discouraged for persisted data:** a
|
|
235
|
+
* missing field may be data corruption, not an old-shape
|
|
236
|
+
* document; silently substituting hides the problem. Prefer an
|
|
237
|
+
* explicit, preferably versioned migration over a decode-time
|
|
238
|
+
* fallback. See file-level note.
|
|
239
|
+
*/
|
|
240
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.succeed(a)))
|
|
69
241
|
}) // todo: copy annotations from original?
|
|
70
242
|
},
|
|
71
|
-
|
|
72
|
-
|
|
243
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-type-assertion -- load-bearing: Object.assign widens the field type without it, breaking `expectTypeOf(l.Default).toEqualTypeOf<"a">()` in tests
|
|
244
|
+
Default: literals[0] as Literals[0],
|
|
245
|
+
/**
|
|
246
|
+
* Construction-only default `literals[0]`. Applied only when the
|
|
247
|
+
* field is omitted from `.make(...)` input. NOT applied during
|
|
248
|
+
* decode — cannot be used to JIT-migrate database fields. See
|
|
249
|
+
* file-level note.
|
|
250
|
+
*/
|
|
251
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.succeed(literals[0]))),
|
|
252
|
+
/**
|
|
253
|
+
* Decode-time default `literals[0]`. **Discouraged for persisted
|
|
254
|
+
* data:** a missing field may be data corruption, not an old-shape
|
|
255
|
+
* document; silently substituting hides the problem. Prefer an
|
|
256
|
+
* explicit, preferably versioned migration over a decode-time
|
|
257
|
+
* fallback. See file-level note.
|
|
258
|
+
*/
|
|
259
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.succeed(literals[0])))
|
|
73
260
|
})
|
|
74
261
|
)
|
|
75
262
|
|
|
76
|
-
/**
|
|
77
|
-
* Like the default Schema `Array` but with `withDefault` => []
|
|
78
|
-
*/
|
|
263
|
+
/** Like the default Schema `Array` but with default helpers. */
|
|
79
264
|
export function Array<ValueSchema extends S.Top>(value: ValueSchema) {
|
|
80
265
|
return pipe(
|
|
81
|
-
S.Array(value),
|
|
82
|
-
(s) =>
|
|
266
|
+
S.Array(value).annotate(concurrencyUnbounded),
|
|
267
|
+
(s) =>
|
|
268
|
+
Object.assign(s, {
|
|
269
|
+
/**
|
|
270
|
+
* Construction-only default `[]`. Applied only when the field is
|
|
271
|
+
* omitted from `.make(...)` input. NOT applied during decode —
|
|
272
|
+
* cannot be used to JIT-migrate database fields. See file-level
|
|
273
|
+
* note.
|
|
274
|
+
*/
|
|
275
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.sync(() => []))),
|
|
276
|
+
/**
|
|
277
|
+
* Decode-time default `[]`. **Discouraged for persisted data:** a
|
|
278
|
+
* missing field may be data corruption, not an old-shape document;
|
|
279
|
+
* silently substituting `[]` hides the problem. Prefer an explicit,
|
|
280
|
+
* preferably versioned migration over a decode-time fallback. See
|
|
281
|
+
* file-level note.
|
|
282
|
+
*/
|
|
283
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.sync(() => [])))
|
|
284
|
+
})
|
|
83
285
|
)
|
|
84
286
|
}
|
|
85
287
|
|
|
86
288
|
/**
|
|
87
|
-
*
|
|
289
|
+
* An annotated `S.Array` of unique items that decodes to a `ReadonlySet`.
|
|
88
290
|
*/
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
(
|
|
291
|
+
export const ReadonlySetFromArray = <ValueSchema extends S.Top>(value: ValueSchema) => {
|
|
292
|
+
const from = S
|
|
293
|
+
.Array(value)
|
|
294
|
+
.annotate({ ...concurrencyUnbounded, expected: "an array of unique items that will be decoded as a ReadonlySet" })
|
|
295
|
+
const to = S.instanceOf(Set) as S.instanceOf<ReadonlySet<ValueSchema["Type"]>>
|
|
296
|
+
const schema = from.pipe(
|
|
297
|
+
S.decodeTo(
|
|
298
|
+
to,
|
|
299
|
+
SchemaTransformation.transform({
|
|
300
|
+
decode: (arr) => new Set(arr) as ReadonlySet<ValueSchema["Type"]>,
|
|
301
|
+
encode: (set) => [...set]
|
|
302
|
+
})
|
|
303
|
+
)
|
|
93
304
|
)
|
|
305
|
+
return schema
|
|
94
306
|
}
|
|
95
307
|
|
|
96
|
-
export { Map_ as Map }
|
|
97
|
-
|
|
98
308
|
/**
|
|
99
|
-
*
|
|
309
|
+
* An annotated `S.Array` of key-value tuples that decodes to a `ReadonlyMap`.
|
|
100
310
|
*/
|
|
311
|
+
export const ReadonlyMapFromArray = <KeySchema extends S.Top, ValueSchema extends S.Top>(pair: {
|
|
312
|
+
readonly key: KeySchema
|
|
313
|
+
readonly value: ValueSchema
|
|
314
|
+
}) => {
|
|
315
|
+
const from = S
|
|
316
|
+
.Array(S.Tuple([pair.key, pair.value]))
|
|
317
|
+
.annotate({
|
|
318
|
+
...concurrencyUnbounded,
|
|
319
|
+
expected: "an array of key-value tuples that will be decoded as a ReadonlyMap"
|
|
320
|
+
})
|
|
321
|
+
const to = S.instanceOf(Map) as S.instanceOf<
|
|
322
|
+
ReadonlyMap<KeySchema["Type"], ValueSchema["Type"]>
|
|
323
|
+
>
|
|
324
|
+
const schema = from.pipe(
|
|
325
|
+
S.decodeTo(
|
|
326
|
+
to,
|
|
327
|
+
SchemaTransformation.transform({
|
|
328
|
+
decode: (
|
|
329
|
+
arr
|
|
330
|
+
) => new Map(arr) as ReadonlyMap<KeySchema["Type"], ValueSchema["Type"]>,
|
|
331
|
+
encode: (
|
|
332
|
+
map
|
|
333
|
+
) => [...map.entries()] as any // fu
|
|
334
|
+
})
|
|
335
|
+
)
|
|
336
|
+
)
|
|
337
|
+
return schema
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
/** Like the default Schema `ReadonlySet` but from Array, with default helpers. */
|
|
101
341
|
export const ReadonlySet = <ValueSchema extends S.Top>(value: ValueSchema) =>
|
|
102
342
|
pipe(
|
|
103
|
-
|
|
343
|
+
ReadonlySetFromArray(value),
|
|
104
344
|
(s) =>
|
|
105
|
-
Object.assign(s, {
|
|
345
|
+
Object.assign(s, {
|
|
346
|
+
/**
|
|
347
|
+
* Construction-only default `new Set()`. Applied only when the field
|
|
348
|
+
* is omitted from `.make(...)` input. NOT applied during decode —
|
|
349
|
+
* cannot be used to JIT-migrate database fields. See file-level
|
|
350
|
+
* note.
|
|
351
|
+
*/
|
|
352
|
+
withConstructorDefault: s.pipe(
|
|
353
|
+
S.withConstructorDefault(Effect.sync(() => new Set<ValueSchema["Type"]>()))
|
|
354
|
+
),
|
|
355
|
+
/**
|
|
356
|
+
* Decode-time default `new Set()`. **Discouraged for persisted
|
|
357
|
+
* data:** a missing field may be data corruption, not an old-shape
|
|
358
|
+
* document; silently substituting an empty set hides the problem.
|
|
359
|
+
* Prefer an explicit, preferably versioned migration over a
|
|
360
|
+
* decode-time fallback. See file-level note.
|
|
361
|
+
*/
|
|
362
|
+
withDecodingDefaultType: s.pipe(
|
|
363
|
+
S.withDecodingDefaultType(Effect.sync(() => new Set<ValueSchema["Type"]>()))
|
|
364
|
+
)
|
|
365
|
+
})
|
|
106
366
|
)
|
|
107
367
|
|
|
108
|
-
/**
|
|
109
|
-
* Like the default Schema `ReadonlyMap` but with `withDefault` => new Map()
|
|
110
|
-
*/
|
|
368
|
+
/** Like the default Schema `ReadonlyMap` but from Array, with default helpers. */
|
|
111
369
|
export const ReadonlyMap = <KeySchema extends S.Top, ValueSchema extends S.Top>(pair: {
|
|
112
370
|
readonly key: KeySchema
|
|
113
371
|
readonly value: ValueSchema
|
|
114
372
|
}) =>
|
|
115
373
|
pipe(
|
|
116
|
-
|
|
117
|
-
(s) =>
|
|
374
|
+
ReadonlyMapFromArray(pair),
|
|
375
|
+
(s) =>
|
|
376
|
+
Object.assign(s, {
|
|
377
|
+
/**
|
|
378
|
+
* Construction-only default `new Map()`. Applied only when the field
|
|
379
|
+
* is omitted from `.make(...)` input. NOT applied during decode —
|
|
380
|
+
* cannot be used to JIT-migrate database fields. See file-level
|
|
381
|
+
* note.
|
|
382
|
+
*/
|
|
383
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.sync(() => new Map()))),
|
|
384
|
+
/**
|
|
385
|
+
* Decode-time default `new Map()`. **Discouraged for persisted
|
|
386
|
+
* data:** a missing field may be data corruption, not an old-shape
|
|
387
|
+
* document; silently substituting an empty map hides the problem.
|
|
388
|
+
* Prefer an explicit, preferably versioned migration over a
|
|
389
|
+
* decode-time fallback. See file-level note.
|
|
390
|
+
*/
|
|
391
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.sync(() => new Map())))
|
|
392
|
+
})
|
|
118
393
|
)
|
|
119
394
|
|
|
120
|
-
/**
|
|
121
|
-
* Like the default Schema `NullOr` but with `withDefault` => null
|
|
122
|
-
*/
|
|
395
|
+
/** Like the default Schema `NullOr` but with default helpers. */
|
|
123
396
|
export const NullOr = <Schema extends S.Top>(self: Schema) =>
|
|
124
397
|
pipe(
|
|
125
398
|
S.NullOr(self),
|
|
126
|
-
(s) =>
|
|
399
|
+
(s) =>
|
|
400
|
+
Object.assign(s, {
|
|
401
|
+
/**
|
|
402
|
+
* Construction-only default `null`. Applied only when the field is
|
|
403
|
+
* omitted from `.make(...)` input. NOT applied during decode —
|
|
404
|
+
* cannot be used to JIT-migrate database fields. See file-level
|
|
405
|
+
* note.
|
|
406
|
+
*/
|
|
407
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.succeed(null))),
|
|
408
|
+
/**
|
|
409
|
+
* Decode-time default `null`. **Discouraged for persisted data:** a
|
|
410
|
+
* missing field may be data corruption, not an old-shape document;
|
|
411
|
+
* silently substituting `null` hides the problem. Prefer an
|
|
412
|
+
* explicit, preferably versioned migration over a decode-time
|
|
413
|
+
* fallback. See file-level note.
|
|
414
|
+
*/
|
|
415
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.succeed(null)))
|
|
416
|
+
})
|
|
127
417
|
)
|
|
128
418
|
|
|
129
|
-
|
|
130
|
-
|
|
419
|
+
/**
|
|
420
|
+
* Attach a `withConstructorDefault` of `new Date()` to any schema.
|
|
421
|
+
*
|
|
422
|
+
* **Construction-only.** Applied only when the field is omitted from
|
|
423
|
+
* `.make(...)` input. NOT applied during decode — cannot be used to
|
|
424
|
+
* JIT-migrate database fields. See file-level note.
|
|
425
|
+
*/
|
|
426
|
+
export const defaultDate = <Schema extends S.Top & S.WithoutConstructorDefault>(schema: Schema) =>
|
|
427
|
+
schema.pipe(S.withConstructorDefault(Effect.sync(() => new global.Date())))
|
|
131
428
|
|
|
132
|
-
|
|
429
|
+
/**
|
|
430
|
+
* Attach a `withConstructorDefault` of `false` to any schema.
|
|
431
|
+
*
|
|
432
|
+
* **Construction-only.** Applied only when the field is omitted from
|
|
433
|
+
* `.make(...)` input. NOT applied during decode — cannot be used to
|
|
434
|
+
* JIT-migrate database fields. See file-level note.
|
|
435
|
+
*/
|
|
436
|
+
export const defaultBool = <Schema extends S.Top & S.WithoutConstructorDefault>(schema: Schema) =>
|
|
437
|
+
schema.pipe(S.withConstructorDefault(Effect.succeed(false)))
|
|
133
438
|
|
|
134
|
-
|
|
439
|
+
/**
|
|
440
|
+
* Attach a `withConstructorDefault` of `null` to any schema.
|
|
441
|
+
*
|
|
442
|
+
* **Construction-only.** Applied only when the field is omitted from
|
|
443
|
+
* `.make(...)` input. NOT applied during decode — cannot be used to
|
|
444
|
+
* JIT-migrate database fields. See file-level note.
|
|
445
|
+
*/
|
|
446
|
+
export const defaultNullable = <Schema extends S.Top & S.WithoutConstructorDefault>(schema: Schema) =>
|
|
447
|
+
schema.pipe(S.withConstructorDefault(Effect.succeed(null)))
|
|
135
448
|
|
|
136
|
-
|
|
449
|
+
/**
|
|
450
|
+
* Attach a `withConstructorDefault` of `[]` to any schema.
|
|
451
|
+
*
|
|
452
|
+
* **Construction-only.** Applied only when the field is omitted from
|
|
453
|
+
* `.make(...)` input. NOT applied during decode — cannot be used to
|
|
454
|
+
* JIT-migrate database fields. See file-level note.
|
|
455
|
+
*/
|
|
456
|
+
export const defaultArray = <Schema extends S.Top & S.WithoutConstructorDefault>(schema: Schema) =>
|
|
457
|
+
schema.pipe(S.withConstructorDefault(Effect.sync(() => [])))
|
|
137
458
|
|
|
138
|
-
|
|
459
|
+
/**
|
|
460
|
+
* Attach a `withConstructorDefault` of `new Map()` to any schema.
|
|
461
|
+
*
|
|
462
|
+
* **Construction-only.** Applied only when the field is omitted from
|
|
463
|
+
* `.make(...)` input. NOT applied during decode — cannot be used to
|
|
464
|
+
* JIT-migrate database fields. See file-level note.
|
|
465
|
+
*/
|
|
466
|
+
export const defaultMap = <Schema extends S.Top & S.WithoutConstructorDefault>(schema: Schema) =>
|
|
467
|
+
schema.pipe(S.withConstructorDefault(Effect.sync(() => new Map())))
|
|
139
468
|
|
|
140
|
-
|
|
469
|
+
/**
|
|
470
|
+
* Attach a `withConstructorDefault` of `new Set()` to any schema.
|
|
471
|
+
*
|
|
472
|
+
* **Construction-only.** Applied only when the field is omitted from
|
|
473
|
+
* `.make(...)` input. NOT applied during decode — cannot be used to
|
|
474
|
+
* JIT-migrate database fields. See file-level note.
|
|
475
|
+
*/
|
|
476
|
+
export const defaultSet = <Schema extends S.Top & S.WithoutConstructorDefault>(schema: Schema) =>
|
|
477
|
+
schema.pipe(S.withConstructorDefault(Effect.sync(() => new Set())))
|
|
141
478
|
|
|
142
479
|
export const withDefaultMake = <Self extends S.Top>(s: Self) => {
|
|
143
480
|
const a = Object.assign(S.decodeSync(s as any) as WithDefaults<Self>, s)
|
|
@@ -166,9 +503,25 @@ export type WithDefaults<Self extends S.Top> = (
|
|
|
166
503
|
// export type UnionToIntersection3<U> = (U extends any ? (k: U) => void : never) extends ((k: infer I) => void) ? I
|
|
167
504
|
// : never
|
|
168
505
|
|
|
506
|
+
/** Union of `DateValid` and `Date`, with default helpers. */
|
|
169
507
|
export const inputDate = extendM(
|
|
170
|
-
S.Union([S.DateValid,
|
|
171
|
-
(s) => ({
|
|
508
|
+
S.Union([S.DateValid, Date]),
|
|
509
|
+
(s) => ({
|
|
510
|
+
/**
|
|
511
|
+
* Construction-only default `new Date()`. Applied only when the field is
|
|
512
|
+
* omitted from `.make(...)` input. NOT applied during decode — cannot be
|
|
513
|
+
* used to JIT-migrate database fields. See file-level note.
|
|
514
|
+
*/
|
|
515
|
+
withConstructorDefault: s.pipe(S.withConstructorDefault(Effect.sync(() => new globalThis.Date()))),
|
|
516
|
+
/**
|
|
517
|
+
* Decode-time default `new Date()`. **Discouraged for persisted data:** a
|
|
518
|
+
* missing field may be data corruption, not an old-shape document;
|
|
519
|
+
* silently substituting `new Date()` hides the problem. Prefer an
|
|
520
|
+
* explicit, preferably versioned migration over a decode-time fallback.
|
|
521
|
+
* See file-level note.
|
|
522
|
+
*/
|
|
523
|
+
withDecodingDefaultType: s.pipe(S.withDecodingDefaultType(Effect.sync(() => new globalThis.Date())))
|
|
524
|
+
})
|
|
172
525
|
)
|
|
173
526
|
|
|
174
527
|
export interface UnionBrand {}
|
|
@@ -218,7 +571,7 @@ export const transformTo = <To extends S.Top, From extends S.Top>(
|
|
|
218
571
|
{ message: "One way schema transformation, encoding is not allowed" }
|
|
219
572
|
)
|
|
220
573
|
)
|
|
221
|
-
})
|
|
574
|
+
})
|
|
222
575
|
)
|
|
223
576
|
)
|
|
224
577
|
|
|
@@ -235,7 +588,7 @@ export const transformToOrFail = <To extends S.Top, From extends S.Top, RD>(
|
|
|
235
588
|
S.decodeTo(
|
|
236
589
|
to,
|
|
237
590
|
SchemaTransformation.transformOrFail({
|
|
238
|
-
decode
|
|
591
|
+
decode,
|
|
239
592
|
encode: (i: any) =>
|
|
240
593
|
Effect.fail(
|
|
241
594
|
new SchemaIssue.Forbidden(
|
|
@@ -243,34 +596,36 @@ export const transformToOrFail = <To extends S.Top, From extends S.Top, RD>(
|
|
|
243
596
|
{ message: "One way schema transformation, encoding is not allowed" }
|
|
244
597
|
)
|
|
245
598
|
)
|
|
246
|
-
})
|
|
599
|
+
})
|
|
247
600
|
)
|
|
248
601
|
)
|
|
249
602
|
|
|
250
|
-
export const provide
|
|
251
|
-
self: Self,
|
|
252
|
-
context:
|
|
253
|
-
): ProvidedCodec<Self, R> => {
|
|
603
|
+
export const provide: {
|
|
604
|
+
<R>(context: Context.Context<R>): <Self extends S.Top>(self: Self) => ProvidedCodec<Self, R>
|
|
605
|
+
<Self extends S.Top, R>(self: Self, context: Context.Context<R>): ProvidedCodec<Self, R>
|
|
606
|
+
} = Function.dual(2, <Self extends S.Top, R>(self: Self, context: Context.Context<R>): ProvidedCodec<Self, R> => {
|
|
254
607
|
const prov = Effect.provide(context)
|
|
255
608
|
return self.pipe(
|
|
256
609
|
S.middlewareDecoding((effect) => prov(effect)),
|
|
257
610
|
S.middlewareEncoding((effect) => prov(effect))
|
|
258
|
-
)
|
|
259
|
-
}
|
|
260
|
-
export const contextFromServices =
|
|
611
|
+
)
|
|
612
|
+
})
|
|
613
|
+
export const contextFromServices = Effect.fnUntraced(function*<
|
|
261
614
|
Self extends S.Top,
|
|
262
|
-
Tags extends ReadonlyArray<
|
|
615
|
+
Tags extends ReadonlyArray<Context.Key<any, any>>
|
|
616
|
+
>(self: Self, ...services: Tags) {
|
|
617
|
+
const context: Context.Context<Context.Service.Identifier<Tags[number]>> = Context.pick(...services)(
|
|
618
|
+
yield* Effect.context<Context.Service.Identifier<Tags[number]>>()
|
|
619
|
+
)
|
|
620
|
+
return provide(self, context)
|
|
621
|
+
}) as <
|
|
622
|
+
Self extends S.Top,
|
|
623
|
+
Tags extends ReadonlyArray<Context.Key<any, any>>
|
|
263
624
|
>(
|
|
264
625
|
self: Self,
|
|
265
626
|
...services: Tags
|
|
266
|
-
)
|
|
267
|
-
ProvidedCodec<Self,
|
|
627
|
+
) => Effect.Effect<
|
|
628
|
+
ProvidedCodec<Self, Context.Service.Identifier<Tags[number]>>,
|
|
268
629
|
never,
|
|
269
|
-
|
|
270
|
-
>
|
|
271
|
-
Effect.gen(function*() {
|
|
272
|
-
const context: ServiceMap.ServiceMap<ServiceMap.Service.Identifier<Tags[number]>> = ServiceMap.pick(...services)(
|
|
273
|
-
yield* Effect.services<ServiceMap.Service.Identifier<Tags[number]>>()
|
|
274
|
-
)
|
|
275
|
-
return provide(self, context)
|
|
276
|
-
})
|
|
630
|
+
Context.Service.Identifier<Tags[number]>
|
|
631
|
+
>
|