@pennyfarthing/core 11.3.2 → 11.3.4

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 (1129) hide show
  1. package/README.md +1 -1
  2. package/package.json +3 -2
  3. package/packages/core/dist/cli/commands/doctor.d.ts +1 -9
  4. package/packages/core/dist/cli/commands/doctor.d.ts.map +1 -1
  5. package/packages/core/dist/cli/commands/doctor.js +51 -107
  6. package/packages/core/dist/cli/commands/doctor.js.map +1 -1
  7. package/packages/core/dist/cli/commands/stale-artifacts-cleanup.test.d.ts +17 -0
  8. package/packages/core/dist/cli/commands/stale-artifacts-cleanup.test.d.ts.map +1 -0
  9. package/packages/core/dist/cli/commands/stale-artifacts-cleanup.test.js +470 -0
  10. package/packages/core/dist/cli/commands/stale-artifacts-cleanup.test.js.map +1 -0
  11. package/packages/core/dist/cli/commands/update.d.ts.map +1 -1
  12. package/packages/core/dist/cli/commands/update.js +26 -1
  13. package/packages/core/dist/cli/commands/update.js.map +1 -1
  14. package/packages/core/dist/cli/cyclist-migration.test.d.ts +16 -0
  15. package/packages/core/dist/cli/cyclist-migration.test.d.ts.map +1 -0
  16. package/packages/core/dist/cli/cyclist-migration.test.js +229 -0
  17. package/packages/core/dist/cli/cyclist-migration.test.js.map +1 -0
  18. package/packages/core/dist/cli/utils/python.d.ts +0 -1
  19. package/packages/core/dist/cli/utils/python.d.ts.map +1 -1
  20. package/packages/core/dist/cli/utils/python.js +1 -11
  21. package/packages/core/dist/cli/utils/python.js.map +1 -1
  22. package/packages/core/dist/cli/utils/stale-artifacts.d.ts +59 -0
  23. package/packages/core/dist/cli/utils/stale-artifacts.d.ts.map +1 -0
  24. package/packages/core/dist/cli/utils/stale-artifacts.js +163 -0
  25. package/packages/core/dist/cli/utils/stale-artifacts.js.map +1 -0
  26. package/packages/core/dist/scripts/benchmark-integration.d.ts +182 -0
  27. package/packages/core/dist/scripts/benchmark-integration.d.ts.map +1 -0
  28. package/packages/core/dist/scripts/benchmark-integration.js +691 -0
  29. package/packages/core/dist/scripts/benchmark-integration.js.map +1 -0
  30. package/packages/core/dist/scripts/job-fair-aggregator.d.ts +150 -0
  31. package/packages/core/dist/scripts/job-fair-aggregator.d.ts.map +1 -0
  32. package/packages/core/dist/scripts/job-fair-aggregator.js +547 -0
  33. package/packages/core/dist/scripts/job-fair-aggregator.js.map +1 -0
  34. package/packages/core/dist/scripts/theme-detail.test.d.ts.map +1 -0
  35. package/packages/core/dist/scripts/theme-detail.test.js.map +1 -0
  36. package/packages/core/dist/server/paths.d.ts.map +1 -1
  37. package/packages/core/dist/server/paths.js +0 -6
  38. package/packages/core/dist/server/paths.js.map +1 -1
  39. package/pennyfarthing-dist/agents/sm-finish.md +2 -2
  40. package/pennyfarthing-dist/agents/testing-runner.md +1 -1
  41. package/pennyfarthing-dist/commands/pf-prime.md +1 -1
  42. package/pennyfarthing-dist/commands/pf-setup.md +4 -2
  43. package/pennyfarthing-dist/guides/agent-behavior.md +1 -1
  44. package/pennyfarthing-dist/guides/bikerack.md +1 -1
  45. package/pennyfarthing-dist/guides/brownfield-tools.md +7 -7
  46. package/pennyfarthing-dist/guides/gates.md +3 -3
  47. package/pennyfarthing-dist/guides/handoff-cli.md +6 -6
  48. package/pennyfarthing-dist/guides/prime.md +6 -6
  49. package/pennyfarthing-dist/guides/scale-levels.md +1 -1
  50. package/pennyfarthing-dist/personas/themes/discworld.yaml +16 -24
  51. package/pennyfarthing-dist/personas/themes/firefly.yaml +4 -4
  52. package/pennyfarthing-dist/pf/CLAUDE.md +213 -0
  53. package/pennyfarthing-dist/pf/README.md +66 -0
  54. package/pennyfarthing-dist/pf/__pycache__/__init__.cpython-311.pyc +0 -0
  55. package/pennyfarthing-dist/pf/__pycache__/__init__.cpython-314.pyc +0 -0
  56. package/pennyfarthing-dist/pf/__pycache__/bellmode_hook.cpython-314.pyc +0 -0
  57. package/pennyfarthing-dist/pf/__pycache__/cli.cpython-311.pyc +0 -0
  58. package/pennyfarthing-dist/pf/__pycache__/cli.cpython-314.pyc +0 -0
  59. package/pennyfarthing-dist/pf/__pycache__/config.cpython-314.pyc +0 -0
  60. package/pennyfarthing-dist/pf/__pycache__/context.cpython-311.pyc +0 -0
  61. package/pennyfarthing-dist/pf/__pycache__/context.cpython-314.pyc +0 -0
  62. package/pennyfarthing-dist/pf/__pycache__/hooks.cpython-314.pyc +0 -0
  63. package/pennyfarthing-dist/pf/__pycache__/jira.cpython-314.pyc +0 -0
  64. package/pennyfarthing-dist/pf/__pycache__/jira_bidirectional_sync.cpython-314.pyc +0 -0
  65. package/pennyfarthing-dist/pf/__pycache__/jira_epic_creation.cpython-314.pyc +0 -0
  66. package/pennyfarthing-dist/pf/__pycache__/jira_sync.cpython-314.pyc +0 -0
  67. package/pennyfarthing-dist/pf/__pycache__/jira_sync_story.cpython-314.pyc +0 -0
  68. package/pennyfarthing-dist/pf/__pycache__/output.cpython-314.pyc +0 -0
  69. package/pennyfarthing-dist/pf/__pycache__/patch_mode.cpython-314.pyc +0 -0
  70. package/pennyfarthing-dist/pf/__pycache__/pretooluse_hook.cpython-314.pyc +0 -0
  71. package/pennyfarthing-dist/pf/__pycache__/schema_validation_hook.cpython-314.pyc +0 -0
  72. package/pennyfarthing-dist/pf/__pycache__/session_start_hook.cpython-314.pyc +0 -0
  73. package/pennyfarthing-dist/pf/__pycache__/sprint.cpython-314.pyc +0 -0
  74. package/pennyfarthing-dist/pf/__pycache__/workflow.cpython-311.pyc +0 -0
  75. package/pennyfarthing-dist/pf/__pycache__/workflow.cpython-314.pyc +0 -0
  76. package/pennyfarthing-dist/pf/bc/__pycache__/__init__.cpython-311.pyc +0 -0
  77. package/pennyfarthing-dist/pf/bc/__pycache__/__init__.cpython-314.pyc +0 -0
  78. package/pennyfarthing-dist/pf/bc/__pycache__/cli.cpython-311.pyc +0 -0
  79. package/pennyfarthing-dist/pf/bc/__pycache__/cli.cpython-314.pyc +0 -0
  80. package/pennyfarthing-dist/pf/bc/__pycache__/focus.cpython-311.pyc +0 -0
  81. package/pennyfarthing-dist/pf/bc/__pycache__/focus.cpython-314.pyc +0 -0
  82. package/pennyfarthing-dist/pf/bc/__pycache__/split.cpython-314.pyc +0 -0
  83. package/pennyfarthing-dist/pf/bc/cli.py +242 -0
  84. package/pennyfarthing-dist/pf/bc/focus.py +334 -0
  85. package/pennyfarthing-dist/pf/bc/split.py +52 -0
  86. package/pennyfarthing-dist/pf/bellmode_hook.py +22 -0
  87. package/pennyfarthing-dist/pf/bikerack/__init__.py +44 -0
  88. package/pennyfarthing-dist/pf/bikerack/__main__.py +5 -0
  89. package/pennyfarthing-dist/pf/bikerack/__pycache__/__init__.cpython-311.pyc +0 -0
  90. package/pennyfarthing-dist/pf/bikerack/__pycache__/__init__.cpython-314.pyc +0 -0
  91. package/pennyfarthing-dist/pf/bikerack/__pycache__/__main__.cpython-314.pyc +0 -0
  92. package/pennyfarthing-dist/pf/bikerack/__pycache__/audit_log_panel.cpython-314.pyc +0 -0
  93. package/pennyfarthing-dist/pf/bikerack/__pycache__/background_panel.cpython-314.pyc +0 -0
  94. package/pennyfarthing-dist/pf/bikerack/__pycache__/base_panel.cpython-314.pyc +0 -0
  95. package/pennyfarthing-dist/pf/bikerack/__pycache__/changed_panel.cpython-314.pyc +0 -0
  96. package/pennyfarthing-dist/pf/bikerack/__pycache__/cli.cpython-311.pyc +0 -0
  97. package/pennyfarthing-dist/pf/bikerack/__pycache__/cli.cpython-314.pyc +0 -0
  98. package/pennyfarthing-dist/pf/bikerack/__pycache__/context_meter_footer.cpython-314.pyc +0 -0
  99. package/pennyfarthing-dist/pf/bikerack/__pycache__/debug_panel.cpython-314.pyc +0 -0
  100. package/pennyfarthing-dist/pf/bikerack/__pycache__/diffs_panel.cpython-314.pyc +0 -0
  101. package/pennyfarthing-dist/pf/bikerack/__pycache__/events.cpython-314.pyc +0 -0
  102. package/pennyfarthing-dist/pf/bikerack/__pycache__/git_panel.cpython-314.pyc +0 -0
  103. package/pennyfarthing-dist/pf/bikerack/__pycache__/launcher.cpython-311.pyc +0 -0
  104. package/pennyfarthing-dist/pf/bikerack/__pycache__/launcher.cpython-314.pyc +0 -0
  105. package/pennyfarthing-dist/pf/bikerack/__pycache__/portrait_resolver.cpython-314.pyc +0 -0
  106. package/pennyfarthing-dist/pf/bikerack/__pycache__/progress_panel.cpython-314.pyc +0 -0
  107. package/pennyfarthing-dist/pf/bikerack/__pycache__/sprint_panel.cpython-314.pyc +0 -0
  108. package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_data.cpython-314.pyc +0 -0
  109. package/pennyfarthing-dist/pf/bikerack/__pycache__/story_detail_screen.cpython-314.pyc +0 -0
  110. package/pennyfarthing-dist/pf/bikerack/__pycache__/tui.cpython-314.pyc +0 -0
  111. package/pennyfarthing-dist/pf/bikerack/__pycache__/ws_client.cpython-314.pyc +0 -0
  112. package/pennyfarthing-dist/pf/bikerack/audit_log_panel.py +161 -0
  113. package/pennyfarthing-dist/pf/bikerack/background_panel.py +162 -0
  114. package/pennyfarthing-dist/pf/bikerack/changed_panel.py +201 -0
  115. package/pennyfarthing-dist/pf/bikerack/cli.py +145 -0
  116. package/pennyfarthing-dist/pf/bikerack/context_meter_footer.py +138 -0
  117. package/pennyfarthing-dist/pf/bikerack/debug_panel.py +248 -0
  118. package/pennyfarthing-dist/pf/bikerack/diffs_panel.py +409 -0
  119. package/pennyfarthing-dist/pf/bikerack/git_panel.py +139 -0
  120. package/pennyfarthing-dist/pf/bikerack/launcher.py +240 -0
  121. package/pennyfarthing-dist/pf/bikerack/portrait_resolver.py +139 -0
  122. package/pennyfarthing-dist/pf/bikerack/progress_panel.py +314 -0
  123. package/pennyfarthing-dist/pf/bikerack/sprint_panel.py +435 -0
  124. package/pennyfarthing-dist/pf/bikerack/story_detail_data.py +247 -0
  125. package/pennyfarthing-dist/pf/bikerack/story_detail_screen.py +177 -0
  126. package/pennyfarthing-dist/pf/bikerack/tui.py +935 -0
  127. package/pennyfarthing-dist/pf/bmad/__pycache__/__init__.cpython-314.pyc +0 -0
  128. package/pennyfarthing-dist/pf/bmad/__pycache__/cli.cpython-314.pyc +0 -0
  129. package/pennyfarthing-dist/pf/bmad/cli.py +197 -0
  130. package/pennyfarthing-dist/pf/bmad/importer.py +200 -0
  131. package/pennyfarthing-dist/pf/bmad/sync.py +464 -0
  132. package/pennyfarthing-dist/pf/bmad/test_parser.py +253 -0
  133. package/pennyfarthing-dist/pf/bmad/test_sync.py +223 -0
  134. package/pennyfarthing-dist/pf/brownfield/__init__.py +35 -0
  135. package/pennyfarthing-dist/pf/brownfield/__main__.py +8 -0
  136. package/pennyfarthing-dist/pf/brownfield/__pycache__/__init__.cpython-314.pyc +0 -0
  137. package/pennyfarthing-dist/pf/brownfield/__pycache__/__main__.cpython-314.pyc +0 -0
  138. package/pennyfarthing-dist/pf/brownfield/__pycache__/cli.cpython-314.pyc +0 -0
  139. package/pennyfarthing-dist/pf/brownfield/__pycache__/discover.cpython-314.pyc +0 -0
  140. package/pennyfarthing-dist/pf/brownfield/cli.py +130 -0
  141. package/pennyfarthing-dist/pf/cli.py +354 -0
  142. package/pennyfarthing-dist/pf/codemarkers/__init__.py +23 -0
  143. package/pennyfarthing-dist/pf/codemarkers/__main__.py +6 -0
  144. package/pennyfarthing-dist/pf/codemarkers/__pycache__/__init__.cpython-314.pyc +0 -0
  145. package/pennyfarthing-dist/pf/codemarkers/__pycache__/__main__.cpython-314.pyc +0 -0
  146. package/pennyfarthing-dist/pf/codemarkers/__pycache__/analyze.cpython-314.pyc +0 -0
  147. package/pennyfarthing-dist/pf/codemarkers/__pycache__/cli.cpython-314.pyc +0 -0
  148. package/pennyfarthing-dist/pf/codemarkers/__pycache__/formatters.cpython-314.pyc +0 -0
  149. package/pennyfarthing-dist/pf/codemarkers/__pycache__/models.cpython-314.pyc +0 -0
  150. package/pennyfarthing-dist/pf/codemarkers/analyze.py +501 -0
  151. package/pennyfarthing-dist/pf/codemarkers/cli.py +179 -0
  152. package/pennyfarthing-dist/pf/codemarkers/formatters.py +88 -0
  153. package/pennyfarthing-dist/pf/common/__init__.py +48 -0
  154. package/pennyfarthing-dist/pf/common/__pycache__/__init__.cpython-311.pyc +0 -0
  155. package/pennyfarthing-dist/pf/common/__pycache__/__init__.cpython-314.pyc +0 -0
  156. package/pennyfarthing-dist/pf/common/__pycache__/config.cpython-311.pyc +0 -0
  157. package/pennyfarthing-dist/pf/common/__pycache__/config.cpython-314.pyc +0 -0
  158. package/pennyfarthing-dist/pf/common/__pycache__/output.cpython-311.pyc +0 -0
  159. package/pennyfarthing-dist/pf/common/__pycache__/output.cpython-314.pyc +0 -0
  160. package/pennyfarthing-dist/pf/common/__pycache__/pr_config.cpython-314.pyc +0 -0
  161. package/pennyfarthing-dist/pf/common/__pycache__/themes.cpython-314.pyc +0 -0
  162. package/pennyfarthing-dist/pf/common/pr_config.py +38 -0
  163. package/pennyfarthing-dist/pf/common/themes.py +253 -0
  164. package/pennyfarthing-dist/pf/complexity/__init__.py +15 -0
  165. package/pennyfarthing-dist/pf/complexity/__main__.py +6 -0
  166. package/pennyfarthing-dist/pf/complexity/__pycache__/__init__.cpython-314.pyc +0 -0
  167. package/pennyfarthing-dist/pf/complexity/__pycache__/__main__.cpython-314.pyc +0 -0
  168. package/pennyfarthing-dist/pf/complexity/__pycache__/analyze.cpython-314.pyc +0 -0
  169. package/pennyfarthing-dist/pf/complexity/__pycache__/cli.cpython-314.pyc +0 -0
  170. package/pennyfarthing-dist/pf/complexity/__pycache__/formatters.cpython-314.pyc +0 -0
  171. package/pennyfarthing-dist/pf/complexity/__pycache__/models.cpython-314.pyc +0 -0
  172. package/pennyfarthing-dist/pf/complexity/analyze.py +207 -0
  173. package/pennyfarthing-dist/pf/complexity/cli.py +82 -0
  174. package/pennyfarthing-dist/pf/complexity/formatters.py +64 -0
  175. package/pennyfarthing-dist/pf/config.py +21 -0
  176. package/pennyfarthing-dist/pf/consultation/__pycache__/__init__.cpython-311.pyc +0 -0
  177. package/pennyfarthing-dist/pf/consultation/__pycache__/__init__.cpython-314.pyc +0 -0
  178. package/pennyfarthing-dist/pf/consultation/__pycache__/cli.cpython-311.pyc +0 -0
  179. package/pennyfarthing-dist/pf/consultation/__pycache__/cli.cpython-314.pyc +0 -0
  180. package/pennyfarthing-dist/pf/consultation/__pycache__/dialogue_manager.cpython-314.pyc +0 -0
  181. package/pennyfarthing-dist/pf/consultation/cli.py +149 -0
  182. package/pennyfarthing-dist/pf/deadcode/__main__.py +6 -0
  183. package/pennyfarthing-dist/pf/deadcode/__pycache__/__init__.cpython-311.pyc +0 -0
  184. package/pennyfarthing-dist/pf/deadcode/__pycache__/__init__.cpython-314.pyc +0 -0
  185. package/pennyfarthing-dist/pf/deadcode/__pycache__/__main__.cpython-314.pyc +0 -0
  186. package/pennyfarthing-dist/pf/deadcode/__pycache__/analyze.cpython-314.pyc +0 -0
  187. package/pennyfarthing-dist/pf/deadcode/__pycache__/cli.cpython-311.pyc +0 -0
  188. package/pennyfarthing-dist/pf/deadcode/__pycache__/cli.cpython-314.pyc +0 -0
  189. package/pennyfarthing-dist/pf/deadcode/__pycache__/formatters.cpython-314.pyc +0 -0
  190. package/pennyfarthing-dist/pf/deadcode/__pycache__/models.cpython-314.pyc +0 -0
  191. package/pennyfarthing-dist/pf/deadcode/analyze.py +322 -0
  192. package/pennyfarthing-dist/pf/deadcode/cli.py +163 -0
  193. package/pennyfarthing-dist/pf/deadcode/formatters.py +106 -0
  194. package/pennyfarthing-dist/pf/dependencies/__init__.py +20 -0
  195. package/pennyfarthing-dist/pf/dependencies/__main__.py +5 -0
  196. package/pennyfarthing-dist/pf/dependencies/__pycache__/__init__.cpython-314.pyc +0 -0
  197. package/pennyfarthing-dist/pf/dependencies/__pycache__/__main__.cpython-314.pyc +0 -0
  198. package/pennyfarthing-dist/pf/dependencies/__pycache__/analyze.cpython-314.pyc +0 -0
  199. package/pennyfarthing-dist/pf/dependencies/__pycache__/cli.cpython-314.pyc +0 -0
  200. package/pennyfarthing-dist/pf/dependencies/__pycache__/formatters.cpython-314.pyc +0 -0
  201. package/pennyfarthing-dist/pf/dependencies/__pycache__/models.cpython-314.pyc +0 -0
  202. package/pennyfarthing-dist/pf/dependencies/analyze.py +155 -0
  203. package/pennyfarthing-dist/pf/dependencies/cli.py +76 -0
  204. package/pennyfarthing-dist/pf/dependencies/formatters.py +63 -0
  205. package/pennyfarthing-dist/pf/epic/__pycache__/__init__.cpython-311.pyc +0 -0
  206. package/pennyfarthing-dist/pf/epic/__pycache__/__init__.cpython-314.pyc +0 -0
  207. package/pennyfarthing-dist/pf/epic/__pycache__/cli.cpython-311.pyc +0 -0
  208. package/pennyfarthing-dist/pf/epic/__pycache__/cli.cpython-314.pyc +0 -0
  209. package/pennyfarthing-dist/pf/gate/cli.py +56 -0
  210. package/pennyfarthing-dist/pf/git/__init__.py +40 -0
  211. package/pennyfarthing-dist/pf/git/__pycache__/__init__.cpython-314.pyc +0 -0
  212. package/pennyfarthing-dist/pf/git/__pycache__/create_branches.cpython-314.pyc +0 -0
  213. package/pennyfarthing-dist/pf/git/__pycache__/status_all.cpython-314.pyc +0 -0
  214. package/pennyfarthing-dist/pf/git/create_branches.py +439 -0
  215. package/pennyfarthing-dist/pf/git/hooks_installer.py +151 -0
  216. package/pennyfarthing-dist/pf/git/repos.py +196 -0
  217. package/pennyfarthing-dist/pf/git/status_all.py +326 -0
  218. package/pennyfarthing-dist/pf/git/worktree.py +302 -0
  219. package/pennyfarthing-dist/pf/git_group/__pycache__/__init__.cpython-311.pyc +0 -0
  220. package/pennyfarthing-dist/pf/git_group/__pycache__/__init__.cpython-314.pyc +0 -0
  221. package/pennyfarthing-dist/pf/git_group/__pycache__/cli.cpython-311.pyc +0 -0
  222. package/pennyfarthing-dist/pf/git_group/__pycache__/cli.cpython-314.pyc +0 -0
  223. package/pennyfarthing-dist/pf/git_group/cli.py +203 -0
  224. package/pennyfarthing-dist/pf/handoff/__pycache__/__init__.cpython-311.pyc +0 -0
  225. package/pennyfarthing-dist/pf/handoff/__pycache__/__init__.cpython-314.pyc +0 -0
  226. package/pennyfarthing-dist/pf/handoff/__pycache__/cli.cpython-311.pyc +0 -0
  227. package/pennyfarthing-dist/pf/handoff/__pycache__/cli.cpython-314.pyc +0 -0
  228. package/pennyfarthing-dist/pf/handoff/__pycache__/complete_phase.cpython-314.pyc +0 -0
  229. package/pennyfarthing-dist/pf/handoff/__pycache__/gate_file.cpython-314.pyc +0 -0
  230. package/pennyfarthing-dist/pf/handoff/__pycache__/gate_runner.cpython-314.pyc +0 -0
  231. package/pennyfarthing-dist/pf/handoff/__pycache__/marker.cpython-314.pyc +0 -0
  232. package/pennyfarthing-dist/pf/handoff/__pycache__/phase_check.cpython-314.pyc +0 -0
  233. package/pennyfarthing-dist/pf/handoff/__pycache__/resolve_gate.cpython-314.pyc +0 -0
  234. package/pennyfarthing-dist/pf/handoff/cli.py +152 -0
  235. package/pennyfarthing-dist/pf/handoff/marker.py +109 -0
  236. package/pennyfarthing-dist/pf/handoff/phase_check.py +96 -0
  237. package/pennyfarthing-dist/pf/healthscore/__init__.py +21 -0
  238. package/pennyfarthing-dist/pf/healthscore/__main__.py +14 -0
  239. package/pennyfarthing-dist/pf/healthscore/__pycache__/__init__.cpython-311.pyc +0 -0
  240. package/pennyfarthing-dist/pf/healthscore/__pycache__/__init__.cpython-314.pyc +0 -0
  241. package/pennyfarthing-dist/pf/healthscore/__pycache__/__main__.cpython-314.pyc +0 -0
  242. package/pennyfarthing-dist/pf/healthscore/__pycache__/analyze.cpython-311.pyc +0 -0
  243. package/pennyfarthing-dist/pf/healthscore/__pycache__/analyze.cpython-314.pyc +0 -0
  244. package/pennyfarthing-dist/pf/healthscore/__pycache__/cli.cpython-311.pyc +0 -0
  245. package/pennyfarthing-dist/pf/healthscore/__pycache__/cli.cpython-314.pyc +0 -0
  246. package/pennyfarthing-dist/pf/healthscore/__pycache__/formatters.cpython-314.pyc +0 -0
  247. package/pennyfarthing-dist/pf/healthscore/__pycache__/models.cpython-311.pyc +0 -0
  248. package/pennyfarthing-dist/pf/healthscore/__pycache__/models.cpython-314.pyc +0 -0
  249. package/pennyfarthing-dist/pf/healthscore/analyze.py +606 -0
  250. package/pennyfarthing-dist/pf/healthscore/cli.py +80 -0
  251. package/pennyfarthing-dist/pf/healthscore/formatters.py +46 -0
  252. package/pennyfarthing-dist/pf/hooks/__pycache__/__init__.cpython-311.pyc +0 -0
  253. package/pennyfarthing-dist/pf/hooks/__pycache__/__init__.cpython-314.pyc +0 -0
  254. package/pennyfarthing-dist/pf/hooks/__pycache__/bell_mode.cpython-311.pyc +0 -0
  255. package/pennyfarthing-dist/pf/hooks/__pycache__/bell_mode.cpython-314.pyc +0 -0
  256. package/pennyfarthing-dist/pf/hooks/__pycache__/cli.cpython-311.pyc +0 -0
  257. package/pennyfarthing-dist/pf/hooks/__pycache__/cli.cpython-314.pyc +0 -0
  258. package/pennyfarthing-dist/pf/hooks/__pycache__/context_breaker.cpython-311.pyc +0 -0
  259. package/pennyfarthing-dist/pf/hooks/__pycache__/context_breaker.cpython-314.pyc +0 -0
  260. package/pennyfarthing-dist/pf/hooks/__pycache__/context_warning.cpython-311.pyc +0 -0
  261. package/pennyfarthing-dist/pf/hooks/__pycache__/context_warning.cpython-314.pyc +0 -0
  262. package/pennyfarthing-dist/pf/hooks/__pycache__/cyclist_pretooluse.cpython-311.pyc +0 -0
  263. package/pennyfarthing-dist/pf/hooks/__pycache__/cyclist_pretooluse.cpython-314.pyc +0 -0
  264. package/pennyfarthing-dist/pf/hooks/__pycache__/pre_edit_check.cpython-311.pyc +0 -0
  265. package/pennyfarthing-dist/pf/hooks/__pycache__/pre_edit_check.cpython-314.pyc +0 -0
  266. package/pennyfarthing-dist/pf/hooks/__pycache__/reflector_check.cpython-311.pyc +0 -0
  267. package/pennyfarthing-dist/pf/hooks/__pycache__/reflector_check.cpython-314.pyc +0 -0
  268. package/pennyfarthing-dist/pf/hooks/__pycache__/schema_validation.cpython-311.pyc +0 -0
  269. package/pennyfarthing-dist/pf/hooks/__pycache__/schema_validation.cpython-314.pyc +0 -0
  270. package/pennyfarthing-dist/pf/hooks/__pycache__/session_start.cpython-311.pyc +0 -0
  271. package/pennyfarthing-dist/pf/hooks/__pycache__/session_start.cpython-314.pyc +0 -0
  272. package/pennyfarthing-dist/pf/hooks/__pycache__/session_stop.cpython-314.pyc +0 -0
  273. package/pennyfarthing-dist/pf/hooks/__pycache__/sprint_yaml_validation.cpython-311.pyc +0 -0
  274. package/pennyfarthing-dist/pf/hooks/__pycache__/sprint_yaml_validation.cpython-314.pyc +0 -0
  275. package/pennyfarthing-dist/pf/hooks/__pycache__/statusline.cpython-311.pyc +0 -0
  276. package/pennyfarthing-dist/pf/hooks/__pycache__/statusline.cpython-314.pyc +0 -0
  277. package/pennyfarthing-dist/pf/hooks/bell_mode.py +214 -0
  278. package/pennyfarthing-dist/pf/hooks/cli.py +96 -0
  279. package/pennyfarthing-dist/pf/hooks/context_breaker.py +104 -0
  280. package/pennyfarthing-dist/pf/hooks/context_warning.py +66 -0
  281. package/pennyfarthing-dist/pf/hooks/cyclist_pretooluse.py +129 -0
  282. package/pennyfarthing-dist/pf/hooks/schema_validation.py +202 -0
  283. package/pennyfarthing-dist/pf/hooks/session_start.py +294 -0
  284. package/pennyfarthing-dist/pf/hooks/sprint_yaml_validation.py +97 -0
  285. package/pennyfarthing-dist/pf/hooks/statusline.py +429 -0
  286. package/pennyfarthing-dist/pf/hooks.py +32 -0
  287. package/pennyfarthing-dist/pf/hotspots/__init__.py +31 -0
  288. package/pennyfarthing-dist/pf/hotspots/__main__.py +6 -0
  289. package/pennyfarthing-dist/pf/hotspots/__pycache__/__init__.cpython-311.pyc +0 -0
  290. package/pennyfarthing-dist/pf/hotspots/__pycache__/__init__.cpython-314.pyc +0 -0
  291. package/pennyfarthing-dist/pf/hotspots/__pycache__/__main__.cpython-314.pyc +0 -0
  292. package/pennyfarthing-dist/pf/hotspots/__pycache__/analyze.cpython-311.pyc +0 -0
  293. package/pennyfarthing-dist/pf/hotspots/__pycache__/analyze.cpython-314.pyc +0 -0
  294. package/pennyfarthing-dist/pf/hotspots/__pycache__/cli.cpython-311.pyc +0 -0
  295. package/pennyfarthing-dist/pf/hotspots/__pycache__/cli.cpython-314.pyc +0 -0
  296. package/pennyfarthing-dist/pf/hotspots/__pycache__/formatters.cpython-314.pyc +0 -0
  297. package/pennyfarthing-dist/pf/hotspots/__pycache__/models.cpython-311.pyc +0 -0
  298. package/pennyfarthing-dist/pf/hotspots/__pycache__/models.cpython-314.pyc +0 -0
  299. package/pennyfarthing-dist/pf/hotspots/analyze.py +613 -0
  300. package/pennyfarthing-dist/pf/hotspots/cli.py +154 -0
  301. package/pennyfarthing-dist/pf/hotspots/formatters.py +109 -0
  302. package/pennyfarthing-dist/pf/jira/__init__.py +97 -0
  303. package/pennyfarthing-dist/pf/jira/__main__.py +10 -0
  304. package/pennyfarthing-dist/pf/jira/__pycache__/__init__.cpython-311.pyc +0 -0
  305. package/pennyfarthing-dist/pf/jira/__pycache__/__init__.cpython-314.pyc +0 -0
  306. package/pennyfarthing-dist/pf/jira/__pycache__/__main__.cpython-314.pyc +0 -0
  307. package/pennyfarthing-dist/pf/jira/__pycache__/bidirectional.cpython-311.pyc +0 -0
  308. package/pennyfarthing-dist/pf/jira/__pycache__/bidirectional.cpython-314.pyc +0 -0
  309. package/pennyfarthing-dist/pf/jira/__pycache__/claim.cpython-311.pyc +0 -0
  310. package/pennyfarthing-dist/pf/jira/__pycache__/claim.cpython-314.pyc +0 -0
  311. package/pennyfarthing-dist/pf/jira/__pycache__/cli.cpython-311.pyc +0 -0
  312. package/pennyfarthing-dist/pf/jira/__pycache__/cli.cpython-314.pyc +0 -0
  313. package/pennyfarthing-dist/pf/jira/__pycache__/client.cpython-311.pyc +0 -0
  314. package/pennyfarthing-dist/pf/jira/__pycache__/client.cpython-314.pyc +0 -0
  315. package/pennyfarthing-dist/pf/jira/__pycache__/compat.cpython-314.pyc +0 -0
  316. package/pennyfarthing-dist/pf/jira/__pycache__/create.cpython-311.pyc +0 -0
  317. package/pennyfarthing-dist/pf/jira/__pycache__/create.cpython-314.pyc +0 -0
  318. package/pennyfarthing-dist/pf/jira/__pycache__/epic.cpython-311.pyc +0 -0
  319. package/pennyfarthing-dist/pf/jira/__pycache__/epic.cpython-314.pyc +0 -0
  320. package/pennyfarthing-dist/pf/jira/__pycache__/mappings.cpython-314.pyc +0 -0
  321. package/pennyfarthing-dist/pf/jira/__pycache__/models.cpython-314.pyc +0 -0
  322. package/pennyfarthing-dist/pf/jira/__pycache__/operations.cpython-311.pyc +0 -0
  323. package/pennyfarthing-dist/pf/jira/__pycache__/operations.cpython-314.pyc +0 -0
  324. package/pennyfarthing-dist/pf/jira/__pycache__/reconcile.cpython-311.pyc +0 -0
  325. package/pennyfarthing-dist/pf/jira/__pycache__/reconcile.cpython-314.pyc +0 -0
  326. package/pennyfarthing-dist/pf/jira/__pycache__/story.cpython-311.pyc +0 -0
  327. package/pennyfarthing-dist/pf/jira/__pycache__/story.cpython-314.pyc +0 -0
  328. package/pennyfarthing-dist/pf/jira/__pycache__/sync.cpython-311.pyc +0 -0
  329. package/pennyfarthing-dist/pf/jira/__pycache__/sync.cpython-314.pyc +0 -0
  330. package/pennyfarthing-dist/pf/jira/bidirectional.py +587 -0
  331. package/pennyfarthing-dist/pf/jira/claim.py +232 -0
  332. package/pennyfarthing-dist/pf/jira/cli.py +362 -0
  333. package/pennyfarthing-dist/pf/jira/client.py +802 -0
  334. package/pennyfarthing-dist/pf/jira/create.py +311 -0
  335. package/pennyfarthing-dist/pf/jira/epic.py +177 -0
  336. package/pennyfarthing-dist/pf/jira/operations.py +124 -0
  337. package/pennyfarthing-dist/pf/jira/reconcile.py +276 -0
  338. package/pennyfarthing-dist/pf/jira/story.py +221 -0
  339. package/pennyfarthing-dist/pf/jira/sync.py +350 -0
  340. package/pennyfarthing-dist/pf/jira_bidirectional_sync.py +37 -0
  341. package/pennyfarthing-dist/pf/jira_epic_creation.py +30 -0
  342. package/pennyfarthing-dist/pf/jira_sync.py +36 -0
  343. package/pennyfarthing-dist/pf/jira_sync_story.py +30 -0
  344. package/pennyfarthing-dist/pf/launch/__pycache__/__init__.cpython-311.pyc +0 -0
  345. package/pennyfarthing-dist/pf/launch/__pycache__/__init__.cpython-314.pyc +0 -0
  346. package/pennyfarthing-dist/pf/launch/__pycache__/cli.cpython-311.pyc +0 -0
  347. package/pennyfarthing-dist/pf/launch/__pycache__/cli.cpython-314.pyc +0 -0
  348. package/pennyfarthing-dist/pf/launch/cli.py +231 -0
  349. package/pennyfarthing-dist/pf/migration/__init__.py +39 -0
  350. package/pennyfarthing-dist/pf/migration/__main__.py +10 -0
  351. package/pennyfarthing-dist/pf/migration/__pycache__/__init__.cpython-314.pyc +0 -0
  352. package/pennyfarthing-dist/pf/migration/__pycache__/__main__.cpython-314.pyc +0 -0
  353. package/pennyfarthing-dist/pf/migration/__pycache__/cli.cpython-314.pyc +0 -0
  354. package/pennyfarthing-dist/pf/migration/__pycache__/session.cpython-314.pyc +0 -0
  355. package/pennyfarthing-dist/pf/migration/__pycache__/skill.cpython-314.pyc +0 -0
  356. package/pennyfarthing-dist/pf/migration/__pycache__/step.cpython-314.pyc +0 -0
  357. package/pennyfarthing-dist/pf/migration/__pycache__/validate.cpython-314.pyc +0 -0
  358. package/pennyfarthing-dist/pf/migration/cli.py +304 -0
  359. package/pennyfarthing-dist/pf/migration/validate.py +285 -0
  360. package/pennyfarthing-dist/pf/output.py +37 -0
  361. package/pennyfarthing-dist/pf/patch_mode.py +449 -0
  362. package/pennyfarthing-dist/pf/preflight/__init__.py +17 -0
  363. package/pennyfarthing-dist/pf/preflight/__main__.py +10 -0
  364. package/pennyfarthing-dist/pf/preflight/__pycache__/__init__.cpython-314.pyc +0 -0
  365. package/pennyfarthing-dist/pf/preflight/__pycache__/__main__.cpython-314.pyc +0 -0
  366. package/pennyfarthing-dist/pf/preflight/__pycache__/cli.cpython-314.pyc +0 -0
  367. package/pennyfarthing-dist/pf/preflight/__pycache__/finish.cpython-314.pyc +0 -0
  368. package/pennyfarthing-dist/pf/preflight/cli.py +141 -0
  369. package/pennyfarthing-dist/pf/pretooluse_hook.py +10 -0
  370. package/pennyfarthing-dist/pf/prime/__init__.py +127 -0
  371. package/pennyfarthing-dist/pf/prime/__main__.py +8 -0
  372. package/pennyfarthing-dist/pf/prime/__pycache__/__init__.cpython-314.pyc +0 -0
  373. package/pennyfarthing-dist/pf/prime/__pycache__/__main__.cpython-314.pyc +0 -0
  374. package/pennyfarthing-dist/pf/prime/__pycache__/cli.cpython-314.pyc +0 -0
  375. package/pennyfarthing-dist/pf/prime/__pycache__/loader.cpython-314.pyc +0 -0
  376. package/pennyfarthing-dist/pf/prime/__pycache__/models.cpython-314.pyc +0 -0
  377. package/pennyfarthing-dist/pf/prime/__pycache__/persona.cpython-314.pyc +0 -0
  378. package/pennyfarthing-dist/pf/prime/__pycache__/session.cpython-314.pyc +0 -0
  379. package/pennyfarthing-dist/pf/prime/__pycache__/tiers.cpython-314.pyc +0 -0
  380. package/pennyfarthing-dist/pf/prime/__pycache__/version_sentinel.cpython-314.pyc +0 -0
  381. package/pennyfarthing-dist/pf/prime/__pycache__/workflow.cpython-314.pyc +0 -0
  382. package/pennyfarthing-dist/pf/prime/cli.py +662 -0
  383. package/pennyfarthing-dist/pf/prime/heatmap.py +643 -0
  384. package/pennyfarthing-dist/pf/prime/loader.py +308 -0
  385. package/pennyfarthing-dist/pf/prime/persona.py +311 -0
  386. package/pennyfarthing-dist/pf/prime/session.py +183 -0
  387. package/pennyfarthing-dist/pf/prime/tiers.py +214 -0
  388. package/pennyfarthing-dist/pf/prime/workflow.py +317 -0
  389. package/pennyfarthing-dist/pf/schema_validation_hook.py +10 -0
  390. package/pennyfarthing-dist/pf/session/__pycache__/__init__.cpython-311.pyc +0 -0
  391. package/pennyfarthing-dist/pf/session/__pycache__/__init__.cpython-314.pyc +0 -0
  392. package/pennyfarthing-dist/pf/session/__pycache__/cli.cpython-311.pyc +0 -0
  393. package/pennyfarthing-dist/pf/session/__pycache__/cli.cpython-314.pyc +0 -0
  394. package/pennyfarthing-dist/pf/session/cli.py +87 -0
  395. package/pennyfarthing-dist/pf/session_start_hook.py +10 -0
  396. package/pennyfarthing-dist/pf/settings/__pycache__/__init__.cpython-314.pyc +0 -0
  397. package/pennyfarthing-dist/pf/settings/__pycache__/cli.cpython-314.pyc +0 -0
  398. package/pennyfarthing-dist/pf/settings/__pycache__/settings.cpython-314.pyc +0 -0
  399. package/pennyfarthing-dist/pf/settings/cli.py +55 -0
  400. package/pennyfarthing-dist/pf/settings/settings.py +98 -0
  401. package/pennyfarthing-dist/pf/sprint/__init__.py +64 -0
  402. package/pennyfarthing-dist/pf/sprint/__main__.py +10 -0
  403. package/pennyfarthing-dist/pf/sprint/__pycache__/__init__.cpython-311.pyc +0 -0
  404. package/pennyfarthing-dist/pf/sprint/__pycache__/__init__.cpython-314.pyc +0 -0
  405. package/pennyfarthing-dist/pf/sprint/__pycache__/__main__.cpython-314.pyc +0 -0
  406. package/pennyfarthing-dist/pf/sprint/__pycache__/archive.cpython-311.pyc +0 -0
  407. package/pennyfarthing-dist/pf/sprint/__pycache__/archive.cpython-314.pyc +0 -0
  408. package/pennyfarthing-dist/pf/sprint/__pycache__/archive_epic.cpython-314.pyc +0 -0
  409. package/pennyfarthing-dist/pf/sprint/__pycache__/cli.cpython-311.pyc +0 -0
  410. package/pennyfarthing-dist/pf/sprint/__pycache__/cli.cpython-314.pyc +0 -0
  411. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_add.cpython-311.pyc +0 -0
  412. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_add.cpython-314.pyc +0 -0
  413. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_update.cpython-311.pyc +0 -0
  414. package/pennyfarthing-dist/pf/sprint/__pycache__/epic_update.cpython-314.pyc +0 -0
  415. package/pennyfarthing-dist/pf/sprint/__pycache__/import_epic.cpython-314.pyc +0 -0
  416. package/pennyfarthing-dist/pf/sprint/__pycache__/loader.cpython-311.pyc +0 -0
  417. package/pennyfarthing-dist/pf/sprint/__pycache__/loader.cpython-314.pyc +0 -0
  418. package/pennyfarthing-dist/pf/sprint/__pycache__/status.cpython-311.pyc +0 -0
  419. package/pennyfarthing-dist/pf/sprint/__pycache__/status.cpython-314.pyc +0 -0
  420. package/pennyfarthing-dist/pf/sprint/__pycache__/story_add.cpython-311.pyc +0 -0
  421. package/pennyfarthing-dist/pf/sprint/__pycache__/story_add.cpython-314.pyc +0 -0
  422. package/pennyfarthing-dist/pf/sprint/__pycache__/story_finish.cpython-314.pyc +0 -0
  423. package/pennyfarthing-dist/pf/sprint/__pycache__/story_update.cpython-311.pyc +0 -0
  424. package/pennyfarthing-dist/pf/sprint/__pycache__/story_update.cpython-314.pyc +0 -0
  425. package/pennyfarthing-dist/pf/sprint/__pycache__/validate_cmd.cpython-311.pyc +0 -0
  426. package/pennyfarthing-dist/pf/sprint/__pycache__/validate_cmd.cpython-314.pyc +0 -0
  427. package/pennyfarthing-dist/pf/sprint/__pycache__/validator.cpython-311.pyc +0 -0
  428. package/pennyfarthing-dist/pf/sprint/__pycache__/validator.cpython-314.pyc +0 -0
  429. package/pennyfarthing-dist/pf/sprint/__pycache__/work.cpython-311.pyc +0 -0
  430. package/pennyfarthing-dist/pf/sprint/__pycache__/work.cpython-314.pyc +0 -0
  431. package/pennyfarthing-dist/pf/sprint/__pycache__/yaml_io.cpython-311.pyc +0 -0
  432. package/pennyfarthing-dist/pf/sprint/__pycache__/yaml_io.cpython-314.pyc +0 -0
  433. package/pennyfarthing-dist/pf/sprint/archive.py +164 -0
  434. package/pennyfarthing-dist/pf/sprint/archive_epic.py +625 -0
  435. package/pennyfarthing-dist/pf/sprint/cli.py +2021 -0
  436. package/pennyfarthing-dist/pf/sprint/epic_add.py +186 -0
  437. package/pennyfarthing-dist/pf/sprint/epic_update.py +148 -0
  438. package/pennyfarthing-dist/pf/sprint/import_epic.py +455 -0
  439. package/pennyfarthing-dist/pf/sprint/loader.py +441 -0
  440. package/pennyfarthing-dist/pf/sprint/status.py +121 -0
  441. package/pennyfarthing-dist/pf/sprint/story_add.py +372 -0
  442. package/pennyfarthing-dist/pf/sprint/story_finish.py +227 -0
  443. package/pennyfarthing-dist/pf/sprint/story_update.py +222 -0
  444. package/pennyfarthing-dist/pf/sprint/validate_cmd.py +307 -0
  445. package/pennyfarthing-dist/pf/sprint/validator.py +694 -0
  446. package/pennyfarthing-dist/pf/sprint/work.py +229 -0
  447. package/pennyfarthing-dist/pf/story/__init__.py +65 -0
  448. package/pennyfarthing-dist/pf/story/__main__.py +10 -0
  449. package/pennyfarthing-dist/pf/story/__pycache__/__init__.cpython-314.pyc +0 -0
  450. package/pennyfarthing-dist/pf/story/__pycache__/__main__.cpython-314.pyc +0 -0
  451. package/pennyfarthing-dist/pf/story/__pycache__/cli.cpython-314.pyc +0 -0
  452. package/pennyfarthing-dist/pf/story/__pycache__/create.cpython-314.pyc +0 -0
  453. package/pennyfarthing-dist/pf/story/__pycache__/size.cpython-314.pyc +0 -0
  454. package/pennyfarthing-dist/pf/story/__pycache__/template.cpython-314.pyc +0 -0
  455. package/pennyfarthing-dist/pf/story/cli.py +105 -0
  456. package/pennyfarthing-dist/pf/story/create.py +167 -0
  457. package/pennyfarthing-dist/pf/tests/__init__.py +1 -0
  458. package/pennyfarthing-dist/pf/tests/__pycache__/__init__.cpython-314.pyc +0 -0
  459. package/pennyfarthing-dist/pf/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
  460. package/pennyfarthing-dist/pf/tests/__pycache__/test_108_1_gate_migration.cpython-314-pytest-9.0.2.pyc +0 -0
  461. package/pennyfarthing-dist/pf/tests/__pycache__/test_archive_epic.cpython-314-pytest-9.0.2.pyc +0 -0
  462. package/pennyfarthing-dist/pf/tests/__pycache__/test_bc.cpython-314-pytest-9.0.2.pyc +0 -0
  463. package/pennyfarthing-dist/pf/tests/__pycache__/test_bikerack.cpython-314-pytest-9.0.2.pyc +0 -0
  464. package/pennyfarthing-dist/pf/tests/__pycache__/test_brownfield.cpython-314-pytest-9.0.2.pyc +0 -0
  465. package/pennyfarthing-dist/pf/tests/__pycache__/test_cli_modules.cpython-314-pytest-9.0.2.pyc +0 -0
  466. package/pennyfarthing-dist/pf/tests/__pycache__/test_cli_normalization.cpython-314-pytest-9.0.2.pyc +0 -0
  467. package/pennyfarthing-dist/pf/tests/__pycache__/test_codemarkers.cpython-314-pytest-9.0.2.pyc +0 -0
  468. package/pennyfarthing-dist/pf/tests/__pycache__/test_common.cpython-314-pytest-9.0.2.pyc +0 -0
  469. package/pennyfarthing-dist/pf/tests/__pycache__/test_confidence_sm_evaluation.cpython-314-pytest-9.0.2.pyc +0 -0
  470. package/pennyfarthing-dist/pf/tests/__pycache__/test_confidence_sm_gate.cpython-314-pytest-9.0.2.pyc +0 -0
  471. package/pennyfarthing-dist/pf/tests/__pycache__/test_dialogue_manager.cpython-314-pytest-9.0.2.pyc +0 -0
  472. package/pennyfarthing-dist/pf/tests/__pycache__/test_epic_shard_validation.cpython-314-pytest-9.0.2.pyc +0 -0
  473. package/pennyfarthing-dist/pf/tests/__pycache__/test_git_utils.cpython-314-pytest-9.0.2.pyc +0 -0
  474. package/pennyfarthing-dist/pf/tests/__pycache__/test_handoff_cli.cpython-314-pytest-9.0.2.pyc +0 -0
  475. package/pennyfarthing-dist/pf/tests/__pycache__/test_handoff_e2e.cpython-314-pytest-9.0.2.pyc +0 -0
  476. package/pennyfarthing-dist/pf/tests/__pycache__/test_healthscore.cpython-314-pytest-9.0.2.pyc +0 -0
  477. package/pennyfarthing-dist/pf/tests/__pycache__/test_jira_package.cpython-314-pytest-9.0.2.pyc +0 -0
  478. package/pennyfarthing-dist/pf/tests/__pycache__/test_package_structure.cpython-314-pytest-9.0.2.pyc +0 -0
  479. package/pennyfarthing-dist/pf/tests/__pycache__/test_patch_mode.cpython-314-pytest-9.0.2.pyc +0 -0
  480. package/pennyfarthing-dist/pf/tests/__pycache__/test_prime.cpython-314-pytest-9.0.2.pyc +0 -0
  481. package/pennyfarthing-dist/pf/tests/__pycache__/test_sprint_package.cpython-314-pytest-9.0.2.pyc +0 -0
  482. package/pennyfarthing-dist/pf/tests/__pycache__/test_sprint_panel.cpython-314-pytest-9.0.2.pyc +0 -0
  483. package/pennyfarthing-dist/pf/tests/__pycache__/test_sprint_validator.cpython-314-pytest-9.0.2.pyc +0 -0
  484. package/pennyfarthing-dist/pf/tests/__pycache__/test_story_add.cpython-314-pytest-9.0.2.pyc +0 -0
  485. package/pennyfarthing-dist/pf/tests/__pycache__/test_story_package.cpython-314-pytest-9.0.2.pyc +0 -0
  486. package/pennyfarthing-dist/pf/tests/__pycache__/test_story_update.cpython-314-pytest-9.0.2.pyc +0 -0
  487. package/pennyfarthing-dist/pf/tests/__pycache__/test_tiers.cpython-314-pytest-9.0.2.pyc +0 -0
  488. package/pennyfarthing-dist/pf/tests/__pycache__/test_token_counting.cpython-314-pytest-9.0.2.pyc +0 -0
  489. package/pennyfarthing-dist/pf/tests/__pycache__/test_topology_loader.cpython-314-pytest-9.0.2.pyc +0 -0
  490. package/pennyfarthing-dist/pf/tests/__pycache__/test_tui_focus.cpython-314-pytest-9.0.2.pyc +0 -0
  491. package/pennyfarthing-dist/pf/tests/__pycache__/test_tui_panel_persistence.cpython-314-pytest-9.0.2.pyc +0 -0
  492. package/pennyfarthing-dist/pf/tests/__pycache__/test_validate_cmd.cpython-314-pytest-9.0.2.pyc +0 -0
  493. package/pennyfarthing-dist/pf/tests/__pycache__/test_version_sentinel.cpython-314-pytest-9.0.2.pyc +0 -0
  494. package/pennyfarthing-dist/pf/tests/__pycache__/test_workflow_check.cpython-314-pytest-9.0.2.pyc +0 -0
  495. package/pennyfarthing-dist/pf/tests/__pycache__/test_workflow_cli.cpython-314-pytest-9.0.2.pyc +0 -0
  496. package/pennyfarthing-dist/pf/tests/__pycache__/test_yaml_io.cpython-314-pytest-9.0.2.pyc +0 -0
  497. package/pennyfarthing-dist/pf/tests/conftest.py +105 -0
  498. package/pennyfarthing-dist/pf/tests/test_108_1_gate_migration.py +540 -0
  499. package/pennyfarthing-dist/pf/tests/test_108_2_remove_handoff_fallback.py +339 -0
  500. package/pennyfarthing-dist/pf/tests/test_archive_epic.py +185 -0
  501. package/pennyfarthing-dist/pf/tests/test_bc.py +490 -0
  502. package/pennyfarthing-dist/pf/tests/test_bikerack.py +785 -0
  503. package/pennyfarthing-dist/pf/tests/test_brownfield.py +839 -0
  504. package/pennyfarthing-dist/pf/tests/test_cli_modules.py +241 -0
  505. package/pennyfarthing-dist/pf/tests/test_cli_normalization.py +295 -0
  506. package/pennyfarthing-dist/pf/tests/test_codemarkers.py +687 -0
  507. package/pennyfarthing-dist/pf/tests/test_common.py +185 -0
  508. package/pennyfarthing-dist/pf/tests/test_confidence_sm_evaluation.py +253 -0
  509. package/pennyfarthing-dist/pf/tests/test_confidence_sm_gate.py +316 -0
  510. package/pennyfarthing-dist/pf/tests/test_dialogue_manager.py +810 -0
  511. package/pennyfarthing-dist/pf/tests/test_epic_shard_validation.py +699 -0
  512. package/pennyfarthing-dist/pf/tests/test_gate_file_resolution.py +341 -0
  513. package/pennyfarthing-dist/pf/tests/test_gate_runner.py +620 -0
  514. package/pennyfarthing-dist/pf/tests/test_git_utils.py +863 -0
  515. package/pennyfarthing-dist/pf/tests/test_handoff_cli.py +934 -0
  516. package/pennyfarthing-dist/pf/tests/test_handoff_e2e.py +454 -0
  517. package/pennyfarthing-dist/pf/tests/test_healthscore.py +516 -0
  518. package/pennyfarthing-dist/pf/tests/test_jira_package.py +331 -0
  519. package/pennyfarthing-dist/pf/tests/test_package_structure.py +359 -0
  520. package/pennyfarthing-dist/pf/tests/test_patch_mode.py +826 -0
  521. package/pennyfarthing-dist/pf/tests/test_prime.py +1068 -0
  522. package/pennyfarthing-dist/pf/tests/test_resolve_gate_file_field.py +460 -0
  523. package/pennyfarthing-dist/pf/tests/test_sprint_package.py +397 -0
  524. package/pennyfarthing-dist/pf/tests/test_sprint_panel.py +610 -0
  525. package/pennyfarthing-dist/pf/tests/test_sprint_validator.py +779 -0
  526. package/pennyfarthing-dist/pf/tests/test_story_add.py +917 -0
  527. package/pennyfarthing-dist/pf/tests/test_story_package.py +153 -0
  528. package/pennyfarthing-dist/pf/tests/test_story_update.py +764 -0
  529. package/pennyfarthing-dist/pf/tests/test_tiers.py +1091 -0
  530. package/pennyfarthing-dist/pf/tests/test_token_counting.py +565 -0
  531. package/pennyfarthing-dist/pf/tests/test_topology_loader.py +620 -0
  532. package/pennyfarthing-dist/pf/tests/test_tui_focus.py +411 -0
  533. package/pennyfarthing-dist/pf/tests/test_tui_panel_persistence.py +411 -0
  534. package/pennyfarthing-dist/pf/tests/test_validate_cmd.py +495 -0
  535. package/pennyfarthing-dist/pf/tests/test_version_sentinel.py +122 -0
  536. package/pennyfarthing-dist/pf/tests/test_workflow_check.py +338 -0
  537. package/pennyfarthing-dist/pf/tests/test_workflow_list_team.py +143 -0
  538. package/pennyfarthing-dist/pf/tests/test_yaml_io.py +812 -0
  539. package/pennyfarthing-dist/pf/theme/__main__.py +6 -0
  540. package/pennyfarthing-dist/pf/theme/__pycache__/__init__.cpython-311.pyc +0 -0
  541. package/pennyfarthing-dist/pf/theme/__pycache__/__init__.cpython-314.pyc +0 -0
  542. package/pennyfarthing-dist/pf/theme/__pycache__/cli.cpython-311.pyc +0 -0
  543. package/pennyfarthing-dist/pf/theme/__pycache__/cli.cpython-314.pyc +0 -0
  544. package/pennyfarthing-dist/pf/theme/cli.py +310 -0
  545. package/pennyfarthing-dist/pf/validate/__pycache__/__init__.cpython-311.pyc +0 -0
  546. package/pennyfarthing-dist/pf/validate/__pycache__/__init__.cpython-314.pyc +0 -0
  547. package/pennyfarthing-dist/pf/validate/__pycache__/cli.cpython-311.pyc +0 -0
  548. package/pennyfarthing-dist/pf/validate/__pycache__/cli.cpython-314.pyc +0 -0
  549. package/pennyfarthing-dist/pf/validate/adapters/__pycache__/__init__.cpython-314.pyc +0 -0
  550. package/pennyfarthing-dist/pf/validate/adapters/__pycache__/agent.cpython-314.pyc +0 -0
  551. package/pennyfarthing-dist/pf/validate/adapters/__pycache__/schema.cpython-314.pyc +0 -0
  552. package/pennyfarthing-dist/pf/validate/adapters/__pycache__/skill_command.cpython-314.pyc +0 -0
  553. package/pennyfarthing-dist/pf/validate/adapters/__pycache__/sprint.cpython-314.pyc +0 -0
  554. package/pennyfarthing-dist/pf/validate/adapters/__pycache__/tandem_awareness.cpython-314.pyc +0 -0
  555. package/pennyfarthing-dist/pf/validate/adapters/__pycache__/workflow.cpython-314.pyc +0 -0
  556. package/pennyfarthing-dist/pf/validate/adapters/agent.py +239 -0
  557. package/pennyfarthing-dist/pf/validate/adapters/schema.py +30 -0
  558. package/pennyfarthing-dist/pf/validate/adapters/skill_command.py +491 -0
  559. package/pennyfarthing-dist/pf/validate/adapters/sprint.py +69 -0
  560. package/pennyfarthing-dist/pf/validate/adapters/tandem_awareness.py +254 -0
  561. package/pennyfarthing-dist/pf/validate/adapters/team_mode.py +323 -0
  562. package/pennyfarthing-dist/pf/validate/adapters/workflow.py +403 -0
  563. package/pennyfarthing-dist/pf/validate/cli.py +164 -0
  564. package/pennyfarthing-dist/pf/welcome_hook.py +10 -0
  565. package/pennyfarthing-dist/pf/workflow/__init__.py +40 -0
  566. package/pennyfarthing-dist/pf/workflow/__pycache__/__init__.cpython-311.pyc +0 -0
  567. package/pennyfarthing-dist/pf/workflow/__pycache__/__init__.cpython-314.pyc +0 -0
  568. package/pennyfarthing-dist/pf/workflow/__pycache__/cli.cpython-311.pyc +0 -0
  569. package/pennyfarthing-dist/pf/workflow/__pycache__/cli.cpython-314.pyc +0 -0
  570. package/pennyfarthing-dist/pf/workflow/__pycache__/helpers.cpython-314.pyc +0 -0
  571. package/pennyfarthing-dist/pf/workflow/__pycache__/scale.cpython-311.pyc +0 -0
  572. package/pennyfarthing-dist/pf/workflow/__pycache__/scale.cpython-314.pyc +0 -0
  573. package/pennyfarthing-dist/pf/workflow/__pycache__/state.cpython-311.pyc +0 -0
  574. package/pennyfarthing-dist/pf/workflow/__pycache__/state.cpython-314.pyc +0 -0
  575. package/pennyfarthing-dist/pf/workflow/cli.py +1101 -0
  576. package/pennyfarthing-dist/pf/workflow/helpers.py +241 -0
  577. package/pennyfarthing-dist/pyproject.toml +18 -0
  578. package/pennyfarthing-dist/scripts/core/agent-session.sh +3 -3
  579. package/pennyfarthing-dist/scripts/core/check-context.sh +8 -8
  580. package/pennyfarthing-dist/scripts/hooks/__pycache__/question_reflector_check.cpython-314.pyc +0 -0
  581. package/pennyfarthing-dist/scripts/hooks/pre-commit.sh +3 -3
  582. package/pennyfarthing-dist/scripts/jira/README.md +1 -1
  583. package/pennyfarthing-dist/scripts/jira/create-jira-epic.sh +1 -1
  584. package/pennyfarthing-dist/scripts/jira/create-jira-story.sh +1 -1
  585. package/pennyfarthing-dist/scripts/jira/jira-claim-story.sh +1 -1
  586. package/pennyfarthing-dist/scripts/jira/jira-reconcile.sh +1 -1
  587. package/pennyfarthing-dist/scripts/jira/jira-sync-story.sh +1 -1
  588. package/pennyfarthing-dist/scripts/jira/sync-epic-jira.sh +1 -1
  589. package/pennyfarthing-dist/scripts/lib/common.sh +3 -3
  590. package/pennyfarthing-dist/scripts/lib/run-pf.sh +11 -13
  591. package/pennyfarthing-dist/scripts/sprint/README.md +1 -1
  592. package/pennyfarthing-dist/scripts/story/create-story.sh +1 -1
  593. package/pennyfarthing-dist/scripts/story/size-story.sh +1 -1
  594. package/pennyfarthing-dist/scripts/story/story-template.sh +1 -1
  595. package/pennyfarthing-dist/scripts/theme/list-themes.sh +3 -3
  596. package/pennyfarthing-dist/templates/pyproject.toml +3 -3
  597. package/pennyfarthing-dist/workflows/project-setup/steps/step-08-theme-packs.md +1 -1
  598. package/pennyfarthing-dist/workflows/project-setup/steps/step-09-jira.md +92 -0
  599. package/pennyfarthing-dist/workflows/project-setup/steps/step-10-cyclist.md +245 -0
  600. package/pennyfarthing-dist/workflows/project-setup/steps/step-11-complete.md +205 -0
  601. package/packages/core/dist/cli/utils/settings-pf-wrapper.test.d.ts +0 -16
  602. package/packages/core/dist/cli/utils/settings-pf-wrapper.test.d.ts.map +0 -1
  603. package/packages/core/dist/cli/utils/settings-pf-wrapper.test.js +0 -377
  604. package/packages/core/dist/cli/utils/settings-pf-wrapper.test.js.map +0 -1
  605. package/packages/core/dist/workflow/team-lifecycle.d.ts +0 -169
  606. package/packages/core/dist/workflow/team-lifecycle.d.ts.map +0 -1
  607. package/packages/core/dist/workflow/team-lifecycle.js +0 -217
  608. package/packages/core/dist/workflow/team-lifecycle.js.map +0 -1
  609. package/packages/core/dist/workflow/team-lifecycle.test.d.ts +0 -20
  610. package/packages/core/dist/workflow/team-lifecycle.test.d.ts.map +0 -1
  611. package/packages/core/dist/workflow/team-lifecycle.test.js +0 -966
  612. package/packages/core/dist/workflow/team-lifecycle.test.js.map +0 -1
  613. package/packages/core/dist/workflow/workflow-graph-validation.d.ts +0 -65
  614. package/packages/core/dist/workflow/workflow-graph-validation.d.ts.map +0 -1
  615. package/packages/core/dist/workflow/workflow-graph-validation.js +0 -190
  616. package/packages/core/dist/workflow/workflow-graph-validation.js.map +0 -1
  617. package/packages/core/dist/workflow/workflow-graph-validation.test.d.ts +0 -18
  618. package/packages/core/dist/workflow/workflow-graph-validation.test.d.ts.map +0 -1
  619. package/packages/core/dist/workflow/workflow-graph-validation.test.js +0 -706
  620. package/packages/core/dist/workflow/workflow-graph-validation.test.js.map +0 -1
  621. package/pennyfarthing-dist/workflows/project-setup/steps/step-09-cyclist.md +0 -245
  622. package/pennyfarthing-dist/workflows/project-setup/steps/step-10-complete.md +0 -204
  623. package/pennyfarthing_scripts/CLAUDE.md +0 -213
  624. package/pennyfarthing_scripts/README.md +0 -66
  625. package/pennyfarthing_scripts/__pycache__/__init__.cpython-314.pyc +0 -0
  626. package/pennyfarthing_scripts/__pycache__/bellmode_hook.cpython-314.pyc +0 -0
  627. package/pennyfarthing_scripts/__pycache__/cli.cpython-314.pyc +0 -0
  628. package/pennyfarthing_scripts/__pycache__/config.cpython-314.pyc +0 -0
  629. package/pennyfarthing_scripts/__pycache__/context.cpython-314.pyc +0 -0
  630. package/pennyfarthing_scripts/__pycache__/hooks.cpython-314.pyc +0 -0
  631. package/pennyfarthing_scripts/__pycache__/jira_bidirectional_sync.cpython-314.pyc +0 -0
  632. package/pennyfarthing_scripts/__pycache__/jira_epic_creation.cpython-314.pyc +0 -0
  633. package/pennyfarthing_scripts/__pycache__/jira_sync.cpython-314.pyc +0 -0
  634. package/pennyfarthing_scripts/__pycache__/jira_sync_story.cpython-314.pyc +0 -0
  635. package/pennyfarthing_scripts/__pycache__/output.cpython-314.pyc +0 -0
  636. package/pennyfarthing_scripts/__pycache__/patch_mode.cpython-314.pyc +0 -0
  637. package/pennyfarthing_scripts/__pycache__/pretooluse_hook.cpython-314.pyc +0 -0
  638. package/pennyfarthing_scripts/__pycache__/schema_validation_hook.cpython-314.pyc +0 -0
  639. package/pennyfarthing_scripts/__pycache__/session_start_hook.cpython-314.pyc +0 -0
  640. package/pennyfarthing_scripts/__pycache__/workflow.cpython-314.pyc +0 -0
  641. package/pennyfarthing_scripts/bc/__pycache__/__init__.cpython-314.pyc +0 -0
  642. package/pennyfarthing_scripts/bc/__pycache__/cli.cpython-314.pyc +0 -0
  643. package/pennyfarthing_scripts/bc/__pycache__/focus.cpython-314.pyc +0 -0
  644. package/pennyfarthing_scripts/bc/__pycache__/split.cpython-314.pyc +0 -0
  645. package/pennyfarthing_scripts/bc/cli.py +0 -242
  646. package/pennyfarthing_scripts/bc/focus.py +0 -334
  647. package/pennyfarthing_scripts/bc/split.py +0 -52
  648. package/pennyfarthing_scripts/bellmode_hook.py +0 -22
  649. package/pennyfarthing_scripts/bikerack/__init__.py +0 -44
  650. package/pennyfarthing_scripts/bikerack/__main__.py +0 -5
  651. package/pennyfarthing_scripts/bikerack/__pycache__/__init__.cpython-314.pyc +0 -0
  652. package/pennyfarthing_scripts/bikerack/__pycache__/__main__.cpython-314.pyc +0 -0
  653. package/pennyfarthing_scripts/bikerack/__pycache__/audit_log_panel.cpython-314.pyc +0 -0
  654. package/pennyfarthing_scripts/bikerack/__pycache__/background_panel.cpython-314.pyc +0 -0
  655. package/pennyfarthing_scripts/bikerack/__pycache__/base_panel.cpython-314.pyc +0 -0
  656. package/pennyfarthing_scripts/bikerack/__pycache__/changed_panel.cpython-314.pyc +0 -0
  657. package/pennyfarthing_scripts/bikerack/__pycache__/cli.cpython-314.pyc +0 -0
  658. package/pennyfarthing_scripts/bikerack/__pycache__/context_meter_footer.cpython-314.pyc +0 -0
  659. package/pennyfarthing_scripts/bikerack/__pycache__/debug_panel.cpython-314.pyc +0 -0
  660. package/pennyfarthing_scripts/bikerack/__pycache__/diffs_panel.cpython-314.pyc +0 -0
  661. package/pennyfarthing_scripts/bikerack/__pycache__/events.cpython-314.pyc +0 -0
  662. package/pennyfarthing_scripts/bikerack/__pycache__/git_panel.cpython-314.pyc +0 -0
  663. package/pennyfarthing_scripts/bikerack/__pycache__/launcher.cpython-314.pyc +0 -0
  664. package/pennyfarthing_scripts/bikerack/__pycache__/portrait.cpython-314.pyc +0 -0
  665. package/pennyfarthing_scripts/bikerack/__pycache__/portrait_resolver.cpython-314.pyc +0 -0
  666. package/pennyfarthing_scripts/bikerack/__pycache__/progress_panel.cpython-314.pyc +0 -0
  667. package/pennyfarthing_scripts/bikerack/__pycache__/sprint_panel.cpython-314.pyc +0 -0
  668. package/pennyfarthing_scripts/bikerack/__pycache__/story_detail_data.cpython-314.pyc +0 -0
  669. package/pennyfarthing_scripts/bikerack/__pycache__/story_detail_screen.cpython-314.pyc +0 -0
  670. package/pennyfarthing_scripts/bikerack/__pycache__/tui.cpython-314.pyc +0 -0
  671. package/pennyfarthing_scripts/bikerack/__pycache__/ws_client.cpython-314.pyc +0 -0
  672. package/pennyfarthing_scripts/bikerack/audit_log_panel.py +0 -161
  673. package/pennyfarthing_scripts/bikerack/background_panel.py +0 -162
  674. package/pennyfarthing_scripts/bikerack/changed_panel.py +0 -201
  675. package/pennyfarthing_scripts/bikerack/cli.py +0 -145
  676. package/pennyfarthing_scripts/bikerack/context_meter_footer.py +0 -138
  677. package/pennyfarthing_scripts/bikerack/debug_panel.py +0 -248
  678. package/pennyfarthing_scripts/bikerack/diffs_panel.py +0 -409
  679. package/pennyfarthing_scripts/bikerack/git_panel.py +0 -139
  680. package/pennyfarthing_scripts/bikerack/launcher.py +0 -240
  681. package/pennyfarthing_scripts/bikerack/portrait_resolver.py +0 -139
  682. package/pennyfarthing_scripts/bikerack/progress_panel.py +0 -314
  683. package/pennyfarthing_scripts/bikerack/sprint_panel.py +0 -435
  684. package/pennyfarthing_scripts/bikerack/story_detail_data.py +0 -247
  685. package/pennyfarthing_scripts/bikerack/story_detail_screen.py +0 -177
  686. package/pennyfarthing_scripts/bikerack/tui.py +0 -935
  687. package/pennyfarthing_scripts/bmad/__pycache__/__init__.cpython-314.pyc +0 -0
  688. package/pennyfarthing_scripts/bmad/__pycache__/cli.cpython-314.pyc +0 -0
  689. package/pennyfarthing_scripts/bmad/__pycache__/parser.cpython-314.pyc +0 -0
  690. package/pennyfarthing_scripts/bmad/__pycache__/sync.cpython-314.pyc +0 -0
  691. package/pennyfarthing_scripts/bmad/__pycache__/test_parser.cpython-314-pytest-9.0.2.pyc +0 -0
  692. package/pennyfarthing_scripts/bmad/__pycache__/test_sync.cpython-314-pytest-9.0.2.pyc +0 -0
  693. package/pennyfarthing_scripts/bmad/cli.py +0 -197
  694. package/pennyfarthing_scripts/bmad/importer.py +0 -200
  695. package/pennyfarthing_scripts/bmad/sync.py +0 -464
  696. package/pennyfarthing_scripts/bmad/test_parser.py +0 -253
  697. package/pennyfarthing_scripts/bmad/test_sync.py +0 -223
  698. package/pennyfarthing_scripts/brownfield/__init__.py +0 -35
  699. package/pennyfarthing_scripts/brownfield/__main__.py +0 -8
  700. package/pennyfarthing_scripts/brownfield/__pycache__/__init__.cpython-314.pyc +0 -0
  701. package/pennyfarthing_scripts/brownfield/__pycache__/__main__.cpython-314.pyc +0 -0
  702. package/pennyfarthing_scripts/brownfield/__pycache__/cli.cpython-314.pyc +0 -0
  703. package/pennyfarthing_scripts/brownfield/__pycache__/discover.cpython-314.pyc +0 -0
  704. package/pennyfarthing_scripts/brownfield/cli.py +0 -130
  705. package/pennyfarthing_scripts/cli.py +0 -354
  706. package/pennyfarthing_scripts/codemarkers/__init__.py +0 -23
  707. package/pennyfarthing_scripts/codemarkers/__main__.py +0 -6
  708. package/pennyfarthing_scripts/codemarkers/__pycache__/__init__.cpython-314.pyc +0 -0
  709. package/pennyfarthing_scripts/codemarkers/__pycache__/__main__.cpython-314.pyc +0 -0
  710. package/pennyfarthing_scripts/codemarkers/__pycache__/analyze.cpython-314.pyc +0 -0
  711. package/pennyfarthing_scripts/codemarkers/__pycache__/cli.cpython-314.pyc +0 -0
  712. package/pennyfarthing_scripts/codemarkers/__pycache__/formatters.cpython-314.pyc +0 -0
  713. package/pennyfarthing_scripts/codemarkers/__pycache__/models.cpython-314.pyc +0 -0
  714. package/pennyfarthing_scripts/codemarkers/analyze.py +0 -501
  715. package/pennyfarthing_scripts/codemarkers/cli.py +0 -179
  716. package/pennyfarthing_scripts/codemarkers/formatters.py +0 -88
  717. package/pennyfarthing_scripts/common/__init__.py +0 -48
  718. package/pennyfarthing_scripts/common/__pycache__/__init__.cpython-314.pyc +0 -0
  719. package/pennyfarthing_scripts/common/__pycache__/config.cpython-314.pyc +0 -0
  720. package/pennyfarthing_scripts/common/__pycache__/output.cpython-314.pyc +0 -0
  721. package/pennyfarthing_scripts/common/__pycache__/themes.cpython-314.pyc +0 -0
  722. package/pennyfarthing_scripts/common/pr_config.py +0 -38
  723. package/pennyfarthing_scripts/common/themes.py +0 -253
  724. package/pennyfarthing_scripts/complexity/__init__.py +0 -15
  725. package/pennyfarthing_scripts/complexity/__main__.py +0 -6
  726. package/pennyfarthing_scripts/complexity/__pycache__/__init__.cpython-314.pyc +0 -0
  727. package/pennyfarthing_scripts/complexity/__pycache__/__main__.cpython-314.pyc +0 -0
  728. package/pennyfarthing_scripts/complexity/__pycache__/analyze.cpython-314.pyc +0 -0
  729. package/pennyfarthing_scripts/complexity/__pycache__/cli.cpython-314.pyc +0 -0
  730. package/pennyfarthing_scripts/complexity/__pycache__/formatters.cpython-314.pyc +0 -0
  731. package/pennyfarthing_scripts/complexity/__pycache__/models.cpython-314.pyc +0 -0
  732. package/pennyfarthing_scripts/complexity/analyze.py +0 -207
  733. package/pennyfarthing_scripts/complexity/cli.py +0 -82
  734. package/pennyfarthing_scripts/complexity/formatters.py +0 -64
  735. package/pennyfarthing_scripts/config.py +0 -21
  736. package/pennyfarthing_scripts/consultation/__pycache__/__init__.cpython-314.pyc +0 -0
  737. package/pennyfarthing_scripts/consultation/__pycache__/cli.cpython-314.pyc +0 -0
  738. package/pennyfarthing_scripts/consultation/cli.py +0 -149
  739. package/pennyfarthing_scripts/deadcode/__main__.py +0 -6
  740. package/pennyfarthing_scripts/deadcode/__pycache__/__init__.cpython-314.pyc +0 -0
  741. package/pennyfarthing_scripts/deadcode/__pycache__/__main__.cpython-314.pyc +0 -0
  742. package/pennyfarthing_scripts/deadcode/__pycache__/analyze.cpython-314.pyc +0 -0
  743. package/pennyfarthing_scripts/deadcode/__pycache__/cli.cpython-314.pyc +0 -0
  744. package/pennyfarthing_scripts/deadcode/__pycache__/formatters.cpython-314.pyc +0 -0
  745. package/pennyfarthing_scripts/deadcode/__pycache__/models.cpython-314.pyc +0 -0
  746. package/pennyfarthing_scripts/deadcode/analyze.py +0 -322
  747. package/pennyfarthing_scripts/deadcode/cli.py +0 -163
  748. package/pennyfarthing_scripts/deadcode/formatters.py +0 -106
  749. package/pennyfarthing_scripts/dependencies/__init__.py +0 -20
  750. package/pennyfarthing_scripts/dependencies/__main__.py +0 -5
  751. package/pennyfarthing_scripts/dependencies/__pycache__/__init__.cpython-314.pyc +0 -0
  752. package/pennyfarthing_scripts/dependencies/__pycache__/__main__.cpython-314.pyc +0 -0
  753. package/pennyfarthing_scripts/dependencies/__pycache__/analyze.cpython-314.pyc +0 -0
  754. package/pennyfarthing_scripts/dependencies/__pycache__/cli.cpython-314.pyc +0 -0
  755. package/pennyfarthing_scripts/dependencies/__pycache__/formatters.cpython-314.pyc +0 -0
  756. package/pennyfarthing_scripts/dependencies/__pycache__/models.cpython-314.pyc +0 -0
  757. package/pennyfarthing_scripts/dependencies/analyze.py +0 -155
  758. package/pennyfarthing_scripts/dependencies/cli.py +0 -76
  759. package/pennyfarthing_scripts/dependencies/formatters.py +0 -63
  760. package/pennyfarthing_scripts/epic/__pycache__/__init__.cpython-314.pyc +0 -0
  761. package/pennyfarthing_scripts/epic/__pycache__/cli.cpython-314.pyc +0 -0
  762. package/pennyfarthing_scripts/gate/__pycache__/__init__.cpython-314.pyc +0 -0
  763. package/pennyfarthing_scripts/gate/__pycache__/cli.cpython-314.pyc +0 -0
  764. package/pennyfarthing_scripts/gate/__pycache__/validate.cpython-314.pyc +0 -0
  765. package/pennyfarthing_scripts/gate/cli.py +0 -56
  766. package/pennyfarthing_scripts/git/__init__.py +0 -40
  767. package/pennyfarthing_scripts/git/__pycache__/__init__.cpython-314.pyc +0 -0
  768. package/pennyfarthing_scripts/git/__pycache__/create_branches.cpython-314.pyc +0 -0
  769. package/pennyfarthing_scripts/git/__pycache__/hooks_installer.cpython-314.pyc +0 -0
  770. package/pennyfarthing_scripts/git/__pycache__/repos.cpython-314.pyc +0 -0
  771. package/pennyfarthing_scripts/git/__pycache__/status_all.cpython-314.pyc +0 -0
  772. package/pennyfarthing_scripts/git/__pycache__/worktree.cpython-314.pyc +0 -0
  773. package/pennyfarthing_scripts/git/create_branches.py +0 -439
  774. package/pennyfarthing_scripts/git/hooks_installer.py +0 -151
  775. package/pennyfarthing_scripts/git/repos.py +0 -196
  776. package/pennyfarthing_scripts/git/status_all.py +0 -326
  777. package/pennyfarthing_scripts/git/worktree.py +0 -302
  778. package/pennyfarthing_scripts/git_group/__pycache__/__init__.cpython-314.pyc +0 -0
  779. package/pennyfarthing_scripts/git_group/__pycache__/cli.cpython-314.pyc +0 -0
  780. package/pennyfarthing_scripts/git_group/cli.py +0 -203
  781. package/pennyfarthing_scripts/handoff/__pycache__/__init__.cpython-314.pyc +0 -0
  782. package/pennyfarthing_scripts/handoff/__pycache__/cli.cpython-314.pyc +0 -0
  783. package/pennyfarthing_scripts/handoff/__pycache__/complete_phase.cpython-314.pyc +0 -0
  784. package/pennyfarthing_scripts/handoff/__pycache__/gate_file.cpython-314.pyc +0 -0
  785. package/pennyfarthing_scripts/handoff/__pycache__/gate_runner.cpython-314.pyc +0 -0
  786. package/pennyfarthing_scripts/handoff/__pycache__/marker.cpython-314.pyc +0 -0
  787. package/pennyfarthing_scripts/handoff/__pycache__/phase_check.cpython-314.pyc +0 -0
  788. package/pennyfarthing_scripts/handoff/__pycache__/resolve_gate.cpython-314.pyc +0 -0
  789. package/pennyfarthing_scripts/handoff/cli.py +0 -152
  790. package/pennyfarthing_scripts/handoff/marker.py +0 -109
  791. package/pennyfarthing_scripts/handoff/phase_check.py +0 -96
  792. package/pennyfarthing_scripts/healthscore/__init__.py +0 -21
  793. package/pennyfarthing_scripts/healthscore/__main__.py +0 -14
  794. package/pennyfarthing_scripts/healthscore/__pycache__/__init__.cpython-314.pyc +0 -0
  795. package/pennyfarthing_scripts/healthscore/__pycache__/__main__.cpython-314.pyc +0 -0
  796. package/pennyfarthing_scripts/healthscore/__pycache__/analyze.cpython-314.pyc +0 -0
  797. package/pennyfarthing_scripts/healthscore/__pycache__/cli.cpython-314.pyc +0 -0
  798. package/pennyfarthing_scripts/healthscore/__pycache__/formatters.cpython-314.pyc +0 -0
  799. package/pennyfarthing_scripts/healthscore/__pycache__/models.cpython-314.pyc +0 -0
  800. package/pennyfarthing_scripts/healthscore/analyze.py +0 -606
  801. package/pennyfarthing_scripts/healthscore/cli.py +0 -80
  802. package/pennyfarthing_scripts/healthscore/formatters.py +0 -46
  803. package/pennyfarthing_scripts/hooks/__pycache__/__init__.cpython-314.pyc +0 -0
  804. package/pennyfarthing_scripts/hooks/__pycache__/bell_mode.cpython-314.pyc +0 -0
  805. package/pennyfarthing_scripts/hooks/__pycache__/cli.cpython-314.pyc +0 -0
  806. package/pennyfarthing_scripts/hooks/__pycache__/context_breaker.cpython-314.pyc +0 -0
  807. package/pennyfarthing_scripts/hooks/__pycache__/context_warning.cpython-314.pyc +0 -0
  808. package/pennyfarthing_scripts/hooks/__pycache__/cyclist_pretooluse.cpython-314.pyc +0 -0
  809. package/pennyfarthing_scripts/hooks/__pycache__/pre_edit_check.cpython-314.pyc +0 -0
  810. package/pennyfarthing_scripts/hooks/__pycache__/reflector_check.cpython-314.pyc +0 -0
  811. package/pennyfarthing_scripts/hooks/__pycache__/schema_validation.cpython-314.pyc +0 -0
  812. package/pennyfarthing_scripts/hooks/__pycache__/session_start.cpython-314.pyc +0 -0
  813. package/pennyfarthing_scripts/hooks/__pycache__/session_stop.cpython-314.pyc +0 -0
  814. package/pennyfarthing_scripts/hooks/__pycache__/sprint_yaml_validation.cpython-314.pyc +0 -0
  815. package/pennyfarthing_scripts/hooks/__pycache__/statusline.cpython-314.pyc +0 -0
  816. package/pennyfarthing_scripts/hooks/bell_mode.py +0 -214
  817. package/pennyfarthing_scripts/hooks/cli.py +0 -96
  818. package/pennyfarthing_scripts/hooks/context_breaker.py +0 -104
  819. package/pennyfarthing_scripts/hooks/context_warning.py +0 -66
  820. package/pennyfarthing_scripts/hooks/cyclist_pretooluse.py +0 -129
  821. package/pennyfarthing_scripts/hooks/schema_validation.py +0 -202
  822. package/pennyfarthing_scripts/hooks/session_start.py +0 -294
  823. package/pennyfarthing_scripts/hooks/sprint_yaml_validation.py +0 -97
  824. package/pennyfarthing_scripts/hooks/statusline.py +0 -429
  825. package/pennyfarthing_scripts/hooks.py +0 -32
  826. package/pennyfarthing_scripts/hotspots/__init__.py +0 -31
  827. package/pennyfarthing_scripts/hotspots/__main__.py +0 -6
  828. package/pennyfarthing_scripts/hotspots/__pycache__/__init__.cpython-314.pyc +0 -0
  829. package/pennyfarthing_scripts/hotspots/__pycache__/__main__.cpython-314.pyc +0 -0
  830. package/pennyfarthing_scripts/hotspots/__pycache__/analyze.cpython-314.pyc +0 -0
  831. package/pennyfarthing_scripts/hotspots/__pycache__/cli.cpython-314.pyc +0 -0
  832. package/pennyfarthing_scripts/hotspots/__pycache__/formatters.cpython-314.pyc +0 -0
  833. package/pennyfarthing_scripts/hotspots/__pycache__/models.cpython-314.pyc +0 -0
  834. package/pennyfarthing_scripts/hotspots/analyze.py +0 -613
  835. package/pennyfarthing_scripts/hotspots/cli.py +0 -154
  836. package/pennyfarthing_scripts/hotspots/formatters.py +0 -109
  837. package/pennyfarthing_scripts/jira/__init__.py +0 -97
  838. package/pennyfarthing_scripts/jira/__main__.py +0 -10
  839. package/pennyfarthing_scripts/jira/__pycache__/__init__.cpython-314.pyc +0 -0
  840. package/pennyfarthing_scripts/jira/__pycache__/__main__.cpython-314.pyc +0 -0
  841. package/pennyfarthing_scripts/jira/__pycache__/bidirectional.cpython-314.pyc +0 -0
  842. package/pennyfarthing_scripts/jira/__pycache__/claim.cpython-314.pyc +0 -0
  843. package/pennyfarthing_scripts/jira/__pycache__/cli.cpython-314.pyc +0 -0
  844. package/pennyfarthing_scripts/jira/__pycache__/client.cpython-314.pyc +0 -0
  845. package/pennyfarthing_scripts/jira/__pycache__/create.cpython-314.pyc +0 -0
  846. package/pennyfarthing_scripts/jira/__pycache__/epic.cpython-314.pyc +0 -0
  847. package/pennyfarthing_scripts/jira/__pycache__/operations.cpython-314.pyc +0 -0
  848. package/pennyfarthing_scripts/jira/__pycache__/reconcile.cpython-314.pyc +0 -0
  849. package/pennyfarthing_scripts/jira/__pycache__/story.cpython-314.pyc +0 -0
  850. package/pennyfarthing_scripts/jira/__pycache__/sync.cpython-314.pyc +0 -0
  851. package/pennyfarthing_scripts/jira/bidirectional.py +0 -587
  852. package/pennyfarthing_scripts/jira/claim.py +0 -232
  853. package/pennyfarthing_scripts/jira/cli.py +0 -362
  854. package/pennyfarthing_scripts/jira/client.py +0 -790
  855. package/pennyfarthing_scripts/jira/create.py +0 -311
  856. package/pennyfarthing_scripts/jira/epic.py +0 -177
  857. package/pennyfarthing_scripts/jira/operations.py +0 -124
  858. package/pennyfarthing_scripts/jira/reconcile.py +0 -276
  859. package/pennyfarthing_scripts/jira/story.py +0 -221
  860. package/pennyfarthing_scripts/jira/sync.py +0 -350
  861. package/pennyfarthing_scripts/jira_bidirectional_sync.py +0 -37
  862. package/pennyfarthing_scripts/jira_epic_creation.py +0 -30
  863. package/pennyfarthing_scripts/jira_sync.py +0 -36
  864. package/pennyfarthing_scripts/jira_sync_story.py +0 -30
  865. package/pennyfarthing_scripts/launch/__pycache__/__init__.cpython-314.pyc +0 -0
  866. package/pennyfarthing_scripts/launch/__pycache__/cli.cpython-314.pyc +0 -0
  867. package/pennyfarthing_scripts/launch/cli.py +0 -231
  868. package/pennyfarthing_scripts/migration/__init__.py +0 -39
  869. package/pennyfarthing_scripts/migration/__main__.py +0 -10
  870. package/pennyfarthing_scripts/migration/__pycache__/__init__.cpython-314.pyc +0 -0
  871. package/pennyfarthing_scripts/migration/__pycache__/session.cpython-314.pyc +0 -0
  872. package/pennyfarthing_scripts/migration/__pycache__/skill.cpython-314.pyc +0 -0
  873. package/pennyfarthing_scripts/migration/__pycache__/step.cpython-314.pyc +0 -0
  874. package/pennyfarthing_scripts/migration/__pycache__/validate.cpython-314.pyc +0 -0
  875. package/pennyfarthing_scripts/migration/cli.py +0 -304
  876. package/pennyfarthing_scripts/migration/validate.py +0 -285
  877. package/pennyfarthing_scripts/output.py +0 -37
  878. package/pennyfarthing_scripts/patch_mode.py +0 -449
  879. package/pennyfarthing_scripts/preflight/__init__.py +0 -17
  880. package/pennyfarthing_scripts/preflight/__main__.py +0 -10
  881. package/pennyfarthing_scripts/preflight/__pycache__/__init__.cpython-314.pyc +0 -0
  882. package/pennyfarthing_scripts/preflight/__pycache__/__main__.cpython-314.pyc +0 -0
  883. package/pennyfarthing_scripts/preflight/__pycache__/cli.cpython-314.pyc +0 -0
  884. package/pennyfarthing_scripts/preflight/__pycache__/finish.cpython-314.pyc +0 -0
  885. package/pennyfarthing_scripts/preflight/cli.py +0 -141
  886. package/pennyfarthing_scripts/pretooluse_hook.py +0 -10
  887. package/pennyfarthing_scripts/prime/__init__.py +0 -127
  888. package/pennyfarthing_scripts/prime/__main__.py +0 -8
  889. package/pennyfarthing_scripts/prime/__pycache__/__init__.cpython-314.pyc +0 -0
  890. package/pennyfarthing_scripts/prime/__pycache__/cli.cpython-314.pyc +0 -0
  891. package/pennyfarthing_scripts/prime/__pycache__/heatmap.cpython-314.pyc +0 -0
  892. package/pennyfarthing_scripts/prime/__pycache__/loader.cpython-314.pyc +0 -0
  893. package/pennyfarthing_scripts/prime/__pycache__/models.cpython-314.pyc +0 -0
  894. package/pennyfarthing_scripts/prime/__pycache__/persona.cpython-314.pyc +0 -0
  895. package/pennyfarthing_scripts/prime/__pycache__/session.cpython-314.pyc +0 -0
  896. package/pennyfarthing_scripts/prime/__pycache__/tiers.cpython-314.pyc +0 -0
  897. package/pennyfarthing_scripts/prime/__pycache__/version_sentinel.cpython-314.pyc +0 -0
  898. package/pennyfarthing_scripts/prime/__pycache__/workflow.cpython-314.pyc +0 -0
  899. package/pennyfarthing_scripts/prime/cli.py +0 -662
  900. package/pennyfarthing_scripts/prime/heatmap.py +0 -643
  901. package/pennyfarthing_scripts/prime/loader.py +0 -308
  902. package/pennyfarthing_scripts/prime/persona.py +0 -311
  903. package/pennyfarthing_scripts/prime/session.py +0 -183
  904. package/pennyfarthing_scripts/prime/tiers.py +0 -214
  905. package/pennyfarthing_scripts/prime/workflow.py +0 -317
  906. package/pennyfarthing_scripts/schema_validation_hook.py +0 -10
  907. package/pennyfarthing_scripts/session/__pycache__/__init__.cpython-314.pyc +0 -0
  908. package/pennyfarthing_scripts/session/__pycache__/cli.cpython-314.pyc +0 -0
  909. package/pennyfarthing_scripts/session/cli.py +0 -87
  910. package/pennyfarthing_scripts/session_start_hook.py +0 -10
  911. package/pennyfarthing_scripts/settings/__pycache__/__init__.cpython-314.pyc +0 -0
  912. package/pennyfarthing_scripts/settings/__pycache__/cli.cpython-314.pyc +0 -0
  913. package/pennyfarthing_scripts/settings/__pycache__/settings.cpython-314.pyc +0 -0
  914. package/pennyfarthing_scripts/settings/cli.py +0 -55
  915. package/pennyfarthing_scripts/settings/settings.py +0 -98
  916. package/pennyfarthing_scripts/sprint/__init__.py +0 -64
  917. package/pennyfarthing_scripts/sprint/__main__.py +0 -10
  918. package/pennyfarthing_scripts/sprint/__pycache__/__init__.cpython-314.pyc +0 -0
  919. package/pennyfarthing_scripts/sprint/__pycache__/__main__.cpython-314.pyc +0 -0
  920. package/pennyfarthing_scripts/sprint/__pycache__/archive.cpython-314.pyc +0 -0
  921. package/pennyfarthing_scripts/sprint/__pycache__/archive_epic.cpython-314.pyc +0 -0
  922. package/pennyfarthing_scripts/sprint/__pycache__/cli.cpython-314.pyc +0 -0
  923. package/pennyfarthing_scripts/sprint/__pycache__/epic_add.cpython-314.pyc +0 -0
  924. package/pennyfarthing_scripts/sprint/__pycache__/epic_update.cpython-314.pyc +0 -0
  925. package/pennyfarthing_scripts/sprint/__pycache__/import_epic.cpython-314.pyc +0 -0
  926. package/pennyfarthing_scripts/sprint/__pycache__/loader.cpython-314.pyc +0 -0
  927. package/pennyfarthing_scripts/sprint/__pycache__/status.cpython-314.pyc +0 -0
  928. package/pennyfarthing_scripts/sprint/__pycache__/story_add.cpython-314.pyc +0 -0
  929. package/pennyfarthing_scripts/sprint/__pycache__/story_finish.cpython-314.pyc +0 -0
  930. package/pennyfarthing_scripts/sprint/__pycache__/story_update.cpython-314.pyc +0 -0
  931. package/pennyfarthing_scripts/sprint/__pycache__/validate_cmd.cpython-314.pyc +0 -0
  932. package/pennyfarthing_scripts/sprint/__pycache__/validator.cpython-314.pyc +0 -0
  933. package/pennyfarthing_scripts/sprint/__pycache__/work.cpython-314.pyc +0 -0
  934. package/pennyfarthing_scripts/sprint/__pycache__/yaml_io.cpython-314.pyc +0 -0
  935. package/pennyfarthing_scripts/sprint/archive.py +0 -164
  936. package/pennyfarthing_scripts/sprint/archive_epic.py +0 -625
  937. package/pennyfarthing_scripts/sprint/cli.py +0 -2021
  938. package/pennyfarthing_scripts/sprint/epic_add.py +0 -186
  939. package/pennyfarthing_scripts/sprint/epic_update.py +0 -148
  940. package/pennyfarthing_scripts/sprint/import_epic.py +0 -455
  941. package/pennyfarthing_scripts/sprint/loader.py +0 -441
  942. package/pennyfarthing_scripts/sprint/status.py +0 -121
  943. package/pennyfarthing_scripts/sprint/story_add.py +0 -372
  944. package/pennyfarthing_scripts/sprint/story_finish.py +0 -227
  945. package/pennyfarthing_scripts/sprint/story_update.py +0 -222
  946. package/pennyfarthing_scripts/sprint/validate_cmd.py +0 -307
  947. package/pennyfarthing_scripts/sprint/validator.py +0 -694
  948. package/pennyfarthing_scripts/sprint/work.py +0 -229
  949. package/pennyfarthing_scripts/story/__init__.py +0 -65
  950. package/pennyfarthing_scripts/story/__main__.py +0 -10
  951. package/pennyfarthing_scripts/story/__pycache__/__init__.cpython-314.pyc +0 -0
  952. package/pennyfarthing_scripts/story/__pycache__/__main__.cpython-314.pyc +0 -0
  953. package/pennyfarthing_scripts/story/__pycache__/cli.cpython-314.pyc +0 -0
  954. package/pennyfarthing_scripts/story/__pycache__/create.cpython-314.pyc +0 -0
  955. package/pennyfarthing_scripts/story/__pycache__/size.cpython-314.pyc +0 -0
  956. package/pennyfarthing_scripts/story/__pycache__/template.cpython-314.pyc +0 -0
  957. package/pennyfarthing_scripts/story/cli.py +0 -105
  958. package/pennyfarthing_scripts/story/create.py +0 -167
  959. package/pennyfarthing_scripts/tests/__init__.py +0 -1
  960. package/pennyfarthing_scripts/tests/__pycache__/__init__.cpython-314.pyc +0 -0
  961. package/pennyfarthing_scripts/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
  962. package/pennyfarthing_scripts/tests/__pycache__/test_108_2_remove_handoff_fallback.cpython-314-pytest-9.0.2.pyc +0 -0
  963. package/pennyfarthing_scripts/tests/__pycache__/test_archive_epic.cpython-314-pytest-9.0.2.pyc +0 -0
  964. package/pennyfarthing_scripts/tests/__pycache__/test_bc.cpython-314-pytest-9.0.2.pyc +0 -0
  965. package/pennyfarthing_scripts/tests/__pycache__/test_bikerack.cpython-314-pytest-9.0.2.pyc +0 -0
  966. package/pennyfarthing_scripts/tests/__pycache__/test_brownfield.cpython-314-pytest-9.0.2.pyc +0 -0
  967. package/pennyfarthing_scripts/tests/__pycache__/test_cli_modules.cpython-314-pytest-9.0.2.pyc +0 -0
  968. package/pennyfarthing_scripts/tests/__pycache__/test_cli_normalization.cpython-314-pytest-9.0.2.pyc +0 -0
  969. package/pennyfarthing_scripts/tests/__pycache__/test_codemarkers.cpython-314-pytest-9.0.2.pyc +0 -0
  970. package/pennyfarthing_scripts/tests/__pycache__/test_common.cpython-314-pytest-9.0.2.pyc +0 -0
  971. package/pennyfarthing_scripts/tests/__pycache__/test_confidence_sm_evaluation.cpython-314-pytest-9.0.2.pyc +0 -0
  972. package/pennyfarthing_scripts/tests/__pycache__/test_confidence_sm_gate.cpython-314-pytest-9.0.2.pyc +0 -0
  973. package/pennyfarthing_scripts/tests/__pycache__/test_epic_shard_validation.cpython-314-pytest-9.0.2.pyc +0 -0
  974. package/pennyfarthing_scripts/tests/__pycache__/test_gate_file_resolution.cpython-314-pytest-9.0.2.pyc +0 -0
  975. package/pennyfarthing_scripts/tests/__pycache__/test_gate_runner.cpython-314-pytest-9.0.2.pyc +0 -0
  976. package/pennyfarthing_scripts/tests/__pycache__/test_git_utils.cpython-314-pytest-9.0.2.pyc +0 -0
  977. package/pennyfarthing_scripts/tests/__pycache__/test_handoff_cli.cpython-314-pytest-9.0.2.pyc +0 -0
  978. package/pennyfarthing_scripts/tests/__pycache__/test_handoff_e2e.cpython-314-pytest-9.0.2.pyc +0 -0
  979. package/pennyfarthing_scripts/tests/__pycache__/test_healthscore.cpython-314-pytest-9.0.2.pyc +0 -0
  980. package/pennyfarthing_scripts/tests/__pycache__/test_jira_package.cpython-314-pytest-9.0.2.pyc +0 -0
  981. package/pennyfarthing_scripts/tests/__pycache__/test_package_structure.cpython-314-pytest-9.0.2.pyc +0 -0
  982. package/pennyfarthing_scripts/tests/__pycache__/test_patch_mode.cpython-314-pytest-9.0.2.pyc +0 -0
  983. package/pennyfarthing_scripts/tests/__pycache__/test_prime.cpython-314-pytest-9.0.2.pyc +0 -0
  984. package/pennyfarthing_scripts/tests/__pycache__/test_resolve_gate_file_field.cpython-314-pytest-9.0.2.pyc +0 -0
  985. package/pennyfarthing_scripts/tests/__pycache__/test_sprint_package.cpython-314-pytest-9.0.2.pyc +0 -0
  986. package/pennyfarthing_scripts/tests/__pycache__/test_sprint_panel.cpython-314-pytest-9.0.2.pyc +0 -0
  987. package/pennyfarthing_scripts/tests/__pycache__/test_sprint_validator.cpython-314-pytest-9.0.2.pyc +0 -0
  988. package/pennyfarthing_scripts/tests/__pycache__/test_story_add.cpython-314-pytest-9.0.2.pyc +0 -0
  989. package/pennyfarthing_scripts/tests/__pycache__/test_story_package.cpython-314-pytest-9.0.2.pyc +0 -0
  990. package/pennyfarthing_scripts/tests/__pycache__/test_story_update.cpython-314-pytest-9.0.2.pyc +0 -0
  991. package/pennyfarthing_scripts/tests/__pycache__/test_tiers.cpython-314-pytest-9.0.2.pyc +0 -0
  992. package/pennyfarthing_scripts/tests/__pycache__/test_token_counting.cpython-314-pytest-9.0.2.pyc +0 -0
  993. package/pennyfarthing_scripts/tests/__pycache__/test_topology_loader.cpython-314-pytest-9.0.2.pyc +0 -0
  994. package/pennyfarthing_scripts/tests/__pycache__/test_tui_focus.cpython-314-pytest-9.0.2.pyc +0 -0
  995. package/pennyfarthing_scripts/tests/__pycache__/test_tui_panel_persistence.cpython-314-pytest-9.0.2.pyc +0 -0
  996. package/pennyfarthing_scripts/tests/__pycache__/test_validate_cmd.cpython-314-pytest-9.0.2.pyc +0 -0
  997. package/pennyfarthing_scripts/tests/__pycache__/test_version_sentinel.cpython-314-pytest-9.0.2.pyc +0 -0
  998. package/pennyfarthing_scripts/tests/__pycache__/test_workflow_check.cpython-314-pytest-9.0.2.pyc +0 -0
  999. package/pennyfarthing_scripts/tests/__pycache__/test_workflow_list_team.cpython-314-pytest-9.0.2.pyc +0 -0
  1000. package/pennyfarthing_scripts/tests/__pycache__/test_workflow_list_team.cpython-314.pyc +0 -0
  1001. package/pennyfarthing_scripts/tests/__pycache__/test_yaml_io.cpython-314-pytest-9.0.2.pyc +0 -0
  1002. package/pennyfarthing_scripts/tests/conftest.py +0 -105
  1003. package/pennyfarthing_scripts/tests/test_108_1_gate_migration.py +0 -540
  1004. package/pennyfarthing_scripts/tests/test_108_2_remove_handoff_fallback.py +0 -339
  1005. package/pennyfarthing_scripts/tests/test_archive_epic.py +0 -185
  1006. package/pennyfarthing_scripts/tests/test_bc.py +0 -490
  1007. package/pennyfarthing_scripts/tests/test_bikerack.py +0 -785
  1008. package/pennyfarthing_scripts/tests/test_brownfield.py +0 -839
  1009. package/pennyfarthing_scripts/tests/test_cli_modules.py +0 -241
  1010. package/pennyfarthing_scripts/tests/test_cli_normalization.py +0 -295
  1011. package/pennyfarthing_scripts/tests/test_codemarkers.py +0 -687
  1012. package/pennyfarthing_scripts/tests/test_common.py +0 -185
  1013. package/pennyfarthing_scripts/tests/test_confidence_sm_evaluation.py +0 -253
  1014. package/pennyfarthing_scripts/tests/test_confidence_sm_gate.py +0 -316
  1015. package/pennyfarthing_scripts/tests/test_dialogue_manager.py +0 -810
  1016. package/pennyfarthing_scripts/tests/test_epic_shard_validation.py +0 -699
  1017. package/pennyfarthing_scripts/tests/test_gate_file_resolution.py +0 -341
  1018. package/pennyfarthing_scripts/tests/test_gate_runner.py +0 -620
  1019. package/pennyfarthing_scripts/tests/test_git_utils.py +0 -863
  1020. package/pennyfarthing_scripts/tests/test_handoff_cli.py +0 -934
  1021. package/pennyfarthing_scripts/tests/test_handoff_e2e.py +0 -454
  1022. package/pennyfarthing_scripts/tests/test_healthscore.py +0 -516
  1023. package/pennyfarthing_scripts/tests/test_jira_package.py +0 -331
  1024. package/pennyfarthing_scripts/tests/test_package_structure.py +0 -359
  1025. package/pennyfarthing_scripts/tests/test_patch_mode.py +0 -826
  1026. package/pennyfarthing_scripts/tests/test_prime.py +0 -1068
  1027. package/pennyfarthing_scripts/tests/test_resolve_gate_file_field.py +0 -462
  1028. package/pennyfarthing_scripts/tests/test_sprint_package.py +0 -397
  1029. package/pennyfarthing_scripts/tests/test_sprint_panel.py +0 -610
  1030. package/pennyfarthing_scripts/tests/test_sprint_validator.py +0 -779
  1031. package/pennyfarthing_scripts/tests/test_story_add.py +0 -917
  1032. package/pennyfarthing_scripts/tests/test_story_package.py +0 -153
  1033. package/pennyfarthing_scripts/tests/test_story_update.py +0 -764
  1034. package/pennyfarthing_scripts/tests/test_tiers.py +0 -1091
  1035. package/pennyfarthing_scripts/tests/test_token_counting.py +0 -565
  1036. package/pennyfarthing_scripts/tests/test_topology_loader.py +0 -620
  1037. package/pennyfarthing_scripts/tests/test_tui_focus.py +0 -411
  1038. package/pennyfarthing_scripts/tests/test_tui_panel_persistence.py +0 -411
  1039. package/pennyfarthing_scripts/tests/test_validate_cmd.py +0 -495
  1040. package/pennyfarthing_scripts/tests/test_version_sentinel.py +0 -122
  1041. package/pennyfarthing_scripts/tests/test_workflow_check.py +0 -338
  1042. package/pennyfarthing_scripts/tests/test_workflow_list_team.py +0 -143
  1043. package/pennyfarthing_scripts/tests/test_yaml_io.py +0 -812
  1044. package/pennyfarthing_scripts/theme/__main__.py +0 -6
  1045. package/pennyfarthing_scripts/theme/__pycache__/__init__.cpython-314.pyc +0 -0
  1046. package/pennyfarthing_scripts/theme/__pycache__/cli.cpython-314.pyc +0 -0
  1047. package/pennyfarthing_scripts/theme/cli.py +0 -310
  1048. package/pennyfarthing_scripts/validate/__pycache__/__init__.cpython-314.pyc +0 -0
  1049. package/pennyfarthing_scripts/validate/__pycache__/cli.cpython-314.pyc +0 -0
  1050. package/pennyfarthing_scripts/validate/adapters/__pycache__/__init__.cpython-314.pyc +0 -0
  1051. package/pennyfarthing_scripts/validate/adapters/__pycache__/agent.cpython-314.pyc +0 -0
  1052. package/pennyfarthing_scripts/validate/adapters/__pycache__/schema.cpython-314.pyc +0 -0
  1053. package/pennyfarthing_scripts/validate/adapters/__pycache__/skill_command.cpython-314.pyc +0 -0
  1054. package/pennyfarthing_scripts/validate/adapters/__pycache__/sprint.cpython-314.pyc +0 -0
  1055. package/pennyfarthing_scripts/validate/adapters/__pycache__/tandem_awareness.cpython-314.pyc +0 -0
  1056. package/pennyfarthing_scripts/validate/adapters/__pycache__/team_mode.cpython-314.pyc +0 -0
  1057. package/pennyfarthing_scripts/validate/adapters/__pycache__/workflow.cpython-314.pyc +0 -0
  1058. package/pennyfarthing_scripts/validate/adapters/agent.py +0 -239
  1059. package/pennyfarthing_scripts/validate/adapters/schema.py +0 -30
  1060. package/pennyfarthing_scripts/validate/adapters/skill_command.py +0 -491
  1061. package/pennyfarthing_scripts/validate/adapters/sprint.py +0 -69
  1062. package/pennyfarthing_scripts/validate/adapters/tandem_awareness.py +0 -254
  1063. package/pennyfarthing_scripts/validate/adapters/team_mode.py +0 -323
  1064. package/pennyfarthing_scripts/validate/adapters/workflow.py +0 -403
  1065. package/pennyfarthing_scripts/validate/cli.py +0 -164
  1066. package/pennyfarthing_scripts/welcome_hook.py +0 -10
  1067. package/pennyfarthing_scripts/workflow/__init__.py +0 -40
  1068. package/pennyfarthing_scripts/workflow/__pycache__/__init__.cpython-314.pyc +0 -0
  1069. package/pennyfarthing_scripts/workflow/__pycache__/cli.cpython-314.pyc +0 -0
  1070. package/pennyfarthing_scripts/workflow/__pycache__/helpers.cpython-314.pyc +0 -0
  1071. package/pennyfarthing_scripts/workflow/__pycache__/scale.cpython-314.pyc +0 -0
  1072. package/pennyfarthing_scripts/workflow/__pycache__/state.cpython-314.pyc +0 -0
  1073. package/pennyfarthing_scripts/workflow/__pycache__/team_lifecycle.cpython-314.pyc +0 -0
  1074. package/pennyfarthing_scripts/workflow/cli.py +0 -1101
  1075. package/pennyfarthing_scripts/workflow/helpers.py +0 -241
  1076. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/__init__.py +0 -0
  1077. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/bc/__init__.py +0 -0
  1078. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/bikerack/base_panel.py +0 -0
  1079. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/bikerack/events.py +0 -0
  1080. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/bikerack/ws_client.py +0 -0
  1081. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/bmad/__init__.py +0 -0
  1082. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/bmad/parser.py +0 -0
  1083. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/brownfield/discover.py +0 -0
  1084. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/codemarkers/models.py +0 -0
  1085. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/common/config.py +0 -0
  1086. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/common/output.py +0 -0
  1087. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/complexity/models.py +0 -0
  1088. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/consultation/__init__.py +0 -0
  1089. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/consultation/dialogue_manager.py +0 -0
  1090. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/context.py +0 -0
  1091. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/deadcode/__init__.py +0 -0
  1092. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/deadcode/models.py +0 -0
  1093. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/dependencies/models.py +0 -0
  1094. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/epic/__init__.py +0 -0
  1095. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/epic/cli.py +0 -0
  1096. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/gate/__init__.py +0 -0
  1097. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/gate/validate.py +0 -0
  1098. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/git_group/__init__.py +0 -0
  1099. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/handoff/__init__.py +0 -0
  1100. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/handoff/complete_phase.py +0 -0
  1101. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/handoff/gate_file.py +0 -0
  1102. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/handoff/gate_runner.py +0 -0
  1103. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/handoff/resolve_gate.py +0 -0
  1104. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/healthscore/models.py +0 -0
  1105. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/hooks/__init__.py +0 -0
  1106. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/hooks/cyclist-pretooluse-hook.sh +0 -0
  1107. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/hooks/pre_edit_check.py +0 -0
  1108. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/hooks/reflector_check.py +0 -0
  1109. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/hooks/session_stop.py +0 -0
  1110. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/hotspots/models.py +0 -0
  1111. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/launch/__init__.py +0 -0
  1112. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/migration/session.py +0 -0
  1113. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/migration/skill.py +0 -0
  1114. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/migration/step.py +0 -0
  1115. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/preflight/finish.py +0 -0
  1116. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/prime/models.py +0 -0
  1117. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/prime/version_sentinel.py +0 -0
  1118. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/session/__init__.py +0 -0
  1119. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/settings/__init__.py +0 -0
  1120. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/sprint/yaml_io.py +0 -0
  1121. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/story/size.py +0 -0
  1122. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/story/template.py +0 -0
  1123. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/swebench.py +0 -0
  1124. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/theme/__init__.py +0 -0
  1125. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/validate/__init__.py +0 -0
  1126. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/validate/adapters/__init__.py +0 -0
  1127. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/workflow/scale.py +0 -0
  1128. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/workflow/state.py +0 -0
  1129. /package/{pennyfarthing_scripts → pennyfarthing-dist/pf}/workflow/team_lifecycle.py +0 -0
@@ -1,966 +0,0 @@
1
- /**
2
- * Tests for Story 86-10: Phase-scoped team lifecycle + gate hooks
3
- *
4
- * RED state tests for native Agent Teams lifecycle in phased workflows.
5
- * These tests cover all 9 acceptance criteria:
6
- *
7
- * AC1: Lead creates team on phase entry when workflow has team block
8
- * AC2: Lead spawns teammates per workflow YAML teammates config
9
- * AC3: TaskCompleted hook enforces gate checks
10
- * AC4: TeammateIdle hook validates teammate work
11
- * AC5: Lead shuts down all teammates before exit protocol
12
- * AC6: TeamDelete runs before pf handoff (full cleanup before marker)
13
- * AC7: Session file updated with teammate activity summary
14
- * AC8: Sidecar file locking for concurrent teammate writes
15
- * AC9: Graceful degradation when teammate crashes
16
- *
17
- * Run with: pnpm build && pnpm test
18
- */
19
- import { describe, it, beforeEach, afterEach } from 'node:test';
20
- import assert from 'node:assert';
21
- import { mkdirSync, rmSync, existsSync } from 'node:fs';
22
- import { join, dirname } from 'node:path';
23
- import { fileURLToPath } from 'node:url';
24
- import { createTeam, spawnTeammates, shutdownAllTeammates, cleanupTeam, checkGateOnTaskCompleted, checkGateOnTeammateIdle, generateTeamSummary, acquireSidecarLock, releaseSidecarLock, getActiveTeam, _resetForTesting, } from './team-lifecycle.js';
25
- // Get directory for test fixtures
26
- const __dirname = dirname(fileURLToPath(import.meta.url));
27
- const TEST_DIR = join(__dirname, '__test_team_lifecycle__');
28
- // =============================================================================
29
- // Test Fixtures
30
- // =============================================================================
31
- /** Phase with team config (Dev + Architect) */
32
- const PHASE_WITH_TEAM = {
33
- name: 'green',
34
- agent: 'dev',
35
- input: ['failing_tests'],
36
- output: ['implementation', 'passing_tests'],
37
- team: {
38
- teammates: [
39
- { agent: 'architect', task: 'Review implementation approach and patterns' },
40
- { agent: 'tea', task: 'Verify tests stay green, flag regressions' },
41
- ],
42
- model: 'sonnet',
43
- display: 'in-process',
44
- },
45
- };
46
- /** Phase with team config (single teammate) */
47
- const PHASE_WITH_SINGLE_TEAMMATE = {
48
- name: 'review',
49
- agent: 'reviewer',
50
- team: {
51
- teammates: [
52
- { agent: 'architect', task: 'Validate architectural patterns' },
53
- ],
54
- },
55
- };
56
- /** Phase with team and gate */
57
- const PHASE_WITH_TEAM_AND_GATE = {
58
- name: 'green',
59
- agent: 'dev',
60
- team: {
61
- teammates: [
62
- { agent: 'tea', task: 'Verify tests pass' },
63
- ],
64
- },
65
- gate: {
66
- type: 'tests_pass',
67
- condition: 'All tests must pass before phase completion',
68
- },
69
- };
70
- /** Phase without team config */
71
- const PHASE_WITHOUT_TEAM = {
72
- name: 'red',
73
- agent: 'tea',
74
- input: ['session_file'],
75
- output: ['failing_tests'],
76
- };
77
- /** No-op adapter for tests that don't need real process interaction */
78
- function createMockAdapter(overrides) {
79
- return {
80
- createTeam: async (params) => ({ teamName: params.teamName }),
81
- deleteTeam: async () => { },
82
- spawnTeammate: async (params) => ({ agentId: `mock-${params.agent}-001` }),
83
- shutdownTeammate: async () => { },
84
- ...overrides,
85
- };
86
- }
87
- // =============================================================================
88
- // Test Suite
89
- // =============================================================================
90
- describe('86-10: Phase-scoped Team Lifecycle', () => {
91
- beforeEach(() => {
92
- _resetForTesting();
93
- if (existsSync(TEST_DIR)) {
94
- rmSync(TEST_DIR, { recursive: true });
95
- }
96
- mkdirSync(TEST_DIR, { recursive: true });
97
- mkdirSync(join(TEST_DIR, '.session'), { recursive: true });
98
- });
99
- afterEach(() => {
100
- _resetForTesting();
101
- if (existsSync(TEST_DIR)) {
102
- rmSync(TEST_DIR, { recursive: true });
103
- }
104
- });
105
- // ===========================================================================
106
- // AC1: Lead creates team on phase entry when workflow has team block
107
- // ===========================================================================
108
- describe('AC1: Create team on phase entry', () => {
109
- it('should create team when phase has team config', async () => {
110
- const params = {
111
- phase: PHASE_WITH_TEAM,
112
- storyId: '86-10',
113
- sessionDir: join(TEST_DIR, '.session'),
114
- adapter: createMockAdapter(),
115
- };
116
- const result = await createTeam(params);
117
- assert.strictEqual(result.success, true, 'Team creation should succeed');
118
- assert.ok(result.data, 'Should return team handle');
119
- assert.ok(result.data.teamName, 'Handle should have team name');
120
- assert.strictEqual(result.data.storyId, '86-10', 'Story ID should match');
121
- assert.strictEqual(result.data.phase, 'green', 'Phase should match');
122
- });
123
- it('should generate team name from storyId and phase', async () => {
124
- const result = await createTeam({
125
- phase: PHASE_WITH_TEAM,
126
- storyId: '86-10',
127
- sessionDir: join(TEST_DIR, '.session'),
128
- adapter: createMockAdapter(),
129
- });
130
- assert.strictEqual(result.success, true);
131
- assert.ok(result.data.teamName.includes('86-10'), 'Team name should include story ID');
132
- assert.ok(result.data.teamName.includes('green'), 'Team name should include phase name');
133
- });
134
- it('should set createdAt timestamp on team handle', async () => {
135
- const result = await createTeam({
136
- phase: PHASE_WITH_TEAM,
137
- storyId: '86-10',
138
- sessionDir: join(TEST_DIR, '.session'),
139
- adapter: createMockAdapter(),
140
- });
141
- assert.strictEqual(result.success, true);
142
- assert.ok(result.data.createdAt, 'Should have createdAt timestamp');
143
- // Verify it's a valid ISO date string
144
- const date = new Date(result.data.createdAt);
145
- assert.ok(!isNaN(date.getTime()), 'createdAt should be valid ISO date');
146
- });
147
- it('should register team in active registry', async () => {
148
- await createTeam({
149
- phase: PHASE_WITH_TEAM,
150
- storyId: '86-10',
151
- sessionDir: join(TEST_DIR, '.session'),
152
- adapter: createMockAdapter(),
153
- });
154
- const active = getActiveTeam('86-10');
155
- assert.ok(active, 'Team should be in active registry');
156
- assert.strictEqual(active.storyId, '86-10');
157
- });
158
- it('should call adapter.createTeam when adapter provided', async () => {
159
- let createCalled = false;
160
- let createParams = {};
161
- const adapter = createMockAdapter({
162
- createTeam: async (params) => {
163
- createCalled = true;
164
- createParams = params;
165
- return { teamName: params.teamName };
166
- },
167
- });
168
- await createTeam({
169
- phase: PHASE_WITH_TEAM,
170
- storyId: '86-10',
171
- sessionDir: join(TEST_DIR, '.session'),
172
- adapter,
173
- });
174
- assert.strictEqual(createCalled, true, 'Adapter createTeam must be called');
175
- assert.ok(createParams.teamName, 'Should pass team name to adapter');
176
- });
177
- it('should return no-op for phase without team config', async () => {
178
- const result = await createTeam({
179
- phase: PHASE_WITHOUT_TEAM,
180
- storyId: '86-10',
181
- sessionDir: join(TEST_DIR, '.session'),
182
- adapter: createMockAdapter(),
183
- });
184
- assert.strictEqual(result.success, true, 'Should succeed as no-op');
185
- assert.strictEqual(result.data, undefined, 'No handle for non-team phase');
186
- });
187
- it('should not register anything for non-team phase', async () => {
188
- await createTeam({
189
- phase: PHASE_WITHOUT_TEAM,
190
- storyId: '86-10',
191
- sessionDir: join(TEST_DIR, '.session'),
192
- });
193
- const active = getActiveTeam('86-10');
194
- assert.strictEqual(active, null, 'No active team for non-team phase');
195
- });
196
- it('should clean up existing team before creating new one for same story', async () => {
197
- const deleted = [];
198
- const adapter = createMockAdapter({
199
- deleteTeam: async (teamName) => { deleted.push(teamName); },
200
- });
201
- // First team
202
- await createTeam({
203
- phase: PHASE_WITH_TEAM,
204
- storyId: '86-10',
205
- sessionDir: join(TEST_DIR, '.session'),
206
- adapter,
207
- });
208
- // Second team for same story (different phase)
209
- await createTeam({
210
- phase: PHASE_WITH_SINGLE_TEAMMATE,
211
- storyId: '86-10',
212
- sessionDir: join(TEST_DIR, '.session'),
213
- adapter,
214
- });
215
- assert.strictEqual(deleted.length, 1, 'First team should be deleted');
216
- const active = getActiveTeam('86-10');
217
- assert.strictEqual(active.phase, 'review', 'Second team should be active');
218
- });
219
- });
220
- // ===========================================================================
221
- // AC2: Lead spawns teammates per workflow YAML teammates config
222
- // ===========================================================================
223
- describe('AC2: Spawn teammates per YAML config', () => {
224
- it('should spawn all teammates from team config', async () => {
225
- const createResult = await createTeam({
226
- phase: PHASE_WITH_TEAM,
227
- storyId: '86-10',
228
- sessionDir: join(TEST_DIR, '.session'),
229
- adapter: createMockAdapter(),
230
- });
231
- const handle = createResult.data;
232
- const result = await spawnTeammates(handle, PHASE_WITH_TEAM.team, '86-10', 'green', createMockAdapter());
233
- assert.strictEqual(result.success, true, 'Spawn should succeed');
234
- assert.ok(result.data, 'Should return teammate handles');
235
- assert.strictEqual(result.data.length, 2, 'Should spawn 2 teammates');
236
- });
237
- it('should spawn teammates with correct agent names', async () => {
238
- const createResult = await createTeam({
239
- phase: PHASE_WITH_TEAM,
240
- storyId: '86-10',
241
- sessionDir: join(TEST_DIR, '.session'),
242
- adapter: createMockAdapter(),
243
- });
244
- const result = await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', createMockAdapter());
245
- const agents = result.data.map((t) => t.agent).sort();
246
- assert.deepStrictEqual(agents, ['architect', 'tea'], 'Should match YAML config');
247
- });
248
- it('should pass task descriptions to spawned teammates', async () => {
249
- const createResult = await createTeam({
250
- phase: PHASE_WITH_TEAM,
251
- storyId: '86-10',
252
- sessionDir: join(TEST_DIR, '.session'),
253
- adapter: createMockAdapter(),
254
- });
255
- const result = await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', createMockAdapter());
256
- const architect = result.data.find((t) => t.agent === 'architect');
257
- assert.ok(architect, 'Should have architect teammate');
258
- assert.strictEqual(architect.task, 'Review implementation approach and patterns', 'Task should match YAML config');
259
- });
260
- it('should call adapter.spawnTeammate for each teammate', async () => {
261
- const spawnedAgents = [];
262
- const adapter = createMockAdapter({
263
- spawnTeammate: async (params) => {
264
- spawnedAgents.push(params.agent);
265
- return { agentId: `mock-${params.agent}` };
266
- },
267
- });
268
- const createResult = await createTeam({
269
- phase: PHASE_WITH_TEAM,
270
- storyId: '86-10',
271
- sessionDir: join(TEST_DIR, '.session'),
272
- adapter,
273
- });
274
- await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', adapter);
275
- assert.deepStrictEqual(spawnedAgents.sort(), ['architect', 'tea'], 'Should call adapter for each teammate');
276
- });
277
- it('should set initial teammate status to spawned', async () => {
278
- const createResult = await createTeam({
279
- phase: PHASE_WITH_TEAM,
280
- storyId: '86-10',
281
- sessionDir: join(TEST_DIR, '.session'),
282
- adapter: createMockAdapter(),
283
- });
284
- const result = await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', createMockAdapter());
285
- for (const teammate of result.data) {
286
- assert.strictEqual(teammate.status, 'spawned', `Teammate ${teammate.agent} should have status 'spawned'`);
287
- }
288
- });
289
- it('should update team handle with spawned teammates', async () => {
290
- const createResult = await createTeam({
291
- phase: PHASE_WITH_TEAM,
292
- storyId: '86-10',
293
- sessionDir: join(TEST_DIR, '.session'),
294
- adapter: createMockAdapter(),
295
- });
296
- await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', createMockAdapter());
297
- const active = getActiveTeam('86-10');
298
- assert.ok(active, 'Team should still be in registry');
299
- assert.strictEqual(active.teammates.length, 2, 'Active team should have spawned teammates');
300
- });
301
- it('should handle single teammate config', async () => {
302
- const createResult = await createTeam({
303
- phase: PHASE_WITH_SINGLE_TEAMMATE,
304
- storyId: '86-10',
305
- sessionDir: join(TEST_DIR, '.session'),
306
- adapter: createMockAdapter(),
307
- });
308
- const result = await spawnTeammates(createResult.data, PHASE_WITH_SINGLE_TEAMMATE.team, '86-10', 'review', createMockAdapter());
309
- assert.strictEqual(result.success, true);
310
- assert.strictEqual(result.data.length, 1, 'Should spawn exactly 1 teammate');
311
- assert.strictEqual(result.data[0].agent, 'architect');
312
- });
313
- });
314
- // ===========================================================================
315
- // AC3: TaskCompleted hook enforces gate checks
316
- // ===========================================================================
317
- describe('AC3: TaskCompleted gate enforcement', () => {
318
- it('should check gate when task completed event fires', () => {
319
- const handle = {
320
- teamName: '86-10-green',
321
- storyId: '86-10',
322
- phase: 'green',
323
- teammates: [
324
- { agent: 'architect', status: 'idle' },
325
- { agent: 'tea', status: 'idle' },
326
- ],
327
- createdAt: new Date().toISOString(),
328
- };
329
- const result = checkGateOnTaskCompleted(handle, PHASE_WITH_TEAM_AND_GATE);
330
- assert.ok('passed' in result, 'Should return gate check result');
331
- assert.ok('gate' in result, 'Should include gate type');
332
- assert.ok(typeof result.passed === 'boolean', 'passed must be boolean');
333
- });
334
- it('should return gate type from phase config', () => {
335
- const handle = {
336
- teamName: '86-10-green',
337
- storyId: '86-10',
338
- phase: 'green',
339
- teammates: [{ agent: 'tea', status: 'idle' }],
340
- createdAt: new Date().toISOString(),
341
- };
342
- const result = checkGateOnTaskCompleted(handle, PHASE_WITH_TEAM_AND_GATE);
343
- assert.strictEqual(result.gate, 'tests_pass', 'Should reflect phase gate type');
344
- });
345
- it('should fail gate when teammates still active', () => {
346
- const handle = {
347
- teamName: '86-10-green',
348
- storyId: '86-10',
349
- phase: 'green',
350
- teammates: [
351
- { agent: 'tea', status: 'active' }, // still working
352
- ],
353
- createdAt: new Date().toISOString(),
354
- };
355
- const result = checkGateOnTaskCompleted(handle, PHASE_WITH_TEAM_AND_GATE);
356
- assert.strictEqual(result.passed, false, 'Gate should fail when teammates still active');
357
- assert.ok(result.reason, 'Should provide failure reason');
358
- });
359
- it('should pass gate when all teammates idle and criteria met', () => {
360
- const handle = {
361
- teamName: '86-10-green',
362
- storyId: '86-10',
363
- phase: 'green',
364
- teammates: [
365
- { agent: 'tea', status: 'idle' },
366
- ],
367
- createdAt: new Date().toISOString(),
368
- };
369
- const result = checkGateOnTaskCompleted(handle, PHASE_WITH_TEAM_AND_GATE);
370
- assert.strictEqual(result.passed, true, 'Gate should pass when all idle');
371
- });
372
- it('should pass trivially when phase has no gate', () => {
373
- const handle = {
374
- teamName: '86-10-green',
375
- storyId: '86-10',
376
- phase: 'green',
377
- teammates: [{ agent: 'architect', status: 'active' }],
378
- createdAt: new Date().toISOString(),
379
- };
380
- // Phase with team but no gate
381
- const phaseNoGate = {
382
- name: 'green',
383
- agent: 'dev',
384
- team: { teammates: [{ agent: 'architect' }] },
385
- };
386
- const result = checkGateOnTaskCompleted(handle, phaseNoGate);
387
- assert.strictEqual(result.passed, true, 'No gate = always passes');
388
- });
389
- });
390
- // ===========================================================================
391
- // AC4: TeammateIdle hook validates teammate work
392
- // ===========================================================================
393
- describe('AC4: TeammateIdle validation', () => {
394
- it('should validate teammate when idle event fires', () => {
395
- const handle = {
396
- teamName: '86-10-green',
397
- storyId: '86-10',
398
- phase: 'green',
399
- teammates: [
400
- { agent: 'tea', task: 'Verify tests pass', status: 'idle' },
401
- ],
402
- createdAt: new Date().toISOString(),
403
- };
404
- const teammate = handle.teammates[0];
405
- const result = checkGateOnTeammateIdle(handle, teammate, PHASE_WITH_TEAM_AND_GATE);
406
- assert.ok('passed' in result, 'Should return gate check result');
407
- assert.ok(typeof result.passed === 'boolean');
408
- });
409
- it('should include gate type in idle check result', () => {
410
- const handle = {
411
- teamName: '86-10-green',
412
- storyId: '86-10',
413
- phase: 'green',
414
- teammates: [{ agent: 'tea', status: 'idle' }],
415
- createdAt: new Date().toISOString(),
416
- };
417
- const result = checkGateOnTeammateIdle(handle, handle.teammates[0], PHASE_WITH_TEAM_AND_GATE);
418
- assert.strictEqual(result.gate, 'tests_pass', 'Gate type should match phase config');
419
- });
420
- it('should fail if teammate task implies tests but tests not passing', () => {
421
- const handle = {
422
- teamName: '86-10-green',
423
- storyId: '86-10',
424
- phase: 'green',
425
- teammates: [
426
- { agent: 'tea', task: 'Verify tests pass', status: 'crashed' },
427
- ],
428
- createdAt: new Date().toISOString(),
429
- };
430
- const result = checkGateOnTeammateIdle(handle, handle.teammates[0], PHASE_WITH_TEAM_AND_GATE);
431
- assert.strictEqual(result.passed, false, 'Crashed teammate should fail gate');
432
- });
433
- it('should pass when idle teammate completed task successfully', () => {
434
- const handle = {
435
- teamName: '86-10-green',
436
- storyId: '86-10',
437
- phase: 'green',
438
- teammates: [
439
- { agent: 'tea', task: 'Verify tests pass', status: 'idle' },
440
- ],
441
- createdAt: new Date().toISOString(),
442
- };
443
- const result = checkGateOnTeammateIdle(handle, handle.teammates[0], PHASE_WITH_TEAM_AND_GATE);
444
- assert.strictEqual(result.passed, true, 'Idle teammate should pass');
445
- });
446
- it('should pass trivially when phase has no gate', () => {
447
- const handle = {
448
- teamName: '86-10-green',
449
- storyId: '86-10',
450
- phase: 'green',
451
- teammates: [{ agent: 'architect', status: 'idle' }],
452
- createdAt: new Date().toISOString(),
453
- };
454
- const phaseNoGate = {
455
- name: 'green',
456
- agent: 'dev',
457
- team: { teammates: [{ agent: 'architect' }] },
458
- };
459
- const result = checkGateOnTeammateIdle(handle, handle.teammates[0], phaseNoGate);
460
- assert.strictEqual(result.passed, true, 'No gate = always passes');
461
- });
462
- });
463
- // ===========================================================================
464
- // AC5: Lead shuts down all teammates before exit protocol
465
- // ===========================================================================
466
- describe('AC5: Shutdown all teammates', () => {
467
- it('should shut down all active teammates', async () => {
468
- const shutdownAgents = [];
469
- const adapter = createMockAdapter({
470
- shutdownTeammate: async (params) => { shutdownAgents.push(params.agent); },
471
- });
472
- const createResult = await createTeam({
473
- phase: PHASE_WITH_TEAM,
474
- storyId: '86-10',
475
- sessionDir: join(TEST_DIR, '.session'),
476
- adapter,
477
- });
478
- const handle = createResult.data;
479
- // Manually add teammate handles to simulate spawned state
480
- handle.teammates = [
481
- { agent: 'architect', status: 'active' },
482
- { agent: 'tea', status: 'idle' },
483
- ];
484
- const result = await shutdownAllTeammates(handle, adapter);
485
- assert.strictEqual(result.success, true, 'Shutdown should succeed');
486
- assert.deepStrictEqual(shutdownAgents.sort(), ['architect', 'tea'], 'Should shut down all teammates');
487
- });
488
- it('should return count of teammates shut down', async () => {
489
- const adapter = createMockAdapter();
490
- const createResult = await createTeam({
491
- phase: PHASE_WITH_TEAM,
492
- storyId: '86-10',
493
- sessionDir: join(TEST_DIR, '.session'),
494
- adapter,
495
- });
496
- const handle = createResult.data;
497
- handle.teammates = [
498
- { agent: 'architect', status: 'active' },
499
- { agent: 'tea', status: 'idle' },
500
- ];
501
- const result = await shutdownAllTeammates(handle, adapter);
502
- assert.strictEqual(result.success, true);
503
- assert.strictEqual(result.data?.shutdownCount, 2, 'Should report 2 shutdowns');
504
- });
505
- it('should update teammate statuses to shutdown', async () => {
506
- const adapter = createMockAdapter();
507
- const createResult = await createTeam({
508
- phase: PHASE_WITH_TEAM,
509
- storyId: '86-10',
510
- sessionDir: join(TEST_DIR, '.session'),
511
- adapter,
512
- });
513
- const handle = createResult.data;
514
- handle.teammates = [
515
- { agent: 'architect', status: 'active' },
516
- { agent: 'tea', status: 'idle' },
517
- ];
518
- await shutdownAllTeammates(handle, adapter);
519
- for (const teammate of handle.teammates) {
520
- assert.strictEqual(teammate.status, 'shutdown', `${teammate.agent} should be marked shutdown`);
521
- }
522
- });
523
- it('should skip already-shutdown teammates', async () => {
524
- const shutdownAgents = [];
525
- const adapter = createMockAdapter({
526
- shutdownTeammate: async (params) => { shutdownAgents.push(params.agent); },
527
- });
528
- const createResult = await createTeam({
529
- phase: PHASE_WITH_TEAM,
530
- storyId: '86-10',
531
- sessionDir: join(TEST_DIR, '.session'),
532
- adapter,
533
- });
534
- const handle = createResult.data;
535
- handle.teammates = [
536
- { agent: 'architect', status: 'shutdown' }, // already done
537
- { agent: 'tea', status: 'active' },
538
- ];
539
- await shutdownAllTeammates(handle, adapter);
540
- assert.deepStrictEqual(shutdownAgents, ['tea'], 'Should only shut down non-shutdown teammates');
541
- });
542
- it('should succeed even with empty teammates list', async () => {
543
- const adapter = createMockAdapter();
544
- const createResult = await createTeam({
545
- phase: PHASE_WITH_TEAM,
546
- storyId: '86-10',
547
- sessionDir: join(TEST_DIR, '.session'),
548
- adapter,
549
- });
550
- const handle = createResult.data;
551
- handle.teammates = [];
552
- const result = await shutdownAllTeammates(handle, adapter);
553
- assert.strictEqual(result.success, true, 'Should succeed with no teammates');
554
- assert.strictEqual(result.data?.shutdownCount, 0);
555
- });
556
- });
557
- // ===========================================================================
558
- // AC6: TeamDelete runs before pf handoff
559
- // ===========================================================================
560
- describe('AC6: TeamDelete before handoff', () => {
561
- it('should call adapter.deleteTeam with team name', async () => {
562
- let deletedTeam = '';
563
- const adapter = createMockAdapter({
564
- deleteTeam: async (teamName) => { deletedTeam = teamName; },
565
- });
566
- const createResult = await createTeam({
567
- phase: PHASE_WITH_TEAM,
568
- storyId: '86-10',
569
- sessionDir: join(TEST_DIR, '.session'),
570
- adapter,
571
- });
572
- const handle = createResult.data;
573
- await cleanupTeam(handle, adapter);
574
- assert.strictEqual(deletedTeam, handle.teamName, 'Should delete team by name');
575
- });
576
- it('should remove team from active registry', async () => {
577
- const adapter = createMockAdapter();
578
- const createResult = await createTeam({
579
- phase: PHASE_WITH_TEAM,
580
- storyId: '86-10',
581
- sessionDir: join(TEST_DIR, '.session'),
582
- adapter,
583
- });
584
- const handle = createResult.data;
585
- await cleanupTeam(handle, adapter);
586
- const active = getActiveTeam('86-10');
587
- assert.strictEqual(active, null, 'Team should not be in registry after cleanup');
588
- });
589
- it('should return cleaned=true on success', async () => {
590
- const adapter = createMockAdapter();
591
- const createResult = await createTeam({
592
- phase: PHASE_WITH_TEAM,
593
- storyId: '86-10',
594
- sessionDir: join(TEST_DIR, '.session'),
595
- adapter,
596
- });
597
- const result = await cleanupTeam(createResult.data, adapter);
598
- assert.strictEqual(result.success, true);
599
- assert.strictEqual(result.data?.cleaned, true);
600
- });
601
- it('should succeed even when adapter.deleteTeam fails', async () => {
602
- const adapter = createMockAdapter({
603
- deleteTeam: async () => { throw new Error('Team not found'); },
604
- });
605
- const createResult = await createTeam({
606
- phase: PHASE_WITH_TEAM,
607
- storyId: '86-10',
608
- sessionDir: join(TEST_DIR, '.session'),
609
- adapter: createMockAdapter(), // working adapter for create
610
- });
611
- // Cleanup with broken adapter — should not throw
612
- const result = await cleanupTeam(createResult.data, adapter);
613
- assert.strictEqual(result.success, true, 'Cleanup should succeed even if delete fails');
614
- });
615
- it('should handle cleanup for non-existent team gracefully', async () => {
616
- const handle = {
617
- teamName: 'nonexistent-team',
618
- storyId: 'no-story',
619
- phase: 'green',
620
- teammates: [],
621
- createdAt: new Date().toISOString(),
622
- };
623
- const result = await cleanupTeam(handle, createMockAdapter());
624
- assert.strictEqual(result.success, true, 'Should succeed for non-existent team');
625
- });
626
- });
627
- // ===========================================================================
628
- // AC7: Session file updated with teammate activity summary
629
- // ===========================================================================
630
- describe('AC7: Team activity summary for audit', () => {
631
- it('should generate summary with team name', () => {
632
- const handle = {
633
- teamName: '86-10-green',
634
- storyId: '86-10',
635
- phase: 'green',
636
- teammates: [
637
- { agent: 'architect', task: 'Review patterns', status: 'shutdown' },
638
- { agent: 'tea', task: 'Verify tests', status: 'shutdown' },
639
- ],
640
- createdAt: new Date().toISOString(),
641
- };
642
- const summary = generateTeamSummary(handle);
643
- assert.strictEqual(summary.teamName, '86-10-green', 'Summary should include team name');
644
- });
645
- it('should include all team members with status', () => {
646
- const handle = {
647
- teamName: '86-10-green',
648
- storyId: '86-10',
649
- phase: 'green',
650
- teammates: [
651
- { agent: 'architect', task: 'Review patterns', status: 'shutdown' },
652
- { agent: 'tea', task: 'Verify tests', status: 'crashed' },
653
- ],
654
- createdAt: new Date().toISOString(),
655
- };
656
- const summary = generateTeamSummary(handle);
657
- assert.strictEqual(summary.members.length, 2, 'Should list all members');
658
- const architectMember = summary.members.find((m) => m.agent === 'architect');
659
- assert.ok(architectMember, 'Should include architect');
660
- assert.strictEqual(architectMember.status, 'shutdown');
661
- });
662
- it('should include task descriptions in member summaries', () => {
663
- const handle = {
664
- teamName: '86-10-green',
665
- storyId: '86-10',
666
- phase: 'green',
667
- teammates: [
668
- { agent: 'architect', task: 'Review patterns', status: 'shutdown' },
669
- ],
670
- createdAt: new Date().toISOString(),
671
- };
672
- const summary = generateTeamSummary(handle);
673
- assert.strictEqual(summary.members[0].task, 'Review patterns');
674
- });
675
- it('should report clean shutdown status', () => {
676
- const handle = {
677
- teamName: '86-10-green',
678
- storyId: '86-10',
679
- phase: 'green',
680
- teammates: [
681
- { agent: 'architect', status: 'shutdown' },
682
- { agent: 'tea', status: 'shutdown' },
683
- ],
684
- createdAt: new Date().toISOString(),
685
- };
686
- const summary = generateTeamSummary(handle);
687
- assert.strictEqual(summary.cleanShutdown, true, 'All shutdown = clean');
688
- });
689
- it('should report unclean shutdown when teammate crashed', () => {
690
- const handle = {
691
- teamName: '86-10-green',
692
- storyId: '86-10',
693
- phase: 'green',
694
- teammates: [
695
- { agent: 'architect', status: 'shutdown' },
696
- { agent: 'tea', status: 'crashed' },
697
- ],
698
- createdAt: new Date().toISOString(),
699
- };
700
- const summary = generateTeamSummary(handle);
701
- assert.strictEqual(summary.cleanShutdown, false, 'Crashed teammate = unclean');
702
- });
703
- it('should include story ID and phase for audit context', () => {
704
- const handle = {
705
- teamName: '86-10-green',
706
- storyId: '86-10',
707
- phase: 'green',
708
- teammates: [],
709
- createdAt: new Date().toISOString(),
710
- };
711
- const summary = generateTeamSummary(handle);
712
- assert.strictEqual(summary.storyId, '86-10');
713
- assert.strictEqual(summary.phase, 'green');
714
- });
715
- });
716
- // ===========================================================================
717
- // AC8: Sidecar file locking for concurrent teammate writes
718
- // ===========================================================================
719
- describe('AC8: Sidecar file locking', () => {
720
- it('should acquire lock for sidecar file', async () => {
721
- const filePath = join(TEST_DIR, '.pennyfarthing', 'sidecars', 'patterns.md');
722
- const result = await acquireSidecarLock(filePath, '86-10');
723
- assert.strictEqual(result.success, true, 'Lock acquisition should succeed');
724
- assert.ok(result.data, 'Should return lock handle');
725
- assert.ok(result.data.lockPath, 'Lock handle should have lock path');
726
- });
727
- it('should include story ID in lock', async () => {
728
- const filePath = join(TEST_DIR, 'sidecars', 'patterns.md');
729
- const result = await acquireSidecarLock(filePath, '86-10');
730
- assert.strictEqual(result.success, true);
731
- assert.strictEqual(result.data.storyId, '86-10', 'Lock should track story ID');
732
- });
733
- it('should set acquiredAt timestamp', async () => {
734
- const filePath = join(TEST_DIR, 'sidecars', 'patterns.md');
735
- const result = await acquireSidecarLock(filePath, '86-10');
736
- assert.strictEqual(result.success, true);
737
- const date = new Date(result.data.acquiredAt);
738
- assert.ok(!isNaN(date.getTime()), 'acquiredAt should be valid ISO date');
739
- });
740
- it('should fail to acquire when already locked by different story', async () => {
741
- const filePath = join(TEST_DIR, 'sidecars', 'patterns.md');
742
- // First lock succeeds
743
- const first = await acquireSidecarLock(filePath, '86-10');
744
- assert.strictEqual(first.success, true);
745
- // Second lock for different story should fail
746
- const second = await acquireSidecarLock(filePath, '86-11');
747
- assert.strictEqual(second.success, false, 'Should fail when already locked');
748
- assert.ok(second.error, 'Should include error message');
749
- });
750
- it('should allow re-entrant lock from same story', async () => {
751
- const filePath = join(TEST_DIR, 'sidecars', 'patterns.md');
752
- const first = await acquireSidecarLock(filePath, '86-10');
753
- assert.strictEqual(first.success, true);
754
- // Same story re-acquiring should succeed (re-entrant)
755
- const second = await acquireSidecarLock(filePath, '86-10');
756
- assert.strictEqual(second.success, true, 'Re-entrant lock should succeed');
757
- });
758
- it('should release lock successfully', async () => {
759
- const filePath = join(TEST_DIR, 'sidecars', 'patterns.md');
760
- const acquired = await acquireSidecarLock(filePath, '86-10');
761
- assert.strictEqual(acquired.success, true);
762
- const released = releaseSidecarLock(acquired.data);
763
- assert.strictEqual(released.success, true, 'Release should succeed');
764
- });
765
- it('should allow new lock after release', async () => {
766
- const filePath = join(TEST_DIR, 'sidecars', 'patterns.md');
767
- // Acquire and release
768
- const first = await acquireSidecarLock(filePath, '86-10');
769
- releaseSidecarLock(first.data);
770
- // New lock should succeed
771
- const second = await acquireSidecarLock(filePath, '86-11');
772
- assert.strictEqual(second.success, true, 'Should acquire after release');
773
- });
774
- it('should handle release of already-released lock gracefully', () => {
775
- const stalelock = {
776
- lockPath: join(TEST_DIR, 'stale.lock'),
777
- storyId: '86-10',
778
- acquiredAt: new Date().toISOString(),
779
- };
780
- const result = releaseSidecarLock(stalelock);
781
- assert.strictEqual(result.success, true, 'Should succeed for stale lock');
782
- });
783
- });
784
- // ===========================================================================
785
- // AC9: Graceful degradation when teammate crashes
786
- // ===========================================================================
787
- describe('AC9: Graceful degradation on teammate crash', () => {
788
- it('should continue spawning remaining teammates when one fails', async () => {
789
- let spawnCount = 0;
790
- const adapter = createMockAdapter({
791
- spawnTeammate: async (params) => {
792
- spawnCount++;
793
- if (params.agent === 'architect') {
794
- throw new Error('Spawn failed: connection refused');
795
- }
796
- return { agentId: `mock-${params.agent}` };
797
- },
798
- });
799
- const createResult = await createTeam({
800
- phase: PHASE_WITH_TEAM,
801
- storyId: '86-10',
802
- sessionDir: join(TEST_DIR, '.session'),
803
- adapter: createMockAdapter(), // working adapter for create
804
- });
805
- const result = await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', adapter);
806
- // Should still succeed overall (degraded mode)
807
- assert.strictEqual(result.success, true, 'Should succeed in degraded mode');
808
- assert.strictEqual(spawnCount, 2, 'Should attempt both spawns');
809
- });
810
- it('should mark crashed teammate with crashed status', async () => {
811
- const adapter = createMockAdapter({
812
- spawnTeammate: async (params) => {
813
- if (params.agent === 'architect') {
814
- throw new Error('Spawn failed');
815
- }
816
- return { agentId: `mock-${params.agent}` };
817
- },
818
- });
819
- const createResult = await createTeam({
820
- phase: PHASE_WITH_TEAM,
821
- storyId: '86-10',
822
- sessionDir: join(TEST_DIR, '.session'),
823
- adapter: createMockAdapter(),
824
- });
825
- const result = await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', adapter);
826
- const crashed = result.data.find((t) => t.agent === 'architect');
827
- assert.ok(crashed, 'Crashed teammate should still appear in list');
828
- assert.strictEqual(crashed.status, 'crashed', 'Failed spawn should be marked crashed');
829
- });
830
- it('should not block shutdown when teammate already crashed', async () => {
831
- const adapter = createMockAdapter({
832
- shutdownTeammate: async (params) => {
833
- if (params.agent === 'architect') {
834
- throw new Error('Cannot shutdown: process not found');
835
- }
836
- },
837
- });
838
- const handle = {
839
- teamName: '86-10-green',
840
- storyId: '86-10',
841
- phase: 'green',
842
- teammates: [
843
- { agent: 'architect', status: 'crashed' },
844
- { agent: 'tea', status: 'active' },
845
- ],
846
- createdAt: new Date().toISOString(),
847
- };
848
- const result = await shutdownAllTeammates(handle, adapter);
849
- assert.strictEqual(result.success, true, 'Shutdown should succeed despite crash');
850
- });
851
- it('should not block cleanup when team deletion fails', async () => {
852
- const adapter = createMockAdapter({
853
- deleteTeam: async () => { throw new Error('Team already deleted'); },
854
- });
855
- const handle = {
856
- teamName: '86-10-green',
857
- storyId: '86-10',
858
- phase: 'green',
859
- teammates: [],
860
- createdAt: new Date().toISOString(),
861
- };
862
- const result = await cleanupTeam(handle, adapter);
863
- assert.strictEqual(result.success, true, 'Cleanup should not throw');
864
- });
865
- it('should return error info for crashed teammates in spawn result', async () => {
866
- const adapter = createMockAdapter({
867
- spawnTeammate: async (params) => {
868
- if (params.agent === 'architect') {
869
- throw new Error('Connection refused');
870
- }
871
- return { agentId: `mock-${params.agent}` };
872
- },
873
- });
874
- const createResult = await createTeam({
875
- phase: PHASE_WITH_TEAM,
876
- storyId: '86-10',
877
- sessionDir: join(TEST_DIR, '.session'),
878
- adapter: createMockAdapter(),
879
- });
880
- const result = await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', adapter);
881
- // Successful teammates should still be in the result
882
- const tea = result.data.find((t) => t.agent === 'tea');
883
- assert.ok(tea, 'Successful teammate should be in result');
884
- assert.strictEqual(tea.status, 'spawned');
885
- });
886
- });
887
- // ===========================================================================
888
- // Result format compliance
889
- // ===========================================================================
890
- describe('Result format compliance', () => {
891
- it('should return {success, data?, error?} from createTeam', async () => {
892
- const result = await createTeam({
893
- phase: PHASE_WITH_TEAM,
894
- storyId: '86-10',
895
- sessionDir: join(TEST_DIR, '.session'),
896
- adapter: createMockAdapter(),
897
- });
898
- assert.ok('success' in result, 'Must have success field');
899
- assert.strictEqual(typeof result.success, 'boolean', 'success must be boolean');
900
- });
901
- it('should return {success, data?, error?} from spawnTeammates', async () => {
902
- const createResult = await createTeam({
903
- phase: PHASE_WITH_TEAM,
904
- storyId: '86-10',
905
- sessionDir: join(TEST_DIR, '.session'),
906
- adapter: createMockAdapter(),
907
- });
908
- const result = await spawnTeammates(createResult.data, PHASE_WITH_TEAM.team, '86-10', 'green', createMockAdapter());
909
- assert.ok('success' in result, 'Must have success field');
910
- assert.strictEqual(typeof result.success, 'boolean');
911
- });
912
- it('should return {success, data?, error?} from shutdownAllTeammates', async () => {
913
- const handle = {
914
- teamName: '86-10-green',
915
- storyId: '86-10',
916
- phase: 'green',
917
- teammates: [],
918
- createdAt: new Date().toISOString(),
919
- };
920
- const result = await shutdownAllTeammates(handle, createMockAdapter());
921
- assert.ok('success' in result, 'Must have success field');
922
- assert.strictEqual(typeof result.success, 'boolean');
923
- });
924
- it('should return {success, data?, error?} from cleanupTeam', async () => {
925
- const handle = {
926
- teamName: '86-10-green',
927
- storyId: '86-10',
928
- phase: 'green',
929
- teammates: [],
930
- createdAt: new Date().toISOString(),
931
- };
932
- const result = await cleanupTeam(handle, createMockAdapter());
933
- assert.ok('success' in result, 'Must have success field');
934
- assert.strictEqual(typeof result.success, 'boolean');
935
- });
936
- });
937
- // ===========================================================================
938
- // ProcessAdapter integration
939
- // ===========================================================================
940
- describe('ProcessAdapter integration', () => {
941
- it('should work without adapter (in-memory test mode)', async () => {
942
- const result = await createTeam({
943
- phase: PHASE_WITH_TEAM,
944
- storyId: '86-10',
945
- sessionDir: join(TEST_DIR, '.session'),
946
- // No adapter — test mode
947
- });
948
- assert.strictEqual(result.success, true, 'Should work without adapter');
949
- assert.ok(result.data, 'Should return handle even without adapter');
950
- });
951
- it('should propagate adapter errors as result errors, not exceptions', async () => {
952
- const adapter = createMockAdapter({
953
- createTeam: async () => { throw new Error('Network timeout'); },
954
- });
955
- const result = await createTeam({
956
- phase: PHASE_WITH_TEAM,
957
- storyId: '86-10',
958
- sessionDir: join(TEST_DIR, '.session'),
959
- adapter,
960
- });
961
- assert.strictEqual(result.success, false, 'Should fail, not throw');
962
- assert.ok(result.error?.includes('Network timeout'), 'Should include error message');
963
- });
964
- });
965
- });
966
- //# sourceMappingURL=team-lifecycle.test.js.map