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.
- package/.claude/commands/subtask-planning/prp-base-create.md +120 -0
- package/.claude/commands/subtask-planning/prp-base-execute.md +65 -0
- package/.claude/commands/task-breakdown.md +94 -0
- package/.claude/system_prompts/task-breakdown.md +1 -0
- package/CHANGELOG.md +188 -0
- package/PRD.md +543 -0
- package/README.md +99 -5
- package/examples/README.md +15 -1
- package/examples/examples/11-reparenting-workflows.ts +269 -0
- package/examples/index.ts +4 -0
- package/package-lock.json +2398 -0
- package/package.json +3 -1
- package/plan/001_d3bb02af4886/TEST_RESULTS.md +259 -0
- package/plan/001_d3bb02af4886/bug_fix_tasks.json +484 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S1/PRP.md +488 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S2/PRP.md +581 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S3/PRP.md +687 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S1/PRP.md +492 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/PRP.md +932 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/concurrent_error_testing_patterns.md +1109 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/vitest_concurrent_testing.md +802 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/workflow_engine_test_references.md +603 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S1/PRP.md +564 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S3/PRP.md +518 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S4/PRP.md +1252 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/PRP.md +364 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/CODEBASE_INVENTORY.md +114 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/DECORATOR_DOCUMENTATION_PATTERNS.md +205 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/PRD_LOCATION_ANALYSIS.md +199 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/ULTRATHINK_PRP_PLAN.md +134 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/PRP.md +495 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/research/console_error_inventory.md +435 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S2/PRP.md +506 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S3/PRP.md +612 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/PRP.md +558 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/research/external_research.md +788 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S2/PRP.md +460 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S3/PRP.md +454 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/PRP.md +520 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/RECOMMENDATION.md +417 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/external_workflow_engines_research.md +760 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/security_implications_analysis.md +245 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S2/PRP.md +792 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/PRP.md +535 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/TEST_EXECUTION_REPORT.md +190 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/PRP.md +654 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/TEST_FIX_REPORT.md +227 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/KEY_FINDINGS.md +345 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/QUICK_REFERENCE.md +193 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/test_maintenance_research.md +1323 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/BREAKING_CHANGES_AUDIT.md +1011 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/PRP.md +927 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S2/PRP.md +505 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/architecture/logger_child_signature_analysis.md +401 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/child_implementation_research.md +142 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/test_patterns_research.md +112 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/vitest_patterns_research.md +159 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/PRP.md +549 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/VERIFICATION_REPORT.md +368 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/edge_case_analysis.md +172 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/usage_inventory.md +175 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S2/PRP.md +696 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S4/PRP.md +860 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/PRP.md +1066 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01-testing-aggregated-errors.md +1103 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01_typescript_error_aggregation_patterns.md +789 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02-error-merge-strategy-testing-guide.md +1098 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02_aggregate_error_patterns.md +1037 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03-promise-allsettled-testing-patterns.md +916 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03_error_merging_strategies.md +1045 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/04_github_stackoverflow_examples.md +890 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/05_comprehensive_summary.md +822 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/INDEX.md +668 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/QUICK_REFERENCE.md +706 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/README.md +265 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/RESEARCH_REPORT.md +655 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S4/research/vitest_testing_patterns.md +1103 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T3S2/PRP.md +426 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/PRP.md +506 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/QUICK_REFERENCE.md +114 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/RESEARCH_SUMMARY.md +316 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/vitest_observer_error_logging_best_practices.md +754 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S3/PRP.md +612 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/PRP.md +719 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/README.md +215 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/analysis.md +765 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S3/PRP.md +718 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/DECISION.md +149 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/PRP.md +470 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/ULTRATHINK_PLAN.md +332 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/codebase_workflow_name_analysis.md +167 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/external_best_practices.md +265 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/validation_patterns.md +273 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S1/workflow_engine_ancestry_api_research.md +760 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S3-PRP.md +434 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S1/PRP.md +717 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/PRP.md +472 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/VALIDATION_REPORT.md +125 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/research/ULTRATHINK_PRP_PLAN.md +301 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/error-logging-best-practices.md +1170 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/research_typescript_partial_and_overloads.md +940 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-quick-reference.md +151 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-research.md +650 -0
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/prd_snapshot.md +259 -0
- package/plan/001_d3bb02af4886/bugfix/P1M1T1S1/PRP.md +457 -0
- package/plan/001_d3bb02af4886/bugfix/RESEARCH_SUMMARY.md +346 -0
- package/plan/001_d3bb02af4886/bugfix/architecture/codebase_structure.md +311 -0
- package/plan/001_d3bb02af4886/bugfix/architecture/concurrent_execution_best_practices.md +1565 -0
- package/plan/001_d3bb02af4886/bugfix/architecture/error_handling_patterns.md +288 -0
- package/plan/001_d3bb02af4886/bugfix/architecture/promise_all_analysis.md +741 -0
- package/plan/001_d3bb02af4886/docs/PRP/P1M1T1S4-functional-workflow-error-state-capture-test.md +652 -0
- package/plan/001_d3bb02af4886/docs/PRP/PRP.md +527 -0
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S1-PRP.md +415 -0
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S2-PRP.md +378 -0
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S4-PRP.md +713 -0
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M2T1S4-PRP.md +370 -0
- package/plan/001_d3bb02af4886/docs/PRP_P1M3T1S3.md +499 -0
- package/plan/001_d3bb02af4886/docs/TEST_RESULTS.md +230 -0
- package/plan/001_d3bb02af4886/docs/bugfix/ANALYSIS_PRD_VS_IMPLEMENTATION.md +1134 -0
- package/plan/001_d3bb02af4886/docs/bugfix/GAP_ANALYSIS_SUMMARY.md +179 -0
- package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/PRP.md +629 -0
- package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/validation-report.md +214 -0
- package/plan/001_d3bb02af4886/docs/bugfix/PRP_P1M4T2S3.md +629 -0
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_PRP.md +529 -0
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_QUICK_REFERENCE.md +142 -0
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_README.md +304 -0
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_TEST_RESULTS.md +558 -0
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_VALIDATION_SUMMARY.md +256 -0
- package/plan/001_d3bb02af4886/docs/bugfix/system_context.md +346 -0
- package/plan/001_d3bb02af4886/docs/bugfix-architecture/bug_analysis.md +415 -0
- package/plan/001_d3bb02af4886/docs/bugfix-architecture/implementation_patterns.md +489 -0
- package/plan/001_d3bb02af4886/docs/bugfix-architecture/system_context.md +218 -0
- package/plan/001_d3bb02af4886/docs/bugfix_INITIATION_SUMMARY.md +380 -0
- package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_PATTERNS.md +1923 -0
- package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_QUICK_REF.md +319 -0
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/codebase-context.md +115 -0
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/cycle-detection-algorithms.md +134 -0
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/test-patterns.md +153 -0
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/workflow-class.md +132 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_BEST_PRACTICES.md +716 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_QUICK_REF.md +186 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/GROUNDSWELL_DECORATOR_EXAMPLES.md +604 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/INDEX.md +213 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/codebase_structure.md +30 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/existing_test_pattern.md +56 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/getRootObservers_implementation.md +53 -0
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/test_conventions.md +49 -0
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/PRP.md +958 -0
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/QUICK_REFERENCE.md +339 -0
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/README.md +305 -0
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/SUMMARY.md +433 -0
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/bidirectional-tree-consistency-testing.md +1574 -0
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/test-pattern-examples.md +1014 -0
- package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_QUICK_REF.md +376 -0
- package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_RESEARCH.md +1507 -0
- package/plan/001_d3bb02af4886/docs/research/bugfix_typescript_patterns.md +949 -0
- package/plan/001_d3bb02af4886/docs/research/error-testing-research.md +619 -0
- package/plan/001_d3bb02af4886/docs/research/error_handling_patterns.md +723 -0
- package/plan/{research → 001_d3bb02af4886/docs/research/general}/introspection-security-guide.md +56 -0
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/PRP_TEMPLATE.md +460 -0
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/QUICK_REFERENCE.md +324 -0
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/README.md +175 -0
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/RESEARCH_REPORT.md +499 -0
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/SUMMARY.md +163 -0
- package/plan/bugfix/BUG_FIX_SUMMARY.md +961 -0
- package/src/__tests__/adversarial/attachChild-performance.test.ts +216 -0
- package/src/__tests__/adversarial/circular-reference.test.ts +101 -0
- package/src/__tests__/adversarial/complex-circular-reference.test.ts +139 -0
- package/src/__tests__/adversarial/concurrent-task-failures.test.ts +571 -0
- package/src/__tests__/adversarial/deep-analysis.test.ts +729 -0
- package/src/__tests__/adversarial/deep-hierarchy-stress.test.ts +213 -0
- package/src/__tests__/adversarial/e2e-prd-validation.test.ts +448 -0
- package/src/__tests__/adversarial/edge-case.test.ts +703 -0
- package/src/__tests__/adversarial/error-merge-strategy.test.ts +760 -0
- package/src/__tests__/adversarial/incremental-performance.test.ts +140 -0
- package/src/__tests__/adversarial/node-map-update-benchmarks.test.ts +457 -0
- package/src/__tests__/adversarial/observer-propagation.test.ts +487 -0
- package/src/__tests__/adversarial/parent-validation.test.ts +143 -0
- package/src/__tests__/adversarial/prd-12-2-compliance.test.ts +611 -0
- package/src/__tests__/adversarial/prd-compliance.test.ts +731 -0
- package/src/__tests__/compatibility/backward-compatibility.test.ts +1572 -0
- package/src/__tests__/helpers/index.ts +18 -0
- package/src/__tests__/helpers/tree-verification.ts +257 -0
- package/src/__tests__/integration/bidirectional-consistency.test.ts +847 -0
- package/src/__tests__/integration/observer-logging.test.ts +643 -0
- package/src/__tests__/integration/tree-mirroring.test.ts +37 -0
- package/src/__tests__/integration/workflow-reparenting.test.ts +303 -0
- package/src/__tests__/unit/context.test.ts +79 -0
- package/src/__tests__/unit/logger.test.ts +293 -0
- package/src/__tests__/unit/observable.test.ts +321 -0
- package/src/__tests__/unit/tree-debugger-incremental.test.ts +170 -0
- package/src/__tests__/unit/utils/workflow-error-utils.test.ts +209 -0
- package/src/__tests__/unit/workflow-detachChild.test.ts +100 -0
- package/src/__tests__/unit/workflow-emitEvent-childDetached.test.ts +153 -0
- package/src/__tests__/unit/workflow-isDescendantOf.test.ts +180 -0
- package/src/__tests__/unit/workflow.test.ts +277 -1
- package/src/core/agent.ts +21 -1
- package/src/core/logger.ts +27 -2
- package/src/core/workflow-context.ts +6 -4
- package/src/core/workflow.ts +252 -14
- package/src/debugger/tree-debugger.ts +52 -7
- package/src/decorators/task.ts +65 -2
- package/src/index.ts +4 -2
- package/src/types/decorators.ts +8 -1
- package/src/types/events.ts +1 -0
- package/src/utils/index.ts +1 -0
- package/src/utils/observable.ts +32 -3
- package/src/utils/workflow-error-utils.ts +56 -0
- package/tsconfig.json +1 -1
- package/llms_full.txt +0 -5890
- package/tasks.json +0 -0
- /package/plan/{backlog.json → 001_d3bb02af4886/backlog.json} +0 -0
- /package/plan/{P1P2/PRP.md → 001_d3bb02af4886/docs/PRP/P1P2-PRP.md} +0 -0
- /package/plan/{P3P4/PRP.md → 001_d3bb02af4886/docs/PRP/P3P4-PRP.md} +0 -0
- /package/plan/{P4P5/PRP.md → 001_d3bb02af4886/docs/PRP/P4P5-PRP.md} +0 -0
- /package/plan/{architecture → 001_d3bb02af4886/docs/architecture}/external_deps.md +0 -0
- /package/plan/{architecture → 001_d3bb02af4886/docs/architecture}/system_context.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_BEST_PRACTICES.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_CODE_PATTERNS.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_INTEGRATION_GUIDE.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_RESEARCH_INDEX.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/REFLECTION_INDEX.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/REFLECTION_RESEARCH_REPORT.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/RESEARCH_SUMMARY.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/anthropic-sdk.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/async-local-storage.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-code-patterns.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-decision-matrix.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-implementation-guide.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-integration-guide.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-patterns.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-quick-reference.md +0 -0
- /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/zod-schema.md +0 -0
- /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/caching-lru.md +0 -0
- /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/introspection-tools.md +0 -0
- /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/reflection-patterns.md +0 -0
- /package/plan/{P4P5/research → 001_d3bb02af4886/docs/research/P4P5}/RESEARCH_SUMMARY.md +0 -0
- /package/plan/{research → 001_d3bb02af4886/docs/research/general}/INTROSPECTION_RESEARCH_SUMMARY.md +0 -0
- /package/plan/{research → 001_d3bb02af4886/docs/research/general}/README-INTROSPECTION.md +0 -0
- /package/plan/{research → 001_d3bb02af4886/docs/research/general}/agent-introspection-patterns.md +0 -0
- /package/plan/{research → 001_d3bb02af4886/docs/research/general}/introspection-tool-examples.md +0 -0
- /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)
|