groundswell 0.0.2 → 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +26 -9
- package/dist/cache/cache-key.d.ts +86 -0
- package/dist/cache/cache-key.d.ts.map +1 -0
- package/dist/cache/cache-key.js +204 -0
- package/dist/cache/cache-key.js.map +1 -0
- package/dist/cache/cache.d.ts +104 -0
- package/dist/cache/cache.d.ts.map +1 -0
- package/dist/cache/cache.js +179 -0
- package/dist/cache/cache.js.map +1 -0
- package/{src/cache/index.ts → dist/cache/index.d.ts} +1 -1
- package/dist/cache/index.d.ts.map +1 -0
- package/dist/cache/index.js +6 -0
- package/dist/cache/index.js.map +1 -0
- package/dist/core/agent.d.ts +203 -0
- package/dist/core/agent.d.ts.map +1 -0
- package/dist/core/agent.js +833 -0
- package/dist/core/agent.js.map +1 -0
- package/{src/core/context.ts → dist/core/context.d.ts} +16 -67
- package/dist/core/context.d.ts.map +1 -0
- package/dist/core/context.js +80 -0
- package/dist/core/context.js.map +1 -0
- package/dist/core/event-tree.d.ts +72 -0
- package/dist/core/event-tree.d.ts.map +1 -0
- package/dist/core/event-tree.js +211 -0
- package/dist/core/event-tree.js.map +1 -0
- package/{src/core/factory.ts → dist/core/factory.d.ts} +6 -27
- package/dist/core/factory.d.ts.map +1 -0
- package/dist/core/factory.js +110 -0
- package/dist/core/factory.js.map +1 -0
- package/{src/core/index.ts → dist/core/index.d.ts} +2 -10
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +9 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/logger.d.ts +50 -0
- package/dist/core/logger.d.ts.map +1 -0
- package/dist/core/logger.js +91 -0
- package/dist/core/logger.js.map +1 -0
- package/dist/core/mcp-handler.d.ts +127 -0
- package/dist/core/mcp-handler.d.ts.map +1 -0
- package/dist/core/mcp-handler.js +323 -0
- package/dist/core/mcp-handler.js.map +1 -0
- package/dist/core/prompt.d.ts +80 -0
- package/dist/core/prompt.d.ts.map +1 -0
- package/dist/core/prompt.js +120 -0
- package/dist/core/prompt.js.map +1 -0
- package/dist/core/workflow-context.d.ts +61 -0
- package/dist/core/workflow-context.d.ts.map +1 -0
- package/dist/core/workflow-context.js +358 -0
- package/dist/core/workflow-context.js.map +1 -0
- package/dist/core/workflow.d.ts +543 -0
- package/dist/core/workflow.d.ts.map +1 -0
- package/dist/core/workflow.js +986 -0
- package/dist/core/workflow.js.map +1 -0
- package/dist/debugger/event-replayer.d.ts +422 -0
- package/dist/debugger/event-replayer.d.ts.map +1 -0
- package/dist/debugger/event-replayer.js +639 -0
- package/dist/debugger/event-replayer.js.map +1 -0
- package/dist/debugger/index.d.ts +2 -0
- package/dist/debugger/index.d.ts.map +1 -0
- package/{src/debugger/index.ts → dist/debugger/index.js} +1 -0
- package/dist/debugger/index.js.map +1 -0
- package/dist/debugger/tree-debugger.d.ts +240 -0
- package/dist/debugger/tree-debugger.d.ts.map +1 -0
- package/dist/debugger/tree-debugger.js +620 -0
- package/dist/debugger/tree-debugger.js.map +1 -0
- package/dist/decorators/index.d.ts +4 -0
- package/dist/decorators/index.d.ts.map +1 -0
- package/{src/decorators/index.ts → dist/decorators/index.js} +1 -0
- package/dist/decorators/index.js.map +1 -0
- package/dist/decorators/observed-state.d.ts +32 -0
- package/dist/decorators/observed-state.d.ts.map +1 -0
- package/dist/decorators/observed-state.js +79 -0
- package/dist/decorators/observed-state.js.map +1 -0
- package/dist/decorators/step.d.ts +15 -0
- package/dist/decorators/step.d.ts.map +1 -0
- package/dist/decorators/step.js +192 -0
- package/dist/decorators/step.js.map +1 -0
- package/dist/decorators/task.d.ts +50 -0
- package/dist/decorators/task.d.ts.map +1 -0
- package/dist/decorators/task.js +118 -0
- package/dist/decorators/task.js.map +1 -0
- package/dist/examples/index.d.ts +3 -0
- package/dist/examples/index.d.ts.map +1 -0
- package/{src/examples/index.ts → dist/examples/index.js} +1 -0
- package/dist/examples/index.js.map +1 -0
- package/dist/examples/tdd-orchestrator.d.ts +15 -0
- package/dist/examples/tdd-orchestrator.d.ts.map +1 -0
- package/dist/examples/tdd-orchestrator.js +121 -0
- package/dist/examples/tdd-orchestrator.js.map +1 -0
- package/dist/examples/test-cycle-workflow.d.ts +14 -0
- package/dist/examples/test-cycle-workflow.d.ts.map +1 -0
- package/dist/examples/test-cycle-workflow.js +116 -0
- package/dist/examples/test-cycle-workflow.js.map +1 -0
- package/dist/harnesses/claude-code-harness.d.ts +391 -0
- package/dist/harnesses/claude-code-harness.d.ts.map +1 -0
- package/dist/harnesses/claude-code-harness.js +1076 -0
- package/dist/harnesses/claude-code-harness.js.map +1 -0
- package/dist/harnesses/harness-registry.d.ts +440 -0
- package/dist/harnesses/harness-registry.d.ts.map +1 -0
- package/dist/harnesses/harness-registry.js +543 -0
- package/dist/harnesses/harness-registry.js.map +1 -0
- package/dist/harnesses/index.d.ts +12 -0
- package/dist/harnesses/index.d.ts.map +1 -0
- package/dist/harnesses/index.js +11 -0
- package/dist/harnesses/index.js.map +1 -0
- package/dist/harnesses/pi-harness.d.ts +219 -0
- package/dist/harnesses/pi-harness.d.ts.map +1 -0
- package/dist/harnesses/pi-harness.js +676 -0
- package/dist/harnesses/pi-harness.js.map +1 -0
- package/dist/harnesses/pi-schema-converter.d.ts +24 -0
- package/dist/harnesses/pi-schema-converter.d.ts.map +1 -0
- package/dist/harnesses/pi-schema-converter.js +81 -0
- package/dist/harnesses/pi-schema-converter.js.map +1 -0
- package/dist/harnesses/register-defaults.d.ts +24 -0
- package/dist/harnesses/register-defaults.d.ts.map +1 -0
- package/dist/harnesses/register-defaults.js +40 -0
- package/dist/harnesses/register-defaults.js.map +1 -0
- package/dist/harnesses/session-store.d.ts +201 -0
- package/dist/harnesses/session-store.d.ts.map +1 -0
- package/dist/harnesses/session-store.js +254 -0
- package/dist/harnesses/session-store.js.map +1 -0
- package/dist/index.d.ts +37 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +57 -0
- package/dist/index.js.map +1 -0
- package/dist/reflection/index.d.ts +5 -0
- package/dist/reflection/index.d.ts.map +1 -0
- package/{src/reflection/index.ts → dist/reflection/index.js} +1 -1
- package/dist/reflection/index.js.map +1 -0
- package/dist/reflection/reflection.d.ts +84 -0
- package/dist/reflection/reflection.d.ts.map +1 -0
- package/dist/reflection/reflection.js +344 -0
- package/dist/reflection/reflection.js.map +1 -0
- package/dist/tools/index.d.ts +6 -0
- package/dist/tools/index.d.ts.map +1 -0
- package/dist/tools/index.js +11 -0
- package/dist/tools/index.js.map +1 -0
- package/dist/tools/introspection.d.ts +165 -0
- package/dist/tools/introspection.d.ts.map +1 -0
- package/dist/tools/introspection.js +324 -0
- package/dist/tools/introspection.js.map +1 -0
- package/dist/types/agent.d.ts +1317 -0
- package/dist/types/agent.d.ts.map +1 -0
- package/dist/types/agent.js +423 -0
- package/dist/types/agent.js.map +1 -0
- package/dist/types/decorators.d.ts +40 -0
- package/dist/types/decorators.d.ts.map +1 -0
- package/dist/types/decorators.js +2 -0
- package/dist/types/decorators.js.map +1 -0
- package/dist/types/error-strategy.d.ts +13 -0
- package/dist/types/error-strategy.d.ts.map +1 -0
- package/dist/types/error-strategy.js +2 -0
- package/dist/types/error-strategy.js.map +1 -0
- package/dist/types/error.d.ts +20 -0
- package/dist/types/error.d.ts.map +1 -0
- package/dist/types/error.js +2 -0
- package/dist/types/error.js.map +1 -0
- package/dist/types/events.d.ts +113 -0
- package/dist/types/events.d.ts.map +1 -0
- package/dist/types/events.js +2 -0
- package/dist/types/events.js.map +1 -0
- package/dist/types/harnesses.d.ts +474 -0
- package/dist/types/harnesses.d.ts.map +1 -0
- package/dist/types/harnesses.js +2 -0
- package/dist/types/harnesses.js.map +1 -0
- package/dist/types/index.d.ts +23 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +8 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types/logging.d.ts +24 -0
- package/dist/types/logging.d.ts.map +1 -0
- package/dist/types/logging.js +2 -0
- package/dist/types/logging.js.map +1 -0
- package/dist/types/observer.d.ts +18 -0
- package/dist/types/observer.d.ts.map +1 -0
- package/dist/types/observer.js +2 -0
- package/dist/types/observer.js.map +1 -0
- package/dist/types/prompt.d.ts +31 -0
- package/dist/types/prompt.d.ts.map +1 -0
- package/dist/types/prompt.js +6 -0
- package/dist/types/prompt.js.map +1 -0
- package/dist/types/providers.d.ts +691 -0
- package/dist/types/providers.d.ts.map +1 -0
- package/dist/types/providers.js +14 -0
- package/dist/types/providers.js.map +1 -0
- package/dist/types/reflection.d.ts +96 -0
- package/dist/types/reflection.d.ts.map +1 -0
- package/dist/types/reflection.js +24 -0
- package/dist/types/reflection.js.map +1 -0
- package/dist/types/restart.d.ts +132 -0
- package/dist/types/restart.d.ts.map +1 -0
- package/dist/types/restart.js +2 -0
- package/dist/types/restart.js.map +1 -0
- package/dist/types/sdk-primitives.d.ts +118 -0
- package/dist/types/sdk-primitives.d.ts.map +1 -0
- package/dist/types/sdk-primitives.js +6 -0
- package/dist/types/sdk-primitives.js.map +1 -0
- package/{src/types/snapshot.ts → dist/types/snapshot.d.ts} +5 -5
- package/dist/types/snapshot.d.ts.map +1 -0
- package/dist/types/snapshot.js +2 -0
- package/dist/types/snapshot.js.map +1 -0
- package/dist/types/streaming.d.ts +194 -0
- package/dist/types/streaming.d.ts.map +1 -0
- package/dist/types/streaming.js +67 -0
- package/dist/types/streaming.js.map +1 -0
- package/dist/types/workflow-context.d.ts +275 -0
- package/dist/types/workflow-context.d.ts.map +1 -0
- package/dist/types/workflow-context.js +8 -0
- package/dist/types/workflow-context.js.map +1 -0
- package/dist/types/workflow.d.ts +30 -0
- package/dist/types/workflow.d.ts.map +1 -0
- package/dist/types/workflow.js +2 -0
- package/dist/types/workflow.js.map +1 -0
- package/dist/utils/agent-validation.d.ts +88 -0
- package/dist/utils/agent-validation.d.ts.map +1 -0
- package/dist/utils/agent-validation.js +87 -0
- package/dist/utils/agent-validation.js.map +1 -0
- package/dist/utils/delay.d.ts +7 -0
- package/dist/utils/delay.d.ts.map +1 -0
- package/dist/utils/delay.js +9 -0
- package/dist/utils/delay.js.map +1 -0
- package/dist/utils/harness-config.d.ts +180 -0
- package/dist/utils/harness-config.d.ts.map +1 -0
- package/dist/utils/harness-config.js +311 -0
- package/dist/utils/harness-config.js.map +1 -0
- package/dist/utils/id.d.ts +6 -0
- package/dist/utils/id.d.ts.map +1 -0
- package/dist/utils/id.js +12 -0
- package/dist/utils/id.js.map +1 -0
- package/dist/utils/index.d.ts +13 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +11 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/model-spec.d.ts +110 -0
- package/dist/utils/model-spec.d.ts.map +1 -0
- package/dist/utils/model-spec.js +149 -0
- package/dist/utils/model-spec.js.map +1 -0
- package/dist/utils/observable.d.ts +54 -0
- package/dist/utils/observable.d.ts.map +1 -0
- package/dist/utils/observable.js +82 -0
- package/dist/utils/observable.js.map +1 -0
- package/dist/utils/provider-config.d.ts +10 -0
- package/dist/utils/provider-config.d.ts.map +1 -0
- package/dist/utils/provider-config.js +10 -0
- package/dist/utils/provider-config.js.map +1 -0
- package/dist/utils/restart-analysis.d.ts +202 -0
- package/dist/utils/restart-analysis.d.ts.map +1 -0
- package/dist/utils/restart-analysis.js +426 -0
- package/dist/utils/restart-analysis.js.map +1 -0
- package/dist/utils/session-serialization.d.ts +118 -0
- package/dist/utils/session-serialization.d.ts.map +1 -0
- package/dist/utils/session-serialization.js +217 -0
- package/dist/utils/session-serialization.js.map +1 -0
- package/dist/utils/workflow-error-utils.d.ts +22 -0
- package/dist/utils/workflow-error-utils.d.ts.map +1 -0
- package/dist/utils/workflow-error-utils.js +45 -0
- package/dist/utils/workflow-error-utils.js.map +1 -0
- package/package.json +34 -5
- package/.claude/commands/subtask-planning/prp-base-create.md +0 -120
- package/.claude/commands/subtask-planning/prp-base-execute.md +0 -65
- package/.claude/commands/task-breakdown.md +0 -94
- package/.claude/settings.local.json +0 -9
- package/.claude/system_prompts/task-breakdown.md +0 -101
- package/CHANGELOG.md +0 -188
- package/PRD.md +0 -543
- package/PRPs/001-hierarchical-workflow-engine.md +0 -2438
- package/PRPs/PRDs/002-agent-prompt.md +0 -390
- package/PRPs/PRDs/003-agent-prompt.md +0 -943
- package/PRPs/PRDs/004-agent-prompt.md +0 -1136
- package/PRPs/PRDs/tasks-001.json +0 -492
- package/PRPs/README.md +0 -83
- package/PRPs/templates/prp_base.md +0 -222
- package/docs/agent.md +0 -422
- package/docs/prompt.md +0 -419
- package/docs/workflow.md +0 -600
- package/examples/README.md +0 -258
- package/examples/examples/01-basic-workflow.ts +0 -100
- package/examples/examples/02-decorator-options.ts +0 -217
- package/examples/examples/03-parent-child.ts +0 -241
- package/examples/examples/04-observers-debugger.ts +0 -340
- package/examples/examples/05-error-handling.ts +0 -387
- package/examples/examples/06-concurrent-tasks.ts +0 -352
- package/examples/examples/07-agent-loops.ts +0 -432
- package/examples/examples/08-sdk-features.ts +0 -667
- package/examples/examples/09-reflection.ts +0 -573
- package/examples/examples/10-introspection.ts +0 -550
- package/examples/examples/11-reparenting-workflows.ts +0 -269
- package/examples/index.ts +0 -147
- package/examples/utils/helpers.ts +0 -57
- package/package-lock.json +0 -2398
- package/plan/001_d3bb02af4886/TEST_RESULTS.md +0 -259
- package/plan/001_d3bb02af4886/backlog.json +0 -867
- package/plan/001_d3bb02af4886/bug_fix_tasks.json +0 -484
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S1/PRP.md +0 -488
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S2/PRP.md +0 -581
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S3/PRP.md +0 -687
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S1/PRP.md +0 -492
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/PRP.md +0 -932
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/concurrent_error_testing_patterns.md +0 -1109
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/vitest_concurrent_testing.md +0 -802
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/workflow_engine_test_references.md +0 -603
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S1/PRP.md +0 -564
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S3/PRP.md +0 -518
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S4/PRP.md +0 -1252
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/PRP.md +0 -364
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/CODEBASE_INVENTORY.md +0 -114
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/DECORATOR_DOCUMENTATION_PATTERNS.md +0 -205
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/PRD_LOCATION_ANALYSIS.md +0 -199
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/ULTRATHINK_PRP_PLAN.md +0 -134
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/PRP.md +0 -495
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/research/console_error_inventory.md +0 -435
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S2/PRP.md +0 -506
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S3/PRP.md +0 -612
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/PRP.md +0 -558
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/research/external_research.md +0 -788
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S2/PRP.md +0 -460
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S3/PRP.md +0 -454
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/PRP.md +0 -520
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/RECOMMENDATION.md +0 -417
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/external_workflow_engines_research.md +0 -760
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/security_implications_analysis.md +0 -245
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S2/PRP.md +0 -792
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/PRP.md +0 -535
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/TEST_EXECUTION_REPORT.md +0 -190
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/PRP.md +0 -654
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/TEST_FIX_REPORT.md +0 -227
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/KEY_FINDINGS.md +0 -345
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/QUICK_REFERENCE.md +0 -193
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/test_maintenance_research.md +0 -1323
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/BREAKING_CHANGES_AUDIT.md +0 -1011
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/PRP.md +0 -927
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S2/PRP.md +0 -505
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/architecture/logger_child_signature_analysis.md +0 -401
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/child_implementation_research.md +0 -142
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/test_patterns_research.md +0 -112
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/vitest_patterns_research.md +0 -159
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/PRP.md +0 -549
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/VERIFICATION_REPORT.md +0 -368
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/edge_case_analysis.md +0 -172
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/usage_inventory.md +0 -175
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S2/PRP.md +0 -696
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S4/PRP.md +0 -860
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/PRP.md +0 -1066
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01-testing-aggregated-errors.md +0 -1103
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01_typescript_error_aggregation_patterns.md +0 -789
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02-error-merge-strategy-testing-guide.md +0 -1098
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02_aggregate_error_patterns.md +0 -1037
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03-promise-allsettled-testing-patterns.md +0 -916
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03_error_merging_strategies.md +0 -1045
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/04_github_stackoverflow_examples.md +0 -890
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/05_comprehensive_summary.md +0 -822
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/INDEX.md +0 -668
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/QUICK_REFERENCE.md +0 -706
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/README.md +0 -265
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/RESEARCH_REPORT.md +0 -655
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S4/research/vitest_testing_patterns.md +0 -1103
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T3S2/PRP.md +0 -426
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/PRP.md +0 -506
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/QUICK_REFERENCE.md +0 -114
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/RESEARCH_SUMMARY.md +0 -316
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/vitest_observer_error_logging_best_practices.md +0 -754
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S3/PRP.md +0 -612
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/PRP.md +0 -719
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/README.md +0 -215
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/analysis.md +0 -765
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S3/PRP.md +0 -718
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/DECISION.md +0 -149
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/PRP.md +0 -470
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/ULTRATHINK_PLAN.md +0 -332
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/codebase_workflow_name_analysis.md +0 -167
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/external_best_practices.md +0 -265
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/validation_patterns.md +0 -273
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S1/workflow_engine_ancestry_api_research.md +0 -760
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S3-PRP.md +0 -434
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S1/PRP.md +0 -717
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/PRP.md +0 -472
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/VALIDATION_REPORT.md +0 -125
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/research/ULTRATHINK_PRP_PLAN.md +0 -301
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/error-logging-best-practices.md +0 -1170
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/research_typescript_partial_and_overloads.md +0 -940
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-quick-reference.md +0 -151
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-research.md +0 -650
- package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/prd_snapshot.md +0 -259
- package/plan/001_d3bb02af4886/bugfix/P1M1T1S1/PRP.md +0 -457
- package/plan/001_d3bb02af4886/bugfix/RESEARCH_SUMMARY.md +0 -346
- package/plan/001_d3bb02af4886/bugfix/architecture/codebase_structure.md +0 -311
- package/plan/001_d3bb02af4886/bugfix/architecture/concurrent_execution_best_practices.md +0 -1565
- package/plan/001_d3bb02af4886/bugfix/architecture/error_handling_patterns.md +0 -288
- package/plan/001_d3bb02af4886/bugfix/architecture/promise_all_analysis.md +0 -741
- package/plan/001_d3bb02af4886/docs/PRP/P1M1T1S4-functional-workflow-error-state-capture-test.md +0 -652
- package/plan/001_d3bb02af4886/docs/PRP/P1P2-PRP.md +0 -527
- package/plan/001_d3bb02af4886/docs/PRP/P3P4-PRP.md +0 -1388
- package/plan/001_d3bb02af4886/docs/PRP/P4P5-PRP.md +0 -1136
- package/plan/001_d3bb02af4886/docs/PRP/PRP.md +0 -527
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S1-PRP.md +0 -415
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S2-PRP.md +0 -378
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S4-PRP.md +0 -713
- package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M2T1S4-PRP.md +0 -370
- package/plan/001_d3bb02af4886/docs/PRP_P1M3T1S3.md +0 -499
- package/plan/001_d3bb02af4886/docs/TEST_RESULTS.md +0 -230
- package/plan/001_d3bb02af4886/docs/architecture/external_deps.md +0 -358
- package/plan/001_d3bb02af4886/docs/architecture/system_context.md +0 -242
- package/plan/001_d3bb02af4886/docs/bugfix/ANALYSIS_PRD_VS_IMPLEMENTATION.md +0 -1134
- package/plan/001_d3bb02af4886/docs/bugfix/GAP_ANALYSIS_SUMMARY.md +0 -179
- package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/PRP.md +0 -629
- package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/validation-report.md +0 -214
- package/plan/001_d3bb02af4886/docs/bugfix/PRP_P1M4T2S3.md +0 -629
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_PRP.md +0 -529
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_QUICK_REFERENCE.md +0 -142
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_README.md +0 -304
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_TEST_RESULTS.md +0 -558
- package/plan/001_d3bb02af4886/docs/bugfix/bugfix_VALIDATION_SUMMARY.md +0 -256
- package/plan/001_d3bb02af4886/docs/bugfix/system_context.md +0 -346
- package/plan/001_d3bb02af4886/docs/bugfix-architecture/bug_analysis.md +0 -415
- package/plan/001_d3bb02af4886/docs/bugfix-architecture/implementation_patterns.md +0 -489
- package/plan/001_d3bb02af4886/docs/bugfix-architecture/system_context.md +0 -218
- package/plan/001_d3bb02af4886/docs/bugfix_INITIATION_SUMMARY.md +0 -380
- package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_PATTERNS.md +0 -1923
- package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_QUICK_REF.md +0 -319
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/codebase-context.md +0 -115
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/cycle-detection-algorithms.md +0 -134
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/test-patterns.md +0 -153
- package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/workflow-class.md +0 -132
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_BEST_PRACTICES.md +0 -716
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_QUICK_REF.md +0 -186
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/GROUNDSWELL_DECORATOR_EXAMPLES.md +0 -604
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/INDEX.md +0 -213
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/codebase_structure.md +0 -30
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/existing_test_pattern.md +0 -56
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/getRootObservers_implementation.md +0 -53
- package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/test_conventions.md +0 -49
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/PRP.md +0 -958
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/QUICK_REFERENCE.md +0 -339
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/README.md +0 -305
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/SUMMARY.md +0 -433
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/bidirectional-tree-consistency-testing.md +0 -1574
- package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/test-pattern-examples.md +0 -1014
- package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_BEST_PRACTICES.md +0 -1929
- package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_CODE_PATTERNS.md +0 -857
- package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_INTEGRATION_GUIDE.md +0 -738
- package/plan/001_d3bb02af4886/docs/research/P1P2/LRU_CACHE_RESEARCH_INDEX.md +0 -424
- package/plan/001_d3bb02af4886/docs/research/P1P2/REFLECTION_INDEX.md +0 -291
- package/plan/001_d3bb02af4886/docs/research/P1P2/REFLECTION_RESEARCH_REPORT.md +0 -1342
- package/plan/001_d3bb02af4886/docs/research/P1P2/RESEARCH_SUMMARY.md +0 -342
- package/plan/001_d3bb02af4886/docs/research/P1P2/anthropic-sdk.md +0 -174
- package/plan/001_d3bb02af4886/docs/research/P1P2/async-local-storage.md +0 -200
- package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-code-patterns.md +0 -1205
- package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-decision-matrix.md +0 -421
- package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-implementation-guide.md +0 -1341
- package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-integration-guide.md +0 -834
- package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-patterns.md +0 -1468
- package/plan/001_d3bb02af4886/docs/research/P1P2/reflection-quick-reference.md +0 -558
- package/plan/001_d3bb02af4886/docs/research/P1P2/zod-schema.md +0 -152
- package/plan/001_d3bb02af4886/docs/research/P3P4/caching-lru.md +0 -116
- package/plan/001_d3bb02af4886/docs/research/P3P4/introspection-tools.md +0 -177
- package/plan/001_d3bb02af4886/docs/research/P3P4/reflection-patterns.md +0 -117
- package/plan/001_d3bb02af4886/docs/research/P4P5/RESEARCH_SUMMARY.md +0 -151
- package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_QUICK_REF.md +0 -376
- package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_RESEARCH.md +0 -1507
- package/plan/001_d3bb02af4886/docs/research/bugfix_typescript_patterns.md +0 -949
- package/plan/001_d3bb02af4886/docs/research/error-testing-research.md +0 -619
- package/plan/001_d3bb02af4886/docs/research/error_handling_patterns.md +0 -723
- package/plan/001_d3bb02af4886/docs/research/general/INTROSPECTION_RESEARCH_SUMMARY.md +0 -378
- package/plan/001_d3bb02af4886/docs/research/general/README-INTROSPECTION.md +0 -352
- package/plan/001_d3bb02af4886/docs/research/general/agent-introspection-patterns.md +0 -1085
- package/plan/001_d3bb02af4886/docs/research/general/introspection-security-guide.md +0 -984
- package/plan/001_d3bb02af4886/docs/research/general/introspection-tool-examples.md +0 -875
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/PRP_TEMPLATE.md +0 -460
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/QUICK_REFERENCE.md +0 -324
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/README.md +0 -175
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/RESEARCH_REPORT.md +0 -499
- package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/SUMMARY.md +0 -163
- package/plan/001_d3bb02af4886/prd_snapshot.md +0 -543
- package/plan/bugfix/BUG_FIX_SUMMARY.md +0 -961
- package/scripts/generate-llms-full.ts +0 -206
- package/src/__tests__/adversarial/attachChild-performance.test.ts +0 -216
- package/src/__tests__/adversarial/circular-reference.test.ts +0 -101
- package/src/__tests__/adversarial/complex-circular-reference.test.ts +0 -139
- package/src/__tests__/adversarial/concurrent-task-failures.test.ts +0 -571
- package/src/__tests__/adversarial/deep-analysis.test.ts +0 -729
- package/src/__tests__/adversarial/deep-hierarchy-stress.test.ts +0 -213
- package/src/__tests__/adversarial/e2e-prd-validation.test.ts +0 -448
- package/src/__tests__/adversarial/edge-case.test.ts +0 -703
- package/src/__tests__/adversarial/error-merge-strategy.test.ts +0 -760
- package/src/__tests__/adversarial/incremental-performance.test.ts +0 -140
- package/src/__tests__/adversarial/node-map-update-benchmarks.test.ts +0 -457
- package/src/__tests__/adversarial/observer-propagation.test.ts +0 -487
- package/src/__tests__/adversarial/parent-validation.test.ts +0 -143
- package/src/__tests__/adversarial/prd-12-2-compliance.test.ts +0 -611
- package/src/__tests__/adversarial/prd-compliance.test.ts +0 -731
- package/src/__tests__/compatibility/backward-compatibility.test.ts +0 -1572
- package/src/__tests__/helpers/index.ts +0 -18
- package/src/__tests__/helpers/tree-verification.ts +0 -257
- package/src/__tests__/integration/agent-workflow.test.ts +0 -256
- package/src/__tests__/integration/bidirectional-consistency.test.ts +0 -847
- package/src/__tests__/integration/observer-logging.test.ts +0 -643
- package/src/__tests__/integration/tree-mirroring.test.ts +0 -151
- package/src/__tests__/integration/workflow-reparenting.test.ts +0 -303
- package/src/__tests__/unit/agent.test.ts +0 -169
- package/src/__tests__/unit/cache-key.test.ts +0 -182
- package/src/__tests__/unit/cache.test.ts +0 -172
- package/src/__tests__/unit/context.test.ts +0 -217
- package/src/__tests__/unit/decorators.test.ts +0 -100
- package/src/__tests__/unit/introspection-tools.test.ts +0 -277
- package/src/__tests__/unit/logger.test.ts +0 -293
- package/src/__tests__/unit/observable.test.ts +0 -321
- package/src/__tests__/unit/prompt.test.ts +0 -135
- package/src/__tests__/unit/reflection.test.ts +0 -210
- package/src/__tests__/unit/tree-debugger-incremental.test.ts +0 -170
- package/src/__tests__/unit/tree-debugger.test.ts +0 -85
- package/src/__tests__/unit/utils/workflow-error-utils.test.ts +0 -209
- package/src/__tests__/unit/workflow-detachChild.test.ts +0 -100
- package/src/__tests__/unit/workflow-emitEvent-childDetached.test.ts +0 -153
- package/src/__tests__/unit/workflow-isDescendantOf.test.ts +0 -180
- package/src/__tests__/unit/workflow.test.ts +0 -357
- package/src/cache/cache-key.ts +0 -244
- package/src/cache/cache.ts +0 -236
- package/src/core/agent.ts +0 -593
- package/src/core/event-tree.ts +0 -260
- package/src/core/logger.ts +0 -112
- package/src/core/mcp-handler.ts +0 -184
- package/src/core/prompt.ts +0 -150
- package/src/core/workflow-context.ts +0 -351
- package/src/core/workflow.ts +0 -540
- package/src/debugger/tree-debugger.ts +0 -255
- package/src/decorators/observed-state.ts +0 -95
- package/src/decorators/step.ts +0 -139
- package/src/decorators/task.ts +0 -159
- package/src/examples/tdd-orchestrator.ts +0 -65
- package/src/examples/test-cycle-workflow.ts +0 -64
- package/src/index.ts +0 -142
- package/src/reflection/reflection.ts +0 -407
- package/src/tools/index.ts +0 -36
- package/src/tools/introspection.ts +0 -464
- package/src/types/agent.ts +0 -90
- package/src/types/decorators.ts +0 -32
- package/src/types/error-strategy.ts +0 -13
- package/src/types/error.ts +0 -20
- package/src/types/events.ts +0 -75
- package/src/types/index.ts +0 -55
- package/src/types/logging.ts +0 -24
- package/src/types/observer.ts +0 -18
- package/src/types/prompt.ts +0 -40
- package/src/types/reflection.ts +0 -117
- package/src/types/sdk-primitives.ts +0 -128
- package/src/types/workflow-context.ts +0 -163
- package/src/types/workflow.ts +0 -37
- package/src/utils/id.ts +0 -11
- package/src/utils/index.ts +0 -4
- package/src/utils/observable.ts +0 -106
- package/src/utils/workflow-error-utils.ts +0 -56
- package/tsconfig.json +0 -22
- package/vitest.config.ts +0 -16
|
@@ -1,489 +0,0 @@
|
|
|
1
|
-
# Implementation Patterns: Tree Integrity Best Practices
|
|
2
|
-
|
|
3
|
-
## Overview
|
|
4
|
-
|
|
5
|
-
This document outlines the proven patterns and best practices for implementing tree integrity in hierarchical data structures, specifically applied to the Workflow engine bug fix.
|
|
6
|
-
|
|
7
|
-
## Core Patterns for attachChild() Fix
|
|
8
|
-
|
|
9
|
-
### Pattern 1: Parent State Validation
|
|
10
|
-
|
|
11
|
-
**Principle**: Always validate the child's current parent state before attaching.
|
|
12
|
-
|
|
13
|
-
```typescript
|
|
14
|
-
public attachChild(child: Workflow): void {
|
|
15
|
-
// Validation 1: Prevent duplicate attachment
|
|
16
|
-
if (this.children.includes(child)) {
|
|
17
|
-
throw new Error(
|
|
18
|
-
`Child '${child.node.name}' is already attached to workflow '${this.node.name}'`
|
|
19
|
-
);
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
// Validation 2: CRITICAL - Check if child already has a different parent
|
|
23
|
-
if (child.parent !== null && child.parent !== this) {
|
|
24
|
-
throw new Error(
|
|
25
|
-
`Child '${child.node.name}' already has parent '${child.parent.node.name}'. ` +
|
|
26
|
-
`A workflow can only have one parent. ` +
|
|
27
|
-
`Use detachChild() on the current parent first if you need to reparent.`
|
|
28
|
-
);
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
// Update child's parent if it's currently null
|
|
32
|
-
if (child.parent === null) {
|
|
33
|
-
child.parent = this;
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
// Add to both trees
|
|
37
|
-
this.children.push(child);
|
|
38
|
-
this.node.children.push(child.node);
|
|
39
|
-
|
|
40
|
-
// Emit event
|
|
41
|
-
this.emitEvent({
|
|
42
|
-
type: 'childAttached',
|
|
43
|
-
parentId: this.id,
|
|
44
|
-
child: child.node,
|
|
45
|
-
});
|
|
46
|
-
}
|
|
47
|
-
```
|
|
48
|
-
|
|
49
|
-
**Key Points**:
|
|
50
|
-
- Check `child.parent !== null` before allowing attachment
|
|
51
|
-
- Check `child.parent !== this` to allow re-attaching to same parent (idempotent)
|
|
52
|
-
- Provide clear error message with current parent's name
|
|
53
|
-
- Suggest solution (use detachChild first)
|
|
54
|
-
|
|
55
|
-
### Pattern 2: Circular Reference Detection
|
|
56
|
-
|
|
57
|
-
**Principle**: Prevent attaching an ancestor as a child (would create cycle).
|
|
58
|
-
|
|
59
|
-
```typescript
|
|
60
|
-
public attachChild(child: Workflow): void {
|
|
61
|
-
// ... other validations ...
|
|
62
|
-
|
|
63
|
-
// Validation 3: Prevent circular references
|
|
64
|
-
if (this.isDescendantOf(child)) {
|
|
65
|
-
throw new Error(
|
|
66
|
-
`Cannot attach child '${child.node.name}' - it is an ancestor of '${this.node.name}'. ` +
|
|
67
|
-
`This would create a circular reference.`
|
|
68
|
-
);
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
// ... rest of method ...
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
/**
|
|
75
|
-
* Check if this workflow is a descendant of another workflow
|
|
76
|
-
* Used to prevent circular references in attachChild()
|
|
77
|
-
*/
|
|
78
|
-
private isDescendantOf(ancestor: Workflow): boolean {
|
|
79
|
-
let current: Workflow | null = this;
|
|
80
|
-
const visited = new Set<Workflow>();
|
|
81
|
-
|
|
82
|
-
while (current !== null) {
|
|
83
|
-
// Cycle detection during traversal itself
|
|
84
|
-
if (visited.has(current)) {
|
|
85
|
-
throw new Error('Circular reference detected in tree structure');
|
|
86
|
-
}
|
|
87
|
-
visited.add(current);
|
|
88
|
-
|
|
89
|
-
if (current === ancestor) {
|
|
90
|
-
return true;
|
|
91
|
-
}
|
|
92
|
-
current = current.parent;
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
return false;
|
|
96
|
-
}
|
|
97
|
-
```
|
|
98
|
-
|
|
99
|
-
**Key Points**:
|
|
100
|
-
- Traverse parent chain from `this` upwards
|
|
101
|
-
- If we encounter `ancestor`, it means `ancestor` is above `this` in tree
|
|
102
|
-
- Attaching it would create a cycle
|
|
103
|
-
- Include cycle detection in traversal itself (defensive)
|
|
104
|
-
|
|
105
|
-
### Pattern 3: Bidirectional Tree Synchronization
|
|
106
|
-
|
|
107
|
-
**Principle**: Always update both the workflow tree and node tree together.
|
|
108
|
-
|
|
109
|
-
```typescript
|
|
110
|
-
public attachChild(child: Workflow): void {
|
|
111
|
-
// ... validations ...
|
|
112
|
-
|
|
113
|
-
// Update workflow tree
|
|
114
|
-
if (child.parent === null) {
|
|
115
|
-
child.parent = this;
|
|
116
|
-
}
|
|
117
|
-
this.children.push(child);
|
|
118
|
-
|
|
119
|
-
// Update node tree (must mirror workflow tree)
|
|
120
|
-
this.node.children.push(child.node);
|
|
121
|
-
|
|
122
|
-
// Emit event for observers
|
|
123
|
-
this.emitEvent({
|
|
124
|
-
type: 'childAttached',
|
|
125
|
-
parentId: this.id,
|
|
126
|
-
child: child.node,
|
|
127
|
-
});
|
|
128
|
-
}
|
|
129
|
-
```
|
|
130
|
-
|
|
131
|
-
**Key Points**:
|
|
132
|
-
- Both trees must be updated atomically (in same method)
|
|
133
|
-
- Order matters: update child.parent, then this.children, then this.node.children
|
|
134
|
-
- Event emission happens after both trees are updated
|
|
135
|
-
- This ensures "1:1 tree mirror" requirement from PRD
|
|
136
|
-
|
|
137
|
-
### Pattern 4: Detach Child for Reparenting
|
|
138
|
-
|
|
139
|
-
**Principle**: Provide a method to properly remove children for reparenting scenarios.
|
|
140
|
-
|
|
141
|
-
```typescript
|
|
142
|
-
/**
|
|
143
|
-
* Detach a child workflow from this parent
|
|
144
|
-
* @throws {Error} If child is not attached to this workflow
|
|
145
|
-
*/
|
|
146
|
-
public detachChild(child: Workflow): void {
|
|
147
|
-
const index = this.children.indexOf(child);
|
|
148
|
-
if (index === -1) {
|
|
149
|
-
throw new Error(
|
|
150
|
-
`Child '${child.node.name}' is not attached to workflow '${this.node.name}'`
|
|
151
|
-
);
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
// Remove from workflow children array
|
|
155
|
-
this.children.splice(index, 1);
|
|
156
|
-
|
|
157
|
-
// Remove from node children array
|
|
158
|
-
const nodeIndex = this.node.children.indexOf(child.node);
|
|
159
|
-
if (nodeIndex !== -1) {
|
|
160
|
-
this.node.children.splice(nodeIndex, 1);
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
// Clear child's parent reference
|
|
164
|
-
child.parent = null;
|
|
165
|
-
|
|
166
|
-
// Emit detached event for observers
|
|
167
|
-
this.emitEvent({
|
|
168
|
-
type: 'childDetached',
|
|
169
|
-
parentId: this.id,
|
|
170
|
-
childId: child.id,
|
|
171
|
-
});
|
|
172
|
-
}
|
|
173
|
-
```
|
|
174
|
-
|
|
175
|
-
**Key Points**:
|
|
176
|
-
- Validate child is actually attached before removing
|
|
177
|
-
- Remove from both trees (workflow and node)
|
|
178
|
-
- Clear child.parent reference
|
|
179
|
-
- Emit event so observers can update
|
|
180
|
-
- Enables reparenting workflow: `oldParent.detachChild(child); newParent.attachChild(child);`
|
|
181
|
-
|
|
182
|
-
## Observer Pattern Patterns
|
|
183
|
-
|
|
184
|
-
### Pattern 5: Root-Following with Validation
|
|
185
|
-
|
|
186
|
-
**Principle**: Observer propagation relies on getRoot() working correctly.
|
|
187
|
-
|
|
188
|
-
```typescript
|
|
189
|
-
/**
|
|
190
|
-
* Get the root workflow with robust cycle detection
|
|
191
|
-
*/
|
|
192
|
-
protected getRoot(): Workflow {
|
|
193
|
-
const visited = new Set<Workflow>();
|
|
194
|
-
let current: Workflow | null = this;
|
|
195
|
-
|
|
196
|
-
while (current) {
|
|
197
|
-
// Cycle detection - prevents infinite loops
|
|
198
|
-
if (visited.has(current)) {
|
|
199
|
-
throw new Error(
|
|
200
|
-
`Circular parent-child relationship detected at workflow '${current.node.name}'. ` +
|
|
201
|
-
`Tree integrity violated.`
|
|
202
|
-
);
|
|
203
|
-
}
|
|
204
|
-
visited.add(current);
|
|
205
|
-
|
|
206
|
-
// Move to parent
|
|
207
|
-
current = current.parent;
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
// Return the last non-null node (the root)
|
|
211
|
-
return Array.from(visited).pop()!;
|
|
212
|
-
}
|
|
213
|
-
```
|
|
214
|
-
|
|
215
|
-
**Key Points**:
|
|
216
|
-
- With attachChild() fix, this method now works correctly
|
|
217
|
-
- Child can only have one parent, so parent chain is deterministic
|
|
218
|
-
- Cycle detection is still important for edge cases
|
|
219
|
-
- Returns the root workflow that has observers array
|
|
220
|
-
|
|
221
|
-
### Pattern 6: Observer Error Isolation
|
|
222
|
-
|
|
223
|
-
**Principle**: Observer errors should not crash the workflow execution.
|
|
224
|
-
|
|
225
|
-
```typescript
|
|
226
|
-
public emitEvent(event: WorkflowEvent): void {
|
|
227
|
-
// Add to this node's event log
|
|
228
|
-
this.node.events.push(event);
|
|
229
|
-
|
|
230
|
-
// Get root with validation
|
|
231
|
-
const root = this.getRoot();
|
|
232
|
-
|
|
233
|
-
// Notify all observers with error isolation
|
|
234
|
-
for (const obs of root.observers) {
|
|
235
|
-
try {
|
|
236
|
-
obs.onEvent(event);
|
|
237
|
-
|
|
238
|
-
// Also notify tree changed for tree update events
|
|
239
|
-
if (event.type === 'treeUpdated' || event.type === 'childAttached' || event.type === 'childDetached') {
|
|
240
|
-
obs.onTreeChanged(root.node);
|
|
241
|
-
}
|
|
242
|
-
} catch (err) {
|
|
243
|
-
// Isolate observer errors to prevent cascading failures
|
|
244
|
-
console.error('Observer onEvent error:', err);
|
|
245
|
-
|
|
246
|
-
// Optionally emit error event
|
|
247
|
-
this.emitEvent({
|
|
248
|
-
type: 'error',
|
|
249
|
-
node: this.node,
|
|
250
|
-
error: {
|
|
251
|
-
message: err instanceof Error ? err.message : 'Observer error',
|
|
252
|
-
original: err,
|
|
253
|
-
workflowId: this.id,
|
|
254
|
-
stack: err instanceof Error ? err.stack : undefined,
|
|
255
|
-
state: null,
|
|
256
|
-
logs: [],
|
|
257
|
-
},
|
|
258
|
-
});
|
|
259
|
-
}
|
|
260
|
-
}
|
|
261
|
-
}
|
|
262
|
-
```
|
|
263
|
-
|
|
264
|
-
**Key Points**:
|
|
265
|
-
- Wrap each observer callback in try-catch
|
|
266
|
-
- Log errors but continue processing other observers
|
|
267
|
-
- This pattern is already implemented and should be preserved
|
|
268
|
-
- Add childDetached to tree update events
|
|
269
|
-
|
|
270
|
-
## Testing Patterns
|
|
271
|
-
|
|
272
|
-
### Pattern 7: Tree Integrity Test Suite
|
|
273
|
-
|
|
274
|
-
```typescript
|
|
275
|
-
describe('Tree Integrity Tests', () => {
|
|
276
|
-
describe('attachChild() Validation', () => {
|
|
277
|
-
it('should prevent attaching child with existing parent', () => {
|
|
278
|
-
const parent1 = new Workflow('Parent1');
|
|
279
|
-
const parent2 = new Workflow('Parent2');
|
|
280
|
-
const child = new Workflow('Child', parent1);
|
|
281
|
-
|
|
282
|
-
// Should throw
|
|
283
|
-
expect(() => parent2.attachChild(child)).toThrow(/already has a parent/);
|
|
284
|
-
});
|
|
285
|
-
|
|
286
|
-
it('should allow attaching child with null parent', () => {
|
|
287
|
-
const parent = new Workflow('Parent');
|
|
288
|
-
const child = new Workflow('Child'); // No parent
|
|
289
|
-
|
|
290
|
-
parent.attachChild(child);
|
|
291
|
-
|
|
292
|
-
expect(child.parent).toBe(parent);
|
|
293
|
-
expect(parent.children).toContain(child);
|
|
294
|
-
});
|
|
295
|
-
|
|
296
|
-
it('should prevent circular references', () => {
|
|
297
|
-
const root = new Workflow('Root');
|
|
298
|
-
const child = new Workflow('Child', root);
|
|
299
|
-
|
|
300
|
-
// Try to create cycle
|
|
301
|
-
expect(() => {
|
|
302
|
-
child.attachChild(root as any);
|
|
303
|
-
}).toThrow(/circular|cycle/i);
|
|
304
|
-
});
|
|
305
|
-
|
|
306
|
-
it('should prevent duplicate children', () => {
|
|
307
|
-
const parent = new Workflow('Parent');
|
|
308
|
-
const child = new Workflow('Child', parent);
|
|
309
|
-
|
|
310
|
-
// Try to attach again
|
|
311
|
-
expect(() => parent.attachChild(child)).toThrow(/already attached/);
|
|
312
|
-
});
|
|
313
|
-
});
|
|
314
|
-
|
|
315
|
-
describe('Observer Propagation', () => {
|
|
316
|
-
it('should propagate events from child to root observers', async () => {
|
|
317
|
-
const events: WorkflowEvent[] = [];
|
|
318
|
-
|
|
319
|
-
const root = new Workflow('Root');
|
|
320
|
-
root.addObserver({
|
|
321
|
-
onLog: () => {},
|
|
322
|
-
onEvent: (e) => events.push(e),
|
|
323
|
-
onStateUpdated: () => {},
|
|
324
|
-
onTreeChanged: () => {},
|
|
325
|
-
});
|
|
326
|
-
|
|
327
|
-
const child = new Workflow('Child', root);
|
|
328
|
-
child.emitEvent({ type: 'stepStart', node: child.node, step: 'test' });
|
|
329
|
-
|
|
330
|
-
// Event should be received by root observer
|
|
331
|
-
const stepEvents = events.filter(e => e.type === 'stepStart');
|
|
332
|
-
expect(stepEvents.length).toBe(1);
|
|
333
|
-
});
|
|
334
|
-
|
|
335
|
-
it('should not propagate events after reparenting', () => {
|
|
336
|
-
const events1: WorkflowEvent[] = [];
|
|
337
|
-
const events2: WorkflowEvent[] = [];
|
|
338
|
-
|
|
339
|
-
const parent1 = new Workflow('Parent1');
|
|
340
|
-
const parent2 = new Workflow('Parent2');
|
|
341
|
-
|
|
342
|
-
parent1.addObserver({
|
|
343
|
-
onLog: () => {},
|
|
344
|
-
onEvent: (e) => events1.push(e),
|
|
345
|
-
onStateUpdated: () => {},
|
|
346
|
-
onTreeChanged: () => {},
|
|
347
|
-
});
|
|
348
|
-
|
|
349
|
-
parent2.addObserver({
|
|
350
|
-
onLog: () => {},
|
|
351
|
-
onEvent: (e) => events2.push(e),
|
|
352
|
-
onStateUpdated: () => {},
|
|
353
|
-
onTreeChanged: () => {},
|
|
354
|
-
});
|
|
355
|
-
|
|
356
|
-
const child = new Workflow('Child', parent1);
|
|
357
|
-
|
|
358
|
-
// Reparent
|
|
359
|
-
parent1.detachChild(child);
|
|
360
|
-
parent2.attachChild(child);
|
|
361
|
-
|
|
362
|
-
// Emit event from child
|
|
363
|
-
child.emitEvent({ type: 'stepStart', node: child.node, step: 'test' });
|
|
364
|
-
|
|
365
|
-
// Should only reach parent2's observers
|
|
366
|
-
expect(events1.filter(e => e.type === 'stepStart').length).toBe(0);
|
|
367
|
-
expect(events2.filter(e => e.type === 'stepStart').length).toBe(1);
|
|
368
|
-
});
|
|
369
|
-
});
|
|
370
|
-
|
|
371
|
-
describe('Bidirectional Consistency', () => {
|
|
372
|
-
it('should maintain consistency between workflow and node trees', () => {
|
|
373
|
-
const parent = new Workflow('Parent');
|
|
374
|
-
const child = new Workflow('Child', parent);
|
|
375
|
-
|
|
376
|
-
// Workflow tree
|
|
377
|
-
expect(child.parent).toBe(parent);
|
|
378
|
-
expect(parent.children).toContain(child);
|
|
379
|
-
|
|
380
|
-
// Node tree
|
|
381
|
-
expect(child.node.parent).toBe(parent.node);
|
|
382
|
-
expect(parent.node.children).toContain(child.node);
|
|
383
|
-
});
|
|
384
|
-
|
|
385
|
-
it('should maintain consistency after detach', () => {
|
|
386
|
-
const parent = new Workflow('Parent');
|
|
387
|
-
const child = new Workflow('Child', parent);
|
|
388
|
-
|
|
389
|
-
parent.detachChild(child);
|
|
390
|
-
|
|
391
|
-
// Workflow tree
|
|
392
|
-
expect(child.parent).toBeNull();
|
|
393
|
-
expect(parent.children).toContain(child).toBe(false);
|
|
394
|
-
|
|
395
|
-
// Node tree
|
|
396
|
-
expect(parent.node.children).toContain(child.node).toBe(false);
|
|
397
|
-
});
|
|
398
|
-
});
|
|
399
|
-
});
|
|
400
|
-
```
|
|
401
|
-
|
|
402
|
-
### Pattern 8: Adversarial Testing
|
|
403
|
-
|
|
404
|
-
```typescript
|
|
405
|
-
describe('Adversarial Tree Tests', () => {
|
|
406
|
-
it('should prevent manual parent mutation creating inconsistency', () => {
|
|
407
|
-
const parent1 = new Workflow('Parent1');
|
|
408
|
-
const parent2 = new Workflow('Parent2');
|
|
409
|
-
const child = new Workflow('Child', parent1);
|
|
410
|
-
|
|
411
|
-
// Try to manually mutate (TypeScript should prevent this at compile time)
|
|
412
|
-
// But at runtime, this could happen with 'as any'
|
|
413
|
-
expect(() => {
|
|
414
|
-
(child as any).parent = parent2;
|
|
415
|
-
}).not.toThrow();
|
|
416
|
-
|
|
417
|
-
// Even if mutation succeeds, tree should be validated
|
|
418
|
-
// This is why we need defensive checks in attachChild()
|
|
419
|
-
});
|
|
420
|
-
|
|
421
|
-
it('should handle deep hierarchies without stack overflow', () => {
|
|
422
|
-
const root = new Workflow('Root');
|
|
423
|
-
let current = root;
|
|
424
|
-
|
|
425
|
-
// Create 1000 levels deep
|
|
426
|
-
for (let i = 0; i < 1000; i++) {
|
|
427
|
-
const child = new Workflow(`Child-${i}`);
|
|
428
|
-
current.attachChild(child);
|
|
429
|
-
current = child;
|
|
430
|
-
}
|
|
431
|
-
|
|
432
|
-
// getRoot() should work without stack overflow
|
|
433
|
-
expect(current.getRoot()).toBe(root);
|
|
434
|
-
});
|
|
435
|
-
|
|
436
|
-
it('should detect complex circular references', () => {
|
|
437
|
-
const root = new Workflow('Root');
|
|
438
|
-
const child1 = new Workflow('Child1', root);
|
|
439
|
-
const child2 = new Workflow('Child2', child1);
|
|
440
|
-
const child3 = new Workflow('Child3', child2);
|
|
441
|
-
|
|
442
|
-
// Try to create cycle: child3 -> root
|
|
443
|
-
expect(() => {
|
|
444
|
-
child3.attachChild(root as any);
|
|
445
|
-
}).toThrow(/circular|cycle/i);
|
|
446
|
-
});
|
|
447
|
-
});
|
|
448
|
-
```
|
|
449
|
-
|
|
450
|
-
## Error Message Patterns
|
|
451
|
-
|
|
452
|
-
### Pattern 9: Clear, Actionable Error Messages
|
|
453
|
-
|
|
454
|
-
```typescript
|
|
455
|
-
// Bad error message
|
|
456
|
-
throw new Error('Invalid parent');
|
|
457
|
-
|
|
458
|
-
// Good error message
|
|
459
|
-
throw new Error(
|
|
460
|
-
`Child '${child.node.name}' already has parent '${child.parent.node.name}'. ` +
|
|
461
|
-
`A workflow can only have one parent. ` +
|
|
462
|
-
`Use detachChild() on '${child.parent.node.name}' first if you need to reparent.`
|
|
463
|
-
);
|
|
464
|
-
```
|
|
465
|
-
|
|
466
|
-
**Components of a Good Error Message**:
|
|
467
|
-
1. **What happened**: Child already has a parent
|
|
468
|
-
2. **Who is involved**: Names of the workflows
|
|
469
|
-
3. **Why it's a problem**: A workflow can only have one parent
|
|
470
|
-
4. **How to fix it**: Use detachChild() first
|
|
471
|
-
|
|
472
|
-
## Implementation Checklist
|
|
473
|
-
|
|
474
|
-
- [ ] Add parent validation to attachChild()
|
|
475
|
-
- [ ] Add circular reference detection to attachChild()
|
|
476
|
-
- [ ] Implement isDescendantOf() helper method
|
|
477
|
-
- [ ] Implement detachChild() method
|
|
478
|
-
- [ ] Update emitEvent() to handle childDetached events
|
|
479
|
-
- [ ] Add tree integrity tests
|
|
480
|
-
- [ ] Add observer propagation tests
|
|
481
|
-
- [ ] Add adversarial tests
|
|
482
|
-
- [ ] Verify all 241 existing tests still pass
|
|
483
|
-
- [ ] Update documentation if needed
|
|
484
|
-
|
|
485
|
-
## References
|
|
486
|
-
|
|
487
|
-
- DOM Tree API: https://dom.spec.whatwg.org/#concept-tree-parent
|
|
488
|
-
- React Fiber Tree: https://github.com/facebook/react/blob/main/packages/react-reconciler/src/ReactFiber.js
|
|
489
|
-
- TypeScript Private Fields: https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-8.html#ecmascript-private-fields
|
|
@@ -1,218 +0,0 @@
|
|
|
1
|
-
# System Context: Hierarchical Workflow Engine
|
|
2
|
-
|
|
3
|
-
## Project Overview
|
|
4
|
-
|
|
5
|
-
**Project Name**: Hierarchical Workflow Engine
|
|
6
|
-
**Technology Stack**: TypeScript, Vitest
|
|
7
|
-
**Current Status**: Production-ready with critical bug
|
|
8
|
-
**Test Coverage**: 241 tests across 16 test files (all passing)
|
|
9
|
-
|
|
10
|
-
## Core Architecture
|
|
11
|
-
|
|
12
|
-
### 1. Dual Tree Representation
|
|
13
|
-
|
|
14
|
-
The workflow engine maintains two synchronized tree structures:
|
|
15
|
-
|
|
16
|
-
1. **Workflow Instance Tree**: Runtime objects with parent/child relationships
|
|
17
|
-
2. **Node Tree**: Serializable data structure for logs and events
|
|
18
|
-
|
|
19
|
-
```
|
|
20
|
-
Workflow Instance Tree Node Tree
|
|
21
|
-
====================== =========
|
|
22
|
-
Workflow (parent) WorkflowNode (parent)
|
|
23
|
-
├─ children: Workflow[] ├─ children: WorkflowNode[]
|
|
24
|
-
└─ parent: Workflow └─ parent: WorkflowNode
|
|
25
|
-
```
|
|
26
|
-
|
|
27
|
-
**Critical Invariant**: Both trees must maintain perfect 1:1 mirroring as specified in PRD Section 12.2.
|
|
28
|
-
|
|
29
|
-
### 2. Core Components
|
|
30
|
-
|
|
31
|
-
#### Workflow Class (`src/core/workflow.ts`)
|
|
32
|
-
|
|
33
|
-
**Key Properties**:
|
|
34
|
-
- `parent: Workflow | null` - Reference to parent workflow (line 49)
|
|
35
|
-
- `children: Workflow[]` - Array of child workflows (line 52)
|
|
36
|
-
- `node: WorkflowNode` - Serializable node representation (line 102)
|
|
37
|
-
- `id: string` - Unique workflow identifier (immutable)
|
|
38
|
-
- `status: WorkflowStatus` - Current workflow status
|
|
39
|
-
- `observers: WorkflowObserver[]` - Array of observers (root-only)
|
|
40
|
-
|
|
41
|
-
**Key Methods**:
|
|
42
|
-
- `constructor(name: string, parent?: Workflow)` - Creates workflow, auto-attaches if parent provided (line 115)
|
|
43
|
-
- `attachChild(child: Workflow): void` - **BUGGY METHOD** (lines 187-201)
|
|
44
|
-
- `detachChild(child: Workflow): void` - Does not exist (needs to be added)
|
|
45
|
-
- `getRoot(): Workflow` - Traverses parent chain with cycle detection
|
|
46
|
-
- `emitEvent(event: WorkflowEvent): void` - Centralized event emission
|
|
47
|
-
- `run(): Promise<any>` - Execute workflow with @Step decorated methods
|
|
48
|
-
|
|
49
|
-
#### Observer Pattern (`src/types/observer.ts`)
|
|
50
|
-
|
|
51
|
-
**Observer Interface**:
|
|
52
|
-
```typescript
|
|
53
|
-
interface WorkflowObserver {
|
|
54
|
-
onEvent(event: WorkflowEvent): void;
|
|
55
|
-
onLog(log: WorkflowLog): void;
|
|
56
|
-
onStateUpdated(state: Record<string, unknown>): void;
|
|
57
|
-
onTreeChanged(root: WorkflowNode): void;
|
|
58
|
-
}
|
|
59
|
-
```
|
|
60
|
-
|
|
61
|
-
**Key Constraints**:
|
|
62
|
-
- Observers can **only** be attached to root workflows
|
|
63
|
-
- Events propagate via `getRootObservers()` which follows parent chain
|
|
64
|
-
- Observer errors are isolated (logged to console, don't crash execution)
|
|
65
|
-
|
|
66
|
-
#### Event System (`src/types/events.ts`)
|
|
67
|
-
|
|
68
|
-
**7 Event Types** (discriminated union):
|
|
69
|
-
1. `stepStart` - Step execution started
|
|
70
|
-
2. `stepFinish` - Step execution completed
|
|
71
|
-
3. `stateUpdated` - @ObservedState property changed
|
|
72
|
-
4. `childAttached` - Child workflow added to tree
|
|
73
|
-
5. `childDetached` - Child workflow removed from tree
|
|
74
|
-
6. `treeUpdated` - Tree structure changed
|
|
75
|
-
7. `error` - Error occurred with state snapshot
|
|
76
|
-
|
|
77
|
-
### 3. Decorator System
|
|
78
|
-
|
|
79
|
-
#### @Step Decorator (`src/decorators/step.ts`)
|
|
80
|
-
- Wraps method execution with error handling
|
|
81
|
-
- Captures state snapshots via @ObservedState
|
|
82
|
-
- Emits stepStart/stepFinish events
|
|
83
|
-
- Options: name, snapshotState, trackTiming, logStart, logFinish
|
|
84
|
-
|
|
85
|
-
#### @Task Decorator (`src/decorators/task.ts`)
|
|
86
|
-
- Decorates methods for concurrent task execution
|
|
87
|
-
- Supports parallel task execution with Promise.all()
|
|
88
|
-
|
|
89
|
-
#### @ObservedState Decorator (`src/decorators/observed-state.ts`)
|
|
90
|
-
- Tracks property changes automatically
|
|
91
|
-
- Supports redaction (redact option)
|
|
92
|
-
- Supports hiding from logs (hidden option)
|
|
93
|
-
- Emits stateUpdated events on change
|
|
94
|
-
|
|
95
|
-
### 4. Testing Infrastructure
|
|
96
|
-
|
|
97
|
-
**Test Organization**:
|
|
98
|
-
```
|
|
99
|
-
src/__tests__/
|
|
100
|
-
├── unit/ - 12 files, core functionality
|
|
101
|
-
├── integration/ - 2 files, component interaction
|
|
102
|
-
└── adversarial/ - 4 files, edge cases and bugs
|
|
103
|
-
├── edge-case.test.ts - 23 tests
|
|
104
|
-
├── deep-analysis.test.ts - 18 tests
|
|
105
|
-
├── prd-compliance.test.ts - 17 tests
|
|
106
|
-
└── e2e-prd-validation.test.ts - End-to-end validation
|
|
107
|
-
```
|
|
108
|
-
|
|
109
|
-
**Test Framework**: Vitest
|
|
110
|
-
**Total Tests**: 241 test cases
|
|
111
|
-
**Current Status**: All passing (including the buggy behavior - tests don't catch this bug)
|
|
112
|
-
|
|
113
|
-
### 5. Current Bug: attachChild() Tree Integrity Violation
|
|
114
|
-
|
|
115
|
-
**Location**: `src/core/workflow.ts:187-201`
|
|
116
|
-
|
|
117
|
-
**Current Implementation**:
|
|
118
|
-
```typescript
|
|
119
|
-
public attachChild(child: Workflow): void {
|
|
120
|
-
if (this.children.includes(child)) {
|
|
121
|
-
throw new Error('Child already attached to this workflow');
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
this.children.push(child);
|
|
125
|
-
this.node.children.push(child.node);
|
|
126
|
-
|
|
127
|
-
this.emitEvent({
|
|
128
|
-
type: 'childAttached',
|
|
129
|
-
parentId: this.id,
|
|
130
|
-
child: child.node,
|
|
131
|
-
});
|
|
132
|
-
}
|
|
133
|
-
```
|
|
134
|
-
|
|
135
|
-
**Problem**: Does not validate if `child.parent` is already set to a different workflow.
|
|
136
|
-
|
|
137
|
-
**Impact**:
|
|
138
|
-
1. Child appears in multiple parents' children arrays
|
|
139
|
-
2. child.parent only points to original parent
|
|
140
|
-
3. Observer events only propagate to original parent
|
|
141
|
-
4. getRoot() returns wrong root
|
|
142
|
-
5. Violates PRD's "1:1 tree mirror" requirement
|
|
143
|
-
|
|
144
|
-
## Architecture Constraints & Requirements
|
|
145
|
-
|
|
146
|
-
### Must Maintain
|
|
147
|
-
|
|
148
|
-
1. **Dual Tree Synchronization**: Any update must modify both workflow tree and node tree
|
|
149
|
-
2. **Observer Event Propagation**: Events must reach all root observers via parent chain
|
|
150
|
-
3. **Circular Reference Detection**: getRoot() must detect and prevent infinite loops
|
|
151
|
-
4. **Type Safety**: All TypeScript types must be maintained
|
|
152
|
-
5. **Backward Compatibility**: Existing API must not break (except buggy behavior)
|
|
153
|
-
6. **Event Emission**: All tree operations must emit appropriate events
|
|
154
|
-
7. **Immutability**: workflow.id and node references must remain immutable
|
|
155
|
-
|
|
156
|
-
### Must Add
|
|
157
|
-
|
|
158
|
-
1. **Parent Validation**: Check if child already has a different parent before attaching
|
|
159
|
-
2. **Circular Reference Prevention**: Check if child is an ancestor before attaching
|
|
160
|
-
3. **detachChild() Method**: New method to properly remove children from tree
|
|
161
|
-
4. **Comprehensive Tests**: Test the new validation and edge cases
|
|
162
|
-
|
|
163
|
-
## File Locations
|
|
164
|
-
|
|
165
|
-
**Core Files**:
|
|
166
|
-
- `/home/dustin/projects/groundswell/src/core/workflow.ts` - Main Workflow class
|
|
167
|
-
- `/home/dustin/projects/groundswell/src/core/event-tree.ts` - Tree querying
|
|
168
|
-
- `/home/dustin/projects/groundswell/src/core/logger.ts` - Logging
|
|
169
|
-
- `/home/dustin/projects/groundswell/src/core/context.ts` - Execution context
|
|
170
|
-
|
|
171
|
-
**Decorator Files**:
|
|
172
|
-
- `/home/dustin/projects/groundswell/src/decorators/step.ts`
|
|
173
|
-
- `/home/dustin/projects/groundswell/src/decorators/task.ts`
|
|
174
|
-
- `/home/dustin/projects/groundswell/src/decorators/observed-state.ts`
|
|
175
|
-
|
|
176
|
-
**Type Definition Files**:
|
|
177
|
-
- `/home/dustin/projects/groundswell/src/types/workflow.ts`
|
|
178
|
-
- `/home/dustin/projects/groundswell/src/types/observer.ts`
|
|
179
|
-
- `/home/dustin/projects/groundswell/src/types/events.ts`
|
|
180
|
-
|
|
181
|
-
**Test Files**:
|
|
182
|
-
- `/home/dustin/projects/groundswell/src/__tests__/adversarial/edge-case.test.ts` - Add new tests here
|
|
183
|
-
- All other test files in `src/__tests__/` directory
|
|
184
|
-
|
|
185
|
-
## PRD Requirements (Reference)
|
|
186
|
-
|
|
187
|
-
The implementation must comply with PRD Section 12.2 - Workflow Base Class:
|
|
188
|
-
|
|
189
|
-
> "The workflow tree should maintain consistency:
|
|
190
|
-
> - A child workflow should have exactly one parent
|
|
191
|
-
> - The `child.parent` property should always match the parent that contains it
|
|
192
|
-
> - A child should only appear in one parent's `children` array"
|
|
193
|
-
|
|
194
|
-
Additionally:
|
|
195
|
-
- PRD Section 7: Observer pattern and event propagation
|
|
196
|
-
- PRD Section 11: Tree debugger API
|
|
197
|
-
- PRD Section 4: Logging and events model (7 event types)
|
|
198
|
-
|
|
199
|
-
## Development Workflow
|
|
200
|
-
|
|
201
|
-
1. **Write Failing Test First**: Follow TDD approach
|
|
202
|
-
2. **Implement Fix**: Update attachChild() method
|
|
203
|
-
3. **Add detachChild() Method**: New method for reparenting support
|
|
204
|
-
4. **Update Tests**: Add comprehensive edge case coverage
|
|
205
|
-
5. **Verify Observer Propagation**: Ensure events reach all observers
|
|
206
|
-
6. **Check Tree Consistency**: Verify 1:1 mirror is maintained
|
|
207
|
-
7. **Run All Tests**: Ensure no regressions (241 tests)
|
|
208
|
-
|
|
209
|
-
## Success Criteria
|
|
210
|
-
|
|
211
|
-
- [ ] attachChild() validates child.parent before attaching
|
|
212
|
-
- [ ] attachChild() prevents circular references
|
|
213
|
-
- [ ] attachChild() throws clear, actionable error messages
|
|
214
|
-
- [ ] detachChild() method properly removes children
|
|
215
|
-
- [ ] All observer events propagate correctly
|
|
216
|
-
- [ ] Tree debugger shows consistent tree structure
|
|
217
|
-
- [ ] All 241+ tests pass
|
|
218
|
-
- [ ] New tests cover adversarial scenarios
|