flingit 0.0.63 → 0.0.65

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.
Files changed (517) hide show
  1. package/dist/cli/commands/dev.d.ts.map +1 -1
  2. package/dist/cli/commands/dev.js +9 -5
  3. package/dist/cli/commands/dev.js.map +1 -1
  4. package/dist/cli/commands/workflow.d.ts +51 -0
  5. package/dist/cli/commands/workflow.d.ts.map +1 -0
  6. package/dist/cli/commands/workflow.js +479 -0
  7. package/dist/cli/commands/workflow.js.map +1 -0
  8. package/dist/cli/deploy/bundler.d.ts.map +1 -1
  9. package/dist/cli/deploy/bundler.js +17 -1
  10. package/dist/cli/deploy/bundler.js.map +1 -1
  11. package/dist/cli/index.d.ts.map +1 -1
  12. package/dist/cli/index.js +2 -0
  13. package/dist/cli/index.js.map +1 -1
  14. package/dist/index.d.ts +4 -1
  15. package/dist/index.d.ts.map +1 -1
  16. package/dist/index.js +3 -1
  17. package/dist/index.js.map +1 -1
  18. package/dist/runtime/entry.d.ts +7 -0
  19. package/dist/runtime/entry.d.ts.map +1 -1
  20. package/dist/runtime/entry.js +71 -2
  21. package/dist/runtime/entry.js.map +1 -1
  22. package/dist/types/workflow.d.ts +82 -0
  23. package/dist/types/workflow.d.ts.map +1 -0
  24. package/dist/types/workflow.js +8 -0
  25. package/dist/types/workflow.js.map +1 -0
  26. package/dist/worker-runtime/d1-event-store.d.ts +22 -0
  27. package/dist/worker-runtime/d1-event-store.d.ts.map +1 -0
  28. package/dist/worker-runtime/d1-event-store.js +227 -0
  29. package/dist/worker-runtime/d1-event-store.js.map +1 -0
  30. package/dist/worker-runtime/entry-extract.d.ts.map +1 -1
  31. package/dist/worker-runtime/entry-extract.js +6 -3
  32. package/dist/worker-runtime/entry-extract.js.map +1 -1
  33. package/dist/worker-runtime/entry.d.ts.map +1 -1
  34. package/dist/worker-runtime/entry.js +105 -3
  35. package/dist/worker-runtime/entry.js.map +1 -1
  36. package/dist/worker-runtime/index.d.ts +2 -0
  37. package/dist/worker-runtime/index.d.ts.map +1 -1
  38. package/dist/worker-runtime/index.js +5 -0
  39. package/dist/worker-runtime/index.js.map +1 -1
  40. package/dist/workflow/runtime.d.ts +56 -0
  41. package/dist/workflow/runtime.d.ts.map +1 -0
  42. package/dist/workflow/runtime.js +450 -0
  43. package/dist/workflow/runtime.js.map +1 -0
  44. package/node_modules/@glideapps/ts-necessities/LICENSE +21 -0
  45. package/node_modules/@glideapps/ts-necessities/README.md +16 -0
  46. package/node_modules/@glideapps/ts-necessities/dist/branded-strings.d.ts +39 -0
  47. package/node_modules/@glideapps/ts-necessities/dist/default-map.d.ts +53 -0
  48. package/node_modules/@glideapps/ts-necessities/dist/index.d.mts +113 -0
  49. package/node_modules/@glideapps/ts-necessities/dist/index.d.ts +113 -0
  50. package/node_modules/@glideapps/ts-necessities/dist/index.js +237 -0
  51. package/node_modules/@glideapps/ts-necessities/dist/index.mjs +189 -0
  52. package/node_modules/@glideapps/ts-necessities/package.json +52 -0
  53. package/node_modules/@rollup/rollup-linux-x64-gnu/README.md +3 -0
  54. package/node_modules/@rollup/rollup-linux-x64-gnu/package.json +25 -0
  55. package/node_modules/@rollup/rollup-linux-x64-gnu/rollup.linux-x64-gnu.node +0 -0
  56. package/node_modules/base64-js/LICENSE +21 -0
  57. package/node_modules/base64-js/README.md +34 -0
  58. package/node_modules/base64-js/base64js.min.js +1 -0
  59. package/node_modules/base64-js/index.d.ts +3 -0
  60. package/node_modules/base64-js/index.js +150 -0
  61. package/node_modules/base64-js/package.json +47 -0
  62. package/node_modules/better-sqlite3/LICENSE +21 -0
  63. package/node_modules/better-sqlite3/README.md +99 -0
  64. package/node_modules/better-sqlite3/binding.gyp +38 -0
  65. package/node_modules/better-sqlite3/deps/common.gypi +68 -0
  66. package/node_modules/better-sqlite3/deps/copy.js +31 -0
  67. package/node_modules/better-sqlite3/deps/defines.gypi +41 -0
  68. package/node_modules/better-sqlite3/deps/download.sh +122 -0
  69. package/node_modules/better-sqlite3/deps/patches/1208.patch +15 -0
  70. package/node_modules/better-sqlite3/deps/sqlite3/sqlite3.c +265994 -0
  71. package/node_modules/better-sqlite3/deps/sqlite3/sqlite3.h +13968 -0
  72. package/node_modules/better-sqlite3/deps/sqlite3/sqlite3ext.h +730 -0
  73. package/node_modules/better-sqlite3/deps/sqlite3.gyp +80 -0
  74. package/node_modules/better-sqlite3/deps/test_extension.c +21 -0
  75. package/node_modules/better-sqlite3/lib/database.js +90 -0
  76. package/node_modules/better-sqlite3/lib/index.js +3 -0
  77. package/node_modules/better-sqlite3/lib/methods/aggregate.js +43 -0
  78. package/node_modules/better-sqlite3/lib/methods/backup.js +67 -0
  79. package/node_modules/better-sqlite3/lib/methods/function.js +31 -0
  80. package/node_modules/better-sqlite3/lib/methods/inspect.js +7 -0
  81. package/node_modules/better-sqlite3/lib/methods/pragma.js +12 -0
  82. package/node_modules/better-sqlite3/lib/methods/serialize.js +16 -0
  83. package/node_modules/better-sqlite3/lib/methods/table.js +189 -0
  84. package/node_modules/better-sqlite3/lib/methods/transaction.js +78 -0
  85. package/node_modules/better-sqlite3/lib/methods/wrappers.js +54 -0
  86. package/node_modules/better-sqlite3/lib/sqlite-error.js +20 -0
  87. package/node_modules/better-sqlite3/lib/util.js +12 -0
  88. package/node_modules/better-sqlite3/package.json +59 -0
  89. package/node_modules/better-sqlite3/src/addon.cpp +47 -0
  90. package/node_modules/better-sqlite3/src/better_sqlite3.cpp +74 -0
  91. package/node_modules/better-sqlite3/src/objects/backup.cpp +120 -0
  92. package/node_modules/better-sqlite3/src/objects/backup.hpp +36 -0
  93. package/node_modules/better-sqlite3/src/objects/database.cpp +417 -0
  94. package/node_modules/better-sqlite3/src/objects/database.hpp +103 -0
  95. package/node_modules/better-sqlite3/src/objects/statement-iterator.cpp +113 -0
  96. package/node_modules/better-sqlite3/src/objects/statement-iterator.hpp +50 -0
  97. package/node_modules/better-sqlite3/src/objects/statement.cpp +383 -0
  98. package/node_modules/better-sqlite3/src/objects/statement.hpp +58 -0
  99. package/node_modules/better-sqlite3/src/util/bind-map.cpp +73 -0
  100. package/node_modules/better-sqlite3/src/util/binder.cpp +193 -0
  101. package/node_modules/better-sqlite3/src/util/constants.cpp +172 -0
  102. package/node_modules/better-sqlite3/src/util/custom-aggregate.cpp +121 -0
  103. package/node_modules/better-sqlite3/src/util/custom-function.cpp +59 -0
  104. package/node_modules/better-sqlite3/src/util/custom-table.cpp +409 -0
  105. package/node_modules/better-sqlite3/src/util/data-converter.cpp +17 -0
  106. package/node_modules/better-sqlite3/src/util/data.cpp +194 -0
  107. package/node_modules/better-sqlite3/src/util/helpers.cpp +109 -0
  108. package/node_modules/better-sqlite3/src/util/macros.cpp +83 -0
  109. package/node_modules/better-sqlite3/src/util/query-macros.cpp +71 -0
  110. package/node_modules/better-sqlite3/src/util/row-builder.cpp +49 -0
  111. package/node_modules/bindings/LICENSE.md +22 -0
  112. package/node_modules/bindings/README.md +98 -0
  113. package/node_modules/bindings/bindings.js +221 -0
  114. package/node_modules/bindings/package.json +28 -0
  115. package/node_modules/bl/.travis.yml +17 -0
  116. package/node_modules/bl/BufferList.js +396 -0
  117. package/node_modules/bl/LICENSE.md +13 -0
  118. package/node_modules/bl/README.md +247 -0
  119. package/node_modules/bl/bl.js +84 -0
  120. package/node_modules/bl/package.json +37 -0
  121. package/node_modules/bl/test/convert.js +21 -0
  122. package/node_modules/bl/test/indexOf.js +492 -0
  123. package/node_modules/bl/test/isBufferList.js +32 -0
  124. package/node_modules/bl/test/test.js +869 -0
  125. package/node_modules/buffer/AUTHORS.md +70 -0
  126. package/node_modules/buffer/LICENSE +21 -0
  127. package/node_modules/buffer/README.md +410 -0
  128. package/node_modules/buffer/index.d.ts +186 -0
  129. package/node_modules/buffer/index.js +1817 -0
  130. package/node_modules/buffer/package.json +96 -0
  131. package/node_modules/chownr/LICENSE +15 -0
  132. package/node_modules/chownr/README.md +3 -0
  133. package/node_modules/chownr/chownr.js +167 -0
  134. package/node_modules/chownr/package.json +29 -0
  135. package/node_modules/decompress-response/index.d.ts +22 -0
  136. package/node_modules/decompress-response/index.js +58 -0
  137. package/node_modules/decompress-response/license +9 -0
  138. package/node_modules/decompress-response/package.json +56 -0
  139. package/node_modules/decompress-response/readme.md +48 -0
  140. package/node_modules/deep-extend/LICENSE +20 -0
  141. package/node_modules/deep-extend/README.md +91 -0
  142. package/node_modules/deep-extend/index.js +1 -0
  143. package/node_modules/deep-extend/lib/deep-extend.js +150 -0
  144. package/node_modules/deep-extend/package.json +62 -0
  145. package/node_modules/detect-libc/LICENSE +201 -0
  146. package/node_modules/detect-libc/README.md +163 -0
  147. package/node_modules/detect-libc/index.d.ts +14 -0
  148. package/node_modules/detect-libc/lib/detect-libc.js +313 -0
  149. package/node_modules/detect-libc/lib/elf.js +39 -0
  150. package/node_modules/detect-libc/lib/filesystem.js +51 -0
  151. package/node_modules/detect-libc/lib/process.js +24 -0
  152. package/node_modules/detect-libc/package.json +44 -0
  153. package/node_modules/determined/LICENSE +18 -0
  154. package/node_modules/determined/README.md +317 -0
  155. package/node_modules/determined/dist/index.cjs +370 -0
  156. package/node_modules/determined/dist/index.d.cts +110 -0
  157. package/node_modules/determined/dist/index.d.ts +110 -0
  158. package/node_modules/determined/dist/index.js +332 -0
  159. package/node_modules/determined/package.json +45 -0
  160. package/node_modules/end-of-stream/LICENSE +21 -0
  161. package/node_modules/end-of-stream/README.md +54 -0
  162. package/node_modules/end-of-stream/index.js +96 -0
  163. package/node_modules/end-of-stream/package.json +37 -0
  164. package/node_modules/expand-template/.travis.yml +6 -0
  165. package/node_modules/expand-template/LICENSE +21 -0
  166. package/node_modules/expand-template/README.md +43 -0
  167. package/node_modules/expand-template/index.js +26 -0
  168. package/node_modules/expand-template/package.json +29 -0
  169. package/node_modules/expand-template/test.js +67 -0
  170. package/node_modules/file-uri-to-path/.npmignore +1 -0
  171. package/node_modules/file-uri-to-path/.travis.yml +30 -0
  172. package/node_modules/file-uri-to-path/History.md +21 -0
  173. package/node_modules/file-uri-to-path/LICENSE +20 -0
  174. package/node_modules/file-uri-to-path/README.md +74 -0
  175. package/node_modules/file-uri-to-path/index.d.ts +2 -0
  176. package/node_modules/file-uri-to-path/index.js +66 -0
  177. package/node_modules/file-uri-to-path/package.json +32 -0
  178. package/node_modules/file-uri-to-path/test/test.js +24 -0
  179. package/node_modules/file-uri-to-path/test/tests.json +13 -0
  180. package/node_modules/flingflow/.nvmrc +1 -0
  181. package/node_modules/flingflow/AGENTS.md +5 -0
  182. package/node_modules/flingflow/README.md +679 -0
  183. package/node_modules/flingflow/SPEC.md +554 -0
  184. package/node_modules/flingflow/TESTING.md +506 -0
  185. package/node_modules/flingflow/dist/backoff.d.ts +9 -0
  186. package/node_modules/flingflow/dist/backoff.js +14 -0
  187. package/node_modules/flingflow/dist/backoff.js.map +1 -0
  188. package/node_modules/flingflow/dist/clock.d.ts +13 -0
  189. package/node_modules/flingflow/dist/clock.js +21 -0
  190. package/node_modules/flingflow/dist/clock.js.map +1 -0
  191. package/node_modules/flingflow/dist/context.d.ts +8 -0
  192. package/node_modules/flingflow/dist/context.js +45 -0
  193. package/node_modules/flingflow/dist/context.js.map +1 -0
  194. package/node_modules/flingflow/dist/engine.d.ts +29 -0
  195. package/node_modules/flingflow/dist/engine.js +306 -0
  196. package/node_modules/flingflow/dist/engine.js.map +1 -0
  197. package/node_modules/flingflow/dist/index.d.ts +18 -0
  198. package/node_modules/flingflow/dist/index.js +14 -0
  199. package/node_modules/flingflow/dist/index.js.map +1 -0
  200. package/node_modules/flingflow/dist/recovery.d.ts +16 -0
  201. package/node_modules/flingflow/dist/recovery.js +118 -0
  202. package/node_modules/flingflow/dist/recovery.js.map +1 -0
  203. package/node_modules/flingflow/dist/registry.d.ts +24 -0
  204. package/node_modules/flingflow/dist/registry.js +29 -0
  205. package/node_modules/flingflow/dist/registry.js.map +1 -0
  206. package/node_modules/flingflow/dist/store-memory.d.ts +29 -0
  207. package/node_modules/flingflow/dist/store-memory.js +349 -0
  208. package/node_modules/flingflow/dist/store-memory.js.map +1 -0
  209. package/node_modules/flingflow/dist/store-sqlite.d.ts +30 -0
  210. package/node_modules/flingflow/dist/store-sqlite.js +400 -0
  211. package/node_modules/flingflow/dist/store-sqlite.js.map +1 -0
  212. package/node_modules/flingflow/dist/store.d.ts +39 -0
  213. package/node_modules/flingflow/dist/store.js +21 -0
  214. package/node_modules/flingflow/dist/store.js.map +1 -0
  215. package/node_modules/flingflow/dist/stress.d.ts +1 -0
  216. package/node_modules/flingflow/dist/stress.js +377 -0
  217. package/node_modules/flingflow/dist/stress.js.map +1 -0
  218. package/node_modules/flingflow/dist/transitions.d.ts +14 -0
  219. package/node_modules/flingflow/dist/transitions.js +28 -0
  220. package/node_modules/flingflow/dist/transitions.js.map +1 -0
  221. package/node_modules/flingflow/dist/types.d.ts +114 -0
  222. package/node_modules/flingflow/dist/types.js +10 -0
  223. package/node_modules/flingflow/dist/types.js.map +1 -0
  224. package/node_modules/flingflow/eslint.config.js +94 -0
  225. package/node_modules/flingflow/package.json +66 -0
  226. package/node_modules/flingflow/src/backoff.ts +14 -0
  227. package/node_modules/flingflow/src/clock.ts +29 -0
  228. package/node_modules/flingflow/src/context.ts +60 -0
  229. package/node_modules/flingflow/src/engine.ts +367 -0
  230. package/node_modules/flingflow/src/index.ts +52 -0
  231. package/node_modules/flingflow/src/recovery.ts +144 -0
  232. package/node_modules/flingflow/src/registry.ts +52 -0
  233. package/node_modules/flingflow/src/store-memory.ts +378 -0
  234. package/node_modules/flingflow/src/store-sqlite.ts +451 -0
  235. package/node_modules/flingflow/src/store.ts +55 -0
  236. package/node_modules/flingflow/src/stress.ts +423 -0
  237. package/node_modules/flingflow/src/transitions.ts +38 -0
  238. package/node_modules/flingflow/src/types.ts +84 -0
  239. package/node_modules/flingflow/test/backoff.test.ts +54 -0
  240. package/node_modules/flingflow/test/context.test.ts +94 -0
  241. package/node_modules/flingflow/test/engine.test.ts +362 -0
  242. package/node_modules/flingflow/test/fixtures.ts +58 -0
  243. package/node_modules/flingflow/test/recovery.test.ts +176 -0
  244. package/node_modules/flingflow/test/simulation.test.ts +395 -0
  245. package/node_modules/flingflow/test/store-conformance.ts +344 -0
  246. package/node_modules/flingflow/test/store-memory.test.ts +8 -0
  247. package/node_modules/flingflow/test/store-sqlite.test.ts +8 -0
  248. package/node_modules/flingflow/tsconfig.json +18 -0
  249. package/node_modules/flingflow/tsconfig.lint.json +5 -0
  250. package/node_modules/flingflow/tsconfig.typecheck.json +20 -0
  251. package/node_modules/flingflow/vitest.config.ts +10 -0
  252. package/node_modules/fs-constants/LICENSE +21 -0
  253. package/node_modules/fs-constants/README.md +26 -0
  254. package/node_modules/fs-constants/browser.js +1 -0
  255. package/node_modules/fs-constants/index.js +1 -0
  256. package/node_modules/fs-constants/package.json +19 -0
  257. package/node_modules/github-from-package/.travis.yml +4 -0
  258. package/node_modules/github-from-package/LICENSE +18 -0
  259. package/node_modules/github-from-package/example/package.json +8 -0
  260. package/node_modules/github-from-package/example/url.js +3 -0
  261. package/node_modules/github-from-package/index.js +17 -0
  262. package/node_modules/github-from-package/package.json +30 -0
  263. package/node_modules/github-from-package/readme.markdown +53 -0
  264. package/node_modules/github-from-package/test/a.json +8 -0
  265. package/node_modules/github-from-package/test/b.json +5 -0
  266. package/node_modules/github-from-package/test/c.json +5 -0
  267. package/node_modules/github-from-package/test/d.json +7 -0
  268. package/node_modules/github-from-package/test/e.json +5 -0
  269. package/node_modules/github-from-package/test/url.js +19 -0
  270. package/node_modules/ieee754/LICENSE +11 -0
  271. package/node_modules/ieee754/README.md +51 -0
  272. package/node_modules/ieee754/index.d.ts +10 -0
  273. package/node_modules/ieee754/index.js +85 -0
  274. package/node_modules/ieee754/package.json +52 -0
  275. package/node_modules/inherits/LICENSE +16 -0
  276. package/node_modules/inherits/README.md +42 -0
  277. package/node_modules/inherits/inherits.js +9 -0
  278. package/node_modules/inherits/inherits_browser.js +27 -0
  279. package/node_modules/inherits/package.json +29 -0
  280. package/node_modules/ini/LICENSE +15 -0
  281. package/node_modules/ini/README.md +102 -0
  282. package/node_modules/ini/ini.js +206 -0
  283. package/node_modules/ini/package.json +33 -0
  284. package/node_modules/mimic-response/index.d.ts +17 -0
  285. package/node_modules/mimic-response/index.js +77 -0
  286. package/node_modules/mimic-response/license +9 -0
  287. package/node_modules/mimic-response/package.json +42 -0
  288. package/node_modules/mimic-response/readme.md +78 -0
  289. package/node_modules/minimist/.eslintrc +29 -0
  290. package/node_modules/minimist/.github/FUNDING.yml +12 -0
  291. package/node_modules/minimist/.nycrc +14 -0
  292. package/node_modules/minimist/CHANGELOG.md +298 -0
  293. package/node_modules/minimist/LICENSE +18 -0
  294. package/node_modules/minimist/README.md +121 -0
  295. package/node_modules/minimist/example/parse.js +4 -0
  296. package/node_modules/minimist/index.js +263 -0
  297. package/node_modules/minimist/package.json +75 -0
  298. package/node_modules/minimist/test/all_bool.js +34 -0
  299. package/node_modules/minimist/test/bool.js +177 -0
  300. package/node_modules/minimist/test/dash.js +43 -0
  301. package/node_modules/minimist/test/default_bool.js +37 -0
  302. package/node_modules/minimist/test/dotted.js +24 -0
  303. package/node_modules/minimist/test/kv_short.js +32 -0
  304. package/node_modules/minimist/test/long.js +33 -0
  305. package/node_modules/minimist/test/num.js +38 -0
  306. package/node_modules/minimist/test/parse.js +209 -0
  307. package/node_modules/minimist/test/parse_modified.js +11 -0
  308. package/node_modules/minimist/test/proto.js +64 -0
  309. package/node_modules/minimist/test/short.js +69 -0
  310. package/node_modules/minimist/test/stop_early.js +17 -0
  311. package/node_modules/minimist/test/unknown.js +104 -0
  312. package/node_modules/minimist/test/whitespace.js +10 -0
  313. package/node_modules/mkdirp-classic/LICENSE +21 -0
  314. package/node_modules/mkdirp-classic/README.md +18 -0
  315. package/node_modules/mkdirp-classic/index.js +98 -0
  316. package/node_modules/mkdirp-classic/package.json +18 -0
  317. package/node_modules/napi-build-utils/.github/workflows/run-npm-tests.yml +31 -0
  318. package/node_modules/napi-build-utils/LICENSE +21 -0
  319. package/node_modules/napi-build-utils/README.md +52 -0
  320. package/node_modules/napi-build-utils/index.js +214 -0
  321. package/node_modules/napi-build-utils/index.md +0 -0
  322. package/node_modules/napi-build-utils/package.json +42 -0
  323. package/node_modules/neverthrow/LICENSE +22 -0
  324. package/node_modules/neverthrow/README.md +1683 -0
  325. package/node_modules/neverthrow/dist/index.cjs.js +510 -0
  326. package/node_modules/neverthrow/dist/index.d.ts +408 -0
  327. package/node_modules/neverthrow/dist/index.es.js +497 -0
  328. package/node_modules/neverthrow/package.json +64 -0
  329. package/node_modules/node-abi/LICENSE +21 -0
  330. package/node_modules/node-abi/README.md +54 -0
  331. package/node_modules/node-abi/abi_registry.json +425 -0
  332. package/node_modules/node-abi/index.js +179 -0
  333. package/node_modules/node-abi/package.json +45 -0
  334. package/node_modules/once/LICENSE +15 -0
  335. package/node_modules/once/README.md +79 -0
  336. package/node_modules/once/once.js +42 -0
  337. package/node_modules/once/package.json +33 -0
  338. package/node_modules/prebuild-install/CHANGELOG.md +131 -0
  339. package/node_modules/prebuild-install/CONTRIBUTING.md +6 -0
  340. package/node_modules/prebuild-install/LICENSE +21 -0
  341. package/node_modules/prebuild-install/README.md +163 -0
  342. package/node_modules/prebuild-install/asset.js +44 -0
  343. package/node_modules/prebuild-install/bin.js +78 -0
  344. package/node_modules/prebuild-install/download.js +142 -0
  345. package/node_modules/prebuild-install/error.js +14 -0
  346. package/node_modules/prebuild-install/help.txt +16 -0
  347. package/node_modules/prebuild-install/index.js +1 -0
  348. package/node_modules/prebuild-install/log.js +33 -0
  349. package/node_modules/prebuild-install/package.json +67 -0
  350. package/node_modules/prebuild-install/proxy.js +35 -0
  351. package/node_modules/prebuild-install/rc.js +64 -0
  352. package/node_modules/prebuild-install/util.js +143 -0
  353. package/node_modules/pump/.github/FUNDING.yml +2 -0
  354. package/node_modules/pump/.travis.yml +5 -0
  355. package/node_modules/pump/LICENSE +21 -0
  356. package/node_modules/pump/README.md +74 -0
  357. package/node_modules/pump/SECURITY.md +5 -0
  358. package/node_modules/pump/index.js +86 -0
  359. package/node_modules/pump/package.json +24 -0
  360. package/node_modules/pump/test-browser.js +66 -0
  361. package/node_modules/pump/test-node.js +53 -0
  362. package/node_modules/rc/LICENSE.APACHE2 +15 -0
  363. package/node_modules/rc/LICENSE.BSD +26 -0
  364. package/node_modules/rc/LICENSE.MIT +24 -0
  365. package/node_modules/rc/README.md +227 -0
  366. package/node_modules/rc/browser.js +7 -0
  367. package/node_modules/rc/cli.js +4 -0
  368. package/node_modules/rc/index.js +53 -0
  369. package/node_modules/rc/lib/utils.js +104 -0
  370. package/node_modules/rc/package.json +29 -0
  371. package/node_modules/rc/test/ini.js +16 -0
  372. package/node_modules/rc/test/nested-env-vars.js +50 -0
  373. package/node_modules/rc/test/test.js +59 -0
  374. package/node_modules/readable-stream/CONTRIBUTING.md +38 -0
  375. package/node_modules/readable-stream/GOVERNANCE.md +136 -0
  376. package/node_modules/readable-stream/LICENSE +47 -0
  377. package/node_modules/readable-stream/README.md +106 -0
  378. package/node_modules/readable-stream/errors-browser.js +127 -0
  379. package/node_modules/readable-stream/errors.js +116 -0
  380. package/node_modules/readable-stream/experimentalWarning.js +17 -0
  381. package/node_modules/readable-stream/lib/_stream_duplex.js +126 -0
  382. package/node_modules/readable-stream/lib/_stream_passthrough.js +37 -0
  383. package/node_modules/readable-stream/lib/_stream_readable.js +1027 -0
  384. package/node_modules/readable-stream/lib/_stream_transform.js +190 -0
  385. package/node_modules/readable-stream/lib/_stream_writable.js +641 -0
  386. package/node_modules/readable-stream/lib/internal/streams/async_iterator.js +180 -0
  387. package/node_modules/readable-stream/lib/internal/streams/buffer_list.js +183 -0
  388. package/node_modules/readable-stream/lib/internal/streams/destroy.js +96 -0
  389. package/node_modules/readable-stream/lib/internal/streams/end-of-stream.js +86 -0
  390. package/node_modules/readable-stream/lib/internal/streams/from-browser.js +3 -0
  391. package/node_modules/readable-stream/lib/internal/streams/from.js +52 -0
  392. package/node_modules/readable-stream/lib/internal/streams/pipeline.js +86 -0
  393. package/node_modules/readable-stream/lib/internal/streams/state.js +22 -0
  394. package/node_modules/readable-stream/lib/internal/streams/stream-browser.js +1 -0
  395. package/node_modules/readable-stream/lib/internal/streams/stream.js +1 -0
  396. package/node_modules/readable-stream/package.json +68 -0
  397. package/node_modules/readable-stream/readable-browser.js +9 -0
  398. package/node_modules/readable-stream/readable.js +16 -0
  399. package/node_modules/safe-buffer/LICENSE +21 -0
  400. package/node_modules/safe-buffer/README.md +584 -0
  401. package/node_modules/safe-buffer/index.d.ts +187 -0
  402. package/node_modules/safe-buffer/index.js +65 -0
  403. package/node_modules/safe-buffer/package.json +51 -0
  404. package/node_modules/semver/LICENSE +15 -0
  405. package/node_modules/semver/README.md +664 -0
  406. package/node_modules/semver/bin/semver.js +191 -0
  407. package/node_modules/semver/classes/comparator.js +143 -0
  408. package/node_modules/semver/classes/index.js +7 -0
  409. package/node_modules/semver/classes/range.js +557 -0
  410. package/node_modules/semver/classes/semver.js +333 -0
  411. package/node_modules/semver/functions/clean.js +8 -0
  412. package/node_modules/semver/functions/cmp.js +54 -0
  413. package/node_modules/semver/functions/coerce.js +62 -0
  414. package/node_modules/semver/functions/compare-build.js +9 -0
  415. package/node_modules/semver/functions/compare-loose.js +5 -0
  416. package/node_modules/semver/functions/compare.js +7 -0
  417. package/node_modules/semver/functions/diff.js +60 -0
  418. package/node_modules/semver/functions/eq.js +5 -0
  419. package/node_modules/semver/functions/gt.js +5 -0
  420. package/node_modules/semver/functions/gte.js +5 -0
  421. package/node_modules/semver/functions/inc.js +21 -0
  422. package/node_modules/semver/functions/lt.js +5 -0
  423. package/node_modules/semver/functions/lte.js +5 -0
  424. package/node_modules/semver/functions/major.js +5 -0
  425. package/node_modules/semver/functions/minor.js +5 -0
  426. package/node_modules/semver/functions/neq.js +5 -0
  427. package/node_modules/semver/functions/parse.js +18 -0
  428. package/node_modules/semver/functions/patch.js +5 -0
  429. package/node_modules/semver/functions/prerelease.js +8 -0
  430. package/node_modules/semver/functions/rcompare.js +5 -0
  431. package/node_modules/semver/functions/rsort.js +5 -0
  432. package/node_modules/semver/functions/satisfies.js +12 -0
  433. package/node_modules/semver/functions/sort.js +5 -0
  434. package/node_modules/semver/functions/valid.js +8 -0
  435. package/node_modules/semver/index.js +91 -0
  436. package/node_modules/semver/internal/constants.js +37 -0
  437. package/node_modules/semver/internal/debug.js +11 -0
  438. package/node_modules/semver/internal/identifiers.js +29 -0
  439. package/node_modules/semver/internal/lrucache.js +42 -0
  440. package/node_modules/semver/internal/parse-options.js +17 -0
  441. package/node_modules/semver/internal/re.js +223 -0
  442. package/node_modules/semver/package.json +78 -0
  443. package/node_modules/semver/preload.js +4 -0
  444. package/node_modules/semver/range.bnf +16 -0
  445. package/node_modules/semver/ranges/gtr.js +6 -0
  446. package/node_modules/semver/ranges/intersects.js +9 -0
  447. package/node_modules/semver/ranges/ltr.js +6 -0
  448. package/node_modules/semver/ranges/max-satisfying.js +27 -0
  449. package/node_modules/semver/ranges/min-satisfying.js +26 -0
  450. package/node_modules/semver/ranges/min-version.js +63 -0
  451. package/node_modules/semver/ranges/outside.js +82 -0
  452. package/node_modules/semver/ranges/simplify.js +49 -0
  453. package/node_modules/semver/ranges/subset.js +249 -0
  454. package/node_modules/semver/ranges/to-comparators.js +10 -0
  455. package/node_modules/semver/ranges/valid.js +13 -0
  456. package/node_modules/simple-concat/.travis.yml +3 -0
  457. package/node_modules/simple-concat/LICENSE +20 -0
  458. package/node_modules/simple-concat/README.md +44 -0
  459. package/node_modules/simple-concat/index.js +15 -0
  460. package/node_modules/simple-concat/package.json +47 -0
  461. package/node_modules/simple-concat/test/basic.js +41 -0
  462. package/node_modules/simple-get/.github/dependabot.yml +15 -0
  463. package/node_modules/simple-get/.github/workflows/ci.yml +23 -0
  464. package/node_modules/simple-get/LICENSE +20 -0
  465. package/node_modules/simple-get/README.md +333 -0
  466. package/node_modules/simple-get/index.js +108 -0
  467. package/node_modules/simple-get/package.json +67 -0
  468. package/node_modules/string_decoder/LICENSE +48 -0
  469. package/node_modules/string_decoder/README.md +47 -0
  470. package/node_modules/string_decoder/lib/string_decoder.js +296 -0
  471. package/node_modules/string_decoder/package.json +34 -0
  472. package/node_modules/strip-json-comments/index.js +70 -0
  473. package/node_modules/strip-json-comments/license +21 -0
  474. package/node_modules/strip-json-comments/package.json +42 -0
  475. package/node_modules/strip-json-comments/readme.md +64 -0
  476. package/node_modules/tar-fs/.travis.yml +6 -0
  477. package/node_modules/tar-fs/LICENSE +21 -0
  478. package/node_modules/tar-fs/README.md +165 -0
  479. package/node_modules/tar-fs/index.js +363 -0
  480. package/node_modules/tar-fs/package.json +41 -0
  481. package/node_modules/tar-fs/test/fixtures/a/hello.txt +1 -0
  482. package/node_modules/tar-fs/test/fixtures/b/a/test.txt +1 -0
  483. package/node_modules/tar-fs/test/fixtures/d/file1 +0 -0
  484. package/node_modules/tar-fs/test/fixtures/d/file2 +0 -0
  485. package/node_modules/tar-fs/test/fixtures/d/sub-dir/file5 +0 -0
  486. package/node_modules/tar-fs/test/fixtures/d/sub-files/file3 +0 -0
  487. package/node_modules/tar-fs/test/fixtures/d/sub-files/file4 +0 -0
  488. package/node_modules/tar-fs/test/fixtures/e/directory/.ignore +0 -0
  489. package/node_modules/tar-fs/test/fixtures/e/file +0 -0
  490. package/node_modules/tar-fs/test/fixtures/invalid.tar +0 -0
  491. package/node_modules/tar-fs/test/index.js +346 -0
  492. package/node_modules/tar-stream/LICENSE +21 -0
  493. package/node_modules/tar-stream/README.md +168 -0
  494. package/node_modules/tar-stream/extract.js +257 -0
  495. package/node_modules/tar-stream/headers.js +295 -0
  496. package/node_modules/tar-stream/index.js +2 -0
  497. package/node_modules/tar-stream/pack.js +255 -0
  498. package/node_modules/tar-stream/package.json +58 -0
  499. package/node_modules/tar-stream/sandbox.js +11 -0
  500. package/node_modules/tunnel-agent/LICENSE +55 -0
  501. package/node_modules/tunnel-agent/README.md +4 -0
  502. package/node_modules/tunnel-agent/index.js +244 -0
  503. package/node_modules/tunnel-agent/package.json +22 -0
  504. package/node_modules/util-deprecate/History.md +16 -0
  505. package/node_modules/util-deprecate/LICENSE +24 -0
  506. package/node_modules/util-deprecate/README.md +53 -0
  507. package/node_modules/util-deprecate/browser.js +67 -0
  508. package/node_modules/util-deprecate/node.js +6 -0
  509. package/node_modules/util-deprecate/package.json +27 -0
  510. package/node_modules/wrappy/LICENSE +15 -0
  511. package/node_modules/wrappy/README.md +36 -0
  512. package/node_modules/wrappy/package.json +29 -0
  513. package/node_modules/wrappy/wrappy.js +33 -0
  514. package/package.json +12 -2
  515. package/templates/default/dot-claude/skills/fling/.hash +1 -1
  516. package/templates/default/dot-claude/skills/fling/SKILL.md +56 -29
  517. package/templates/default/dot-claude/skills/fling/references/WORKFLOWS.md +368 -0
@@ -0,0 +1,176 @@
1
+ import { describe, it, expect, beforeEach } from "vitest";
2
+ import { runRecovery } from "../src/recovery.js";
3
+ import { Engine } from "../src/engine.js";
4
+ import { MemoryEventStore } from "../src/store-memory.js";
5
+ import { WorkflowRegistry } from "../src/registry.js";
6
+ import { ManualClock } from "../src/clock.js";
7
+ import type { EventStore } from "../src/store.js";
8
+ import type { WorkflowConfig } from "../src/types.js";
9
+ import { noSim } from "./fixtures.js";
10
+
11
+ const TEST_WF_CONFIG: WorkflowConfig = { maxAttempts: 3, initialBackoffMS: 0, backoffMultiplier: 2, maxBackoffMS: 60_000, stepTimeoutMS: 5000 };
12
+
13
+ describe("recovery", () => {
14
+ let store: EventStore;
15
+ let registry: WorkflowRegistry;
16
+ let clock: ManualClock;
17
+ let idCounter: number;
18
+
19
+ beforeEach(async () => {
20
+ store = new MemoryEventStore();
21
+ (await store.initialize())._unsafeUnwrap();
22
+ registry = new WorkflowRegistry();
23
+ clock = new ManualClock(1000);
24
+ idCounter = 0;
25
+
26
+ registry.register({
27
+ name: "test-wf",
28
+ fn: async (_ctx) => ({ done: true, result: "ok" }),
29
+ maxAttempts: 3,
30
+ stepTimeoutMS: 5000,
31
+ initialBackoffMS: 0,
32
+ });
33
+ });
34
+
35
+ it("detects timed-out step_started", async () => {
36
+ // Manually create a workflow with a step_started that's timed out
37
+ (await store.appendBatch([
38
+ { run_id: "run-1", workflow_id: "wf-1", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
39
+ { run_id: "run-1", workflow_id: "wf-1", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1000 },
40
+ ], noSim))._unsafeUnwrap();
41
+
42
+ // Not yet timed out
43
+ clock.set(5999);
44
+ let recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
45
+ expect(recovered).toHaveLength(0);
46
+
47
+ // Now timed out
48
+ clock.set(6000);
49
+ recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
50
+ expect(recovered).toEqual([{ runId: "run-1", reason: "timeout" }]);
51
+
52
+ // step_failed should have been appended
53
+ const events = (await store.getAllEvents("run-1", noSim))._unsafeUnwrap();
54
+ expect(events).toHaveLength(3);
55
+ expect(events[2].event_type).toBe("step_failed");
56
+ expect((events[2].payload as any).reason).toBe("timeout");
57
+ expect((events[2].payload as any).retryable).toBe(true);
58
+ });
59
+
60
+ it("detects stuck workflow at workflow_created", async () => {
61
+ (await store.append({
62
+ run_id: "run-stuck", workflow_id: "wf-stuck", seq: 0, event_type: "workflow_created",
63
+ payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000,
64
+ }, noSim))._unsafeUnwrap();
65
+
66
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
67
+ expect(recovered).toEqual([{ runId: "run-stuck", reason: "stuck_created" }]);
68
+ });
69
+
70
+ it("detects stuck workflow at step_completed with next_step", async () => {
71
+ (await store.appendBatch([
72
+ { run_id: "run-2", workflow_id: "wf-2", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
73
+ { run_id: "run-2", workflow_id: "wf-2", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1000 },
74
+ { run_id: "run-2", workflow_id: "wf-2", seq: 2, event_type: "step_completed", payload: { step: "start", writes: {}, next_step: "two" }, created_at: 1000 },
75
+ ], noSim))._unsafeUnwrap();
76
+
77
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
78
+ expect(recovered).toEqual([{ runId: "run-2", reason: "stuck_completed" }]);
79
+ });
80
+
81
+ it("detects stuck workflow at retryable step_failed", async () => {
82
+ (await store.appendBatch([
83
+ { run_id: "run-3", workflow_id: "wf-3", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
84
+ { run_id: "run-3", workflow_id: "wf-3", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1000 },
85
+ { run_id: "run-3", workflow_id: "wf-3", seq: 2, event_type: "step_failed", payload: { step: "start", attempt: 1, reason: "oops", retryable: true }, created_at: 1000 },
86
+ ], noSim))._unsafeUnwrap();
87
+
88
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
89
+ expect(recovered).toEqual([{ runId: "run-3", reason: "stuck_failed" }]);
90
+ });
91
+
92
+ it("ignores terminal workflows", async () => {
93
+ (await store.appendBatch([
94
+ { run_id: "run-done", workflow_id: "wf-done", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
95
+ { run_id: "run-done", workflow_id: "wf-done", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1001 },
96
+ { run_id: "run-done", workflow_id: "wf-done", seq: 2, event_type: "step_completed", payload: { step: "start", writes: {} }, created_at: 1002 },
97
+ { run_id: "run-done", workflow_id: "wf-done", seq: 3, event_type: "workflow_completed", payload: { result: "ok" }, created_at: 1003 },
98
+ ], noSim))._unsafeUnwrap();
99
+
100
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
101
+ expect(recovered).toHaveLength(0);
102
+ });
103
+
104
+ it("ignores non-timed-out in-progress steps", async () => {
105
+ (await store.appendBatch([
106
+ { run_id: "run-running", workflow_id: "wf-running", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
107
+ { run_id: "run-running", workflow_id: "wf-running", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1000 },
108
+ ], noSim))._unsafeUnwrap();
109
+
110
+ clock.set(3000); // Only 2s elapsed, timeout is 5s
111
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
112
+ expect(recovered).toHaveLength(0);
113
+ });
114
+
115
+ it("recovery + engine integration: timed-out workflow recovers and completes", async () => {
116
+ const engine = new Engine({
117
+ store,
118
+ registry,
119
+ clock,
120
+ generateId: () => `run-${++idCounter}`,
121
+ });
122
+
123
+ // Simulate a step that was claimed but the worker died
124
+ (await store.appendBatch([
125
+ { run_id: "run-10", workflow_id: "wf-10", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
126
+ { run_id: "run-10", workflow_id: "wf-10", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1000 },
127
+ ], noSim))._unsafeUnwrap();
128
+
129
+ // Advance past timeout
130
+ clock.set(7000);
131
+
132
+ // Recovery marks it as failed
133
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
134
+ expect(recovered).toEqual([{ runId: "run-10", reason: "timeout" }]);
135
+
136
+ // Engine retries and completes
137
+ (await engine.runToCompletion("run-10", noSim))._unsafeUnwrap();
138
+
139
+ const events = (await store.getAllEvents("run-10", noSim))._unsafeUnwrap();
140
+ const last = events[events.length - 1];
141
+ expect(last.event_type).toBe("workflow_completed");
142
+ });
143
+
144
+ it("detects step_completed without next_step as needing workflow_completed", async () => {
145
+ (await store.appendBatch([
146
+ { run_id: "run-4", workflow_id: "wf-4", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
147
+ { run_id: "run-4", workflow_id: "wf-4", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1000 },
148
+ { run_id: "run-4", workflow_id: "wf-4", seq: 2, event_type: "step_completed", payload: { step: "start", writes: {} }, created_at: 1000 },
149
+ ], noSim))._unsafeUnwrap();
150
+
151
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
152
+ expect(recovered).toEqual([{ runId: "run-4", reason: "stuck_terminal" }]);
153
+ });
154
+
155
+ it("detects non-retryable step_failed as needing workflow_failed", async () => {
156
+ (await store.appendBatch([
157
+ { run_id: "run-5", workflow_id: "wf-5", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
158
+ { run_id: "run-5", workflow_id: "wf-5", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 1 }, created_at: 1000 },
159
+ { run_id: "run-5", workflow_id: "wf-5", seq: 2, event_type: "step_failed", payload: { step: "start", attempt: 1, reason: "fatal", retryable: false }, created_at: 1000 },
160
+ ], noSim))._unsafeUnwrap();
161
+
162
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
163
+ expect(recovered).toEqual([{ runId: "run-5", reason: "stuck_terminal" }]);
164
+ });
165
+
166
+ it("detects max-attempts-exhausted step_failed as needing workflow_failed", async () => {
167
+ (await store.appendBatch([
168
+ { run_id: "run-6", workflow_id: "wf-6", seq: 0, event_type: "workflow_created", payload: { workflowName: "test-wf", writes: {}, config: TEST_WF_CONFIG }, created_at: 1000 },
169
+ { run_id: "run-6", workflow_id: "wf-6", seq: 1, event_type: "step_started", payload: { step: "start", attempt: 3 }, created_at: 1000 },
170
+ { run_id: "run-6", workflow_id: "wf-6", seq: 2, event_type: "step_failed", payload: { step: "start", attempt: 3, reason: "oops", retryable: true }, created_at: 1000 },
171
+ ], noSim))._unsafeUnwrap();
172
+
173
+ const recovered = (await runRecovery({ store, registry, clock }, noSim))._unsafeUnwrap();
174
+ expect(recovered).toEqual([{ runId: "run-6", reason: "stuck_terminal" }]);
175
+ });
176
+ });
@@ -0,0 +1,395 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import {
3
+ SimulationImpl,
4
+ SimpleEntropySource,
5
+ type SimulationTask,
6
+ type TaskSpec,
7
+ } from "determined";
8
+ import { Engine } from "../src/engine.js";
9
+ import { MemoryEventStore } from "../src/store-memory.js";
10
+ import { SqliteEventStore } from "../src/store-sqlite.js";
11
+ import { WorkflowRegistry } from "../src/registry.js";
12
+ import { ManualClock } from "../src/clock.js";
13
+ import { runRecovery } from "../src/recovery.js";
14
+ import type { EventStore, EventStoreCheck } from "../src/store.js";
15
+ import type { Continuation, Ctx, WorkflowEvent } from "../src/types.js";
16
+ import { noSim } from "./fixtures.js";
17
+
18
+ function makeRegistry(): WorkflowRegistry {
19
+ const registry = new WorkflowRegistry();
20
+
21
+ registry.register({
22
+ name: "counter",
23
+ fn: async (ctx: Ctx): Promise<Continuation> => {
24
+ switch (ctx.step) {
25
+ case "start": {
26
+ ctx.set("count", 0);
27
+ return { step: "increment" };
28
+ }
29
+ case "increment": {
30
+ const count = (await ctx.get("count")) as number;
31
+ if (count >= 3) {
32
+ return { done: true, result: { finalCount: count } };
33
+ }
34
+ ctx.set("count", count + 1);
35
+ return { step: "increment" };
36
+ }
37
+ default:
38
+ throw new Error(`Unknown step: ${ctx.step}`);
39
+ }
40
+ },
41
+ maxAttempts: 3,
42
+ stepTimeoutMS: 30_000,
43
+ initialBackoffMS: 0,
44
+ });
45
+
46
+ registry.register({
47
+ name: "simple",
48
+ fn: async (ctx: Ctx): Promise<Continuation> => {
49
+ const data = await ctx.get("seed") ?? await ctx.get("test");
50
+ ctx.set("done", true);
51
+ return { done: true, result: { data } };
52
+ },
53
+ maxAttempts: 3,
54
+ stepTimeoutMS: 30_000,
55
+ initialBackoffMS: 0,
56
+ });
57
+
58
+ return registry;
59
+ }
60
+
61
+ const validTransitions: Record<string, string[]> = {
62
+ workflow_created: ["step_started"],
63
+ step_started: ["step_completed", "step_failed"],
64
+ step_completed: ["step_started", "workflow_completed"],
65
+ step_failed: ["step_started", "workflow_failed"],
66
+ };
67
+
68
+ function makeCheck(): EventStoreCheck {
69
+ return (runId: string, events: WorkflowEvent[]) => {
70
+ if (events.length === 0) return;
71
+
72
+ // Ascending seq with no gaps
73
+ for (let i = 0; i < events.length; i++) {
74
+ if (events[i].seq !== i) {
75
+ throw new Error(
76
+ `Check failed [${runId}]: expected seq ${i}, got ${events[i].seq}`,
77
+ );
78
+ }
79
+ }
80
+
81
+ // First event must be workflow_created
82
+ if (events[0].event_type !== "workflow_created") {
83
+ throw new Error(
84
+ `Check failed [${runId}]: first event is ${events[0].event_type}, expected workflow_created`,
85
+ );
86
+ }
87
+
88
+ // All transitions are valid
89
+ for (let i = 0; i < events.length - 1; i++) {
90
+ const from = events[i].event_type;
91
+ const to = events[i + 1].event_type;
92
+ const allowed = validTransitions[from];
93
+ if (!allowed?.includes(to)) {
94
+ throw new Error(
95
+ `Check failed [${runId}]: invalid transition ${from} → ${to} at seq ${i}→${i + 1}`,
96
+ );
97
+ }
98
+ }
99
+
100
+ // At most one terminal event
101
+ const terminals = events.filter(
102
+ (e) =>
103
+ e.event_type === "workflow_completed" ||
104
+ e.event_type === "workflow_failed",
105
+ );
106
+ if (terminals.length > 1) {
107
+ throw new Error(
108
+ `Check failed [${runId}]: ${terminals.length} terminal events`,
109
+ );
110
+ }
111
+ };
112
+ }
113
+
114
+ function validateEventTrace(events: WorkflowEvent[]): void {
115
+ // Check ascending seq
116
+ for (let i = 0; i < events.length; i++) {
117
+ if (events[i].seq !== i) {
118
+ throw new Error(`Expected seq ${i}, got ${events[i].seq}`);
119
+ }
120
+ }
121
+
122
+ // Check valid transitions
123
+ for (let i = 0; i < events.length - 1; i++) {
124
+ const from = events[i].event_type;
125
+ const to = events[i + 1].event_type;
126
+ const allowed = validTransitions[from];
127
+ if (!allowed?.includes(to)) {
128
+ throw new Error(
129
+ `Invalid transition: ${from} → ${to} at seq ${i}→${i + 1}`,
130
+ );
131
+ }
132
+ }
133
+
134
+ // Check terminal
135
+ const last = events[events.length - 1];
136
+ if (
137
+ last.event_type !== "workflow_completed" &&
138
+ last.event_type !== "workflow_failed"
139
+ ) {
140
+ throw new Error(`Workflow not terminal: last event is ${last.event_type}`);
141
+ }
142
+ }
143
+
144
+ function simulationTests(
145
+ name: string,
146
+ createStore: () => Promise<EventStore>,
147
+ ) {
148
+ describe(`simulation tests (${name})`, () => {
149
+ it("concurrent workers: exactly one claims each step", async () => {
150
+ const store = await createStore();
151
+ const registry = makeRegistry();
152
+ const clock = new ManualClock(1000);
153
+ let idCounter = 0;
154
+
155
+ const engine = new Engine({
156
+ store,
157
+ registry,
158
+ clock,
159
+ generateId: () => `run-${++idCounter}`,
160
+ });
161
+
162
+ const runId = (await engine.createWorkflow("counter-wf", "counter", {}, noSim))._unsafeUnwrap();
163
+
164
+ const logger = { log: () => {}, error: () => {} };
165
+ const entropy = new SimpleEntropySource();
166
+
167
+ const sim = new SimulationImpl(logger, entropy, 0);
168
+
169
+ const workerSpec = (workerName: string): TaskSpec<void> => ({
170
+ name: workerName,
171
+ f: async (task: SimulationTask) => {
172
+ for (let i = 0; i < 20; i++) {
173
+ const result = await engine.executeStep(runId, task);
174
+ if (result.isErr()) continue; // treat err as crash
175
+ }
176
+ },
177
+ });
178
+
179
+ const result = await sim.runTasks([
180
+ workerSpec("worker-1"),
181
+ workerSpec("worker-2"),
182
+ workerSpec("worker-3"),
183
+ ] as const);
184
+
185
+ expect(result.isOk()).toBe(true);
186
+
187
+ const events = (await store.getAllEvents(runId, noSim))._unsafeUnwrap();
188
+ validateEventTrace(events);
189
+
190
+ const last = events[events.length - 1];
191
+ expect(last.event_type).toBe("workflow_completed");
192
+ expect((last.payload as any).result).toEqual({ finalCount: 3 });
193
+ });
194
+
195
+ it("concurrent workers across multiple seeds", async () => {
196
+ for (let seed = 0; seed < 100; seed++) {
197
+ const store = await createStore();
198
+ const registry = makeRegistry();
199
+ const clock = new ManualClock(1000);
200
+ let idCounter = 0;
201
+
202
+ const engine = new Engine({
203
+ store,
204
+ registry,
205
+ clock,
206
+ generateId: () => `run-${++idCounter}`,
207
+ });
208
+
209
+ const runId = (await engine.createWorkflow("simple-wf", "simple", {
210
+ seed,
211
+ }, noSim))._unsafeUnwrap();
212
+
213
+ const logger = { log: () => {}, error: () => {} };
214
+
215
+ const entropy = {
216
+ random: () => {
217
+ const x = Math.sin(seed * 9301 + 49297) * 49297;
218
+ return x - Math.floor(x);
219
+ },
220
+ };
221
+
222
+ const sim = new SimulationImpl(logger, entropy, 0);
223
+
224
+ const workerSpec = (workerName: string): TaskSpec<void> => ({
225
+ name: workerName,
226
+ f: async (task: SimulationTask) => {
227
+ for (let i = 0; i < 10; i++) {
228
+ const result = await engine.executeStep(runId, task);
229
+ if (result.isErr()) continue; // treat err as crash
230
+ }
231
+ },
232
+ });
233
+
234
+ const result = await sim.runTasks([
235
+ workerSpec("w1"),
236
+ workerSpec("w2"),
237
+ ] as const);
238
+
239
+ expect(result.isOk(), `Seed ${seed} failed`).toBe(true);
240
+
241
+ const events = (await store.getAllEvents(runId, noSim))._unsafeUnwrap();
242
+ validateEventTrace(events);
243
+
244
+ const last = events[events.length - 1];
245
+ expect(last.event_type).toBe("workflow_completed");
246
+ }
247
+ });
248
+
249
+ it("failure injection with recovery", async () => {
250
+ const store = await createStore();
251
+ const registry = makeRegistry();
252
+ const clock = new ManualClock(1000);
253
+ let idCounter = 0;
254
+
255
+ const engine = new Engine({
256
+ store,
257
+ registry,
258
+ clock,
259
+ generateId: () => `run-${++idCounter}`,
260
+ });
261
+
262
+ const runId = (await engine.createWorkflow("counter-wf", "counter", {}, noSim))._unsafeUnwrap();
263
+
264
+ const logger = { log: () => {}, error: () => {} };
265
+ const entropy = new SimpleEntropySource();
266
+
267
+ const sim = new SimulationImpl(logger, entropy, 0.3);
268
+
269
+ const workerSpec = (workerName: string): TaskSpec<void> => ({
270
+ name: workerName,
271
+ f: async (task: SimulationTask) => {
272
+ for (let i = 0; i < 50; i++) {
273
+ const result = await engine.executeStep(runId, task);
274
+ if (result.isErr()) {
275
+ // Err — simulate crash, just keep trying
276
+ }
277
+ }
278
+ },
279
+ });
280
+
281
+ const recoverySpec: TaskSpec<void> = {
282
+ name: "recovery",
283
+ f: async (task: SimulationTask) => {
284
+ for (let i = 0; i < 30; i++) {
285
+ clock.advance(35_000);
286
+
287
+ try {
288
+ const recoveryResult = await runRecovery(
289
+ { store, registry, clock },
290
+ task,
291
+ );
292
+ if (recoveryResult.isErr()) continue; // treat err as crash
293
+ const recovered = recoveryResult.value;
294
+
295
+ for (const r of recovered) {
296
+ const stepResult = await engine.executeStep(r.runId, task);
297
+ if (stepResult.isErr()) {
298
+ // failpoint
299
+ }
300
+ }
301
+ } catch {
302
+ // failpoint in store during recovery
303
+ }
304
+ }
305
+ },
306
+ };
307
+
308
+ await sim.runTasks([
309
+ workerSpec("worker-1"),
310
+ workerSpec("worker-2"),
311
+ recoverySpec,
312
+ ] as const);
313
+
314
+ const events = (await store.getAllEvents(runId, noSim))._unsafeUnwrap();
315
+
316
+ if (events.length > 0) {
317
+ for (let i = 0; i < events.length; i++) {
318
+ expect(events[i].seq).toBe(i);
319
+ }
320
+
321
+ for (let i = 0; i < events.length - 1; i++) {
322
+ const from = events[i].event_type;
323
+ const to = events[i + 1].event_type;
324
+ const allowed = validTransitions[from];
325
+ expect(allowed, `No transitions defined from ${from}`).toBeDefined();
326
+ expect(
327
+ allowed,
328
+ `Invalid transition: ${from} → ${to} at seq ${i}→${i + 1}`,
329
+ ).toContain(to);
330
+ }
331
+ }
332
+ });
333
+
334
+ it("appendBatch atomicity under concurrency", async () => {
335
+ const store = await createStore();
336
+ const registry = makeRegistry();
337
+ const clock = new ManualClock(1000);
338
+ let idCounter = 0;
339
+
340
+ const engine = new Engine({
341
+ store,
342
+ registry,
343
+ clock,
344
+ generateId: () => `run-${++idCounter}`,
345
+ });
346
+
347
+ const runId = (await engine.createWorkflow("simple-wf", "simple", { test: true }, noSim))._unsafeUnwrap();
348
+
349
+ const logger = { log: () => {}, error: () => {} };
350
+ const entropy = new SimpleEntropySource();
351
+ const sim = new SimulationImpl(logger, entropy, 0);
352
+
353
+ const workerSpec = (workerName: string): TaskSpec<void> => ({
354
+ name: workerName,
355
+ f: async (task: SimulationTask) => {
356
+ for (let i = 0; i < 10; i++) {
357
+ const result = await engine.executeStep(runId, task);
358
+ if (result.isErr()) continue; // treat err as crash
359
+ }
360
+ },
361
+ });
362
+
363
+ const result = await sim.runTasks([
364
+ workerSpec("w1"),
365
+ workerSpec("w2"),
366
+ workerSpec("w3"),
367
+ ] as const);
368
+
369
+ expect(result.isOk()).toBe(true);
370
+
371
+ const events = (await store.getAllEvents(runId, noSim))._unsafeUnwrap();
372
+ validateEventTrace(events);
373
+
374
+ const terminals = events.filter(
375
+ (e) =>
376
+ e.event_type === "workflow_completed" ||
377
+ e.event_type === "workflow_failed",
378
+ );
379
+ expect(terminals).toHaveLength(1);
380
+ });
381
+ });
382
+ }
383
+
384
+ // Run against both store implementations, with check callback
385
+ simulationTests("MemoryEventStore", async () => {
386
+ const store = new MemoryEventStore({ check: makeCheck() });
387
+ (await store.initialize())._unsafeUnwrap();
388
+ return store;
389
+ });
390
+
391
+ simulationTests("SqliteEventStore", async () => {
392
+ const store = new SqliteEventStore(":memory:", { check: makeCheck() });
393
+ (await store.initialize())._unsafeUnwrap();
394
+ return store;
395
+ });