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.
- package/README.en.md +3 -2
- package/README.md +3 -2
- package/dist/backend/claude.d.ts +2 -0
- package/dist/backend/claude.js +57 -54
- package/dist/backend/claude.js.map +1 -1
- package/dist/backend/codex.d.ts +2 -0
- package/dist/backend/codex.js +27 -0
- package/dist/backend/codex.js.map +1 -1
- package/dist/backend/factory.d.ts +43 -0
- package/dist/backend/factory.js +109 -29
- package/dist/backend/factory.js.map +1 -1
- package/dist/backend/probe.d.ts +27 -0
- package/dist/backend/probe.js +85 -0
- package/dist/backend/probe.js.map +1 -0
- package/dist/backend/qwen.d.ts +59 -0
- package/dist/backend/qwen.js +372 -0
- package/dist/backend/qwen.js.map +1 -0
- package/dist/backend/registry.d.ts +58 -0
- package/dist/backend/registry.js +23 -0
- package/dist/backend/registry.js.map +1 -0
- package/dist/backend/types.d.ts +6 -1
- package/dist/bridge/admin-config.d.ts +47 -0
- package/dist/bridge/admin-config.js +141 -0
- package/dist/bridge/admin-config.js.map +1 -0
- package/dist/bridge/collab-commands.d.ts +42 -0
- package/dist/bridge/collab-commands.js +254 -0
- package/dist/bridge/collab-commands.js.map +1 -0
- package/dist/bridge/commands.d.ts +1 -1
- package/dist/bridge/commands.js +10 -6
- package/dist/bridge/commands.js.map +1 -1
- package/dist/bridge/feishu-commands.d.ts +27 -0
- package/dist/bridge/feishu-commands.js +462 -0
- package/dist/bridge/feishu-commands.js.map +1 -0
- package/dist/bridge/intent-classifier.d.ts +7 -2
- package/dist/bridge/intent-classifier.js +1 -1
- package/dist/bridge/intent-classifier.js.map +1 -1
- package/dist/bridge/lifecycle.d.ts +46 -0
- package/dist/bridge/lifecycle.js +228 -0
- package/dist/bridge/lifecycle.js.map +1 -0
- package/dist/bridge/memory-commands.d.ts +26 -0
- package/dist/bridge/memory-commands.js +330 -0
- package/dist/bridge/memory-commands.js.map +1 -0
- package/dist/bridge/reply-builders.d.ts +30 -0
- package/dist/bridge/reply-builders.js +72 -0
- package/dist/bridge/reply-builders.js.map +1 -0
- package/dist/bridge/run-pipeline.d.ts +86 -0
- package/dist/bridge/run-pipeline.js +453 -0
- package/dist/bridge/run-pipeline.js.map +1 -0
- package/dist/bridge/run-scheduler.d.ts +47 -0
- package/dist/bridge/run-scheduler.js +121 -0
- package/dist/bridge/run-scheduler.js.map +1 -0
- package/dist/bridge/service-utils.d.ts +47 -0
- package/dist/bridge/service-utils.js +309 -0
- package/dist/bridge/service-utils.js.map +1 -0
- package/dist/bridge/service.d.ts +114 -66
- package/dist/bridge/service.js +230 -2199
- package/dist/bridge/service.js.map +1 -1
- package/dist/config/load.js +1 -1
- package/dist/config/load.js.map +1 -1
- package/dist/config/paths.js +1 -20
- package/dist/config/paths.js.map +1 -1
- package/dist/config/schema.d.ts +50 -16
- package/dist/config/schema.js +41 -2
- package/dist/config/schema.js.map +1 -1
- package/dist/feishu/long-connection.js +1 -0
- package/dist/feishu/long-connection.js.map +1 -1
- package/dist/feishu/webhook.js +1 -0
- package/dist/feishu/webhook.js.map +1 -1
- 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
|