@clayroach/effect 3.19.14-source-capture.8 → 3.19.14-source-trace.2
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/SourceLocation/package.json +6 -0
- package/dist/cjs/Effect.js +2 -28
- package/dist/cjs/Effect.js.map +1 -1
- package/dist/cjs/FiberRef.js +12 -1
- package/dist/cjs/FiberRef.js.map +1 -1
- package/dist/cjs/Layer.js +2 -24
- package/dist/cjs/Layer.js.map +1 -1
- package/dist/cjs/RuntimeFlags.js +1 -29
- package/dist/cjs/RuntimeFlags.js.map +1 -1
- package/dist/cjs/SourceLocation.js +60 -0
- package/dist/cjs/SourceLocation.js.map +1 -0
- package/dist/cjs/Tracer.js +1 -15
- package/dist/cjs/Tracer.js.map +1 -1
- package/dist/cjs/Utils.js +1 -1
- package/dist/cjs/Utils.js.map +1 -1
- package/dist/cjs/index.js +3 -1
- package/dist/cjs/index.js.map +1 -1
- package/dist/cjs/internal/clock.js +1 -1
- package/dist/cjs/internal/clock.js.map +1 -1
- package/dist/cjs/internal/core.js +17 -50
- package/dist/cjs/internal/core.js.map +1 -1
- package/dist/cjs/internal/effect/circular.js +18 -30
- package/dist/cjs/internal/effect/circular.js.map +1 -1
- package/dist/cjs/internal/fiberRuntime.js +16 -65
- package/dist/cjs/internal/fiberRuntime.js.map +1 -1
- package/dist/cjs/internal/layer/circular.js +1 -5
- package/dist/cjs/internal/layer/circular.js.map +1 -1
- package/dist/cjs/internal/layer.js +1 -3
- package/dist/cjs/internal/layer.js.map +1 -1
- package/dist/cjs/internal/logger.js +25 -2
- package/dist/cjs/internal/logger.js.map +1 -1
- package/dist/cjs/internal/runtimeFlags.js +2 -11
- package/dist/cjs/internal/runtimeFlags.js.map +1 -1
- package/dist/cjs/internal/tracer.js +1 -114
- package/dist/cjs/internal/tracer.js.map +1 -1
- package/dist/dts/Config.d.ts +2 -2
- package/dist/dts/Config.d.ts.map +1 -1
- package/dist/dts/Effect.d.ts +8 -29
- package/dist/dts/Effect.d.ts.map +1 -1
- package/dist/dts/FiberRef.d.ts +12 -0
- package/dist/dts/FiberRef.d.ts.map +1 -1
- package/dist/dts/Layer.d.ts +0 -22
- package/dist/dts/Layer.d.ts.map +1 -1
- package/dist/dts/RuntimeFlags.d.ts +0 -28
- package/dist/dts/RuntimeFlags.d.ts.map +1 -1
- package/dist/dts/SourceLocation.d.ts +88 -0
- package/dist/dts/SourceLocation.d.ts.map +1 -0
- package/dist/dts/Tracer.d.ts +0 -15
- package/dist/dts/Tracer.d.ts.map +1 -1
- package/dist/dts/index.d.ts +6 -0
- package/dist/dts/index.d.ts.map +1 -1
- package/dist/dts/internal/core.d.ts.map +1 -1
- package/dist/dts/internal/layer.d.ts.map +1 -1
- package/dist/dts/internal/runtimeFlags.d.ts.map +1 -1
- package/dist/esm/Effect.js +0 -26
- package/dist/esm/Effect.js.map +1 -1
- package/dist/esm/FiberRef.js +11 -0
- package/dist/esm/FiberRef.js.map +1 -1
- package/dist/esm/Layer.js +0 -22
- package/dist/esm/Layer.js.map +1 -1
- package/dist/esm/RuntimeFlags.js +0 -28
- package/dist/esm/RuntimeFlags.js.map +1 -1
- package/dist/esm/SourceLocation.js +51 -0
- package/dist/esm/SourceLocation.js.map +1 -0
- package/dist/esm/Tracer.js +0 -14
- package/dist/esm/Tracer.js.map +1 -1
- package/dist/esm/Utils.js +1 -1
- package/dist/esm/Utils.js.map +1 -1
- package/dist/esm/index.js +6 -0
- package/dist/esm/index.js.map +1 -1
- package/dist/esm/internal/clock.js +1 -1
- package/dist/esm/internal/clock.js.map +1 -1
- package/dist/esm/internal/core.js +12 -45
- package/dist/esm/internal/core.js.map +1 -1
- package/dist/esm/internal/effect/circular.js +18 -30
- package/dist/esm/internal/effect/circular.js.map +1 -1
- package/dist/esm/internal/fiberRuntime.js +13 -60
- package/dist/esm/internal/fiberRuntime.js.map +1 -1
- package/dist/esm/internal/layer/circular.js +0 -4
- package/dist/esm/internal/layer/circular.js.map +1 -1
- package/dist/esm/internal/layer.js +0 -2
- package/dist/esm/internal/layer.js.map +1 -1
- package/dist/esm/internal/logger.js +25 -2
- package/dist/esm/internal/logger.js.map +1 -1
- package/dist/esm/internal/runtimeFlags.js +1 -9
- package/dist/esm/internal/runtimeFlags.js.map +1 -1
- package/dist/esm/internal/tracer.js +0 -111
- package/dist/esm/internal/tracer.js.map +1 -1
- package/package.json +12 -1
- package/src/Arbitrary.ts +1101 -0
- package/src/Array.ts +3589 -0
- package/src/BigDecimal.ts +1349 -0
- package/src/BigInt.ts +643 -0
- package/src/Boolean.ts +287 -0
- package/src/Brand.ts +360 -0
- package/src/Cache.ts +281 -0
- package/src/Cause.ts +1555 -0
- package/src/Channel.ts +2355 -0
- package/src/ChildExecutorDecision.ts +146 -0
- package/src/Chunk.ts +1495 -0
- package/src/Clock.ts +111 -0
- package/src/Config.ts +542 -0
- package/src/ConfigError.ts +270 -0
- package/src/ConfigProvider.ts +333 -0
- package/src/ConfigProviderPathPatch.ts +100 -0
- package/src/Console.ts +226 -0
- package/src/Context.ts +585 -0
- package/src/Cron.ts +706 -0
- package/src/Data.ts +596 -0
- package/src/DateTime.ts +1686 -0
- package/src/DefaultServices.ts +34 -0
- package/src/Deferred.ts +301 -0
- package/src/Differ.ts +450 -0
- package/src/Duration.ts +1000 -0
- package/src/Effect.ts +14817 -0
- package/src/Effectable.ts +107 -0
- package/src/Either.ts +1040 -0
- package/src/Encoding.ts +195 -0
- package/src/Equal.ts +98 -0
- package/src/Equivalence.ts +235 -0
- package/src/ExecutionPlan.ts +308 -0
- package/src/ExecutionStrategy.ts +119 -0
- package/src/Exit.ts +467 -0
- package/src/FastCheck.ts +9 -0
- package/src/Fiber.ts +744 -0
- package/src/FiberHandle.ts +540 -0
- package/src/FiberId.ts +195 -0
- package/src/FiberMap.ts +656 -0
- package/src/FiberRef.ts +444 -0
- package/src/FiberRefs.ts +204 -0
- package/src/FiberRefsPatch.ts +105 -0
- package/src/FiberSet.ts +491 -0
- package/src/FiberStatus.ts +108 -0
- package/src/Function.ts +1222 -0
- package/src/GlobalValue.ts +53 -0
- package/src/Graph.ts +3732 -0
- package/src/GroupBy.ts +103 -0
- package/src/HKT.ts +45 -0
- package/src/Hash.ts +195 -0
- package/src/HashMap.ts +519 -0
- package/src/HashRing.ts +317 -0
- package/src/HashSet.ts +2346 -0
- package/src/Inspectable.ts +287 -0
- package/src/Iterable.ts +1119 -0
- package/src/JSONSchema.ts +1044 -0
- package/src/KeyedPool.ts +167 -0
- package/src/Layer.ts +1228 -0
- package/src/LayerMap.ts +436 -0
- package/src/List.ts +977 -0
- package/src/LogLevel.ts +285 -0
- package/src/LogSpan.ts +25 -0
- package/src/Logger.ts +702 -0
- package/src/Mailbox.ts +268 -0
- package/src/ManagedRuntime.ts +180 -0
- package/src/Match.ts +1477 -0
- package/src/MergeDecision.ts +95 -0
- package/src/MergeState.ts +172 -0
- package/src/MergeStrategy.ts +107 -0
- package/src/Metric.ts +780 -0
- package/src/MetricBoundaries.ts +69 -0
- package/src/MetricHook.ts +151 -0
- package/src/MetricKey.ts +224 -0
- package/src/MetricKeyType.ts +262 -0
- package/src/MetricLabel.ts +47 -0
- package/src/MetricPair.ts +71 -0
- package/src/MetricPolling.ts +148 -0
- package/src/MetricRegistry.ts +48 -0
- package/src/MetricState.ts +257 -0
- package/src/Micro.ts +4405 -0
- package/src/ModuleVersion.ts +18 -0
- package/src/MutableHashMap.ts +411 -0
- package/src/MutableHashSet.ts +706 -0
- package/src/MutableList.ts +297 -0
- package/src/MutableQueue.ts +227 -0
- package/src/MutableRef.ts +202 -0
- package/src/NonEmptyIterable.ts +32 -0
- package/src/Number.ts +1071 -0
- package/src/Option.ts +2170 -0
- package/src/Order.ts +373 -0
- package/src/Ordering.ts +111 -0
- package/src/ParseResult.ts +2031 -0
- package/src/PartitionedSemaphore.ts +200 -0
- package/src/Pipeable.ts +566 -0
- package/src/Pool.ts +204 -0
- package/src/Predicate.ts +1405 -0
- package/src/Pretty.ts +205 -0
- package/src/PrimaryKey.ts +23 -0
- package/src/PubSub.ts +182 -0
- package/src/Queue.ts +644 -0
- package/src/Random.ts +204 -0
- package/src/RateLimiter.ts +138 -0
- package/src/RcMap.ts +141 -0
- package/src/RcRef.ts +122 -0
- package/src/Readable.ts +93 -0
- package/src/Record.ts +1274 -0
- package/src/RedBlackTree.ts +421 -0
- package/src/Redacted.ts +144 -0
- package/src/Ref.ts +180 -0
- package/src/RegExp.ts +38 -0
- package/src/Reloadable.ts +127 -0
- package/src/Request.ts +347 -0
- package/src/RequestBlock.ts +118 -0
- package/src/RequestResolver.ts +366 -0
- package/src/Resource.ts +119 -0
- package/src/Runtime.ts +383 -0
- package/src/RuntimeFlags.ts +336 -0
- package/src/RuntimeFlagsPatch.ts +183 -0
- package/src/STM.ts +2045 -0
- package/src/Schedule.ts +2219 -0
- package/src/ScheduleDecision.ts +62 -0
- package/src/ScheduleInterval.ts +151 -0
- package/src/ScheduleIntervals.ts +122 -0
- package/src/Scheduler.ts +353 -0
- package/src/Schema.ts +10914 -0
- package/src/SchemaAST.ts +3043 -0
- package/src/Scope.ts +204 -0
- package/src/ScopedCache.ts +151 -0
- package/src/ScopedRef.ts +117 -0
- package/src/Secret.ts +88 -0
- package/src/SingleProducerAsyncInput.ts +67 -0
- package/src/Sink.ts +1461 -0
- package/src/SortedMap.ts +287 -0
- package/src/SortedSet.ts +390 -0
- package/src/SourceLocation.ts +108 -0
- package/src/Stream.ts +6468 -0
- package/src/StreamEmit.ts +136 -0
- package/src/StreamHaltStrategy.ts +123 -0
- package/src/Streamable.ts +45 -0
- package/src/String.ts +778 -0
- package/src/Struct.ts +243 -0
- package/src/Subscribable.ts +100 -0
- package/src/SubscriptionRef.ts +298 -0
- package/src/Supervisor.ts +240 -0
- package/src/Symbol.ts +29 -0
- package/src/SynchronizedRef.ts +270 -0
- package/src/TArray.ts +495 -0
- package/src/TDeferred.ts +100 -0
- package/src/TMap.ts +515 -0
- package/src/TPriorityQueue.ts +223 -0
- package/src/TPubSub.ts +200 -0
- package/src/TQueue.ts +432 -0
- package/src/TRandom.ts +129 -0
- package/src/TReentrantLock.ts +224 -0
- package/src/TRef.ts +178 -0
- package/src/TSemaphore.ts +129 -0
- package/src/TSet.ts +365 -0
- package/src/TSubscriptionRef.ts +192 -0
- package/src/Take.ts +258 -0
- package/src/TestAnnotation.ts +158 -0
- package/src/TestAnnotationMap.ts +119 -0
- package/src/TestAnnotations.ts +117 -0
- package/src/TestClock.ts +556 -0
- package/src/TestConfig.ts +47 -0
- package/src/TestContext.ts +36 -0
- package/src/TestLive.ts +53 -0
- package/src/TestServices.ts +390 -0
- package/src/TestSized.ts +55 -0
- package/src/Tracer.ts +182 -0
- package/src/Trie.ts +840 -0
- package/src/Tuple.ts +305 -0
- package/src/Types.ts +353 -0
- package/src/Unify.ts +113 -0
- package/src/UpstreamPullRequest.ts +117 -0
- package/src/UpstreamPullStrategy.ts +121 -0
- package/src/Utils.ts +809 -0
- package/src/index.ts +1568 -0
- package/src/internal/array.ts +8 -0
- package/src/internal/blockedRequests.ts +520 -0
- package/src/internal/cache.ts +733 -0
- package/src/internal/cause.ts +1050 -0
- package/src/internal/channel/channelExecutor.ts +1200 -0
- package/src/internal/channel/channelState.ts +134 -0
- package/src/internal/channel/childExecutorDecision.ts +96 -0
- package/src/internal/channel/continuation.ts +200 -0
- package/src/internal/channel/mergeDecision.ts +113 -0
- package/src/internal/channel/mergeState.ts +120 -0
- package/src/internal/channel/mergeStrategy.ts +72 -0
- package/src/internal/channel/singleProducerAsyncInput.ts +259 -0
- package/src/internal/channel/subexecutor.ts +229 -0
- package/src/internal/channel/upstreamPullRequest.ts +84 -0
- package/src/internal/channel/upstreamPullStrategy.ts +87 -0
- package/src/internal/channel.ts +2603 -0
- package/src/internal/clock.ts +95 -0
- package/src/internal/completedRequestMap.ts +9 -0
- package/src/internal/concurrency.ts +54 -0
- package/src/internal/config.ts +716 -0
- package/src/internal/configError.ts +304 -0
- package/src/internal/configProvider/pathPatch.ts +97 -0
- package/src/internal/configProvider.ts +799 -0
- package/src/internal/console.ts +153 -0
- package/src/internal/context.ts +337 -0
- package/src/internal/core-effect.ts +2293 -0
- package/src/internal/core-stream.ts +998 -0
- package/src/internal/core.ts +3189 -0
- package/src/internal/data.ts +36 -0
- package/src/internal/dataSource.ts +327 -0
- package/src/internal/dateTime.ts +1277 -0
- package/src/internal/defaultServices/console.ts +100 -0
- package/src/internal/defaultServices.ts +163 -0
- package/src/internal/deferred.ts +46 -0
- package/src/internal/differ/chunkPatch.ts +211 -0
- package/src/internal/differ/contextPatch.ts +232 -0
- package/src/internal/differ/hashMapPatch.ts +220 -0
- package/src/internal/differ/hashSetPatch.ts +176 -0
- package/src/internal/differ/orPatch.ts +311 -0
- package/src/internal/differ/readonlyArrayPatch.ts +210 -0
- package/src/internal/differ.ts +200 -0
- package/src/internal/doNotation.ts +80 -0
- package/src/internal/effect/circular.ts +895 -0
- package/src/internal/effectable.ts +131 -0
- package/src/internal/either.ts +110 -0
- package/src/internal/encoding/base64.ts +286 -0
- package/src/internal/encoding/base64Url.ts +29 -0
- package/src/internal/encoding/common.ts +51 -0
- package/src/internal/encoding/hex.ts +315 -0
- package/src/internal/errors.ts +7 -0
- package/src/internal/executionPlan.ts +114 -0
- package/src/internal/executionStrategy.ts +74 -0
- package/src/internal/fiber.ts +388 -0
- package/src/internal/fiberId.ts +267 -0
- package/src/internal/fiberMessage.ts +82 -0
- package/src/internal/fiberRefs/patch.ts +144 -0
- package/src/internal/fiberRefs.ts +297 -0
- package/src/internal/fiberRuntime.ts +3842 -0
- package/src/internal/fiberScope.ts +71 -0
- package/src/internal/fiberStatus.ts +119 -0
- package/src/internal/groupBy.ts +530 -0
- package/src/internal/hashMap/array.ts +49 -0
- package/src/internal/hashMap/bitwise.ts +32 -0
- package/src/internal/hashMap/config.ts +14 -0
- package/src/internal/hashMap/keySet.ts +8 -0
- package/src/internal/hashMap/node.ts +391 -0
- package/src/internal/hashMap.ts +586 -0
- package/src/internal/hashSet.ts +323 -0
- package/src/internal/keyedPool.ts +244 -0
- package/src/internal/layer/circular.ts +214 -0
- package/src/internal/layer.ts +1483 -0
- package/src/internal/logSpan.ts +20 -0
- package/src/internal/logger-circular.ts +24 -0
- package/src/internal/logger.ts +522 -0
- package/src/internal/mailbox.ts +561 -0
- package/src/internal/managedRuntime/circular.ts +6 -0
- package/src/internal/managedRuntime.ts +134 -0
- package/src/internal/matcher.ts +652 -0
- package/src/internal/metric/boundaries.ts +75 -0
- package/src/internal/metric/hook.ts +483 -0
- package/src/internal/metric/key.ts +167 -0
- package/src/internal/metric/keyType.ts +238 -0
- package/src/internal/metric/label.ts +41 -0
- package/src/internal/metric/pair.ts +48 -0
- package/src/internal/metric/polling.ts +149 -0
- package/src/internal/metric/registry.ts +187 -0
- package/src/internal/metric/state.ts +290 -0
- package/src/internal/metric.ts +577 -0
- package/src/internal/opCodes/cause.ts +35 -0
- package/src/internal/opCodes/channel.ts +83 -0
- package/src/internal/opCodes/channelChildExecutorDecision.ts +17 -0
- package/src/internal/opCodes/channelMergeDecision.ts +11 -0
- package/src/internal/opCodes/channelMergeState.ts +17 -0
- package/src/internal/opCodes/channelMergeStrategy.ts +11 -0
- package/src/internal/opCodes/channelState.ts +23 -0
- package/src/internal/opCodes/channelUpstreamPullRequest.ts +11 -0
- package/src/internal/opCodes/channelUpstreamPullStrategy.ts +11 -0
- package/src/internal/opCodes/config.ts +65 -0
- package/src/internal/opCodes/configError.ts +35 -0
- package/src/internal/opCodes/continuation.ts +11 -0
- package/src/internal/opCodes/deferred.ts +11 -0
- package/src/internal/opCodes/effect.ts +89 -0
- package/src/internal/opCodes/layer.ts +59 -0
- package/src/internal/opCodes/streamHaltStrategy.ts +23 -0
- package/src/internal/option.ts +80 -0
- package/src/internal/pool.ts +432 -0
- package/src/internal/pubsub.ts +1762 -0
- package/src/internal/query.ts +204 -0
- package/src/internal/queue.ts +766 -0
- package/src/internal/random.ts +161 -0
- package/src/internal/rateLimiter.ts +93 -0
- package/src/internal/rcMap.ts +285 -0
- package/src/internal/rcRef.ts +192 -0
- package/src/internal/redBlackTree/iterator.ts +200 -0
- package/src/internal/redBlackTree/node.ts +68 -0
- package/src/internal/redBlackTree.ts +1245 -0
- package/src/internal/redacted.ts +73 -0
- package/src/internal/ref.ts +171 -0
- package/src/internal/reloadable.ts +140 -0
- package/src/internal/request.ts +177 -0
- package/src/internal/resource.ts +76 -0
- package/src/internal/ringBuffer.ts +68 -0
- package/src/internal/runtime.ts +558 -0
- package/src/internal/runtimeFlags.ts +178 -0
- package/src/internal/runtimeFlagsPatch.ts +103 -0
- package/src/internal/schedule/decision.ts +47 -0
- package/src/internal/schedule/interval.ts +101 -0
- package/src/internal/schedule/intervals.ts +180 -0
- package/src/internal/schedule.ts +2199 -0
- package/src/internal/schema/errors.ts +191 -0
- package/src/internal/schema/schemaId.ts +106 -0
- package/src/internal/schema/util.ts +50 -0
- package/src/internal/scopedCache.ts +644 -0
- package/src/internal/scopedRef.ts +118 -0
- package/src/internal/secret.ts +89 -0
- package/src/internal/singleShotGen.ts +35 -0
- package/src/internal/sink.ts +2120 -0
- package/src/internal/stack.ts +10 -0
- package/src/internal/stm/core.ts +817 -0
- package/src/internal/stm/entry.ts +59 -0
- package/src/internal/stm/journal.ts +123 -0
- package/src/internal/stm/opCodes/stm.ts +71 -0
- package/src/internal/stm/opCodes/stmState.ts +17 -0
- package/src/internal/stm/opCodes/strategy.ts +17 -0
- package/src/internal/stm/opCodes/tExit.ts +29 -0
- package/src/internal/stm/opCodes/tryCommit.ts +11 -0
- package/src/internal/stm/stm.ts +1453 -0
- package/src/internal/stm/stmState.ts +136 -0
- package/src/internal/stm/tArray.ts +550 -0
- package/src/internal/stm/tDeferred.ts +81 -0
- package/src/internal/stm/tExit.ts +190 -0
- package/src/internal/stm/tMap.ts +824 -0
- package/src/internal/stm/tPriorityQueue.ts +267 -0
- package/src/internal/stm/tPubSub.ts +551 -0
- package/src/internal/stm/tQueue.ts +393 -0
- package/src/internal/stm/tRandom.ts +140 -0
- package/src/internal/stm/tReentrantLock.ts +352 -0
- package/src/internal/stm/tRef.ts +195 -0
- package/src/internal/stm/tSemaphore.ts +113 -0
- package/src/internal/stm/tSet.ts +259 -0
- package/src/internal/stm/tSubscriptionRef.ts +286 -0
- package/src/internal/stm/tryCommit.ts +34 -0
- package/src/internal/stm/txnId.ts +14 -0
- package/src/internal/stm/versioned.ts +4 -0
- package/src/internal/stream/debounceState.ts +57 -0
- package/src/internal/stream/emit.ts +123 -0
- package/src/internal/stream/haltStrategy.ts +94 -0
- package/src/internal/stream/handoff.ts +187 -0
- package/src/internal/stream/handoffSignal.ts +59 -0
- package/src/internal/stream/pull.ts +34 -0
- package/src/internal/stream/sinkEndReason.ts +30 -0
- package/src/internal/stream/zipAllState.ts +88 -0
- package/src/internal/stream/zipChunksState.ts +56 -0
- package/src/internal/stream.ts +8801 -0
- package/src/internal/string-utils.ts +107 -0
- package/src/internal/subscriptionRef.ts +138 -0
- package/src/internal/supervisor/patch.ts +190 -0
- package/src/internal/supervisor.ts +303 -0
- package/src/internal/synchronizedRef.ts +114 -0
- package/src/internal/take.ts +199 -0
- package/src/internal/testing/sleep.ts +27 -0
- package/src/internal/testing/suspendedWarningData.ts +85 -0
- package/src/internal/testing/warningData.ts +94 -0
- package/src/internal/tracer.ts +150 -0
- package/src/internal/trie.ts +722 -0
- package/src/internal/version.ts +7 -0
package/src/Boolean.ts
ADDED
|
@@ -0,0 +1,287 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* This module provides utility functions and type class instances for working with the `boolean` type in TypeScript.
|
|
3
|
+
* It includes functions for basic boolean operations, as well as type class instances for
|
|
4
|
+
* `Equivalence` and `Order`.
|
|
5
|
+
*
|
|
6
|
+
* @since 2.0.0
|
|
7
|
+
*/
|
|
8
|
+
import * as equivalence from "./Equivalence.js"
|
|
9
|
+
import type { LazyArg } from "./Function.js"
|
|
10
|
+
import { dual } from "./Function.js"
|
|
11
|
+
import * as order from "./Order.js"
|
|
12
|
+
import * as predicate from "./Predicate.js"
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Tests if a value is a `boolean`.
|
|
16
|
+
*
|
|
17
|
+
* @example
|
|
18
|
+
* ```ts
|
|
19
|
+
* import * as assert from "node:assert"
|
|
20
|
+
* import { isBoolean } from "effect/Boolean"
|
|
21
|
+
*
|
|
22
|
+
* assert.deepStrictEqual(isBoolean(true), true)
|
|
23
|
+
* assert.deepStrictEqual(isBoolean("true"), false)
|
|
24
|
+
* ```
|
|
25
|
+
*
|
|
26
|
+
* @category guards
|
|
27
|
+
* @since 2.0.0
|
|
28
|
+
*/
|
|
29
|
+
export const isBoolean: (input: unknown) => input is boolean = predicate.isBoolean
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* This function returns the result of either of the given functions depending on the value of the boolean parameter.
|
|
33
|
+
* It is useful when you have to run one of two functions depending on the boolean value.
|
|
34
|
+
*
|
|
35
|
+
* @example
|
|
36
|
+
* ```ts
|
|
37
|
+
* import * as assert from "node:assert"
|
|
38
|
+
* import { Boolean } from "effect"
|
|
39
|
+
*
|
|
40
|
+
* assert.deepStrictEqual(Boolean.match(true, { onFalse: () => "It's false!", onTrue: () => "It's true!" }), "It's true!")
|
|
41
|
+
* ```
|
|
42
|
+
*
|
|
43
|
+
* @category pattern matching
|
|
44
|
+
* @since 2.0.0
|
|
45
|
+
*/
|
|
46
|
+
export const match: {
|
|
47
|
+
<A, B = A>(options: {
|
|
48
|
+
readonly onFalse: LazyArg<A>
|
|
49
|
+
readonly onTrue: LazyArg<B>
|
|
50
|
+
}): (value: boolean) => A | B
|
|
51
|
+
<A, B>(value: boolean, options: {
|
|
52
|
+
readonly onFalse: LazyArg<A>
|
|
53
|
+
readonly onTrue: LazyArg<B>
|
|
54
|
+
}): A | B
|
|
55
|
+
} = dual(2, <A, B>(value: boolean, options: {
|
|
56
|
+
readonly onFalse: LazyArg<A>
|
|
57
|
+
readonly onTrue: LazyArg<B>
|
|
58
|
+
}): A | B => value ? options.onTrue() : options.onFalse())
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* @category instances
|
|
62
|
+
* @since 2.0.0
|
|
63
|
+
*/
|
|
64
|
+
export const Equivalence: equivalence.Equivalence<boolean> = equivalence.boolean
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* @category instances
|
|
68
|
+
* @since 2.0.0
|
|
69
|
+
*/
|
|
70
|
+
export const Order: order.Order<boolean> = order.boolean
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Negates the given boolean: `!self`
|
|
74
|
+
*
|
|
75
|
+
* @example
|
|
76
|
+
* ```ts
|
|
77
|
+
* import * as assert from "node:assert"
|
|
78
|
+
* import { not } from "effect/Boolean"
|
|
79
|
+
*
|
|
80
|
+
* assert.deepStrictEqual(not(true), false)
|
|
81
|
+
* assert.deepStrictEqual(not(false), true)
|
|
82
|
+
* ```
|
|
83
|
+
*
|
|
84
|
+
* @category combinators
|
|
85
|
+
* @since 2.0.0
|
|
86
|
+
*/
|
|
87
|
+
export const not = (self: boolean): boolean => !self
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Combines two boolean using AND: `self && that`.
|
|
91
|
+
*
|
|
92
|
+
* @example
|
|
93
|
+
* ```ts
|
|
94
|
+
* import * as assert from "node:assert"
|
|
95
|
+
* import { and } from "effect/Boolean"
|
|
96
|
+
*
|
|
97
|
+
* assert.deepStrictEqual(and(true, true), true)
|
|
98
|
+
* assert.deepStrictEqual(and(true, false), false)
|
|
99
|
+
* assert.deepStrictEqual(and(false, true), false)
|
|
100
|
+
* assert.deepStrictEqual(and(false, false), false)
|
|
101
|
+
* ```
|
|
102
|
+
*
|
|
103
|
+
* @category combinators
|
|
104
|
+
* @since 2.0.0
|
|
105
|
+
*/
|
|
106
|
+
export const and: {
|
|
107
|
+
(that: boolean): (self: boolean) => boolean
|
|
108
|
+
(self: boolean, that: boolean): boolean
|
|
109
|
+
} = dual(2, (self: boolean, that: boolean): boolean => self && that)
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Combines two boolean using NAND: `!(self && that)`.
|
|
113
|
+
*
|
|
114
|
+
* @example
|
|
115
|
+
* ```ts
|
|
116
|
+
* import * as assert from "node:assert"
|
|
117
|
+
* import { nand } from "effect/Boolean"
|
|
118
|
+
*
|
|
119
|
+
* assert.deepStrictEqual(nand(true, true), false)
|
|
120
|
+
* assert.deepStrictEqual(nand(true, false), true)
|
|
121
|
+
* assert.deepStrictEqual(nand(false, true), true)
|
|
122
|
+
* assert.deepStrictEqual(nand(false, false), true)
|
|
123
|
+
* ```
|
|
124
|
+
*
|
|
125
|
+
* @category combinators
|
|
126
|
+
* @since 2.0.0
|
|
127
|
+
*/
|
|
128
|
+
export const nand: {
|
|
129
|
+
(that: boolean): (self: boolean) => boolean
|
|
130
|
+
(self: boolean, that: boolean): boolean
|
|
131
|
+
} = dual(2, (self: boolean, that: boolean): boolean => !(self && that))
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Combines two boolean using OR: `self || that`.
|
|
135
|
+
*
|
|
136
|
+
* @example
|
|
137
|
+
* ```ts
|
|
138
|
+
* import * as assert from "node:assert"
|
|
139
|
+
* import { or } from "effect/Boolean"
|
|
140
|
+
*
|
|
141
|
+
* assert.deepStrictEqual(or(true, true), true)
|
|
142
|
+
* assert.deepStrictEqual(or(true, false), true)
|
|
143
|
+
* assert.deepStrictEqual(or(false, true), true)
|
|
144
|
+
* assert.deepStrictEqual(or(false, false), false)
|
|
145
|
+
* ```
|
|
146
|
+
*
|
|
147
|
+
* @category combinators
|
|
148
|
+
* @since 2.0.0
|
|
149
|
+
*/
|
|
150
|
+
export const or: {
|
|
151
|
+
(that: boolean): (self: boolean) => boolean
|
|
152
|
+
(self: boolean, that: boolean): boolean
|
|
153
|
+
} = dual(2, (self: boolean, that: boolean): boolean => self || that)
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* Combines two booleans using NOR: `!(self || that)`.
|
|
157
|
+
*
|
|
158
|
+
* @example
|
|
159
|
+
* ```ts
|
|
160
|
+
* import * as assert from "node:assert"
|
|
161
|
+
* import { nor } from "effect/Boolean"
|
|
162
|
+
*
|
|
163
|
+
* assert.deepStrictEqual(nor(true, true), false)
|
|
164
|
+
* assert.deepStrictEqual(nor(true, false), false)
|
|
165
|
+
* assert.deepStrictEqual(nor(false, true), false)
|
|
166
|
+
* assert.deepStrictEqual(nor(false, false), true)
|
|
167
|
+
* ```
|
|
168
|
+
*
|
|
169
|
+
* @category combinators
|
|
170
|
+
* @since 2.0.0
|
|
171
|
+
*/
|
|
172
|
+
export const nor: {
|
|
173
|
+
(that: boolean): (self: boolean) => boolean
|
|
174
|
+
(self: boolean, that: boolean): boolean
|
|
175
|
+
} = dual(2, (self: boolean, that: boolean): boolean => !(self || that))
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* Combines two booleans using XOR: `(!self && that) || (self && !that)`.
|
|
179
|
+
*
|
|
180
|
+
* @example
|
|
181
|
+
* ```ts
|
|
182
|
+
* import * as assert from "node:assert"
|
|
183
|
+
* import { xor } from "effect/Boolean"
|
|
184
|
+
*
|
|
185
|
+
* assert.deepStrictEqual(xor(true, true), false)
|
|
186
|
+
* assert.deepStrictEqual(xor(true, false), true)
|
|
187
|
+
* assert.deepStrictEqual(xor(false, true), true)
|
|
188
|
+
* assert.deepStrictEqual(xor(false, false), false)
|
|
189
|
+
* ```
|
|
190
|
+
*
|
|
191
|
+
* @category combinators
|
|
192
|
+
* @since 2.0.0
|
|
193
|
+
*/
|
|
194
|
+
export const xor: {
|
|
195
|
+
(that: boolean): (self: boolean) => boolean
|
|
196
|
+
(self: boolean, that: boolean): boolean
|
|
197
|
+
} = dual(2, (self: boolean, that: boolean): boolean => (!self && that) || (self && !that))
|
|
198
|
+
|
|
199
|
+
/**
|
|
200
|
+
* Combines two booleans using EQV (aka XNOR): `!xor(self, that)`.
|
|
201
|
+
*
|
|
202
|
+
* @example
|
|
203
|
+
* ```ts
|
|
204
|
+
* import * as assert from "node:assert"
|
|
205
|
+
* import { eqv } from "effect/Boolean"
|
|
206
|
+
*
|
|
207
|
+
* assert.deepStrictEqual(eqv(true, true), true)
|
|
208
|
+
* assert.deepStrictEqual(eqv(true, false), false)
|
|
209
|
+
* assert.deepStrictEqual(eqv(false, true), false)
|
|
210
|
+
* assert.deepStrictEqual(eqv(false, false), true)
|
|
211
|
+
* ```
|
|
212
|
+
*
|
|
213
|
+
* @category combinators
|
|
214
|
+
* @since 2.0.0
|
|
215
|
+
*/
|
|
216
|
+
export const eqv: {
|
|
217
|
+
(that: boolean): (self: boolean) => boolean
|
|
218
|
+
(self: boolean, that: boolean): boolean
|
|
219
|
+
} = dual(2, (self: boolean, that: boolean): boolean => !xor(self, that))
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* Combines two booleans using an implication: `(!self || that)`.
|
|
223
|
+
*
|
|
224
|
+
* @example
|
|
225
|
+
* ```ts
|
|
226
|
+
* import * as assert from "node:assert"
|
|
227
|
+
* import { implies } from "effect/Boolean"
|
|
228
|
+
*
|
|
229
|
+
* assert.deepStrictEqual(implies(true, true), true)
|
|
230
|
+
* assert.deepStrictEqual(implies(true, false), false)
|
|
231
|
+
* assert.deepStrictEqual(implies(false, true), true)
|
|
232
|
+
* assert.deepStrictEqual(implies(false, false), true)
|
|
233
|
+
* ```
|
|
234
|
+
*
|
|
235
|
+
* @category combinators
|
|
236
|
+
* @since 2.0.0
|
|
237
|
+
*/
|
|
238
|
+
export const implies: {
|
|
239
|
+
(that: boolean): (self: boolean) => boolean
|
|
240
|
+
(self: boolean, that: boolean): boolean
|
|
241
|
+
} = dual(2, (self, that) => self ? that : true)
|
|
242
|
+
|
|
243
|
+
/**
|
|
244
|
+
* This utility function is used to check if all the elements in a collection of boolean values are `true`.
|
|
245
|
+
*
|
|
246
|
+
* @example
|
|
247
|
+
* ```ts
|
|
248
|
+
* import * as assert from "node:assert"
|
|
249
|
+
* import { every } from "effect/Boolean"
|
|
250
|
+
*
|
|
251
|
+
* assert.deepStrictEqual(every([true, true, true]), true)
|
|
252
|
+
* assert.deepStrictEqual(every([true, false, true]), false)
|
|
253
|
+
* ```
|
|
254
|
+
*
|
|
255
|
+
* @since 2.0.0
|
|
256
|
+
*/
|
|
257
|
+
export const every = (collection: Iterable<boolean>): boolean => {
|
|
258
|
+
for (const b of collection) {
|
|
259
|
+
if (!b) {
|
|
260
|
+
return false
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
return true
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
/**
|
|
267
|
+
* This utility function is used to check if at least one of the elements in a collection of boolean values is `true`.
|
|
268
|
+
*
|
|
269
|
+
* @example
|
|
270
|
+
* ```ts
|
|
271
|
+
* import * as assert from "node:assert"
|
|
272
|
+
* import { some } from "effect/Boolean"
|
|
273
|
+
*
|
|
274
|
+
* assert.deepStrictEqual(some([true, false, true]), true)
|
|
275
|
+
* assert.deepStrictEqual(some([false, false, false]), false)
|
|
276
|
+
* ```
|
|
277
|
+
*
|
|
278
|
+
* @since 2.0.0
|
|
279
|
+
*/
|
|
280
|
+
export const some = (collection: Iterable<boolean>): boolean => {
|
|
281
|
+
for (const b of collection) {
|
|
282
|
+
if (b) {
|
|
283
|
+
return true
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
return false
|
|
287
|
+
}
|
package/src/Brand.ts
ADDED
|
@@ -0,0 +1,360 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* This module provides types and utility functions to create and work with branded types,
|
|
3
|
+
* which are TypeScript types with an added type tag to prevent accidental usage of a value in the wrong context.
|
|
4
|
+
*
|
|
5
|
+
* The `refined` and `nominal` functions are both used to create branded types in TypeScript.
|
|
6
|
+
* The main difference between them is that `refined` allows for validation of the data, while `nominal` does not.
|
|
7
|
+
*
|
|
8
|
+
* The `nominal` function is used to create a new branded type that has the same underlying type as the input, but with a different name.
|
|
9
|
+
* This is useful when you want to distinguish between two values of the same type that have different meanings.
|
|
10
|
+
* The `nominal` function does not perform any validation of the input data.
|
|
11
|
+
*
|
|
12
|
+
* On the other hand, the `refined` function is used to create a new branded type that has the same underlying type as the input,
|
|
13
|
+
* but with a different name, and it also allows for validation of the input data.
|
|
14
|
+
* The `refined` function takes a predicate that is used to validate the input data.
|
|
15
|
+
* If the input data fails the validation, a `BrandErrors` is returned, which provides information about the specific validation failure.
|
|
16
|
+
*
|
|
17
|
+
* @since 2.0.0
|
|
18
|
+
*/
|
|
19
|
+
import * as Arr from "./Array.js"
|
|
20
|
+
import * as Either from "./Either.js"
|
|
21
|
+
import { identity, unsafeCoerce } from "./Function.js"
|
|
22
|
+
import * as Option from "./Option.js"
|
|
23
|
+
import type { Predicate } from "./Predicate.js"
|
|
24
|
+
import type * as Types from "./Types.js"
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* @since 2.0.0
|
|
28
|
+
* @category symbols
|
|
29
|
+
*/
|
|
30
|
+
export const BrandTypeId: unique symbol = Symbol.for("effect/Brand")
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* @since 2.0.0
|
|
34
|
+
* @category symbols
|
|
35
|
+
*/
|
|
36
|
+
export type BrandTypeId = typeof BrandTypeId
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* @since 2.0.0
|
|
40
|
+
* @category symbols
|
|
41
|
+
*/
|
|
42
|
+
export const RefinedConstructorsTypeId: unique symbol = Symbol.for("effect/Brand/Refined")
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* @since 2.0.0
|
|
46
|
+
* @category symbols
|
|
47
|
+
*/
|
|
48
|
+
export type RefinedConstructorsTypeId = typeof RefinedConstructorsTypeId
|
|
49
|
+
|
|
50
|
+
/**
|
|
51
|
+
* A generic interface that defines a branded type.
|
|
52
|
+
*
|
|
53
|
+
* @since 2.0.0
|
|
54
|
+
* @category models
|
|
55
|
+
*/
|
|
56
|
+
export interface Brand<in out K extends string | symbol> {
|
|
57
|
+
readonly [BrandTypeId]: {
|
|
58
|
+
readonly [k in K]: K
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/**
|
|
63
|
+
* @since 2.0.0
|
|
64
|
+
*/
|
|
65
|
+
export declare namespace Brand {
|
|
66
|
+
/**
|
|
67
|
+
* Represents a list of refinement errors.
|
|
68
|
+
*
|
|
69
|
+
* @since 2.0.0
|
|
70
|
+
* @category models
|
|
71
|
+
*/
|
|
72
|
+
export interface BrandErrors extends Array<RefinementError> {}
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* Represents an error that occurs when the provided value of the branded type does not pass the refinement predicate.
|
|
76
|
+
*
|
|
77
|
+
* @since 2.0.0
|
|
78
|
+
* @category models
|
|
79
|
+
*/
|
|
80
|
+
export interface RefinementError {
|
|
81
|
+
readonly meta: unknown
|
|
82
|
+
readonly message: string
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
/**
|
|
86
|
+
* @since 2.0.0
|
|
87
|
+
* @category models
|
|
88
|
+
*/
|
|
89
|
+
export interface Constructor<in out A extends Brand<any>> {
|
|
90
|
+
readonly [RefinedConstructorsTypeId]: RefinedConstructorsTypeId
|
|
91
|
+
/**
|
|
92
|
+
* Constructs a branded type from a value of type `A`, throwing an error if
|
|
93
|
+
* the provided `A` is not valid.
|
|
94
|
+
*/
|
|
95
|
+
(args: Brand.Unbranded<A>): A
|
|
96
|
+
/**
|
|
97
|
+
* Constructs a branded type from a value of type `A`, returning `Some<A>`
|
|
98
|
+
* if the provided `A` is valid, `None` otherwise.
|
|
99
|
+
*/
|
|
100
|
+
option(args: Brand.Unbranded<A>): Option.Option<A>
|
|
101
|
+
/**
|
|
102
|
+
* Constructs a branded type from a value of type `A`, returning `Right<A>`
|
|
103
|
+
* if the provided `A` is valid, `Left<BrandError>` otherwise.
|
|
104
|
+
*/
|
|
105
|
+
either(args: Brand.Unbranded<A>): Either.Either<A, Brand.BrandErrors>
|
|
106
|
+
/**
|
|
107
|
+
* Attempts to refine the provided value of type `A`, returning `true` if
|
|
108
|
+
* the provided `A` is valid, `false` otherwise.
|
|
109
|
+
*/
|
|
110
|
+
is(a: Brand.Unbranded<A>): a is Brand.Unbranded<A> & A
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
/**
|
|
114
|
+
* A utility type to extract a branded type from a `Brand.Constructor`.
|
|
115
|
+
*
|
|
116
|
+
* @since 2.0.0
|
|
117
|
+
* @category models
|
|
118
|
+
*/
|
|
119
|
+
export type FromConstructor<A> = A extends Brand.Constructor<infer B> ? B : never
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* A utility type to extract the value type from a brand.
|
|
123
|
+
*
|
|
124
|
+
* @since 2.0.0
|
|
125
|
+
* @category models
|
|
126
|
+
*/
|
|
127
|
+
export type Unbranded<P> = P extends infer Q & Brands<P> ? Q : P
|
|
128
|
+
|
|
129
|
+
/**
|
|
130
|
+
* A utility type to extract the brands from a branded type.
|
|
131
|
+
*
|
|
132
|
+
* @since 2.0.0
|
|
133
|
+
* @category models
|
|
134
|
+
*/
|
|
135
|
+
export type Brands<P> = P extends Brand<any> ? Types.UnionToIntersection<
|
|
136
|
+
{
|
|
137
|
+
[k in keyof P[BrandTypeId]]: k extends string | symbol ? Brand<k>
|
|
138
|
+
: never
|
|
139
|
+
}[keyof P[BrandTypeId]]
|
|
140
|
+
>
|
|
141
|
+
: never
|
|
142
|
+
|
|
143
|
+
/**
|
|
144
|
+
* A utility type that checks that all brands have the same base type.
|
|
145
|
+
*
|
|
146
|
+
* @since 2.0.0
|
|
147
|
+
* @category models
|
|
148
|
+
*/
|
|
149
|
+
export type EnsureCommonBase<
|
|
150
|
+
Brands extends readonly [Brand.Constructor<any>, ...Array<Brand.Constructor<any>>]
|
|
151
|
+
> = {
|
|
152
|
+
[B in keyof Brands]: Brand.Unbranded<Brand.FromConstructor<Brands[0]>> extends
|
|
153
|
+
Brand.Unbranded<Brand.FromConstructor<Brands[B]>>
|
|
154
|
+
? Brand.Unbranded<Brand.FromConstructor<Brands[B]>> extends Brand.Unbranded<Brand.FromConstructor<Brands[0]>>
|
|
155
|
+
? Brands[B]
|
|
156
|
+
: Brands[B]
|
|
157
|
+
: "ERROR: All brands should have the same base type"
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* @category alias
|
|
163
|
+
* @since 2.0.0
|
|
164
|
+
*/
|
|
165
|
+
export type Branded<A, K extends string | symbol> = A & Brand<K>
|
|
166
|
+
|
|
167
|
+
/**
|
|
168
|
+
* Returns a `BrandErrors` that contains a single `RefinementError`.
|
|
169
|
+
*
|
|
170
|
+
* @since 2.0.0
|
|
171
|
+
* @category constructors
|
|
172
|
+
*/
|
|
173
|
+
export const error = (message: string, meta?: unknown): Brand.BrandErrors => [{
|
|
174
|
+
message,
|
|
175
|
+
meta
|
|
176
|
+
}]
|
|
177
|
+
|
|
178
|
+
/**
|
|
179
|
+
* Takes a variable number of `BrandErrors` and returns a single `BrandErrors` that contains all refinement errors.
|
|
180
|
+
*
|
|
181
|
+
* @since 2.0.0
|
|
182
|
+
* @category constructors
|
|
183
|
+
*/
|
|
184
|
+
export const errors: (...errors: Array<Brand.BrandErrors>) => Brand.BrandErrors = (
|
|
185
|
+
...errors: Array<Brand.BrandErrors>
|
|
186
|
+
): Brand.BrandErrors => Arr.flatten(errors)
|
|
187
|
+
|
|
188
|
+
/**
|
|
189
|
+
* Returns a `Brand.Constructor` that can construct a branded type from an unbranded value using the provided `refinement`
|
|
190
|
+
* predicate as validation of the input data.
|
|
191
|
+
*
|
|
192
|
+
* If you don't want to perform any validation but only distinguish between two values of the same type but with different meanings,
|
|
193
|
+
* see {@link nominal}.
|
|
194
|
+
*
|
|
195
|
+
* **Example**
|
|
196
|
+
*
|
|
197
|
+
* ```ts
|
|
198
|
+
* import * as assert from "node:assert"
|
|
199
|
+
* import { Brand } from "effect"
|
|
200
|
+
*
|
|
201
|
+
* type Int = number & Brand.Brand<"Int">
|
|
202
|
+
*
|
|
203
|
+
* const Int = Brand.refined<Int>(
|
|
204
|
+
* (n) => Number.isInteger(n),
|
|
205
|
+
* (n) => Brand.error(`Expected ${n} to be an integer`)
|
|
206
|
+
* )
|
|
207
|
+
*
|
|
208
|
+
* console.log(Int(1))
|
|
209
|
+
* // 1
|
|
210
|
+
*
|
|
211
|
+
* assert.throws(() => Int(1.1))
|
|
212
|
+
* ```
|
|
213
|
+
*
|
|
214
|
+
* @since 2.0.0
|
|
215
|
+
* @category constructors
|
|
216
|
+
*/
|
|
217
|
+
export function refined<A extends Brand<any>>(
|
|
218
|
+
f: (unbranded: Brand.Unbranded<A>) => Option.Option<Brand.BrandErrors>
|
|
219
|
+
): Brand.Constructor<A>
|
|
220
|
+
export function refined<A extends Brand<any>>(
|
|
221
|
+
refinement: Predicate<Brand.Unbranded<A>>,
|
|
222
|
+
onFailure: (unbranded: Brand.Unbranded<A>) => Brand.BrandErrors
|
|
223
|
+
): Brand.Constructor<A>
|
|
224
|
+
export function refined<A extends Brand<any>>(
|
|
225
|
+
...args: [(unbranded: Brand.Unbranded<A>) => Option.Option<Brand.BrandErrors>] | [
|
|
226
|
+
Predicate<Brand.Unbranded<A>>,
|
|
227
|
+
(unbranded: Brand.Unbranded<A>) => Brand.BrandErrors
|
|
228
|
+
]
|
|
229
|
+
): Brand.Constructor<A> {
|
|
230
|
+
const either: (unbranded: Brand.Unbranded<A>) => Either.Either<A, Brand.BrandErrors> = args.length === 2 ?
|
|
231
|
+
(unbranded) => args[0](unbranded) ? Either.right(unbranded as A) : Either.left(args[1](unbranded)) :
|
|
232
|
+
(unbranded) => {
|
|
233
|
+
return Option.match(args[0](unbranded), {
|
|
234
|
+
onNone: () => Either.right(unbranded as A),
|
|
235
|
+
onSome: Either.left
|
|
236
|
+
})
|
|
237
|
+
}
|
|
238
|
+
return Object.assign((unbranded: Brand.Unbranded<A>) => Either.getOrThrowWith(either(unbranded), identity), {
|
|
239
|
+
[RefinedConstructorsTypeId]: RefinedConstructorsTypeId,
|
|
240
|
+
option: (args: any) => Option.getRight(either(args)),
|
|
241
|
+
either,
|
|
242
|
+
is: (args: any): args is Brand.Unbranded<A> & A => Either.isRight(either(args))
|
|
243
|
+
}) as any
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
/**
|
|
247
|
+
* This function returns a `Brand.Constructor` that **does not apply any runtime checks**, it just returns the provided value.
|
|
248
|
+
* It can be used to create nominal types that allow distinguishing between two values of the same type but with different meanings.
|
|
249
|
+
*
|
|
250
|
+
* If you also want to perform some validation, see {@link refined}.
|
|
251
|
+
*
|
|
252
|
+
* **Example**
|
|
253
|
+
*
|
|
254
|
+
* ```ts
|
|
255
|
+
* import * as assert from "node:assert"
|
|
256
|
+
* import { Brand } from "effect"
|
|
257
|
+
*
|
|
258
|
+
* type UserId = number & Brand.Brand<"UserId">
|
|
259
|
+
*
|
|
260
|
+
* const UserId = Brand.nominal<UserId>()
|
|
261
|
+
*
|
|
262
|
+
* console.log(UserId(1))
|
|
263
|
+
* // 1
|
|
264
|
+
* ```
|
|
265
|
+
*
|
|
266
|
+
* @since 2.0.0
|
|
267
|
+
* @category constructors
|
|
268
|
+
*/
|
|
269
|
+
export const nominal = <A extends Brand<any>>(): Brand.Constructor<
|
|
270
|
+
A
|
|
271
|
+
> => {
|
|
272
|
+
// @ts-expect-error
|
|
273
|
+
return Object.assign((args) => args, {
|
|
274
|
+
[RefinedConstructorsTypeId]: RefinedConstructorsTypeId,
|
|
275
|
+
option: (args: any) => Option.some(args),
|
|
276
|
+
either: (args: any) => Either.right(args),
|
|
277
|
+
is: (_args: any): _args is Brand.Unbranded<A> & A => true
|
|
278
|
+
})
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
/**
|
|
282
|
+
* Combines two or more brands together to form a single branded type.
|
|
283
|
+
* This API is useful when you want to validate that the input data passes multiple brand validators.
|
|
284
|
+
*
|
|
285
|
+
* **Example**
|
|
286
|
+
*
|
|
287
|
+
* ```ts
|
|
288
|
+
* import * as assert from "node:assert"
|
|
289
|
+
* import { Brand } from "effect"
|
|
290
|
+
*
|
|
291
|
+
* type Int = number & Brand.Brand<"Int">
|
|
292
|
+
* const Int = Brand.refined<Int>(
|
|
293
|
+
* (n) => Number.isInteger(n),
|
|
294
|
+
* (n) => Brand.error(`Expected ${n} to be an integer`)
|
|
295
|
+
* )
|
|
296
|
+
* type Positive = number & Brand.Brand<"Positive">
|
|
297
|
+
* const Positive = Brand.refined<Positive>(
|
|
298
|
+
* (n) => n > 0,
|
|
299
|
+
* (n) => Brand.error(`Expected ${n} to be positive`)
|
|
300
|
+
* )
|
|
301
|
+
*
|
|
302
|
+
* const PositiveInt = Brand.all(Int, Positive)
|
|
303
|
+
*
|
|
304
|
+
* console.log(PositiveInt(1))
|
|
305
|
+
* // 1
|
|
306
|
+
*
|
|
307
|
+
* assert.throws(() => PositiveInt(1.1))
|
|
308
|
+
* ```
|
|
309
|
+
*
|
|
310
|
+
* @since 2.0.0
|
|
311
|
+
* @category combining
|
|
312
|
+
*/
|
|
313
|
+
export const all: <Brands extends readonly [Brand.Constructor<any>, ...Array<Brand.Constructor<any>>]>(
|
|
314
|
+
...brands: Brand.EnsureCommonBase<Brands>
|
|
315
|
+
) => Brand.Constructor<
|
|
316
|
+
Types.UnionToIntersection<{ [B in keyof Brands]: Brand.FromConstructor<Brands[B]> }[number]> extends
|
|
317
|
+
infer X extends Brand<any> ? X : Brand<any>
|
|
318
|
+
> = <
|
|
319
|
+
Brands extends readonly [Brand.Constructor<any>, ...Array<Brand.Constructor<any>>]
|
|
320
|
+
>(...brands: Brand.EnsureCommonBase<Brands>): Brand.Constructor<
|
|
321
|
+
Types.UnionToIntersection<
|
|
322
|
+
{
|
|
323
|
+
[B in keyof Brands]: Brand.FromConstructor<Brands[B]>
|
|
324
|
+
}[number]
|
|
325
|
+
> extends infer X extends Brand<any> ? X : Brand<any>
|
|
326
|
+
> => {
|
|
327
|
+
const either = (args: any): Either.Either<any, Brand.BrandErrors> => {
|
|
328
|
+
let result: Either.Either<any, Brand.BrandErrors> = Either.right(args)
|
|
329
|
+
for (const brand of brands) {
|
|
330
|
+
const nextResult = brand.either(args)
|
|
331
|
+
if (Either.isLeft(result) && Either.isLeft(nextResult)) {
|
|
332
|
+
result = Either.left([...result.left, ...nextResult.left])
|
|
333
|
+
} else {
|
|
334
|
+
result = Either.isLeft(result) ? result : nextResult
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
return result
|
|
338
|
+
}
|
|
339
|
+
// @ts-expect-error
|
|
340
|
+
return Object.assign((args) =>
|
|
341
|
+
Either.match(either(args), {
|
|
342
|
+
onLeft: (e) => {
|
|
343
|
+
throw e
|
|
344
|
+
},
|
|
345
|
+
onRight: identity
|
|
346
|
+
}), {
|
|
347
|
+
[RefinedConstructorsTypeId]: RefinedConstructorsTypeId,
|
|
348
|
+
option: (args: any) => Option.getRight(either(args)),
|
|
349
|
+
either,
|
|
350
|
+
is: (args: any): args is any => Either.isRight(either(args))
|
|
351
|
+
})
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
/**
|
|
355
|
+
* Retrieves the unbranded value from a `Brand` instance.
|
|
356
|
+
*
|
|
357
|
+
* @since 3.15.0
|
|
358
|
+
* @category getters
|
|
359
|
+
*/
|
|
360
|
+
export const unbranded: <A extends Brand<any>>(branded: A) => Brand.Unbranded<A> = unsafeCoerce
|