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,100 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { Workflow, WorkflowObserver, WorkflowEvent } from '../../index.js';
|
|
3
|
+
|
|
4
|
+
class SimpleWorkflow extends Workflow {
|
|
5
|
+
async run(): Promise<string> {
|
|
6
|
+
this.setStatus('running');
|
|
7
|
+
this.setStatus('completed');
|
|
8
|
+
return 'done';
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
describe('Workflow.detachChild()', () => {
|
|
13
|
+
it('should remove child from parent.children array', () => {
|
|
14
|
+
// Arrange: Create parent with child
|
|
15
|
+
const parent = new SimpleWorkflow('Parent');
|
|
16
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
17
|
+
|
|
18
|
+
// Assert: Verify child is in parent.children
|
|
19
|
+
expect(parent.children).toContain(child);
|
|
20
|
+
|
|
21
|
+
// Act: Call detachChild (will fail - method doesn't exist)
|
|
22
|
+
parent.detachChild(child);
|
|
23
|
+
|
|
24
|
+
// Assert: Verify child removed from parent.children
|
|
25
|
+
expect(parent.children).not.toContain(child);
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
it('should clear child.parent to null', () => {
|
|
29
|
+
// Arrange: Create parent with child
|
|
30
|
+
const parent = new SimpleWorkflow('Parent');
|
|
31
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
32
|
+
|
|
33
|
+
// Assert: Verify child.parent is set
|
|
34
|
+
expect(child.parent).toBe(parent);
|
|
35
|
+
|
|
36
|
+
// Act: Call detachChild
|
|
37
|
+
parent.detachChild(child);
|
|
38
|
+
|
|
39
|
+
// Assert: Verify child.parent is null
|
|
40
|
+
expect(child.parent).toBeNull();
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
it('should remove child.node from parent.node.children array', () => {
|
|
44
|
+
// Arrange: Create parent with child
|
|
45
|
+
const parent = new SimpleWorkflow('Parent');
|
|
46
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
47
|
+
|
|
48
|
+
// Assert: Verify child.node is in parent.node.children
|
|
49
|
+
expect(parent.getNode().children).toContain(child.getNode());
|
|
50
|
+
|
|
51
|
+
// Act: Call detachChild
|
|
52
|
+
parent.detachChild(child);
|
|
53
|
+
|
|
54
|
+
// Assert: Verify child.node removed from parent.node.children
|
|
55
|
+
expect(parent.getNode().children).not.toContain(child.getNode());
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it('should emit childDetached event with correct payload', () => {
|
|
59
|
+
// Arrange: Create parent with observer
|
|
60
|
+
const parent = new SimpleWorkflow('Parent');
|
|
61
|
+
const events: WorkflowEvent[] = [];
|
|
62
|
+
|
|
63
|
+
const observer: WorkflowObserver = {
|
|
64
|
+
onLog: () => {},
|
|
65
|
+
onEvent: (event) => events.push(event),
|
|
66
|
+
onStateUpdated: () => {},
|
|
67
|
+
onTreeChanged: () => {},
|
|
68
|
+
};
|
|
69
|
+
|
|
70
|
+
parent.addObserver(observer);
|
|
71
|
+
|
|
72
|
+
// Act: Create child (triggers attachChild event), then detach
|
|
73
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
74
|
+
events.length = 0; // Clear attachChild events
|
|
75
|
+
|
|
76
|
+
parent.detachChild(child);
|
|
77
|
+
|
|
78
|
+
// Assert: Verify childDetached event was emitted
|
|
79
|
+
const detachEvent = events.find((e) => e.type === 'childDetached');
|
|
80
|
+
expect(detachEvent).toBeDefined();
|
|
81
|
+
|
|
82
|
+
// Assert: Verify event payload (with type guard for discriminated union)
|
|
83
|
+
expect(detachEvent?.type === 'childDetached' && detachEvent.parentId).toBe(parent.id);
|
|
84
|
+
expect(detachEvent?.type === 'childDetached' && detachEvent.childId).toBe(child.id);
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
it('should throw error when child is not attached to parent', () => {
|
|
88
|
+
// Arrange: Create parent and child separately (no attachment)
|
|
89
|
+
const parent = new SimpleWorkflow('Parent');
|
|
90
|
+
const child = new SimpleWorkflow('Child');
|
|
91
|
+
|
|
92
|
+
// Assert: Verify child is NOT in parent.children
|
|
93
|
+
expect(parent.children).not.toContain(child);
|
|
94
|
+
|
|
95
|
+
// Act & Assert: Calling detachChild should throw error
|
|
96
|
+
expect(() => parent.detachChild(child)).toThrow(
|
|
97
|
+
/not attached/i
|
|
98
|
+
);
|
|
99
|
+
});
|
|
100
|
+
});
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { Workflow, WorkflowObserver, WorkflowEvent } from '../../index.js';
|
|
3
|
+
|
|
4
|
+
class SimpleWorkflow extends Workflow {
|
|
5
|
+
async run(): Promise<string> {
|
|
6
|
+
this.setStatus('running');
|
|
7
|
+
this.setStatus('completed');
|
|
8
|
+
return 'done';
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
describe('emitEvent() - childDetached Tree Update Events', () => {
|
|
13
|
+
it('should call onTreeChanged() when childDetached event is emitted', () => {
|
|
14
|
+
// Arrange: Create parent with observer tracking callbacks
|
|
15
|
+
const parent = new SimpleWorkflow('Parent');
|
|
16
|
+
const events: WorkflowEvent[] = [];
|
|
17
|
+
const treeChanges: any[] = [];
|
|
18
|
+
|
|
19
|
+
const observer: WorkflowObserver = {
|
|
20
|
+
onLog: () => {},
|
|
21
|
+
onEvent: (e) => events.push(e),
|
|
22
|
+
onStateUpdated: () => {},
|
|
23
|
+
onTreeChanged: (root) => treeChanges.push(root),
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
parent.addObserver(observer);
|
|
27
|
+
|
|
28
|
+
// Act: Detach child (which emits childDetached event)
|
|
29
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
30
|
+
events.length = 0; // Clear attachChild events
|
|
31
|
+
treeChanges.length = 0; // Clear attachChild tree changes
|
|
32
|
+
|
|
33
|
+
parent.detachChild(child);
|
|
34
|
+
|
|
35
|
+
// Assert: Verify childDetached event was emitted
|
|
36
|
+
const detachEvent = events.find((e) => e.type === 'childDetached');
|
|
37
|
+
expect(detachEvent).toBeDefined();
|
|
38
|
+
expect(detachEvent?.type === 'childDetached' && detachEvent.parentId).toBe(parent.id);
|
|
39
|
+
|
|
40
|
+
// CRITICAL ASSERTION: onTreeChanged() must be called
|
|
41
|
+
expect(treeChanges.length).toBe(1);
|
|
42
|
+
expect(treeChanges[0]).toBe(parent.getNode()); // Receives root node
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('should call onEvent() before onTreeChanged() for childDetached', () => {
|
|
46
|
+
// Arrange: Track call order
|
|
47
|
+
const parent = new SimpleWorkflow('Parent');
|
|
48
|
+
const callOrder: string[] = [];
|
|
49
|
+
|
|
50
|
+
const observer: WorkflowObserver = {
|
|
51
|
+
onLog: () => {},
|
|
52
|
+
onEvent: () => callOrder.push('onEvent'),
|
|
53
|
+
onStateUpdated: () => {},
|
|
54
|
+
onTreeChanged: () => callOrder.push('onTreeChanged'),
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
parent.addObserver(observer);
|
|
58
|
+
|
|
59
|
+
// Act
|
|
60
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
61
|
+
callOrder.length = 0; // Clear attachChild call order
|
|
62
|
+
parent.detachChild(child);
|
|
63
|
+
|
|
64
|
+
// Assert: Verify order
|
|
65
|
+
expect(callOrder).toEqual(['onEvent', 'onTreeChanged']);
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
it('should include childDetached in tree update event types', () => {
|
|
69
|
+
// This test verifies the implementation at the code level
|
|
70
|
+
// Access the emitEvent method and verify it triggers onTreeChanged for childDetached
|
|
71
|
+
const workflow = new SimpleWorkflow('Test');
|
|
72
|
+
|
|
73
|
+
// Access private emitEvent method for testing
|
|
74
|
+
const emitEvent = (workflow as any).emitEvent.bind(workflow);
|
|
75
|
+
|
|
76
|
+
// Track if onTreeChanged was called
|
|
77
|
+
let onTreeChangedCalled = false;
|
|
78
|
+
const mockObserver = {
|
|
79
|
+
onLog: () => {},
|
|
80
|
+
onEvent: () => {},
|
|
81
|
+
onStateUpdated: () => {},
|
|
82
|
+
onTreeChanged: () => { onTreeChangedCalled = true; },
|
|
83
|
+
};
|
|
84
|
+
|
|
85
|
+
// Mock getRootObservers to return test observer
|
|
86
|
+
(workflow as any).getRootObservers = () => [mockObserver];
|
|
87
|
+
|
|
88
|
+
// Emit childDetached event
|
|
89
|
+
emitEvent({
|
|
90
|
+
type: 'childDetached',
|
|
91
|
+
parentId: workflow.id,
|
|
92
|
+
childId: 'test-child-id',
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
// Assert: onTreeChanged should be called
|
|
96
|
+
expect(onTreeChangedCalled).toBe(true);
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
it('should pass correct root node to onTreeChanged() after detach', () => {
|
|
100
|
+
// Arrange: Create multi-level tree
|
|
101
|
+
const root = new SimpleWorkflow('Root');
|
|
102
|
+
const parent = new SimpleWorkflow('Parent', root);
|
|
103
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
104
|
+
|
|
105
|
+
const treeChangedRoots: any[] = [];
|
|
106
|
+
|
|
107
|
+
const observer: WorkflowObserver = {
|
|
108
|
+
onLog: () => {},
|
|
109
|
+
onEvent: () => {},
|
|
110
|
+
onStateUpdated: () => {},
|
|
111
|
+
onTreeChanged: (rootNode) => treeChangedRoots.push(rootNode),
|
|
112
|
+
};
|
|
113
|
+
|
|
114
|
+
root.addObserver(observer);
|
|
115
|
+
|
|
116
|
+
// Clear events from tree construction
|
|
117
|
+
treeChangedRoots.length = 0;
|
|
118
|
+
|
|
119
|
+
// Act: Detach child from parent
|
|
120
|
+
parent.detachChild(child);
|
|
121
|
+
|
|
122
|
+
// Assert: onTreeChanged should receive root node (from root workflow)
|
|
123
|
+
expect(treeChangedRoots.length).toBe(1);
|
|
124
|
+
expect(treeChangedRoots[0]).toBe(root.getNode());
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
it('should trigger onTreeChanged for both attach and detach operations', () => {
|
|
128
|
+
// Arrange: Track tree changes
|
|
129
|
+
const parent = new SimpleWorkflow('Parent');
|
|
130
|
+
const treeChanges: any[] = [];
|
|
131
|
+
|
|
132
|
+
const observer: WorkflowObserver = {
|
|
133
|
+
onLog: () => {},
|
|
134
|
+
onEvent: () => {},
|
|
135
|
+
onStateUpdated: () => {},
|
|
136
|
+
onTreeChanged: (root) => treeChanges.push({ operation: 'treeChanged', root }),
|
|
137
|
+
};
|
|
138
|
+
|
|
139
|
+
parent.addObserver(observer);
|
|
140
|
+
|
|
141
|
+
// Act 1: Attach child
|
|
142
|
+
const child = new SimpleWorkflow('Child', parent);
|
|
143
|
+
const attachCount = treeChanges.length;
|
|
144
|
+
|
|
145
|
+
// Act 2: Detach child
|
|
146
|
+
treeChanges.length = 0; // Clear to only track detach
|
|
147
|
+
parent.detachChild(child);
|
|
148
|
+
|
|
149
|
+
// Assert: Both attach and detach should trigger onTreeChanged
|
|
150
|
+
expect(treeChanges.length).toBe(1);
|
|
151
|
+
expect(treeChanges[0].root).toBe(parent.getNode());
|
|
152
|
+
});
|
|
153
|
+
});
|
|
@@ -0,0 +1,180 @@
|
|
|
1
|
+
import { describe, it, expect } from "vitest";
|
|
2
|
+
import { Workflow } from "../../core/workflow.js";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Test suite for public isDescendantOf() API.
|
|
6
|
+
*
|
|
7
|
+
* These tests verify that the isDescendantOf() method is publicly accessible
|
|
8
|
+
* without requiring (as any) type casting, following the P1.M3.T4.S2 PRP
|
|
9
|
+
* implementation for making this method public.
|
|
10
|
+
*/
|
|
11
|
+
describe("Workflow.isDescendantOf() - Public API", () => {
|
|
12
|
+
describe("Public API Accessibility", () => {
|
|
13
|
+
it("should be publicly accessible without casting to any", () => {
|
|
14
|
+
const root = new Workflow("root");
|
|
15
|
+
const child = new Workflow("child", root);
|
|
16
|
+
|
|
17
|
+
// This should work without (as any) cast - verifies public visibility
|
|
18
|
+
const result = child.isDescendantOf(root);
|
|
19
|
+
|
|
20
|
+
expect(result).toBe(true);
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
it("should return true for direct parent relationship", () => {
|
|
24
|
+
const root = new Workflow("root");
|
|
25
|
+
const child = new Workflow("child", root);
|
|
26
|
+
|
|
27
|
+
const result = child.isDescendantOf(root);
|
|
28
|
+
|
|
29
|
+
expect(result).toBe(true);
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
it("should return true for nested descendant relationship", () => {
|
|
33
|
+
const root = new Workflow("root");
|
|
34
|
+
const level1 = new Workflow("level1", root);
|
|
35
|
+
const level2 = new Workflow("level2", level1);
|
|
36
|
+
const level3 = new Workflow("level3", level2);
|
|
37
|
+
|
|
38
|
+
expect(level1.isDescendantOf(root)).toBe(true);
|
|
39
|
+
expect(level2.isDescendantOf(root)).toBe(true);
|
|
40
|
+
expect(level3.isDescendantOf(root)).toBe(true);
|
|
41
|
+
expect(level3.isDescendantOf(level1)).toBe(true);
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
it("should return false for unrelated workflows", () => {
|
|
45
|
+
const tree1 = new Workflow("tree1");
|
|
46
|
+
const tree2 = new Workflow("tree2");
|
|
47
|
+
|
|
48
|
+
const result = tree1.isDescendantOf(tree2);
|
|
49
|
+
|
|
50
|
+
expect(result).toBe(false);
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
it("should return false when checking root against descendant", () => {
|
|
54
|
+
const root = new Workflow("root");
|
|
55
|
+
const child = new Workflow("child", root);
|
|
56
|
+
|
|
57
|
+
const result = root.isDescendantOf(child);
|
|
58
|
+
|
|
59
|
+
expect(result).toBe(false);
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it("should return false for root checking itself", () => {
|
|
63
|
+
const root = new Workflow("root");
|
|
64
|
+
|
|
65
|
+
const result = root.isDescendantOf(root);
|
|
66
|
+
|
|
67
|
+
expect(result).toBe(false);
|
|
68
|
+
});
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
describe("Use Case: Hierarchy Validation", () => {
|
|
72
|
+
it("should support validating workflow belongs to specific hierarchy", () => {
|
|
73
|
+
const productionRoot = new Workflow("production-root");
|
|
74
|
+
const stagingRoot = new Workflow("staging-root");
|
|
75
|
+
|
|
76
|
+
const prodWorkflow = new Workflow("prod-workflow", productionRoot);
|
|
77
|
+
const stagingWorkflow = new Workflow("staging-workflow", stagingRoot);
|
|
78
|
+
|
|
79
|
+
expect(prodWorkflow.isDescendantOf(productionRoot)).toBe(true);
|
|
80
|
+
expect(prodWorkflow.isDescendantOf(stagingRoot)).toBe(false);
|
|
81
|
+
expect(stagingWorkflow.isDescendantOf(stagingRoot)).toBe(true);
|
|
82
|
+
expect(stagingWorkflow.isDescendantOf(productionRoot)).toBe(false);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
describe("Use Case: Circular Reference Prevention", () => {
|
|
87
|
+
it("should support validation before attaching to prevent circular references", () => {
|
|
88
|
+
const root = new Workflow("root");
|
|
89
|
+
const child = new Workflow("child", root);
|
|
90
|
+
const grandchild = new Workflow("grandchild", child);
|
|
91
|
+
|
|
92
|
+
// grandchild should NOT be attachable to root (would create cycle)
|
|
93
|
+
expect(grandchild.isDescendantOf(root)).toBe(true);
|
|
94
|
+
|
|
95
|
+
// Unrelated workflow should be attachable
|
|
96
|
+
const unrelated = new Workflow("unrelated");
|
|
97
|
+
expect(unrelated.isDescendantOf(root)).toBe(false);
|
|
98
|
+
});
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
describe("Use Case: Conditional Logic Based on Ancestry", () => {
|
|
102
|
+
it("should support conditional execution based on hierarchy position", () => {
|
|
103
|
+
const productionRoot = new Workflow("production-root");
|
|
104
|
+
const developmentRoot = new Workflow("development-root");
|
|
105
|
+
|
|
106
|
+
const prodBatch = new Workflow("prod-batch", productionRoot);
|
|
107
|
+
const devBatch = new Workflow("dev-batch", developmentRoot);
|
|
108
|
+
|
|
109
|
+
// Simulate conditional logic
|
|
110
|
+
const isInProduction = prodBatch.isDescendantOf(productionRoot);
|
|
111
|
+
const isProductionWorkflow = devBatch.isDescendantOf(productionRoot);
|
|
112
|
+
|
|
113
|
+
expect(isInProduction).toBe(true);
|
|
114
|
+
expect(isProductionWorkflow).toBe(false);
|
|
115
|
+
});
|
|
116
|
+
});
|
|
117
|
+
|
|
118
|
+
describe("Edge Cases", () => {
|
|
119
|
+
it("should handle workflows with no parent", () => {
|
|
120
|
+
const orphan = new Workflow("orphan");
|
|
121
|
+
const root = new Workflow("root");
|
|
122
|
+
|
|
123
|
+
const result = orphan.isDescendantOf(root);
|
|
124
|
+
|
|
125
|
+
expect(result).toBe(false);
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
it("should handle sibling relationships correctly", () => {
|
|
129
|
+
const root = new Workflow("root");
|
|
130
|
+
const sibling1 = new Workflow("sibling1", root);
|
|
131
|
+
const sibling2 = new Workflow("sibling2", root);
|
|
132
|
+
|
|
133
|
+
// Siblings are not descendants of each other
|
|
134
|
+
expect(sibling1.isDescendantOf(sibling2)).toBe(false);
|
|
135
|
+
expect(sibling2.isDescendantOf(sibling1)).toBe(false);
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
it("should handle complex multi-level hierarchies", () => {
|
|
139
|
+
const root = new Workflow("root");
|
|
140
|
+
const branch1 = new Workflow("branch1", root);
|
|
141
|
+
const branch2 = new Workflow("branch2", root);
|
|
142
|
+
const leaf1a = new Workflow("leaf1a", branch1);
|
|
143
|
+
const leaf1b = new Workflow("leaf1b", branch1);
|
|
144
|
+
const leaf2a = new Workflow("leaf2a", branch2);
|
|
145
|
+
|
|
146
|
+
// All leafs should be descendants of root
|
|
147
|
+
expect(leaf1a.isDescendantOf(root)).toBe(true);
|
|
148
|
+
expect(leaf1b.isDescendantOf(root)).toBe(true);
|
|
149
|
+
expect(leaf2a.isDescendantOf(root)).toBe(true);
|
|
150
|
+
|
|
151
|
+
// Leaf1a should not be descendant of branch2
|
|
152
|
+
expect(leaf1a.isDescendantOf(branch2)).toBe(false);
|
|
153
|
+
|
|
154
|
+
// Leaf2a should not be descendant of branch1
|
|
155
|
+
expect(leaf2a.isDescendantOf(branch1)).toBe(false);
|
|
156
|
+
});
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
describe("Edge Cases: Circular Reference Detection", () => {
|
|
160
|
+
it("should throw error when circular reference is detected", () => {
|
|
161
|
+
// Arrange: Create parent and child workflows with circular reference
|
|
162
|
+
const parent = new Workflow("Parent");
|
|
163
|
+
const child = new Workflow("Child", parent);
|
|
164
|
+
const unrelated = new Workflow("Unrelated");
|
|
165
|
+
|
|
166
|
+
// Act: Create circular reference manually
|
|
167
|
+
// This simulates a bug or malicious input that creates a cycle
|
|
168
|
+
// Normal attachChild() prevents this, so we bypass normal safeguards
|
|
169
|
+
parent.parent = child;
|
|
170
|
+
|
|
171
|
+
// Assert: isDescendantOf should throw error for circular reference
|
|
172
|
+
// We use 'unrelated' as the ancestor to force traversal through the cycle
|
|
173
|
+
// If we used 'child', the method would return true immediately before
|
|
174
|
+
// detecting the cycle (since child === child at first iteration)
|
|
175
|
+
expect(() => parent.isDescendantOf(unrelated)).toThrow(
|
|
176
|
+
"Circular parent-child relationship detected",
|
|
177
|
+
);
|
|
178
|
+
});
|
|
179
|
+
});
|
|
180
|
+
});
|