groundswell 0.0.1 → 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (242) hide show
  1. package/.claude/commands/subtask-planning/prp-base-create.md +120 -0
  2. package/.claude/commands/subtask-planning/prp-base-execute.md +65 -0
  3. package/.claude/commands/task-breakdown.md +94 -0
  4. package/.claude/system_prompts/task-breakdown.md +1 -0
  5. package/CHANGELOG.md +188 -0
  6. package/PRD.md +543 -0
  7. package/README.md +99 -5
  8. package/examples/README.md +15 -1
  9. package/examples/examples/11-reparenting-workflows.ts +269 -0
  10. package/examples/index.ts +4 -0
  11. package/package-lock.json +2398 -0
  12. package/package.json +3 -1
  13. package/plan/001_d3bb02af4886/TEST_RESULTS.md +259 -0
  14. package/plan/001_d3bb02af4886/bug_fix_tasks.json +484 -0
  15. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S1/PRP.md +488 -0
  16. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S2/PRP.md +581 -0
  17. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M1T1S3/PRP.md +687 -0
  18. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S1/PRP.md +492 -0
  19. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/PRP.md +932 -0
  20. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/concurrent_error_testing_patterns.md +1109 -0
  21. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/vitest_concurrent_testing.md +802 -0
  22. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T1S3/research/workflow_engine_test_references.md +603 -0
  23. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S1/PRP.md +564 -0
  24. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S3/PRP.md +518 -0
  25. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T2S4/PRP.md +1252 -0
  26. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/PRP.md +364 -0
  27. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/CODEBASE_INVENTORY.md +114 -0
  28. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/DECORATOR_DOCUMENTATION_PATTERNS.md +205 -0
  29. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/PRD_LOCATION_ANALYSIS.md +199 -0
  30. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M2T3S1/research/ULTRATHINK_PRP_PLAN.md +134 -0
  31. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/PRP.md +495 -0
  32. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S1/research/console_error_inventory.md +435 -0
  33. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S2/PRP.md +506 -0
  34. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T1S3/PRP.md +612 -0
  35. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/PRP.md +558 -0
  36. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T2S2/research/external_research.md +788 -0
  37. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S2/PRP.md +460 -0
  38. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T3S3/PRP.md +454 -0
  39. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/PRP.md +520 -0
  40. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/RECOMMENDATION.md +417 -0
  41. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/external_workflow_engines_research.md +760 -0
  42. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S1/research/security_implications_analysis.md +245 -0
  43. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M3T4S2/PRP.md +792 -0
  44. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/PRP.md +535 -0
  45. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S1/TEST_EXECUTION_REPORT.md +190 -0
  46. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/PRP.md +654 -0
  47. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/TEST_FIX_REPORT.md +227 -0
  48. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/KEY_FINDINGS.md +345 -0
  49. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/QUICK_REFERENCE.md +193 -0
  50. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T1S2/research/test_maintenance_research.md +1323 -0
  51. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/BREAKING_CHANGES_AUDIT.md +1011 -0
  52. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S1/PRP.md +927 -0
  53. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/P1M4T3S2/PRP.md +505 -0
  54. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/architecture/logger_child_signature_analysis.md +401 -0
  55. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/child_implementation_research.md +142 -0
  56. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/test_patterns_research.md +112 -0
  57. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S3/vitest_patterns_research.md +159 -0
  58. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/PRP.md +549 -0
  59. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/VERIFICATION_REPORT.md +368 -0
  60. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/edge_case_analysis.md +172 -0
  61. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M1T1S4/usage_inventory.md +175 -0
  62. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S2/PRP.md +696 -0
  63. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T1S4/PRP.md +860 -0
  64. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/PRP.md +1066 -0
  65. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01-testing-aggregated-errors.md +1103 -0
  66. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/01_typescript_error_aggregation_patterns.md +789 -0
  67. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02-error-merge-strategy-testing-guide.md +1098 -0
  68. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/02_aggregate_error_patterns.md +1037 -0
  69. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03-promise-allsettled-testing-patterns.md +916 -0
  70. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/03_error_merging_strategies.md +1045 -0
  71. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/04_github_stackoverflow_examples.md +890 -0
  72. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/05_comprehensive_summary.md +822 -0
  73. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/INDEX.md +668 -0
  74. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/QUICK_REFERENCE.md +706 -0
  75. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/README.md +265 -0
  76. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S2/research/RESEARCH_REPORT.md +655 -0
  77. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T2S4/research/vitest_testing_patterns.md +1103 -0
  78. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M2T3S2/PRP.md +426 -0
  79. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/PRP.md +506 -0
  80. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/QUICK_REFERENCE.md +114 -0
  81. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/RESEARCH_SUMMARY.md +316 -0
  82. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S2/research/vitest_observer_error_logging_best_practices.md +754 -0
  83. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T1S3/PRP.md +612 -0
  84. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/PRP.md +719 -0
  85. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/README.md +215 -0
  86. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S1/analysis.md +765 -0
  87. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T2S3/PRP.md +718 -0
  88. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/DECISION.md +149 -0
  89. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/PRP.md +470 -0
  90. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/ULTRATHINK_PLAN.md +332 -0
  91. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/codebase_workflow_name_analysis.md +167 -0
  92. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/external_best_practices.md +265 -0
  93. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T3S1/research/validation_patterns.md +273 -0
  94. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S1/workflow_engine_ancestry_api_research.md +760 -0
  95. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M3T4S3-PRP.md +434 -0
  96. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S1/PRP.md +717 -0
  97. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/PRP.md +472 -0
  98. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/VALIDATION_REPORT.md +125 -0
  99. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/P1M4T2S2/research/ULTRATHINK_PRP_PLAN.md +301 -0
  100. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/error-logging-best-practices.md +1170 -0
  101. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/research_typescript_partial_and_overloads.md +940 -0
  102. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-quick-reference.md +151 -0
  103. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/docs/vitest-research.md +650 -0
  104. package/plan/001_d3bb02af4886/bugfix/001_e8e04329daf3/prd_snapshot.md +259 -0
  105. package/plan/001_d3bb02af4886/bugfix/P1M1T1S1/PRP.md +457 -0
  106. package/plan/001_d3bb02af4886/bugfix/RESEARCH_SUMMARY.md +346 -0
  107. package/plan/001_d3bb02af4886/bugfix/architecture/codebase_structure.md +311 -0
  108. package/plan/001_d3bb02af4886/bugfix/architecture/concurrent_execution_best_practices.md +1565 -0
  109. package/plan/001_d3bb02af4886/bugfix/architecture/error_handling_patterns.md +288 -0
  110. package/plan/001_d3bb02af4886/bugfix/architecture/promise_all_analysis.md +741 -0
  111. package/plan/001_d3bb02af4886/docs/PRP/P1M1T1S4-functional-workflow-error-state-capture-test.md +652 -0
  112. package/plan/001_d3bb02af4886/docs/PRP/PRP.md +527 -0
  113. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S1-PRP.md +415 -0
  114. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S2-PRP.md +378 -0
  115. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M1T2S4-PRP.md +713 -0
  116. package/plan/001_d3bb02af4886/docs/PRP/bugfix/P1M2T1S4-PRP.md +370 -0
  117. package/plan/001_d3bb02af4886/docs/PRP_P1M3T1S3.md +499 -0
  118. package/plan/001_d3bb02af4886/docs/TEST_RESULTS.md +230 -0
  119. package/plan/001_d3bb02af4886/docs/bugfix/ANALYSIS_PRD_VS_IMPLEMENTATION.md +1134 -0
  120. package/plan/001_d3bb02af4886/docs/bugfix/GAP_ANALYSIS_SUMMARY.md +179 -0
  121. package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/PRP.md +629 -0
  122. package/plan/001_d3bb02af4886/docs/bugfix/P1M4T2S1/validation-report.md +214 -0
  123. package/plan/001_d3bb02af4886/docs/bugfix/PRP_P1M4T2S3.md +629 -0
  124. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_PRP.md +529 -0
  125. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_QUICK_REFERENCE.md +142 -0
  126. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_README.md +304 -0
  127. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_TEST_RESULTS.md +558 -0
  128. package/plan/001_d3bb02af4886/docs/bugfix/bugfix_VALIDATION_SUMMARY.md +256 -0
  129. package/plan/001_d3bb02af4886/docs/bugfix/system_context.md +346 -0
  130. package/plan/001_d3bb02af4886/docs/bugfix-architecture/bug_analysis.md +415 -0
  131. package/plan/001_d3bb02af4886/docs/bugfix-architecture/implementation_patterns.md +489 -0
  132. package/plan/001_d3bb02af4886/docs/bugfix-architecture/system_context.md +218 -0
  133. package/plan/001_d3bb02af4886/docs/bugfix_INITIATION_SUMMARY.md +380 -0
  134. package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_PATTERNS.md +1923 -0
  135. package/plan/001_d3bb02af4886/docs/research/CYCLE_DETECTION_QUICK_REF.md +319 -0
  136. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/codebase-context.md +115 -0
  137. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/cycle-detection-algorithms.md +134 -0
  138. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/test-patterns.md +153 -0
  139. package/plan/001_d3bb02af4886/docs/research/P1M1T2S1/workflow-class.md +132 -0
  140. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_BEST_PRACTICES.md +716 -0
  141. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/DECORATOR_DOCUMENTATION_QUICK_REF.md +186 -0
  142. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/GROUNDSWELL_DECORATOR_EXAMPLES.md +604 -0
  143. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/INDEX.md +213 -0
  144. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/codebase_structure.md +30 -0
  145. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/existing_test_pattern.md +56 -0
  146. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/getRootObservers_implementation.md +53 -0
  147. package/plan/001_d3bb02af4886/docs/research/P1M2T1S4/test_conventions.md +49 -0
  148. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/PRP.md +958 -0
  149. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/QUICK_REFERENCE.md +339 -0
  150. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/README.md +305 -0
  151. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/SUMMARY.md +433 -0
  152. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/bidirectional-tree-consistency-testing.md +1574 -0
  153. package/plan/001_d3bb02af4886/docs/research/P1M3T1S4/test-pattern-examples.md +1014 -0
  154. package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_QUICK_REF.md +376 -0
  155. package/plan/001_d3bb02af4886/docs/research/PROMISE_ALLSETTLED_RESEARCH.md +1507 -0
  156. package/plan/001_d3bb02af4886/docs/research/bugfix_typescript_patterns.md +949 -0
  157. package/plan/001_d3bb02af4886/docs/research/error-testing-research.md +619 -0
  158. package/plan/001_d3bb02af4886/docs/research/error_handling_patterns.md +723 -0
  159. package/plan/{research → 001_d3bb02af4886/docs/research/general}/introspection-security-guide.md +56 -0
  160. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/PRP_TEMPLATE.md +460 -0
  161. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/QUICK_REFERENCE.md +324 -0
  162. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/README.md +175 -0
  163. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/RESEARCH_REPORT.md +499 -0
  164. package/plan/001_d3bb02af4886/docs/research/incremental-tree-map-updates/SUMMARY.md +163 -0
  165. package/plan/bugfix/BUG_FIX_SUMMARY.md +961 -0
  166. package/src/__tests__/adversarial/attachChild-performance.test.ts +216 -0
  167. package/src/__tests__/adversarial/circular-reference.test.ts +101 -0
  168. package/src/__tests__/adversarial/complex-circular-reference.test.ts +139 -0
  169. package/src/__tests__/adversarial/concurrent-task-failures.test.ts +571 -0
  170. package/src/__tests__/adversarial/deep-analysis.test.ts +729 -0
  171. package/src/__tests__/adversarial/deep-hierarchy-stress.test.ts +213 -0
  172. package/src/__tests__/adversarial/e2e-prd-validation.test.ts +448 -0
  173. package/src/__tests__/adversarial/edge-case.test.ts +703 -0
  174. package/src/__tests__/adversarial/error-merge-strategy.test.ts +760 -0
  175. package/src/__tests__/adversarial/incremental-performance.test.ts +140 -0
  176. package/src/__tests__/adversarial/node-map-update-benchmarks.test.ts +457 -0
  177. package/src/__tests__/adversarial/observer-propagation.test.ts +487 -0
  178. package/src/__tests__/adversarial/parent-validation.test.ts +143 -0
  179. package/src/__tests__/adversarial/prd-12-2-compliance.test.ts +611 -0
  180. package/src/__tests__/adversarial/prd-compliance.test.ts +731 -0
  181. package/src/__tests__/compatibility/backward-compatibility.test.ts +1572 -0
  182. package/src/__tests__/helpers/index.ts +18 -0
  183. package/src/__tests__/helpers/tree-verification.ts +257 -0
  184. package/src/__tests__/integration/bidirectional-consistency.test.ts +847 -0
  185. package/src/__tests__/integration/observer-logging.test.ts +643 -0
  186. package/src/__tests__/integration/tree-mirroring.test.ts +37 -0
  187. package/src/__tests__/integration/workflow-reparenting.test.ts +303 -0
  188. package/src/__tests__/unit/context.test.ts +79 -0
  189. package/src/__tests__/unit/logger.test.ts +293 -0
  190. package/src/__tests__/unit/observable.test.ts +321 -0
  191. package/src/__tests__/unit/tree-debugger-incremental.test.ts +170 -0
  192. package/src/__tests__/unit/utils/workflow-error-utils.test.ts +209 -0
  193. package/src/__tests__/unit/workflow-detachChild.test.ts +100 -0
  194. package/src/__tests__/unit/workflow-emitEvent-childDetached.test.ts +153 -0
  195. package/src/__tests__/unit/workflow-isDescendantOf.test.ts +180 -0
  196. package/src/__tests__/unit/workflow.test.ts +277 -1
  197. package/src/core/agent.ts +21 -1
  198. package/src/core/logger.ts +27 -2
  199. package/src/core/workflow-context.ts +6 -4
  200. package/src/core/workflow.ts +252 -14
  201. package/src/debugger/tree-debugger.ts +52 -7
  202. package/src/decorators/task.ts +65 -2
  203. package/src/index.ts +4 -2
  204. package/src/types/decorators.ts +8 -1
  205. package/src/types/events.ts +1 -0
  206. package/src/utils/index.ts +1 -0
  207. package/src/utils/observable.ts +32 -3
  208. package/src/utils/workflow-error-utils.ts +56 -0
  209. package/tsconfig.json +1 -1
  210. package/llms_full.txt +0 -5890
  211. package/tasks.json +0 -0
  212. /package/plan/{backlog.json → 001_d3bb02af4886/backlog.json} +0 -0
  213. /package/plan/{P1P2/PRP.md → 001_d3bb02af4886/docs/PRP/P1P2-PRP.md} +0 -0
  214. /package/plan/{P3P4/PRP.md → 001_d3bb02af4886/docs/PRP/P3P4-PRP.md} +0 -0
  215. /package/plan/{P4P5/PRP.md → 001_d3bb02af4886/docs/PRP/P4P5-PRP.md} +0 -0
  216. /package/plan/{architecture → 001_d3bb02af4886/docs/architecture}/external_deps.md +0 -0
  217. /package/plan/{architecture → 001_d3bb02af4886/docs/architecture}/system_context.md +0 -0
  218. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_BEST_PRACTICES.md +0 -0
  219. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_CODE_PATTERNS.md +0 -0
  220. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_INTEGRATION_GUIDE.md +0 -0
  221. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/LRU_CACHE_RESEARCH_INDEX.md +0 -0
  222. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/REFLECTION_INDEX.md +0 -0
  223. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/REFLECTION_RESEARCH_REPORT.md +0 -0
  224. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/RESEARCH_SUMMARY.md +0 -0
  225. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/anthropic-sdk.md +0 -0
  226. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/async-local-storage.md +0 -0
  227. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-code-patterns.md +0 -0
  228. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-decision-matrix.md +0 -0
  229. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-implementation-guide.md +0 -0
  230. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-integration-guide.md +0 -0
  231. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-patterns.md +0 -0
  232. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/reflection-quick-reference.md +0 -0
  233. /package/plan/{P1P2/research → 001_d3bb02af4886/docs/research/P1P2}/zod-schema.md +0 -0
  234. /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/caching-lru.md +0 -0
  235. /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/introspection-tools.md +0 -0
  236. /package/plan/{P3P4/research → 001_d3bb02af4886/docs/research/P3P4}/reflection-patterns.md +0 -0
  237. /package/plan/{P4P5/research → 001_d3bb02af4886/docs/research/P4P5}/RESEARCH_SUMMARY.md +0 -0
  238. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/INTROSPECTION_RESEARCH_SUMMARY.md +0 -0
  239. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/README-INTROSPECTION.md +0 -0
  240. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/agent-introspection-patterns.md +0 -0
  241. /package/plan/{research → 001_d3bb02af4886/docs/research/general}/introspection-tool-examples.md +0 -0
  242. /package/{PRPs/PRDs/001-hierarchical-workflow-engine.md → plan/001_d3bb02af4886/prd_snapshot.md} +0 -0
@@ -0,0 +1,802 @@
1
+ # Vitest Concurrent Testing Research for Promise.allSettled
2
+
3
+ **Research Date:** 2026-01-12
4
+ **Status:** Vitest Testing Patterns Research
5
+ **Target:** P1.M2.T1.S3 - Testing Promise.allSettled Implementation
6
+
7
+ ---
8
+
9
+ ## Table of Contents
10
+
11
+ 1. [Official Vitest Documentation](#1-official-vitest-documentation)
12
+ 2. [Async Testing Patterns in Vitest](#2-async-testing-patterns-in-vitest)
13
+ 3. [Promise.allSettled Testing Patterns](#3-promiseallsettled-testing-patterns)
14
+ 4. [Error Handling and Collection Testing](#4-error-handling-and-collection-testing)
15
+ 5. [Concurrent Operations Testing](#5-concurrent-operations-testing)
16
+ 6. [Event Emission Verification](#6-event-emission-verification)
17
+ 7. [Best Practices Summary](#7-best-practices-summary)
18
+ 8. [Real-World Patterns from Groundswell](#8-real-world-patterns-from-groundswell)
19
+
20
+ ---
21
+
22
+ ## 1. Official Vitest Documentation
23
+
24
+ ### Primary Resources
25
+
26
+ **Main Documentation**
27
+ - URL: https://vitest.dev
28
+ - Coverage: Async testing, assertions, mocking, test organization
29
+
30
+ **API Reference**
31
+ - URL: https://vitest.dev/api/
32
+ - Coverage: Complete API reference for all vitest functions
33
+
34
+ **Async Testing Guide**
35
+ - URL: https://vitest.dev/guide/async.html
36
+ - Coverage: Async/await patterns, promise testing, timeouts
37
+
38
+ **Mocking Guide**
39
+ - URL: https://vitest.dev/guide/mocking.html
40
+ - Coverage: Mock functions, spies, timers, modules
41
+
42
+ **Expect API**
43
+ - URL: https://vitest.dev/api/expect.html
44
+ - Coverage: All assertion methods including `.resolves` and `.rejects`
45
+
46
+ ### Key Vitest Features for Concurrent Testing
47
+
48
+ 1. **Native Async/Await Support**: First-class async/await in tests
49
+ 2. **Promise Assertions**: `.resolves` and `.rejects` modifiers
50
+ 3. **Mock Functions**: `vi.fn()`, `vi.spyOn()`, `vi.mocked()`
51
+ 4. **Test Isolation**: Each test runs in isolated context
52
+ 5. **Concurrent Test Execution**: `test.concurrent()` for parallel test runs
53
+
54
+ ---
55
+
56
+ ## 2. Async Testing Patterns in Vitest
57
+
58
+ ### 2.1 Basic Async Test Structure
59
+
60
+ ```typescript
61
+ import { describe, it, expect } from 'vitest';
62
+
63
+ describe('Async operations', () => {
64
+ it('should handle async/await', async () => {
65
+ const result = await fetchData();
66
+ expect(result).toBeDefined();
67
+ });
68
+
69
+ it('should handle promises', () => {
70
+ return fetchData().then(result => {
71
+ expect(result).toBeDefined();
72
+ });
73
+ });
74
+ });
75
+ ```
76
+
77
+ ### 2.2 Promise Resolution Testing
78
+
79
+ ```typescript
80
+ describe('Promise resolution', () => {
81
+ it('should resolve successfully', async () => {
82
+ await expect(Promise.resolve('success')).resolves.toBe('success');
83
+ });
84
+
85
+ it('should reject with error', async () => {
86
+ await expect(Promise.reject(new Error('failure'))).rejects.toThrow('failure');
87
+ });
88
+
89
+ it('should match error shape', async () => {
90
+ await expect(Promise.reject({ message: 'error', code: 500 }))
91
+ .rejects.toMatchObject({
92
+ message: 'error',
93
+ code: 500
94
+ });
95
+ });
96
+ });
97
+ ```
98
+
99
+ ### 2.3 Multiple Async Operations
100
+
101
+ ```typescript
102
+ describe('Multiple async operations', () => {
103
+ it('should complete multiple operations', async () => {
104
+ const results = await Promise.all([
105
+ fetchItem(1),
106
+ fetchItem(2),
107
+ fetchItem(3)
108
+ ]);
109
+
110
+ expect(results).toHaveLength(3);
111
+ expect(results[0]).toBeDefined();
112
+ });
113
+
114
+ it('should handle mixed success/failure with allSettled', async () => {
115
+ const results = await Promise.allSettled([
116
+ Promise.resolve('success'),
117
+ Promise.reject(new Error('failure')),
118
+ Promise.resolve(42)
119
+ ]);
120
+
121
+ expect(results).toHaveLength(3);
122
+ });
123
+ });
124
+ ```
125
+
126
+ ---
127
+
128
+ ## 3. Promise.allSettled Testing Patterns
129
+
130
+ ### 3.1 Basic Promise.allSettled Testing
131
+
132
+ ```typescript
133
+ import { describe, it, expect } from 'vitest';
134
+
135
+ describe('Promise.allSettled basics', () => {
136
+ it('should return all results regardless of fulfillment', async () => {
137
+ const promises = [
138
+ Promise.resolve('success'),
139
+ Promise.reject(new Error('failure')),
140
+ Promise.resolve(42)
141
+ ];
142
+
143
+ const results = await Promise.allSettled(promises);
144
+
145
+ expect(results).toHaveLength(3);
146
+ });
147
+
148
+ it('should contain status for each result', async () => {
149
+ const results = await Promise.allSettled([
150
+ Promise.resolve('value'),
151
+ Promise.reject(new Error('error'))
152
+ ]);
153
+
154
+ expect(results[0].status).toBe('fulfilled');
155
+ expect(results[1].status).toBe('rejected');
156
+ });
157
+ });
158
+ ```
159
+
160
+ ### 3.2 Type-Safe Result Testing
161
+
162
+ ```typescript
163
+ describe('Type-safe Promise.allSettled testing', () => {
164
+ it('should handle fulfilled results with type guard', async () => {
165
+ const results = await Promise.allSettled([
166
+ Promise.resolve('success'),
167
+ Promise.resolve(42)
168
+ ]);
169
+
170
+ const fulfilled = results.filter(
171
+ (r): r is PromiseFulfilledResult<unknown> => r.status === 'fulfilled'
172
+ );
173
+
174
+ expect(fulfilled).toHaveLength(2);
175
+ expect(fulfilled[0].value).toBe('success');
176
+ expect(fulfilled[1].value).toBe(42);
177
+ });
178
+
179
+ it('should handle rejected results with type guard', async () => {
180
+ const results = await Promise.allSettled([
181
+ Promise.reject(new Error('error1')),
182
+ Promise.reject(new Error('error2'))
183
+ ]);
184
+
185
+ const rejected = results.filter(
186
+ (r): r is PromiseRejectedResult => r.status === 'rejected'
187
+ );
188
+
189
+ expect(rejected).toHaveLength(2);
190
+ expect(rejected[0].reason).toBeInstanceOf(Error);
191
+ expect(rejected[0].reason.message).toBe('error1');
192
+ });
193
+ });
194
+ ```
195
+
196
+ ### 3.3 Counting Success vs Failure
197
+
198
+ ```typescript
199
+ describe('Counting results from Promise.allSettled', () => {
200
+ it('should count successful operations', async () => {
201
+ const results = await Promise.allSettled([
202
+ Promise.resolve('a'),
203
+ Promise.reject(new Error('b')),
204
+ Promise.resolve('c'),
205
+ Promise.reject(new Error('d'))
206
+ ]);
207
+
208
+ const successful = results.filter(r => r.status === 'fulfilled');
209
+ const failed = results.filter(r => r.status === 'rejected');
210
+
211
+ expect(successful).toHaveLength(2);
212
+ expect(failed).toHaveLength(2);
213
+ });
214
+
215
+ it('should calculate success rate', async () => {
216
+ const results = await Promise.allSettled([
217
+ Promise.resolve('a'),
218
+ Promise.reject(new Error('b')),
219
+ Promise.resolve('c')
220
+ ]);
221
+
222
+ const successRate = results.filter(r => r.status === 'fulfilled').length / results.length;
223
+
224
+ expect(successRate).toBeCloseTo(0.667, 2);
225
+ });
226
+ });
227
+ ```
228
+
229
+ ---
230
+
231
+ ## 4. Error Handling and Collection Testing
232
+
233
+ ### 4.1 Error Aggregation Testing
234
+
235
+ ```typescript
236
+ describe('Error aggregation with Promise.allSettled', () => {
237
+ it('should collect all errors from rejected promises', async () => {
238
+ const results = await Promise.allSettled([
239
+ Promise.reject(new Error('Error 1')),
240
+ Promise.resolve('success'),
241
+ Promise.reject(new Error('Error 2'))
242
+ ]);
243
+
244
+ const errors = results
245
+ .filter(r => r.status === 'rejected')
246
+ .map(r => (r as PromiseRejectedResult).reason);
247
+
248
+ expect(errors).toHaveLength(2);
249
+ expect(errors[0].message).toBe('Error 1');
250
+ expect(errors[1].message).toBe('Error 2');
251
+ });
252
+
253
+ it('should preserve error stack traces', async () => {
254
+ const results = await Promise.allSettled([
255
+ Promise.reject(new Error('Stack trace test'))
256
+ ]);
257
+
258
+ const rejected = results.find(r => r.status === 'rejected') as PromiseRejectedResult;
259
+
260
+ expect(rejected.reason.stack).toBeDefined();
261
+ expect(rejected.reason.stack).toContain('Stack trace test');
262
+ });
263
+ });
264
+ ```
265
+
266
+ ### 4.2 Error Message Validation
267
+
268
+ ```typescript
269
+ describe('Error message validation', () => {
270
+ it('should validate error messages', async () => {
271
+ const results = await Promise.allSettled([
272
+ Promise.reject(new Error('Validation failed')),
273
+ Promise.reject(new Error('Network error'))
274
+ ]);
275
+
276
+ const errorMessages = results
277
+ .filter(r => r.status === 'rejected')
278
+ .map(r => (r as PromiseRejectedResult).reason.message);
279
+
280
+ expect(errorMessages).toContain('Validation failed');
281
+ expect(errorMessages).toContain('Network error');
282
+ });
283
+
284
+ it('should match error patterns', async () => {
285
+ const results = await Promise.allSettled([
286
+ Promise.reject(new Error('Error: Code 500')),
287
+ Promise.reject(new Error('Error: Code 404'))
288
+ ]);
289
+
290
+ const errorMessages = results
291
+ .filter(r => r.status === 'rejected')
292
+ .map(r => (r as PromiseRejectedResult).reason.message);
293
+
294
+ errorMessages.forEach(msg => {
295
+ expect(msg).toMatch(/^Error: Code \d+$/);
296
+ });
297
+ });
298
+ });
299
+ ```
300
+
301
+ ### 4.3 Custom Error Objects
302
+
303
+ ```typescript
304
+ interface WorkflowError {
305
+ message: string;
306
+ workflowId: string;
307
+ code?: number;
308
+ }
309
+
310
+ describe('Custom error handling', () => {
311
+ it('should handle custom error objects', async () => {
312
+ const errors: WorkflowError[] = [
313
+ { message: 'Task failed', workflowId: 'wf-1', code: 500 },
314
+ { message: 'Timeout', workflowId: 'wf-2' }
315
+ ];
316
+
317
+ const results = await Promise.allSettled(
318
+ errors.map(e => Promise.reject(e))
319
+ );
320
+
321
+ const rejected = results.filter(r => r.status === 'rejected') as PromiseRejectedResult[];
322
+
323
+ expect(rejected[0].reason).toMatchObject({
324
+ message: 'Task failed',
325
+ workflowId: 'wf-1'
326
+ });
327
+ });
328
+ });
329
+ ```
330
+
331
+ ---
332
+
333
+ ## 5. Concurrent Operations Testing
334
+
335
+ ### 5.1 Testing Concurrent Execution
336
+
337
+ ```typescript
338
+ describe('Concurrent operations', () => {
339
+ it('should execute operations concurrently', async () => {
340
+ let operation1Complete = false;
341
+ let operation2Complete = false;
342
+
343
+ const results = await Promise.allSettled([
344
+ (async () => {
345
+ await delay(100);
346
+ operation1Complete = true;
347
+ return 'op1';
348
+ })(),
349
+ (async () => {
350
+ await delay(50);
351
+ operation2Complete = true;
352
+ return 'op2';
353
+ })()
354
+ ]);
355
+
356
+ // Both should complete despite different delays
357
+ expect(operation1Complete).toBe(true);
358
+ expect(operation2Complete).toBe(true);
359
+ expect(results).toHaveLength(2);
360
+ });
361
+ });
362
+ ```
363
+
364
+ ### 5.2 Testing Ordering Guarantees
365
+
366
+ ```typescript
367
+ describe('Result ordering with Promise.allSettled', () => {
368
+ it('should maintain input order regardless of completion time', async () => {
369
+ const results = await Promise.allSettled([
370
+ delay(100).then(() => 'first'),
371
+ delay(10).then(() => 'second'),
372
+ delay(50).then(() => 'third')
373
+ ]);
374
+
375
+ const values = results
376
+ .filter(r => r.status === 'fulfilled')
377
+ .map(r => (r as PromiseFulfilledResult<string>).value);
378
+
379
+ expect(values).toEqual(['first', 'second', 'third']);
380
+ });
381
+ });
382
+ ```
383
+
384
+ ### 5.3 Testing Race Conditions
385
+
386
+ ```typescript
387
+ describe('Race condition testing', () => {
388
+ it('should handle concurrent modifications safely', async () => {
389
+ const counter = { value: 0 };
390
+ const operations = 10;
391
+
392
+ const results = await Promise.allSettled(
393
+ Array.from({ length: operations }, (_, i) =>
394
+ Promise.resolve().then(() => {
395
+ counter.value += 1;
396
+ return i;
397
+ })
398
+ )
399
+ );
400
+
401
+ expect(counter.value).toBe(operations);
402
+ expect(results.filter(r => r.status === 'fulfilled')).toHaveLength(operations);
403
+ });
404
+ });
405
+ ```
406
+
407
+ ---
408
+
409
+ ## 6. Event Emission Verification
410
+
411
+ ### 6.1 Testing Event Collection
412
+
413
+ ```typescript
414
+ describe('Event emission verification', () => {
415
+ it('should collect events from concurrent operations', async () => {
416
+ const events: string[] = [];
417
+
418
+ const results = await Promise.allSettled([
419
+ emitEvent('start', events),
420
+ emitEvent('middle', events),
421
+ emitEvent('end', events)
422
+ ]);
423
+
424
+ expect(events).toHaveLength(3);
425
+ expect(events).toContain('start');
426
+ expect(events).toContain('middle');
427
+ expect(events).toContain('end');
428
+ });
429
+ });
430
+
431
+ async function emitEvent(name: string, events: string[]): Promise<void> {
432
+ await Promise.resolve();
433
+ events.push(name);
434
+ }
435
+ ```
436
+
437
+ ### 6.2 Verifying Event Ordering
438
+
439
+ ```typescript
440
+ describe('Event ordering verification', () => {
441
+ it('should maintain event order across concurrent operations', async () => {
442
+ const events: { timestamp: number; name: string }[] = [];
443
+
444
+ await Promise.allSettled([
445
+ recordEvent('event1', events),
446
+ recordEvent('event2', events),
447
+ recordEvent('event3', events)
448
+ ]);
449
+
450
+ // Verify events were recorded in order
451
+ const timestamps = events.map(e => e.timestamp);
452
+ const sortedTimestamps = [...timestamps].sort((a, b) => a - b);
453
+
454
+ expect(timestamps).toEqual(sortedTimestamps);
455
+ });
456
+ });
457
+
458
+ async function recordEvent(name: string, events: any[]): Promise<void> {
459
+ events.push({ timestamp: Date.now(), name });
460
+ await Promise.resolve();
461
+ }
462
+ ```
463
+
464
+ ### 6.3 Error Event Verification
465
+
466
+ ```typescript
467
+ describe('Error event verification', () => {
468
+ it('should emit error events for failed operations', async () => {
469
+ const errorEvents: Error[] = [];
470
+
471
+ const results = await Promise.allSettled([
472
+ Promise.resolve('success'),
473
+ Promise.reject(new Error('Operation failed')).catch(e => {
474
+ errorEvents.push(e);
475
+ throw e;
476
+ })
477
+ ]);
478
+
479
+ expect(errorEvents).toHaveLength(1);
480
+ expect(errorEvents[0].message).toBe('Operation failed');
481
+ });
482
+ });
483
+ ```
484
+
485
+ ---
486
+
487
+ ## 7. Best Practices Summary
488
+
489
+ ### 7.1 Test Organization
490
+
491
+ 1. **Group related tests with `describe` blocks**
492
+ - Organize by feature or functionality
493
+ - Use nested describes for hierarchical testing
494
+
495
+ 2. **Use descriptive test names**
496
+ - Test names should describe what is being tested
497
+ - Include the expected outcome
498
+
499
+ 3. **Follow AAA pattern** (Arrange, Act, Assert)
500
+ ```typescript
501
+ it('should handle error correctly', async () => {
502
+ // Arrange
503
+ const error = new Error('Test error');
504
+
505
+ // Act
506
+ const result = await Promise.allSettled([
507
+ Promise.reject(error)
508
+ ]);
509
+
510
+ // Assert
511
+ expect(result[0].status).toBe('rejected');
512
+ });
513
+ ```
514
+
515
+ ### 7.2 Async Testing Best Practices
516
+
517
+ 1. **Always use async/await for async operations**
518
+ ```typescript
519
+ it('good - using async/await', async () => {
520
+ const result = await fetchData();
521
+ expect(result).toBeDefined();
522
+ });
523
+ ```
524
+
525
+ 2. **Use `.resolves` and `.rejects` for promise assertions**
526
+ ```typescript
527
+ await expect(promise).resolves.toBe(value);
528
+ await expect(promise).rejects.toThrow(error);
529
+ ```
530
+
531
+ 3. **Return promises when not using async/await**
532
+ ```typescript
533
+ it('good - returning promise', () => {
534
+ return fetchData().then(result => {
535
+ expect(result).toBeDefined();
536
+ });
537
+ });
538
+ ```
539
+
540
+ ### 7.3 Mocking Best Practices
541
+
542
+ 1. **Use `vi.spyOn()` for existing functions**
543
+ ```typescript
544
+ const consoleSpy = vi.spyOn(console, 'log').mockImplementation(() => {});
545
+ // test code
546
+ consoleSpy.mockRestore();
547
+ ```
548
+
549
+ 2. **Use `vi.fn()` for new mock functions**
550
+ ```typescript
551
+ const mockFn = vi.fn().mockResolvedValue('result');
552
+ ```
553
+
554
+ 3. **Always restore mocks in cleanup**
555
+ ```typescript
556
+ afterEach(() => {
557
+ vi.restoreAllMocks();
558
+ });
559
+ ```
560
+
561
+ ### 7.4 Concurrent Testing Best Practices
562
+
563
+ 1. **Test with mixed success/failure scenarios**
564
+ ```typescript
565
+ const results = await Promise.allSettled([
566
+ Promise.resolve('success'),
567
+ Promise.reject(new Error('failure'))
568
+ ]);
569
+ ```
570
+
571
+ 2. **Verify all operations complete**
572
+ ```typescript
573
+ expect(results).toHaveLength(expectedCount);
574
+ ```
575
+
576
+ 3. **Validate error collection**
577
+ ```typescript
578
+ const errors = results
579
+ .filter(r => r.status === 'rejected')
580
+ .map(r => r.reason);
581
+
582
+ expect(errors).toHaveLength(expectedErrorCount);
583
+ ```
584
+
585
+ 4. **Use type guards for type safety**
586
+ ```typescript
587
+ const fulfilled = results.filter(
588
+ (r): r is PromiseFulfilledResult<T> => r.status === 'fulfilled'
589
+ );
590
+ ```
591
+
592
+ ---
593
+
594
+ ## 8. Real-World Patterns from Groundswell
595
+
596
+ ### 8.1 Current Test Patterns in Groundswell
597
+
598
+ Based on analysis of `/home/dustin/projects/groundswell/src/__tests__/`:
599
+
600
+ **Location:** `/home/dustin/projects/groundswell/src/__tests__/unit/decorators.test.ts`
601
+
602
+ ```typescript
603
+ describe('@Step decorator', () => {
604
+ it('should wrap errors in WorkflowError', async () => {
605
+ class FailingWorkflow extends Workflow {
606
+ @Step()
607
+ async failingStep(): Promise<void> {
608
+ throw new Error('Step failed');
609
+ }
610
+
611
+ async run(): Promise<void> {
612
+ await this.failingStep();
613
+ }
614
+ }
615
+
616
+ const wf = new FailingWorkflow();
617
+
618
+ await expect(wf.run()).rejects.toMatchObject({
619
+ message: 'Step failed',
620
+ workflowId: wf.id,
621
+ });
622
+ });
623
+ });
624
+ ```
625
+
626
+ ### 8.2 Event Testing Pattern
627
+
628
+ **Location:** `/home/dustin/projects/groundswell/src/__tests__/unit/decorators.test.ts`
629
+
630
+ ```typescript
631
+ it('should emit stepStart and stepEnd events', async () => {
632
+ const wf = new StepTestWorkflow();
633
+ const events: WorkflowEvent[] = [];
634
+
635
+ wf.addObserver({
636
+ onLog: () => {},
637
+ onEvent: (e) => events.push(e),
638
+ onStateUpdated: () => {},
639
+ onTreeChanged: () => {},
640
+ });
641
+
642
+ await wf.run();
643
+
644
+ const startEvent = events.find((e) => e.type === 'stepStart');
645
+ const endEvent = events.find((e) => e.type === 'stepEnd');
646
+
647
+ expect(startEvent).toBeDefined();
648
+ expect(endEvent).toBeDefined();
649
+ });
650
+ ```
651
+
652
+ ### 8.3 Async Error Handling Pattern
653
+
654
+ **Common pattern across Groundswell tests:**
655
+
656
+ ```typescript
657
+ await expect(workflow.run()).rejects.toThrow('Test error from step');
658
+ await expect(workflow.run()).rejects.toMatchObject({
659
+ message: 'Error message',
660
+ workflowId: workflow.id,
661
+ });
662
+ ```
663
+
664
+ ### 8.4 Vitest Configuration
665
+
666
+ **Location:** `/home/dustin/projects/groundswell/vitest.config.ts`
667
+
668
+ ```typescript
669
+ import { defineConfig } from 'vitest/config';
670
+
671
+ export default defineConfig({
672
+ test: {
673
+ include: ['src/__tests__/**/*.test.ts'],
674
+ globals: true,
675
+ },
676
+ resolve: {
677
+ alias: {
678
+ // Handle .js extensions in imports for TypeScript files
679
+ },
680
+ },
681
+ esbuild: {
682
+ target: 'node18',
683
+ },
684
+ });
685
+ ```
686
+
687
+ ---
688
+
689
+ ## 9. Testing Checklist for Promise.allSettled Implementation
690
+
691
+ ### 9.1 Core Functionality Tests
692
+
693
+ - [ ] All operations complete regardless of success/failure
694
+ - [ ] Results array matches input promise order
695
+ - [ ] Status is correctly set ('fulfilled' or 'rejected')
696
+ - [ ] Values are preserved for fulfilled promises
697
+ - [ ] Reasons are preserved for rejected promises
698
+
699
+ ### 9.2 Error Collection Tests
700
+
701
+ - [ ] All errors are collected from rejected promises
702
+ - [ ] Error objects maintain their structure
703
+ - [ ] Error stack traces are preserved
704
+ - [ ] Error messages are accessible
705
+ - [ ] Custom error objects are handled correctly
706
+
707
+ ### 9.3 Concurrent Execution Tests
708
+
709
+ - [ ] Operations run concurrently (not sequentially)
710
+ - [ ] All operations complete even if some fail
711
+ - [ ] No race conditions in result collection
712
+ - [ ] Result order is maintained regardless of completion time
713
+
714
+ ### 9.4 Integration Tests
715
+
716
+ - [ ] Event emissions occur for all operations
717
+ - [ ] Logs are collected from all operations
718
+ - [ ] State snapshots are captured for failures
719
+ - [ ] Error aggregation works in production scenarios
720
+
721
+ ### 9.5 Edge Cases
722
+
723
+ - [ ] Empty promise array
724
+ - [ ] All promises fail
725
+ - [ ] All promises succeed
726
+ - [ ] Mixed error types
727
+ - [ ] Rejected promises with undefined reasons
728
+
729
+ ---
730
+
731
+ ## 10. Recommended Test File Structure
732
+
733
+ ```typescript
734
+ import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
735
+ import { Workflow, Task, Step } from '../../index.js';
736
+
737
+ describe('Promise.allSettled in @Task decorator', () => {
738
+ describe('Concurrent execution', () => {
739
+ describe('Basic functionality', () => {
740
+ it('should execute all workflows concurrently', async () => {
741
+ // Test implementation
742
+ });
743
+
744
+ it('should complete all operations regardless of failures', async () => {
745
+ // Test implementation
746
+ });
747
+ });
748
+
749
+ describe('Error collection', () => {
750
+ it('should collect all errors from failed workflows', async () => {
751
+ // Test implementation
752
+ });
753
+
754
+ it('should preserve error details', async () => {
755
+ // Test implementation
756
+ });
757
+ });
758
+
759
+ describe('Event emission', () => {
760
+ it('should emit events for all workflows', async () => {
761
+ // Test implementation
762
+ });
763
+
764
+ it('should maintain event order', async () => {
765
+ // Test implementation
766
+ });
767
+ });
768
+
769
+ describe('State management', () => {
770
+ it('should capture state for all workflows', async () => {
771
+ // Test implementation
772
+ });
773
+
774
+ it('should handle partial failures gracefully', async () => {
775
+ // Test implementation
776
+ });
777
+ });
778
+ });
779
+ });
780
+ ```
781
+
782
+ ---
783
+
784
+ ## Additional Resources
785
+
786
+ ### Community Resources
787
+
788
+ - **Vitest GitHub Discussions**: https://github.com/vitest-dev/vitest/discussions
789
+ - **Stack Overflow [vitest tag]**: https://stackoverflow.com/questions/tagged/vitest
790
+ - **Vitest Awesome List**: https://github.com/vitest-dev/vitest/issues
791
+
792
+ ### Related Documentation
793
+
794
+ - **Testing Library**: https://testing-library.com/
795
+ - **JavaScript Promises**: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise
796
+ - **TypeScript Promise Types**: https://www.typescriptlang.org/docs/handbook/2/types-from-types.html#promise-types
797
+
798
+ ---
799
+
800
+ **Document Version:** 1.0
801
+ **Last Updated:** 2026-01-12
802
+ **Maintained By:** P1.M2.T1.S3 Research Team