feique 1.3.3 → 1.5.0

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 (69) hide show
  1. package/README.en.md +3 -2
  2. package/README.md +3 -2
  3. package/dist/backend/claude.d.ts +2 -0
  4. package/dist/backend/claude.js +57 -54
  5. package/dist/backend/claude.js.map +1 -1
  6. package/dist/backend/codex.d.ts +2 -0
  7. package/dist/backend/codex.js +27 -0
  8. package/dist/backend/codex.js.map +1 -1
  9. package/dist/backend/factory.d.ts +43 -0
  10. package/dist/backend/factory.js +109 -29
  11. package/dist/backend/factory.js.map +1 -1
  12. package/dist/backend/probe.d.ts +27 -0
  13. package/dist/backend/probe.js +85 -0
  14. package/dist/backend/probe.js.map +1 -0
  15. package/dist/backend/qwen.d.ts +59 -0
  16. package/dist/backend/qwen.js +372 -0
  17. package/dist/backend/qwen.js.map +1 -0
  18. package/dist/backend/registry.d.ts +58 -0
  19. package/dist/backend/registry.js +23 -0
  20. package/dist/backend/registry.js.map +1 -0
  21. package/dist/backend/types.d.ts +6 -1
  22. package/dist/bridge/admin-config.d.ts +47 -0
  23. package/dist/bridge/admin-config.js +141 -0
  24. package/dist/bridge/admin-config.js.map +1 -0
  25. package/dist/bridge/collab-commands.d.ts +42 -0
  26. package/dist/bridge/collab-commands.js +254 -0
  27. package/dist/bridge/collab-commands.js.map +1 -0
  28. package/dist/bridge/commands.d.ts +1 -1
  29. package/dist/bridge/commands.js +10 -6
  30. package/dist/bridge/commands.js.map +1 -1
  31. package/dist/bridge/feishu-commands.d.ts +27 -0
  32. package/dist/bridge/feishu-commands.js +462 -0
  33. package/dist/bridge/feishu-commands.js.map +1 -0
  34. package/dist/bridge/intent-classifier.d.ts +7 -2
  35. package/dist/bridge/intent-classifier.js +1 -1
  36. package/dist/bridge/intent-classifier.js.map +1 -1
  37. package/dist/bridge/lifecycle.d.ts +46 -0
  38. package/dist/bridge/lifecycle.js +228 -0
  39. package/dist/bridge/lifecycle.js.map +1 -0
  40. package/dist/bridge/memory-commands.d.ts +26 -0
  41. package/dist/bridge/memory-commands.js +330 -0
  42. package/dist/bridge/memory-commands.js.map +1 -0
  43. package/dist/bridge/reply-builders.d.ts +30 -0
  44. package/dist/bridge/reply-builders.js +72 -0
  45. package/dist/bridge/reply-builders.js.map +1 -0
  46. package/dist/bridge/run-pipeline.d.ts +86 -0
  47. package/dist/bridge/run-pipeline.js +453 -0
  48. package/dist/bridge/run-pipeline.js.map +1 -0
  49. package/dist/bridge/run-scheduler.d.ts +47 -0
  50. package/dist/bridge/run-scheduler.js +121 -0
  51. package/dist/bridge/run-scheduler.js.map +1 -0
  52. package/dist/bridge/service-utils.d.ts +47 -0
  53. package/dist/bridge/service-utils.js +309 -0
  54. package/dist/bridge/service-utils.js.map +1 -0
  55. package/dist/bridge/service.d.ts +114 -66
  56. package/dist/bridge/service.js +230 -2199
  57. package/dist/bridge/service.js.map +1 -1
  58. package/dist/config/load.js +1 -1
  59. package/dist/config/load.js.map +1 -1
  60. package/dist/config/paths.js +1 -20
  61. package/dist/config/paths.js.map +1 -1
  62. package/dist/config/schema.d.ts +50 -16
  63. package/dist/config/schema.js +41 -2
  64. package/dist/config/schema.js.map +1 -1
  65. package/dist/feishu/long-connection.js +1 -0
  66. package/dist/feishu/long-connection.js.map +1 -1
  67. package/dist/feishu/webhook.js +1 -0
  68. package/dist/feishu/webhook.js.map +1 -1
  69. package/package.json +1 -1
@@ -0,0 +1,453 @@
1
+ import path from 'node:path';
2
+ import { randomUUID } from 'node:crypto';
3
+ import { resolveProjectBackendWithFailover } from '../backend/factory.js';
4
+ import { retrieveMemoryContext } from '../memory/retrieve.js';
5
+ import { summarizeThreadTurn } from '../memory/summarize.js';
6
+ import { extractInsights } from '../collaboration/knowledge.js';
7
+ import { recordRunOutcome, DEFAULT_TRUST_POLICY } from '../collaboration/trust.js';
8
+ import { buildProjectTimeline, buildOnboardingContext, isNewActor } from '../collaboration/timeline.js';
9
+ import { truncateForFeishuCard } from '../feishu/text.js';
10
+ import { estimateCost } from '../observability/cost.js';
11
+ import { extractFileMarkers, friendlyErrorMessage, truncateExcerpt } from './service-utils.js';
12
+ export async function executePrompt(host, input) {
13
+ const conversation = (await host.sessionStore.getConversation(input.sessionKey)) ??
14
+ (await host.sessionStore.ensureConversation(input.sessionKey, {
15
+ chat_id: input.chatId,
16
+ actor_id: input.actorId,
17
+ tenant_key: input.tenantKey,
18
+ scope: input.project.session_scope,
19
+ }));
20
+ let currentSession = conversation.projects[input.projectAlias];
21
+ // Auto-adopt latest local session when no active session exists
22
+ if (!currentSession?.thread_id && host.config.service.project_switch_auto_adopt_latest) {
23
+ try {
24
+ const sessionBackendOverrideForAdopt = await host.sessionStore.getProjectBackend(input.sessionKey, input.projectAlias);
25
+ const backendForAdopt = host.resolveBackendByName(input.projectAlias, sessionBackendOverrideForAdopt);
26
+ const latestLocal = await backendForAdopt.findLatestSession(input.project.root);
27
+ if (latestLocal) {
28
+ await host.sessionStore.upsertProjectSession(input.sessionKey, input.projectAlias, {
29
+ thread_id: latestLocal.sessionId,
30
+ });
31
+ const refreshed = await host.sessionStore.getConversation(input.sessionKey);
32
+ currentSession = refreshed?.projects[input.projectAlias];
33
+ host.logger.info({ projectAlias: input.projectAlias, sessionId: latestLocal.sessionId, backend: latestLocal.backend }, 'Auto-adopted latest local session for prompt execution');
34
+ }
35
+ }
36
+ catch { /* auto-adopt is best-effort */ }
37
+ }
38
+ if (host.config.service.memory_enabled) {
39
+ await host.memoryStore.cleanupExpiredMemories();
40
+ }
41
+ const memoryContext = host.config.service.memory_enabled
42
+ ? await retrieveMemoryContext(host.memoryStore, {
43
+ conversationKey: input.sessionKey,
44
+ projectAlias: input.projectAlias,
45
+ threadId: currentSession?.thread_id,
46
+ query: input.prompt,
47
+ searchLimit: host.config.service.memory_search_limit,
48
+ groupChatId: input.incomingMessage.chat_type === 'group' ? input.incomingMessage.chat_id : undefined,
49
+ includeGroupMemories: host.config.service.memory_group_enabled && input.incomingMessage.chat_type === 'group',
50
+ })
51
+ : { pinnedMemories: [], relevantMemories: [], pinnedGroupMemories: [], relevantGroupMemories: [] };
52
+ // Direction 6: Inject onboarding context for new actors
53
+ let onboardingPrefix = '';
54
+ if (input.actorId && host.config.service.memory_enabled) {
55
+ try {
56
+ const allRuns = await host.runStateStore.listRuns();
57
+ if (isNewActor(input.actorId, allRuns, input.projectAlias)) {
58
+ const memories = await host.memoryStore.listRecentMemories({ scope: 'project', project_alias: input.projectAlias }, 10);
59
+ const timeline = buildProjectTimeline(allRuns, memories, [], input.projectAlias, 10);
60
+ onboardingPrefix = buildOnboardingContext(timeline, memories, input.projectAlias);
61
+ }
62
+ }
63
+ catch { /* onboarding injection is best-effort */ }
64
+ }
65
+ const effectivePrompt = onboardingPrefix
66
+ ? `${onboardingPrefix}\n\n${input.prompt}`
67
+ : input.prompt;
68
+ const bridgePrompt = await host.buildBridgePrompt(input.projectAlias, input.project, input.incomingMessage, effectivePrompt, memoryContext);
69
+ const startedAt = Date.now();
70
+ const projectRoot = path.resolve(input.project.root);
71
+ const runId = input.runId ?? randomUUID();
72
+ let lastProgressUpdate = 0;
73
+ const activeRun = {
74
+ runId,
75
+ controller: new AbortController(),
76
+ };
77
+ host.activeRuns.set(input.queueKey, activeRun);
78
+ const sessionBackendOverride = await host.sessionStore.getProjectBackend(input.sessionKey, input.projectAlias);
79
+ const failoverResolution = await resolveProjectBackendWithFailover(host.config, input.projectAlias, sessionBackendOverride, host.codexSessionIndex);
80
+ const backend = failoverResolution.backend;
81
+ if (failoverResolution.failover) {
82
+ await host.handleBackendFailover(input.chatId, input.projectAlias, runId, failoverResolution.failover);
83
+ }
84
+ const backendLabel = backend.name === 'claude' ? 'Claude' : backend.name === 'qwen' ? 'Qwen' : 'Codex';
85
+ await host.updateRunStartedReply(input.chatId, input.projectAlias, runId, backendLabel);
86
+ await host.runStateStore.upsertRun(runId, {
87
+ queue_key: input.queueKey,
88
+ conversation_key: input.sessionKey,
89
+ project_alias: input.projectAlias,
90
+ chat_id: input.chatId,
91
+ actor_id: input.actorId,
92
+ actor_name: input.incomingMessage.actor_name,
93
+ session_id: currentSession?.thread_id,
94
+ project_root: projectRoot,
95
+ prompt_excerpt: truncateExcerpt(input.prompt),
96
+ status: 'running',
97
+ status_detail: undefined,
98
+ });
99
+ await host.auditLog.append({
100
+ type: 'codex.run.started',
101
+ run_id: runId,
102
+ chat_id: input.chatId,
103
+ actor_id: input.actorId,
104
+ project_alias: input.projectAlias,
105
+ conversation_key: input.sessionKey,
106
+ session_id: currentSession?.thread_id,
107
+ prompt: input.prompt,
108
+ });
109
+ await host.appendProjectAuditEvent(input.projectAlias, input.project, {
110
+ type: 'codex.run.started',
111
+ run_id: runId,
112
+ chat_id: input.chatId,
113
+ actor_id: input.actorId,
114
+ session_id: currentSession?.thread_id,
115
+ project_root: projectRoot,
116
+ });
117
+ host.logger.info({
118
+ runId,
119
+ queueKey: input.queueKey,
120
+ sessionKey: input.sessionKey,
121
+ projectAlias: input.projectAlias,
122
+ projectRoot,
123
+ sessionId: currentSession?.thread_id,
124
+ }, 'Codex run started');
125
+ host.metrics?.recordCodexTurnStarted(input.projectAlias, runId);
126
+ try {
127
+ const outputTokenLimit = backend.name === 'claude'
128
+ ? (host.config.claude?.output_token_limit ?? host.config.codex.output_token_limit)
129
+ : backend.name === 'qwen'
130
+ ? (host.config.qwen?.output_token_limit ?? host.config.codex.output_token_limit)
131
+ : host.config.codex.output_token_limit;
132
+ const result = await backend.run({
133
+ workdir: input.project.root,
134
+ prompt: bridgePrompt,
135
+ sessionId: currentSession?.thread_id,
136
+ timeoutMs: backend.name === 'claude'
137
+ ? (host.config.claude?.run_timeout_ms ?? host.config.codex.run_timeout_ms)
138
+ : backend.name === 'qwen'
139
+ ? (host.config.qwen?.run_timeout_ms ?? host.config.codex.run_timeout_ms)
140
+ : host.config.codex.run_timeout_ms,
141
+ signal: activeRun.controller.signal,
142
+ logger: host.logger,
143
+ projectConfig: backend.name === 'codex'
144
+ ? {
145
+ profile: input.project.profile ?? host.config.codex.default_profile,
146
+ model: input.project.codex_model,
147
+ sandbox: input.project.codex_sandbox ?? input.project.sandbox ?? host.config.codex.default_sandbox,
148
+ tempDir: host.resolveProjectTempDir(input.projectAlias, input.project),
149
+ cacheDir: host.resolveProjectCacheDir(input.projectAlias, input.project),
150
+ }
151
+ : backend.name === 'qwen'
152
+ ? {
153
+ approvalMode: input.project.qwen_approval_mode ?? host.config.qwen?.default_approval_mode,
154
+ model: input.project.qwen_model ?? host.config.qwen?.default_model,
155
+ allowedTools: input.project.qwen_allowed_tools ?? host.config.qwen?.allowed_tools,
156
+ systemPromptAppend: input.project.qwen_system_prompt_append ?? host.config.qwen?.system_prompt_append,
157
+ }
158
+ : {
159
+ permissionMode: input.project.claude_permission_mode ?? host.config.claude?.default_permission_mode,
160
+ model: input.project.claude_model ?? host.config.claude?.default_model,
161
+ maxBudgetUsd: input.project.claude_max_budget_usd ?? host.config.claude?.max_budget_usd,
162
+ allowedTools: input.project.claude_allowed_tools ?? host.config.claude?.allowed_tools,
163
+ systemPromptAppend: input.project.claude_system_prompt_append ?? host.config.claude?.system_prompt_append,
164
+ },
165
+ onSpawn: async (pid) => {
166
+ activeRun.pid = pid;
167
+ await host.runStateStore.upsertRun(runId, {
168
+ queue_key: input.queueKey,
169
+ conversation_key: input.sessionKey,
170
+ project_alias: input.projectAlias,
171
+ chat_id: input.chatId,
172
+ actor_id: input.actorId,
173
+ session_id: currentSession?.thread_id,
174
+ project_root: projectRoot,
175
+ prompt_excerpt: truncateExcerpt(input.prompt),
176
+ status: 'running',
177
+ status_detail: undefined,
178
+ pid,
179
+ });
180
+ },
181
+ onEvent: async (event) => {
182
+ if (!host.config.service.emit_progress_updates) {
183
+ return;
184
+ }
185
+ const message = backend.summarizeEvent(event);
186
+ if (!message) {
187
+ return;
188
+ }
189
+ const now = Date.now();
190
+ if (now - lastProgressUpdate < host.config.service.progress_update_interval_ms) {
191
+ return;
192
+ }
193
+ lastProgressUpdate = now;
194
+ await host.updateRunProgressReply(input, runId, message, backendLabel);
195
+ },
196
+ });
197
+ const excerpt = result.finalMessage.slice(0, outputTokenLimit);
198
+ if (!excerpt.trim()) {
199
+ host.logger.warn({
200
+ runId,
201
+ queueKey: input.queueKey,
202
+ sessionKey: input.sessionKey,
203
+ projectAlias: input.projectAlias,
204
+ sessionId: result.sessionId,
205
+ durationMs: Date.now() - startedAt,
206
+ }, 'Codex run completed without a displayable final message');
207
+ }
208
+ // Extract and send any [SEND_FILE:path] markers before text reply
209
+ const { cleanText: excerptWithoutFiles, filePaths } = extractFileMarkers(excerpt);
210
+ if (filePaths.length > 0) {
211
+ for (const filePath of filePaths) {
212
+ try {
213
+ await host.feishuClient.sendFile(input.chatId, filePath);
214
+ host.logger.info({ chatId: input.chatId, filePath }, 'Sent file to Feishu');
215
+ }
216
+ catch (err) {
217
+ const msg = err instanceof Error ? err.message : String(err);
218
+ host.logger.warn({ chatId: input.chatId, filePath, error: msg }, 'Failed to send file to Feishu');
219
+ // Notify user about the failure inline
220
+ excerptWithoutFiles === excerpt || await host.feishuClient.sendText(input.chatId, `⚠️ 文件发送失败: ${filePath}\n${msg}`);
221
+ }
222
+ }
223
+ }
224
+ const finalExcerpt = excerptWithoutFiles.trim() || excerpt;
225
+ const cardSummary = truncateForFeishuCard(finalExcerpt || `${backendLabel} 已完成,但没有返回可显示文本。`);
226
+ await host.auditLog.append({
227
+ type: 'codex.run.completed',
228
+ run_id: runId,
229
+ chat_id: input.chatId,
230
+ actor_id: input.actorId,
231
+ project_alias: input.projectAlias,
232
+ conversation_key: input.sessionKey,
233
+ session_id: result.sessionId,
234
+ exit_code: result.exitCode,
235
+ duration_ms: Date.now() - startedAt,
236
+ backend: backend.name,
237
+ });
238
+ await host.appendProjectAuditEvent(input.projectAlias, input.project, {
239
+ type: 'codex.run.completed',
240
+ run_id: runId,
241
+ chat_id: input.chatId,
242
+ actor_id: input.actorId,
243
+ session_id: result.sessionId,
244
+ duration_ms: Date.now() - startedAt,
245
+ backend: backend.name,
246
+ });
247
+ host.logger.info({
248
+ runId,
249
+ queueKey: input.queueKey,
250
+ sessionKey: input.sessionKey,
251
+ projectAlias: input.projectAlias,
252
+ sessionId: result.sessionId,
253
+ exitCode: result.exitCode,
254
+ finalMessageChars: excerpt.length,
255
+ durationMs: Date.now() - startedAt,
256
+ }, 'Codex run completed');
257
+ await host.sessionStore.upsertProjectSession(input.sessionKey, input.projectAlias, {
258
+ thread_id: result.sessionId,
259
+ last_prompt: input.prompt,
260
+ last_response_excerpt: excerpt,
261
+ });
262
+ if (host.config.service.memory_enabled && result.sessionId) {
263
+ const summaryDraft = summarizeThreadTurn({
264
+ previousSummary: memoryContext.threadSummary?.summary,
265
+ prompt: input.prompt,
266
+ responseExcerpt: excerpt,
267
+ maxChars: host.config.service.thread_summary_max_chars,
268
+ });
269
+ const threadSummary = await host.memoryStore.upsertThreadSummary({
270
+ conversation_key: input.sessionKey,
271
+ project_alias: input.projectAlias,
272
+ thread_id: result.sessionId,
273
+ summary: summaryDraft.summary,
274
+ recent_prompt: input.prompt,
275
+ recent_response_excerpt: excerpt,
276
+ files_touched: summaryDraft.filesTouched,
277
+ open_tasks: summaryDraft.openTasks,
278
+ decisions: summaryDraft.decisions,
279
+ });
280
+ await host.auditLog.append({
281
+ type: 'memory.thread_summary.updated',
282
+ run_id: runId,
283
+ project_alias: input.projectAlias,
284
+ conversation_key: input.sessionKey,
285
+ thread_id: result.sessionId,
286
+ files_touched: threadSummary.files_touched,
287
+ });
288
+ }
289
+ await host.enforceSessionHistoryLimit(input.sessionKey, input.projectAlias);
290
+ await host.runStateStore.upsertRun(runId, {
291
+ queue_key: input.queueKey,
292
+ conversation_key: input.sessionKey,
293
+ project_alias: input.projectAlias,
294
+ chat_id: input.chatId,
295
+ actor_id: input.actorId,
296
+ session_id: result.sessionId,
297
+ project_root: projectRoot,
298
+ pid: activeRun.pid,
299
+ prompt_excerpt: truncateExcerpt(input.prompt),
300
+ status: 'success',
301
+ status_detail: undefined,
302
+ input_tokens: result.inputTokens,
303
+ output_tokens: result.outputTokens,
304
+ estimated_cost_usd: estimateCost(result.inputTokens, result.outputTokens, backend.name),
305
+ });
306
+ host.metrics?.recordCodexTurn('success', input.projectAlias, (Date.now() - startedAt) / 1000, runId);
307
+ // Record cost and token metrics
308
+ if (result.inputTokens || result.outputTokens) {
309
+ const costUsd = estimateCost(result.inputTokens, result.outputTokens, backend.name) ?? 0;
310
+ host.metrics?.recordCost(input.projectAlias, backend.name, costUsd);
311
+ host.metrics?.recordTokens(input.projectAlias, backend.name, result.inputTokens ?? 0, result.outputTokens ?? 0);
312
+ }
313
+ // Direction 5: Record trust outcome
314
+ try {
315
+ const trustState = await host.trustStore.getOrCreate(input.projectAlias);
316
+ const updated = recordRunOutcome(trustState, true, DEFAULT_TRUST_POLICY);
317
+ await host.trustStore.update(input.projectAlias, updated);
318
+ host.metrics?.recordTrustLevel(input.projectAlias, updated.current_level);
319
+ }
320
+ catch { /* trust tracking is best-effort */ }
321
+ // Proactive alerts: check if this run triggers any team alerts
322
+ try {
323
+ const completedRunState = await host.runStateStore.getRun(runId);
324
+ if (completedRunState) {
325
+ await host.checkAndSendAlerts(completedRunState);
326
+ }
327
+ }
328
+ catch { /* alerts are best-effort */ }
329
+ // Direction 2: Auto-extract knowledge
330
+ if (host.config.service.memory_enabled && excerpt.length >= 100) {
331
+ try {
332
+ const insight = extractInsights(input.prompt, excerpt, input.projectAlias);
333
+ if (insight) {
334
+ await host.memoryStore.saveProjectMemory({
335
+ project_alias: insight.project_alias,
336
+ title: insight.title,
337
+ content: insight.content,
338
+ tags: insight.tags,
339
+ source: 'auto',
340
+ created_by: input.actorId,
341
+ });
342
+ }
343
+ }
344
+ catch { /* auto-extraction is best-effort */ }
345
+ }
346
+ await host.sendOrUpdateRunOutcome({
347
+ input,
348
+ runId,
349
+ title: `${backendLabel} 已完成`,
350
+ body: finalExcerpt || `${backendLabel} 已完成,但没有返回可显示文本。`,
351
+ runStatus: 'success',
352
+ runPhase: '已完成',
353
+ cardSummary,
354
+ sessionId: result.sessionId,
355
+ });
356
+ }
357
+ catch (error) {
358
+ const message = error instanceof Error ? error.message : String(error);
359
+ const cancelled = error instanceof Error && error.name === 'AbortError' && activeRun.cancelReason === 'user';
360
+ const status = cancelled ? 'cancelled' : 'failure';
361
+ if (!cancelled && error instanceof Error && error.name === 'AbortError') {
362
+ activeRun.cancelReason = 'timeout';
363
+ }
364
+ if (!cancelled && activeRun.cancelReason === 'timeout') {
365
+ host.metrics?.recordCodexTurn('failure', input.projectAlias, (Date.now() - startedAt) / 1000, runId);
366
+ }
367
+ else {
368
+ host.metrics?.recordCodexTurn(cancelled ? 'cancelled' : 'failure', input.projectAlias, (Date.now() - startedAt) / 1000, runId);
369
+ }
370
+ await host.runStateStore.upsertRun(runId, {
371
+ queue_key: input.queueKey,
372
+ conversation_key: input.sessionKey,
373
+ project_alias: input.projectAlias,
374
+ chat_id: input.chatId,
375
+ actor_id: input.actorId,
376
+ session_id: currentSession?.thread_id,
377
+ project_root: projectRoot,
378
+ pid: activeRun.pid,
379
+ prompt_excerpt: truncateExcerpt(input.prompt),
380
+ status,
381
+ status_detail: undefined,
382
+ error: message,
383
+ });
384
+ await host.auditLog.append({
385
+ type: cancelled ? 'codex.run.cancelled' : 'codex.run.failed',
386
+ run_id: runId,
387
+ chat_id: input.chatId,
388
+ actor_id: input.actorId,
389
+ project_alias: input.projectAlias,
390
+ conversation_key: input.sessionKey,
391
+ error: message,
392
+ });
393
+ await host.appendProjectAuditEvent(input.projectAlias, input.project, {
394
+ type: cancelled ? 'codex.run.cancelled' : 'codex.run.failed',
395
+ run_id: runId,
396
+ chat_id: input.chatId,
397
+ actor_id: input.actorId,
398
+ error: message,
399
+ });
400
+ // Direction 5: Record trust failure (only for actual failures, not cancellations)
401
+ if (!cancelled) {
402
+ try {
403
+ const trustState = await host.trustStore.getOrCreate(input.projectAlias);
404
+ const updated = recordRunOutcome(trustState, false, DEFAULT_TRUST_POLICY);
405
+ await host.trustStore.update(input.projectAlias, updated);
406
+ }
407
+ catch { /* trust tracking is best-effort */ }
408
+ // Notify project chats about the failure
409
+ await host.notifyProjectChats(input.projectAlias, `❌ 运行失败 [${input.projectAlias}]\n${message.slice(0, 200)}`);
410
+ // Proactive alerts on failure
411
+ try {
412
+ const failedRunState = await host.runStateStore.getRun(runId);
413
+ if (failedRunState) {
414
+ await host.checkAndSendAlerts(failedRunState);
415
+ }
416
+ }
417
+ catch { /* alerts are best-effort */ }
418
+ }
419
+ if (cancelled) {
420
+ host.logger.warn({
421
+ runId,
422
+ queueKey: input.queueKey,
423
+ sessionKey: input.sessionKey,
424
+ projectAlias: input.projectAlias,
425
+ durationMs: Date.now() - startedAt,
426
+ }, 'Codex run cancelled');
427
+ }
428
+ else {
429
+ host.logger.error({
430
+ error,
431
+ runId,
432
+ queueKey: input.queueKey,
433
+ sessionKey: input.sessionKey,
434
+ projectAlias: input.projectAlias,
435
+ durationMs: Date.now() - startedAt,
436
+ }, 'Codex run failed');
437
+ }
438
+ await host.sendOrUpdateRunOutcome({
439
+ input,
440
+ runId,
441
+ title: cancelled ? '运行已取消' : '执行失败',
442
+ body: cancelled ? '当前运行已取消。' : ['执行失败。', '', friendlyErrorMessage(message)].join('\n'),
443
+ runStatus: cancelled ? 'cancelled' : 'failure',
444
+ runPhase: cancelled ? '已取消' : '失败',
445
+ cardSummary: truncateForFeishuCard(cancelled ? '当前运行已取消。' : friendlyErrorMessage(message)),
446
+ });
447
+ }
448
+ finally {
449
+ host.activeRuns.delete(input.queueKey);
450
+ host.runReplyTargets.delete(runId);
451
+ }
452
+ }
453
+ //# sourceMappingURL=run-pipeline.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-pipeline.js","sourceRoot":"","sources":["../../src/bridge/run-pipeline.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,47 @@
1
+ import type { BridgeConfig, ProjectConfig } from '../config/schema.js';
2
+ import type { Logger } from '../logging.js';
3
+ import type { AuditLog } from '../state/audit-log.js';
4
+ import { RunStateStore, type RunState } from '../state/run-state-store.js';
5
+ import type { MetricsRegistry } from '../observability/metrics.js';
6
+ import type { TaskQueue } from './task-queue.js';
7
+ /**
8
+ * Subset of FeiqueService needed by the run scheduler. The scheduler owns
9
+ * nothing — it reaches back through the host to enqueue work on the
10
+ * shared TaskQueues, update run state, write audit events, and emit
11
+ * metrics.
12
+ */
13
+ export interface RunSchedulerHost {
14
+ readonly queue: TaskQueue;
15
+ readonly projectRootQueue: TaskQueue;
16
+ readonly auditLog: AuditLog;
17
+ readonly runStateStore: RunStateStore;
18
+ readonly logger: Logger;
19
+ readonly metrics?: MetricsRegistry;
20
+ }
21
+ export interface QueuedExecutionNotice {
22
+ runId: string;
23
+ detail: string;
24
+ reason: 'project' | 'project-root';
25
+ }
26
+ export interface ScheduledProjectExecution {
27
+ runId: string;
28
+ queued: QueuedExecutionNotice | null;
29
+ release: () => void;
30
+ completion: Promise<void>;
31
+ }
32
+ export interface ScheduleProjectContext {
33
+ projectAlias: string;
34
+ project: ProjectConfig;
35
+ sessionKey: string;
36
+ queueKey: string;
37
+ }
38
+ export interface ScheduleMetadata {
39
+ chatId: string;
40
+ actorId?: string;
41
+ actorName?: string;
42
+ prompt: string;
43
+ }
44
+ export declare function buildAcknowledgedRunReply(projectAlias: string, phase: '已接收' | '排队中' | '处理中', detail: string, mode: BridgeConfig['service']['reply_mode']): string;
45
+ export declare function buildQueuedStatusDetail(projectAlias: string, reason: QueuedExecutionNotice['reason'], frontCount: number, blockingRun: RunState | null): string;
46
+ export declare function buildRunStatusSummary(lastResponseExcerpt?: string, activeRun?: RunState | null): string;
47
+ export declare function scheduleProjectExecution(host: RunSchedulerHost, projectContext: ScheduleProjectContext, metadata: ScheduleMetadata, task: (runId?: string) => Promise<void>): Promise<ScheduledProjectExecution>;
@@ -0,0 +1,121 @@
1
+ import path from 'node:path';
2
+ import { randomUUID } from 'node:crypto';
3
+ import { buildProjectRootQueueKey, createDeferred, truncateExcerpt } from './service-utils.js';
4
+ // ---------------------------------------------------------------------------
5
+ // Pure status-line builders (exported so service.ts reply paths can inline them)
6
+ // ---------------------------------------------------------------------------
7
+ export function buildAcknowledgedRunReply(projectAlias, phase, detail, mode) {
8
+ if (mode === 'text') {
9
+ return [`项目: ${projectAlias}`, `状态: ${phase}`, '', detail].join('\n');
10
+ }
11
+ return detail;
12
+ }
13
+ export function buildQueuedStatusDetail(projectAlias, reason, frontCount, blockingRun) {
14
+ const lines = [
15
+ reason === 'project' ? `当前项目 ${projectAlias} 已有任务在处理,已进入排队。` : '当前仓库正在被其他会话操作,已进入排队。',
16
+ frontCount > 0 ? `前方还有 ${frontCount} 个任务。` : null,
17
+ ];
18
+ if (blockingRun) {
19
+ const actorName = blockingRun.actor_name ?? blockingRun.actor_id ?? '其他成员';
20
+ lines.push(`当前执行: ${actorName}`);
21
+ const elapsedMs = Date.now() - new Date(blockingRun.started_at).getTime();
22
+ const elapsedMin = Math.round(elapsedMs / 60_000);
23
+ if (elapsedMin > 0) {
24
+ lines.push(`已运行: ${elapsedMin} 分钟`);
25
+ }
26
+ if (reason === 'project-root' && blockingRun.project_alias && blockingRun.project_alias !== projectAlias) {
27
+ lines.push(`占用项目: ${blockingRun.project_alias}`);
28
+ }
29
+ }
30
+ lines.push(`排队时间: ${new Date().toLocaleTimeString('zh-CN', { hour: '2-digit', minute: '2-digit' })}`);
31
+ return lines.filter((line) => line !== null).join('\n');
32
+ }
33
+ export function buildRunStatusSummary(lastResponseExcerpt, activeRun) {
34
+ if (activeRun?.status === 'queued' && activeRun.status_detail) {
35
+ return [activeRun.status_detail, lastResponseExcerpt ? `\n上一轮摘要:\n${lastResponseExcerpt}` : null].filter(Boolean).join('\n');
36
+ }
37
+ return lastResponseExcerpt ?? '暂无会话摘要。';
38
+ }
39
+ // ---------------------------------------------------------------------------
40
+ // Queue preparation (module-private)
41
+ // ---------------------------------------------------------------------------
42
+ async function prepareQueuedExecution(host, projectContext, metadata, runId) {
43
+ const queuePending = host.queue.getPendingCount(projectContext.queueKey);
44
+ const rootKey = buildProjectRootQueueKey(projectContext.project.root);
45
+ const rootPending = host.projectRootQueue.getPendingCount(rootKey);
46
+ if (queuePending <= 0 && rootPending <= 0) {
47
+ return null;
48
+ }
49
+ const projectRoot = path.resolve(projectContext.project.root);
50
+ const reason = queuePending > 0 ? 'project' : 'project-root';
51
+ const frontCount = reason === 'project' ? queuePending : rootPending;
52
+ const blockingRun = reason === 'project'
53
+ ? await host.runStateStore.getActiveRun(projectContext.queueKey)
54
+ : await host.runStateStore.getExecutionRunByProjectRoot(projectRoot);
55
+ const detail = buildQueuedStatusDetail(projectContext.projectAlias, reason, frontCount, blockingRun);
56
+ await host.runStateStore.upsertRun(runId, {
57
+ queue_key: projectContext.queueKey,
58
+ conversation_key: projectContext.sessionKey,
59
+ project_alias: projectContext.projectAlias,
60
+ chat_id: metadata.chatId,
61
+ actor_id: metadata.actorId,
62
+ actor_name: metadata.actorName,
63
+ project_root: projectRoot,
64
+ prompt_excerpt: truncateExcerpt(metadata.prompt),
65
+ status: 'queued',
66
+ status_detail: detail,
67
+ });
68
+ await host.auditLog.append({
69
+ type: 'codex.run.queued',
70
+ run_id: runId,
71
+ chat_id: metadata.chatId,
72
+ actor_id: metadata.actorId,
73
+ project_alias: projectContext.projectAlias,
74
+ conversation_key: projectContext.sessionKey,
75
+ project_root: projectRoot,
76
+ queue_reason: reason,
77
+ blocking_run_id: blockingRun?.run_id,
78
+ front_count: frontCount,
79
+ });
80
+ host.logger.warn({
81
+ runId,
82
+ queueKey: projectContext.queueKey,
83
+ sessionKey: projectContext.sessionKey,
84
+ projectAlias: projectContext.projectAlias,
85
+ projectRoot,
86
+ reason,
87
+ frontCount,
88
+ blockingStatus: blockingRun?.status,
89
+ blockingProjectAlias: blockingRun?.project_alias,
90
+ }, 'Codex run queued');
91
+ return {
92
+ runId,
93
+ detail,
94
+ reason,
95
+ };
96
+ }
97
+ // ---------------------------------------------------------------------------
98
+ // Main entry: schedule a project execution on the dual-queue system
99
+ // ---------------------------------------------------------------------------
100
+ export async function scheduleProjectExecution(host, projectContext, metadata, task) {
101
+ const runId = randomUUID();
102
+ const queued = await prepareQueuedExecution(host, projectContext, metadata, runId);
103
+ const rootKey = buildProjectRootQueueKey(projectContext.project.root);
104
+ const startGate = createDeferred();
105
+ // Record queue depth when enqueuing
106
+ host.metrics?.recordQueueDepth(projectContext.projectAlias, host.queue.getPendingCount(projectContext.queueKey) + 1);
107
+ return {
108
+ runId,
109
+ queued,
110
+ release: () => startGate.resolve(),
111
+ completion: host.queue.run(projectContext.queueKey, async () => {
112
+ await host.projectRootQueue.run(rootKey, async () => {
113
+ await startGate.promise;
114
+ await task(runId);
115
+ }, { priority: projectContext.project.run_priority });
116
+ // Record queue depth after dequeue
117
+ host.metrics?.recordQueueDepth(projectContext.projectAlias, host.queue.getPendingCount(projectContext.queueKey));
118
+ }),
119
+ };
120
+ }
121
+ //# sourceMappingURL=run-scheduler.js.map