groundswell 0.0.2 → 1.0.0

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 (554) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +26 -9
  3. package/dist/cache/cache-key.d.ts +86 -0
  4. package/dist/cache/cache-key.d.ts.map +1 -0
  5. package/dist/cache/cache-key.js +204 -0
  6. package/dist/cache/cache-key.js.map +1 -0
  7. package/dist/cache/cache.d.ts +104 -0
  8. package/dist/cache/cache.d.ts.map +1 -0
  9. package/dist/cache/cache.js +179 -0
  10. package/dist/cache/cache.js.map +1 -0
  11. package/{src/cache/index.ts → dist/cache/index.d.ts} +1 -1
  12. package/dist/cache/index.d.ts.map +1 -0
  13. package/dist/cache/index.js +6 -0
  14. package/dist/cache/index.js.map +1 -0
  15. package/dist/core/agent.d.ts +203 -0
  16. package/dist/core/agent.d.ts.map +1 -0
  17. package/dist/core/agent.js +833 -0
  18. package/dist/core/agent.js.map +1 -0
  19. package/{src/core/context.ts → dist/core/context.d.ts} +16 -67
  20. package/dist/core/context.d.ts.map +1 -0
  21. package/dist/core/context.js +80 -0
  22. package/dist/core/context.js.map +1 -0
  23. package/dist/core/event-tree.d.ts +72 -0
  24. package/dist/core/event-tree.d.ts.map +1 -0
  25. package/dist/core/event-tree.js +211 -0
  26. package/dist/core/event-tree.js.map +1 -0
  27. package/{src/core/factory.ts → dist/core/factory.d.ts} +6 -27
  28. package/dist/core/factory.d.ts.map +1 -0
  29. package/dist/core/factory.js +110 -0
  30. package/dist/core/factory.js.map +1 -0
  31. package/{src/core/index.ts → dist/core/index.d.ts} +2 -10
  32. package/dist/core/index.d.ts.map +1 -0
  33. package/dist/core/index.js +9 -0
  34. package/dist/core/index.js.map +1 -0
  35. package/dist/core/logger.d.ts +50 -0
  36. package/dist/core/logger.d.ts.map +1 -0
  37. package/dist/core/logger.js +91 -0
  38. package/dist/core/logger.js.map +1 -0
  39. package/dist/core/mcp-handler.d.ts +127 -0
  40. package/dist/core/mcp-handler.d.ts.map +1 -0
  41. package/dist/core/mcp-handler.js +323 -0
  42. package/dist/core/mcp-handler.js.map +1 -0
  43. package/dist/core/prompt.d.ts +80 -0
  44. package/dist/core/prompt.d.ts.map +1 -0
  45. package/dist/core/prompt.js +120 -0
  46. package/dist/core/prompt.js.map +1 -0
  47. package/dist/core/workflow-context.d.ts +61 -0
  48. package/dist/core/workflow-context.d.ts.map +1 -0
  49. package/dist/core/workflow-context.js +358 -0
  50. package/dist/core/workflow-context.js.map +1 -0
  51. package/dist/core/workflow.d.ts +543 -0
  52. package/dist/core/workflow.d.ts.map +1 -0
  53. package/dist/core/workflow.js +986 -0
  54. package/dist/core/workflow.js.map +1 -0
  55. package/dist/debugger/event-replayer.d.ts +422 -0
  56. package/dist/debugger/event-replayer.d.ts.map +1 -0
  57. package/dist/debugger/event-replayer.js +639 -0
  58. package/dist/debugger/event-replayer.js.map +1 -0
  59. package/dist/debugger/index.d.ts +2 -0
  60. package/dist/debugger/index.d.ts.map +1 -0
  61. package/{src/debugger/index.ts → dist/debugger/index.js} +1 -0
  62. package/dist/debugger/index.js.map +1 -0
  63. package/dist/debugger/tree-debugger.d.ts +240 -0
  64. package/dist/debugger/tree-debugger.d.ts.map +1 -0
  65. package/dist/debugger/tree-debugger.js +620 -0
  66. package/dist/debugger/tree-debugger.js.map +1 -0
  67. package/dist/decorators/index.d.ts +4 -0
  68. package/dist/decorators/index.d.ts.map +1 -0
  69. package/{src/decorators/index.ts → dist/decorators/index.js} +1 -0
  70. package/dist/decorators/index.js.map +1 -0
  71. package/dist/decorators/observed-state.d.ts +32 -0
  72. package/dist/decorators/observed-state.d.ts.map +1 -0
  73. package/dist/decorators/observed-state.js +79 -0
  74. package/dist/decorators/observed-state.js.map +1 -0
  75. package/dist/decorators/step.d.ts +15 -0
  76. package/dist/decorators/step.d.ts.map +1 -0
  77. package/dist/decorators/step.js +192 -0
  78. package/dist/decorators/step.js.map +1 -0
  79. package/dist/decorators/task.d.ts +50 -0
  80. package/dist/decorators/task.d.ts.map +1 -0
  81. package/dist/decorators/task.js +118 -0
  82. package/dist/decorators/task.js.map +1 -0
  83. package/dist/examples/index.d.ts +3 -0
  84. package/dist/examples/index.d.ts.map +1 -0
  85. package/{src/examples/index.ts → dist/examples/index.js} +1 -0
  86. package/dist/examples/index.js.map +1 -0
  87. package/dist/examples/tdd-orchestrator.d.ts +15 -0
  88. package/dist/examples/tdd-orchestrator.d.ts.map +1 -0
  89. package/dist/examples/tdd-orchestrator.js +121 -0
  90. package/dist/examples/tdd-orchestrator.js.map +1 -0
  91. package/dist/examples/test-cycle-workflow.d.ts +14 -0
  92. package/dist/examples/test-cycle-workflow.d.ts.map +1 -0
  93. package/dist/examples/test-cycle-workflow.js +116 -0
  94. package/dist/examples/test-cycle-workflow.js.map +1 -0
  95. package/dist/harnesses/claude-code-harness.d.ts +391 -0
  96. package/dist/harnesses/claude-code-harness.d.ts.map +1 -0
  97. package/dist/harnesses/claude-code-harness.js +1076 -0
  98. package/dist/harnesses/claude-code-harness.js.map +1 -0
  99. package/dist/harnesses/harness-registry.d.ts +440 -0
  100. package/dist/harnesses/harness-registry.d.ts.map +1 -0
  101. package/dist/harnesses/harness-registry.js +543 -0
  102. package/dist/harnesses/harness-registry.js.map +1 -0
  103. package/dist/harnesses/index.d.ts +12 -0
  104. package/dist/harnesses/index.d.ts.map +1 -0
  105. package/dist/harnesses/index.js +11 -0
  106. package/dist/harnesses/index.js.map +1 -0
  107. package/dist/harnesses/pi-harness.d.ts +219 -0
  108. package/dist/harnesses/pi-harness.d.ts.map +1 -0
  109. package/dist/harnesses/pi-harness.js +676 -0
  110. package/dist/harnesses/pi-harness.js.map +1 -0
  111. package/dist/harnesses/pi-schema-converter.d.ts +24 -0
  112. package/dist/harnesses/pi-schema-converter.d.ts.map +1 -0
  113. package/dist/harnesses/pi-schema-converter.js +81 -0
  114. package/dist/harnesses/pi-schema-converter.js.map +1 -0
  115. package/dist/harnesses/register-defaults.d.ts +24 -0
  116. package/dist/harnesses/register-defaults.d.ts.map +1 -0
  117. package/dist/harnesses/register-defaults.js +40 -0
  118. package/dist/harnesses/register-defaults.js.map +1 -0
  119. package/dist/harnesses/session-store.d.ts +201 -0
  120. package/dist/harnesses/session-store.d.ts.map +1 -0
  121. package/dist/harnesses/session-store.js +254 -0
  122. package/dist/harnesses/session-store.js.map +1 -0
  123. package/dist/index.d.ts +37 -0
  124. package/dist/index.d.ts.map +1 -0
  125. package/dist/index.js +57 -0
  126. package/dist/index.js.map +1 -0
  127. package/dist/reflection/index.d.ts +5 -0
  128. package/dist/reflection/index.d.ts.map +1 -0
  129. package/{src/reflection/index.ts → dist/reflection/index.js} +1 -1
  130. package/dist/reflection/index.js.map +1 -0
  131. package/dist/reflection/reflection.d.ts +84 -0
  132. package/dist/reflection/reflection.d.ts.map +1 -0
  133. package/dist/reflection/reflection.js +344 -0
  134. package/dist/reflection/reflection.js.map +1 -0
  135. package/dist/tools/index.d.ts +6 -0
  136. package/dist/tools/index.d.ts.map +1 -0
  137. package/dist/tools/index.js +11 -0
  138. package/dist/tools/index.js.map +1 -0
  139. package/dist/tools/introspection.d.ts +165 -0
  140. package/dist/tools/introspection.d.ts.map +1 -0
  141. package/dist/tools/introspection.js +324 -0
  142. package/dist/tools/introspection.js.map +1 -0
  143. package/dist/types/agent.d.ts +1317 -0
  144. package/dist/types/agent.d.ts.map +1 -0
  145. package/dist/types/agent.js +423 -0
  146. package/dist/types/agent.js.map +1 -0
  147. package/dist/types/decorators.d.ts +40 -0
  148. package/dist/types/decorators.d.ts.map +1 -0
  149. package/dist/types/decorators.js +2 -0
  150. package/dist/types/decorators.js.map +1 -0
  151. package/dist/types/error-strategy.d.ts +13 -0
  152. package/dist/types/error-strategy.d.ts.map +1 -0
  153. package/dist/types/error-strategy.js +2 -0
  154. package/dist/types/error-strategy.js.map +1 -0
  155. package/dist/types/error.d.ts +20 -0
  156. package/dist/types/error.d.ts.map +1 -0
  157. package/dist/types/error.js +2 -0
  158. package/dist/types/error.js.map +1 -0
  159. package/dist/types/events.d.ts +113 -0
  160. package/dist/types/events.d.ts.map +1 -0
  161. package/dist/types/events.js +2 -0
  162. package/dist/types/events.js.map +1 -0
  163. package/dist/types/harnesses.d.ts +474 -0
  164. package/dist/types/harnesses.d.ts.map +1 -0
  165. package/dist/types/harnesses.js +2 -0
  166. package/dist/types/harnesses.js.map +1 -0
  167. package/dist/types/index.d.ts +23 -0
  168. package/dist/types/index.d.ts.map +1 -0
  169. package/dist/types/index.js +8 -0
  170. package/dist/types/index.js.map +1 -0
  171. package/dist/types/logging.d.ts +24 -0
  172. package/dist/types/logging.d.ts.map +1 -0
  173. package/dist/types/logging.js +2 -0
  174. package/dist/types/logging.js.map +1 -0
  175. package/dist/types/observer.d.ts +18 -0
  176. package/dist/types/observer.d.ts.map +1 -0
  177. package/dist/types/observer.js +2 -0
  178. package/dist/types/observer.js.map +1 -0
  179. package/dist/types/prompt.d.ts +31 -0
  180. package/dist/types/prompt.d.ts.map +1 -0
  181. package/dist/types/prompt.js +6 -0
  182. package/dist/types/prompt.js.map +1 -0
  183. package/dist/types/providers.d.ts +691 -0
  184. package/dist/types/providers.d.ts.map +1 -0
  185. package/dist/types/providers.js +14 -0
  186. package/dist/types/providers.js.map +1 -0
  187. package/dist/types/reflection.d.ts +96 -0
  188. package/dist/types/reflection.d.ts.map +1 -0
  189. package/dist/types/reflection.js +24 -0
  190. package/dist/types/reflection.js.map +1 -0
  191. package/dist/types/restart.d.ts +132 -0
  192. package/dist/types/restart.d.ts.map +1 -0
  193. package/dist/types/restart.js +2 -0
  194. package/dist/types/restart.js.map +1 -0
  195. package/dist/types/sdk-primitives.d.ts +118 -0
  196. package/dist/types/sdk-primitives.d.ts.map +1 -0
  197. package/dist/types/sdk-primitives.js +6 -0
  198. package/dist/types/sdk-primitives.js.map +1 -0
  199. package/{src/types/snapshot.ts → dist/types/snapshot.d.ts} +5 -5
  200. package/dist/types/snapshot.d.ts.map +1 -0
  201. package/dist/types/snapshot.js +2 -0
  202. package/dist/types/snapshot.js.map +1 -0
  203. package/dist/types/streaming.d.ts +194 -0
  204. package/dist/types/streaming.d.ts.map +1 -0
  205. package/dist/types/streaming.js +67 -0
  206. package/dist/types/streaming.js.map +1 -0
  207. package/dist/types/workflow-context.d.ts +275 -0
  208. package/dist/types/workflow-context.d.ts.map +1 -0
  209. package/dist/types/workflow-context.js +8 -0
  210. package/dist/types/workflow-context.js.map +1 -0
  211. package/dist/types/workflow.d.ts +30 -0
  212. package/dist/types/workflow.d.ts.map +1 -0
  213. package/dist/types/workflow.js +2 -0
  214. package/dist/types/workflow.js.map +1 -0
  215. package/dist/utils/agent-validation.d.ts +88 -0
  216. package/dist/utils/agent-validation.d.ts.map +1 -0
  217. package/dist/utils/agent-validation.js +87 -0
  218. package/dist/utils/agent-validation.js.map +1 -0
  219. package/dist/utils/delay.d.ts +7 -0
  220. package/dist/utils/delay.d.ts.map +1 -0
  221. package/dist/utils/delay.js +9 -0
  222. package/dist/utils/delay.js.map +1 -0
  223. package/dist/utils/harness-config.d.ts +180 -0
  224. package/dist/utils/harness-config.d.ts.map +1 -0
  225. package/dist/utils/harness-config.js +311 -0
  226. package/dist/utils/harness-config.js.map +1 -0
  227. package/dist/utils/id.d.ts +6 -0
  228. package/dist/utils/id.d.ts.map +1 -0
  229. package/dist/utils/id.js +12 -0
  230. package/dist/utils/id.js.map +1 -0
  231. package/dist/utils/index.d.ts +13 -0
  232. package/dist/utils/index.d.ts.map +1 -0
  233. package/dist/utils/index.js +11 -0
  234. package/dist/utils/index.js.map +1 -0
  235. package/dist/utils/model-spec.d.ts +110 -0
  236. package/dist/utils/model-spec.d.ts.map +1 -0
  237. package/dist/utils/model-spec.js +149 -0
  238. package/dist/utils/model-spec.js.map +1 -0
  239. package/dist/utils/observable.d.ts +54 -0
  240. package/dist/utils/observable.d.ts.map +1 -0
  241. package/dist/utils/observable.js +82 -0
  242. package/dist/utils/observable.js.map +1 -0
  243. package/dist/utils/provider-config.d.ts +10 -0
  244. package/dist/utils/provider-config.d.ts.map +1 -0
  245. package/dist/utils/provider-config.js +10 -0
  246. package/dist/utils/provider-config.js.map +1 -0
  247. package/dist/utils/restart-analysis.d.ts +202 -0
  248. package/dist/utils/restart-analysis.d.ts.map +1 -0
  249. package/dist/utils/restart-analysis.js +426 -0
  250. package/dist/utils/restart-analysis.js.map +1 -0
  251. package/dist/utils/session-serialization.d.ts +118 -0
  252. package/dist/utils/session-serialization.d.ts.map +1 -0
  253. package/dist/utils/session-serialization.js +217 -0
  254. package/dist/utils/session-serialization.js.map +1 -0
  255. package/dist/utils/workflow-error-utils.d.ts +22 -0
  256. package/dist/utils/workflow-error-utils.d.ts.map +1 -0
  257. package/dist/utils/workflow-error-utils.js +45 -0
  258. package/dist/utils/workflow-error-utils.js.map +1 -0
  259. package/package.json +34 -5
  260. package/.claude/commands/subtask-planning/prp-base-create.md +0 -120
  261. package/.claude/commands/subtask-planning/prp-base-execute.md +0 -65
  262. package/.claude/commands/task-breakdown.md +0 -94
  263. package/.claude/settings.local.json +0 -9
  264. package/.claude/system_prompts/task-breakdown.md +0 -101
  265. package/CHANGELOG.md +0 -188
  266. package/PRD.md +0 -543
  267. package/PRPs/001-hierarchical-workflow-engine.md +0 -2438
  268. package/PRPs/PRDs/002-agent-prompt.md +0 -390
  269. package/PRPs/PRDs/003-agent-prompt.md +0 -943
  270. package/PRPs/PRDs/004-agent-prompt.md +0 -1136
  271. package/PRPs/PRDs/tasks-001.json +0 -492
  272. package/PRPs/README.md +0 -83
  273. package/PRPs/templates/prp_base.md +0 -222
  274. package/docs/agent.md +0 -422
  275. package/docs/prompt.md +0 -419
  276. package/docs/workflow.md +0 -600
  277. package/examples/README.md +0 -258
  278. package/examples/examples/01-basic-workflow.ts +0 -100
  279. package/examples/examples/02-decorator-options.ts +0 -217
  280. package/examples/examples/03-parent-child.ts +0 -241
  281. package/examples/examples/04-observers-debugger.ts +0 -340
  282. package/examples/examples/05-error-handling.ts +0 -387
  283. package/examples/examples/06-concurrent-tasks.ts +0 -352
  284. package/examples/examples/07-agent-loops.ts +0 -432
  285. package/examples/examples/08-sdk-features.ts +0 -667
  286. package/examples/examples/09-reflection.ts +0 -573
  287. package/examples/examples/10-introspection.ts +0 -550
  288. package/examples/examples/11-reparenting-workflows.ts +0 -269
  289. package/examples/index.ts +0 -147
  290. package/examples/utils/helpers.ts +0 -57
  291. package/package-lock.json +0 -2398
  292. package/plan/001_d3bb02af4886/TEST_RESULTS.md +0 -259
  293. package/plan/001_d3bb02af4886/backlog.json +0 -867
  294. package/plan/001_d3bb02af4886/bug_fix_tasks.json +0 -484
  295. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S1/PRP.md +0 -488
  296. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S2/PRP.md +0 -581
  297. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S3/PRP.md +0 -687
  298. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S1/PRP.md +0 -492
  299. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/PRP.md +0 -932
  300. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/concurrent_error_testing_patterns.md +0 -1109
  301. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/vitest_concurrent_testing.md +0 -802
  302. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/workflow_engine_test_references.md +0 -603
  303. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S1/PRP.md +0 -564
  304. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S3/PRP.md +0 -518
  305. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S4/PRP.md +0 -1252
  306. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/PRP.md +0 -364
  307. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/CODEBASE_INVENTORY.md +0 -114
  308. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/DECORATOR_DOCUMENTATION_PATTERNS.md +0 -205
  309. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/PRD_LOCATION_ANALYSIS.md +0 -199
  310. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/ULTRATHINK_PRP_PLAN.md +0 -134
  311. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/PRP.md +0 -495
  312. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/research/console_error_inventory.md +0 -435
  313. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S2/PRP.md +0 -506
  314. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S3/PRP.md +0 -612
  315. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/PRP.md +0 -558
  316. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/research/external_research.md +0 -788
  317. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S2/PRP.md +0 -460
  318. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S3/PRP.md +0 -454
  319. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/PRP.md +0 -520
  320. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/RECOMMENDATION.md +0 -417
  321. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/external_workflow_engines_research.md +0 -760
  322. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/security_implications_analysis.md +0 -245
  323. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S2/PRP.md +0 -792
  324. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/PRP.md +0 -535
  325. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/TEST_EXECUTION_REPORT.md +0 -190
  326. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/PRP.md +0 -654
  327. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/TEST_FIX_REPORT.md +0 -227
  328. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/KEY_FINDINGS.md +0 -345
  329. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/QUICK_REFERENCE.md +0 -193
  330. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/test_maintenance_research.md +0 -1323
  331. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/BREAKING_CHANGES_AUDIT.md +0 -1011
  332. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/PRP.md +0 -927
  333. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S2/PRP.md +0 -505
  334. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/architecture/logger_child_signature_analysis.md +0 -401
  335. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/child_implementation_research.md +0 -142
  336. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/test_patterns_research.md +0 -112
  337. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/vitest_patterns_research.md +0 -159
  338. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/PRP.md +0 -549
  339. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/VERIFICATION_REPORT.md +0 -368
  340. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/edge_case_analysis.md +0 -172
  341. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/usage_inventory.md +0 -175
  342. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S2/PRP.md +0 -696
  343. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S4/PRP.md +0 -860
  344. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/PRP.md +0 -1066
  345. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01-testing-aggregated-errors.md +0 -1103
  346. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01_typescript_error_aggregation_patterns.md +0 -789
  347. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02-error-merge-strategy-testing-guide.md +0 -1098
  348. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02_aggregate_error_patterns.md +0 -1037
  349. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03-promise-allsettled-testing-patterns.md +0 -916
  350. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03_error_merging_strategies.md +0 -1045
  351. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/04_github_stackoverflow_examples.md +0 -890
  352. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/05_comprehensive_summary.md +0 -822
  353. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/INDEX.md +0 -668
  354. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/QUICK_REFERENCE.md +0 -706
  355. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/README.md +0 -265
  356. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/RESEARCH_REPORT.md +0 -655
  357. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S4/research/vitest_testing_patterns.md +0 -1103
  358. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T3S2/PRP.md +0 -426
  359. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/PRP.md +0 -506
  360. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/QUICK_REFERENCE.md +0 -114
  361. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/RESEARCH_SUMMARY.md +0 -316
  362. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/vitest_observer_error_logging_best_practices.md +0 -754
  363. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S3/PRP.md +0 -612
  364. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/PRP.md +0 -719
  365. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/README.md +0 -215
  366. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/analysis.md +0 -765
  367. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S3/PRP.md +0 -718
  368. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/DECISION.md +0 -149
  369. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/PRP.md +0 -470
  370. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/ULTRATHINK_PLAN.md +0 -332
  371. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/codebase_workflow_name_analysis.md +0 -167
  372. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/external_best_practices.md +0 -265
  373. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/validation_patterns.md +0 -273
  374. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S1/workflow_engine_ancestry_api_research.md +0 -760
  375. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S3-PRP.md +0 -434
  376. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S1/PRP.md +0 -717
  377. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/PRP.md +0 -472
  378. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/VALIDATION_REPORT.md +0 -125
  379. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/research/ULTRATHINK_PRP_PLAN.md +0 -301
  380. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/error-logging-best-practices.md +0 -1170
  381. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/research_typescript_partial_and_overloads.md +0 -940
  382. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-quick-reference.md +0 -151
  383. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-research.md +0 -650
  384. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/prd_snapshot.md +0 -259
  385. package/plan/001_d3bb02af4886/bugfix/P1M1T1S1/PRP.md +0 -457
  386. package/plan/001_d3bb02af4886/bugfix/RESEARCH_SUMMARY.md +0 -346
  387. package/plan/001_d3bb02af4886/bugfix/architecture/codebase_structure.md +0 -311
  388. package/plan/001_d3bb02af4886/bugfix/architecture/concurrent_execution_best_practices.md +0 -1565
  389. package/plan/001_d3bb02af4886/bugfix/architecture/error_handling_patterns.md +0 -288
  390. package/plan/001_d3bb02af4886/bugfix/architecture/promise_all_analysis.md +0 -741
  391. package/plan/001_d3bb02af4886/docs/PRP/P1M1T1S4-functional-workflow-error-state-capture-test.md +0 -652
  392. package/plan/001_d3bb02af4886/docs/PRP/P1P2-PRP.md +0 -527
  393. package/plan/001_d3bb02af4886/docs/PRP/P3P4-PRP.md +0 -1388
  394. package/plan/001_d3bb02af4886/docs/PRP/P4P5-PRP.md +0 -1136
  395. package/plan/001_d3bb02af4886/docs/PRP/PRP.md +0 -527
  396. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S1-PRP.md +0 -415
  397. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S2-PRP.md +0 -378
  398. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S4-PRP.md +0 -713
  399. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M2T1S4-PRP.md +0 -370
  400. package/plan/001_d3bb02af4886/docs/PRP_P1M3T1S3.md +0 -499
  401. package/plan/001_d3bb02af4886/docs/TEST_RESULTS.md +0 -230
  402. package/plan/001_d3bb02af4886/docs/architecture/external_deps.md +0 -358
  403. package/plan/001_d3bb02af4886/docs/architecture/system_context.md +0 -242
  404. package/plan/001_d3bb02af4886/docs/bugfix/ANALYSIS_PRD_VS_IMPLEMENTATION.md +0 -1134
  405. package/plan/001_d3bb02af4886/docs/bugfix/GAP_ANALYSIS_SUMMARY.md +0 -179
  406. package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/PRP.md +0 -629
  407. package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/validation-report.md +0 -214
  408. package/plan/001_d3bb02af4886/docs/bugfix/PRP_P1M4T2S3.md +0 -629
  409. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_PRP.md +0 -529
  410. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_QUICK_REFERENCE.md +0 -142
  411. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_README.md +0 -304
  412. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_TEST_RESULTS.md +0 -558
  413. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_VALIDATION_SUMMARY.md +0 -256
  414. package/plan/001_d3bb02af4886/docs/bugfix/system_context.md +0 -346
  415. package/plan/001_d3bb02af4886/docs/bugfix-architecture/bug_analysis.md +0 -415
  416. package/plan/001_d3bb02af4886/docs/bugfix-architecture/implementation_patterns.md +0 -489
  417. package/plan/001_d3bb02af4886/docs/bugfix-architecture/system_context.md +0 -218
  418. package/plan/001_d3bb02af4886/docs/bugfix_INITIATION_SUMMARY.md +0 -380
  419. package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_PATTERNS.md +0 -1923
  420. package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_QUICK_REF.md +0 -319
  421. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/codebase-context.md +0 -115
  422. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/cycle-detection-algorithms.md +0 -134
  423. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/test-patterns.md +0 -153
  424. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/workflow-class.md +0 -132
  425. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_BEST_PRACTICES.md +0 -716
  426. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_QUICK_REF.md +0 -186
  427. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/GROUNDSWELL_DECORATOR_EXAMPLES.md +0 -604
  428. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/INDEX.md +0 -213
  429. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/codebase_structure.md +0 -30
  430. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/existing_test_pattern.md +0 -56
  431. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/getRootObservers_implementation.md +0 -53
  432. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/test_conventions.md +0 -49
  433. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/PRP.md +0 -958
  434. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/QUICK_REFERENCE.md +0 -339
  435. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/README.md +0 -305
  436. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/SUMMARY.md +0 -433
  437. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/bidirectional-tree-consistency-testing.md +0 -1574
  438. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/test-pattern-examples.md +0 -1014
  439. package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_BEST_PRACTICES.md +0 -1929
  440. package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_CODE_PATTERNS.md +0 -857
  441. package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_INTEGRATION_GUIDE.md +0 -738
  442. package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_RESEARCH_INDEX.md +0 -424
  443. package/plan/001_d3bb02af4886/docs/research/P1P2/REFLECTION_INDEX.md +0 -291
  444. package/plan/001_d3bb02af4886/docs/research/P1P2/REFLECTION_RESEARCH_REPORT.md +0 -1342
  445. package/plan/001_d3bb02af4886/docs/research/P1P2/RESEARCH_SUMMARY.md +0 -342
  446. package/plan/001_d3bb02af4886/docs/research/P1P2/anthropic-sdk.md +0 -174
  447. package/plan/001_d3bb02af4886/docs/research/P1P2/async-local-storage.md +0 -200
  448. package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-code-patterns.md +0 -1205
  449. package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-decision-matrix.md +0 -421
  450. package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-implementation-guide.md +0 -1341
  451. package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-integration-guide.md +0 -834
  452. package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-patterns.md +0 -1468
  453. package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-quick-reference.md +0 -558
  454. package/plan/001_d3bb02af4886/docs/research/P1P2/zod-schema.md +0 -152
  455. package/plan/001_d3bb02af4886/docs/research/P3P4/caching-lru.md +0 -116
  456. package/plan/001_d3bb02af4886/docs/research/P3P4/introspection-tools.md +0 -177
  457. package/plan/001_d3bb02af4886/docs/research/P3P4/reflection-patterns.md +0 -117
  458. package/plan/001_d3bb02af4886/docs/research/P4P5/RESEARCH_SUMMARY.md +0 -151
  459. package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_QUICK_REF.md +0 -376
  460. package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_RESEARCH.md +0 -1507
  461. package/plan/001_d3bb02af4886/docs/research/bugfix_typescript_patterns.md +0 -949
  462. package/plan/001_d3bb02af4886/docs/research/error-testing-research.md +0 -619
  463. package/plan/001_d3bb02af4886/docs/research/error_handling_patterns.md +0 -723
  464. package/plan/001_d3bb02af4886/docs/research/general/INTROSPECTION_RESEARCH_SUMMARY.md +0 -378
  465. package/plan/001_d3bb02af4886/docs/research/general/README-INTROSPECTION.md +0 -352
  466. package/plan/001_d3bb02af4886/docs/research/general/agent-introspection-patterns.md +0 -1085
  467. package/plan/001_d3bb02af4886/docs/research/general/introspection-security-guide.md +0 -984
  468. package/plan/001_d3bb02af4886/docs/research/general/introspection-tool-examples.md +0 -875
  469. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/PRP_TEMPLATE.md +0 -460
  470. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/QUICK_REFERENCE.md +0 -324
  471. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/README.md +0 -175
  472. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/RESEARCH_REPORT.md +0 -499
  473. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/SUMMARY.md +0 -163
  474. package/plan/001_d3bb02af4886/prd_snapshot.md +0 -543
  475. package/plan/bugfix/BUG_FIX_SUMMARY.md +0 -961
  476. package/scripts/generate-llms-full.ts +0 -206
  477. package/src/__tests__/adversarial/attachChild-performance.test.ts +0 -216
  478. package/src/__tests__/adversarial/circular-reference.test.ts +0 -101
  479. package/src/__tests__/adversarial/complex-circular-reference.test.ts +0 -139
  480. package/src/__tests__/adversarial/concurrent-task-failures.test.ts +0 -571
  481. package/src/__tests__/adversarial/deep-analysis.test.ts +0 -729
  482. package/src/__tests__/adversarial/deep-hierarchy-stress.test.ts +0 -213
  483. package/src/__tests__/adversarial/e2e-prd-validation.test.ts +0 -448
  484. package/src/__tests__/adversarial/edge-case.test.ts +0 -703
  485. package/src/__tests__/adversarial/error-merge-strategy.test.ts +0 -760
  486. package/src/__tests__/adversarial/incremental-performance.test.ts +0 -140
  487. package/src/__tests__/adversarial/node-map-update-benchmarks.test.ts +0 -457
  488. package/src/__tests__/adversarial/observer-propagation.test.ts +0 -487
  489. package/src/__tests__/adversarial/parent-validation.test.ts +0 -143
  490. package/src/__tests__/adversarial/prd-12-2-compliance.test.ts +0 -611
  491. package/src/__tests__/adversarial/prd-compliance.test.ts +0 -731
  492. package/src/__tests__/compatibility/backward-compatibility.test.ts +0 -1572
  493. package/src/__tests__/helpers/index.ts +0 -18
  494. package/src/__tests__/helpers/tree-verification.ts +0 -257
  495. package/src/__tests__/integration/agent-workflow.test.ts +0 -256
  496. package/src/__tests__/integration/bidirectional-consistency.test.ts +0 -847
  497. package/src/__tests__/integration/observer-logging.test.ts +0 -643
  498. package/src/__tests__/integration/tree-mirroring.test.ts +0 -151
  499. package/src/__tests__/integration/workflow-reparenting.test.ts +0 -303
  500. package/src/__tests__/unit/agent.test.ts +0 -169
  501. package/src/__tests__/unit/cache-key.test.ts +0 -182
  502. package/src/__tests__/unit/cache.test.ts +0 -172
  503. package/src/__tests__/unit/context.test.ts +0 -217
  504. package/src/__tests__/unit/decorators.test.ts +0 -100
  505. package/src/__tests__/unit/introspection-tools.test.ts +0 -277
  506. package/src/__tests__/unit/logger.test.ts +0 -293
  507. package/src/__tests__/unit/observable.test.ts +0 -321
  508. package/src/__tests__/unit/prompt.test.ts +0 -135
  509. package/src/__tests__/unit/reflection.test.ts +0 -210
  510. package/src/__tests__/unit/tree-debugger-incremental.test.ts +0 -170
  511. package/src/__tests__/unit/tree-debugger.test.ts +0 -85
  512. package/src/__tests__/unit/utils/workflow-error-utils.test.ts +0 -209
  513. package/src/__tests__/unit/workflow-detachChild.test.ts +0 -100
  514. package/src/__tests__/unit/workflow-emitEvent-childDetached.test.ts +0 -153
  515. package/src/__tests__/unit/workflow-isDescendantOf.test.ts +0 -180
  516. package/src/__tests__/unit/workflow.test.ts +0 -357
  517. package/src/cache/cache-key.ts +0 -244
  518. package/src/cache/cache.ts +0 -236
  519. package/src/core/agent.ts +0 -593
  520. package/src/core/event-tree.ts +0 -260
  521. package/src/core/logger.ts +0 -112
  522. package/src/core/mcp-handler.ts +0 -184
  523. package/src/core/prompt.ts +0 -150
  524. package/src/core/workflow-context.ts +0 -351
  525. package/src/core/workflow.ts +0 -540
  526. package/src/debugger/tree-debugger.ts +0 -255
  527. package/src/decorators/observed-state.ts +0 -95
  528. package/src/decorators/step.ts +0 -139
  529. package/src/decorators/task.ts +0 -159
  530. package/src/examples/tdd-orchestrator.ts +0 -65
  531. package/src/examples/test-cycle-workflow.ts +0 -64
  532. package/src/index.ts +0 -142
  533. package/src/reflection/reflection.ts +0 -407
  534. package/src/tools/index.ts +0 -36
  535. package/src/tools/introspection.ts +0 -464
  536. package/src/types/agent.ts +0 -90
  537. package/src/types/decorators.ts +0 -32
  538. package/src/types/error-strategy.ts +0 -13
  539. package/src/types/error.ts +0 -20
  540. package/src/types/events.ts +0 -75
  541. package/src/types/index.ts +0 -55
  542. package/src/types/logging.ts +0 -24
  543. package/src/types/observer.ts +0 -18
  544. package/src/types/prompt.ts +0 -40
  545. package/src/types/reflection.ts +0 -117
  546. package/src/types/sdk-primitives.ts +0 -128
  547. package/src/types/workflow-context.ts +0 -163
  548. package/src/types/workflow.ts +0 -37
  549. package/src/utils/id.ts +0 -11
  550. package/src/utils/index.ts +0 -4
  551. package/src/utils/observable.ts +0 -106
  552. package/src/utils/workflow-error-utils.ts +0 -56
  553. package/tsconfig.json +0 -22
  554. package/vitest.config.ts +0 -16
@@ -1,153 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { Workflow, WorkflowObserver, WorkflowEvent } from '../../index.js';
3
-
4
- class SimpleWorkflow extends Workflow {
5
- async run(): Promise<string> {
6
- this.setStatus('running');
7
- this.setStatus('completed');
8
- return 'done';
9
- }
10
- }
11
-
12
- describe('emitEvent() - childDetached Tree Update Events', () => {
13
- it('should call onTreeChanged() when childDetached event is emitted', () => {
14
- // Arrange: Create parent with observer tracking callbacks
15
- const parent = new SimpleWorkflow('Parent');
16
- const events: WorkflowEvent[] = [];
17
- const treeChanges: any[] = [];
18
-
19
- const observer: WorkflowObserver = {
20
- onLog: () => {},
21
- onEvent: (e) => events.push(e),
22
- onStateUpdated: () => {},
23
- onTreeChanged: (root) => treeChanges.push(root),
24
- };
25
-
26
- parent.addObserver(observer);
27
-
28
- // Act: Detach child (which emits childDetached event)
29
- const child = new SimpleWorkflow('Child', parent);
30
- events.length = 0; // Clear attachChild events
31
- treeChanges.length = 0; // Clear attachChild tree changes
32
-
33
- parent.detachChild(child);
34
-
35
- // Assert: Verify childDetached event was emitted
36
- const detachEvent = events.find((e) => e.type === 'childDetached');
37
- expect(detachEvent).toBeDefined();
38
- expect(detachEvent?.type === 'childDetached' && detachEvent.parentId).toBe(parent.id);
39
-
40
- // CRITICAL ASSERTION: onTreeChanged() must be called
41
- expect(treeChanges.length).toBe(1);
42
- expect(treeChanges[0]).toBe(parent.getNode()); // Receives root node
43
- });
44
-
45
- it('should call onEvent() before onTreeChanged() for childDetached', () => {
46
- // Arrange: Track call order
47
- const parent = new SimpleWorkflow('Parent');
48
- const callOrder: string[] = [];
49
-
50
- const observer: WorkflowObserver = {
51
- onLog: () => {},
52
- onEvent: () => callOrder.push('onEvent'),
53
- onStateUpdated: () => {},
54
- onTreeChanged: () => callOrder.push('onTreeChanged'),
55
- };
56
-
57
- parent.addObserver(observer);
58
-
59
- // Act
60
- const child = new SimpleWorkflow('Child', parent);
61
- callOrder.length = 0; // Clear attachChild call order
62
- parent.detachChild(child);
63
-
64
- // Assert: Verify order
65
- expect(callOrder).toEqual(['onEvent', 'onTreeChanged']);
66
- });
67
-
68
- it('should include childDetached in tree update event types', () => {
69
- // This test verifies the implementation at the code level
70
- // Access the emitEvent method and verify it triggers onTreeChanged for childDetached
71
- const workflow = new SimpleWorkflow('Test');
72
-
73
- // Access private emitEvent method for testing
74
- const emitEvent = (workflow as any).emitEvent.bind(workflow);
75
-
76
- // Track if onTreeChanged was called
77
- let onTreeChangedCalled = false;
78
- const mockObserver = {
79
- onLog: () => {},
80
- onEvent: () => {},
81
- onStateUpdated: () => {},
82
- onTreeChanged: () => { onTreeChangedCalled = true; },
83
- };
84
-
85
- // Mock getRootObservers to return test observer
86
- (workflow as any).getRootObservers = () => [mockObserver];
87
-
88
- // Emit childDetached event
89
- emitEvent({
90
- type: 'childDetached',
91
- parentId: workflow.id,
92
- childId: 'test-child-id',
93
- });
94
-
95
- // Assert: onTreeChanged should be called
96
- expect(onTreeChangedCalled).toBe(true);
97
- });
98
-
99
- it('should pass correct root node to onTreeChanged() after detach', () => {
100
- // Arrange: Create multi-level tree
101
- const root = new SimpleWorkflow('Root');
102
- const parent = new SimpleWorkflow('Parent', root);
103
- const child = new SimpleWorkflow('Child', parent);
104
-
105
- const treeChangedRoots: any[] = [];
106
-
107
- const observer: WorkflowObserver = {
108
- onLog: () => {},
109
- onEvent: () => {},
110
- onStateUpdated: () => {},
111
- onTreeChanged: (rootNode) => treeChangedRoots.push(rootNode),
112
- };
113
-
114
- root.addObserver(observer);
115
-
116
- // Clear events from tree construction
117
- treeChangedRoots.length = 0;
118
-
119
- // Act: Detach child from parent
120
- parent.detachChild(child);
121
-
122
- // Assert: onTreeChanged should receive root node (from root workflow)
123
- expect(treeChangedRoots.length).toBe(1);
124
- expect(treeChangedRoots[0]).toBe(root.getNode());
125
- });
126
-
127
- it('should trigger onTreeChanged for both attach and detach operations', () => {
128
- // Arrange: Track tree changes
129
- const parent = new SimpleWorkflow('Parent');
130
- const treeChanges: any[] = [];
131
-
132
- const observer: WorkflowObserver = {
133
- onLog: () => {},
134
- onEvent: () => {},
135
- onStateUpdated: () => {},
136
- onTreeChanged: (root) => treeChanges.push({ operation: 'treeChanged', root }),
137
- };
138
-
139
- parent.addObserver(observer);
140
-
141
- // Act 1: Attach child
142
- const child = new SimpleWorkflow('Child', parent);
143
- const attachCount = treeChanges.length;
144
-
145
- // Act 2: Detach child
146
- treeChanges.length = 0; // Clear to only track detach
147
- parent.detachChild(child);
148
-
149
- // Assert: Both attach and detach should trigger onTreeChanged
150
- expect(treeChanges.length).toBe(1);
151
- expect(treeChanges[0].root).toBe(parent.getNode());
152
- });
153
- });
@@ -1,180 +0,0 @@
1
- import { describe, it, expect } from "vitest";
2
- import { Workflow } from "../../core/workflow.js";
3
-
4
- /**
5
- * Test suite for public isDescendantOf() API.
6
- *
7
- * These tests verify that the isDescendantOf() method is publicly accessible
8
- * without requiring (as any) type casting, following the P1.M3.T4.S2 PRP
9
- * implementation for making this method public.
10
- */
11
- describe("Workflow.isDescendantOf() - Public API", () => {
12
- describe("Public API Accessibility", () => {
13
- it("should be publicly accessible without casting to any", () => {
14
- const root = new Workflow("root");
15
- const child = new Workflow("child", root);
16
-
17
- // This should work without (as any) cast - verifies public visibility
18
- const result = child.isDescendantOf(root);
19
-
20
- expect(result).toBe(true);
21
- });
22
-
23
- it("should return true for direct parent relationship", () => {
24
- const root = new Workflow("root");
25
- const child = new Workflow("child", root);
26
-
27
- const result = child.isDescendantOf(root);
28
-
29
- expect(result).toBe(true);
30
- });
31
-
32
- it("should return true for nested descendant relationship", () => {
33
- const root = new Workflow("root");
34
- const level1 = new Workflow("level1", root);
35
- const level2 = new Workflow("level2", level1);
36
- const level3 = new Workflow("level3", level2);
37
-
38
- expect(level1.isDescendantOf(root)).toBe(true);
39
- expect(level2.isDescendantOf(root)).toBe(true);
40
- expect(level3.isDescendantOf(root)).toBe(true);
41
- expect(level3.isDescendantOf(level1)).toBe(true);
42
- });
43
-
44
- it("should return false for unrelated workflows", () => {
45
- const tree1 = new Workflow("tree1");
46
- const tree2 = new Workflow("tree2");
47
-
48
- const result = tree1.isDescendantOf(tree2);
49
-
50
- expect(result).toBe(false);
51
- });
52
-
53
- it("should return false when checking root against descendant", () => {
54
- const root = new Workflow("root");
55
- const child = new Workflow("child", root);
56
-
57
- const result = root.isDescendantOf(child);
58
-
59
- expect(result).toBe(false);
60
- });
61
-
62
- it("should return false for root checking itself", () => {
63
- const root = new Workflow("root");
64
-
65
- const result = root.isDescendantOf(root);
66
-
67
- expect(result).toBe(false);
68
- });
69
- });
70
-
71
- describe("Use Case: Hierarchy Validation", () => {
72
- it("should support validating workflow belongs to specific hierarchy", () => {
73
- const productionRoot = new Workflow("production-root");
74
- const stagingRoot = new Workflow("staging-root");
75
-
76
- const prodWorkflow = new Workflow("prod-workflow", productionRoot);
77
- const stagingWorkflow = new Workflow("staging-workflow", stagingRoot);
78
-
79
- expect(prodWorkflow.isDescendantOf(productionRoot)).toBe(true);
80
- expect(prodWorkflow.isDescendantOf(stagingRoot)).toBe(false);
81
- expect(stagingWorkflow.isDescendantOf(stagingRoot)).toBe(true);
82
- expect(stagingWorkflow.isDescendantOf(productionRoot)).toBe(false);
83
- });
84
- });
85
-
86
- describe("Use Case: Circular Reference Prevention", () => {
87
- it("should support validation before attaching to prevent circular references", () => {
88
- const root = new Workflow("root");
89
- const child = new Workflow("child", root);
90
- const grandchild = new Workflow("grandchild", child);
91
-
92
- // grandchild should NOT be attachable to root (would create cycle)
93
- expect(grandchild.isDescendantOf(root)).toBe(true);
94
-
95
- // Unrelated workflow should be attachable
96
- const unrelated = new Workflow("unrelated");
97
- expect(unrelated.isDescendantOf(root)).toBe(false);
98
- });
99
- });
100
-
101
- describe("Use Case: Conditional Logic Based on Ancestry", () => {
102
- it("should support conditional execution based on hierarchy position", () => {
103
- const productionRoot = new Workflow("production-root");
104
- const developmentRoot = new Workflow("development-root");
105
-
106
- const prodBatch = new Workflow("prod-batch", productionRoot);
107
- const devBatch = new Workflow("dev-batch", developmentRoot);
108
-
109
- // Simulate conditional logic
110
- const isInProduction = prodBatch.isDescendantOf(productionRoot);
111
- const isProductionWorkflow = devBatch.isDescendantOf(productionRoot);
112
-
113
- expect(isInProduction).toBe(true);
114
- expect(isProductionWorkflow).toBe(false);
115
- });
116
- });
117
-
118
- describe("Edge Cases", () => {
119
- it("should handle workflows with no parent", () => {
120
- const orphan = new Workflow("orphan");
121
- const root = new Workflow("root");
122
-
123
- const result = orphan.isDescendantOf(root);
124
-
125
- expect(result).toBe(false);
126
- });
127
-
128
- it("should handle sibling relationships correctly", () => {
129
- const root = new Workflow("root");
130
- const sibling1 = new Workflow("sibling1", root);
131
- const sibling2 = new Workflow("sibling2", root);
132
-
133
- // Siblings are not descendants of each other
134
- expect(sibling1.isDescendantOf(sibling2)).toBe(false);
135
- expect(sibling2.isDescendantOf(sibling1)).toBe(false);
136
- });
137
-
138
- it("should handle complex multi-level hierarchies", () => {
139
- const root = new Workflow("root");
140
- const branch1 = new Workflow("branch1", root);
141
- const branch2 = new Workflow("branch2", root);
142
- const leaf1a = new Workflow("leaf1a", branch1);
143
- const leaf1b = new Workflow("leaf1b", branch1);
144
- const leaf2a = new Workflow("leaf2a", branch2);
145
-
146
- // All leafs should be descendants of root
147
- expect(leaf1a.isDescendantOf(root)).toBe(true);
148
- expect(leaf1b.isDescendantOf(root)).toBe(true);
149
- expect(leaf2a.isDescendantOf(root)).toBe(true);
150
-
151
- // Leaf1a should not be descendant of branch2
152
- expect(leaf1a.isDescendantOf(branch2)).toBe(false);
153
-
154
- // Leaf2a should not be descendant of branch1
155
- expect(leaf2a.isDescendantOf(branch1)).toBe(false);
156
- });
157
- });
158
-
159
- describe("Edge Cases: Circular Reference Detection", () => {
160
- it("should throw error when circular reference is detected", () => {
161
- // Arrange: Create parent and child workflows with circular reference
162
- const parent = new Workflow("Parent");
163
- const child = new Workflow("Child", parent);
164
- const unrelated = new Workflow("Unrelated");
165
-
166
- // Act: Create circular reference manually
167
- // This simulates a bug or malicious input that creates a cycle
168
- // Normal attachChild() prevents this, so we bypass normal safeguards
169
- parent.parent = child;
170
-
171
- // Assert: isDescendantOf should throw error for circular reference
172
- // We use 'unrelated' as the ancestor to force traversal through the cycle
173
- // If we used 'child', the method would return true immediately before
174
- // detecting the cycle (since child === child at first iteration)
175
- expect(() => parent.isDescendantOf(unrelated)).toThrow(
176
- "Circular parent-child relationship detected",
177
- );
178
- });
179
- });
180
- });
@@ -1,357 +0,0 @@
1
- import { describe, it, expect } from 'vitest';
2
- import { Workflow, WorkflowObserver, WorkflowNode, LogEntry, WorkflowEvent, ObservedState, getObservedState } from '../../index.js';
3
-
4
- class SimpleWorkflow extends Workflow {
5
- async run(): Promise<string> {
6
- this.setStatus('running');
7
- this.logger.info('Running simple workflow');
8
- this.setStatus('completed');
9
- return 'done';
10
- }
11
- }
12
-
13
- describe('Workflow Name Validation', () => {
14
- it('should reject empty string name', () => {
15
- expect(() => new SimpleWorkflow('')).toThrow('Workflow name cannot be empty or whitespace only');
16
- });
17
-
18
- it('should reject whitespace-only name (spaces)', () => {
19
- expect(() => new SimpleWorkflow(' ')).toThrow('Workflow name cannot be empty or whitespace only');
20
- });
21
-
22
- it('should reject whitespace-only name (tabs)', () => {
23
- expect(() => new SimpleWorkflow('\t\t')).toThrow('Workflow name cannot be empty or whitespace only');
24
- });
25
-
26
- it('should reject whitespace-only name (newlines)', () => {
27
- expect(() => new SimpleWorkflow('\n\n')).toThrow('Workflow name cannot be empty or whitespace only');
28
- });
29
-
30
- it('should reject whitespace-only name (mixed whitespace)', () => {
31
- expect(() => new SimpleWorkflow(' \t\n ')).toThrow('Workflow name cannot be empty or whitespace only');
32
- });
33
-
34
- it('should reject name exceeding 100 characters', () => {
35
- const longName = 'a'.repeat(101);
36
- expect(() => new SimpleWorkflow(longName)).toThrow('Workflow name cannot exceed 100 characters');
37
- });
38
-
39
- it('should accept name with exactly 100 characters', () => {
40
- const exactly100 = 'a'.repeat(100);
41
- const wf = new SimpleWorkflow(exactly100);
42
- expect(wf.getNode().name).toBe(exactly100);
43
- expect(wf.getNode().name.length).toBe(100);
44
- });
45
-
46
- it('should accept valid names with leading/trailing whitespace', () => {
47
- const wf1 = new SimpleWorkflow(' MyWorkflow ');
48
- expect(wf1.getNode().name).toBe(' MyWorkflow ');
49
-
50
- const wf2 = new SimpleWorkflow('\tValidName\t');
51
- expect(wf2.getNode().name).toBe('\tValidName\t');
52
- });
53
-
54
- it('should use class name when name is undefined', () => {
55
- const wf = new SimpleWorkflow();
56
- expect(wf.getNode().name).toBe('SimpleWorkflow');
57
- });
58
-
59
- it('should use class name when name is null', () => {
60
- const wf = new SimpleWorkflow(null as any);
61
- expect(wf.getNode().name).toBe('SimpleWorkflow');
62
- });
63
-
64
- it('should validate both constructor patterns - class-based with empty name', () => {
65
- expect(() => new SimpleWorkflow('')).toThrow('Workflow name cannot be empty or whitespace only');
66
- });
67
-
68
- it('should validate both constructor patterns - functional with empty name', () => {
69
- expect(() => new Workflow({ name: '' }, async () => {})).toThrow('Workflow name cannot be empty or whitespace only');
70
- });
71
-
72
- it('should validate both constructor patterns - functional with whitespace name', () => {
73
- expect(() => new Workflow({ name: ' ' }, async () => {})).toThrow('Workflow name cannot be empty or whitespace only');
74
- });
75
-
76
- it('should validate both constructor patterns - functional with name exceeding 100 characters', () => {
77
- const longName = 'a'.repeat(101);
78
- expect(() => new Workflow({ name: longName }, async () => {})).toThrow('Workflow name cannot exceed 100 characters');
79
- });
80
-
81
- it('should accept valid name in functional pattern', () => {
82
- const wf = new Workflow({ name: 'ValidFunctionalWorkflow' }, async () => 'done');
83
- expect(wf.getNode().name).toBe('ValidFunctionalWorkflow');
84
- });
85
- });
86
-
87
- describe('Workflow', () => {
88
- it('should create with unique id', () => {
89
- const wf1 = new SimpleWorkflow();
90
- const wf2 = new SimpleWorkflow();
91
- expect(wf1.id).not.toBe(wf2.id);
92
- });
93
-
94
- it('should use class name as default name', () => {
95
- const wf = new SimpleWorkflow();
96
- expect(wf.getNode().name).toBe('SimpleWorkflow');
97
- });
98
-
99
- it('should use custom name when provided', () => {
100
- const wf = new SimpleWorkflow('CustomName');
101
- expect(wf.getNode().name).toBe('CustomName');
102
- });
103
-
104
- it('should start with idle status', () => {
105
- const wf = new SimpleWorkflow();
106
- expect(wf.status).toBe('idle');
107
- expect(wf.getNode().status).toBe('idle');
108
- });
109
-
110
- it('should attach child to parent', () => {
111
- const parent = new SimpleWorkflow('Parent');
112
- const child = new SimpleWorkflow('Child', parent);
113
-
114
- expect(child.parent).toBe(parent);
115
- expect(parent.children).toContain(child);
116
- expect(parent.getNode().children).toContain(child.getNode());
117
- });
118
-
119
- it('should emit logs to observers', async () => {
120
- const wf = new SimpleWorkflow();
121
- const logs: LogEntry[] = [];
122
-
123
- const observer: WorkflowObserver = {
124
- onLog: (entry) => logs.push(entry),
125
- onEvent: () => {},
126
- onStateUpdated: () => {},
127
- onTreeChanged: () => {},
128
- };
129
-
130
- wf.addObserver(observer);
131
- await wf.run();
132
-
133
- expect(logs.length).toBeGreaterThan(0);
134
- expect(logs[0].message).toBe('Running simple workflow');
135
- });
136
-
137
- it('should emit childAttached event', () => {
138
- const parent = new SimpleWorkflow('Parent');
139
- const events: WorkflowEvent[] = [];
140
-
141
- const observer: WorkflowObserver = {
142
- onLog: () => {},
143
- onEvent: (event) => events.push(event),
144
- onStateUpdated: () => {},
145
- onTreeChanged: () => {},
146
- };
147
-
148
- parent.addObserver(observer);
149
- const child = new SimpleWorkflow('Child', parent);
150
-
151
- const attachEvent = events.find((e) => e.type === 'childAttached');
152
- expect(attachEvent).toBeDefined();
153
- expect(attachEvent?.type === 'childAttached' && attachEvent.parentId).toBe(parent.id);
154
- });
155
-
156
- it('should capture state and logs in functional workflow error', async () => {
157
- // Arrange: Create observer to capture events
158
- const events: WorkflowEvent[] = [];
159
-
160
- const observer: WorkflowObserver = {
161
- onLog: () => {},
162
- onEvent: (event) => events.push(event),
163
- onStateUpdated: () => {},
164
- onTreeChanged: () => {},
165
- };
166
-
167
- // Arrange: Create functional workflow with a step that throws an error
168
- const workflow = new Workflow<void>(
169
- { name: 'ErrorCaptureTest' },
170
- async (ctx) => {
171
- // Execute a step that will fail
172
- await ctx.step('failing-step', async () => {
173
- throw new Error('Test error from step');
174
- });
175
- }
176
- );
177
-
178
- // Act: Attach observer and run workflow
179
- workflow.addObserver(observer);
180
- await expect(workflow.run()).rejects.toThrow('Test error from step');
181
-
182
- // Assert: Verify error event was emitted
183
- const errorEvents = events.filter((e) => e.type === 'error');
184
- expect(errorEvents.length).toBeGreaterThanOrEqual(1);
185
-
186
- // Assert: Verify error structure
187
- const errorEvent = errorEvents[0];
188
- expect(errorEvent.error).toBeDefined();
189
- expect(errorEvent.error.message).toBe('Test error from step');
190
-
191
- // Assert: Verify logs array was captured (even if empty for step errors)
192
- expect(errorEvent.error.logs).toBeDefined();
193
- expect(Array.isArray(errorEvent.error.logs)).toBe(true);
194
-
195
- // Assert: Verify state was captured (may be empty object for pure functional workflows)
196
- expect(errorEvent.error.state).toBeDefined();
197
- expect(typeof errorEvent.error.state).toBe('object');
198
-
199
- // Assert: Verify workflow status
200
- expect(workflow.status).toBe('failed');
201
-
202
- // Assert: Verify workflowId is captured
203
- expect(errorEvent.error.workflowId).toBe(workflow.id);
204
- });
205
-
206
- it('should capture @ObservedState fields in workflow error state', async () => {
207
- // Test workflow class with @ObservedState decorated fields
208
- // Using functional pattern (executor) so error events are emitted via runFunctional()
209
- class StatefulWorkflowClass extends Workflow {
210
- @ObservedState()
211
- stepCount: number = 0;
212
-
213
- @ObservedState({ redact: true })
214
- apiKey: string = 'secret-key-123';
215
-
216
- @ObservedState({ hidden: true })
217
- internalCounter: number = 42;
218
- }
219
-
220
- // Arrange: Create observer to capture error events
221
- const events: WorkflowEvent[] = [];
222
-
223
- const observer: WorkflowObserver = {
224
- onLog: () => {},
225
- onEvent: (event) => events.push(event),
226
- onStateUpdated: () => {},
227
- onTreeChanged: () => {},
228
- };
229
-
230
- // Arrange: Create workflow with @ObservedState fields using functional pattern
231
- const workflow = new StatefulWorkflowClass(
232
- { name: 'StatefulErrorTest' },
233
- async (ctx) => {
234
- // Modify @ObservedState fields on the workflow instance
235
- (workflow as any).stepCount = 5;
236
- (workflow as any).apiKey = 'updated-key';
237
- (workflow as any).internalCounter = 99;
238
-
239
- // Execute a step that will fail
240
- await ctx.step('failing-step', async () => {
241
- throw new Error('Error after state update');
242
- });
243
- }
244
- );
245
-
246
- // Act: Attach observer and trigger error
247
- workflow.addObserver(observer);
248
- await expect(workflow.run()).rejects.toThrow('Error after state update');
249
-
250
- // Assert: Verify error event was emitted
251
- const errorEvents = events.filter((e) => e.type === 'error');
252
- expect(errorEvents.length).toBeGreaterThanOrEqual(1);
253
-
254
- // Assert: Verify error structure
255
- const errorEvent = errorEvents[0];
256
- expect(errorEvent.error).toBeDefined();
257
- expect(errorEvent.error.message).toBe('Error after state update');
258
-
259
- // Assert: Verify @ObservedState fields were captured
260
- expect(errorEvent.error.state).toBeDefined();
261
- expect(typeof errorEvent.error.state).toBe('object');
262
-
263
- // Assert: Verify public field value is captured
264
- expect(errorEvent.error.state.stepCount).toBe(5);
265
-
266
- // Assert: Verify redacted field shows '***'
267
- expect(errorEvent.error.state.apiKey).toBe('***');
268
-
269
- // Assert: Verify hidden field is NOT in state
270
- expect('internalCounter' in errorEvent.error.state).toBe(false);
271
-
272
- // Assert: Verify logs array is present (may be empty)
273
- expect(errorEvent.error.logs).toBeDefined();
274
- expect(Array.isArray(errorEvent.error.logs)).toBe(true);
275
-
276
- // Assert: Verify workflow status
277
- expect(workflow.status).toBe('failed');
278
-
279
- // Assert: Verify workflowId is captured
280
- expect(errorEvent.error.workflowId).toBe(workflow.id);
281
- });
282
-
283
- it('should detect circular parent relationship', () => {
284
- // Arrange: Create parent and child workflows
285
- const parent = new SimpleWorkflow('Parent');
286
- const child = new SimpleWorkflow('Child', parent);
287
-
288
- // Act: Create circular reference manually
289
- // This simulates a bug or malicious input that creates a cycle
290
- parent.parent = child;
291
-
292
- // Assert: getRoot() should throw error for circular reference
293
- // Note: getRoot() is protected, so we cast to any to access it
294
- expect(() => (parent as any).getRoot()).toThrow(
295
- 'Circular parent-child relationship detected'
296
- );
297
- });
298
-
299
- it('should detect circular relationship in getRootObservers', () => {
300
- // Arrange: Create parent and child workflows
301
- const parent = new SimpleWorkflow('Parent');
302
- const child = new SimpleWorkflow('Child', parent);
303
-
304
- // Act: Create circular reference manually
305
- // This simulates a bug or malicious input that creates a cycle
306
- parent.parent = child;
307
-
308
- // Assert: getRootObservers() should throw error for circular reference
309
- // Note: getRootObservers() is private, so we cast to any to access it
310
- expect(() => (parent as any).getRootObservers()).toThrow(
311
- 'Circular parent-child relationship detected'
312
- );
313
- });
314
-
315
- it('should throw error when duplicate attachment attempted', () => {
316
- // Arrange: Create parent and child workflows with first attachment
317
- const parent = new SimpleWorkflow('Parent');
318
- const child = new SimpleWorkflow('Child', parent);
319
-
320
- // Act & Assert: Second attachment attempt should throw error
321
- expect(() => parent.attachChild(child)).toThrow(
322
- 'Child already attached to this workflow'
323
- );
324
- });
325
-
326
- it('should emit treeUpdated event when status changes', () => {
327
- // Arrange: Create workflow instance
328
- const wf = new SimpleWorkflow();
329
-
330
- // Arrange: Create arrays to track event emissions and callback invocations
331
- const events: WorkflowEvent[] = [];
332
- const treeChangedCalls: WorkflowNode[] = [];
333
-
334
- // Arrange: Create observer with callbacks
335
- const observer: WorkflowObserver = {
336
- onLog: () => {}, // Empty - not testing logs
337
- onEvent: (event) => events.push(event), // Capture events
338
- onStateUpdated: () => {}, // Empty - not testing state updates
339
- onTreeChanged: (root) => treeChangedCalls.push(root), // Capture tree changes
340
- };
341
-
342
- // Act: Attach observer and trigger status change
343
- wf.addObserver(observer);
344
- wf.setStatus('running');
345
-
346
- // Assert: Verify treeUpdated event was emitted
347
- const treeUpdatedEvent = events.find((e) => e.type === 'treeUpdated');
348
- expect(treeUpdatedEvent).toBeDefined();
349
-
350
- // Assert: Verify event payload contains root node (type narrowing for discriminated union)
351
- expect(treeUpdatedEvent?.type === 'treeUpdated' && treeUpdatedEvent.root).toBe(wf.getNode());
352
-
353
- // Assert: Verify onTreeChanged callback was invoked with root node
354
- expect(treeChangedCalls).toHaveLength(1);
355
- expect(treeChangedCalls[0]).toBe(wf.getNode());
356
- });
357
- });