@shepai/cli 1.150.0 → 1.150.1-pr441.b309db7

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 (467) hide show
  1. package/apis/json-schema/FeatureFlags.yaml +5 -0
  2. package/apis/json-schema/NotificationEventConfig.yaml +15 -0
  3. package/apis/json-schema/NotificationEventType.yaml +3 -0
  4. package/apis/json-schema/ScheduledWorkflow.yaml +46 -0
  5. package/apis/json-schema/WorkflowExecution.yaml +38 -0
  6. package/apis/json-schema/WorkflowExecutionStatus.yaml +10 -0
  7. package/apis/json-schema/WorkflowTriggerType.yaml +7 -0
  8. package/dist/packages/core/src/application/ports/output/repositories/index.d.ts +2 -0
  9. package/dist/packages/core/src/application/ports/output/repositories/index.d.ts.map +1 -1
  10. package/dist/packages/core/src/application/ports/output/repositories/workflow-execution-repository.interface.d.ts +66 -0
  11. package/dist/packages/core/src/application/ports/output/repositories/workflow-execution-repository.interface.d.ts.map +1 -0
  12. package/dist/packages/core/src/application/ports/output/repositories/workflow-execution-repository.interface.js +11 -0
  13. package/dist/packages/core/src/application/ports/output/repositories/workflow-repository.interface.d.ts +81 -0
  14. package/dist/packages/core/src/application/ports/output/repositories/workflow-repository.interface.d.ts.map +1 -0
  15. package/dist/packages/core/src/application/ports/output/repositories/workflow-repository.interface.js +11 -0
  16. package/dist/packages/core/src/application/ports/output/services/clock.interface.d.ts +28 -0
  17. package/dist/packages/core/src/application/ports/output/services/clock.interface.d.ts.map +1 -0
  18. package/dist/packages/core/src/application/ports/output/services/clock.interface.js +13 -0
  19. package/dist/packages/core/src/application/ports/output/services/index.d.ts +1 -0
  20. package/dist/packages/core/src/application/ports/output/services/index.d.ts.map +1 -1
  21. package/dist/packages/core/src/application/use-cases/workflows/create-workflow.use-case.d.ts +28 -0
  22. package/dist/packages/core/src/application/use-cases/workflows/create-workflow.use-case.d.ts.map +1 -0
  23. package/dist/packages/core/src/application/use-cases/workflows/create-workflow.use-case.js +72 -0
  24. package/dist/packages/core/src/application/use-cases/workflows/cron-helpers.d.ts +27 -0
  25. package/dist/packages/core/src/application/use-cases/workflows/cron-helpers.d.ts.map +1 -0
  26. package/dist/packages/core/src/application/use-cases/workflows/cron-helpers.js +43 -0
  27. package/dist/packages/core/src/application/use-cases/workflows/delete-workflow.use-case.d.ts +18 -0
  28. package/dist/packages/core/src/application/use-cases/workflows/delete-workflow.use-case.d.ts.map +1 -0
  29. package/dist/packages/core/src/application/use-cases/workflows/delete-workflow.use-case.js +64 -0
  30. package/dist/packages/core/src/application/use-cases/workflows/get-workflow-history.use-case.d.ts +16 -0
  31. package/dist/packages/core/src/application/use-cases/workflows/get-workflow-history.use-case.d.ts.map +1 -0
  32. package/dist/packages/core/src/application/use-cases/workflows/get-workflow-history.use-case.js +46 -0
  33. package/dist/packages/core/src/application/use-cases/workflows/get-workflow.use-case.d.ts +13 -0
  34. package/dist/packages/core/src/application/use-cases/workflows/get-workflow.use-case.d.ts.map +1 -0
  35. package/dist/packages/core/src/application/use-cases/workflows/get-workflow.use-case.js +42 -0
  36. package/dist/packages/core/src/application/use-cases/workflows/list-workflows.use-case.d.ts +13 -0
  37. package/dist/packages/core/src/application/use-cases/workflows/list-workflows.use-case.d.ts.map +1 -0
  38. package/dist/packages/core/src/application/use-cases/workflows/list-workflows.use-case.js +33 -0
  39. package/dist/packages/core/src/application/use-cases/workflows/run-workflow.use-case.d.ts +19 -0
  40. package/dist/packages/core/src/application/use-cases/workflows/run-workflow.use-case.d.ts.map +1 -0
  41. package/dist/packages/core/src/application/use-cases/workflows/run-workflow.use-case.js +62 -0
  42. package/dist/packages/core/src/application/use-cases/workflows/schedule-workflow.use-case.d.ts +22 -0
  43. package/dist/packages/core/src/application/use-cases/workflows/schedule-workflow.use-case.d.ts.map +1 -0
  44. package/dist/packages/core/src/application/use-cases/workflows/schedule-workflow.use-case.js +73 -0
  45. package/dist/packages/core/src/application/use-cases/workflows/templates/branch-rebase.template.d.ts +9 -0
  46. package/dist/packages/core/src/application/use-cases/workflows/templates/branch-rebase.template.d.ts.map +1 -0
  47. package/dist/packages/core/src/application/use-cases/workflows/templates/branch-rebase.template.js +27 -0
  48. package/dist/packages/core/src/application/use-cases/workflows/templates/issue-triage.template.d.ts +11 -0
  49. package/dist/packages/core/src/application/use-cases/workflows/templates/issue-triage.template.d.ts.map +1 -0
  50. package/dist/packages/core/src/application/use-cases/workflows/templates/issue-triage.template.js +27 -0
  51. package/dist/packages/core/src/application/use-cases/workflows/toggle-workflow.use-case.d.ts +16 -0
  52. package/dist/packages/core/src/application/use-cases/workflows/toggle-workflow.use-case.d.ts.map +1 -0
  53. package/dist/packages/core/src/application/use-cases/workflows/toggle-workflow.use-case.js +60 -0
  54. package/dist/packages/core/src/application/use-cases/workflows/update-workflow.use-case.d.ts +25 -0
  55. package/dist/packages/core/src/application/use-cases/workflows/update-workflow.use-case.d.ts.map +1 -0
  56. package/dist/packages/core/src/application/use-cases/workflows/update-workflow.use-case.js +83 -0
  57. package/dist/packages/core/src/domain/factories/settings-defaults.factory.d.ts.map +1 -1
  58. package/dist/packages/core/src/domain/factories/settings-defaults.factory.js +4 -0
  59. package/dist/packages/core/src/domain/generated/output.d.ts +114 -1
  60. package/dist/packages/core/src/domain/generated/output.d.ts.map +1 -1
  61. package/dist/packages/core/src/domain/generated/output.js +16 -0
  62. package/dist/packages/core/src/infrastructure/di/container.d.ts.map +1 -1
  63. package/dist/packages/core/src/infrastructure/di/container.js +66 -0
  64. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/settings.mapper.d.ts +4 -0
  65. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/settings.mapper.d.ts.map +1 -1
  66. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/settings.mapper.js +8 -0
  67. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/workflow-execution.mapper.d.ts +46 -0
  68. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/workflow-execution.mapper.d.ts.map +1 -0
  69. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/workflow-execution.mapper.js +57 -0
  70. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/workflow.mapper.d.ts +51 -0
  71. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/workflow.mapper.d.ts.map +1 -0
  72. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/workflow.mapper.js +71 -0
  73. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/044-add-workflow-notification-events.d.ts +12 -0
  74. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/044-add-workflow-notification-events.d.ts.map +1 -0
  75. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/044-add-workflow-notification-events.js +22 -0
  76. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/045-create-scheduled-workflows.d.ts +14 -0
  77. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/045-create-scheduled-workflows.d.ts.map +1 -0
  78. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/045-create-scheduled-workflows.js +65 -0
  79. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/046-add-feature-flag-scheduled-workflows.d.ts +11 -0
  80. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/046-add-feature-flag-scheduled-workflows.d.ts.map +1 -0
  81. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/046-add-feature-flag-scheduled-workflows.js +17 -0
  82. package/dist/packages/core/src/infrastructure/repositories/sqlite-workflow-execution.repository.d.ts +24 -0
  83. package/dist/packages/core/src/infrastructure/repositories/sqlite-workflow-execution.repository.d.ts.map +1 -0
  84. package/dist/packages/core/src/infrastructure/repositories/sqlite-workflow-execution.repository.js +92 -0
  85. package/dist/packages/core/src/infrastructure/repositories/sqlite-workflow.repository.d.ts +26 -0
  86. package/dist/packages/core/src/infrastructure/repositories/sqlite-workflow.repository.d.ts.map +1 -0
  87. package/dist/packages/core/src/infrastructure/repositories/sqlite-workflow.repository.js +117 -0
  88. package/dist/packages/core/src/infrastructure/services/agents/feature-agent/nodes/evidence.node.d.ts.map +1 -1
  89. package/dist/packages/core/src/infrastructure/services/agents/feature-agent/nodes/evidence.node.js +2 -1
  90. package/dist/packages/core/src/infrastructure/services/agents/feature-agent/nodes/fast-implement.node.d.ts.map +1 -1
  91. package/dist/packages/core/src/infrastructure/services/agents/feature-agent/nodes/fast-implement.node.js +11 -42
  92. package/dist/packages/core/src/infrastructure/services/agents/feature-agent/nodes/implement.node.d.ts.map +1 -1
  93. package/dist/packages/core/src/infrastructure/services/agents/feature-agent/nodes/implement.node.js +2 -3
  94. package/dist/packages/core/src/infrastructure/services/clock.d.ts +11 -0
  95. package/dist/packages/core/src/infrastructure/services/clock.d.ts.map +1 -0
  96. package/dist/packages/core/src/infrastructure/services/clock.js +22 -0
  97. package/dist/packages/core/src/infrastructure/services/notifications/notification.service.d.ts.map +1 -1
  98. package/dist/packages/core/src/infrastructure/services/notifications/notification.service.js +3 -0
  99. package/dist/packages/core/src/infrastructure/services/workflow-scheduler/workflow-scheduler.service.d.ts +69 -0
  100. package/dist/packages/core/src/infrastructure/services/workflow-scheduler/workflow-scheduler.service.d.ts.map +1 -0
  101. package/dist/packages/core/src/infrastructure/services/workflow-scheduler/workflow-scheduler.service.js +270 -0
  102. package/dist/src/presentation/cli/commands/_serve.command.d.ts.map +1 -1
  103. package/dist/src/presentation/cli/commands/_serve.command.js +14 -0
  104. package/dist/src/presentation/cli/commands/workflow/create.command.d.ts +13 -0
  105. package/dist/src/presentation/cli/commands/workflow/create.command.d.ts.map +1 -0
  106. package/dist/src/presentation/cli/commands/workflow/create.command.js +83 -0
  107. package/dist/src/presentation/cli/commands/workflow/delete.command.d.ts +13 -0
  108. package/dist/src/presentation/cli/commands/workflow/delete.command.d.ts.map +1 -0
  109. package/dist/src/presentation/cli/commands/workflow/delete.command.js +50 -0
  110. package/dist/src/presentation/cli/commands/workflow/disable.command.d.ts +11 -0
  111. package/dist/src/presentation/cli/commands/workflow/disable.command.d.ts.map +1 -0
  112. package/dist/src/presentation/cli/commands/workflow/disable.command.js +32 -0
  113. package/dist/src/presentation/cli/commands/workflow/enable.command.d.ts +11 -0
  114. package/dist/src/presentation/cli/commands/workflow/enable.command.d.ts.map +1 -0
  115. package/dist/src/presentation/cli/commands/workflow/enable.command.js +35 -0
  116. package/dist/src/presentation/cli/commands/workflow/format-helpers.d.ts +35 -0
  117. package/dist/src/presentation/cli/commands/workflow/format-helpers.d.ts.map +1 -0
  118. package/dist/src/presentation/cli/commands/workflow/format-helpers.js +89 -0
  119. package/dist/src/presentation/cli/commands/workflow/history.command.d.ts +12 -0
  120. package/dist/src/presentation/cli/commands/workflow/history.command.d.ts.map +1 -0
  121. package/dist/src/presentation/cli/commands/workflow/history.command.js +46 -0
  122. package/dist/src/presentation/cli/commands/workflow/index.d.ts +26 -0
  123. package/dist/src/presentation/cli/commands/workflow/index.d.ts.map +1 -0
  124. package/dist/src/presentation/cli/commands/workflow/index.js +48 -0
  125. package/dist/src/presentation/cli/commands/workflow/list.command.d.ts +11 -0
  126. package/dist/src/presentation/cli/commands/workflow/list.command.d.ts.map +1 -0
  127. package/dist/src/presentation/cli/commands/workflow/list.command.js +52 -0
  128. package/dist/src/presentation/cli/commands/workflow/run.command.d.ts +11 -0
  129. package/dist/src/presentation/cli/commands/workflow/run.command.d.ts.map +1 -0
  130. package/dist/src/presentation/cli/commands/workflow/run.command.js +37 -0
  131. package/dist/src/presentation/cli/commands/workflow/schedule.command.d.ts +13 -0
  132. package/dist/src/presentation/cli/commands/workflow/schedule.command.d.ts.map +1 -0
  133. package/dist/src/presentation/cli/commands/workflow/schedule.command.js +76 -0
  134. package/dist/src/presentation/cli/commands/workflow/show.command.d.ts +11 -0
  135. package/dist/src/presentation/cli/commands/workflow/show.command.d.ts.map +1 -0
  136. package/dist/src/presentation/cli/commands/workflow/show.command.js +93 -0
  137. package/dist/src/presentation/cli/commands/workflow/update.command.d.ts +13 -0
  138. package/dist/src/presentation/cli/commands/workflow/update.command.d.ts.map +1 -0
  139. package/dist/src/presentation/cli/commands/workflow/update.command.js +67 -0
  140. package/dist/src/presentation/cli/index.js +2 -0
  141. package/dist/src/presentation/web/app/actions/get-workflow-history.d.ts +7 -0
  142. package/dist/src/presentation/web/app/actions/get-workflow-history.d.ts.map +1 -0
  143. package/dist/src/presentation/web/app/actions/get-workflow-history.js +17 -0
  144. package/dist/src/presentation/web/app/actions/list-workflows.d.ts +7 -0
  145. package/dist/src/presentation/web/app/actions/list-workflows.d.ts.map +1 -0
  146. package/dist/src/presentation/web/app/actions/list-workflows.js +17 -0
  147. package/dist/src/presentation/web/app/actions/toggle-workflow.d.ts +8 -0
  148. package/dist/src/presentation/web/app/actions/toggle-workflow.d.ts.map +1 -0
  149. package/dist/src/presentation/web/app/actions/toggle-workflow.js +19 -0
  150. package/dist/src/presentation/web/app/actions/trigger-workflow.d.ts +8 -0
  151. package/dist/src/presentation/web/app/actions/trigger-workflow.d.ts.map +1 -0
  152. package/dist/src/presentation/web/app/actions/trigger-workflow.js +19 -0
  153. package/dist/src/presentation/web/app/workflows/page.d.ts +4 -0
  154. package/dist/src/presentation/web/app/workflows/page.d.ts.map +1 -0
  155. package/dist/src/presentation/web/app/workflows/page.js +18 -0
  156. package/dist/src/presentation/web/components/common/feature-node/derive-feature-state.d.ts.map +1 -1
  157. package/dist/src/presentation/web/components/common/feature-node/derive-feature-state.js +4 -0
  158. package/dist/src/presentation/web/components/common/repository-node/repository-drawer.stories.d.ts.map +1 -1
  159. package/dist/src/presentation/web/components/common/repository-node/repository-drawer.stories.js +1 -0
  160. package/dist/src/presentation/web/components/features/settings/feature-flags-settings-section.d.ts.map +1 -1
  161. package/dist/src/presentation/web/components/features/settings/feature-flags-settings-section.js +3 -0
  162. package/dist/src/presentation/web/components/features/settings/feature-flags-settings-section.stories.d.ts.map +1 -1
  163. package/dist/src/presentation/web/components/features/settings/feature-flags-settings-section.stories.js +3 -0
  164. package/dist/src/presentation/web/components/features/settings/notification-settings-section.d.ts.map +1 -1
  165. package/dist/src/presentation/web/components/features/settings/notification-settings-section.js +6 -1
  166. package/dist/src/presentation/web/components/features/settings/notification-settings-section.stories.d.ts.map +1 -1
  167. package/dist/src/presentation/web/components/features/settings/notification-settings-section.stories.js +6 -0
  168. package/dist/src/presentation/web/components/features/settings/settings-page-client.d.ts.map +1 -1
  169. package/dist/src/presentation/web/components/features/settings/settings-page-client.js +19 -2
  170. package/dist/src/presentation/web/components/features/settings/settings-page-client.stories.d.ts.map +1 -1
  171. package/dist/src/presentation/web/components/features/settings/settings-page-client.stories.js +1 -0
  172. package/dist/src/presentation/web/components/features/workflows/workflow-execution-history.d.ts +7 -0
  173. package/dist/src/presentation/web/components/features/workflows/workflow-execution-history.d.ts.map +1 -0
  174. package/dist/src/presentation/web/components/features/workflows/workflow-execution-history.js +84 -0
  175. package/dist/src/presentation/web/components/features/workflows/workflow-execution-history.stories.d.ts +11 -0
  176. package/dist/src/presentation/web/components/features/workflows/workflow-execution-history.stories.d.ts.map +1 -0
  177. package/dist/src/presentation/web/components/features/workflows/workflow-execution-history.stories.js +104 -0
  178. package/dist/src/presentation/web/components/features/workflows/workflow-list-item.d.ts +8 -0
  179. package/dist/src/presentation/web/components/features/workflows/workflow-list-item.d.ts.map +1 -0
  180. package/dist/src/presentation/web/components/features/workflows/workflow-list-item.js +68 -0
  181. package/dist/src/presentation/web/components/features/workflows/workflow-list-item.stories.d.ts +11 -0
  182. package/dist/src/presentation/web/components/features/workflows/workflow-list-item.stories.d.ts.map +1 -0
  183. package/dist/src/presentation/web/components/features/workflows/workflow-list-item.stories.js +89 -0
  184. package/dist/src/presentation/web/components/features/workflows/workflow-list.d.ts +8 -0
  185. package/dist/src/presentation/web/components/features/workflows/workflow-list.d.ts.map +1 -0
  186. package/dist/src/presentation/web/components/features/workflows/workflow-list.js +10 -0
  187. package/dist/src/presentation/web/components/features/workflows/workflow-list.stories.d.ts +11 -0
  188. package/dist/src/presentation/web/components/features/workflows/workflow-list.stories.d.ts.map +1 -0
  189. package/dist/src/presentation/web/components/features/workflows/workflow-list.stories.js +90 -0
  190. package/dist/src/presentation/web/components/features/workflows/workflows-page-client.d.ts +6 -0
  191. package/dist/src/presentation/web/components/features/workflows/workflows-page-client.d.ts.map +1 -0
  192. package/dist/src/presentation/web/components/features/workflows/workflows-page-client.js +14 -0
  193. package/dist/src/presentation/web/components/features/workflows/workflows-page-client.stories.d.ts +10 -0
  194. package/dist/src/presentation/web/components/features/workflows/workflows-page-client.stories.d.ts.map +1 -0
  195. package/dist/src/presentation/web/components/features/workflows/workflows-page-client.stories.js +70 -0
  196. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.d.ts.map +1 -1
  197. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.js +2 -2
  198. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.stories.d.ts.map +1 -1
  199. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.stories.js +1 -0
  200. package/dist/src/presentation/web/hooks/feature-flags-context.d.ts.map +1 -1
  201. package/dist/src/presentation/web/hooks/feature-flags-context.js +1 -0
  202. package/dist/src/presentation/web/lib/feature-flags.d.ts +2 -0
  203. package/dist/src/presentation/web/lib/feature-flags.d.ts.map +1 -1
  204. package/dist/src/presentation/web/lib/feature-flags.js +5 -0
  205. package/dist/tsconfig.build.tsbuildinfo +1 -1
  206. package/package.json +2 -1
  207. package/web/.next/BUILD_ID +1 -1
  208. package/web/.next/app-path-routes-manifest.json +2 -1
  209. package/web/.next/build-manifest.json +2 -2
  210. package/web/.next/fallback-build-manifest.json +2 -2
  211. package/web/.next/prerender-manifest.json +3 -3
  212. package/web/.next/required-server-files.js +3 -3
  213. package/web/.next/required-server-files.json +3 -3
  214. package/web/.next/routes-manifest.json +6 -0
  215. package/web/.next/server/app/(dashboard)/@drawer/adopt/page/server-reference-manifest.json +28 -28
  216. package/web/.next/server/app/(dashboard)/@drawer/adopt/page.js +2 -2
  217. package/web/.next/server/app/(dashboard)/@drawer/adopt/page.js.nft.json +1 -1
  218. package/web/.next/server/app/(dashboard)/@drawer/adopt/page_client-reference-manifest.js +1 -1
  219. package/web/.next/server/app/(dashboard)/@drawer/create/page/server-reference-manifest.json +28 -28
  220. package/web/.next/server/app/(dashboard)/@drawer/create/page.js +2 -2
  221. package/web/.next/server/app/(dashboard)/@drawer/create/page.js.nft.json +1 -1
  222. package/web/.next/server/app/(dashboard)/@drawer/create/page_client-reference-manifest.js +1 -1
  223. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
  224. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page.js +2 -2
  225. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  226. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  227. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page/server-reference-manifest.json +36 -36
  228. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page.js +2 -2
  229. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page.js.nft.json +1 -1
  230. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page_client-reference-manifest.js +1 -1
  231. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
  232. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page.js +2 -2
  233. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page.js.nft.json +1 -1
  234. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  235. package/web/.next/server/app/(dashboard)/create/page/server-reference-manifest.json +28 -28
  236. package/web/.next/server/app/(dashboard)/create/page.js +2 -2
  237. package/web/.next/server/app/(dashboard)/create/page.js.nft.json +1 -1
  238. package/web/.next/server/app/(dashboard)/create/page_client-reference-manifest.js +1 -1
  239. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
  240. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page.js +2 -2
  241. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  242. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  243. package/web/.next/server/app/(dashboard)/feature/[featureId]/page/server-reference-manifest.json +36 -36
  244. package/web/.next/server/app/(dashboard)/feature/[featureId]/page.js +2 -2
  245. package/web/.next/server/app/(dashboard)/feature/[featureId]/page.js.nft.json +1 -1
  246. package/web/.next/server/app/(dashboard)/feature/[featureId]/page_client-reference-manifest.js +1 -1
  247. package/web/.next/server/app/(dashboard)/page/server-reference-manifest.json +26 -26
  248. package/web/.next/server/app/(dashboard)/page.js +2 -2
  249. package/web/.next/server/app/(dashboard)/page.js.nft.json +1 -1
  250. package/web/.next/server/app/(dashboard)/page_client-reference-manifest.js +1 -1
  251. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
  252. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page.js +2 -2
  253. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page.js.nft.json +1 -1
  254. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  255. package/web/.next/server/app/_global-error/page.js +1 -1
  256. package/web/.next/server/app/_global-error/page.js.nft.json +1 -1
  257. package/web/.next/server/app/_global-error.html +2 -2
  258. package/web/.next/server/app/_global-error.rsc +1 -1
  259. package/web/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  260. package/web/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  261. package/web/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  262. package/web/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  263. package/web/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  264. package/web/.next/server/app/_not-found/page/server-reference-manifest.json +3 -3
  265. package/web/.next/server/app/_not-found/page.js +1 -1
  266. package/web/.next/server/app/_not-found/page.js.nft.json +1 -1
  267. package/web/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  268. package/web/.next/server/app/api/attachments/preview/route.js.nft.json +1 -1
  269. package/web/.next/server/app/api/evidence/route.js.nft.json +1 -1
  270. package/web/.next/server/app/api/graph-data/route.js.nft.json +1 -1
  271. package/web/.next/server/app/settings/page/server-reference-manifest.json +8 -8
  272. package/web/.next/server/app/settings/page.js +3 -3
  273. package/web/.next/server/app/settings/page.js.nft.json +1 -1
  274. package/web/.next/server/app/settings/page_client-reference-manifest.js +1 -1
  275. package/web/.next/server/app/skills/page/server-reference-manifest.json +8 -8
  276. package/web/.next/server/app/skills/page.js +2 -2
  277. package/web/.next/server/app/skills/page.js.nft.json +1 -1
  278. package/web/.next/server/app/skills/page_client-reference-manifest.js +1 -1
  279. package/web/.next/server/app/tools/page/server-reference-manifest.json +8 -8
  280. package/web/.next/server/app/tools/page.js +2 -2
  281. package/web/.next/server/app/tools/page.js.nft.json +1 -1
  282. package/web/.next/server/app/tools/page_client-reference-manifest.js +1 -1
  283. package/web/.next/server/app/version/page/server-reference-manifest.json +3 -3
  284. package/web/.next/server/app/version/page.js +1 -1
  285. package/web/.next/server/app/version/page.js.nft.json +1 -1
  286. package/web/.next/server/app/version/page_client-reference-manifest.js +1 -1
  287. package/web/.next/server/app/workflows/page/app-paths-manifest.json +3 -0
  288. package/web/.next/server/app/workflows/page/build-manifest.json +18 -0
  289. package/web/.next/server/app/workflows/page/next-font-manifest.json +6 -0
  290. package/web/.next/server/app/workflows/page/react-loadable-manifest.json +8 -0
  291. package/web/.next/server/app/workflows/page/server-reference-manifest.json +110 -0
  292. package/web/.next/server/app/workflows/page.js +19 -0
  293. package/web/.next/server/app/workflows/page.js.map +5 -0
  294. package/web/.next/server/app/workflows/page.js.nft.json +1 -0
  295. package/web/.next/server/app/workflows/page_client-reference-manifest.js +2 -0
  296. package/web/.next/server/app-paths-manifest.json +2 -1
  297. package/web/.next/server/chunks/403f9_next_dist_esm_build_templates_app-route_370c43b1.js.map +1 -1
  298. package/web/.next/server/chunks/[root-of-the-server]__a402b567._.js +1 -1
  299. package/web/.next/server/chunks/[root-of-the-server]__c6e32a23._.js +1 -1
  300. package/web/.next/server/chunks/[root-of-the-server]__c6e32a23._.js.map +1 -1
  301. package/web/.next/server/chunks/[root-of-the-server]__cd67a84c._.js +1 -1
  302. package/web/.next/server/chunks/[root-of-the-server]__cd67a84c._.js.map +1 -1
  303. package/web/.next/server/chunks/ssr/{_6256a985._.js → 403f9_next_8a33ddee._.js} +2 -2
  304. package/web/.next/server/chunks/ssr/403f9_next_8a33ddee._.js.map +1 -0
  305. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js +1 -1
  306. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js.map +1 -1
  307. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_0c7787d0._.js +3 -0
  308. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_0c7787d0._.js.map +1 -0
  309. package/web/.next/server/chunks/ssr/[root-of-the-server]__082758f6._.js +3 -0
  310. package/web/.next/server/chunks/ssr/[root-of-the-server]__082758f6._.js.map +1 -0
  311. package/web/.next/server/chunks/ssr/{[root-of-the-server]__d1040bd1._.js → [root-of-the-server]__090f24cb._.js} +2 -2
  312. package/web/.next/server/chunks/ssr/{[root-of-the-server]__d1040bd1._.js.map → [root-of-the-server]__090f24cb._.js.map} +1 -1
  313. package/web/.next/server/chunks/ssr/[root-of-the-server]__29580090._.js +1 -1
  314. package/web/.next/server/chunks/ssr/[root-of-the-server]__29580090._.js.map +1 -1
  315. package/web/.next/server/chunks/ssr/[root-of-the-server]__32cd6108._.js +3 -0
  316. package/web/.next/server/chunks/ssr/[root-of-the-server]__32cd6108._.js.map +1 -0
  317. package/web/.next/server/chunks/ssr/[root-of-the-server]__357d99f9._.js +1 -1
  318. package/web/.next/server/chunks/ssr/[root-of-the-server]__43f51aa6._.js +1 -1
  319. package/web/.next/server/chunks/ssr/[root-of-the-server]__43f51aa6._.js.map +1 -1
  320. package/web/.next/server/chunks/ssr/[root-of-the-server]__44358910._.js +4 -0
  321. package/web/.next/server/chunks/ssr/[root-of-the-server]__44358910._.js.map +1 -0
  322. package/web/.next/server/chunks/ssr/{[root-of-the-server]__3ef34e4c._.js → [root-of-the-server]__57e0ee34._.js} +3 -3
  323. package/web/.next/server/chunks/ssr/[root-of-the-server]__57e0ee34._.js.map +1 -0
  324. package/web/.next/server/chunks/ssr/{[root-of-the-server]__efeeaed4._.js → [root-of-the-server]__7aa484f5._.js} +2 -2
  325. package/web/.next/server/chunks/ssr/{[root-of-the-server]__efeeaed4._.js.map → [root-of-the-server]__7aa484f5._.js.map} +1 -1
  326. package/web/.next/server/chunks/ssr/[root-of-the-server]__815546bd._.js +1 -1
  327. package/web/.next/server/chunks/ssr/[root-of-the-server]__815546bd._.js.map +1 -1
  328. package/web/.next/server/chunks/ssr/[root-of-the-server]__aad040c0._.js +2 -2
  329. package/web/.next/server/chunks/ssr/[root-of-the-server]__aad040c0._.js.map +1 -1
  330. package/web/.next/server/chunks/ssr/[root-of-the-server]__c094882b._.js +1 -1
  331. package/web/.next/server/chunks/ssr/[root-of-the-server]__c094882b._.js.map +1 -1
  332. package/web/.next/server/chunks/ssr/{[root-of-the-server]__0b150ddf._.js → [root-of-the-server]__c85ef334._.js} +2 -2
  333. package/web/.next/server/chunks/ssr/[root-of-the-server]__c85ef334._.js.map +1 -0
  334. package/web/.next/server/chunks/ssr/[root-of-the-server]__cf7aca8a._.js +3 -0
  335. package/web/.next/server/chunks/ssr/{[root-of-the-server]__df7c1cd3._.js.map → [root-of-the-server]__cf7aca8a._.js.map} +1 -1
  336. package/web/.next/server/chunks/ssr/[root-of-the-server]__dac5dbf1._.js +1 -1
  337. package/web/.next/server/chunks/ssr/[root-of-the-server]__dac5dbf1._.js.map +1 -1
  338. package/web/.next/server/chunks/ssr/{[root-of-the-server]__2138fa7e._.js → [root-of-the-server]__e5851493._.js} +3 -3
  339. package/web/.next/server/chunks/ssr/[root-of-the-server]__e5851493._.js.map +1 -0
  340. package/web/.next/server/chunks/ssr/[root-of-the-server]__fae8b355._.js +1 -1
  341. package/web/.next/server/chunks/ssr/[root-of-the-server]__fae8b355._.js.map +1 -1
  342. package/web/.next/server/chunks/ssr/_0c5f56e3._.js +2 -2
  343. package/web/.next/server/chunks/ssr/_0c5f56e3._.js.map +1 -1
  344. package/web/.next/server/chunks/ssr/_17dbc240._.js +3 -0
  345. package/web/.next/server/chunks/ssr/_17dbc240._.js.map +1 -0
  346. package/web/.next/server/chunks/ssr/_1b719e7f._.js +1 -1
  347. package/web/.next/server/chunks/ssr/_1b719e7f._.js.map +1 -1
  348. package/web/.next/server/chunks/ssr/_20a4a0d5._.js +3 -0
  349. package/web/.next/server/chunks/ssr/_20a4a0d5._.js.map +1 -0
  350. package/web/.next/server/chunks/ssr/{_8fcc39d4._.js → _212aafc7._.js} +2 -2
  351. package/web/.next/server/chunks/ssr/{_8fcc39d4._.js.map → _212aafc7._.js.map} +1 -1
  352. package/web/.next/server/chunks/ssr/_26582473._.js +3 -0
  353. package/web/.next/server/chunks/ssr/_26582473._.js.map +1 -0
  354. package/web/.next/server/chunks/ssr/{_b5fc318a._.js → _277acdde._.js} +2 -2
  355. package/web/.next/server/chunks/ssr/_277acdde._.js.map +1 -0
  356. package/web/.next/server/chunks/ssr/{_92b9e766._.js → _3038739c._.js} +2 -2
  357. package/web/.next/server/chunks/ssr/{_92b9e766._.js.map → _3038739c._.js.map} +1 -1
  358. package/web/.next/server/chunks/ssr/_37e8548b._.js +1 -1
  359. package/web/.next/server/chunks/ssr/_37e8548b._.js.map +1 -1
  360. package/web/.next/server/chunks/ssr/_4711e8cf._.js +3 -0
  361. package/web/.next/server/chunks/ssr/_4711e8cf._.js.map +1 -0
  362. package/web/.next/server/chunks/ssr/_4b432739._.js +1 -1
  363. package/web/.next/server/chunks/ssr/_4b432739._.js.map +1 -1
  364. package/web/.next/server/chunks/ssr/_55d763e2._.js +1 -1
  365. package/web/.next/server/chunks/ssr/_55d763e2._.js.map +1 -1
  366. package/web/.next/server/chunks/ssr/_64bdfc6f._.js +2 -2
  367. package/web/.next/server/chunks/ssr/_64bdfc6f._.js.map +1 -1
  368. package/web/.next/server/chunks/ssr/_89a36303._.js +3 -0
  369. package/web/.next/server/chunks/ssr/_89a36303._.js.map +1 -0
  370. package/web/.next/server/chunks/ssr/{_0020fddd._.js → _a3721140._.js} +2 -2
  371. package/web/.next/server/chunks/ssr/_a3721140._.js.map +1 -0
  372. package/web/.next/server/chunks/ssr/_b71645b4._.js +1 -1
  373. package/web/.next/server/chunks/ssr/_b71645b4._.js.map +1 -1
  374. package/web/.next/server/chunks/ssr/_d8575088._.js +1 -1
  375. package/web/.next/server/chunks/ssr/_d8575088._.js.map +1 -1
  376. package/web/.next/server/chunks/ssr/_e3411387._.js +9 -0
  377. package/web/.next/server/chunks/ssr/_e3411387._.js.map +1 -0
  378. package/web/.next/server/chunks/ssr/_f35785e8._.js +3 -0
  379. package/web/.next/server/chunks/ssr/_f35785e8._.js.map +1 -0
  380. package/web/.next/server/chunks/ssr/_f64ffec0._.js +4 -0
  381. package/web/.next/server/chunks/ssr/_f64ffec0._.js.map +1 -0
  382. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js +1 -1
  383. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js.map +1 -1
  384. package/web/.next/server/chunks/ssr/{_1b7dae9a._.js → f7870_sonner_dist_index_mjs_dbf63824._.js} +2 -2
  385. package/web/.next/server/chunks/ssr/f7870_sonner_dist_index_mjs_dbf63824._.js.map +1 -0
  386. package/web/.next/server/chunks/ssr/node_modules__pnpm_28388798._.js +3 -0
  387. package/web/.next/server/chunks/ssr/node_modules__pnpm_28388798._.js.map +1 -0
  388. package/web/.next/server/chunks/ssr/node_modules__pnpm_ae1705ef._.js +3 -0
  389. package/web/.next/server/chunks/ssr/node_modules__pnpm_ae1705ef._.js.map +1 -0
  390. package/web/.next/server/chunks/ssr/src_presentation_web_47c56a89._.js +3 -0
  391. package/web/.next/server/chunks/ssr/{src_presentation_web_f449e0af._.js.map → src_presentation_web_47c56a89._.js.map} +1 -1
  392. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_1b176e3c.js +1 -1
  393. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_1b176e3c.js.map +1 -1
  394. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_bd9f0dda.js +1 -1
  395. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_bd9f0dda.js.map +1 -1
  396. package/web/.next/server/chunks/ssr/src_presentation_web_app_actions_open-ide_ts_baaca5d5._.js +1 -1
  397. package/web/.next/server/chunks/ssr/src_presentation_web_components_e599bb8c._.js +1 -1
  398. package/web/.next/server/chunks/ssr/src_presentation_web_components_e599bb8c._.js.map +1 -1
  399. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js +1 -1
  400. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js.map +1 -1
  401. package/web/.next/server/pages/500.html +2 -2
  402. package/web/.next/server/server-reference-manifest.js +1 -1
  403. package/web/.next/server/server-reference-manifest.json +128 -47
  404. package/web/.next/static/chunks/012cd198f2f5f69b.js +1 -0
  405. package/web/.next/static/chunks/1717cd5043bdd3bd.js +1 -0
  406. package/web/.next/static/chunks/1760fdd748136b0a.js +1 -0
  407. package/web/.next/static/chunks/18c906505123f6bc.js +1 -0
  408. package/web/.next/static/chunks/{eb7b91c45ec7b468.js → 2be2403fc6ea10fb.js} +2 -2
  409. package/web/.next/static/chunks/2c1ff62df9b34313.js +1 -0
  410. package/web/.next/static/chunks/{672e3da3ad26005b.js → 42c9564e375717ed.js} +1 -1
  411. package/web/.next/static/chunks/5420560352333486.js +1 -0
  412. package/web/.next/static/chunks/581da2fea22a26ed.js +1 -0
  413. package/web/.next/static/chunks/7e664ecef6b0412b.js +1 -0
  414. package/web/.next/static/chunks/8254b51579b9bf14.js +1 -0
  415. package/web/.next/static/chunks/{9b8678597fa1db84.css → 919a5016ff5973a8.css} +1 -1
  416. package/web/.next/static/chunks/955de9c458882838.js +5 -0
  417. package/web/.next/static/chunks/{a18a5348858c6d75.js → 9d58a91a359e0b7d.js} +3 -3
  418. package/web/.next/static/chunks/{10b75a1ea923a690.js → c7865694beb27ce3.js} +1 -1
  419. package/web/.next/static/chunks/d7a464b957f611ba.js +1 -0
  420. package/web/.next/static/chunks/de2a780de5e1d718.js +1 -0
  421. package/web/.next/static/chunks/ed35b4293a81c861.js +9 -0
  422. package/web/.next/static/chunks/{97a773e5a88462a0.js → ed82c6ab93a79fa3.js} +1 -1
  423. package/web/.next/static/chunks/f29c8010a2d10bc5.js +1 -0
  424. package/web/.next/static/chunks/f75142a1b75f1c32.js +1 -0
  425. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_8b618f7d._.js +0 -3
  426. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_8b618f7d._.js.map +0 -1
  427. package/web/.next/server/chunks/ssr/[root-of-the-server]__0b150ddf._.js.map +0 -1
  428. package/web/.next/server/chunks/ssr/[root-of-the-server]__2138fa7e._.js.map +0 -1
  429. package/web/.next/server/chunks/ssr/[root-of-the-server]__3ef34e4c._.js.map +0 -1
  430. package/web/.next/server/chunks/ssr/[root-of-the-server]__d48c5b11._.js +0 -3
  431. package/web/.next/server/chunks/ssr/[root-of-the-server]__d48c5b11._.js.map +0 -1
  432. package/web/.next/server/chunks/ssr/[root-of-the-server]__df7c1cd3._.js +0 -3
  433. package/web/.next/server/chunks/ssr/_0020fddd._.js.map +0 -1
  434. package/web/.next/server/chunks/ssr/_01046927._.js +0 -9
  435. package/web/.next/server/chunks/ssr/_01046927._.js.map +0 -1
  436. package/web/.next/server/chunks/ssr/_05c23ad9._.js +0 -3
  437. package/web/.next/server/chunks/ssr/_05c23ad9._.js.map +0 -1
  438. package/web/.next/server/chunks/ssr/_16eb4fec._.js +0 -3
  439. package/web/.next/server/chunks/ssr/_16eb4fec._.js.map +0 -1
  440. package/web/.next/server/chunks/ssr/_1b7dae9a._.js.map +0 -1
  441. package/web/.next/server/chunks/ssr/_6256a985._.js.map +0 -1
  442. package/web/.next/server/chunks/ssr/_67104d9e._.js +0 -3
  443. package/web/.next/server/chunks/ssr/_67104d9e._.js.map +0 -1
  444. package/web/.next/server/chunks/ssr/_b5fc318a._.js.map +0 -1
  445. package/web/.next/server/chunks/ssr/_bf695af2._.js +0 -3
  446. package/web/.next/server/chunks/ssr/_bf695af2._.js.map +0 -1
  447. package/web/.next/server/chunks/ssr/_d4b20e29._.js +0 -3
  448. package/web/.next/server/chunks/ssr/_d4b20e29._.js.map +0 -1
  449. package/web/.next/server/chunks/ssr/node_modules__pnpm_55c7a131._.js +0 -3
  450. package/web/.next/server/chunks/ssr/node_modules__pnpm_55c7a131._.js.map +0 -1
  451. package/web/.next/server/chunks/ssr/src_presentation_web_f449e0af._.js +0 -3
  452. package/web/.next/static/chunks/08baac5434d9528e.js +0 -13
  453. package/web/.next/static/chunks/1f265c393614cad2.js +0 -1
  454. package/web/.next/static/chunks/413e2ca694528497.js +0 -1
  455. package/web/.next/static/chunks/49057cf8cd37e262.js +0 -1
  456. package/web/.next/static/chunks/56233eb0780a997e.js +0 -1
  457. package/web/.next/static/chunks/601d93593f5f664f.js +0 -1
  458. package/web/.next/static/chunks/688971d5bad5dc7c.js +0 -1
  459. package/web/.next/static/chunks/ad8f9fe681d88d09.js +0 -1
  460. package/web/.next/static/chunks/b70dd4985a814b27.js +0 -1
  461. package/web/.next/static/chunks/c69cea3bed20eea7.js +0 -1
  462. package/web/.next/static/chunks/d2e9df0b38f0d962.js +0 -1
  463. package/web/.next/static/chunks/f17d6d87a3508ba0.js +0 -1
  464. package/web/.next/static/chunks/f9a69279d55bfd3b.js +0 -1
  465. /package/web/.next/static/{e6hBEk7bk-CV1jLaYcZLX → Lsq24p2KQmGstqY0A7dDg}/_buildManifest.js +0 -0
  466. /package/web/.next/static/{e6hBEk7bk-CV1jLaYcZLX → Lsq24p2KQmGstqY0A7dDg}/_clientMiddlewareManifest.json +0 -0
  467. /package/web/.next/static/{e6hBEk7bk-CV1jLaYcZLX → Lsq24p2KQmGstqY0A7dDg}/_ssgManifest.js +0 -0
@@ -11,14 +11,12 @@
11
11
  */
12
12
  import { execSync } from 'node:child_process';
13
13
  import { isGraphBubbleUp } from '@langchain/langgraph';
14
- import { createNodeLogger, buildExecutorOptions, retryExecute, getCompletedPhases, markPhaseComplete, saveEvidenceManifest, } from './node-helpers.js';
14
+ import { createNodeLogger, buildExecutorOptions, retryExecute, getCompletedPhases, markPhaseComplete, } from './node-helpers.js';
15
15
  import { reportNodeStart } from '../heartbeat.js';
16
16
  import { recordPhaseStart, recordPhaseEnd } from '../phase-timing-context.js';
17
17
  import { updateNodeLifecycle } from '../lifecycle-context.js';
18
18
  import { buildFastImplementPrompt } from './prompts/fast-implement.prompt.js';
19
- import { buildEvidencePrompt } from './prompts/evidence-prompts.js';
20
- import { parseEvidenceRecords } from './evidence-output-parser.js';
21
- import { hasSettings, getSettings } from '../../../settings.service.js';
19
+ import { createEvidenceNode } from './evidence.node.js';
22
20
  /**
23
21
  * Factory that creates the fast-implement node function.
24
22
  *
@@ -62,11 +60,15 @@ export function createFastImplementNode(executor) {
62
60
  if (!hasWorktreeChanges(cwd) && !hasNewCommits(cwd)) {
63
61
  throw new Error('[fast-implement] Agent produced no file changes — it may have entered plan mode or asked questions instead of implementing. Retrying.');
64
62
  }
65
- // --- Evidence sub-agent: capture proof of completion (settings-gated) ---
66
- const evidenceEnabled = hasSettings() && getSettings().workflow.enableEvidence;
67
- const evidence = evidenceEnabled ? await collectEvidence(executor, state, log) : [];
68
- if (!evidenceEnabled) {
69
- log.info('Evidence collection disabled via settings — skipping');
63
+ // --- Evidence sub-agent: capture proof of completion (feature-gated) ---
64
+ let evidence = [];
65
+ if (state.enableEvidence) {
66
+ const evidenceNode = createEvidenceNode(executor);
67
+ const evidenceResult = await evidenceNode(state);
68
+ evidence = evidenceResult.evidence ?? [];
69
+ }
70
+ else {
71
+ log.info('Evidence collection disabled — skipping');
70
72
  }
71
73
  await recordPhaseEnd(timingId, durationMs, {
72
74
  inputTokens: result.usage?.inputTokens,
@@ -106,39 +108,6 @@ export function createFastImplementNode(executor) {
106
108
  }
107
109
  };
108
110
  }
109
- /**
110
- * Sub-agent call to collect evidence after fast implementation completes.
111
- * Graceful degradation: returns empty array on any failure so evidence
112
- * collection never blocks the workflow.
113
- */
114
- async function collectEvidence(executor, state, log) {
115
- try {
116
- log.info('Collecting evidence (sub-agent)');
117
- const commitEvidence = hasSettings() && getSettings().workflow.commitEvidence;
118
- const prompt = buildEvidencePrompt(state, { commitEvidence });
119
- const options = buildExecutorOptions(state);
120
- const result = await retryExecute(executor, prompt, options, { logger: log });
121
- try {
122
- const evidence = parseEvidenceRecords(result.result);
123
- log.info(`Parsed ${evidence.length} evidence record(s)`);
124
- saveEvidenceManifest(state, evidence, log);
125
- return evidence;
126
- }
127
- catch (parseErr) {
128
- const msg = parseErr instanceof Error ? parseErr.message : String(parseErr);
129
- log.error(`Warning: evidence parsing failed: ${msg} — continuing with empty evidence`);
130
- return [];
131
- }
132
- }
133
- catch (err) {
134
- // Re-throw LangGraph control-flow exceptions
135
- if (isGraphBubbleUp(err))
136
- throw err;
137
- const msg = err instanceof Error ? err.message : String(err);
138
- log.error(`Evidence collection failed: ${msg} — continuing without evidence`);
139
- return [];
140
- }
141
- }
142
111
  /**
143
112
  * Check whether the worktree has any uncommitted changes (new, modified, or deleted files).
144
113
  * Uses `git status --porcelain` which outputs one line per changed file, or empty if clean.
@@ -1 +1 @@
1
- {"version":3,"file":"implement.node.d.ts","sourceRoot":"","sources":["../../../../../../../../../packages/core/src/infrastructure/services/agents/feature-agent/nodes/implement.node.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAMH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,+DAA+D,CAAC;AACpG,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAqErD,wBAAgB,mBAAmB,CAAC,QAAQ,EAAE,cAAc,IAG5C,OAAO,iBAAiB,KAAG,OAAO,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC,CAiQ7E"}
1
+ {"version":3,"file":"implement.node.d.ts","sourceRoot":"","sources":["../../../../../../../../../packages/core/src/infrastructure/services/agents/feature-agent/nodes/implement.node.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAMH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,+DAA+D,CAAC;AACpG,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAoErD,wBAAgB,mBAAmB,CAAC,QAAQ,EAAE,cAAc,IAG5C,OAAO,iBAAiB,KAAG,OAAO,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC,CAiQ7E"}
@@ -16,7 +16,6 @@ import { recordPhaseStart, recordPhaseEnd, recordApprovalWaitStart, updatePhaseP
16
16
  import { updateNodeLifecycle } from '../lifecycle-context.js';
17
17
  import { buildImplementPhasePrompt, } from './prompts/implement.prompt.js';
18
18
  import { createEvidenceNode } from './evidence.node.js';
19
- import { hasSettings, getSettings } from '../../../settings.service.js';
20
19
  /**
21
20
  * Update feature.yaml with current implementation progress.
22
21
  * Silently no-ops if the file is missing or unparseable.
@@ -190,8 +189,8 @@ export function createImplementNode(executor) {
190
189
  log.info(`All phases complete — ${completedTasks}/${totalTasks} tasks (${elapsed}s)`);
191
190
  updateFeatureProgress(state.specDir, totalTasks, totalTasks, 'implementation-complete', null, log);
192
191
  messages.push(`[implement] Complete: ${totalTasks} tasks across ${totalPhases} phases (${elapsed}s)`);
193
- // --- Evidence sub-agent: capture proof of completion (settings-gated) ---
194
- const evidenceEnabled = hasSettings() && getSettings().workflow.enableEvidence;
192
+ // --- Evidence sub-agent: capture proof of completion (feature-gated) ---
193
+ const evidenceEnabled = state.enableEvidence;
195
194
  let evidenceResult = {};
196
195
  if (evidenceEnabled) {
197
196
  const evidenceNode = createEvidenceNode(executor);
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Real Clock Service
3
+ *
4
+ * Infrastructure implementation of IClock that returns the actual system time.
5
+ * Used in production; tests substitute a MockClock for deterministic behavior.
6
+ */
7
+ import type { IClock } from '../../application/ports/output/services/clock.interface.js';
8
+ export declare class RealClock implements IClock {
9
+ now(): Date;
10
+ }
11
+ //# sourceMappingURL=clock.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"clock.d.ts","sourceRoot":"","sources":["../../../../../../packages/core/src/infrastructure/services/clock.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAIH,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,4DAA4D,CAAC;AAEzF,qBACa,SAAU,YAAW,MAAM;IACtC,GAAG,IAAI,IAAI;CAGZ"}
@@ -0,0 +1,22 @@
1
+ /**
2
+ * Real Clock Service
3
+ *
4
+ * Infrastructure implementation of IClock that returns the actual system time.
5
+ * Used in production; tests substitute a MockClock for deterministic behavior.
6
+ */
7
+ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
8
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
9
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
10
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
11
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
12
+ };
13
+ import { injectable } from 'tsyringe';
14
+ let RealClock = class RealClock {
15
+ now() {
16
+ return new Date();
17
+ }
18
+ };
19
+ RealClock = __decorate([
20
+ injectable()
21
+ ], RealClock);
22
+ export { RealClock };
@@ -1 +1 @@
1
- {"version":3,"file":"notification.service.d.ts","sourceRoot":"","sources":["../../../../../../../packages/core/src/infrastructure/services/notifications/notification.service.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,KAAK,EACV,iBAAiB,EAElB,MAAM,qCAAqC,CAAC;AAE7C,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,8EAA8E,CAAC;AAEzH,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAC7D,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAgB7D,qBAAa,mBAAoB,YAAW,oBAAoB;IAC9D,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAkB;IACtC,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAkB;gBAEtC,GAAG,EAAE,eAAe,EAAE,eAAe,EAAE,eAAe;IAKlE,MAAM,CAAC,KAAK,EAAE,iBAAiB,GAAG,IAAI;CAiBvC"}
1
+ {"version":3,"file":"notification.service.d.ts","sourceRoot":"","sources":["../../../../../../../packages/core/src/infrastructure/services/notifications/notification.service.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,KAAK,EACV,iBAAiB,EAElB,MAAM,qCAAqC,CAAC;AAE7C,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,8EAA8E,CAAC;AAEzH,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAC7D,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AAmB7D,qBAAa,mBAAoB,YAAW,oBAAoB;IAC9D,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAkB;IACtC,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAkB;gBAEtC,GAAG,EAAE,eAAe,EAAE,eAAe,EAAE,eAAe;IAKlE,MAAM,CAAC,KAAK,EAAE,iBAAiB,GAAG,IAAI;CAiBvC"}
@@ -22,6 +22,9 @@ const EVENT_TYPE_TO_CONFIG_KEY = {
22
22
  [NotificationEventType.PrChecksFailed]: 'prChecksFailed',
23
23
  [NotificationEventType.PrBlocked]: 'prBlocked',
24
24
  [NotificationEventType.MergeReviewReady]: 'mergeReviewReady',
25
+ [NotificationEventType.WorkflowStarted]: 'workflowStarted',
26
+ [NotificationEventType.WorkflowCompleted]: 'workflowCompleted',
27
+ [NotificationEventType.WorkflowFailed]: 'workflowFailed',
25
28
  };
26
29
  export class NotificationService {
27
30
  bus;
@@ -0,0 +1,69 @@
1
+ /**
2
+ * Workflow Scheduler Service
3
+ *
4
+ * Polls for due scheduled workflows and executes them sequentially.
5
+ * Follows the NotificationWatcherService/PrSyncWatcherService pattern:
6
+ * - Singleton with start()/stop()/isRunning() lifecycle
7
+ * - setInterval-based polling
8
+ * - Immediate first poll on start()
9
+ * - Graceful shutdown via clearInterval
10
+ *
11
+ * Uses IClock for all time operations (deterministic testing).
12
+ * Uses croner only for validation and nextRun() calculation.
13
+ *
14
+ * Each tick:
15
+ * 1. Query enabled workflows where nextRunAt <= clock.now()
16
+ * 2. Enqueue due workflows into FIFO queue
17
+ * 3. If no execution is running, dequeue and execute next
18
+ * 4. Every 100 ticks: run retention cleanup
19
+ *
20
+ * Crash recovery on startup:
21
+ * - Mark stale 'running' executions as 'failed'
22
+ * - Recalculate nextRunAt for all enabled workflows
23
+ */
24
+ import type { IWorkflowRepository } from '../../../application/ports/output/repositories/workflow-repository.interface.js';
25
+ import type { IWorkflowExecutionRepository } from '../../../application/ports/output/repositories/workflow-execution-repository.interface.js';
26
+ import type { IClock } from '../../../application/ports/output/services/clock.interface.js';
27
+ import type { INotificationService } from '../../../application/ports/output/services/notification-service.interface.js';
28
+ export declare class WorkflowSchedulerService {
29
+ private readonly workflowRepo;
30
+ private readonly executionRepo;
31
+ private readonly clock;
32
+ private readonly notificationService;
33
+ private readonly pollIntervalMs;
34
+ private readonly retentionDays;
35
+ private intervalId;
36
+ private tickCount;
37
+ private isExecuting;
38
+ private readonly queue;
39
+ constructor(workflowRepo: IWorkflowRepository, executionRepo: IWorkflowExecutionRepository, clock: IClock, notificationService: INotificationService, pollIntervalMs?: number, retentionDays?: number);
40
+ isRunning(): boolean;
41
+ start(): Promise<void>;
42
+ stop(): void;
43
+ private poll;
44
+ private executeWorkflow;
45
+ private recoverStaleExecutions;
46
+ private runRetentionCleanup;
47
+ private emitNotification;
48
+ }
49
+ /**
50
+ * Initialize the workflow scheduler singleton.
51
+ * Must be called once during daemon startup.
52
+ */
53
+ export declare function initializeWorkflowScheduler(workflowRepo: IWorkflowRepository, executionRepo: IWorkflowExecutionRepository, clock: IClock, notificationService: INotificationService, pollIntervalMs?: number, retentionDays?: number): void;
54
+ /**
55
+ * Get the workflow scheduler singleton.
56
+ */
57
+ export declare function getWorkflowScheduler(): WorkflowSchedulerService;
58
+ /**
59
+ * Check if the workflow scheduler has been initialized.
60
+ */
61
+ export declare function hasWorkflowScheduler(): boolean;
62
+ /**
63
+ * Reset the workflow scheduler singleton (for testing purposes only).
64
+ * Stops the scheduler if running before resetting.
65
+ *
66
+ * @internal
67
+ */
68
+ export declare function resetWorkflowScheduler(): void;
69
+ //# sourceMappingURL=workflow-scheduler.service.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workflow-scheduler.service.d.ts","sourceRoot":"","sources":["../../../../../../../packages/core/src/infrastructure/services/workflow-scheduler/workflow-scheduler.service.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;GAsBG;AAcH,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,iFAAiF,CAAC;AAC3H,OAAO,KAAK,EAAE,4BAA4B,EAAE,MAAM,2FAA2F,CAAC;AAC9I,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,+DAA+D,CAAC;AAC5F,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,8EAA8E,CAAC;AAOzH,qBAAa,wBAAwB;IACnC,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAsB;IACnD,OAAO,CAAC,QAAQ,CAAC,aAAa,CAA+B;IAC7D,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAS;IAC/B,OAAO,CAAC,QAAQ,CAAC,mBAAmB,CAAuB;IAC3D,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAS;IACxC,OAAO,CAAC,QAAQ,CAAC,aAAa,CAAS;IAEvC,OAAO,CAAC,UAAU,CAA+C;IACjE,OAAO,CAAC,SAAS,CAAK;IACtB,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAgB;gBAGpC,YAAY,EAAE,mBAAmB,EACjC,aAAa,EAAE,4BAA4B,EAC3C,KAAK,EAAE,MAAM,EACb,mBAAmB,EAAE,oBAAoB,EACzC,cAAc,GAAE,MAAiC,EACjD,aAAa,GAAE,MAA+B;IAUhD,SAAS,IAAI,OAAO;IAId,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAc5B,IAAI,IAAI,IAAI;YAOE,IAAI;YA2CJ,eAAe;YA6Hf,sBAAsB;YAqBtB,mBAAmB;IAWjC,OAAO,CAAC,gBAAgB;CAkBzB;AAMD;;;GAGG;AACH,wBAAgB,2BAA2B,CACzC,YAAY,EAAE,mBAAmB,EACjC,aAAa,EAAE,4BAA4B,EAC3C,KAAK,EAAE,MAAM,EACb,mBAAmB,EAAE,oBAAoB,EACzC,cAAc,CAAC,EAAE,MAAM,EACvB,aAAa,CAAC,EAAE,MAAM,GACrB,IAAI,CAaN;AAED;;GAEG;AACH,wBAAgB,oBAAoB,IAAI,wBAAwB,CAQ/D;AAED;;GAEG;AACH,wBAAgB,oBAAoB,IAAI,OAAO,CAE9C;AAED;;;;;GAKG;AACH,wBAAgB,sBAAsB,IAAI,IAAI,CAK7C"}
@@ -0,0 +1,270 @@
1
+ /**
2
+ * Workflow Scheduler Service
3
+ *
4
+ * Polls for due scheduled workflows and executes them sequentially.
5
+ * Follows the NotificationWatcherService/PrSyncWatcherService pattern:
6
+ * - Singleton with start()/stop()/isRunning() lifecycle
7
+ * - setInterval-based polling
8
+ * - Immediate first poll on start()
9
+ * - Graceful shutdown via clearInterval
10
+ *
11
+ * Uses IClock for all time operations (deterministic testing).
12
+ * Uses croner only for validation and nextRun() calculation.
13
+ *
14
+ * Each tick:
15
+ * 1. Query enabled workflows where nextRunAt <= clock.now()
16
+ * 2. Enqueue due workflows into FIFO queue
17
+ * 3. If no execution is running, dequeue and execute next
18
+ * 4. Every 100 ticks: run retention cleanup
19
+ *
20
+ * Crash recovery on startup:
21
+ * - Mark stale 'running' executions as 'failed'
22
+ * - Recalculate nextRunAt for all enabled workflows
23
+ */
24
+ import { randomUUID } from 'node:crypto';
25
+ import { NotificationEventType, NotificationSeverity, WorkflowExecutionStatus, WorkflowTriggerType, } from '../../../domain/generated/output.js';
26
+ import { calculateNextRunAt } from '../../../application/use-cases/workflows/cron-helpers.js';
27
+ const DEFAULT_POLL_INTERVAL_MS = 60_000;
28
+ const RETENTION_CLEANUP_INTERVAL_TICKS = 100;
29
+ const DEFAULT_RETENTION_DAYS = 30;
30
+ export class WorkflowSchedulerService {
31
+ workflowRepo;
32
+ executionRepo;
33
+ clock;
34
+ notificationService;
35
+ pollIntervalMs;
36
+ retentionDays;
37
+ intervalId = null;
38
+ tickCount = 0;
39
+ isExecuting = false;
40
+ queue = []; // workflow IDs
41
+ constructor(workflowRepo, executionRepo, clock, notificationService, pollIntervalMs = DEFAULT_POLL_INTERVAL_MS, retentionDays = DEFAULT_RETENTION_DAYS) {
42
+ this.workflowRepo = workflowRepo;
43
+ this.executionRepo = executionRepo;
44
+ this.clock = clock;
45
+ this.notificationService = notificationService;
46
+ this.pollIntervalMs = pollIntervalMs;
47
+ this.retentionDays = retentionDays;
48
+ }
49
+ isRunning() {
50
+ return this.intervalId !== null;
51
+ }
52
+ async start() {
53
+ if (this.intervalId !== null)
54
+ return;
55
+ // Crash recovery: mark stale 'running' executions as 'failed'
56
+ await this.recoverStaleExecutions();
57
+ // Run first poll immediately
58
+ void this.poll();
59
+ this.intervalId = setInterval(() => {
60
+ void this.poll();
61
+ }, this.pollIntervalMs);
62
+ }
63
+ stop() {
64
+ if (this.intervalId !== null) {
65
+ clearInterval(this.intervalId);
66
+ this.intervalId = null;
67
+ }
68
+ }
69
+ async poll() {
70
+ try {
71
+ this.tickCount++;
72
+ // Find due workflows
73
+ const enabledWorkflows = await this.workflowRepo.findEnabled();
74
+ const now = this.clock.now();
75
+ for (const workflow of enabledWorkflows) {
76
+ if (workflow.nextRunAt &&
77
+ new Date(workflow.nextRunAt).getTime() <= now.getTime() &&
78
+ !this.queue.includes(workflow.id)) {
79
+ this.queue.push(workflow.id);
80
+ }
81
+ }
82
+ // Also process any manually queued executions
83
+ const queuedExecutions = await this.executionRepo.findByStatus(WorkflowExecutionStatus.Queued);
84
+ for (const execution of queuedExecutions) {
85
+ if (!this.queue.includes(execution.workflowId)) {
86
+ this.queue.push(execution.workflowId);
87
+ }
88
+ }
89
+ // Execute one queued workflow if nothing is running
90
+ if (!this.isExecuting && this.queue.length > 0) {
91
+ const workflowId = this.queue.shift();
92
+ void this.executeWorkflow(workflowId, queuedExecutions);
93
+ }
94
+ // Retention cleanup every N ticks
95
+ if (this.tickCount % RETENTION_CLEANUP_INTERVAL_TICKS === 0) {
96
+ await this.runRetentionCleanup();
97
+ }
98
+ }
99
+ catch {
100
+ // DB not ready or query failed — skip this poll cycle
101
+ }
102
+ }
103
+ async executeWorkflow(workflowId, existingQueuedExecutions) {
104
+ this.isExecuting = true;
105
+ try {
106
+ const workflow = await this.workflowRepo.findById(workflowId);
107
+ if (!workflow) {
108
+ this.isExecuting = false;
109
+ return;
110
+ }
111
+ // Find or create execution record
112
+ let execution = existingQueuedExecutions.find((e) => e.workflowId === workflowId && e.status === WorkflowExecutionStatus.Queued);
113
+ const now = this.clock.now();
114
+ if (!execution) {
115
+ // Create execution record for scheduled trigger
116
+ execution = {
117
+ id: randomUUID(),
118
+ workflowId,
119
+ triggerType: WorkflowTriggerType.Scheduled,
120
+ status: WorkflowExecutionStatus.Queued,
121
+ startedAt: now,
122
+ createdAt: now,
123
+ updatedAt: now,
124
+ };
125
+ await this.executionRepo.create(execution);
126
+ }
127
+ // Transition to Running
128
+ execution = {
129
+ ...execution,
130
+ status: WorkflowExecutionStatus.Running,
131
+ startedAt: now,
132
+ updatedAt: now,
133
+ };
134
+ await this.executionRepo.update(execution);
135
+ // Emit WorkflowStarted event
136
+ this.emitNotification(NotificationEventType.WorkflowStarted, NotificationSeverity.Info, workflow, execution, `Workflow "${workflow.name}" started`);
137
+ try {
138
+ // Execute the workflow prompt
139
+ // For now, this is a placeholder — the actual agent execution
140
+ // will be wired in via the CLI/daemon integration (Phase 4).
141
+ // The scheduler creates the record and manages state transitions.
142
+ // Agent execution would happen here via IAgentRunner.
143
+ const completedAt = this.clock.now();
144
+ const durationMs = completedAt.getTime() - now.getTime();
145
+ // Mark as completed
146
+ execution = {
147
+ ...execution,
148
+ status: WorkflowExecutionStatus.Completed,
149
+ completedAt,
150
+ durationMs,
151
+ updatedAt: completedAt,
152
+ };
153
+ await this.executionRepo.update(execution);
154
+ // Emit WorkflowCompleted event
155
+ this.emitNotification(NotificationEventType.WorkflowCompleted, NotificationSeverity.Success, workflow, execution, `Workflow "${workflow.name}" completed successfully`);
156
+ }
157
+ catch (err) {
158
+ const errorMessage = err instanceof Error ? err.message : String(err);
159
+ const failedAt = this.clock.now();
160
+ execution = {
161
+ ...execution,
162
+ status: WorkflowExecutionStatus.Failed,
163
+ completedAt: failedAt,
164
+ durationMs: failedAt.getTime() - now.getTime(),
165
+ errorMessage,
166
+ updatedAt: failedAt,
167
+ };
168
+ await this.executionRepo.update(execution);
169
+ // Emit WorkflowFailed event
170
+ this.emitNotification(NotificationEventType.WorkflowFailed, NotificationSeverity.Error, workflow, execution, `Workflow "${workflow.name}" failed: ${errorMessage}`);
171
+ }
172
+ // Update workflow lastRunAt and recalculate nextRunAt
173
+ const updateNow = this.clock.now();
174
+ const updatedWorkflow = {
175
+ ...workflow,
176
+ lastRunAt: updateNow,
177
+ updatedAt: updateNow,
178
+ };
179
+ if (workflow.cronExpression && workflow.enabled) {
180
+ const nextRunAt = calculateNextRunAt(workflow.cronExpression, workflow.timezone, updateNow);
181
+ updatedWorkflow.nextRunAt = nextRunAt ?? undefined;
182
+ }
183
+ else {
184
+ updatedWorkflow.nextRunAt = undefined;
185
+ }
186
+ await this.workflowRepo.update(updatedWorkflow);
187
+ }
188
+ finally {
189
+ this.isExecuting = false;
190
+ }
191
+ }
192
+ async recoverStaleExecutions() {
193
+ try {
194
+ const staleExecutions = await this.executionRepo.findByStatus(WorkflowExecutionStatus.Running);
195
+ const now = this.clock.now();
196
+ for (const execution of staleExecutions) {
197
+ await this.executionRepo.update({
198
+ ...execution,
199
+ status: WorkflowExecutionStatus.Failed,
200
+ completedAt: now,
201
+ errorMessage: 'Daemon crashed during execution',
202
+ updatedAt: now,
203
+ });
204
+ }
205
+ }
206
+ catch {
207
+ // Best-effort recovery
208
+ }
209
+ }
210
+ async runRetentionCleanup() {
211
+ try {
212
+ const cutoff = new Date(this.clock.now().getTime() - this.retentionDays * 24 * 60 * 60 * 1000);
213
+ await this.executionRepo.deleteOlderThan(cutoff);
214
+ }
215
+ catch {
216
+ // Best-effort cleanup
217
+ }
218
+ }
219
+ emitNotification(eventType, severity, workflow, execution, message) {
220
+ const event = {
221
+ eventType,
222
+ agentRunId: execution.id,
223
+ featureId: workflow.id,
224
+ featureName: workflow.name,
225
+ message,
226
+ severity,
227
+ timestamp: this.clock.now().toISOString(),
228
+ };
229
+ this.notificationService.notify(event);
230
+ }
231
+ }
232
+ // --- Singleton accessors (follows NotificationWatcherService pattern) ---
233
+ let schedulerInstance = null;
234
+ /**
235
+ * Initialize the workflow scheduler singleton.
236
+ * Must be called once during daemon startup.
237
+ */
238
+ export function initializeWorkflowScheduler(workflowRepo, executionRepo, clock, notificationService, pollIntervalMs, retentionDays) {
239
+ if (schedulerInstance !== null) {
240
+ throw new Error('Workflow scheduler already initialized. Cannot re-initialize.');
241
+ }
242
+ schedulerInstance = new WorkflowSchedulerService(workflowRepo, executionRepo, clock, notificationService, pollIntervalMs, retentionDays);
243
+ }
244
+ /**
245
+ * Get the workflow scheduler singleton.
246
+ */
247
+ export function getWorkflowScheduler() {
248
+ if (schedulerInstance === null) {
249
+ throw new Error('Workflow scheduler not initialized. Call initializeWorkflowScheduler() during daemon startup.');
250
+ }
251
+ return schedulerInstance;
252
+ }
253
+ /**
254
+ * Check if the workflow scheduler has been initialized.
255
+ */
256
+ export function hasWorkflowScheduler() {
257
+ return schedulerInstance !== null;
258
+ }
259
+ /**
260
+ * Reset the workflow scheduler singleton (for testing purposes only).
261
+ * Stops the scheduler if running before resetting.
262
+ *
263
+ * @internal
264
+ */
265
+ export function resetWorkflowScheduler() {
266
+ if (schedulerInstance !== null) {
267
+ schedulerInstance.stop();
268
+ }
269
+ schedulerInstance = null;
270
+ }
@@ -1 +1 @@
1
- {"version":3,"file":"_serve.command.d.ts","sourceRoot":"","sources":["../../../../../src/presentation/cli/commands/_serve.command.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AAEH,OAAO,EAAE,OAAO,EAAwB,MAAM,WAAW,CAAC;AAwB1D;;GAEG;AACH,wBAAgB,kBAAkB,IAAI,OAAO,CA0D5C"}
1
+ {"version":3,"file":"_serve.command.d.ts","sourceRoot":"","sources":["../../../../../src/presentation/cli/commands/_serve.command.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AAEH,OAAO,EAAE,OAAO,EAAwB,MAAM,WAAW,CAAC;AAiC1D;;GAEG;AACH,wBAAgB,kBAAkB,IAAI,OAAO,CA0E5C"}
@@ -29,6 +29,8 @@ import { container } from '../../../../packages/core/src/infrastructure/di/conta
29
29
  import { setVersionEnvVars } from '../../../../packages/core/src/infrastructure/services/version.service.js';
30
30
  import { resolveWebDir } from '../../../../packages/core/src/infrastructure/services/web-server.service.js';
31
31
  import { initializeNotificationWatcher, getNotificationWatcher, } from '../../../../packages/core/src/infrastructure/services/notifications/notification-watcher.service.js';
32
+ import { initializeWorkflowScheduler, getWorkflowScheduler, hasWorkflowScheduler, } from '../../../../packages/core/src/infrastructure/services/workflow-scheduler/workflow-scheduler.service.js';
33
+ import { hasSettings, getSettings } from '../../../../packages/core/src/infrastructure/services/settings.service.js';
32
34
  function parsePort(value) {
33
35
  const port = parseInt(value, 10);
34
36
  if (isNaN(port) || port < 1024 || port > 65535) {
@@ -62,6 +64,15 @@ export function createServeCommand() {
62
64
  const notificationService = container.resolve('INotificationService');
63
65
  initializeNotificationWatcher(runRepo, phaseTimingRepo, featureRepo, notificationService);
64
66
  getNotificationWatcher().start();
67
+ // Start workflow scheduler (only when feature flag is enabled)
68
+ const scheduledWorkflowsEnabled = hasSettings() && getSettings().featureFlags?.scheduledWorkflows === true;
69
+ if (scheduledWorkflowsEnabled) {
70
+ const workflowRepo = container.resolve('IWorkflowRepository');
71
+ const executionRepo = container.resolve('IWorkflowExecutionRepository');
72
+ const clock = container.resolve('IClock');
73
+ initializeWorkflowScheduler(workflowRepo, executionRepo, clock, notificationService);
74
+ await getWorkflowScheduler().start();
75
+ }
65
76
  // Graceful shutdown handler — identical pattern to ui.command.ts
66
77
  let isShuttingDown = false;
67
78
  const shutdown = async () => {
@@ -71,6 +82,9 @@ export function createServeCommand() {
71
82
  // Force exit after 5s if graceful shutdown stalls
72
83
  const forceExit = setTimeout(() => process.exit(0), 5000);
73
84
  forceExit.unref();
85
+ if (hasWorkflowScheduler()) {
86
+ getWorkflowScheduler().stop();
87
+ }
74
88
  getNotificationWatcher().stop();
75
89
  const deploymentService = container.resolve('IDeploymentService');
76
90
  deploymentService.stopAll();
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Workflow Create Command
3
+ *
4
+ * Create a new scheduled workflow definition with a name, prompt, and optional
5
+ * description, tool constraints, and template.
6
+ *
7
+ * Usage:
8
+ * shep workflow create <name> --prompt <prompt> [options]
9
+ * shep workflow create <name> --template <template-name> [options]
10
+ */
11
+ import { Command } from 'commander';
12
+ export declare function createCreateCommand(): Command;
13
+ //# sourceMappingURL=create.command.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"create.command.d.ts","sourceRoot":"","sources":["../../../../../../src/presentation/cli/commands/workflow/create.command.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAiBpC,wBAAgB,mBAAmB,IAAI,OAAO,CA+E7C"}
@@ -0,0 +1,83 @@
1
+ /**
2
+ * Workflow Create Command
3
+ *
4
+ * Create a new scheduled workflow definition with a name, prompt, and optional
5
+ * description, tool constraints, and template.
6
+ *
7
+ * Usage:
8
+ * shep workflow create <name> --prompt <prompt> [options]
9
+ * shep workflow create <name> --template <template-name> [options]
10
+ */
11
+ import { Command } from 'commander';
12
+ import { container } from '../../../../../packages/core/src/infrastructure/di/container.js';
13
+ import { CreateWorkflowUseCase } from '../../../../../packages/core/src/application/use-cases/workflows/create-workflow.use-case.js';
14
+ import { colors, messages } from '../../ui/index.js';
15
+ import { getIssueTriageTemplate } from '../../../../../packages/core/src/application/use-cases/workflows/templates/issue-triage.template.js';
16
+ import { getBranchRebaseTemplate } from '../../../../../packages/core/src/application/use-cases/workflows/templates/branch-rebase.template.js';
17
+ function collectToolConstraints(value, previous) {
18
+ return [...previous, value];
19
+ }
20
+ const TEMPLATES = {
21
+ 'issue-triage': getIssueTriageTemplate,
22
+ 'branch-rebase': getBranchRebaseTemplate,
23
+ };
24
+ export function createCreateCommand() {
25
+ return new Command('create')
26
+ .description('Create a new workflow')
27
+ .argument('<name>', 'Unique workflow name')
28
+ .option('-p, --prompt <prompt>', 'Agent prompt (the instruction the AI agent will execute)')
29
+ .option('-d, --description <text>', 'Human-readable description')
30
+ .option('-t, --tool-constraint <tool>', 'Allowed tool (repeatable)', collectToolConstraints, [])
31
+ .option('--template <name>', 'Create from built-in template (issue-triage, branch-rebase)')
32
+ .option('-r, --repo <path>', 'Repository path (defaults to current directory)')
33
+ .option('--disabled', 'Create in disabled state')
34
+ .action(async (name, options) => {
35
+ try {
36
+ let prompt = options.prompt;
37
+ let description = options.description;
38
+ let toolConstraints = options.toolConstraint.length > 0 ? options.toolConstraint : undefined;
39
+ // Resolve template if specified
40
+ if (options.template) {
41
+ const templateFactory = TEMPLATES[options.template];
42
+ if (!templateFactory) {
43
+ messages.error(`Unknown template: "${options.template}". Available templates: ${Object.keys(TEMPLATES).join(', ')}`);
44
+ process.exitCode = 1;
45
+ return;
46
+ }
47
+ const template = templateFactory();
48
+ prompt = prompt ?? template.prompt;
49
+ description = description ?? template.description;
50
+ toolConstraints = toolConstraints ?? template.toolConstraints;
51
+ }
52
+ if (!prompt) {
53
+ messages.error('A prompt is required. Use --prompt <prompt> or --template <name>.');
54
+ process.exitCode = 1;
55
+ return;
56
+ }
57
+ const repositoryPath = options.repo ?? process.cwd();
58
+ const useCase = container.resolve(CreateWorkflowUseCase);
59
+ const workflow = await useCase.execute({
60
+ name,
61
+ prompt,
62
+ repositoryPath,
63
+ ...(description != null && { description }),
64
+ ...(toolConstraints != null && { toolConstraints }),
65
+ ...(options.disabled && { enabled: false }),
66
+ });
67
+ messages.newline();
68
+ messages.success('Workflow created');
69
+ console.log(` ${colors.muted('Name:')} ${colors.accent(workflow.name)}`);
70
+ console.log(` ${colors.muted('ID:')} ${workflow.id.substring(0, 8)}`);
71
+ console.log(` ${colors.muted('Enabled:')} ${workflow.enabled ? colors.success('yes') : colors.muted('no')}`);
72
+ if (workflow.description) {
73
+ console.log(` ${colors.muted('Desc:')} ${workflow.description}`);
74
+ }
75
+ messages.newline();
76
+ }
77
+ catch (error) {
78
+ const err = error instanceof Error ? error : new Error(String(error));
79
+ messages.error('Failed to create workflow', err);
80
+ process.exitCode = 1;
81
+ }
82
+ });
83
+ }