mstro-app 0.5.1 → 0.5.5

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 (240) hide show
  1. package/PRIVACY.md +9 -9
  2. package/README.md +71 -28
  3. package/bin/commands/config.js +1 -1
  4. package/bin/mstro.js +55 -4
  5. package/dist/server/cli/eta-estimator.d.ts +55 -0
  6. package/dist/server/cli/eta-estimator.d.ts.map +1 -0
  7. package/dist/server/cli/eta-estimator.js +222 -0
  8. package/dist/server/cli/eta-estimator.js.map +1 -0
  9. package/dist/server/cli/headless/stall-assessor.d.ts +50 -0
  10. package/dist/server/cli/headless/stall-assessor.d.ts.map +1 -1
  11. package/dist/server/cli/headless/stall-assessor.js +64 -9
  12. package/dist/server/cli/headless/stall-assessor.js.map +1 -1
  13. package/dist/server/cli/headless/tool-watchdog.d.ts +21 -0
  14. package/dist/server/cli/headless/tool-watchdog.d.ts.map +1 -1
  15. package/dist/server/cli/headless/tool-watchdog.js +19 -12
  16. package/dist/server/cli/headless/tool-watchdog.js.map +1 -1
  17. package/dist/server/cli/improvisation-history-store.d.ts.map +1 -1
  18. package/dist/server/cli/improvisation-history-store.js +5 -1
  19. package/dist/server/cli/improvisation-history-store.js.map +1 -1
  20. package/dist/server/cli/improvisation-output-queue.d.ts +5 -1
  21. package/dist/server/cli/improvisation-output-queue.d.ts.map +1 -1
  22. package/dist/server/cli/improvisation-output-queue.js +30 -7
  23. package/dist/server/cli/improvisation-output-queue.js.map +1 -1
  24. package/dist/server/cli/improvisation-session-manager.d.ts +29 -0
  25. package/dist/server/cli/improvisation-session-manager.d.ts.map +1 -1
  26. package/dist/server/cli/improvisation-session-manager.js +50 -1
  27. package/dist/server/cli/improvisation-session-manager.js.map +1 -1
  28. package/dist/server/cli/improvisation-types.d.ts +2 -0
  29. package/dist/server/cli/improvisation-types.d.ts.map +1 -1
  30. package/dist/server/cli/improvisation-types.js.map +1 -1
  31. package/dist/server/engines/EngineEvent.d.ts +126 -0
  32. package/dist/server/engines/EngineEvent.d.ts.map +1 -0
  33. package/dist/server/engines/EngineEvent.js +11 -0
  34. package/dist/server/engines/EngineEvent.js.map +1 -0
  35. package/dist/server/engines/claude/ClaudeCodeEngine.d.ts +47 -0
  36. package/dist/server/engines/claude/ClaudeCodeEngine.d.ts.map +1 -0
  37. package/dist/server/engines/claude/ClaudeCodeEngine.js +338 -0
  38. package/dist/server/engines/claude/ClaudeCodeEngine.js.map +1 -0
  39. package/dist/server/engines/factory.d.ts +21 -0
  40. package/dist/server/engines/factory.d.ts.map +1 -0
  41. package/dist/server/engines/factory.js +152 -0
  42. package/dist/server/engines/factory.js.map +1 -0
  43. package/dist/server/engines/opencode/OpenCodeEngine.d.ts +148 -0
  44. package/dist/server/engines/opencode/OpenCodeEngine.d.ts.map +1 -0
  45. package/dist/server/engines/opencode/OpenCodeEngine.js +630 -0
  46. package/dist/server/engines/opencode/OpenCodeEngine.js.map +1 -0
  47. package/dist/server/engines/opencode/OpenCodeServerManager.d.ts +172 -0
  48. package/dist/server/engines/opencode/OpenCodeServerManager.d.ts.map +1 -0
  49. package/dist/server/engines/opencode/OpenCodeServerManager.js +390 -0
  50. package/dist/server/engines/opencode/OpenCodeServerManager.js.map +1 -0
  51. package/dist/server/engines/opencode/model-catalog.d.ts +94 -0
  52. package/dist/server/engines/opencode/model-catalog.d.ts.map +1 -0
  53. package/dist/server/engines/opencode/model-catalog.js +141 -0
  54. package/dist/server/engines/opencode/model-catalog.js.map +1 -0
  55. package/dist/server/engines/types.d.ts +146 -0
  56. package/dist/server/engines/types.d.ts.map +1 -0
  57. package/dist/server/engines/types.js +4 -0
  58. package/dist/server/engines/types.js.map +1 -0
  59. package/dist/server/index.js +1 -1
  60. package/dist/server/index.js.map +1 -1
  61. package/dist/server/mcp/bouncer-haiku.d.ts +17 -4
  62. package/dist/server/mcp/bouncer-haiku.d.ts.map +1 -1
  63. package/dist/server/mcp/bouncer-haiku.js +8 -124
  64. package/dist/server/mcp/bouncer-haiku.js.map +1 -1
  65. package/dist/server/mcp/bouncer-integration.d.ts +45 -0
  66. package/dist/server/mcp/bouncer-integration.d.ts.map +1 -1
  67. package/dist/server/mcp/bouncer-integration.js +69 -5
  68. package/dist/server/mcp/bouncer-integration.js.map +1 -1
  69. package/dist/server/mcp/classifier/BouncerClassifier.d.ts +34 -0
  70. package/dist/server/mcp/classifier/BouncerClassifier.d.ts.map +1 -0
  71. package/dist/server/mcp/classifier/BouncerClassifier.js +4 -0
  72. package/dist/server/mcp/classifier/BouncerClassifier.js.map +1 -0
  73. package/dist/server/mcp/classifier/ClaudeBouncerClassifier.d.ts +17 -0
  74. package/dist/server/mcp/classifier/ClaudeBouncerClassifier.d.ts.map +1 -0
  75. package/dist/server/mcp/classifier/ClaudeBouncerClassifier.js +142 -0
  76. package/dist/server/mcp/classifier/ClaudeBouncerClassifier.js.map +1 -0
  77. package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.d.ts +68 -0
  78. package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.d.ts.map +1 -0
  79. package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.js +182 -0
  80. package/dist/server/mcp/classifier/OpenCodeBouncerClassifier.js.map +1 -0
  81. package/dist/server/mcp/classifier/factory.d.ts +70 -0
  82. package/dist/server/mcp/classifier/factory.d.ts.map +1 -0
  83. package/dist/server/mcp/classifier/factory.js +155 -0
  84. package/dist/server/mcp/classifier/factory.js.map +1 -0
  85. package/dist/server/services/plan/agent-resolver.d.ts +26 -0
  86. package/dist/server/services/plan/agent-resolver.d.ts.map +1 -0
  87. package/dist/server/services/plan/agent-resolver.js +102 -0
  88. package/dist/server/services/plan/agent-resolver.js.map +1 -0
  89. package/dist/server/services/plan/composer.d.ts.map +1 -1
  90. package/dist/server/services/plan/composer.js +59 -11
  91. package/dist/server/services/plan/composer.js.map +1 -1
  92. package/dist/server/services/plan/executor.d.ts.map +1 -1
  93. package/dist/server/services/plan/executor.js +3 -1
  94. package/dist/server/services/plan/executor.js.map +1 -1
  95. package/dist/server/services/plan/issue-prompt-builder.d.ts.map +1 -1
  96. package/dist/server/services/plan/issue-prompt-builder.js +33 -1
  97. package/dist/server/services/plan/issue-prompt-builder.js.map +1 -1
  98. package/dist/server/services/plan/parser-core.d.ts.map +1 -1
  99. package/dist/server/services/plan/parser-core.js +1 -0
  100. package/dist/server/services/plan/parser-core.js.map +1 -1
  101. package/dist/server/services/plan/types.d.ts +1 -0
  102. package/dist/server/services/plan/types.d.ts.map +1 -1
  103. package/dist/server/services/settings.d.ts +76 -2
  104. package/dist/server/services/settings.d.ts.map +1 -1
  105. package/dist/server/services/settings.js +127 -4
  106. package/dist/server/services/settings.js.map +1 -1
  107. package/dist/server/services/websocket/git-branch-handlers.d.ts.map +1 -1
  108. package/dist/server/services/websocket/git-branch-handlers.js +19 -6
  109. package/dist/server/services/websocket/git-branch-handlers.js.map +1 -1
  110. package/dist/server/services/websocket/handler.d.ts +17 -1
  111. package/dist/server/services/websocket/handler.d.ts.map +1 -1
  112. package/dist/server/services/websocket/handler.js +54 -2
  113. package/dist/server/services/websocket/handler.js.map +1 -1
  114. package/dist/server/services/websocket/quality-complexity.d.ts.map +1 -1
  115. package/dist/server/services/websocket/quality-complexity.js +78 -26
  116. package/dist/server/services/websocket/quality-complexity.js.map +1 -1
  117. package/dist/server/services/websocket/quality-eta.d.ts +47 -0
  118. package/dist/server/services/websocket/quality-eta.d.ts.map +1 -0
  119. package/dist/server/services/websocket/quality-eta.js +110 -0
  120. package/dist/server/services/websocket/quality-eta.js.map +1 -0
  121. package/dist/server/services/websocket/quality-grading.d.ts +27 -4
  122. package/dist/server/services/websocket/quality-grading.d.ts.map +1 -1
  123. package/dist/server/services/websocket/quality-grading.js +369 -201
  124. package/dist/server/services/websocket/quality-grading.js.map +1 -1
  125. package/dist/server/services/websocket/quality-handlers.d.ts.map +1 -1
  126. package/dist/server/services/websocket/quality-handlers.js +145 -7
  127. package/dist/server/services/websocket/quality-handlers.js.map +1 -1
  128. package/dist/server/services/websocket/quality-operations.d.ts +34 -0
  129. package/dist/server/services/websocket/quality-operations.d.ts.map +1 -0
  130. package/dist/server/services/websocket/quality-operations.js +47 -0
  131. package/dist/server/services/websocket/quality-operations.js.map +1 -0
  132. package/dist/server/services/websocket/quality-persistence.d.ts +9 -0
  133. package/dist/server/services/websocket/quality-persistence.d.ts.map +1 -1
  134. package/dist/server/services/websocket/quality-persistence.js +10 -0
  135. package/dist/server/services/websocket/quality-persistence.js.map +1 -1
  136. package/dist/server/services/websocket/quality-review-agent.d.ts +1 -1
  137. package/dist/server/services/websocket/quality-review-agent.d.ts.map +1 -1
  138. package/dist/server/services/websocket/quality-review-agent.js +105 -56
  139. package/dist/server/services/websocket/quality-review-agent.js.map +1 -1
  140. package/dist/server/services/websocket/quality-service.d.ts +9 -1
  141. package/dist/server/services/websocket/quality-service.d.ts.map +1 -1
  142. package/dist/server/services/websocket/quality-service.js +334 -14
  143. package/dist/server/services/websocket/quality-service.js.map +1 -1
  144. package/dist/server/services/websocket/quality-tools.d.ts +21 -0
  145. package/dist/server/services/websocket/quality-tools.d.ts.map +1 -1
  146. package/dist/server/services/websocket/quality-tools.js +49 -0
  147. package/dist/server/services/websocket/quality-tools.js.map +1 -1
  148. package/dist/server/services/websocket/quality-types.d.ts +35 -2
  149. package/dist/server/services/websocket/quality-types.d.ts.map +1 -1
  150. package/dist/server/services/websocket/quality-types.js +1 -1
  151. package/dist/server/services/websocket/quality-types.js.map +1 -1
  152. package/dist/server/services/websocket/session-handlers.d.ts +3 -1
  153. package/dist/server/services/websocket/session-handlers.d.ts.map +1 -1
  154. package/dist/server/services/websocket/session-handlers.js +57 -9
  155. package/dist/server/services/websocket/session-handlers.js.map +1 -1
  156. package/dist/server/services/websocket/session-history.js +3 -0
  157. package/dist/server/services/websocket/session-history.js.map +1 -1
  158. package/dist/server/services/websocket/session-initialization.d.ts.map +1 -1
  159. package/dist/server/services/websocket/session-initialization.js +158 -42
  160. package/dist/server/services/websocket/session-initialization.js.map +1 -1
  161. package/dist/server/services/websocket/session-registry.d.ts +25 -0
  162. package/dist/server/services/websocket/session-registry.d.ts.map +1 -1
  163. package/dist/server/services/websocket/session-registry.js +19 -0
  164. package/dist/server/services/websocket/session-registry.js.map +1 -1
  165. package/dist/server/services/websocket/settings-handlers.d.ts +1 -1
  166. package/dist/server/services/websocket/settings-handlers.d.ts.map +1 -1
  167. package/dist/server/services/websocket/settings-handlers.js +35 -4
  168. package/dist/server/services/websocket/settings-handlers.js.map +1 -1
  169. package/dist/server/services/websocket/tab-broadcast.d.ts +7 -2
  170. package/dist/server/services/websocket/tab-broadcast.d.ts.map +1 -1
  171. package/dist/server/services/websocket/tab-broadcast.js +10 -2
  172. package/dist/server/services/websocket/tab-broadcast.js.map +1 -1
  173. package/dist/server/services/websocket/tab-event-buffer.d.ts +97 -8
  174. package/dist/server/services/websocket/tab-event-buffer.d.ts.map +1 -1
  175. package/dist/server/services/websocket/tab-event-buffer.js +138 -12
  176. package/dist/server/services/websocket/tab-event-buffer.js.map +1 -1
  177. package/dist/server/services/websocket/tab-event-replay.d.ts +29 -13
  178. package/dist/server/services/websocket/tab-event-replay.d.ts.map +1 -1
  179. package/dist/server/services/websocket/tab-event-replay.js +55 -2
  180. package/dist/server/services/websocket/tab-event-replay.js.map +1 -1
  181. package/dist/server/services/websocket/tab-handlers.d.ts +9 -1
  182. package/dist/server/services/websocket/tab-handlers.d.ts.map +1 -1
  183. package/dist/server/services/websocket/tab-handlers.js +47 -2
  184. package/dist/server/services/websocket/tab-handlers.js.map +1 -1
  185. package/dist/server/services/websocket/types.d.ts +28 -5
  186. package/dist/server/services/websocket/types.d.ts.map +1 -1
  187. package/dist/server/services/websocket/types.js +10 -4
  188. package/dist/server/services/websocket/types.js.map +1 -1
  189. package/package.json +5 -3
  190. package/server/cli/eta-estimator.ts +249 -0
  191. package/server/cli/headless/stall-assessor.ts +93 -0
  192. package/server/cli/headless/tool-watchdog.ts +21 -0
  193. package/server/cli/improvisation-history-store.ts +4 -1
  194. package/server/cli/improvisation-output-queue.ts +29 -7
  195. package/server/cli/improvisation-session-manager.ts +54 -1
  196. package/server/cli/improvisation-types.ts +2 -0
  197. package/server/engines/EngineEvent.ts +156 -0
  198. package/server/engines/claude/ClaudeCodeEngine.ts +404 -0
  199. package/server/engines/factory.ts +176 -0
  200. package/server/engines/opencode/OpenCodeEngine.ts +786 -0
  201. package/server/engines/opencode/OpenCodeServerManager.ts +577 -0
  202. package/server/engines/opencode/model-catalog.ts +217 -0
  203. package/server/engines/types.ts +173 -0
  204. package/server/index.ts +1 -1
  205. package/server/mcp/bouncer-haiku.ts +21 -145
  206. package/server/mcp/bouncer-integration.ts +107 -5
  207. package/server/mcp/classifier/BouncerClassifier.ts +40 -0
  208. package/server/mcp/classifier/ClaudeBouncerClassifier.ts +189 -0
  209. package/server/mcp/classifier/OpenCodeBouncerClassifier.ts +305 -0
  210. package/server/mcp/classifier/factory.ts +195 -0
  211. package/server/services/plan/agent-resolver.ts +115 -0
  212. package/server/services/plan/agents/code-review.md +38 -8
  213. package/server/services/plan/composer.ts +63 -11
  214. package/server/services/plan/executor.ts +3 -1
  215. package/server/services/plan/issue-prompt-builder.ts +39 -1
  216. package/server/services/plan/parser-core.ts +1 -0
  217. package/server/services/plan/types.ts +4 -0
  218. package/server/services/settings.ts +161 -4
  219. package/server/services/websocket/git-branch-handlers.ts +20 -6
  220. package/server/services/websocket/handler.ts +59 -2
  221. package/server/services/websocket/quality-complexity.ts +80 -26
  222. package/server/services/websocket/quality-eta.ts +155 -0
  223. package/server/services/websocket/quality-grading.ts +445 -222
  224. package/server/services/websocket/quality-handlers.ts +153 -7
  225. package/server/services/websocket/quality-operations.ts +72 -0
  226. package/server/services/websocket/quality-persistence.ts +17 -0
  227. package/server/services/websocket/quality-review-agent.ts +154 -64
  228. package/server/services/websocket/quality-service.ts +361 -13
  229. package/server/services/websocket/quality-tools.ts +51 -0
  230. package/server/services/websocket/quality-types.ts +41 -2
  231. package/server/services/websocket/session-handlers.ts +64 -10
  232. package/server/services/websocket/session-history.ts +3 -0
  233. package/server/services/websocket/session-initialization.ts +189 -46
  234. package/server/services/websocket/session-registry.ts +37 -0
  235. package/server/services/websocket/settings-handlers.ts +41 -4
  236. package/server/services/websocket/tab-broadcast.ts +10 -2
  237. package/server/services/websocket/tab-event-buffer.ts +143 -11
  238. package/server/services/websocket/tab-event-replay.ts +70 -3
  239. package/server/services/websocket/tab-handlers.ts +53 -5
  240. package/server/services/websocket/types.ts +37 -5
@@ -0,0 +1,195 @@
1
+ // Copyright (c) 2025-present Mstro, Inc. All rights reserved.
2
+ // Licensed under the MIT License. See LICENSE file for details.
3
+
4
+ /**
5
+ * Bouncer classifier factory.
6
+ *
7
+ * Two entry points:
8
+ *
9
+ * - `getClassifier()` — production path. Reads
10
+ * `settings.bouncerClassifier: { engine, model }` and returns the
11
+ * matching `BouncerClassifier` instance. If the persisted config is
12
+ * missing, malformed, or names a non-eligible model, it logs a clear
13
+ * warning and falls back to `ClaudeBouncerClassifier` + Haiku — the
14
+ * Bouncer must always have a classifier to call, so "no config" and
15
+ * "bad config" both collapse to the known-safe default rather than
16
+ * throwing.
17
+ *
18
+ * - `createBouncerClassifier(options?)` — direct-construction helper used
19
+ * by the engineSwap feature-flag gate (see `engine-swap-flag.test.ts`).
20
+ * Accepts an explicit `engineId` and is deliberately feature-flag-aware:
21
+ * when `engineSwap` is disabled, the flag short-circuits to Claude.
22
+ *
23
+ * New callers should prefer `getClassifier()` so the user-selected model
24
+ * takes effect without plumbing. The bouncer-integration layer constructs
25
+ * its default classifier lazily so env var changes and settings edits
26
+ * propagate on the next classification call.
27
+ */
28
+
29
+ import { OpenCodeServerManager } from '../../engines/opencode/OpenCodeServerManager.js';
30
+ import type { EngineId } from '../../engines/types.js';
31
+ import {
32
+ BOUNCER_ELIGIBLE_MODELS,
33
+ type BouncerClassifierConfig,
34
+ DEFAULT_BOUNCER_CLASSIFIER,
35
+ getBouncerClassifier,
36
+ isEngineSwapEnabled,
37
+ } from '../../services/settings.js';
38
+ import type { BouncerClassifier } from './BouncerClassifier.js';
39
+ import { ClaudeBouncerClassifier } from './ClaudeBouncerClassifier.js';
40
+ import { OpenCodeBouncerClassifier } from './OpenCodeBouncerClassifier.js';
41
+
42
+ /** Options accepted by every classifier implementation. */
43
+ export interface ClassifierFactoryOptions {
44
+ /**
45
+ * Which engine backs the classifier. With `engineSwap` off this is
46
+ * ignored and `'claude-code'` is used; with the flag on, non-Claude
47
+ * engines throw until their implementations land (Epic 4).
48
+ */
49
+ engineId?: EngineId;
50
+ }
51
+
52
+ /**
53
+ * Construct the Layer-2 Bouncer classifier by engine id (no settings
54
+ * lookup). Exists for the `engineSwap` feature-flag gate, which asserts
55
+ * that the factory is flag-aware in both on/off states. New production
56
+ * callers should route through {@link getClassifier} instead.
57
+ */
58
+ export function createBouncerClassifier(
59
+ options: ClassifierFactoryOptions = {},
60
+ ): BouncerClassifier {
61
+ if (!isEngineSwapEnabled()) {
62
+ return new ClaudeBouncerClassifier();
63
+ }
64
+ const engineId = options.engineId ?? 'claude-code';
65
+ switch (engineId) {
66
+ case 'claude-code':
67
+ return new ClaudeBouncerClassifier();
68
+ case 'opencode':
69
+ // Wired through `getClassifier()` (settings path). Direct engine-id
70
+ // construction stays intentionally narrow — callers that want the
71
+ // OpenCode classifier should pick it via the Settings UI so the
72
+ // shared `OpenCodeServerManager` is available.
73
+ throw new Error(
74
+ 'OpenCode bouncer classifier is not implemented yet (Epic 4). ' +
75
+ 'Keep engineSwap off until the OpenCode classifier ships.',
76
+ );
77
+ default: {
78
+ const exhaustive: never = engineId;
79
+ throw new Error(`Unknown classifier engine id: ${String(exhaustive)}`);
80
+ }
81
+ }
82
+ }
83
+
84
+ /**
85
+ * Process-lifetime singleton for the `opencode serve` subprocess used by
86
+ * the classifier. Deliberately separate from the engines-side manager so
87
+ * tests can inject a mock client without touching the engine factory.
88
+ * Lazy: never created until an OpenCode classifier is first requested.
89
+ */
90
+ let sharedOpenCodeManager: OpenCodeServerManager | null = null;
91
+ let openCodeManagerFactory: () => OpenCodeServerManager = () =>
92
+ new OpenCodeServerManager({ registerProcessHandlers: true });
93
+
94
+ function getSharedOpenCodeServerManager(): OpenCodeServerManager {
95
+ if (!sharedOpenCodeManager) {
96
+ sharedOpenCodeManager = openCodeManagerFactory();
97
+ }
98
+ return sharedOpenCodeManager;
99
+ }
100
+
101
+ /**
102
+ * Override the OpenCode manager used by the classifier factory. Test-only;
103
+ * production code never calls this. Pass `null` to reset to the default.
104
+ */
105
+ export function __setOpenCodeManagerFactoryForTests(
106
+ factory: (() => OpenCodeServerManager) | null,
107
+ ): void {
108
+ sharedOpenCodeManager = null;
109
+ openCodeManagerFactory = factory
110
+ ?? (() => new OpenCodeServerManager({ registerProcessHandlers: true }));
111
+ }
112
+
113
+ /**
114
+ * Log a fallback reason in a single place so grep + log analysis surface
115
+ * every path where we silently dropped back to Claude+Haiku. Goes to
116
+ * stderr (matching the rest of the Bouncer logs) so it shows up in the
117
+ * CLI's `--trace` output and in audit transcripts.
118
+ */
119
+ function logFallback(reason: string): void {
120
+ console.warn(
121
+ `[Bouncer] Classifier config invalid, falling back to Claude+Haiku: ${reason}`,
122
+ );
123
+ }
124
+
125
+ /**
126
+ * Construct a `BouncerClassifier` for the provided config. Throws on bad
127
+ * config — callers that need fallback semantics should use
128
+ * {@link getClassifier} instead.
129
+ */
130
+ export function createClassifierForConfig(
131
+ config: BouncerClassifierConfig,
132
+ ): BouncerClassifier {
133
+ const eligible = BOUNCER_ELIGIBLE_MODELS[config.engine];
134
+ if (!eligible || !eligible.includes(config.model)) {
135
+ throw new Error(
136
+ `Model '${config.model}' is not bouncer-eligible for engine '${config.engine}'`,
137
+ );
138
+ }
139
+ switch (config.engine) {
140
+ case 'claude-code':
141
+ // The Claude classifier currently hardcodes `--model haiku` in the
142
+ // subprocess call. Passing `sonnet` still returns Haiku until a
143
+ // later issue threads the model through — the eligibility check
144
+ // guards correctness; the subprocess args are a follow-up.
145
+ return new ClaudeBouncerClassifier();
146
+ case 'opencode':
147
+ return new OpenCodeBouncerClassifier({
148
+ manager: getSharedOpenCodeServerManager(),
149
+ model: config.model,
150
+ });
151
+ default: {
152
+ const exhaustive: never = config.engine;
153
+ throw new Error(`Unknown classifier engine id: ${String(exhaustive)}`);
154
+ }
155
+ }
156
+ }
157
+
158
+ /**
159
+ * Production classifier accessor. Reads the user's current Bouncer
160
+ * classifier choice from persistent settings and returns a fresh
161
+ * `BouncerClassifier` instance. Invalid or missing config logs a clear
162
+ * warning and falls back to the default Claude+Haiku classifier — the
163
+ * Bouncer is a required security layer, so "no classifier available" is
164
+ * never an acceptable outcome.
165
+ *
166
+ * Called on every `reviewOperation()` path (indirectly via the
167
+ * integration layer's lazy default); cheap because classifier
168
+ * construction is synchronous and does not spawn subprocesses until the
169
+ * first `classify()` call.
170
+ */
171
+ export function getClassifier(): BouncerClassifier {
172
+ let config: BouncerClassifierConfig;
173
+ try {
174
+ config = getBouncerClassifier();
175
+ } catch (err) {
176
+ logFallback(err instanceof Error ? err.message : String(err));
177
+ return new ClaudeBouncerClassifier();
178
+ }
179
+
180
+ try {
181
+ return createClassifierForConfig(config);
182
+ } catch (err) {
183
+ logFallback(err instanceof Error ? err.message : String(err));
184
+ // Last-resort fallback — if even the default config can't build the
185
+ // classifier (e.g. OpenCode catalogue edit broke the model list), we
186
+ // still return Claude+Haiku so the Bouncer keeps functioning.
187
+ if (
188
+ config.engine === DEFAULT_BOUNCER_CLASSIFIER.engine &&
189
+ config.model === DEFAULT_BOUNCER_CLASSIFIER.model
190
+ ) {
191
+ return new ClaudeBouncerClassifier();
192
+ }
193
+ return new ClaudeBouncerClassifier();
194
+ }
195
+ }
@@ -0,0 +1,115 @@
1
+ // Copyright (c) 2025-present Mstro, Inc. All rights reserved.
2
+
3
+ /**
4
+ * Agent Resolver — Maps issue.agents hints to subagents installed on the user's system.
5
+ *
6
+ * Issue front matter may specify `agents` as either canonical Claude Code subagent
7
+ * names (e.g. `backend-architect`) or general role pointers (e.g. `backend engineer`).
8
+ * This module bridges the two: it consults AgentManager (project / global / bundled
9
+ * `.claude/agents/`) and resolves each hint to a concrete agent name when possible,
10
+ * falling back to the original hint when no match is found so the executor can still
11
+ * surface the user's intent in the prompt.
12
+ */
13
+
14
+ import { type AgentInfo, agentManager } from '../../utils/agent-manager.js';
15
+
16
+ export interface ResolvedAgent {
17
+ /** The original hint as written in the issue front matter. */
18
+ hint: string;
19
+ /** The resolved canonical agent name, or null if no installed agent matched. */
20
+ resolvedName: string | null;
21
+ /** The matching agent info, or null if no installed agent matched. */
22
+ info: AgentInfo | null;
23
+ }
24
+
25
+ const NON_WORD = /[^a-z0-9]+/g;
26
+
27
+ function normalize(input: string): string {
28
+ return input.toLowerCase().replace(NON_WORD, ' ').trim();
29
+ }
30
+
31
+ function tokenize(input: string): string[] {
32
+ return normalize(input).split(' ').filter(Boolean);
33
+ }
34
+
35
+ /**
36
+ * Discover every available agent across project / global / bundled directories.
37
+ * Project entries shadow global, which shadows bundled (deduped by canonical name).
38
+ */
39
+ function listAvailableAgents(workingDir: string): AgentInfo[] {
40
+ const seen = new Map<string, AgentInfo>();
41
+ const layers = [
42
+ agentManager.listProjectAgents(workingDir),
43
+ agentManager.listGlobalAgents(),
44
+ agentManager.listBundledAgents(),
45
+ ];
46
+ for (const layer of layers) {
47
+ for (const agent of layer) {
48
+ if (!seen.has(agent.name)) seen.set(agent.name, agent);
49
+ }
50
+ }
51
+ return Array.from(seen.values());
52
+ }
53
+
54
+ /**
55
+ * Score how well an agent matches a hint. Returns 0 when there is no token overlap.
56
+ * Higher is better. Exact normalized matches return Infinity.
57
+ */
58
+ function matchScore(hint: string, agent: AgentInfo): number {
59
+ const normalizedHint = normalize(hint);
60
+ const normalizedName = normalize(agent.name);
61
+ if (normalizedHint === normalizedName) return Number.POSITIVE_INFINITY;
62
+
63
+ const hintTokens = tokenize(hint);
64
+ if (hintTokens.length === 0) return 0;
65
+
66
+ const haystack = `${normalizedName} ${normalize(agent.description ?? '')}`;
67
+ let matched = 0;
68
+ for (const token of hintTokens) {
69
+ if (token.length < 2) continue;
70
+ if (haystack.includes(token)) matched++;
71
+ }
72
+ if (matched === 0) return 0;
73
+
74
+ // Reward agents whose name (not just description) contains hint tokens.
75
+ const nameMatches = hintTokens.filter(t => t.length >= 2 && normalizedName.includes(t)).length;
76
+ return matched + nameMatches * 0.5;
77
+ }
78
+
79
+ /**
80
+ * Resolve a single hint against the catalog of available agents.
81
+ * Returns the highest-scoring agent, or null when no agent has any token overlap.
82
+ */
83
+ function resolveHint(hint: string, available: AgentInfo[]): AgentInfo | null {
84
+ let bestScore = 0;
85
+ let best: AgentInfo | null = null;
86
+ for (const agent of available) {
87
+ const score = matchScore(hint, agent);
88
+ if (score > bestScore) {
89
+ bestScore = score;
90
+ best = agent;
91
+ }
92
+ }
93
+ return best;
94
+ }
95
+
96
+ /**
97
+ * Resolve every hint in `agents` against the user's installed Claude Code subagents.
98
+ * Hints with no match are preserved (resolvedName: null) so the executor can still
99
+ * mention them in the prompt with a graceful fallback note.
100
+ */
101
+ export function resolveAgentHints(agents: string[], workingDir: string): ResolvedAgent[] {
102
+ if (!agents || agents.length === 0) return [];
103
+ const available = listAvailableAgents(workingDir);
104
+ return agents
105
+ .map(raw => raw.trim())
106
+ .filter(Boolean)
107
+ .map(hint => {
108
+ const info = resolveHint(hint, available);
109
+ return {
110
+ hint,
111
+ resolvedName: info?.name ?? null,
112
+ info,
113
+ };
114
+ });
115
+ }
@@ -74,19 +74,49 @@ For each finding, use this reasoning process:
74
74
 
75
75
  ## Scoring Guidelines
76
76
 
77
- The overall grade is computed deterministically from your findings, not from a number you supply. Severity and category on each finding are what drive the grade — pick them carefully.
77
+ The overall grade is computed deterministically from your findings, not from a number you supply. **Severity and category on each finding are what drive the grade — pick them carefully.** When in doubt, downgrade.
78
78
 
79
- Three independent dimension grades are computed:
79
+ ### Severity Ladder calibrate by likelihood × user impact, not just by topic
80
80
 
81
- - **Security** (category: `security`) — uses a severity-threshold rule: A = 0 findings, B = only low, C = ≥1 medium, D = ≥1 high, F = ≥1 critical.
82
- - **Reliability** (categories: `bugs`, `logic`, `performance`) severity-threshold rule, slightly more lenient: A = 0 findings or ≤1 low, B = ≥2 low or ≤2 medium, C = ≥3 medium or ≥1 high, D = ≥2 high, F = ≥1 critical.
83
- - **Maintainability** (categories: `architecture`, `oop`, `maintainability`) — density-based (issues per 1000 lines), with a severity escape hatch: any high finding caps at C, any critical caps at D.
81
+ Severity should answer two questions:
82
+ 1. **How likely is this to actually trigger?** (Common path vs. edge case vs. theoretical)
83
+ 2. **What happens when it triggers?** (User-visible breakage / data loss vs. internal-only / cosmetic)
84
84
 
85
- Overall grade = the worst of the three dimensions. A single critical security finding caps the entire codebase at F.
85
+ Use this ladder. Worked examples follow each level.
86
86
 
87
- This means **severity is load-bearing**: marking something `high` when it's really `low` will swing the grade unfairly. When in doubt, downgrade. A finding without clear evidence of harm is `low`.
87
+ - **`critical`** Reserved for "this is broken in production today on common code paths." Active data corruption, RCE, auth bypass for normal users, unrecoverable crash on the happy path. If the on-call would page at 3 AM for it, it's critical.
88
+ - ✅ SQL injection on a public form. Hard-coded production credentials in a deployed file. A `null`-deref on the homepage render path.
89
+ - ❌ "Could become a problem if traffic 100×". "Edge case where two clients race within 50ms." A theoretical bug in error-handling code that has never run.
88
90
 
89
- You may still emit `score`, `grade`, and `scoreRationale` for reference they are persisted but ignored when computing the displayed grade. Focus your effort on accurate findings, not on guessing the overall number.
91
+ - **`high`** A real bug or vulnerability that **definitely affects normal users on common code paths** with **user-visible consequences** (broken UI, wrong data shown, action silently fails). Or an exploitable security issue that requires only realistic conditions.
92
+ - ✅ Wrong state shown after a successful save (UI/UX bug). XSS via reflected URL parameter on a logged-in dashboard. Wrong calculation in a money-handling code path. Memory leak that grows on every page-view.
93
+ - ❌ Race condition on degraded shutdown paths. Edge-case exploit gated behind admin auth on a feature that hasn't shipped. A theoretical SSRF on an internal endpoint with no user reach. Defense-in-depth gaps (rate limit absent, header missing) — those are `low`.
94
+
95
+ - **`medium`** — Real issue but affects an edge case OR has limited user impact OR requires unusual conditions to trigger. Worth fixing eventually; not blocking.
96
+ - ✅ Missing error handling on a rarely-failing dependency. Logic bug in an admin-only page. A bug only reachable when two specific feature flags are both on. Performance issue that adds 50 ms but isn't user-perceptible.
97
+ - ❌ "Best practice" preferences with no user impact. Theoretical bugs in unreachable code.
98
+
99
+ - **`low`** — Improbable, theoretical, or cosmetic. Defense-in-depth missing, style/preference, "could be cleaner." Many of these are fine to leave for years.
100
+ - ✅ Missing rate limit on a low-traffic admin endpoint. SQL injection-shaped pattern that ends up safely parameterized. A `console.log` left in code. A nullable field that's only null in a code path that never executes.
101
+
102
+ ### Likelihood-weighted severity rules
103
+
104
+ Apply these as veto rules **after** you've chosen a severity from topic alone:
105
+
106
+ - If the bug only fires on a path that **realistically never executes in production**, downgrade by at least one step (high→medium, medium→low). A bug that requires "the network connection drops between line 42 and 43 of the shutdown handler" is `low` even if its consequences would be severe.
107
+ - If the issue has **no user-visible effect** (no UI/UX impact, no incorrect data shown, no security boundary crossed), it caps at `medium`. UI/UX wiring bugs and broken interactive flows skew higher; pure-internal architecture / observability gaps skew lower.
108
+ - If the issue is a **defense-in-depth gap** (rate limits, hardening headers, additional validation on already-validated input), cap at `low` unless you can articulate the realistic exploit chain that survives the existing defenses.
109
+ - If exploitability requires **conditions that only matter at high traffic / wide user attack surface**, downgrade for early-stage projects: this is `low` or `medium`, not `high`. (Make this explicit in the description so the reader knows the call.)
110
+
111
+ ### Three dimension grades the engine derives
112
+
113
+ - **Security** (category: `security`) — strictest. A = 0 findings, B = only low, C = ≥1 medium, F = ≥1 high, F- = ≥1 critical.
114
+ - **Reliability** (categories: `bugs`, `logic`, `performance`) — density-based grade per KLOC with severity escape: critical → F, any high → caps at C. Multiple medium findings escalate gradually rather than auto-failing.
115
+ - **Maintainability** (categories: `architecture`, `oop`, `maintainability`) — density-based with severity escape: critical → F, any high → C.
116
+
117
+ Overall grade = the worst of the three. A single critical security finding caps the entire codebase at F-.
118
+
119
+ You may still emit `score`, `grade`, and `scoreRationale` for reference — they are persisted but ignored when computing the displayed grade. Focus your effort on accurate severity classification, not on guessing the overall number.
90
120
 
91
121
  ## Output
92
122
 
@@ -10,6 +10,7 @@
10
10
 
11
11
  import { existsSync, readFileSync } from 'node:fs';
12
12
  import { join } from 'node:path';
13
+ import { getEtaProfileCached } from '../../cli/eta-estimator.js';
13
14
  import type { ToolUseEvent } from '../../cli/headless/index.js';
14
15
  import { ResilientRunner } from '../../cli/headless/resilient-runner.js';
15
16
  import { cleanupAttachments, preparePromptAndAttachments } from '../../cli/improvisation-attachments.js';
@@ -28,6 +29,21 @@ const PROMPT_TOOL_MESSAGES: Record<string, string> = {
28
29
  Bash: 'Running commands...',
29
30
  };
30
31
 
32
+ /**
33
+ * Resolve the ETA quantile profile used by the planning indicator. Reads
34
+ * .mstro/history (chat improv movements) since planning-prompt durations
35
+ * cluster in the same range. Failures degrade silently to undefined — the
36
+ * indicator falls back to elapsed-only display.
37
+ */
38
+ async function resolvePromptEtaProfile(workingDir: string): Promise<NonNullable<Awaited<ReturnType<typeof getEtaProfileCached>>> | undefined> {
39
+ try {
40
+ const profile = await getEtaProfileCached(join(workingDir, '.mstro', 'history'));
41
+ return profile ?? undefined;
42
+ } catch {
43
+ return undefined;
44
+ }
45
+ }
46
+
31
47
  function getPromptToolCompleteMessage(event: ToolUseEvent): string | null {
32
48
  const input = event.completeInput;
33
49
  if (!input) return null;
@@ -229,6 +245,7 @@ blocks: [] # Use backlog-relative paths: backlog/IS-NNN.md
229
245
  review_gate: auto
230
246
  output_type: auto # code = modify source files, document = produce written artifact, auto = infer
231
247
  output_file: null
248
+ agents: [] # Agent hints — see "agents field rules" below
232
249
  ---
233
250
 
234
251
  # IS-NNN: Title
@@ -268,6 +285,21 @@ Implementation guidance.
268
285
  - When output_type is \`document\`, "Files to Modify" entries are treated as references, not files to edit. The AI produces a document artifact and is reviewed on document quality.
269
286
  - When output_type is \`code\`, "Files to Modify" lists actual source files the AI must edit. The review gate verifies source files were changed.
270
287
 
288
+ ## agents field rules
289
+
290
+ The \`agents\` field is a list of agent hints for the executing Claude Code session. The executor uses Claude Code's Task tool to delegate work to matching subagents in the user's \`.claude/agents/\` directory (project / global / bundled), with a fallback to the general-purpose agent when no match is found.
291
+
292
+ - ALWAYS populate \`agents\` with the most relevant 1–4 agents for the work the issue describes. Empty arrays mean "no hints" — only use \`[]\` when no agent type plausibly applies (rare).
293
+ - Entries can be specific agent file names (e.g. \`backend-architect\`, \`frontend-developer\`, \`code-reviewer\`, \`security-auditor\`) OR general role pointers the user's system can match (e.g. \`backend engineer\`, \`product designer\`, \`marketing\`). Prefer specific names — they resolve more reliably.
294
+ - Match agents to the actual work, not the issue's surface topic. A "fix login button" issue is frontend work (\`frontend-developer\`); a "design login flow" issue is product/design (\`product-designer\`, \`ux-writer\`).
295
+ - Common pairings:
296
+ - Code implementation: pick from \`frontend-developer\`, \`backend-architect\`, \`typescript-pro\`, \`python-pro\`, \`golang-pro\`, etc., based on stack
297
+ - UI/design work: \`ui-designer\`, \`product-designer\`, \`design-system-architect\`, \`ux-writer\`
298
+ - Data/DB: \`database-architect\`, \`database-optimizer\`, \`data-engineer\`, \`sql-pro\`
299
+ - Quality: pair an implementation agent with \`code-reviewer\`, \`test-automator\`, or \`security-auditor\` for sensitive issues
300
+ - Product/strategy: \`product-manager\`, \`product-marketing\`, \`business-analyst\`
301
+ - YAML format: inline \`agents: [backend-architect, code-reviewer]\` or block list with \`-\` items both work.
302
+
271
303
  ## Epic creation rules
272
304
 
273
305
  - Create an EP-*.md file in ${cc.backlogPath} with type: epic and a children: [] field in front matter
@@ -287,11 +319,27 @@ User request: ${userPrompt}`;
287
319
  prepareAttachmentPrompt(ctx, enrichedPrompt, attachments, workingDir, cc.effectiveBoardId);
288
320
 
289
321
  const streamBoardId = cc.effectiveBoardId ?? null;
322
+ const etaProfile = await resolvePromptEtaProfile(workingDir);
323
+
324
+ // Tracks whether `planPromptResponse` has been broadcast. The web side
325
+ // treats this event as the authoritative completion signal — without it,
326
+ // the composer todo list stays stuck on a spinner. The finally block
327
+ // guarantees a completion broadcast even if the runner throws or exits
328
+ // through an unexpected path.
329
+ let responseSent = false;
330
+ const sendResponse = (response: string, success: boolean, error: string | null) => {
331
+ if (responseSent) return;
332
+ responseSent = true;
333
+ ctx.broadcastToAll({
334
+ type: 'planPromptResponse',
335
+ data: { response, success, error, boardId: streamBoardId },
336
+ });
337
+ };
290
338
 
291
339
  try {
292
340
  ctx.broadcastToAll({
293
341
  type: 'planPromptProgress',
294
- data: { message: 'Starting project planning...', boardId: streamBoardId },
342
+ data: { message: 'Starting project planning...', boardId: streamBoardId, etaProfile },
295
343
  });
296
344
 
297
345
  const runner = new ResilientRunner({
@@ -339,15 +387,11 @@ User request: ${userPrompt}`;
339
387
  data: { message: 'Finalizing project plan...', boardId: streamBoardId },
340
388
  });
341
389
 
342
- ctx.broadcastToAll({
343
- type: 'planPromptResponse',
344
- data: {
345
- response: result.completed ? 'Prompt executed successfully.' : (result.error || 'Unknown error'),
346
- success: result.completed,
347
- error: result.error || null,
348
- boardId: streamBoardId,
349
- },
350
- });
390
+ sendResponse(
391
+ result.completed ? 'Prompt executed successfully.' : (result.error || 'Unknown error'),
392
+ result.completed,
393
+ result.error || null,
394
+ );
351
395
 
352
396
  // Re-parse and broadcast updated state
353
397
  const updatedState = parsePlanDirectory(workingDir);
@@ -355,11 +399,19 @@ User request: ${userPrompt}`;
355
399
  ctx.broadcastToAll({ type: 'planStateUpdated', data: updatedState });
356
400
  }
357
401
  } catch (error) {
402
+ const errorMsg = error instanceof Error ? error.message : String(error);
358
403
  ctx.broadcastToAll({
359
404
  type: 'planError',
360
- data: { error: error instanceof Error ? error.message : String(error), boardId: streamBoardId },
405
+ data: { error: errorMsg, boardId: streamBoardId },
361
406
  });
407
+ // Send a completion signal too — `planError` clears streaming on the web
408
+ // but doesn't set the response banner. Without this, the user sees a
409
+ // half-finished UI (no spinner, no message).
410
+ sendResponse(errorMsg, false, errorMsg);
362
411
  } finally {
363
412
  cleanupAttachments(workingDir, attachmentSessionId);
413
+ // Defense in depth: guarantee a completion broadcast for any control
414
+ // flow not covered above (process abort, unexpected throw types, etc.).
415
+ sendResponse('Prompt execution ended unexpectedly.', false, 'No completion signal');
364
416
  }
365
417
  }
@@ -265,7 +265,6 @@ export class PlanExecutor extends EventEmitter {
265
265
  /** Run waves until done, paused, stopped, or stalled. */
266
266
  private async runWaveLoop(): Promise<'done' | 'stalled' | 'dead'> {
267
267
  let consecutiveZeroCompletions = 0;
268
- const maxParallel = await getBoardMaxParallelAgents(this.context.pmDir, this.effectiveBoardId(), this.emitWarn);
269
268
 
270
269
  while (!this.shouldStop && !this.shouldPause) {
271
270
  const readyIssues = await this.pickReadyIssues();
@@ -274,6 +273,9 @@ export class PlanExecutor extends EventEmitter {
274
273
  return await this.hasDeadIssues() ? 'dead' : 'done';
275
274
  }
276
275
 
276
+ // Re-read on each wave so users can scale agents up/down mid-execution
277
+ // without restarting — the new value takes effect at the next wave boundary.
278
+ const maxParallel = await getBoardMaxParallelAgents(this.context.pmDir, this.effectiveBoardId(), this.emitWarn);
277
279
  const completedCount = await this.executeWave(readyIssues.slice(0, maxParallel));
278
280
 
279
281
  if (completedCount > 0) {
@@ -8,6 +8,7 @@
8
8
  */
9
9
 
10
10
  import { join } from 'node:path';
11
+ import { type ResolvedAgent, resolveAgentHints } from './agent-resolver.js';
11
12
  import { resolveIsCodeTask } from './issue-classification.js';
12
13
  import type { Issue } from './types.js';
13
14
 
@@ -21,6 +22,41 @@ export interface IssuePromptOptions {
21
22
  outputPath: string;
22
23
  }
23
24
 
25
+ /**
26
+ * Render the Agents section of an issue prompt. Splits resolved hints from
27
+ * unresolved ones so Claude knows which names are real subagents to delegate to
28
+ * via the Task tool, vs. role hints for which the user has no installed match.
29
+ */
30
+ function renderAgentsSection(resolved: ResolvedAgent[]): string {
31
+ if (resolved.length === 0) return '';
32
+
33
+ const matched = resolved.filter(r => r.resolvedName);
34
+ const unmatched = resolved.filter(r => !r.resolvedName);
35
+
36
+ const lines: string[] = ['', '## Suggested Agents'];
37
+
38
+ if (matched.length > 0) {
39
+ lines.push('Delegate the relevant portions of this work to these subagents using the Task tool (subagent_type = the agent name). Use them as primary executors when the work matches their expertise:');
40
+ for (const r of matched) {
41
+ const desc = r.info?.description ? ` — ${r.info.description}` : '';
42
+ const labelHint = r.hint.toLowerCase() !== (r.resolvedName ?? '').toLowerCase()
43
+ ? ` (matched from "${r.hint}")`
44
+ : '';
45
+ lines.push(`- \`${r.resolvedName}\`${labelHint}${desc}`);
46
+ }
47
+ }
48
+
49
+ if (unmatched.length > 0) {
50
+ lines.push('');
51
+ lines.push('Role hints with no installed subagent match — use the general-purpose agent (or your best judgment) for work in these areas:');
52
+ for (const r of unmatched) {
53
+ lines.push(`- ${r.hint}`);
54
+ }
55
+ }
56
+
57
+ return `\n${lines.join('\n')}`;
58
+ }
59
+
24
60
  /**
25
61
  * Build a self-contained prompt for one issue.
26
62
  * The resulting Claude Code session will work independently —
@@ -45,6 +81,8 @@ export function buildIssuePrompt(options: IssuePromptOptions): string {
45
81
  ? `\n## Predecessor Outputs\nRead these before starting — they contain context from upstream issues:\n${predecessorDocs.map(d => `- ${d}`).join('\n')}`
46
82
  : '';
47
83
 
84
+ const agentSection = renderAgentsSection(resolveAgentHints(issue.agents, workingDir));
85
+
48
86
  const outDir = boardDir ? join(boardDir, 'out') : pmDir ? join(pmDir, 'out') : join(workingDir, '.mstro', 'pm', 'out');
49
87
 
50
88
  return `You are executing issue ${issue.id}: ${issue.title}.
@@ -67,7 +105,7 @@ ${criteria || 'No specific criteria defined.'}
67
105
 
68
106
  ### Technical Notes
69
107
  ${issue.technicalNotes || 'None'}
70
- ${files}${predecessorSection}
108
+ ${files}${predecessorSection}${agentSection}
71
109
 
72
110
  ## Your Task
73
111
 
@@ -287,6 +287,7 @@ export function parseIssue(content: string, filePath: string): Issue {
287
287
  reviewGate: (['none', 'auto', 'required'].includes(String(fm.review_gate)) ? String(fm.review_gate) : 'auto') as Issue['reviewGate'],
288
288
  outputType: (['code', 'document', 'auto'].includes(String(fm.output_type)) ? String(fm.output_type) : 'auto') as Issue['outputType'],
289
289
  outputFile: optionalString(fm.output_file),
290
+ agents: toStringArray(fm.agents),
290
291
  body,
291
292
  path: filePath,
292
293
  };
@@ -98,6 +98,10 @@ export interface Issue {
98
98
  outputType: 'code' | 'document' | 'auto';
99
99
  // Planned output file path (from front matter output_file, relative to working dir)
100
100
  outputFile: string | null;
101
+ // Agent hints for the executing Claude Code session — names or general roles
102
+ // (e.g. ["backend-architect", "database-architect"], ["product, design"]).
103
+ // Empty = no agent hints; the executor uses default behavior.
104
+ agents: string[];
101
105
  // Full markdown body
102
106
  body: string;
103
107
  // File path relative to .mstro/pm/