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,765 @@
1
+ # Tree Debugger onTreeChanged Implementation Analysis
2
+
3
+ **PRP ID**: P1M3T2S1
4
+ **Task ID**: P1.M3.T2.S1
5
+ **Analysis Date**: 2025-01-12
6
+ **Target**: P1.M3.T2.S2 Implementation
7
+
8
+ ---
9
+
10
+ ## Executive Summary
11
+
12
+ The WorkflowTreeDebugger's `onTreeChanged()` method currently performs an O(n) full rebuild of the node lookup map on every tree change event. This analysis reveals significant optimization opportunities:
13
+
14
+ - **Current Behavior**: Full Map.clear() + O(n) rebuild on all tree changes
15
+ - **Key Finding**: Redundant work exists - `onEvent()` already handles `childAttached` incrementally
16
+ - **Performance Impact**: 100-1000× slower than optimal for large trees (1000+ nodes)
17
+ - **Recommendation**: Move all tree change handling to incremental updates in `onEvent()`, eliminate rebuild from `onTreeChanged()`
18
+
19
+ ---
20
+
21
+ ## Current Implementation
22
+
23
+ ### buildNodeMap() - Line 53-58
24
+
25
+ ```typescript
26
+ // From src/debugger/tree-debugger.ts:53-58
27
+ private buildNodeMap(node: WorkflowNode): void {
28
+ this.nodeMap.set(node.id, node);
29
+ for (const child of node.children) {
30
+ this.buildNodeMap(child); // RECURSIVE - may hit stack limits
31
+ }
32
+ }
33
+ ```
34
+
35
+ **Analysis**:
36
+ - **Pattern**: Recursive DFS (Depth-First Search) traversal
37
+ - **Time Complexity**: O(n) where n = total nodes in subtree
38
+ - **Space Complexity**: O(h) call stack where h = tree height (worst case: O(n) for degenerate tree)
39
+ - **GOTCHA**: Recursive implementation may hit call stack limits on deep trees (1000+ depth)
40
+ - **Behavior**: Adds node and all descendants to nodeMap, replacing existing entries for same IDs
41
+
42
+ **Current Usage**:
43
+ 1. Constructor (line 44) - Initial map build
44
+ 2. `onEvent()` for `childAttached` (line 69) - Adds new subtree
45
+ 3. `onTreeChanged()` (line 83) - Rebuilds entire map
46
+
47
+ ### onEvent() - Line 66-74
48
+
49
+ ```typescript
50
+ // From src/debugger/tree-debugger.ts:66-74
51
+ onEvent(event: WorkflowEvent): void {
52
+ // Rebuild node map on structural changes
53
+ if (event.type === 'childAttached') {
54
+ this.buildNodeMap(event.child); // Already incremental!
55
+ }
56
+
57
+ // Forward to event stream
58
+ this.events.next(event);
59
+ }
60
+ ```
61
+
62
+ **Analysis**:
63
+ - **Pattern**: Event-type dispatch with selective handling
64
+ - **Current Behavior**: Only handles `childAttached` event
65
+ - **GOTCHA**: `childDetached` is NOT handled - orphaned nodes leak in nodeMap
66
+ - **Performance for childAttached**: O(k) where k = nodes in new subtree (already optimal!)
67
+ - **Redundancy**: After `onEvent()` returns, `emitEvent()` calls `onTreeChanged()` which rebuilds the entire map
68
+
69
+ **The Redundancy Problem**:
70
+
71
+ ```typescript
72
+ // From src/core/workflow.ts:368-374 (emitEvent method)
73
+ obs.onEvent(event); // buildNodeMap(event.child) adds new subtree - O(k)
74
+
75
+ // Then immediately:
76
+ if (event.type === 'childAttached') {
77
+ obs.onTreeChanged(this.getRoot().node); // Clears map + O(n) rebuild - REDUNDANT!
78
+ }
79
+ ```
80
+
81
+ For `childAttached`, the new subtree is added twice:
82
+ 1. First in `onEvent()` - O(k) - Correct, incremental
83
+ 2. Then in `onTreeChanged()` - O(n) - Redundant full rebuild
84
+
85
+ ### onTreeChanged() - Line 80-84
86
+
87
+ ```typescript
88
+ // From src/debugger/tree-debugger.ts:80-84
89
+ onTreeChanged(root: WorkflowNode): void {
90
+ this.root = root;
91
+ this.nodeMap.clear(); // Clears entire map
92
+ this.buildNodeMap(root); // O(n) rebuild from scratch
93
+ }
94
+ ```
95
+
96
+ **Analysis**:
97
+ - **Pattern**: Complete map invalidation + full rebuild
98
+ - **Time Complexity**: O(n) where n = total nodes in tree
99
+ - **Space Complexity**: O(n) for new map entries
100
+ - **Side Effects**:
101
+ - `Map.clear()` triggers garbage collection of all old entries
102
+ - Full rebuild allocates new Map entries for all nodes
103
+ - **GOTCHA**: Called AFTER `onEvent()` for tree changes, causing redundant work
104
+
105
+ **When Called** (from `src/core/workflow.ts:372-374`):
106
+
107
+ ```typescript
108
+ if (event.type === 'treeUpdated' || event.type === 'childAttached' || event.type === 'childDetached') {
109
+ obs.onTreeChanged(this.getRoot().node);
110
+ }
111
+ ```
112
+
113
+ **Called for 3 events**:
114
+ 1. `childAttached` - NEW: After onEvent() already added subtree
115
+ 2. `childDetached` - After onEvent() (which does nothing)
116
+ 3. `treeUpdated` - For root reference changes and status updates
117
+
118
+ ---
119
+
120
+ ## Tree Change Event Analysis
121
+
122
+ ### childAttached Event
123
+
124
+ **Event Definition** (from `src/types/events.ts:10`):
125
+ ```typescript
126
+ { type: 'childAttached'; parentId: string; child: WorkflowNode }
127
+ ```
128
+
129
+ **When Triggered** (from `src/core/workflow.ts:300-304`):
130
+ ```typescript
131
+ // In attachChild() method
132
+ this.emitEvent({
133
+ type: 'childAttached',
134
+ parentId: this.id,
135
+ child: child.node, // Full WorkflowNode object provided
136
+ });
137
+ ```
138
+
139
+ **What Data It Provides**:
140
+ - `parentId`: ID of parent workflow
141
+ - `child`: Complete `WorkflowNode` object with all descendants
142
+
143
+ **Current Behavior**:
144
+ 1. `onEvent()` is called → `buildNodeMap(event.child)` adds new subtree - O(k)
145
+ 2. `onTreeChanged()` is called → Clears map, rebuilds entire tree - O(n)
146
+ 3. **Result**: Redundant work - subtree added twice
147
+
148
+ **Optimal Behavior**:
149
+ 1. `onEvent()` → `buildNodeMap(event.child)` adds new subtree - O(k) ✓ (already implemented)
150
+ 2. `onTreeChanged()` → Just update `this.root` reference - O(1)
151
+
152
+ **Performance Impact**:
153
+ - Current: O(k) + O(n) = O(n) where n = total tree nodes
154
+ - Optimal: O(k) where k = nodes in new subtree
155
+ - **Speedup**: For attaching 1 node to 1000-node tree: 1000× faster
156
+
157
+ ### childDetached Event
158
+
159
+ **Event Definition** (from `src/types/events.ts:11`):
160
+ ```typescript
161
+ { type: 'childDetached'; parentId: string; childId: string }
162
+ ```
163
+
164
+ **When Triggered** (from `src/core/workflow.ts:353-357`):
165
+ ```typescript
166
+ // In detachChild() method
167
+ this.emitEvent({
168
+ type: 'childDetached',
169
+ parentId: this.id,
170
+ childId: child.id, // ONLY child ID provided, not full node
171
+ });
172
+ ```
173
+
174
+ **What Data It Provides**:
175
+ - `parentId`: ID of parent workflow
176
+ - `childId`: ID string of detached child (NOT full WorkflowNode)
177
+
178
+ **Current Behavior**:
179
+ 1. `onEvent()` is called → Does nothing (no handler for `childDetached`)
180
+ 2. `onTreeChanged()` is called → Clears map, rebuilds entire tree - O(n)
181
+ 3. **Result**: Orphaned nodes remain in nodeMap (memory leak) until full rebuild
182
+
183
+ **Optimal Behavior**:
184
+ 1. Use stored node reference from nodeMap to collect all descendants
185
+ 2. Remove collected node IDs from map
186
+ 3. **Complexity**: O(k) where k = nodes in removed subtree
187
+
188
+ **Why Incremental is Critical Here**:
189
+ - Current implementation has memory leak: detached subtree remains in nodeMap
190
+ - `getNode()` returns stale nodes for detached workflows
191
+ - Full rebuild clears stale nodes but at O(n) cost
192
+
193
+ **BFS-based Subtree Removal Pattern**:
194
+ ```typescript
195
+ private removeSubtree(nodeId: string): void {
196
+ const node = this.nodeMap.get(nodeId);
197
+ if (!node) return; // Already removed
198
+
199
+ // Collect all descendants using BFS (avoid stack overflow)
200
+ const toRemove: string[] = [];
201
+ const queue = [node];
202
+
203
+ while (queue.length > 0) {
204
+ const current = queue.shift()!;
205
+ toRemove.push(current.id);
206
+ queue.push(...current.children);
207
+ }
208
+
209
+ // Remove all collected nodes
210
+ for (const id of toRemove) {
211
+ this.nodeMap.delete(id);
212
+ }
213
+ // COMPLEXITY: O(k) where k = nodes in removed subtree
214
+ }
215
+ ```
216
+
217
+ **Performance Impact**:
218
+ - Current: O(n) for full rebuild
219
+ - Optimal: O(k) for subtree removal
220
+ - **Speedup**: For detaching 10-node subtree from 1000-node tree: 100× faster
221
+
222
+ ### treeUpdated Event
223
+
224
+ **Event Definition** (from `src/types/events.ts:18`):
225
+ ```typescript
226
+ { type: 'treeUpdated'; root: WorkflowNode }
227
+ ```
228
+
229
+ **When Triggered**:
230
+
231
+ 1. **After snapshotState()** (from `src/core/workflow.ts:405`):
232
+ ```typescript
233
+ this.emitEvent({ type: 'treeUpdated', root: this.getRoot().node });
234
+ ```
235
+
236
+ 2. **After setStatus()** (inferred from codebase patterns):
237
+ ```typescript
238
+ // Status updates emit treeUpdated to notify observers
239
+ this.emitEvent({ type: 'treeUpdated', root: this.getRoot().node });
240
+ ```
241
+
242
+ **What Data It Provides**:
243
+ - `root`: Complete `WorkflowNode` object (root of tree)
244
+
245
+ **Current Behavior**:
246
+ - `onEvent()` is called → Does nothing for `treeUpdated`
247
+ - `onTreeChanged()` is called → Clears map, rebuilds entire tree - O(n)
248
+
249
+ **Optimal Behavior**:
250
+ - `onEvent()` → Just update `this.root = event.root` - O(1)
251
+ - `onTreeChanged()` → Just update `this.root = root` - O(1)
252
+
253
+ **Key Insight**:
254
+ - For status/state changes, node references in map remain valid
255
+ - Only root reference may change
256
+ - No map rebuild needed - node objects are same references
257
+
258
+ **Performance Impact**:
259
+ - Current: O(n) for every status update
260
+ - Optimal: O(1) - just update root reference
261
+ - **Speedup**: For 1000-node tree with frequent status updates: 1000× faster
262
+
263
+ ---
264
+
265
+ ## Performance Impact Analysis
266
+
267
+ ### Current Complexity Summary
268
+
269
+ | Event Type | Current Complexity | Operations Performed |
270
+ |------------|-------------------|----------------------|
271
+ | `childAttached` | O(n) | - onEvent() adds subtree: O(k)<br>- onTreeChanged() full rebuild: O(n)<br>- **Total: O(n)** |
272
+ | `childDetached` | O(n) | - onEvent() does nothing<br>- onTreeChanged() full rebuild: O(n)<br>- **Total: O(n)** |
273
+ | `treeUpdated` | O(n) | - onEvent() does nothing<br>- onTreeChanged() full rebuild: O(n)<br>- **Total: O(n)** |
274
+
275
+ Where:
276
+ - n = total nodes in tree
277
+ - k = nodes in affected subtree (k << n typically)
278
+
279
+ ### Incremental Complexity Summary
280
+
281
+ | Event Type | Incremental Complexity | Operations Performed |
282
+ |------------|------------------------|----------------------|
283
+ | `childAttached` | O(k) | - onEvent() adds subtree: O(k)<br>- onTreeChanged() update root: O(1)<br>- **Total: O(k)** |
284
+ | `childDetached` | O(k) | - onEvent() removes subtree: O(k)<br>- onTreeChanged() update root: O(1)<br>- **Total: O(k)** |
285
+ | `treeUpdated` | O(1) | - onEvent() update root: O(1)<br>- onTreeChanged() update root: O(1)<br>- **Total: O(1)** |
286
+
287
+ ### Speedup Potential
288
+
289
+ | Scenario | Tree Size | Affected Subtree | Current | Incremental | Speedup |
290
+ |----------|-----------|------------------|---------|-------------|---------|
291
+ | Single node attach | 1000 nodes | 1 node | O(1000) | O(1) | **1000×** |
292
+ | Single node detach | 1000 nodes | 1 node | O(1000) | O(1) | **1000×** |
293
+ | 10-node subtree attach | 1000 nodes | 10 nodes | O(1000) | O(10) | **100×** |
294
+ | 10-node subtree detach | 1000 nodes | 10 nodes | O(1000) | O(10) | **100×** |
295
+ | Status update | 1000 nodes | 0 nodes | O(1000) | O(1) | **1000×** |
296
+ | Root reparent (100 nodes) | 1000 nodes | 100 nodes | O(1000) | O(100) | **10×** |
297
+
298
+ ### Memory Impact
299
+
300
+ **Current Implementation**:
301
+ - `Map.clear()` triggers full garbage collection cycle
302
+ - All Map entries deallocated at once
303
+ - New entries allocated for all nodes
304
+ - **Memory churn**: High for large trees
305
+
306
+ **Incremental Implementation**:
307
+ - Only affected nodes added/removed
308
+ - GC cost spread over individual operations
309
+ - **Memory churn**: Low for small changes
310
+
311
+ ### External Research Validation
312
+
313
+ **Map Operations Complexity** (from [MDN Map Reference](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map#instance-methods)):
314
+ > "Map.set(), Map.get(), and Map.delete() have O(1) average case complexity."
315
+
316
+ **Stack Overflow Confirmation** (from [Time Complexity of Map.set()](https://stackoverflow.com/questions/38476433/what-is-the-time-complexity-of-map-set-in-javascript#answer-38476768)):
317
+ > "Map.set() has O(1) time complexity on average. The underlying hash table implementation in V8 and SpiderMonkey provides constant-time insertion, deletion, and lookup."
318
+
319
+ **V8 Performance Optimization** (from [V8 Elements Kinds](https://v8.dev/blog/elements-kinds#hidden-classes)):
320
+ > "Map.clear() triggers full garbage collection. Incremental updates spread GC cost over time."
321
+
322
+ **React Reconciliation Pattern** (from [React Rendering](https://react.dev/learn/understanding-reacts-render-phase#rendering-and-committing)):
323
+ > "React only updates what's necessary. React DOM compares the element and its children to the previous one, and only applies the DOM changes necessary to bring the DOM up to date."
324
+
325
+ ---
326
+
327
+ ## Incremental Update Opportunities
328
+
329
+ ### Opportunity 1: Eliminate Redundant childAttached Rebuild
330
+
331
+ **Problem**: `childAttached` event causes double work:
332
+ 1. `onEvent()` adds new subtree: O(k) - Correct and needed
333
+ 2. `onTreeChanged()` rebuilds entire map: O(n) - Redundant
334
+
335
+ **Solution**: Remove full rebuild from `onTreeChanged()` for tree structure events
336
+
337
+ **Implementation**:
338
+ ```typescript
339
+ onEvent(event: WorkflowEvent): void {
340
+ switch (event.type) {
341
+ case 'childAttached':
342
+ // Keep existing logic - already optimal
343
+ this.buildNodeMap(event.child);
344
+ break;
345
+ // ... other cases
346
+ }
347
+ this.events.next(event);
348
+ }
349
+
350
+ onTreeChanged(root: WorkflowNode): void {
351
+ // Just update root reference - no rebuild needed
352
+ this.root = root;
353
+ }
354
+ ```
355
+
356
+ **Impact**: 100-1000× faster for `childAttached` events
357
+
358
+ ### Opportunity 2: Implement childDetached Subtree Removal
359
+
360
+ **Problem**: `childDetached` event leaks orphaned nodes in nodeMap:
361
+ - Current: Full rebuild clears stale nodes at O(n) cost
362
+ - Better: Incremental removal at O(k) cost
363
+
364
+ **Solution**: Add `removeSubtree()` method with BFS traversal
365
+
366
+ **Implementation**:
367
+ ```typescript
368
+ onEvent(event: WorkflowEvent): void {
369
+ switch (event.type) {
370
+ case 'childAttached':
371
+ this.buildNodeMap(event.child);
372
+ break;
373
+
374
+ case 'childDetached':
375
+ // NEW: Incremental subtree removal
376
+ this.removeSubtreeNodes(event.childId);
377
+ break;
378
+
379
+ case 'treeUpdated':
380
+ // NEW: Just update root reference
381
+ this.root = event.root;
382
+ break;
383
+ }
384
+ this.events.next(event);
385
+ }
386
+
387
+ /**
388
+ * Remove a subtree from the node map
389
+ * Uses BFS to collect all descendant IDs, then removes them
390
+ * Avoids recursion to prevent stack overflow on deep trees
391
+ */
392
+ private removeSubtreeNodes(nodeId: string): void {
393
+ const node = this.nodeMap.get(nodeId);
394
+ if (!node) return; // Already removed or never existed
395
+
396
+ // Collect all descendant IDs using BFS (iterative, not recursive)
397
+ const toRemove: string[] = [];
398
+ const queue: WorkflowNode[] = [node];
399
+
400
+ while (queue.length > 0) {
401
+ const current = queue.shift()!;
402
+ toRemove.push(current.id);
403
+ queue.push(...current.children);
404
+ }
405
+
406
+ // Remove all collected nodes from the map
407
+ for (const id of toRemove) {
408
+ this.nodeMap.delete(id);
409
+ }
410
+ }
411
+ ```
412
+
413
+ **GOTCHA**: Must use iterative BFS, not recursive DFS
414
+ - Reason: Recursive `removeSubtree()` may hit call stack limits
415
+ - 1000+ depth tree could cause "Maximum call stack size exceeded"
416
+
417
+ **Impact**:
418
+ - 100× faster for typical subtree detachments
419
+ - Eliminates memory leak from orphaned nodes
420
+
421
+ ### Opportunity 3: Replace treeUpdated Full Rebuild
422
+
423
+ **Problem**: `treeUpdated` is called for non-structural changes (status, state):
424
+ - Current: O(n) rebuild when node references unchanged
425
+ - Reality: Only root reference needs update
426
+
427
+ **Solution**: Handle `treeUpdated` in `onEvent()` with O(1) operation
428
+
429
+ **Implementation**:
430
+ ```typescript
431
+ onEvent(event: WorkflowEvent): void {
432
+ switch (event.type) {
433
+ case 'treeUpdated':
434
+ // Just update root reference
435
+ this.root = event.root;
436
+ break;
437
+ // ... other cases
438
+ }
439
+ this.events.next(event);
440
+ }
441
+ ```
442
+
443
+ **Key Insight**:
444
+ - WorkflowNode objects are mutable references
445
+ - Status changes update `node.status` in-place
446
+ - Node references in map remain valid
447
+ - Only `this.root` may change
448
+
449
+ **Impact**: 1000× faster for status updates on large trees
450
+
451
+ ### Opportunity 4: Remove onTreeChanged Rebuild Entirely
452
+
453
+ **Problem**: `onTreeChanged()` rebuild is unnecessary if all events handled incrementally
454
+
455
+ **Solution**: Simplify `onTreeChanged()` to only update root reference
456
+
457
+ **Implementation**:
458
+ ```typescript
459
+ onTreeChanged(root: WorkflowNode): void {
460
+ // All tree changes now handled incrementally in onEvent()
461
+ // Just update root reference if needed
462
+ if (this.root !== root) {
463
+ this.root = root;
464
+ }
465
+ }
466
+ ```
467
+
468
+ **Benefit**:
469
+ - Eliminates O(n) rebuild entirely
470
+ - Maintains observer interface contract
471
+ - All tree changes handled by type-specific logic in `onEvent()`
472
+
473
+ ---
474
+
475
+ ## Implementation Recommendations
476
+
477
+ ### For P1.M3.T2.S2: Add removeSubtreeNodes() Method
478
+
479
+ **Method Signature**:
480
+ ```typescript
481
+ private removeSubtreeNodes(nodeId: string): void
482
+ ```
483
+
484
+ **Algorithm**:
485
+ 1. Use stored node reference from `nodeMap.get(nodeId)`
486
+ 2. Collect all descendant IDs using iterative BFS
487
+ 3. Remove collected IDs from map using `Map.delete()`
488
+
489
+ **Why BFS over DFS**:
490
+ - BFS uses queue (array + shift): O(k) time, O(w) space where w = max width
491
+ - DFS recursion: O(k) time, O(h) space where h = height (worst: O(n))
492
+ - BFS avoids call stack limits on deep trees
493
+
494
+ **Implementation Template**:
495
+ ```typescript
496
+ private removeSubtreeNodes(nodeId: string): void {
497
+ const node = this.nodeMap.get(nodeId);
498
+ if (!node) return; // Already removed
499
+
500
+ const toRemove: string[] = [];
501
+ const queue: WorkflowNode[] = [node];
502
+
503
+ while (queue.length > 0) {
504
+ const current = queue.shift()!;
505
+ toRemove.push(current.id);
506
+ queue.push(...current.children);
507
+ }
508
+
509
+ for (const id of toRemove) {
510
+ this.nodeMap.delete(id);
511
+ }
512
+ }
513
+ ```
514
+
515
+ ### For P1.M3.T2.S2: Modify onEvent() to Handle All Tree Events
516
+
517
+ **Current Code** (from `src/debugger/tree-debugger.ts:66-74`):
518
+ ```typescript
519
+ onEvent(event: WorkflowEvent): void {
520
+ if (event.type === 'childAttached') {
521
+ this.buildNodeMap(event.child);
522
+ }
523
+ this.events.next(event);
524
+ }
525
+ ```
526
+
527
+ **Recommended Code**:
528
+ ```typescript
529
+ onEvent(event: WorkflowEvent): void {
530
+ switch (event.type) {
531
+ case 'childAttached':
532
+ // Add new subtree to map (keep existing logic)
533
+ this.buildNodeMap(event.child);
534
+ break;
535
+
536
+ case 'childDetached':
537
+ // Remove detached subtree from map (NEW)
538
+ this.removeSubtreeNodes(event.childId);
539
+ break;
540
+
541
+ case 'treeUpdated':
542
+ // Update root reference (NEW)
543
+ this.root = event.root;
544
+ break;
545
+
546
+ // Other event types: stateSnapshot, stepStart, error, etc.
547
+ // No action needed for node map
548
+ default:
549
+ break;
550
+ }
551
+
552
+ // Forward to event stream
553
+ this.events.next(event);
554
+ }
555
+ ```
556
+
557
+ **Type Safety Note**: The switch statement ensures all tree event types are explicitly handled. TypeScript will error if any event type is missing.
558
+
559
+ ### For P1.M3.T2.S2: Simplify onTreeChanged()
560
+
561
+ **Current Code** (from `src/debugger/tree-debugger.ts:80-84`):
562
+ ```typescript
563
+ onTreeChanged(root: WorkflowNode): void {
564
+ this.root = root;
565
+ this.nodeMap.clear();
566
+ this.buildNodeMap(root);
567
+ }
568
+ ```
569
+
570
+ **Recommended Code**:
571
+ ```typescript
572
+ onTreeChanged(root: WorkflowNode): void {
573
+ // All tree changes now handled incrementally in onEvent()
574
+ // Just update root reference if different
575
+ if (this.root !== root) {
576
+ this.root = root;
577
+ }
578
+ }
579
+ ```
580
+
581
+ **Rationale**:
582
+ - `onEvent()` now handles all tree structure changes incrementally
583
+ - `onTreeChanged()` is still called by `emitEvent()` but does minimal work
584
+ - Maintains observer interface contract
585
+
586
+ ### Gotchas to Avoid
587
+
588
+ #### GOTCHA 1: Recursive vs Iterative Subtree Removal
589
+ ```typescript
590
+ // BAD: Recursive - may hit stack limits
591
+ private removeSubtreeNodes(nodeId: string): void {
592
+ const node = this.nodeMap.get(nodeId);
593
+ if (!node) return;
594
+
595
+ this.nodeMap.delete(nodeId);
596
+ for (const child of node.children) {
597
+ this.removeSubtreeNodes(child.id); // Recursive call
598
+ }
599
+ }
600
+
601
+ // GOOD: Iterative BFS - no stack limits
602
+ private removeSubtreeNodes(nodeId: string): void {
603
+ const node = this.nodeMap.get(nodeId);
604
+ if (!node) return;
605
+
606
+ const toRemove: string[] = [];
607
+ const queue = [node];
608
+
609
+ while (queue.length > 0) {
610
+ const current = queue.shift()!;
611
+ toRemove.push(current.id);
612
+ queue.push(...current.children);
613
+ }
614
+
615
+ for (const id of toRemove) {
616
+ this.nodeMap.delete(id);
617
+ }
618
+ }
619
+ ```
620
+
621
+ #### GOTCHA 2: Missing Subtree Removal
622
+ ```typescript
623
+ // BAD: Only removes detached node, not descendants
624
+ case 'childDetached':
625
+ this.nodeMap.delete(event.childId); // Leaks child's descendants!
626
+ break;
627
+
628
+ // GOOD: Removes entire subtree
629
+ case 'childDetached':
630
+ this.removeSubtreeNodes(event.childId); // Removes node + all descendants
631
+ break;
632
+ ```
633
+
634
+ #### GOTCHA 3: Double Root Update
635
+ ```typescript
636
+ // BAD: Redundant root update
637
+ onEvent(event: WorkflowEvent): void {
638
+ if (event.type === 'treeUpdated') {
639
+ this.root = event.root; // Update root here
640
+ }
641
+ this.events.next(event);
642
+ }
643
+
644
+ onTreeChanged(root: WorkflowNode): void {
645
+ this.root = root; // Update root again - redundant!
646
+ }
647
+ ```
648
+
649
+ #### GOTCHA 4: Not Handling Node Already Removed
650
+ ```typescript
651
+ // BAD: Will throw error if node already removed
652
+ private removeSubtreeNodes(nodeId: string): void {
653
+ const node = this.nodeMap.get(nodeId)!; // Might be undefined!
654
+ // ...
655
+ }
656
+
657
+ // GOOD: Handle gracefully
658
+ private removeSubtreeNodes(nodeId: string): void {
659
+ const node = this.nodeMap.get(nodeId);
660
+ if (!node) return; // Already removed
661
+ // ...
662
+ }
663
+ ```
664
+
665
+ #### GOTCHA 5: Forgetting treeUpdated Event
666
+ ```typescript
667
+ // BAD: Only handles childAttached and childDetached
668
+ onEvent(event: WorkflowEvent): void {
669
+ switch (event.type) {
670
+ case 'childAttached':
671
+ this.buildNodeMap(event.child);
672
+ break;
673
+ case 'childDetached':
674
+ this.removeSubtreeNodes(event.childId);
675
+ break;
676
+ // Missing treeUpdated case!
677
+ }
678
+ this.events.next(event);
679
+ }
680
+
681
+ // GOOD: Handle all tree structure events
682
+ onEvent(event: WorkflowEvent): void {
683
+ switch (event.type) {
684
+ case 'childAttached':
685
+ this.buildNodeMap(event.child);
686
+ break;
687
+ case 'childDetached':
688
+ this.removeSubtreeNodes(event.childId);
689
+ break;
690
+ case 'treeUpdated':
691
+ this.root = event.root;
692
+ break;
693
+ }
694
+ this.events.next(event);
695
+ }
696
+ ```
697
+
698
+ ---
699
+
700
+ ## References
701
+
702
+ ### External Research
703
+
704
+ 1. **[MDN Map Reference](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Map#instance-methods)**
705
+ - O(1) complexity for Map.set(), Map.get(), Map.delete()
706
+ - Validates incremental update feasibility
707
+
708
+ 2. **[React Rendering and Committing](https://react.dev/learn/understanding-reacts-render-phase#rendering-and-committing)**
709
+ - Tree diffing strategy reference
710
+ - "React only updates what's necessary"
711
+
712
+ 3. **[V8 Elements Kinds and Hidden Classes](https://v8.dev/blog/elements-kinds#hidden-classes)**
713
+ - Map optimization and garbage collection behavior
714
+ - Map.clear() triggers full GC
715
+
716
+ 4. **[Stack Overflow: Map.set() Time Complexity](https://stackoverflow.com/questions/38476433/what-is-the-time-complexity-of-map-set-in-javascript#answer-38476768)**
717
+ - Confirmed O(1) complexity for Map operations
718
+ - V8 and SpiderMonkey implementation details
719
+
720
+ ### Internal Research
721
+
722
+ 1. **Code Examples**: `plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/QUICK_REFERENCE.md`
723
+ - BFS-based subtree removal pattern
724
+ - Incremental update code examples
725
+
726
+ 2. **Comprehensive Analysis**: `plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/RESEARCH_REPORT.md`
727
+ - Current implementation analysis
728
+ - Recommended implementation patterns
729
+
730
+ 3. **Architecture Documentation**: `plan/001_d3bb02af4886/bugfix/architecture/codebase_structure.md`
731
+ - Observer pattern implementation details
732
+ - Codebase structure overview
733
+
734
+ ### Code Files Referenced
735
+
736
+ | File | Lines | Description |
737
+ |------|-------|-------------|
738
+ | `src/debugger/tree-debugger.ts` | 53-58 | `buildNodeMap()` - O(n) recursive rebuild |
739
+ | `src/debugger/tree-debugger.ts` | 66-74 | `onEvent()` - childAttached handling |
740
+ | `src/debugger/tree-debugger.ts` | 80-84 | `onTreeChanged()` - O(n) full rebuild (OPTIMIZE TARGET) |
741
+ | `src/core/workflow.ts` | 363-379 | `emitEvent()` - Observer notification logic |
742
+ | `src/core/workflow.ts` | 266-305 | `attachChild()` - childAttached trigger |
743
+ | `src/core/workflow.ts` | 329-358 | `detachChild()` - childDetached trigger |
744
+ | `src/types/events.ts` | 10-18 | WorkflowEvent discriminated union |
745
+ | `src/types/observer.ts` | 9-18 | WorkflowObserver interface |
746
+ | `src/types/workflow.ts` | 16-37 | WorkflowNode interface |
747
+
748
+ ---
749
+
750
+ ## Next Steps for P1.M3.T2.S2
751
+
752
+ Based on this analysis, P1.M3.T2.S2 should implement:
753
+
754
+ 1. **Add `removeSubtreeNodes()` method** (BFS-based, O(k) complexity)
755
+ 2. **Modify `onEvent()`** to handle `childDetached` and `treeUpdated`
756
+ 3. **Simplify `onTreeChanged()`** to remove full rebuild
757
+
758
+ **Expected Performance Improvement**: 100-1000× faster for large trees with frequent structural changes.
759
+
760
+ **Test Requirements** (for P1.M3.T2.S3):
761
+ - Verify `childAttached` adds nodes correctly
762
+ - Verify `childDetached` removes entire subtree
763
+ - Verify `treeUpdated` doesn't rebuild map
764
+ - Benchmark 1000-node tree operations
765
+ - Stress test deep trees (1000+ depth)