oh-my-codex 0.8.11 → 0.8.13

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 (162) hide show
  1. package/README.md +43 -35
  2. package/dist/agents/__tests__/definitions.test.js +1 -0
  3. package/dist/agents/__tests__/definitions.test.js.map +1 -1
  4. package/dist/agents/definitions.d.ts.map +1 -1
  5. package/dist/agents/definitions.js +11 -0
  6. package/dist/agents/definitions.js.map +1 -1
  7. package/dist/cli/__tests__/doctor-invalid-config.test.d.ts +2 -0
  8. package/dist/cli/__tests__/doctor-invalid-config.test.d.ts.map +1 -0
  9. package/dist/cli/__tests__/doctor-invalid-config.test.js +52 -0
  10. package/dist/cli/__tests__/doctor-invalid-config.test.js.map +1 -0
  11. package/dist/cli/__tests__/index.test.js +35 -3
  12. package/dist/cli/__tests__/index.test.js.map +1 -1
  13. package/dist/cli/__tests__/launch-fallback.test.d.ts +2 -0
  14. package/dist/cli/__tests__/launch-fallback.test.d.ts.map +1 -0
  15. package/dist/cli/__tests__/launch-fallback.test.js +60 -0
  16. package/dist/cli/__tests__/launch-fallback.test.js.map +1 -0
  17. package/dist/cli/__tests__/resume.test.d.ts +2 -0
  18. package/dist/cli/__tests__/resume.test.d.ts.map +1 -0
  19. package/dist/cli/__tests__/resume.test.js +78 -0
  20. package/dist/cli/__tests__/resume.test.js.map +1 -0
  21. package/dist/cli/__tests__/session-search-help.test.d.ts +2 -0
  22. package/dist/cli/__tests__/session-search-help.test.d.ts.map +1 -0
  23. package/dist/cli/__tests__/session-search-help.test.js +36 -0
  24. package/dist/cli/__tests__/session-search-help.test.js.map +1 -0
  25. package/dist/cli/__tests__/session-search.test.d.ts +2 -0
  26. package/dist/cli/__tests__/session-search.test.d.ts.map +1 -0
  27. package/dist/cli/__tests__/session-search.test.js +77 -0
  28. package/dist/cli/__tests__/session-search.test.js.map +1 -0
  29. package/dist/cli/__tests__/setup-prompts-overwrite.test.js +2 -0
  30. package/dist/cli/__tests__/setup-prompts-overwrite.test.js.map +1 -1
  31. package/dist/cli/__tests__/team-decompose.test.js +41 -15
  32. package/dist/cli/__tests__/team-decompose.test.js.map +1 -1
  33. package/dist/cli/__tests__/team.test.js +208 -3
  34. package/dist/cli/__tests__/team.test.js.map +1 -1
  35. package/dist/cli/doctor.d.ts.map +1 -1
  36. package/dist/cli/doctor.js +26 -0
  37. package/dist/cli/doctor.js.map +1 -1
  38. package/dist/cli/index.d.ts +4 -3
  39. package/dist/cli/index.d.ts.map +1 -1
  40. package/dist/cli/index.js +73 -27
  41. package/dist/cli/index.js.map +1 -1
  42. package/dist/cli/session-search.d.ts +8 -0
  43. package/dist/cli/session-search.d.ts.map +1 -0
  44. package/dist/cli/session-search.js +133 -0
  45. package/dist/cli/session-search.js.map +1 -0
  46. package/dist/cli/team.d.ts +13 -12
  47. package/dist/cli/team.d.ts.map +1 -1
  48. package/dist/cli/team.js +123 -39
  49. package/dist/cli/team.js.map +1 -1
  50. package/dist/hooks/__tests__/agents-overlay.test.js +33 -1
  51. package/dist/hooks/__tests__/agents-overlay.test.js.map +1 -1
  52. package/dist/hooks/__tests__/notify-fallback-watcher.test.js +219 -0
  53. package/dist/hooks/__tests__/notify-fallback-watcher.test.js.map +1 -1
  54. package/dist/hooks/__tests__/notify-hook-all-workers-idle.test.js +1 -0
  55. package/dist/hooks/__tests__/notify-hook-all-workers-idle.test.js.map +1 -1
  56. package/dist/hooks/__tests__/notify-hook-auto-nudge.test.js +64 -1
  57. package/dist/hooks/__tests__/notify-hook-auto-nudge.test.js.map +1 -1
  58. package/dist/hooks/__tests__/notify-hook-modules.test.js +7 -0
  59. package/dist/hooks/__tests__/notify-hook-modules.test.js.map +1 -1
  60. package/dist/hooks/__tests__/notify-hook-team-dispatch.test.js +2 -1
  61. package/dist/hooks/__tests__/notify-hook-team-dispatch.test.js.map +1 -1
  62. package/dist/hooks/__tests__/notify-hook-team-leader-nudge.test.js +420 -5
  63. package/dist/hooks/__tests__/notify-hook-team-leader-nudge.test.js.map +1 -1
  64. package/dist/hooks/__tests__/notify-hook-tmux-heal.test.js +95 -0
  65. package/dist/hooks/__tests__/notify-hook-tmux-heal.test.js.map +1 -1
  66. package/dist/hooks/__tests__/notify-hook-worker-idle.test.js +3 -0
  67. package/dist/hooks/__tests__/notify-hook-worker-idle.test.js.map +1 -1
  68. package/dist/hooks/__tests__/tmux-hook-engine.test.js +39 -1
  69. package/dist/hooks/__tests__/tmux-hook-engine.test.js.map +1 -1
  70. package/dist/hooks/agents-overlay.d.ts +6 -1
  71. package/dist/hooks/agents-overlay.d.ts.map +1 -1
  72. package/dist/hooks/agents-overlay.js +45 -4
  73. package/dist/hooks/agents-overlay.js.map +1 -1
  74. package/dist/mcp/team-server.js +1 -1
  75. package/dist/mcp/team-server.js.map +1 -1
  76. package/dist/session-history/__tests__/search.test.d.ts +2 -0
  77. package/dist/session-history/__tests__/search.test.d.ts.map +1 -0
  78. package/dist/session-history/__tests__/search.test.js +150 -0
  79. package/dist/session-history/__tests__/search.test.js.map +1 -0
  80. package/dist/session-history/search.d.ts +31 -0
  81. package/dist/session-history/search.d.ts.map +1 -0
  82. package/dist/session-history/search.js +326 -0
  83. package/dist/session-history/search.js.map +1 -0
  84. package/dist/team/__tests__/allocation-policy.test.d.ts +2 -0
  85. package/dist/team/__tests__/allocation-policy.test.d.ts.map +1 -0
  86. package/dist/team/__tests__/allocation-policy.test.js +39 -0
  87. package/dist/team/__tests__/allocation-policy.test.js.map +1 -0
  88. package/dist/team/__tests__/api-interop.test.js +140 -4
  89. package/dist/team/__tests__/api-interop.test.js.map +1 -1
  90. package/dist/team/__tests__/followup-planner.test.js +12 -0
  91. package/dist/team/__tests__/followup-planner.test.js.map +1 -1
  92. package/dist/team/__tests__/idle-nudge.test.js +6 -1
  93. package/dist/team/__tests__/idle-nudge.test.js.map +1 -1
  94. package/dist/team/__tests__/rebalance-policy.test.d.ts +2 -0
  95. package/dist/team/__tests__/rebalance-policy.test.d.ts.map +1 -0
  96. package/dist/team/__tests__/rebalance-policy.test.js +125 -0
  97. package/dist/team/__tests__/rebalance-policy.test.js.map +1 -0
  98. package/dist/team/__tests__/runtime.test.js +315 -12
  99. package/dist/team/__tests__/runtime.test.js.map +1 -1
  100. package/dist/team/__tests__/state.test.js +20 -1
  101. package/dist/team/__tests__/state.test.js.map +1 -1
  102. package/dist/team/__tests__/team-ops-contract.test.js +1 -0
  103. package/dist/team/__tests__/team-ops-contract.test.js.map +1 -1
  104. package/dist/team/__tests__/worker-bootstrap.test.js +20 -3
  105. package/dist/team/__tests__/worker-bootstrap.test.js.map +1 -1
  106. package/dist/team/allocation-policy.d.ts +23 -0
  107. package/dist/team/allocation-policy.d.ts.map +1 -0
  108. package/dist/team/allocation-policy.js +71 -0
  109. package/dist/team/allocation-policy.js.map +1 -0
  110. package/dist/team/api-interop.d.ts +1 -1
  111. package/dist/team/api-interop.d.ts.map +1 -1
  112. package/dist/team/api-interop.js +159 -0
  113. package/dist/team/api-interop.js.map +1 -1
  114. package/dist/team/idle-nudge.js +1 -1
  115. package/dist/team/idle-nudge.js.map +1 -1
  116. package/dist/team/rebalance-policy.d.ts +19 -0
  117. package/dist/team/rebalance-policy.d.ts.map +1 -0
  118. package/dist/team/rebalance-policy.js +48 -0
  119. package/dist/team/rebalance-policy.js.map +1 -0
  120. package/dist/team/runtime.d.ts.map +1 -1
  121. package/dist/team/runtime.js +132 -17
  122. package/dist/team/runtime.js.map +1 -1
  123. package/dist/team/state/types.d.ts +3 -0
  124. package/dist/team/state/types.d.ts.map +1 -1
  125. package/dist/team/state/types.js.map +1 -1
  126. package/dist/team/state.d.ts +8 -0
  127. package/dist/team/state.d.ts.map +1 -1
  128. package/dist/team/state.js +28 -12
  129. package/dist/team/state.js.map +1 -1
  130. package/dist/team/team-ops.d.ts +2 -1
  131. package/dist/team/team-ops.d.ts.map +1 -1
  132. package/dist/team/team-ops.js +1 -0
  133. package/dist/team/team-ops.js.map +1 -1
  134. package/dist/team/tmux-session.d.ts +5 -4
  135. package/dist/team/tmux-session.d.ts.map +1 -1
  136. package/dist/team/tmux-session.js +5 -67
  137. package/dist/team/tmux-session.js.map +1 -1
  138. package/dist/team/worker-bootstrap.d.ts +1 -0
  139. package/dist/team/worker-bootstrap.d.ts.map +1 -1
  140. package/dist/team/worker-bootstrap.js +9 -2
  141. package/dist/team/worker-bootstrap.js.map +1 -1
  142. package/package.json +2 -1
  143. package/prompts/team-executor.md +57 -0
  144. package/prompts/team-orchestrator.md +8 -0
  145. package/scripts/notify-fallback-watcher.js +295 -1
  146. package/scripts/notify-hook/auto-nudge.js +20 -4
  147. package/scripts/notify-hook/team-dispatch.js +11 -58
  148. package/scripts/notify-hook/team-leader-nudge.js +59 -12
  149. package/scripts/notify-hook/team-tmux-guard.js +28 -11
  150. package/scripts/notify-hook/team-worker.js +3 -1
  151. package/scripts/notify-hook/tmux-injection.js +12 -13
  152. package/scripts/tmux-hook-engine.js +56 -0
  153. package/skills/team/SKILL.md +14 -0
  154. package/templates/catalog-manifest.json +5 -0
  155. package/dist/rtk/__tests__/index.test.d.ts +0 -2
  156. package/dist/rtk/__tests__/index.test.d.ts.map +0 -1
  157. package/dist/rtk/__tests__/index.test.js +0 -104
  158. package/dist/rtk/__tests__/index.test.js.map +0 -1
  159. package/dist/rtk/index.d.ts +0 -130
  160. package/dist/rtk/index.d.ts.map +0 -1
  161. package/dist/rtk/index.js +0 -257
  162. package/dist/rtk/index.js.map +0 -1
@@ -0,0 +1,326 @@
1
+ import { createReadStream, existsSync } from 'node:fs';
2
+ import { readdir, stat } from 'node:fs/promises';
3
+ import { join, relative } from 'node:path';
4
+ import { createInterface } from 'node:readline';
5
+ import { codexHome } from '../utils/paths.js';
6
+ const DEFAULT_LIMIT = 10;
7
+ const DEFAULT_CONTEXT = 80;
8
+ const MAX_LIMIT = 100;
9
+ const MAX_CONTEXT = 400;
10
+ const DURATION_RE = /^(\d+)([smhdw])$/i;
11
+ function clampInteger(value, fallback, max) {
12
+ if (!Number.isInteger(value) || value < 0)
13
+ return fallback;
14
+ return Math.min(value, max);
15
+ }
16
+ function normalizeProjectFilter(project, cwd) {
17
+ if (!project)
18
+ return undefined;
19
+ const trimmed = project.trim();
20
+ if (trimmed === '')
21
+ return undefined;
22
+ if (trimmed === 'current')
23
+ return cwd;
24
+ if (trimmed === 'all')
25
+ return undefined;
26
+ return trimmed;
27
+ }
28
+ export function parseSinceSpec(value, now = Date.now()) {
29
+ if (!value)
30
+ return null;
31
+ const trimmed = value.trim();
32
+ if (trimmed === '')
33
+ return null;
34
+ const durationMatch = DURATION_RE.exec(trimmed);
35
+ if (durationMatch) {
36
+ const amount = Number(durationMatch[1]);
37
+ const unit = durationMatch[2].toLowerCase();
38
+ const multipliers = {
39
+ s: 1000,
40
+ m: 60_000,
41
+ h: 3_600_000,
42
+ d: 86_400_000,
43
+ w: 604_800_000,
44
+ };
45
+ return now - amount * multipliers[unit];
46
+ }
47
+ const timestamp = Date.parse(trimmed);
48
+ if (!Number.isNaN(timestamp))
49
+ return timestamp;
50
+ throw new Error(`Invalid --since value "${value}". Use formats like 7d, 24h, or 2026-03-10.`);
51
+ }
52
+ async function listRolloutFiles(root) {
53
+ if (!existsSync(root))
54
+ return [];
55
+ const files = [];
56
+ const queue = [root];
57
+ while (queue.length > 0) {
58
+ const dir = queue.pop();
59
+ if (!dir)
60
+ continue;
61
+ const entries = await readdir(dir, { withFileTypes: true }).catch(() => []);
62
+ for (const entry of entries) {
63
+ const path = join(dir, entry.name);
64
+ if (entry.isDirectory()) {
65
+ queue.push(path);
66
+ continue;
67
+ }
68
+ if (entry.isFile() && entry.name.startsWith('rollout-') && entry.name.endsWith('.jsonl')) {
69
+ files.push(path);
70
+ }
71
+ }
72
+ }
73
+ return files.sort((a, b) => b.localeCompare(a));
74
+ }
75
+ function safeParseJson(line) {
76
+ try {
77
+ const parsed = JSON.parse(line);
78
+ return parsed && typeof parsed === 'object' ? parsed : null;
79
+ }
80
+ catch {
81
+ return null;
82
+ }
83
+ }
84
+ function asString(value) {
85
+ return typeof value === 'string' && value.trim() !== '' ? value : null;
86
+ }
87
+ function collectTextFragments(value, fragments) {
88
+ if (typeof value === 'string') {
89
+ if (value.trim() !== '')
90
+ fragments.push(value);
91
+ return;
92
+ }
93
+ if (Array.isArray(value)) {
94
+ for (const item of value)
95
+ collectTextFragments(item, fragments);
96
+ return;
97
+ }
98
+ if (!value || typeof value !== 'object')
99
+ return;
100
+ for (const [key, child] of Object.entries(value)) {
101
+ if (key === 'base_instructions' || key === 'developer_instructions')
102
+ continue;
103
+ collectTextFragments(child, fragments);
104
+ }
105
+ }
106
+ function extractSessionMeta(parsed) {
107
+ if (!parsed || parsed.type !== 'session_meta')
108
+ return null;
109
+ const payload = parsed.payload;
110
+ if (!payload || typeof payload !== 'object')
111
+ return null;
112
+ const typedPayload = payload;
113
+ const sessionId = asString(typedPayload.id);
114
+ if (!sessionId)
115
+ return null;
116
+ return {
117
+ sessionId,
118
+ timestamp: asString(typedPayload.timestamp),
119
+ cwd: asString(typedPayload.cwd),
120
+ };
121
+ }
122
+ function extractSearchableTexts(parsed, rawLine) {
123
+ if (!parsed) {
124
+ return [{ text: rawLine, recordType: 'raw' }];
125
+ }
126
+ const topType = asString(parsed.type) ?? 'unknown';
127
+ const texts = [];
128
+ if (topType === 'session_meta') {
129
+ const payload = parsed.payload;
130
+ if (payload && typeof payload === 'object') {
131
+ const meta = payload;
132
+ const summary = [meta.id, meta.cwd, meta.agent_role, meta.agent_nickname]
133
+ .flatMap((value) => (typeof value === 'string' && value.trim() !== '') ? [value] : [])
134
+ .join(' ')
135
+ .trim();
136
+ if (summary)
137
+ texts.push({ text: summary, recordType: 'session_meta' });
138
+ }
139
+ return texts;
140
+ }
141
+ if (topType === 'event_msg') {
142
+ const payload = parsed.payload;
143
+ const payloadType = payload && typeof payload === 'object'
144
+ ? asString(payload.type) ?? 'unknown'
145
+ : 'unknown';
146
+ const fragments = [];
147
+ collectTextFragments(payload, fragments);
148
+ const text = fragments.join(' \n ').trim();
149
+ if (text) {
150
+ texts.push({ text, recordType: `event_msg:${payloadType}` });
151
+ }
152
+ return texts;
153
+ }
154
+ if (topType === 'response_item') {
155
+ const payload = parsed.payload;
156
+ if (!payload || typeof payload !== 'object')
157
+ return texts;
158
+ const typedPayload = payload;
159
+ const payloadType = asString(typedPayload.type) ?? 'unknown';
160
+ if (payloadType === 'message') {
161
+ const role = asString(typedPayload.role) ?? 'unknown';
162
+ if (role !== 'assistant' && role !== 'user')
163
+ return texts;
164
+ const fragments = [];
165
+ collectTextFragments(typedPayload.content, fragments);
166
+ const text = fragments.join(' \n ').trim();
167
+ if (text)
168
+ texts.push({ text, recordType: `response_item:${payloadType}:${role}` });
169
+ return texts;
170
+ }
171
+ const fragments = [];
172
+ if (payloadType === 'function_call') {
173
+ const name = asString(typedPayload.name);
174
+ if (name)
175
+ fragments.push(name);
176
+ const argumentsText = asString(typedPayload.arguments);
177
+ if (argumentsText)
178
+ fragments.push(argumentsText);
179
+ }
180
+ else if (payloadType === 'function_call_output') {
181
+ const outputText = typedPayload.output;
182
+ collectTextFragments(outputText, fragments);
183
+ }
184
+ else {
185
+ collectTextFragments(typedPayload, fragments);
186
+ }
187
+ const text = fragments.join(' \n ').trim();
188
+ if (text)
189
+ texts.push({ text, recordType: `response_item:${payloadType}` });
190
+ return texts;
191
+ }
192
+ return [{ text: rawLine, recordType: topType }];
193
+ }
194
+ function buildSnippet(text, query, context, caseSensitive) {
195
+ if (text === '')
196
+ return null;
197
+ const haystack = caseSensitive ? text : text.toLowerCase();
198
+ const needle = caseSensitive ? query : query.toLowerCase();
199
+ const index = haystack.indexOf(needle);
200
+ if (index < 0)
201
+ return null;
202
+ const start = Math.max(0, index - context);
203
+ const end = Math.min(text.length, index + query.length + context);
204
+ const prefix = start > 0 ? '…' : '';
205
+ const suffix = end < text.length ? '…' : '';
206
+ return `${prefix}${text.slice(start, end).replace(/\s+/g, ' ').trim()}${suffix}`;
207
+ }
208
+ function matchesFilter(value, filter, caseSensitive) {
209
+ if (!filter)
210
+ return true;
211
+ if (!value)
212
+ return false;
213
+ if (caseSensitive)
214
+ return value.includes(filter);
215
+ return value.toLowerCase().includes(filter.toLowerCase());
216
+ }
217
+ async function searchRolloutFile(filePath, options) {
218
+ const stream = createReadStream(filePath, 'utf-8');
219
+ const reader = createInterface({ input: stream, crlfDelay: Infinity });
220
+ const results = [];
221
+ let meta = null;
222
+ let lineNumber = 0;
223
+ let skipFile = false;
224
+ try {
225
+ for await (const line of reader) {
226
+ lineNumber += 1;
227
+ const parsed = safeParseJson(line);
228
+ if (lineNumber === 1) {
229
+ meta = extractSessionMeta(parsed);
230
+ const fallbackSessionId = filePath.split('rollout-')[1]?.replace(/\.jsonl$/, '') ?? filePath;
231
+ if (!meta) {
232
+ meta = { sessionId: fallbackSessionId, timestamp: null, cwd: null };
233
+ }
234
+ const sessionTimestamp = meta.timestamp ? Date.parse(meta.timestamp) : Number.NaN;
235
+ if (options.sinceCutoff != null && !Number.isNaN(sessionTimestamp) && sessionTimestamp < options.sinceCutoff) {
236
+ skipFile = true;
237
+ break;
238
+ }
239
+ if (!matchesFilter(meta.sessionId, options.session, options.caseSensitive)) {
240
+ skipFile = true;
241
+ break;
242
+ }
243
+ if (!matchesFilter(meta.cwd, options.projectFilter, options.caseSensitive)) {
244
+ skipFile = true;
245
+ break;
246
+ }
247
+ }
248
+ for (const candidate of extractSearchableTexts(parsed, line)) {
249
+ const snippet = buildSnippet(candidate.text, options.query, options.context, options.caseSensitive);
250
+ if (!snippet || !meta)
251
+ continue;
252
+ results.push({
253
+ session_id: meta.sessionId,
254
+ timestamp: meta.timestamp,
255
+ cwd: meta.cwd,
256
+ transcript_path: filePath,
257
+ transcript_path_relative: relative(options.codexHomeDir, filePath),
258
+ record_type: candidate.recordType,
259
+ line_number: lineNumber,
260
+ snippet,
261
+ });
262
+ if (results.length >= options.limit) {
263
+ return { meta, results };
264
+ }
265
+ }
266
+ }
267
+ }
268
+ finally {
269
+ reader.close();
270
+ stream.destroy();
271
+ }
272
+ return skipFile ? { meta, results: [] } : { meta, results };
273
+ }
274
+ export async function searchSessionHistory(options) {
275
+ const query = options.query.trim();
276
+ if (query === '') {
277
+ throw new Error('Search query must not be empty.');
278
+ }
279
+ const cwd = options.cwd ?? process.cwd();
280
+ const codexHomeDir = options.codexHomeDir ?? codexHome();
281
+ const limit = clampInteger(options.limit ?? DEFAULT_LIMIT, DEFAULT_LIMIT, MAX_LIMIT) || DEFAULT_LIMIT;
282
+ const context = clampInteger(options.context ?? DEFAULT_CONTEXT, DEFAULT_CONTEXT, MAX_CONTEXT) || DEFAULT_CONTEXT;
283
+ const caseSensitive = options.caseSensitive === true;
284
+ const sinceCutoff = parseSinceSpec(options.since, options.now ?? Date.now());
285
+ const projectFilter = normalizeProjectFilter(options.project, cwd);
286
+ const rolloutRoot = join(codexHomeDir, 'sessions');
287
+ const files = await listRolloutFiles(rolloutRoot);
288
+ const results = [];
289
+ let searchedFiles = 0;
290
+ const matchedSessions = new Set();
291
+ for (const filePath of files) {
292
+ if (results.length >= limit)
293
+ break;
294
+ const fileStat = await stat(filePath).catch(() => null);
295
+ if (!fileStat)
296
+ continue;
297
+ if (sinceCutoff != null && fileStat.mtimeMs < sinceCutoff) {
298
+ continue;
299
+ }
300
+ searchedFiles += 1;
301
+ const fileSearch = await searchRolloutFile(filePath, {
302
+ query,
303
+ context,
304
+ caseSensitive,
305
+ limit: limit - results.length,
306
+ session: options.session,
307
+ sinceCutoff,
308
+ projectFilter,
309
+ cwd,
310
+ codexHomeDir,
311
+ });
312
+ for (const result of fileSearch.results) {
313
+ results.push(result);
314
+ matchedSessions.add(result.session_id);
315
+ if (results.length >= limit)
316
+ break;
317
+ }
318
+ }
319
+ return {
320
+ query,
321
+ searched_files: searchedFiles,
322
+ matched_sessions: matchedSessions.size,
323
+ results,
324
+ };
325
+ }
326
+ //# sourceMappingURL=search.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search.js","sourceRoot":"","sources":["../../src/session-history/search.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=allocation-policy.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"allocation-policy.test.d.ts","sourceRoot":"","sources":["../../../src/team/__tests__/allocation-policy.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,39 @@
1
+ import assert from 'node:assert/strict';
2
+ import { describe, it } from 'node:test';
3
+ import { allocateTasksToWorkers, chooseTaskOwner } from '../allocation-policy.js';
4
+ describe('allocation-policy', () => {
5
+ it('prefers matching worker role when no prior assignments exist', () => {
6
+ const decision = chooseTaskOwner({ subject: 'write docs', description: 'document the feature', role: 'writer' }, [
7
+ { name: 'worker-1', role: 'executor' },
8
+ { name: 'worker-2', role: 'writer' },
9
+ ], []);
10
+ assert.equal(decision.owner, 'worker-2');
11
+ assert.match(decision.reason, /matches worker role writer/);
12
+ });
13
+ it('clusters same-role tasks on a specialized lane before spilling over', () => {
14
+ const assignments = allocateTasksToWorkers([
15
+ { subject: 'write docs 1', description: 'd1', role: 'writer' },
16
+ { subject: 'write docs 2', description: 'd2', role: 'writer' },
17
+ { subject: 'implement feature', description: 'd3', role: 'executor' },
18
+ ], [
19
+ { name: 'worker-1', role: 'executor' },
20
+ { name: 'worker-2', role: 'writer' },
21
+ { name: 'worker-3', role: 'test-engineer' },
22
+ ]);
23
+ assert.equal(assignments[0].owner, 'worker-2');
24
+ assert.equal(assignments[1].owner, 'worker-2');
25
+ assert.equal(assignments[2].owner, 'worker-1');
26
+ });
27
+ it('falls back to load balancing when roles do not differentiate the work', () => {
28
+ const assignments = allocateTasksToWorkers([
29
+ { subject: 'task a', description: 'a' },
30
+ { subject: 'task b', description: 'b' },
31
+ { subject: 'task c', description: 'c' },
32
+ ], [
33
+ { name: 'worker-1' },
34
+ { name: 'worker-2' },
35
+ ]);
36
+ assert.deepEqual(assignments.map((task) => task.owner), ['worker-1', 'worker-2', 'worker-1']);
37
+ });
38
+ });
39
+ //# sourceMappingURL=allocation-policy.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"allocation-policy.test.js","sourceRoot":"","sources":["../../../src/team/__tests__/allocation-policy.test.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,WAAW,CAAC;AACzC,OAAO,EAAE,sBAAsB,EAAE,eAAe,EAAE,MAAM,yBAAyB,CAAC;AAElF,QAAQ,CAAC,mBAAmB,EAAE,GAAG,EAAE;IACjC,EAAE,CAAC,8DAA8D,EAAE,GAAG,EAAE;QACtE,MAAM,QAAQ,GAAG,eAAe,CAC9B,EAAE,OAAO,EAAE,YAAY,EAAE,WAAW,EAAE,sBAAsB,EAAE,IAAI,EAAE,QAAQ,EAAE,EAC9E;YACE,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,UAAU,EAAE;YACtC,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,QAAQ,EAAE;SACrC,EACD,EAAE,CACH,CAAC;QAEF,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,EAAE,4BAA4B,CAAC,CAAC;IAC9D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qEAAqE,EAAE,GAAG,EAAE;QAC7E,MAAM,WAAW,GAAG,sBAAsB,CACxC;YACE,EAAE,OAAO,EAAE,cAAc,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE;YAC9D,EAAE,OAAO,EAAE,cAAc,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE,QAAQ,EAAE;YAC9D,EAAE,OAAO,EAAE,mBAAmB,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE,UAAU,EAAE;SACtE,EACD;YACE,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,UAAU,EAAE;YACtC,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,QAAQ,EAAE;YACpC,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,eAAe,EAAE;SAC5C,CACF,CAAC;QAEF,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QAC/C,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QAC/C,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;IACjD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uEAAuE,EAAE,GAAG,EAAE;QAC/E,MAAM,WAAW,GAAG,sBAAsB,CACxC;YACE,EAAE,OAAO,EAAE,QAAQ,EAAE,WAAW,EAAE,GAAG,EAAE;YACvC,EAAE,OAAO,EAAE,QAAQ,EAAE,WAAW,EAAE,GAAG,EAAE;YACvC,EAAE,OAAO,EAAE,QAAQ,EAAE,WAAW,EAAE,GAAG,EAAE;SACxC,EACD;YACE,EAAE,IAAI,EAAE,UAAU,EAAE;YACpB,EAAE,IAAI,EAAE,UAAU,EAAE;SACrB,CACF,CAAC;QAEF,MAAM,CAAC,SAAS,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,UAAU,EAAE,UAAU,EAAE,UAAU,CAAC,CAAC,CAAC;IAChG,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -4,7 +4,7 @@ import { mkdtemp, rm, mkdir, writeFile } from 'node:fs/promises';
4
4
  import { join } from 'node:path';
5
5
  import { tmpdir } from 'node:os';
6
6
  import { resolveTeamApiOperation, buildLegacyTeamDeprecationHint, executeTeamApiOperation, LEGACY_TEAM_MCP_TOOLS, TEAM_API_OPERATIONS, } from '../api-interop.js';
7
- import { initTeamState, createTask, sendDirectMessage, enqueueDispatchRequest, readDispatchRequest, appendTeamEvent, } from '../state.js';
7
+ import { initTeamState, createTask, sendDirectMessage, enqueueDispatchRequest, readDispatchRequest, appendTeamEvent, updateWorkerHeartbeat, writeMonitorSnapshot, writeWorkerStatus, } from '../state.js';
8
8
  async function setupTeam(name) {
9
9
  const cwd = await mkdtemp(join(tmpdir(), `omx-interop-${name}-`));
10
10
  await initTeamState(name, 'test task', 'executor', 2, cwd);
@@ -30,7 +30,7 @@ describe('resolveTeamApiOperation', () => {
30
30
  it('handles whitespace and casing', () => {
31
31
  assert.equal(resolveTeamApiOperation(' SEND_MESSAGE '), 'send-message');
32
32
  });
33
- it('resolves all 30 operations from the operation list', () => {
33
+ it('resolves all 32 operations from the operation list', () => {
34
34
  for (const op of TEAM_API_OPERATIONS) {
35
35
  assert.equal(resolveTeamApiOperation(op), op);
36
36
  }
@@ -64,8 +64,8 @@ describe('LEGACY_TEAM_MCP_TOOLS', () => {
64
64
  });
65
65
  });
66
66
  describe('TEAM_API_OPERATIONS', () => {
67
- it('contains 30 operations', () => {
68
- assert.equal(TEAM_API_OPERATIONS.length, 30);
67
+ it('contains 32 operations', () => {
68
+ assert.equal(TEAM_API_OPERATIONS.length, 32);
69
69
  });
70
70
  it('all use kebab-case', () => {
71
71
  for (const op of TEAM_API_OPERATIONS) {
@@ -941,6 +941,142 @@ describe('executeTeamApiOperation: await-event', () => {
941
941
  }
942
942
  });
943
943
  });
944
+ // ─── read-idle-state ────────────────────────────────────────────────────────
945
+ describe('executeTeamApiOperation: read-idle-state', () => {
946
+ it('returns structured idle state from summary, snapshot, and recent events', async () => {
947
+ const { cwd, cleanup } = await setupTeam('idle-state-team');
948
+ try {
949
+ await writeMonitorSnapshot('idle-state-team', {
950
+ taskStatusById: { '1': 'pending' },
951
+ workerAliveByName: { 'worker-1': true, 'worker-2': true },
952
+ workerStateByName: { 'worker-1': 'idle', 'worker-2': 'working' },
953
+ workerTurnCountByName: { 'worker-1': 3, 'worker-2': 5 },
954
+ workerTaskIdByName: { 'worker-1': '1', 'worker-2': '1' },
955
+ mailboxNotifiedByMessageId: {},
956
+ completedEventTaskIds: {},
957
+ }, cwd);
958
+ await appendTeamEvent('idle-state-team', {
959
+ type: 'worker_idle',
960
+ worker: 'worker-1',
961
+ task_id: '1',
962
+ prev_state: 'working',
963
+ }, cwd);
964
+ const allIdleEvent = await appendTeamEvent('idle-state-team', {
965
+ type: 'all_workers_idle',
966
+ worker: 'worker-1',
967
+ worker_count: 2,
968
+ }, cwd);
969
+ const result = await executeTeamApiOperation('read-idle-state', {
970
+ team_name: 'idle-state-team',
971
+ }, cwd);
972
+ assert.equal(result.ok, true);
973
+ if (result.ok) {
974
+ assert.equal(result.data.team_name, 'idle-state-team');
975
+ assert.equal(result.data.worker_count, 2);
976
+ assert.equal(result.data.idle_worker_count, 1);
977
+ assert.deepEqual(result.data.idle_workers, ['worker-1']);
978
+ assert.deepEqual(result.data.non_idle_workers, ['worker-2']);
979
+ assert.equal(result.data.all_workers_idle, false);
980
+ const byWorker = result.data.last_idle_transition_by_worker;
981
+ assert.equal(byWorker['worker-1']?.source_type, 'worker_idle');
982
+ assert.equal(byWorker['worker-2'], null);
983
+ const lastAllIdle = result.data.last_all_workers_idle_event;
984
+ assert.equal(lastAllIdle?.event_id, allIdleEvent.event_id);
985
+ assert.equal(lastAllIdle?.type, 'all_workers_idle');
986
+ assert.equal(lastAllIdle?.worker_count, 2);
987
+ }
988
+ }
989
+ finally {
990
+ await cleanup();
991
+ }
992
+ });
993
+ });
994
+ // ─── read-stall-state ───────────────────────────────────────────────────────
995
+ describe('executeTeamApiOperation: read-stall-state', () => {
996
+ it('returns structured stall state from summary, snapshot, and recent events', async () => {
997
+ const { cwd, cleanup } = await setupTeam('stall-state-team');
998
+ try {
999
+ const task = await createTask('stall-state-team', {
1000
+ subject: 'Pending work',
1001
+ description: 'Needs attention',
1002
+ status: 'pending',
1003
+ }, cwd);
1004
+ await writeWorkerStatus('stall-state-team', 'worker-1', {
1005
+ state: 'working',
1006
+ current_task_id: task.id,
1007
+ updated_at: '2026-03-10T10:00:00.000Z',
1008
+ }, cwd);
1009
+ await writeWorkerStatus('stall-state-team', 'worker-2', {
1010
+ state: 'idle',
1011
+ updated_at: '2026-03-10T10:00:00.000Z',
1012
+ }, cwd);
1013
+ await updateWorkerHeartbeat('stall-state-team', 'worker-1', {
1014
+ alive: true,
1015
+ pid: 101,
1016
+ turn_count: 1,
1017
+ last_turn_at: '2026-03-10T10:00:00.000Z',
1018
+ }, cwd);
1019
+ await updateWorkerHeartbeat('stall-state-team', 'worker-2', {
1020
+ alive: true,
1021
+ pid: 102,
1022
+ turn_count: 1,
1023
+ last_turn_at: '2026-03-10T10:00:00.000Z',
1024
+ }, cwd);
1025
+ const primed = await executeTeamApiOperation('get-summary', {
1026
+ team_name: 'stall-state-team',
1027
+ }, cwd);
1028
+ assert.equal(primed.ok, true);
1029
+ await updateWorkerHeartbeat('stall-state-team', 'worker-1', {
1030
+ alive: true,
1031
+ pid: 101,
1032
+ turn_count: 8,
1033
+ last_turn_at: '2026-03-10T10:05:00.000Z',
1034
+ }, cwd);
1035
+ await writeMonitorSnapshot('stall-state-team', {
1036
+ taskStatusById: { [task.id]: 'pending' },
1037
+ workerAliveByName: { 'worker-1': true, 'worker-2': true },
1038
+ workerStateByName: { 'worker-1': 'idle', 'worker-2': 'idle' },
1039
+ workerTurnCountByName: { 'worker-1': 8, 'worker-2': 1 },
1040
+ workerTaskIdByName: { 'worker-1': task.id, 'worker-2': '' },
1041
+ mailboxNotifiedByMessageId: {},
1042
+ completedEventTaskIds: {},
1043
+ }, cwd);
1044
+ const idleEvent = await appendTeamEvent('stall-state-team', {
1045
+ type: 'all_workers_idle',
1046
+ worker: 'worker-2',
1047
+ worker_count: 2,
1048
+ }, cwd);
1049
+ const nudgeEvent = await appendTeamEvent('stall-state-team', {
1050
+ type: 'team_leader_nudge',
1051
+ worker: 'leader-fixed',
1052
+ reason: 'all_workers_idle',
1053
+ }, cwd);
1054
+ const result = await executeTeamApiOperation('read-stall-state', {
1055
+ team_name: 'stall-state-team',
1056
+ }, cwd);
1057
+ assert.equal(result.ok, true);
1058
+ if (result.ok) {
1059
+ assert.equal(result.data.team_name, 'stall-state-team');
1060
+ assert.equal(result.data.team_stalled, true);
1061
+ assert.equal(result.data.leader_stale, true);
1062
+ assert.deepEqual(result.data.stalled_workers, ['worker-1']);
1063
+ assert.deepEqual(result.data.dead_workers, []);
1064
+ assert.equal(result.data.pending_task_count, 1);
1065
+ assert.equal(result.data.all_workers_idle, true);
1066
+ assert.match(result.data.reasons.join(' '), /workers_non_reporting:worker-1/);
1067
+ assert.match(result.data.reasons.join(' '), /leader_attention_pending:team_leader_nudge/);
1068
+ const lastAllIdle = result.data.last_all_workers_idle_event;
1069
+ const lastNudge = result.data.last_team_leader_nudge_event;
1070
+ assert.equal(lastAllIdle?.event_id, idleEvent.event_id);
1071
+ assert.equal(lastNudge?.event_id, nudgeEvent.event_id);
1072
+ assert.equal(lastNudge?.reason, 'all_workers_idle');
1073
+ }
1074
+ }
1075
+ finally {
1076
+ await cleanup();
1077
+ }
1078
+ });
1079
+ });
944
1080
  // ─── get-summary ──────────────────────────────────────────────────────────
945
1081
  describe('executeTeamApiOperation: get-summary', () => {
946
1082
  it('returns summary for existing team', async () => {