@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/Cron.ts
ADDED
|
@@ -0,0 +1,706 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @since 2.0.0
|
|
3
|
+
*/
|
|
4
|
+
import * as Arr from "./Array.js"
|
|
5
|
+
import * as Data from "./Data.js"
|
|
6
|
+
import type * as DateTime from "./DateTime.js"
|
|
7
|
+
import * as Either from "./Either.js"
|
|
8
|
+
import * as Equal from "./Equal.js"
|
|
9
|
+
import * as equivalence from "./Equivalence.js"
|
|
10
|
+
import { constVoid, dual, identity, pipe } from "./Function.js"
|
|
11
|
+
import * as Hash from "./Hash.js"
|
|
12
|
+
import { format, type Inspectable, NodeInspectSymbol } from "./Inspectable.js"
|
|
13
|
+
import * as dateTime from "./internal/dateTime.js"
|
|
14
|
+
import * as N from "./Number.js"
|
|
15
|
+
import * as Option from "./Option.js"
|
|
16
|
+
import { type Pipeable, pipeArguments } from "./Pipeable.js"
|
|
17
|
+
import { hasProperty } from "./Predicate.js"
|
|
18
|
+
import * as String from "./String.js"
|
|
19
|
+
import type { Mutable } from "./Types.js"
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* @since 2.0.0
|
|
23
|
+
* @category symbols
|
|
24
|
+
*/
|
|
25
|
+
export const TypeId: unique symbol = Symbol.for("effect/Cron")
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* @since 2.0.0
|
|
29
|
+
* @category symbol
|
|
30
|
+
*/
|
|
31
|
+
export type TypeId = typeof TypeId
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* @since 2.0.0
|
|
35
|
+
* @category models
|
|
36
|
+
*/
|
|
37
|
+
export interface Cron extends Pipeable, Equal.Equal, Inspectable {
|
|
38
|
+
readonly [TypeId]: TypeId
|
|
39
|
+
readonly tz: Option.Option<DateTime.TimeZone>
|
|
40
|
+
readonly seconds: ReadonlySet<number>
|
|
41
|
+
readonly minutes: ReadonlySet<number>
|
|
42
|
+
readonly hours: ReadonlySet<number>
|
|
43
|
+
readonly days: ReadonlySet<number>
|
|
44
|
+
readonly months: ReadonlySet<number>
|
|
45
|
+
readonly weekdays: ReadonlySet<number>
|
|
46
|
+
/** @internal */
|
|
47
|
+
readonly first: {
|
|
48
|
+
readonly second: number
|
|
49
|
+
readonly minute: number
|
|
50
|
+
readonly hour: number
|
|
51
|
+
readonly day: number
|
|
52
|
+
readonly month: number
|
|
53
|
+
readonly weekday: number
|
|
54
|
+
}
|
|
55
|
+
/** @internal */
|
|
56
|
+
readonly next: {
|
|
57
|
+
readonly second: ReadonlyArray<number | undefined>
|
|
58
|
+
readonly minute: ReadonlyArray<number | undefined>
|
|
59
|
+
readonly hour: ReadonlyArray<number | undefined>
|
|
60
|
+
readonly day: ReadonlyArray<number | undefined>
|
|
61
|
+
readonly month: ReadonlyArray<number | undefined>
|
|
62
|
+
readonly weekday: ReadonlyArray<number | undefined>
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
const CronProto = {
|
|
67
|
+
[TypeId]: TypeId,
|
|
68
|
+
[Equal.symbol](this: Cron, that: unknown) {
|
|
69
|
+
return isCron(that) && equals(this, that)
|
|
70
|
+
},
|
|
71
|
+
[Hash.symbol](this: Cron): number {
|
|
72
|
+
return pipe(
|
|
73
|
+
Hash.hash(this.tz),
|
|
74
|
+
Hash.combine(Hash.array(Arr.fromIterable(this.seconds))),
|
|
75
|
+
Hash.combine(Hash.array(Arr.fromIterable(this.minutes))),
|
|
76
|
+
Hash.combine(Hash.array(Arr.fromIterable(this.hours))),
|
|
77
|
+
Hash.combine(Hash.array(Arr.fromIterable(this.days))),
|
|
78
|
+
Hash.combine(Hash.array(Arr.fromIterable(this.months))),
|
|
79
|
+
Hash.combine(Hash.array(Arr.fromIterable(this.weekdays))),
|
|
80
|
+
Hash.cached(this)
|
|
81
|
+
)
|
|
82
|
+
},
|
|
83
|
+
toString(this: Cron) {
|
|
84
|
+
return format(this.toJSON())
|
|
85
|
+
},
|
|
86
|
+
toJSON(this: Cron) {
|
|
87
|
+
return {
|
|
88
|
+
_id: "Cron",
|
|
89
|
+
tz: this.tz,
|
|
90
|
+
seconds: Arr.fromIterable(this.seconds),
|
|
91
|
+
minutes: Arr.fromIterable(this.minutes),
|
|
92
|
+
hours: Arr.fromIterable(this.hours),
|
|
93
|
+
days: Arr.fromIterable(this.days),
|
|
94
|
+
months: Arr.fromIterable(this.months),
|
|
95
|
+
weekdays: Arr.fromIterable(this.weekdays)
|
|
96
|
+
}
|
|
97
|
+
},
|
|
98
|
+
[NodeInspectSymbol](this: Cron) {
|
|
99
|
+
return this.toJSON()
|
|
100
|
+
},
|
|
101
|
+
pipe() {
|
|
102
|
+
return pipeArguments(this, arguments)
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
/**
|
|
107
|
+
* Checks if a given value is a `Cron` instance.
|
|
108
|
+
*
|
|
109
|
+
* @since 2.0.0
|
|
110
|
+
* @category guards
|
|
111
|
+
*/
|
|
112
|
+
export const isCron = (u: unknown): u is Cron => hasProperty(u, TypeId)
|
|
113
|
+
|
|
114
|
+
/**
|
|
115
|
+
* Creates a `Cron` instance.
|
|
116
|
+
*
|
|
117
|
+
* @since 2.0.0
|
|
118
|
+
* @category constructors
|
|
119
|
+
*/
|
|
120
|
+
export const make = (values: {
|
|
121
|
+
readonly seconds?: Iterable<number> | undefined
|
|
122
|
+
readonly minutes: Iterable<number>
|
|
123
|
+
readonly hours: Iterable<number>
|
|
124
|
+
readonly days: Iterable<number>
|
|
125
|
+
readonly months: Iterable<number>
|
|
126
|
+
readonly weekdays: Iterable<number>
|
|
127
|
+
readonly tz?: DateTime.TimeZone | undefined
|
|
128
|
+
}): Cron => {
|
|
129
|
+
const o: Mutable<Cron> = Object.create(CronProto)
|
|
130
|
+
o.seconds = new Set(Arr.sort(values.seconds ?? [0], N.Order))
|
|
131
|
+
o.minutes = new Set(Arr.sort(values.minutes, N.Order))
|
|
132
|
+
o.hours = new Set(Arr.sort(values.hours, N.Order))
|
|
133
|
+
o.days = new Set(Arr.sort(values.days, N.Order))
|
|
134
|
+
o.months = new Set(Arr.sort(values.months, N.Order))
|
|
135
|
+
o.weekdays = new Set(Arr.sort(values.weekdays, N.Order))
|
|
136
|
+
o.tz = Option.fromNullable(values.tz)
|
|
137
|
+
|
|
138
|
+
const seconds = Array.from(o.seconds)
|
|
139
|
+
const minutes = Array.from(o.minutes)
|
|
140
|
+
const hours = Array.from(o.hours)
|
|
141
|
+
const days = Array.from(o.days)
|
|
142
|
+
const months = Array.from(o.months)
|
|
143
|
+
const weekdays = Array.from(o.weekdays)
|
|
144
|
+
|
|
145
|
+
o.first = {
|
|
146
|
+
second: seconds[0] ?? 0,
|
|
147
|
+
minute: minutes[0] ?? 0,
|
|
148
|
+
hour: hours[0] ?? 0,
|
|
149
|
+
day: days[0] ?? 1,
|
|
150
|
+
month: (months[0] ?? 1) - 1,
|
|
151
|
+
weekday: weekdays[0] ?? 0
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
o.next = {
|
|
155
|
+
second: nextLookupTable(seconds, 60),
|
|
156
|
+
minute: nextLookupTable(minutes, 60),
|
|
157
|
+
hour: nextLookupTable(hours, 24),
|
|
158
|
+
day: nextLookupTable(days, 32),
|
|
159
|
+
month: nextLookupTable(months, 13),
|
|
160
|
+
weekday: nextLookupTable(weekdays, 7)
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
return o
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
const nextLookupTable = (values: ReadonlyArray<number>, size: number): Array<number | undefined> => {
|
|
167
|
+
const result = new Array(size).fill(undefined)
|
|
168
|
+
if (values.length === 0) {
|
|
169
|
+
return result
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
let current: number | undefined = undefined
|
|
173
|
+
let index = values.length - 1
|
|
174
|
+
for (let i = size - 1; i >= 0; i--) {
|
|
175
|
+
while (index >= 0 && values[index] >= i) {
|
|
176
|
+
current = values[index--]
|
|
177
|
+
}
|
|
178
|
+
result[i] = current
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
return result
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
/**
|
|
185
|
+
* @since 2.0.0
|
|
186
|
+
* @category symbol
|
|
187
|
+
*/
|
|
188
|
+
export const ParseErrorTypeId: unique symbol = Symbol.for("effect/Cron/errors/ParseError")
|
|
189
|
+
|
|
190
|
+
/**
|
|
191
|
+
* @since 2.0.0
|
|
192
|
+
* @category symbols
|
|
193
|
+
*/
|
|
194
|
+
export type ParseErrorTypeId = typeof ParseErrorTypeId
|
|
195
|
+
|
|
196
|
+
/**
|
|
197
|
+
* Represents a checked exception which occurs when decoding fails.
|
|
198
|
+
*
|
|
199
|
+
* @since 2.0.0
|
|
200
|
+
* @category models
|
|
201
|
+
*/
|
|
202
|
+
export class ParseError extends Data.TaggedError("CronParseError")<{
|
|
203
|
+
readonly message: string
|
|
204
|
+
readonly input?: string
|
|
205
|
+
}> {
|
|
206
|
+
/**
|
|
207
|
+
* @since 2.0.0
|
|
208
|
+
*/
|
|
209
|
+
readonly [ParseErrorTypeId] = ParseErrorTypeId
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
/**
|
|
213
|
+
* Returns `true` if the specified value is an `ParseError`, `false` otherwise.
|
|
214
|
+
*
|
|
215
|
+
* @since 2.0.0
|
|
216
|
+
* @category guards
|
|
217
|
+
*/
|
|
218
|
+
export const isParseError = (u: unknown): u is ParseError => hasProperty(u, ParseErrorTypeId)
|
|
219
|
+
|
|
220
|
+
/**
|
|
221
|
+
* Parses a cron expression into a `Cron` instance.
|
|
222
|
+
*
|
|
223
|
+
* @example
|
|
224
|
+
* ```ts
|
|
225
|
+
* import * as assert from "node:assert"
|
|
226
|
+
* import { Cron, Either } from "effect"
|
|
227
|
+
*
|
|
228
|
+
* // At 04:00 on every day-of-month from 8 through 14.
|
|
229
|
+
* assert.deepStrictEqual(Cron.parse("0 0 4 8-14 * *"), Either.right(Cron.make({
|
|
230
|
+
* seconds: [0],
|
|
231
|
+
* minutes: [0],
|
|
232
|
+
* hours: [4],
|
|
233
|
+
* days: [8, 9, 10, 11, 12, 13, 14],
|
|
234
|
+
* months: [],
|
|
235
|
+
* weekdays: []
|
|
236
|
+
* })))
|
|
237
|
+
* ```
|
|
238
|
+
*
|
|
239
|
+
* @since 2.0.0
|
|
240
|
+
* @category constructors
|
|
241
|
+
*/
|
|
242
|
+
export const parse = (cron: string, tz?: DateTime.TimeZone | string): Either.Either<Cron, ParseError> => {
|
|
243
|
+
const segments = cron.split(" ").filter(String.isNonEmpty)
|
|
244
|
+
if (segments.length !== 5 && segments.length !== 6) {
|
|
245
|
+
return Either.left(
|
|
246
|
+
new ParseError({
|
|
247
|
+
message: `Invalid number of segments in cron expression`,
|
|
248
|
+
input: cron
|
|
249
|
+
})
|
|
250
|
+
)
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
if (segments.length === 5) {
|
|
254
|
+
segments.unshift("0")
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
const [seconds, minutes, hours, days, months, weekdays] = segments
|
|
258
|
+
const zone = tz === undefined || dateTime.isTimeZone(tz) ?
|
|
259
|
+
Either.right(tz) :
|
|
260
|
+
Either.fromOption(dateTime.zoneFromString(tz), () =>
|
|
261
|
+
new ParseError({
|
|
262
|
+
message: `Invalid time zone in cron expression`,
|
|
263
|
+
input: tz
|
|
264
|
+
}))
|
|
265
|
+
|
|
266
|
+
return Either.all({
|
|
267
|
+
tz: zone,
|
|
268
|
+
seconds: parseSegment(seconds, secondOptions),
|
|
269
|
+
minutes: parseSegment(minutes, minuteOptions),
|
|
270
|
+
hours: parseSegment(hours, hourOptions),
|
|
271
|
+
days: parseSegment(days, dayOptions),
|
|
272
|
+
months: parseSegment(months, monthOptions),
|
|
273
|
+
weekdays: parseSegment(weekdays, weekdayOptions)
|
|
274
|
+
}).pipe(Either.map(make))
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
/**
|
|
278
|
+
* Parses a cron expression into a `Cron` instance.
|
|
279
|
+
*
|
|
280
|
+
* **Details**
|
|
281
|
+
*
|
|
282
|
+
* This function takes a cron expression as a string and attempts to parse it
|
|
283
|
+
* into a `Cron` instance. If the expression is valid, the resulting `Cron`
|
|
284
|
+
* instance will represent the schedule defined by the cron expression.
|
|
285
|
+
*
|
|
286
|
+
* If the expression is invalid, the function throws a `ParseError`.
|
|
287
|
+
*
|
|
288
|
+
* You can optionally provide a time zone (`tz`) to interpret the cron
|
|
289
|
+
* expression in a specific time zone. If no time zone is provided, the cron
|
|
290
|
+
* expression will use the default time zone.
|
|
291
|
+
*
|
|
292
|
+
* @example
|
|
293
|
+
* ```ts
|
|
294
|
+
* import { Cron } from "effect"
|
|
295
|
+
*
|
|
296
|
+
* // At 04:00 on every day-of-month from 8 through 14.
|
|
297
|
+
* console.log(Cron.unsafeParse("0 4 8-14 * *"))
|
|
298
|
+
* // Output:
|
|
299
|
+
* // {
|
|
300
|
+
* // _id: 'Cron',
|
|
301
|
+
* // tz: { _id: 'Option', _tag: 'None' },
|
|
302
|
+
* // seconds: [ 0 ],
|
|
303
|
+
* // minutes: [ 0 ],
|
|
304
|
+
* // hours: [ 4 ],
|
|
305
|
+
* // days: [
|
|
306
|
+
* // 8, 9, 10, 11,
|
|
307
|
+
* // 12, 13, 14
|
|
308
|
+
* // ],
|
|
309
|
+
* // months: [],
|
|
310
|
+
* // weekdays: []
|
|
311
|
+
* // }
|
|
312
|
+
* ```
|
|
313
|
+
*
|
|
314
|
+
* @since 2.0.0
|
|
315
|
+
* @category constructors
|
|
316
|
+
*/
|
|
317
|
+
export const unsafeParse = (cron: string, tz?: DateTime.TimeZone | string): Cron =>
|
|
318
|
+
Either.getOrThrowWith(parse(cron, tz), identity)
|
|
319
|
+
|
|
320
|
+
/**
|
|
321
|
+
* Checks if a given `Date` falls within an active `Cron` time window.
|
|
322
|
+
*
|
|
323
|
+
* @example
|
|
324
|
+
* ```ts
|
|
325
|
+
* import * as assert from "node:assert"
|
|
326
|
+
* import { Cron, Either } from "effect"
|
|
327
|
+
*
|
|
328
|
+
* const cron = Either.getOrThrow(Cron.parse("0 4 8-14 * *"))
|
|
329
|
+
* assert.deepStrictEqual(Cron.match(cron, new Date("2021-01-08 04:00:00")), true)
|
|
330
|
+
* assert.deepStrictEqual(Cron.match(cron, new Date("2021-01-08 05:00:00")), false)
|
|
331
|
+
* ```
|
|
332
|
+
*
|
|
333
|
+
* @throws `IllegalArgumentException` if the given `DateTime.Input` is invalid.
|
|
334
|
+
*
|
|
335
|
+
* @since 2.0.0
|
|
336
|
+
*/
|
|
337
|
+
export const match = (cron: Cron, date: DateTime.DateTime.Input): boolean => {
|
|
338
|
+
const parts = dateTime.unsafeMakeZoned(date, {
|
|
339
|
+
timeZone: Option.getOrUndefined(cron.tz)
|
|
340
|
+
}).pipe(dateTime.toParts)
|
|
341
|
+
|
|
342
|
+
if (cron.seconds.size !== 0 && !cron.seconds.has(parts.seconds)) {
|
|
343
|
+
return false
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
if (cron.minutes.size !== 0 && !cron.minutes.has(parts.minutes)) {
|
|
347
|
+
return false
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
if (cron.hours.size !== 0 && !cron.hours.has(parts.hours)) {
|
|
351
|
+
return false
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
if (cron.months.size !== 0 && !cron.months.has(parts.month)) {
|
|
355
|
+
return false
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
if (cron.days.size === 0 && cron.weekdays.size === 0) {
|
|
359
|
+
return true
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
if (cron.weekdays.size === 0) {
|
|
363
|
+
return cron.days.has(parts.day)
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
if (cron.days.size === 0) {
|
|
367
|
+
return cron.weekdays.has(parts.weekDay)
|
|
368
|
+
}
|
|
369
|
+
|
|
370
|
+
return cron.days.has(parts.day) || cron.weekdays.has(parts.weekDay)
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
const daysInMonth = (date: Date): number =>
|
|
374
|
+
new Date(Date.UTC(date.getUTCFullYear(), date.getUTCMonth() + 1, 0)).getUTCDate()
|
|
375
|
+
|
|
376
|
+
/**
|
|
377
|
+
* Returns the next run `Date` for the given `Cron` instance.
|
|
378
|
+
*
|
|
379
|
+
* Uses the current time as a starting point if no value is provided for `now`.
|
|
380
|
+
*
|
|
381
|
+
* @example
|
|
382
|
+
* ```ts
|
|
383
|
+
* import * as assert from "node:assert"
|
|
384
|
+
* import { Cron, Either } from "effect"
|
|
385
|
+
*
|
|
386
|
+
* const after = new Date("2021-01-01 00:00:00")
|
|
387
|
+
* const cron = Either.getOrThrow(Cron.parse("0 4 8-14 * *"))
|
|
388
|
+
* assert.deepStrictEqual(Cron.next(cron, after), new Date("2021-01-08 04:00:00"))
|
|
389
|
+
* ```
|
|
390
|
+
*
|
|
391
|
+
* @throws `IllegalArgumentException` if the given `DateTime.Input` is invalid.
|
|
392
|
+
* @throws `Error` if the next run date cannot be found within 10,000 iterations.
|
|
393
|
+
*
|
|
394
|
+
* @since 2.0.0
|
|
395
|
+
*/
|
|
396
|
+
export const next = (cron: Cron, startFrom?: DateTime.DateTime.Input): Date => {
|
|
397
|
+
const tz = Option.getOrUndefined(cron.tz)
|
|
398
|
+
const zoned = dateTime.unsafeMakeZoned(startFrom ?? new Date(), {
|
|
399
|
+
timeZone: tz
|
|
400
|
+
})
|
|
401
|
+
|
|
402
|
+
const utc = tz !== undefined && dateTime.isTimeZoneNamed(tz) && tz.id === "UTC"
|
|
403
|
+
const adjustDst = utc ? constVoid : (current: Date) => {
|
|
404
|
+
const adjusted = dateTime.unsafeMakeZoned(current, {
|
|
405
|
+
timeZone: zoned.zone,
|
|
406
|
+
adjustForTimeZone: true
|
|
407
|
+
}).pipe(dateTime.toDate)
|
|
408
|
+
|
|
409
|
+
// TODO: This implementation currently only skips forward when transitioning into daylight savings time.
|
|
410
|
+
const drift = current.getTime() - adjusted.getTime()
|
|
411
|
+
if (drift > 0) {
|
|
412
|
+
current.setTime(current.getTime() + drift)
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
|
|
416
|
+
const result = dateTime.mutate(zoned, (current) => {
|
|
417
|
+
current.setUTCSeconds(current.getUTCSeconds() + 1, 0)
|
|
418
|
+
|
|
419
|
+
for (let i = 0; i < 10_000; i++) {
|
|
420
|
+
if (cron.seconds.size !== 0) {
|
|
421
|
+
const currentSecond = current.getUTCSeconds()
|
|
422
|
+
const nextSecond = cron.next.second[currentSecond]
|
|
423
|
+
if (nextSecond === undefined) {
|
|
424
|
+
current.setUTCMinutes(current.getUTCMinutes() + 1, cron.first.second)
|
|
425
|
+
adjustDst(current)
|
|
426
|
+
continue
|
|
427
|
+
}
|
|
428
|
+
if (nextSecond > currentSecond) {
|
|
429
|
+
current.setUTCSeconds(nextSecond)
|
|
430
|
+
adjustDst(current)
|
|
431
|
+
continue
|
|
432
|
+
}
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
if (cron.minutes.size !== 0) {
|
|
436
|
+
const currentMinute = current.getUTCMinutes()
|
|
437
|
+
const nextMinute = cron.next.minute[currentMinute]
|
|
438
|
+
if (nextMinute === undefined) {
|
|
439
|
+
current.setUTCHours(current.getUTCHours() + 1, cron.first.minute, cron.first.second)
|
|
440
|
+
adjustDst(current)
|
|
441
|
+
continue
|
|
442
|
+
}
|
|
443
|
+
if (nextMinute > currentMinute) {
|
|
444
|
+
current.setUTCMinutes(nextMinute, cron.first.second)
|
|
445
|
+
adjustDst(current)
|
|
446
|
+
continue
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
|
|
450
|
+
if (cron.hours.size !== 0) {
|
|
451
|
+
const currentHour = current.getUTCHours()
|
|
452
|
+
const nextHour = cron.next.hour[currentHour]
|
|
453
|
+
if (nextHour === undefined) {
|
|
454
|
+
current.setUTCDate(current.getUTCDate() + 1)
|
|
455
|
+
current.setUTCHours(cron.first.hour, cron.first.minute, cron.first.second)
|
|
456
|
+
adjustDst(current)
|
|
457
|
+
continue
|
|
458
|
+
}
|
|
459
|
+
if (nextHour > currentHour) {
|
|
460
|
+
current.setUTCHours(nextHour, cron.first.minute, cron.first.second)
|
|
461
|
+
adjustDst(current)
|
|
462
|
+
continue
|
|
463
|
+
}
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
if (cron.weekdays.size !== 0 || cron.days.size !== 0) {
|
|
467
|
+
let a: number = Infinity
|
|
468
|
+
let b: number = Infinity
|
|
469
|
+
|
|
470
|
+
if (cron.weekdays.size !== 0) {
|
|
471
|
+
const currentWeekday = current.getUTCDay()
|
|
472
|
+
const nextWeekday = cron.next.weekday[currentWeekday]
|
|
473
|
+
a = nextWeekday === undefined ? 7 - currentWeekday + cron.first.weekday : nextWeekday - currentWeekday
|
|
474
|
+
}
|
|
475
|
+
|
|
476
|
+
if (cron.days.size !== 0 && a !== 0) {
|
|
477
|
+
const currentDay = current.getUTCDate()
|
|
478
|
+
const nextDay = cron.next.day[currentDay]
|
|
479
|
+
b = nextDay === undefined ? daysInMonth(current) - currentDay + cron.first.day : nextDay - currentDay
|
|
480
|
+
}
|
|
481
|
+
|
|
482
|
+
const addDays = Math.min(a, b)
|
|
483
|
+
if (addDays !== 0) {
|
|
484
|
+
current.setUTCDate(current.getUTCDate() + addDays)
|
|
485
|
+
current.setUTCHours(cron.first.hour, cron.first.minute, cron.first.second)
|
|
486
|
+
adjustDst(current)
|
|
487
|
+
continue
|
|
488
|
+
}
|
|
489
|
+
}
|
|
490
|
+
|
|
491
|
+
if (cron.months.size !== 0) {
|
|
492
|
+
const currentMonth = current.getUTCMonth() + 1
|
|
493
|
+
const nextMonth = cron.next.month[currentMonth]
|
|
494
|
+
if (nextMonth === undefined) {
|
|
495
|
+
current.setUTCFullYear(current.getUTCFullYear() + 1)
|
|
496
|
+
current.setUTCMonth(cron.first.month, cron.first.day)
|
|
497
|
+
current.setUTCHours(cron.first.hour, cron.first.minute, cron.first.second)
|
|
498
|
+
adjustDst(current)
|
|
499
|
+
continue
|
|
500
|
+
}
|
|
501
|
+
if (nextMonth > currentMonth) {
|
|
502
|
+
current.setUTCMonth(nextMonth - 1, cron.first.day)
|
|
503
|
+
current.setUTCHours(cron.first.hour, cron.first.minute, cron.first.second)
|
|
504
|
+
adjustDst(current)
|
|
505
|
+
continue
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
return
|
|
510
|
+
}
|
|
511
|
+
|
|
512
|
+
throw new Error("Unable to find next cron date")
|
|
513
|
+
})
|
|
514
|
+
|
|
515
|
+
return dateTime.toDateUtc(result)
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
/**
|
|
519
|
+
* Returns an `IterableIterator` which yields the sequence of `Date`s that match the `Cron` instance.
|
|
520
|
+
*
|
|
521
|
+
* @since 2.0.0
|
|
522
|
+
*/
|
|
523
|
+
export const sequence = function*(cron: Cron, startFrom?: DateTime.DateTime.Input): IterableIterator<Date> {
|
|
524
|
+
while (true) {
|
|
525
|
+
yield startFrom = next(cron, startFrom)
|
|
526
|
+
}
|
|
527
|
+
}
|
|
528
|
+
|
|
529
|
+
/**
|
|
530
|
+
* @category instances
|
|
531
|
+
* @since 2.0.0
|
|
532
|
+
*/
|
|
533
|
+
export const Equivalence: equivalence.Equivalence<Cron> = equivalence.make((self, that) =>
|
|
534
|
+
restrictionsEquals(self.seconds, that.seconds) &&
|
|
535
|
+
restrictionsEquals(self.minutes, that.minutes) &&
|
|
536
|
+
restrictionsEquals(self.hours, that.hours) &&
|
|
537
|
+
restrictionsEquals(self.days, that.days) &&
|
|
538
|
+
restrictionsEquals(self.months, that.months) &&
|
|
539
|
+
restrictionsEquals(self.weekdays, that.weekdays)
|
|
540
|
+
)
|
|
541
|
+
|
|
542
|
+
const restrictionsArrayEquals = equivalence.array(equivalence.number)
|
|
543
|
+
const restrictionsEquals = (self: ReadonlySet<number>, that: ReadonlySet<number>): boolean =>
|
|
544
|
+
restrictionsArrayEquals(Arr.fromIterable(self), Arr.fromIterable(that))
|
|
545
|
+
|
|
546
|
+
/**
|
|
547
|
+
* Checks if two `Cron`s are equal.
|
|
548
|
+
*
|
|
549
|
+
* @since 2.0.0
|
|
550
|
+
* @category predicates
|
|
551
|
+
*/
|
|
552
|
+
export const equals: {
|
|
553
|
+
(that: Cron): (self: Cron) => boolean
|
|
554
|
+
(self: Cron, that: Cron): boolean
|
|
555
|
+
} = dual(2, (self: Cron, that: Cron): boolean => Equivalence(self, that))
|
|
556
|
+
|
|
557
|
+
interface SegmentOptions {
|
|
558
|
+
min: number
|
|
559
|
+
max: number
|
|
560
|
+
aliases?: Record<string, number> | undefined
|
|
561
|
+
}
|
|
562
|
+
|
|
563
|
+
const secondOptions: SegmentOptions = {
|
|
564
|
+
min: 0,
|
|
565
|
+
max: 59
|
|
566
|
+
}
|
|
567
|
+
|
|
568
|
+
const minuteOptions: SegmentOptions = {
|
|
569
|
+
min: 0,
|
|
570
|
+
max: 59
|
|
571
|
+
}
|
|
572
|
+
|
|
573
|
+
const hourOptions: SegmentOptions = {
|
|
574
|
+
min: 0,
|
|
575
|
+
max: 23
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
const dayOptions: SegmentOptions = {
|
|
579
|
+
min: 1,
|
|
580
|
+
max: 31
|
|
581
|
+
}
|
|
582
|
+
|
|
583
|
+
const monthOptions: SegmentOptions = {
|
|
584
|
+
min: 1,
|
|
585
|
+
max: 12,
|
|
586
|
+
aliases: {
|
|
587
|
+
jan: 1,
|
|
588
|
+
feb: 2,
|
|
589
|
+
mar: 3,
|
|
590
|
+
apr: 4,
|
|
591
|
+
may: 5,
|
|
592
|
+
jun: 6,
|
|
593
|
+
jul: 7,
|
|
594
|
+
aug: 8,
|
|
595
|
+
sep: 9,
|
|
596
|
+
oct: 10,
|
|
597
|
+
nov: 11,
|
|
598
|
+
dec: 12
|
|
599
|
+
}
|
|
600
|
+
}
|
|
601
|
+
|
|
602
|
+
const weekdayOptions: SegmentOptions = {
|
|
603
|
+
min: 0,
|
|
604
|
+
max: 6,
|
|
605
|
+
aliases: {
|
|
606
|
+
sun: 0,
|
|
607
|
+
mon: 1,
|
|
608
|
+
tue: 2,
|
|
609
|
+
wed: 3,
|
|
610
|
+
thu: 4,
|
|
611
|
+
fri: 5,
|
|
612
|
+
sat: 6
|
|
613
|
+
}
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
const parseSegment = (
|
|
617
|
+
input: string,
|
|
618
|
+
options: SegmentOptions
|
|
619
|
+
): Either.Either<ReadonlySet<number>, ParseError> => {
|
|
620
|
+
const capacity = options.max - options.min + 1
|
|
621
|
+
const values = new Set<number>()
|
|
622
|
+
const fields = input.split(",")
|
|
623
|
+
|
|
624
|
+
for (const field of fields) {
|
|
625
|
+
const [raw, step] = splitStep(field)
|
|
626
|
+
if (raw === "*" && step === undefined) {
|
|
627
|
+
return Either.right(new Set())
|
|
628
|
+
}
|
|
629
|
+
|
|
630
|
+
if (step !== undefined) {
|
|
631
|
+
if (!Number.isInteger(step)) {
|
|
632
|
+
return Either.left(new ParseError({ message: `Expected step value to be a positive integer`, input }))
|
|
633
|
+
}
|
|
634
|
+
if (step < 1) {
|
|
635
|
+
return Either.left(new ParseError({ message: `Expected step value to be greater than 0`, input }))
|
|
636
|
+
}
|
|
637
|
+
if (step > options.max) {
|
|
638
|
+
return Either.left(new ParseError({ message: `Expected step value to be less than ${options.max}`, input }))
|
|
639
|
+
}
|
|
640
|
+
}
|
|
641
|
+
|
|
642
|
+
if (raw === "*") {
|
|
643
|
+
for (let i = options.min; i <= options.max; i += step ?? 1) {
|
|
644
|
+
values.add(i)
|
|
645
|
+
}
|
|
646
|
+
} else {
|
|
647
|
+
const [left, right] = splitRange(raw, options.aliases)
|
|
648
|
+
if (!Number.isInteger(left)) {
|
|
649
|
+
return Either.left(new ParseError({ message: `Expected a positive integer`, input }))
|
|
650
|
+
}
|
|
651
|
+
if (left < options.min || left > options.max) {
|
|
652
|
+
return Either.left(
|
|
653
|
+
new ParseError({ message: `Expected a value between ${options.min} and ${options.max}`, input })
|
|
654
|
+
)
|
|
655
|
+
}
|
|
656
|
+
|
|
657
|
+
if (right === undefined) {
|
|
658
|
+
values.add(left)
|
|
659
|
+
} else {
|
|
660
|
+
if (!Number.isInteger(right)) {
|
|
661
|
+
return Either.left(new ParseError({ message: `Expected a positive integer`, input }))
|
|
662
|
+
}
|
|
663
|
+
if (right < options.min || right > options.max) {
|
|
664
|
+
return Either.left(
|
|
665
|
+
new ParseError({ message: `Expected a value between ${options.min} and ${options.max}`, input })
|
|
666
|
+
)
|
|
667
|
+
}
|
|
668
|
+
if (left > right) {
|
|
669
|
+
return Either.left(new ParseError({ message: `Invalid value range`, input }))
|
|
670
|
+
}
|
|
671
|
+
|
|
672
|
+
for (let i = left; i <= right; i += step ?? 1) {
|
|
673
|
+
values.add(i)
|
|
674
|
+
}
|
|
675
|
+
}
|
|
676
|
+
}
|
|
677
|
+
|
|
678
|
+
if (values.size >= capacity) {
|
|
679
|
+
return Either.right(new Set())
|
|
680
|
+
}
|
|
681
|
+
}
|
|
682
|
+
|
|
683
|
+
return Either.right(values)
|
|
684
|
+
}
|
|
685
|
+
|
|
686
|
+
const splitStep = (input: string): [string, number | undefined] => {
|
|
687
|
+
const seperator = input.indexOf("/")
|
|
688
|
+
if (seperator !== -1) {
|
|
689
|
+
return [input.slice(0, seperator), Number(input.slice(seperator + 1))]
|
|
690
|
+
}
|
|
691
|
+
|
|
692
|
+
return [input, undefined]
|
|
693
|
+
}
|
|
694
|
+
|
|
695
|
+
const splitRange = (input: string, aliases?: Record<string, number>): [number, number | undefined] => {
|
|
696
|
+
const seperator = input.indexOf("-")
|
|
697
|
+
if (seperator !== -1) {
|
|
698
|
+
return [aliasOrValue(input.slice(0, seperator), aliases), aliasOrValue(input.slice(seperator + 1), aliases)]
|
|
699
|
+
}
|
|
700
|
+
|
|
701
|
+
return [aliasOrValue(input, aliases), undefined]
|
|
702
|
+
}
|
|
703
|
+
|
|
704
|
+
function aliasOrValue(field: string, aliases?: Record<string, number>): number {
|
|
705
|
+
return aliases?.[field.toLocaleLowerCase()] ?? Number(field)
|
|
706
|
+
}
|