groundswell 0.0.1 → 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (242) hide show
  1. package/.claude/commands/subtask-planning/prp-base-create.md +120 -0
  2. package/.claude/commands/subtask-planning/prp-base-execute.md +65 -0
  3. package/.claude/commands/task-breakdown.md +94 -0
  4. package/.claude/system_prompts/task-breakdown.md +1 -0
  5. package/CHANGELOG.md +188 -0
  6. package/PRD.md +543 -0
  7. package/README.md +99 -5
  8. package/examples/README.md +15 -1
  9. package/examples/examples/11-reparenting-workflows.ts +269 -0
  10. package/examples/index.ts +4 -0
  11. package/package-lock.json +2398 -0
  12. package/package.json +3 -1
  13. package/plan/001_d3bb02af4886/TEST_RESULTS.md +259 -0
  14. package/plan/001_d3bb02af4886/bug_fix_tasks.json +484 -0
  15. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S1/PRP.md +488 -0
  16. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S2/PRP.md +581 -0
  17. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S3/PRP.md +687 -0
  18. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S1/PRP.md +492 -0
  19. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/PRP.md +932 -0
  20. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/concurrent_error_testing_patterns.md +1109 -0
  21. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/vitest_concurrent_testing.md +802 -0
  22. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/workflow_engine_test_references.md +603 -0
  23. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S1/PRP.md +564 -0
  24. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S3/PRP.md +518 -0
  25. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S4/PRP.md +1252 -0
  26. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/PRP.md +364 -0
  27. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/CODEBASE_INVENTORY.md +114 -0
  28. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/DECORATOR_DOCUMENTATION_PATTERNS.md +205 -0
  29. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/PRD_LOCATION_ANALYSIS.md +199 -0
  30. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/ULTRATHINK_PRP_PLAN.md +134 -0
  31. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/PRP.md +495 -0
  32. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/research/console_error_inventory.md +435 -0
  33. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S2/PRP.md +506 -0
  34. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S3/PRP.md +612 -0
  35. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/PRP.md +558 -0
  36. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/research/external_research.md +788 -0
  37. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S2/PRP.md +460 -0
  38. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S3/PRP.md +454 -0
  39. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/PRP.md +520 -0
  40. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/RECOMMENDATION.md +417 -0
  41. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/external_workflow_engines_research.md +760 -0
  42. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/security_implications_analysis.md +245 -0
  43. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S2/PRP.md +792 -0
  44. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/PRP.md +535 -0
  45. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/TEST_EXECUTION_REPORT.md +190 -0
  46. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/PRP.md +654 -0
  47. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/TEST_FIX_REPORT.md +227 -0
  48. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/KEY_FINDINGS.md +345 -0
  49. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/QUICK_REFERENCE.md +193 -0
  50. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/test_maintenance_research.md +1323 -0
  51. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/BREAKING_CHANGES_AUDIT.md +1011 -0
  52. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/PRP.md +927 -0
  53. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S2/PRP.md +505 -0
  54. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/architecture/logger_child_signature_analysis.md +401 -0
  55. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/child_implementation_research.md +142 -0
  56. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/test_patterns_research.md +112 -0
  57. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/vitest_patterns_research.md +159 -0
  58. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/PRP.md +549 -0
  59. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/VERIFICATION_REPORT.md +368 -0
  60. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/edge_case_analysis.md +172 -0
  61. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/usage_inventory.md +175 -0
  62. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S2/PRP.md +696 -0
  63. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S4/PRP.md +860 -0
  64. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/PRP.md +1066 -0
  65. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01-testing-aggregated-errors.md +1103 -0
  66. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01_typescript_error_aggregation_patterns.md +789 -0
  67. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02-error-merge-strategy-testing-guide.md +1098 -0
  68. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02_aggregate_error_patterns.md +1037 -0
  69. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03-promise-allsettled-testing-patterns.md +916 -0
  70. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03_error_merging_strategies.md +1045 -0
  71. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/04_github_stackoverflow_examples.md +890 -0
  72. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/05_comprehensive_summary.md +822 -0
  73. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/INDEX.md +668 -0
  74. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/QUICK_REFERENCE.md +706 -0
  75. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/README.md +265 -0
  76. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/RESEARCH_REPORT.md +655 -0
  77. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S4/research/vitest_testing_patterns.md +1103 -0
  78. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T3S2/PRP.md +426 -0
  79. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/PRP.md +506 -0
  80. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/QUICK_REFERENCE.md +114 -0
  81. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/RESEARCH_SUMMARY.md +316 -0
  82. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/vitest_observer_error_logging_best_practices.md +754 -0
  83. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S3/PRP.md +612 -0
  84. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/PRP.md +719 -0
  85. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/README.md +215 -0
  86. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/analysis.md +765 -0
  87. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S3/PRP.md +718 -0
  88. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/DECISION.md +149 -0
  89. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/PRP.md +470 -0
  90. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/ULTRATHINK_PLAN.md +332 -0
  91. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/codebase_workflow_name_analysis.md +167 -0
  92. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/external_best_practices.md +265 -0
  93. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/validation_patterns.md +273 -0
  94. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S1/workflow_engine_ancestry_api_research.md +760 -0
  95. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S3-PRP.md +434 -0
  96. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S1/PRP.md +717 -0
  97. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/PRP.md +472 -0
  98. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/VALIDATION_REPORT.md +125 -0
  99. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/research/ULTRATHINK_PRP_PLAN.md +301 -0
  100. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/error-logging-best-practices.md +1170 -0
  101. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/research_typescript_partial_and_overloads.md +940 -0
  102. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-quick-reference.md +151 -0
  103. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-research.md +650 -0
  104. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/prd_snapshot.md +259 -0
  105. package/plan/001_d3bb02af4886/bugfix/P1M1T1S1/PRP.md +457 -0
  106. package/plan/001_d3bb02af4886/bugfix/RESEARCH_SUMMARY.md +346 -0
  107. package/plan/001_d3bb02af4886/bugfix/architecture/codebase_structure.md +311 -0
  108. package/plan/001_d3bb02af4886/bugfix/architecture/concurrent_execution_best_practices.md +1565 -0
  109. package/plan/001_d3bb02af4886/bugfix/architecture/error_handling_patterns.md +288 -0
  110. package/plan/001_d3bb02af4886/bugfix/architecture/promise_all_analysis.md +741 -0
  111. package/plan/001_d3bb02af4886/docs/PRP/P1M1T1S4-functional-workflow-error-state-capture-test.md +652 -0
  112. package/plan/001_d3bb02af4886/docs/PRP/PRP.md +527 -0
  113. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S1-PRP.md +415 -0
  114. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S2-PRP.md +378 -0
  115. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S4-PRP.md +713 -0
  116. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M2T1S4-PRP.md +370 -0
  117. package/plan/001_d3bb02af4886/docs/PRP_P1M3T1S3.md +499 -0
  118. package/plan/001_d3bb02af4886/docs/TEST_RESULTS.md +230 -0
  119. package/plan/001_d3bb02af4886/docs/bugfix/ANALYSIS_PRD_VS_IMPLEMENTATION.md +1134 -0
  120. package/plan/001_d3bb02af4886/docs/bugfix/GAP_ANALYSIS_SUMMARY.md +179 -0
  121. package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/PRP.md +629 -0
  122. package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/validation-report.md +214 -0
  123. package/plan/001_d3bb02af4886/docs/bugfix/PRP_P1M4T2S3.md +629 -0
  124. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_PRP.md +529 -0
  125. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_QUICK_REFERENCE.md +142 -0
  126. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_README.md +304 -0
  127. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_TEST_RESULTS.md +558 -0
  128. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_VALIDATION_SUMMARY.md +256 -0
  129. package/plan/001_d3bb02af4886/docs/bugfix/system_context.md +346 -0
  130. package/plan/001_d3bb02af4886/docs/bugfix-architecture/bug_analysis.md +415 -0
  131. package/plan/001_d3bb02af4886/docs/bugfix-architecture/implementation_patterns.md +489 -0
  132. package/plan/001_d3bb02af4886/docs/bugfix-architecture/system_context.md +218 -0
  133. package/plan/001_d3bb02af4886/docs/bugfix_INITIATION_SUMMARY.md +380 -0
  134. package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_PATTERNS.md +1923 -0
  135. package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_QUICK_REF.md +319 -0
  136. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/codebase-context.md +115 -0
  137. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/cycle-detection-algorithms.md +134 -0
  138. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/test-patterns.md +153 -0
  139. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/workflow-class.md +132 -0
  140. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_BEST_PRACTICES.md +716 -0
  141. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_QUICK_REF.md +186 -0
  142. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/GROUNDSWELL_DECORATOR_EXAMPLES.md +604 -0
  143. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/INDEX.md +213 -0
  144. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/codebase_structure.md +30 -0
  145. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/existing_test_pattern.md +56 -0
  146. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/getRootObservers_implementation.md +53 -0
  147. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/test_conventions.md +49 -0
  148. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/PRP.md +958 -0
  149. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/QUICK_REFERENCE.md +339 -0
  150. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/README.md +305 -0
  151. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/SUMMARY.md +433 -0
  152. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/bidirectional-tree-consistency-testing.md +1574 -0
  153. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/test-pattern-examples.md +1014 -0
  154. package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_QUICK_REF.md +376 -0
  155. package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_RESEARCH.md +1507 -0
  156. package/plan/001_d3bb02af4886/docs/research/bugfix_typescript_patterns.md +949 -0
  157. package/plan/001_d3bb02af4886/docs/research/error-testing-research.md +619 -0
  158. package/plan/001_d3bb02af4886/docs/research/error_handling_patterns.md +723 -0
  159. package/plan/{research → 001_d3bb02af4886/docs/research/general}/introspection-security-guide.md +56 -0
  160. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/PRP_TEMPLATE.md +460 -0
  161. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/QUICK_REFERENCE.md +324 -0
  162. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/README.md +175 -0
  163. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/RESEARCH_REPORT.md +499 -0
  164. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/SUMMARY.md +163 -0
  165. package/plan/bugfix/BUG_FIX_SUMMARY.md +961 -0
  166. package/src/__tests__/adversarial/attachChild-performance.test.ts +216 -0
  167. package/src/__tests__/adversarial/circular-reference.test.ts +101 -0
  168. package/src/__tests__/adversarial/complex-circular-reference.test.ts +139 -0
  169. package/src/__tests__/adversarial/concurrent-task-failures.test.ts +571 -0
  170. package/src/__tests__/adversarial/deep-analysis.test.ts +729 -0
  171. package/src/__tests__/adversarial/deep-hierarchy-stress.test.ts +213 -0
  172. package/src/__tests__/adversarial/e2e-prd-validation.test.ts +448 -0
  173. package/src/__tests__/adversarial/edge-case.test.ts +703 -0
  174. package/src/__tests__/adversarial/error-merge-strategy.test.ts +760 -0
  175. package/src/__tests__/adversarial/incremental-performance.test.ts +140 -0
  176. package/src/__tests__/adversarial/node-map-update-benchmarks.test.ts +457 -0
  177. package/src/__tests__/adversarial/observer-propagation.test.ts +487 -0
  178. package/src/__tests__/adversarial/parent-validation.test.ts +143 -0
  179. package/src/__tests__/adversarial/prd-12-2-compliance.test.ts +611 -0
  180. package/src/__tests__/adversarial/prd-compliance.test.ts +731 -0
  181. package/src/__tests__/compatibility/backward-compatibility.test.ts +1572 -0
  182. package/src/__tests__/helpers/index.ts +18 -0
  183. package/src/__tests__/helpers/tree-verification.ts +257 -0
  184. package/src/__tests__/integration/bidirectional-consistency.test.ts +847 -0
  185. package/src/__tests__/integration/observer-logging.test.ts +643 -0
  186. package/src/__tests__/integration/tree-mirroring.test.ts +37 -0
  187. package/src/__tests__/integration/workflow-reparenting.test.ts +303 -0
  188. package/src/__tests__/unit/context.test.ts +79 -0
  189. package/src/__tests__/unit/logger.test.ts +293 -0
  190. package/src/__tests__/unit/observable.test.ts +321 -0
  191. package/src/__tests__/unit/tree-debugger-incremental.test.ts +170 -0
  192. package/src/__tests__/unit/utils/workflow-error-utils.test.ts +209 -0
  193. package/src/__tests__/unit/workflow-detachChild.test.ts +100 -0
  194. package/src/__tests__/unit/workflow-emitEvent-childDetached.test.ts +153 -0
  195. package/src/__tests__/unit/workflow-isDescendantOf.test.ts +180 -0
  196. package/src/__tests__/unit/workflow.test.ts +277 -1
  197. package/src/core/agent.ts +21 -1
  198. package/src/core/logger.ts +27 -2
  199. package/src/core/workflow-context.ts +6 -4
  200. package/src/core/workflow.ts +252 -14
  201. package/src/debugger/tree-debugger.ts +52 -7
  202. package/src/decorators/task.ts +65 -2
  203. package/src/index.ts +4 -2
  204. package/src/types/decorators.ts +8 -1
  205. package/src/types/events.ts +1 -0
  206. package/src/utils/index.ts +1 -0
  207. package/src/utils/observable.ts +32 -3
  208. package/src/utils/workflow-error-utils.ts +56 -0
  209. package/tsconfig.json +1 -1
  210. package/llms_full.txt +0 -5890
  211. package/tasks.json +0 -0
  212. /package/plan/{backlog.json → 001_d3bb02af4886/backlog.json} +0 -0
  213. /package/plan/{P1P2/PRP.md → 001_d3bb02af4886/docs/PRP/P1P2-PRP.md} +0 -0
  214. /package/plan/{P3P4/PRP.md → 001_d3bb02af4886/docs/PRP/P3P4-PRP.md} +0 -0
  215. /package/plan/{P4P5/PRP.md → 001_d3bb02af4886/docs/PRP/P4P5-PRP.md} +0 -0
  216. /package/plan/{architecture → 001_d3bb02af4886/docs/architecture}/external_deps.md +0 -0
  217. /package/plan/{architecture → 001_d3bb02af4886/docs/architecture}/system_context.md +0 -0
  218. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_BEST_PRACTICES.md +0 -0
  219. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_CODE_PATTERNS.md +0 -0
  220. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_INTEGRATION_GUIDE.md +0 -0
  221. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_RESEARCH_INDEX.md +0 -0
  222. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/REFLECTION_INDEX.md +0 -0
  223. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/REFLECTION_RESEARCH_REPORT.md +0 -0
  224. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/RESEARCH_SUMMARY.md +0 -0
  225. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/anthropic-sdk.md +0 -0
  226. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/async-local-storage.md +0 -0
  227. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-code-patterns.md +0 -0
  228. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-decision-matrix.md +0 -0
  229. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-implementation-guide.md +0 -0
  230. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-integration-guide.md +0 -0
  231. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-patterns.md +0 -0
  232. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-quick-reference.md +0 -0
  233. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/zod-schema.md +0 -0
  234. /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/caching-lru.md +0 -0
  235. /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/introspection-tools.md +0 -0
  236. /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/reflection-patterns.md +0 -0
  237. /package/plan/{P4P5/research → 001_d3bb02af4886/docs/research/P4P5}/RESEARCH_SUMMARY.md +0 -0
  238. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/INTROSPECTION_RESEARCH_SUMMARY.md +0 -0
  239. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/README-INTROSPECTION.md +0 -0
  240. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/agent-introspection-patterns.md +0 -0
  241. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/introspection-tool-examples.md +0 -0
  242. /package/{PRPs/PRDs/001-hierarchical-workflow-engine.md → plan/001_d3bb02af4886/prd_snapshot.md} +0 -0
@@ -0,0 +1,487 @@
1
+ /**
2
+ * PRD Section 7: Observer Propagation Compliance Test
3
+ *
4
+ * Validates that observer propagation works correctly per PRD Section 7 requirements:
5
+ * - Events from deeply nested children bubble up to root observers via getRoot()
6
+ * - After reparenting, events propagate to new root's observer (not old root's)
7
+ * - getRoot() correctly follows parent chain with cycle detection
8
+ * - All observer callbacks (onEvent, onTreeChanged) are invoked
9
+ *
10
+ * Bug Context: The tree integrity bug (child in multiple parents) broke observer
11
+ * propagation because getRoot() only followed child.parent chain. This test validates
12
+ * the fix ensures events route to correct observers.
13
+ *
14
+ * References:
15
+ * - Bug Analysis: plan/docs/bugfix-architecture/bug_analysis.md (lines 60-91)
16
+ * - getRoot() implementation: src/core/workflow.ts (lines 174-189)
17
+ * - getRootObservers() implementation: src/core/workflow.ts (lines 124-139)
18
+ * - emitEvent() implementation: src/core/workflow.ts (lines 313-329)
19
+ */
20
+
21
+ import { describe, it, expect } from 'vitest';
22
+ import {
23
+ Workflow,
24
+ WorkflowObserver,
25
+ WorkflowEvent,
26
+ } from '../../index.js';
27
+
28
+ /**
29
+ * SimpleWorkflow class for testing
30
+ * Pattern from: src/__tests__/integration/workflow-reparenting.test.ts
31
+ */
32
+ class SimpleWorkflow extends Workflow {
33
+ async run(): Promise<string> {
34
+ this.setStatus('running');
35
+ this.setStatus('completed');
36
+ return 'done';
37
+ }
38
+ }
39
+
40
+ /**
41
+ * Observer creation helper
42
+ * Pattern from: src/__tests__/integration/workflow-reparenting.test.ts:142-147
43
+ */
44
+ const createObserver = (eventsArray: WorkflowEvent[]): WorkflowObserver => ({
45
+ onLog: () => {}, // Empty - not testing logs in this test suite
46
+ onEvent: (e) => eventsArray.push(e), // Capture events for validation
47
+ onStateUpdated: () => {}, // Empty - not testing state updates
48
+ onTreeChanged: () => {}, // Empty - not testing tree changes
49
+ });
50
+
51
+ describe('PRD Section 7: Observer Propagation', () => {
52
+ describe('Event Bubbling: Grandchild to Root Observer', () => {
53
+ it('should propagate events from grandchild to root observer via getRoot()', () => {
54
+ // ============================================================
55
+ // PHASE 1: Setup - Create 3-level tree
56
+ // ============================================================
57
+ // ARRANGE: Create root, child, grandchild hierarchy
58
+ const root = new SimpleWorkflow('Root');
59
+ const child = new SimpleWorkflow('Child', root);
60
+ const grandchild = new SimpleWorkflow('Grandchild', child);
61
+
62
+ // ============================================================
63
+ // PHASE 2: Add observer to root
64
+ // ============================================================
65
+ // ARRANGE: Create event capture array and observer
66
+ const rootEvents: WorkflowEvent[] = [];
67
+ root.addObserver(createObserver(rootEvents));
68
+
69
+ // ============================================================
70
+ // PHASE 3: Emit event from grandchild
71
+ // ============================================================
72
+ // ACT: Clear any construction events, then emit from grandchild
73
+ // PRD Section 7: "Observers attach to root workflow and receive all events"
74
+ rootEvents.length = 0; // Clear any construction events
75
+ grandchild.setStatus('running');
76
+
77
+ // ============================================================
78
+ // PHASE 4: Verify root observer received event
79
+ // ============================================================
80
+ // ASSERT: Root observer should receive event via getRoot() traversal
81
+ // getRoot() from grandchild returns root, then root.observers receive event
82
+ const receivedEvent = rootEvents.find((e) => e.type === 'treeUpdated');
83
+ expect(receivedEvent).toBeDefined();
84
+
85
+ // Type guard for discriminated union
86
+ if (receivedEvent?.type === 'treeUpdated') {
87
+ expect(receivedEvent.root.id).toBe(root.id);
88
+ }
89
+ });
90
+
91
+ it('should propagate events through multiple hierarchy levels', () => {
92
+ // ============================================================
93
+ // Test deeper hierarchy (4+ levels) to verify getRoot() traversal
94
+ // ============================================================
95
+ // ARRANGE: Create 5-level deep hierarchy
96
+ const root = new SimpleWorkflow('Root');
97
+ const level1 = new SimpleWorkflow('L1', root);
98
+ const level2 = new SimpleWorkflow('L2', level1);
99
+ const level3 = new SimpleWorkflow('L3', level2);
100
+ const level4 = new SimpleWorkflow('L4', level3);
101
+
102
+ const rootEvents: WorkflowEvent[] = [];
103
+ root.addObserver(createObserver(rootEvents));
104
+
105
+ // ACT: Emit event from deepest level
106
+ rootEvents.length = 0;
107
+ level4.setStatus('running');
108
+
109
+ // ASSERT: Verify event bubbled through 4 levels to root
110
+ // This validates getRoot() correctly traverses: level4 -> level3 -> level2 -> level1 -> root
111
+ expect(rootEvents.some((e) => e.type === 'treeUpdated')).toBe(true);
112
+
113
+ // Verify the event root is the original root workflow
114
+ const treeUpdatedEvent = rootEvents.find((e) => e.type === 'treeUpdated');
115
+ if (treeUpdatedEvent?.type === 'treeUpdated') {
116
+ expect(treeUpdatedEvent.root.id).toBe(root.id);
117
+ }
118
+ });
119
+ });
120
+
121
+ describe('Observer Routing After Reparenting', () => {
122
+ it('should route events to new root observer after reparenting', () => {
123
+ // ============================================================
124
+ // PHASE 1: Setup - Create parent1, parent2, grandchild with parent1
125
+ // ============================================================
126
+ // ARRANGE: Create initial tree structure
127
+ const parent1 = new SimpleWorkflow('Parent1');
128
+ const child1 = new SimpleWorkflow('Child1', parent1);
129
+ const grandchild = new SimpleWorkflow('Grandchild', child1);
130
+ const parent2 = new SimpleWorkflow('Parent2');
131
+
132
+ // ============================================================
133
+ // PHASE 2: Verify initial propagation to parent1 observer
134
+ // ============================================================
135
+ // ARRANGE: Create observer for parent1
136
+ const parent1Events: WorkflowEvent[] = [];
137
+ parent1.addObserver(createObserver(parent1Events));
138
+
139
+ // ACT: Emit event from grandchild
140
+ parent1Events.length = 0;
141
+ grandchild.setStatus('running');
142
+
143
+ // ASSERT: parent1 observer receives event (initial state)
144
+ expect(parent1Events.some((e) => e.type === 'treeUpdated')).toBe(true);
145
+
146
+ // ============================================================
147
+ // PHASE 3: Reparent grandchild to different tree
148
+ // ============================================================
149
+ // ACT: First detach from child1, then attach to child2
150
+ // Pattern: detachChild() -> attachChild() for proper reparenting
151
+ const child2 = new SimpleWorkflow('Child2', parent2);
152
+ child1.detachChild(grandchild);
153
+ child2.attachChild(grandchild);
154
+
155
+ // ============================================================
156
+ // PHASE 4: Add observer to new root (parent2)
157
+ // ============================================================
158
+ // ARRANGE: Create observer for parent2
159
+ const parent2Events: WorkflowEvent[] = [];
160
+ parent2.addObserver(createObserver(parent2Events));
161
+
162
+ // ============================================================
163
+ // PHASE 5: Emit event from grandchild
164
+ // ============================================================
165
+ // ACT: Clear events to isolate post-reparenting behavior
166
+ parent1Events.length = 0;
167
+ parent2Events.length = 0;
168
+ grandchild.setStatus('completed');
169
+
170
+ // ============================================================
171
+ // PHASE 6: Verify new root observer receives event
172
+ // ============================================================
173
+ // ASSERT: parent2 observer receives event (after reparenting)
174
+ expect(parent2Events.some((e) => e.type === 'treeUpdated')).toBe(true);
175
+ });
176
+
177
+ it('should not route events to old root observer after reparenting', () => {
178
+ // ============================================================
179
+ // CRITICAL VALIDATION: This test validates the bug fix
180
+ // ============================================================
181
+ // Bug analysis (bug_analysis.md:60-91) showed:
182
+ // - child.parent stayed pointing to old parent
183
+ // - getRoot() returned wrong root
184
+ // - Old parent's observers still received events after reparenting
185
+ //
186
+ // This test ensures that after the fix:
187
+ // - child.parent is updated to new parent
188
+ // - getRoot() returns correct root
189
+ // - Old parent's observers do NOT receive events
190
+
191
+ // ARRANGE: Create two separate trees with observers
192
+ const parent1 = new SimpleWorkflow('Parent1');
193
+ const child1 = new SimpleWorkflow('Child1', parent1);
194
+ const grandchild = new SimpleWorkflow('Grandchild', child1);
195
+ const parent2 = new SimpleWorkflow('Parent2');
196
+ const child2 = new SimpleWorkflow('Child2', parent2);
197
+
198
+ const parent1Events: WorkflowEvent[] = [];
199
+ const parent2Events: WorkflowEvent[] = [];
200
+
201
+ parent1.addObserver(createObserver(parent1Events));
202
+ parent2.addObserver(createObserver(parent2Events));
203
+
204
+ // ACT: Reparent grandchild from parent1 tree to parent2 tree
205
+ // Pattern: detach from old parent, then attach to new parent
206
+ child1.detachChild(grandchild);
207
+ child2.attachChild(grandchild);
208
+
209
+ // Clear events to isolate post-reparenting behavior
210
+ parent1Events.length = 0;
211
+ parent2Events.length = 0;
212
+
213
+ // Emit event from grandchild
214
+ grandchild.setStatus('running');
215
+
216
+ // ASSERT: CRITICAL - Old root's observer must NOT receive event
217
+ // This was the bug - old parent's observers still received events
218
+ expect(parent1Events.some((e) => e.type === 'treeUpdated')).toBe(false);
219
+ expect(parent1Events.length).toBe(0);
220
+
221
+ // ASSERT: New root's observer MUST receive event
222
+ expect(parent2Events.some((e) => e.type === 'treeUpdated')).toBe(true);
223
+ });
224
+ });
225
+
226
+ describe('getRoot() Traversal Correctness', () => {
227
+ it('should find root workflow via parent chain traversal', () => {
228
+ // ============================================================
229
+ // Verify getRoot() from grandchild returns root
230
+ // ============================================================
231
+ // This is called internally by getRootObservers()
232
+ // ARRANGE: Create 3-level tree
233
+ const root = new SimpleWorkflow('Root');
234
+ const child = new SimpleWorkflow('Child', root);
235
+ const grandchild = new SimpleWorkflow('Grandchild', child);
236
+
237
+ // ACT: Call getRoot() on grandchild (protected method, access via cast)
238
+ const foundRoot = (grandchild as any).getRoot();
239
+
240
+ // ASSERT: getRoot() should return root workflow
241
+ expect(foundRoot).toBe(root);
242
+ expect(foundRoot).not.toBe(child);
243
+ expect(foundRoot).not.toBe(grandchild);
244
+ });
245
+
246
+ it('should find root through deep parent chain', () => {
247
+ // ============================================================
248
+ // Verify getRoot() works correctly with deep hierarchies
249
+ // ============================================================
250
+ // ARRANGE: Create 10-level deep hierarchy
251
+ const root = new SimpleWorkflow('Root');
252
+ let current = root;
253
+
254
+ for (let i = 0; i < 10; i++) {
255
+ const nextChild = new SimpleWorkflow(`Level${i}`);
256
+ current.attachChild(nextChild);
257
+ current = nextChild;
258
+ }
259
+
260
+ // ACT: Call getRoot() on deepest child
261
+ const foundRoot = (current as any).getRoot();
262
+
263
+ // ASSERT: Deepest child's getRoot() returns original root
264
+ expect(foundRoot).toBe(root);
265
+ });
266
+
267
+ it('should maintain correct root after multiple reparenting cycles', () => {
268
+ // ============================================================
269
+ // Test reparenting: A->B->C => X->Y->C => A->Z->C
270
+ // ============================================================
271
+ // ARRANGE: Create initial tree A->B->C
272
+ const parentA = new SimpleWorkflow('ParentA');
273
+ const parentB = new SimpleWorkflow('ParentB', parentA);
274
+ const childC = new SimpleWorkflow('ChildC', parentB);
275
+
276
+ // ASSERT: Initial - C's root should be A
277
+ expect((childC as any).getRoot()).toBe(parentA);
278
+
279
+ // ACT: Reparent to X->Y->C
280
+ const parentX = new SimpleWorkflow('ParentX');
281
+ const parentY = new SimpleWorkflow('ParentY', parentX);
282
+ parentB.detachChild(childC);
283
+ parentY.attachChild(childC);
284
+
285
+ // ASSERT: After first reparenting, C's root should be X
286
+ expect((childC as any).getRoot()).toBe(parentX);
287
+
288
+ // ACT: Reparent again to A->Z->C
289
+ const parentZ = new SimpleWorkflow('ParentZ', parentA);
290
+ parentY.detachChild(childC);
291
+ parentZ.attachChild(childC);
292
+
293
+ // ASSERT: After second reparenting, C's root should be A again
294
+ expect((childC as any).getRoot()).toBe(parentA);
295
+ });
296
+ });
297
+
298
+ describe('Observer Callback Invocation', () => {
299
+ it('should invoke onEvent() for all event types from children', () => {
300
+ // ============================================================
301
+ // Verify onEvent() callback is invoked
302
+ // ============================================================
303
+ // ARRANGE: Create 2-level tree with tracking observer
304
+ const root = new SimpleWorkflow('Root');
305
+ const child = new SimpleWorkflow('Child', root);
306
+
307
+ let onEventCallCount = 0;
308
+ let receivedEventType: string | undefined;
309
+
310
+ const trackingObserver: WorkflowObserver = {
311
+ onLog: () => {},
312
+ onEvent: (e) => {
313
+ onEventCallCount++;
314
+ receivedEventType = e.type;
315
+ },
316
+ onStateUpdated: () => {},
317
+ onTreeChanged: () => {},
318
+ };
319
+
320
+ root.addObserver(trackingObserver);
321
+
322
+ // ACT: Emit event from child
323
+ child.setStatus('running');
324
+
325
+ // ASSERT: Verify onEvent was called
326
+ expect(onEventCallCount).toBeGreaterThan(0);
327
+ expect(receivedEventType).toBeDefined();
328
+ });
329
+
330
+ it('should invoke onTreeChanged() for tree update events', () => {
331
+ // ============================================================
332
+ // Verify onTreeChanged() callback is invoked
333
+ // ============================================================
334
+ // ARRANGE: Create 2-level tree with tracking observer
335
+ const root = new SimpleWorkflow('Root');
336
+ const child = new SimpleWorkflow('Child', root);
337
+
338
+ let onTreeChangedCallCount = 0;
339
+ let receivedRoot: any;
340
+
341
+ const trackingObserver: WorkflowObserver = {
342
+ onLog: () => {},
343
+ onEvent: () => {},
344
+ onStateUpdated: () => {},
345
+ onTreeChanged: (rootNode) => {
346
+ onTreeChangedCallCount++;
347
+ receivedRoot = rootNode;
348
+ },
349
+ };
350
+
351
+ root.addObserver(trackingObserver);
352
+
353
+ // ACT: Emit tree update event (triggers both onEvent and onTreeChanged)
354
+ // Note: setStatus() emits treeUpdated event which triggers onTreeChanged()
355
+ child.setStatus('running');
356
+
357
+ // ASSERT: Verify onTreeChanged was called with correct root
358
+ expect(onTreeChangedCallCount).toBeGreaterThan(0);
359
+ expect(receivedRoot).toBeDefined();
360
+ expect(receivedRoot.id).toBe(root.id);
361
+ });
362
+
363
+ it('should invoke callbacks in correct order', () => {
364
+ // ============================================================
365
+ // Verify callback order: onEvent before onTreeChanged
366
+ // ============================================================
367
+ // ARRANGE: Create 2-level tree with order-tracking observer
368
+ const root = new SimpleWorkflow('Root');
369
+ const child = new SimpleWorkflow('Child', root);
370
+
371
+ const callOrder: string[] = [];
372
+
373
+ const orderTrackingObserver: WorkflowObserver = {
374
+ onLog: () => callOrder.push('onLog'),
375
+ onEvent: () => callOrder.push('onEvent'),
376
+ onStateUpdated: () => callOrder.push('onStateUpdated'),
377
+ onTreeChanged: () => callOrder.push('onTreeChanged'),
378
+ };
379
+
380
+ root.addObserver(orderTrackingObserver);
381
+
382
+ // ACT: Emit tree update event
383
+ callOrder.length = 0;
384
+ child.setStatus('running');
385
+
386
+ // ASSERT: Verify callback order
387
+ // emitEvent() calls onEvent() first, then onTreeChanged()
388
+ // See workflow.ts:318-324
389
+ const eventIndex = callOrder.indexOf('onEvent');
390
+ const treeChangedIndex = callOrder.indexOf('onTreeChanged');
391
+
392
+ expect(eventIndex).toBeGreaterThanOrEqual(0);
393
+ expect(treeChangedIndex).toBeGreaterThanOrEqual(0);
394
+ expect(eventIndex).toBeLessThan(treeChangedIndex);
395
+ });
396
+ });
397
+
398
+ describe('Edge Cases: Cycle Detection', () => {
399
+ it('should still propagate events after cycle detection validation', () => {
400
+ // ============================================================
401
+ // Verify that cycle detection in getRoot() doesn't break normal propagation
402
+ // ============================================================
403
+ // getRoot() uses visited Set for cycle detection (workflow.ts:175-188)
404
+ // This test ensures normal operation is not affected
405
+
406
+ // ARRANGE: Create 3-level tree with no cycles
407
+ const root = new SimpleWorkflow('Root');
408
+ const child = new SimpleWorkflow('Child', root);
409
+ const grandchild = new SimpleWorkflow('Grandchild', child);
410
+
411
+ const rootEvents: WorkflowEvent[] = [];
412
+ root.addObserver(createObserver(rootEvents));
413
+
414
+ // ACT: Emit event from grandchild
415
+ // This should work normally (no cycle exists)
416
+ grandchild.setStatus('running');
417
+
418
+ // ASSERT: Verify propagation still works
419
+ expect(rootEvents.some((e) => e.type === 'treeUpdated')).toBe(true);
420
+
421
+ // Verify getRoot() didn't throw cycle detection error
422
+ const foundRoot = (grandchild as any).getRoot();
423
+ expect(foundRoot).toBe(root);
424
+ });
425
+ });
426
+
427
+ describe('Multiple Reparenting Cycles', () => {
428
+ it('should handle multiple reparenting cycles correctly', () => {
429
+ // ============================================================
430
+ // Test: A->B->C => X->Y->C => A->Z->C => P->Q->C
431
+ // ============================================================
432
+ // ARRANGE: Create three potential parents
433
+ const parentA = new SimpleWorkflow('ParentA');
434
+ const parentB = new SimpleWorkflow('ParentB', parentA);
435
+ const childC = new SimpleWorkflow('ChildC', parentB);
436
+
437
+ const parentX = new SimpleWorkflow('ParentX');
438
+ const parentY = new SimpleWorkflow('ParentY', parentX);
439
+
440
+ const parentP = new SimpleWorkflow('ParentP');
441
+ const parentQ = new SimpleWorkflow('ParentQ', parentP);
442
+
443
+ const eventsA: WorkflowEvent[] = [];
444
+ const eventsX: WorkflowEvent[] = [];
445
+ const eventsP: WorkflowEvent[] = [];
446
+
447
+ const createObserverWithTracking = (eventsArray: WorkflowEvent[]): WorkflowObserver => ({
448
+ onLog: () => {},
449
+ onEvent: (e) => eventsArray.push(e),
450
+ onStateUpdated: () => {},
451
+ onTreeChanged: () => {},
452
+ });
453
+
454
+ parentA.addObserver(createObserverWithTracking(eventsA));
455
+ parentX.addObserver(createObserverWithTracking(eventsX));
456
+ parentP.addObserver(createObserverWithTracking(eventsP));
457
+
458
+ // ACT & ASSERT: Cycle 1 - Verify initial state A->B->C
459
+ eventsA.length = 0;
460
+ childC.setStatus('running');
461
+ expect(eventsA.some((e) => e.type === 'treeUpdated')).toBe(true);
462
+
463
+ // ACT & ASSERT: Cycle 2 - Reparent to X->Y->C
464
+ parentB.detachChild(childC);
465
+ parentY.attachChild(childC);
466
+
467
+ eventsA.length = 0;
468
+ eventsX.length = 0;
469
+ childC.setStatus('completed');
470
+ expect(eventsX.some((e) => e.type === 'treeUpdated')).toBe(true);
471
+ expect(eventsA.some((e) => e.type === 'treeUpdated')).toBe(false);
472
+
473
+ // ACT & ASSERT: Cycle 3 - Reparent to P->Q->C
474
+ parentY.detachChild(childC);
475
+ parentQ.attachChild(childC);
476
+
477
+ eventsX.length = 0;
478
+ eventsP.length = 0;
479
+ childC.setStatus('running');
480
+ expect(eventsP.some((e) => e.type === 'treeUpdated')).toBe(true);
481
+ expect(eventsX.some((e) => e.type === 'treeUpdated')).toBe(false);
482
+
483
+ // Final: Verify getRoot() returns correct root
484
+ expect((childC as any).getRoot()).toBe(parentP);
485
+ });
486
+ });
487
+ });
@@ -0,0 +1,143 @@
1
+ /**
2
+ * Parent Validation Tests (TDD Red Phase)
3
+ *
4
+ * These tests validate the attachChild() method properly prevents
5
+ * attaching a child workflow that already has a different parent.
6
+ *
7
+ * This is the RED phase of TDD - tests are written to FAIL initially,
8
+ * documenting the expected behavior before implementation.
9
+ *
10
+ * Related: plan/docs/bugfix-architecture/bug_analysis.md
11
+ */
12
+
13
+ import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
14
+ import { Workflow } from '../../index.js';
15
+
16
+ /**
17
+ * SimpleWorkflow class for testing
18
+ * Pattern from: src/__tests__/unit/workflow.test.ts:4-11
19
+ */
20
+ class SimpleWorkflow extends Workflow {
21
+ async run(): Promise<string> {
22
+ this.setStatus('running');
23
+ this.setStatus('completed');
24
+ return 'done';
25
+ }
26
+ }
27
+
28
+ describe('Adversarial: Parent Validation', () => {
29
+ /**
30
+ * Setup: Mock console methods to capture error messages
31
+ * Pattern from: research/console-mocking.md "Basic Spying Patterns"
32
+ */
33
+ beforeEach(() => {
34
+ vi.spyOn(console, 'log').mockImplementation(() => {});
35
+ vi.spyOn(console, 'error').mockImplementation(() => {});
36
+ vi.spyOn(console, 'warn').mockImplementation(() => {});
37
+ });
38
+
39
+ /**
40
+ * Teardown: Restore all mocks to prevent test pollution
41
+ * CRITICAL: Always use vi.restoreAllMocks() in afterEach
42
+ */
43
+ afterEach(() => {
44
+ vi.restoreAllMocks();
45
+ });
46
+
47
+ /**
48
+ * Primary failing test for parent validation bug
49
+ *
50
+ * Bug: attachChild() only checks if child is already attached to THIS workflow
51
+ * It does NOT check if child already has a different parent
52
+ *
53
+ * Expected: Error thrown with message containing 'already has a parent'
54
+ * Actual: No error thrown, inconsistent tree state created
55
+ *
56
+ * Pattern from: research/error-assertions.md "Partial Message Matching"
57
+ */
58
+ it('should throw when attaching child that already has a different parent', () => {
59
+ // ARRANGE: Create two parent workflows
60
+ const parent1 = new SimpleWorkflow('Parent1');
61
+ const parent2 = new SimpleWorkflow('Parent2');
62
+
63
+ // ARRANGE: Create child with parent1 (constructor auto-attaches)
64
+ // CRITICAL: Constructor calls parent.attachChild(this) at workflow.ts:113-116
65
+ const child = new SimpleWorkflow('Child', parent1);
66
+
67
+ // Verify initial state
68
+ expect(child.parent).toBe(parent1);
69
+ expect(parent1.children).toContain(child);
70
+
71
+ // ACT & ASSERT: Attempting to attach child to parent2 should throw
72
+ // This test FAILS because attachChild() doesn't check child.parent !== this
73
+ expect(() => parent2.attachChild(child)).toThrow('already has a parent');
74
+ });
75
+
76
+ /**
77
+ * Test: Manual Parent Mutation with 'as any'
78
+ *
79
+ * Scenario: Even if someone manually mutates child.parent using 'as any',
80
+ * attachChild() should still validate and throw an error.
81
+ *
82
+ * This tests the defensive programming aspect - that the existing validation
83
+ * checks catch inconsistent state even when TypeScript type safety is bypassed.
84
+ *
85
+ * The attachChild() method has two validation checks (in order):
86
+ * 1. Line 217-219: children.includes(child) check
87
+ * 2. Line 222-228: child.parent !== null && child.parent !== this check
88
+ *
89
+ * When we manually mutate (child as any).parent = parent2, the child is
90
+ * still in parent1.children, so the first check throws first.
91
+ *
92
+ * Pattern from: plan/bugfix/P1M3T1S2/PRP.md "Manual Parent Mutation Test"
93
+ */
94
+ it('should throw when manually mutating parent with as any then calling attachChild', () => {
95
+ // ARRANGE: Create two parent workflows
96
+ const parent1 = new SimpleWorkflow('Parent1');
97
+ const parent2 = new SimpleWorkflow('Parent2');
98
+
99
+ // ARRANGE: Create child with parent1 (constructor auto-attaches)
100
+ // CRITICAL: Constructor calls parent.attachChild(this) at workflow.ts:113-116
101
+ const child = new SimpleWorkflow('Child', parent1);
102
+
103
+ // Verify initial state - child should be attached to parent1
104
+ expect(child.parent).toBe(parent1);
105
+ expect(parent1.children).toContain(child);
106
+
107
+ // ARRANGE: Manually mutate child.parent using 'as any' to bypass TypeScript
108
+ // This simulates a developer bypassing the type system
109
+ (child as any).parent = parent2;
110
+
111
+ // Verify manual mutation worked
112
+ expect(child.parent).toBe(parent2); // Now points to parent2
113
+ expect(parent1.children).toContain(child); // But still in parent1's children array!
114
+
115
+ // ACT & ASSERT: parent1.attachChild(child) should throw
116
+ // The validation at workflow.ts:217-219 checks children.includes(child) first
117
+ // Since child is still in parent1.children, it throws "Child already attached"
118
+ // This is defensive programming - even manual mutation is caught
119
+ expect(() => parent1.attachChild(child)).toThrow('Child already attached to this workflow');
120
+ });
121
+
122
+ /**
123
+ * Verify console.error is called with helpful message
124
+ *
125
+ * Pattern from: research/console-mocking.md "Verifying Error Messages"
126
+ */
127
+ it('should log helpful error message to console when attaching child with existing parent', () => {
128
+ // ARRANGE
129
+ const parent1 = new SimpleWorkflow('Parent1');
130
+ const parent2 = new SimpleWorkflow('Parent2');
131
+ const child = new SimpleWorkflow('Child', parent1);
132
+
133
+ // ACT: Attempt the invalid attachment
134
+ try {
135
+ parent2.attachChild(child);
136
+ } catch (err) {
137
+ // Expected error - test will fail because error isn't thrown yet
138
+ }
139
+
140
+ // ASSERT: Console.error should be called with helpful message
141
+ expect(console.error).toHaveBeenCalled();
142
+ });
143
+ });