larkcc 0.1.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/CHANGELOG.md +24 -0
- package/LICENSE +21 -0
- package/README.md +617 -0
- package/dist/agent.d.ts +9 -0
- package/dist/agent.js +177 -0
- package/dist/agent.js.map +1 -0
- package/dist/app.d.ts +14 -0
- package/dist/app.js +570 -0
- package/dist/app.js.map +1 -0
- package/dist/commands.d.ts +8 -0
- package/dist/commands.js +143 -0
- package/dist/commands.js.map +1 -0
- package/dist/config.d.ts +61 -0
- package/dist/config.js +184 -0
- package/dist/config.js.map +1 -0
- package/dist/feishu.d.ts +52 -0
- package/dist/feishu.js +964 -0
- package/dist/feishu.js.map +1 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +213 -0
- package/dist/index.js.map +1 -0
- package/dist/logger.d.ts +10 -0
- package/dist/logger.js +13 -0
- package/dist/logger.js.map +1 -0
- package/dist/multifile.d.ts +48 -0
- package/dist/multifile.js +107 -0
- package/dist/multifile.js.map +1 -0
- package/dist/oauth.d.ts +19 -0
- package/dist/oauth.js +142 -0
- package/dist/oauth.js.map +1 -0
- package/dist/session.d.ts +6 -0
- package/dist/session.js +59 -0
- package/dist/session.js.map +1 -0
- package/dist/setup.d.ts +2 -0
- package/dist/setup.js +45 -0
- package/dist/setup.js.map +1 -0
- package/package.json +58 -0
package/dist/app.js
ADDED
|
@@ -0,0 +1,570 @@
|
|
|
1
|
+
import * as lark from "@larksuiteoapi/node-sdk";
|
|
2
|
+
import fs from "fs";
|
|
3
|
+
import path from "path";
|
|
4
|
+
import os from "os";
|
|
5
|
+
import readline from "readline";
|
|
6
|
+
import { execSync } from "child_process";
|
|
7
|
+
import { createLarkClient, createWSClient, sendText, downloadImage, downloadFile } from "./feishu.js";
|
|
8
|
+
import { runAgent, ensureEnv } from "./agent.js";
|
|
9
|
+
import { saveOwnerOpenId } from "./config.js";
|
|
10
|
+
import { parseCommand } from "./commands.js";
|
|
11
|
+
import { getSession, setSession, getChatId, saveChatId } from "./session.js";
|
|
12
|
+
import { logger } from "./logger.js";
|
|
13
|
+
import * as multifile from "./multifile.js";
|
|
14
|
+
const CLAUDE_SETTINGS_PATH = path.join(os.homedir(), ".claude", "settings.json");
|
|
15
|
+
const LOCK_DIR = path.join(os.homedir(), ".larkcc");
|
|
16
|
+
function lockPath(profile) {
|
|
17
|
+
return path.join(LOCK_DIR, profile ? `lock-${profile}.json` : "lock-default.json");
|
|
18
|
+
}
|
|
19
|
+
function readLock(profile) {
|
|
20
|
+
try {
|
|
21
|
+
const p = lockPath(profile);
|
|
22
|
+
if (!fs.existsSync(p))
|
|
23
|
+
return null;
|
|
24
|
+
return JSON.parse(fs.readFileSync(p, "utf8"));
|
|
25
|
+
}
|
|
26
|
+
catch {
|
|
27
|
+
return null;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
function writeLock(cwd, profile, isContinue) {
|
|
31
|
+
if (!fs.existsSync(LOCK_DIR))
|
|
32
|
+
fs.mkdirSync(LOCK_DIR, { recursive: true });
|
|
33
|
+
fs.writeFileSync(lockPath(profile), JSON.stringify({ pid: process.pid, cwd, startedAt: new Date().toISOString(), continue: isContinue }, null, 2), "utf8");
|
|
34
|
+
}
|
|
35
|
+
function clearLock(profile) {
|
|
36
|
+
try {
|
|
37
|
+
fs.unlinkSync(lockPath(profile));
|
|
38
|
+
}
|
|
39
|
+
catch { }
|
|
40
|
+
}
|
|
41
|
+
function isProcessAlive(pid) {
|
|
42
|
+
try {
|
|
43
|
+
process.kill(pid, 0);
|
|
44
|
+
return true;
|
|
45
|
+
}
|
|
46
|
+
catch {
|
|
47
|
+
return false;
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
async function checkLock(cwd, profile) {
|
|
51
|
+
const lock = readLock(profile);
|
|
52
|
+
if (!lock)
|
|
53
|
+
return;
|
|
54
|
+
if (!isProcessAlive(lock.pid)) {
|
|
55
|
+
clearLock(profile);
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
if (lock.cwd === cwd && lock.pid === process.pid)
|
|
59
|
+
return;
|
|
60
|
+
const profileLabel = profile ? `[${profile}] ` : "";
|
|
61
|
+
logger.warn(`${profileLabel}Already running!`);
|
|
62
|
+
logger.warn(` PID: ${lock.pid}`);
|
|
63
|
+
logger.warn(` Project: ${lock.cwd}`);
|
|
64
|
+
logger.warn(` Started: ${lock.startedAt}`);
|
|
65
|
+
console.log("");
|
|
66
|
+
const answer = await new Promise((resolve) => {
|
|
67
|
+
const rl = readline.createInterface({ input: process.stdin, output: process.stdout });
|
|
68
|
+
rl.question(" Continue anyway? (y/n): ", (a) => { rl.close(); resolve(a.trim().toLowerCase()); });
|
|
69
|
+
});
|
|
70
|
+
if (answer !== "y") {
|
|
71
|
+
logger.info("Aborted.");
|
|
72
|
+
process.exit(0);
|
|
73
|
+
}
|
|
74
|
+
clearLock(profile);
|
|
75
|
+
}
|
|
76
|
+
export function listRunningProcesses() {
|
|
77
|
+
const processes = [];
|
|
78
|
+
const cleaned = [];
|
|
79
|
+
if (!fs.existsSync(LOCK_DIR))
|
|
80
|
+
return { processes, cleaned };
|
|
81
|
+
const files = fs.readdirSync(LOCK_DIR).filter(f => f.startsWith("lock-") && f.endsWith(".json"));
|
|
82
|
+
for (const file of files) {
|
|
83
|
+
const match = file.match(/^lock-(.+)\.json$/);
|
|
84
|
+
if (!match)
|
|
85
|
+
continue;
|
|
86
|
+
const profile = match[1];
|
|
87
|
+
const lock = readLock(profile);
|
|
88
|
+
if (!lock)
|
|
89
|
+
continue;
|
|
90
|
+
const alive = isProcessAlive(lock.pid);
|
|
91
|
+
if (!alive) {
|
|
92
|
+
clearLock(profile);
|
|
93
|
+
cleaned.push(profile);
|
|
94
|
+
continue;
|
|
95
|
+
}
|
|
96
|
+
processes.push({
|
|
97
|
+
profile,
|
|
98
|
+
pid: lock.pid,
|
|
99
|
+
cwd: lock.cwd,
|
|
100
|
+
startedAt: lock.startedAt,
|
|
101
|
+
isContinue: lock.continue ?? false,
|
|
102
|
+
alive,
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
// 按启动时间排序
|
|
106
|
+
processes.sort((a, b) => new Date(a.startedAt).getTime() - new Date(b.startedAt).getTime());
|
|
107
|
+
return { processes, cleaned };
|
|
108
|
+
}
|
|
109
|
+
function injectClaudeEnv() {
|
|
110
|
+
try {
|
|
111
|
+
if (!fs.existsSync(CLAUDE_SETTINGS_PATH))
|
|
112
|
+
return;
|
|
113
|
+
const settings = JSON.parse(fs.readFileSync(CLAUDE_SETTINGS_PATH, "utf8"));
|
|
114
|
+
const env = settings?.env ?? {};
|
|
115
|
+
for (const [key, value] of Object.entries(env)) {
|
|
116
|
+
if (!process.env[key])
|
|
117
|
+
process.env[key] = String(value);
|
|
118
|
+
}
|
|
119
|
+
logger.dim(`injected ${Object.keys(env).length} env vars from ~/.claude/settings.json`);
|
|
120
|
+
}
|
|
121
|
+
catch (err) {
|
|
122
|
+
logger.warn(`Failed to read ~/.claude/settings.json: ${String(err)}`);
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
function ensureClaudeOnboarding() {
|
|
126
|
+
const claudeJsonPath = path.join(os.homedir(), ".claude.json");
|
|
127
|
+
try {
|
|
128
|
+
let json = {};
|
|
129
|
+
if (fs.existsSync(claudeJsonPath))
|
|
130
|
+
json = JSON.parse(fs.readFileSync(claudeJsonPath, "utf8"));
|
|
131
|
+
if (!json.hasCompletedOnboarding) {
|
|
132
|
+
json.hasCompletedOnboarding = true;
|
|
133
|
+
fs.writeFileSync(claudeJsonPath, JSON.stringify(json, null, 2), "utf8");
|
|
134
|
+
logger.dim("wrote hasCompletedOnboarding to ~/.claude.json");
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
catch (err) {
|
|
138
|
+
logger.warn(`Failed to write ~/.claude.json: ${String(err)}`);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
function ensureClaudeInPath() {
|
|
142
|
+
const commonPaths = [
|
|
143
|
+
"/usr/local/bin", "/usr/bin",
|
|
144
|
+
`${os.homedir()}/.npm-global/bin`, `${os.homedir()}/.local/bin`,
|
|
145
|
+
"/opt/homebrew/bin", "/home/linuxbrew/.linuxbrew/bin",
|
|
146
|
+
];
|
|
147
|
+
try {
|
|
148
|
+
const claudePath = execSync("which claude 2>/dev/null || command -v claude 2>/dev/null", {
|
|
149
|
+
shell: "/bin/bash",
|
|
150
|
+
env: { ...process.env, PATH: [...commonPaths, process.env.PATH ?? ""].join(":") },
|
|
151
|
+
}).toString().trim();
|
|
152
|
+
if (claudePath) {
|
|
153
|
+
const dir = path.dirname(claudePath);
|
|
154
|
+
if (!process.env.PATH?.includes(dir))
|
|
155
|
+
process.env.PATH = `${dir}:${process.env.PATH}`;
|
|
156
|
+
logger.dim(`claude found: ${claudePath}`);
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
catch { }
|
|
161
|
+
for (const dir of commonPaths) {
|
|
162
|
+
if (fs.existsSync(path.join(dir, "claude"))) {
|
|
163
|
+
if (!process.env.PATH?.includes(dir))
|
|
164
|
+
process.env.PATH = `${dir}:${process.env.PATH}`;
|
|
165
|
+
logger.dim(`claude found: ${dir}/claude`);
|
|
166
|
+
return;
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
logger.warn("claude CLI not found — make sure it's installed: npm install -g @anthropic-ai/claude-code");
|
|
170
|
+
}
|
|
171
|
+
// 去掉消息里的 @ 提及文字,只保留实际内容
|
|
172
|
+
function stripMentions(text) {
|
|
173
|
+
return text.replace(/@\S+/g, "").trim();
|
|
174
|
+
}
|
|
175
|
+
// ── 主逻辑 ────────────────────────────────────────────────────
|
|
176
|
+
export async function startApp(cwd, config, profile, continueSession = false) {
|
|
177
|
+
const { app_id, app_secret } = config.feishu;
|
|
178
|
+
const getOwnerOpenId = () => config.feishu.owner_open_id;
|
|
179
|
+
const customCommands = config.commands ?? {};
|
|
180
|
+
await checkLock(cwd, profile);
|
|
181
|
+
writeLock(cwd, profile, continueSession);
|
|
182
|
+
injectClaudeEnv();
|
|
183
|
+
ensureClaudeOnboarding();
|
|
184
|
+
ensureEnv();
|
|
185
|
+
ensureClaudeInPath();
|
|
186
|
+
const client = createLarkClient(app_id, app_secret);
|
|
187
|
+
const wsClient = createWSClient(app_id, app_secret);
|
|
188
|
+
// 获取机器人自己的 open_id,用于群消息 @ 识别
|
|
189
|
+
let botOpenId = "";
|
|
190
|
+
try {
|
|
191
|
+
const botInfo = await client.bot.getBotInfo({});
|
|
192
|
+
botOpenId = botInfo.data?.open_id ?? "";
|
|
193
|
+
if (botOpenId)
|
|
194
|
+
logger.dim(`bot open_id: ${botOpenId}`);
|
|
195
|
+
}
|
|
196
|
+
catch { }
|
|
197
|
+
let processing = false;
|
|
198
|
+
let processingStartedAt = 0;
|
|
199
|
+
let currentAbortController = null;
|
|
200
|
+
let knownChatId = getChatId();
|
|
201
|
+
const startupTime = Date.now();
|
|
202
|
+
const recentMessages = new Map();
|
|
203
|
+
const PROCESSING_TIMEOUT_MS = 10 * 60 * 1000;
|
|
204
|
+
if (continueSession) {
|
|
205
|
+
const savedSession = getSession(true);
|
|
206
|
+
if (savedSession) {
|
|
207
|
+
setSession(savedSession);
|
|
208
|
+
logger.info(`Resuming session: ${savedSession}`);
|
|
209
|
+
}
|
|
210
|
+
else
|
|
211
|
+
logger.warn("No saved session found, starting fresh");
|
|
212
|
+
}
|
|
213
|
+
const profileLabel = profile ? ` [${profile}]` : "";
|
|
214
|
+
logger.info(`Project: ${cwd}`);
|
|
215
|
+
logger.info(`Profile: ${profile ?? "default"}`);
|
|
216
|
+
logger.info(`AppID: ${app_id}`);
|
|
217
|
+
logger.info(`Owner: ${getOwnerOpenId() || "(pending first message)"}`);
|
|
218
|
+
logger.info(`Session: ${continueSession ? "continue" : "new"}`);
|
|
219
|
+
logger.info("Connecting to Feishu...");
|
|
220
|
+
wsClient.start({
|
|
221
|
+
eventDispatcher: new lark.EventDispatcher({}).register({
|
|
222
|
+
"im.message.receive_v1": async (data) => {
|
|
223
|
+
const msg = data.message;
|
|
224
|
+
const senderId = data.sender.sender_id?.open_id ?? "";
|
|
225
|
+
const isGroup = msg.chat_type === "group";
|
|
226
|
+
if (!["text", "post", "image", "file"].includes(msg.message_type))
|
|
227
|
+
return;
|
|
228
|
+
const msgTimestamp = Number(msg.create_time);
|
|
229
|
+
const now = Date.now();
|
|
230
|
+
if (msgTimestamp < startupTime) {
|
|
231
|
+
logger.dim(`skipped pre-startup message (${Math.round((now - msgTimestamp) / 1000)}s ago)`);
|
|
232
|
+
return;
|
|
233
|
+
}
|
|
234
|
+
// 群消息:只响应 @ 自己 或 引用了消息(不管引用谁)
|
|
235
|
+
if (isGroup) {
|
|
236
|
+
const mentions = msg.mentions ?? [];
|
|
237
|
+
const atBot = botOpenId
|
|
238
|
+
? mentions.some(m => m.id?.open_id === botOpenId)
|
|
239
|
+
: mentions.length > 0; // botOpenId 未知时,有 @ 就响应
|
|
240
|
+
const hasQuote = !!msg.parent_id;
|
|
241
|
+
if (!atBot && !hasQuote)
|
|
242
|
+
return;
|
|
243
|
+
}
|
|
244
|
+
const dedupeKey = `${senderId}:${msg.message_id}`;
|
|
245
|
+
const lastSeen = recentMessages.get(dedupeKey);
|
|
246
|
+
if (lastSeen && now - lastSeen < 30_000)
|
|
247
|
+
return;
|
|
248
|
+
recentMessages.set(dedupeKey, now);
|
|
249
|
+
for (const [k, t] of recentMessages) {
|
|
250
|
+
if (now - t > 30_000)
|
|
251
|
+
recentMessages.delete(k);
|
|
252
|
+
}
|
|
253
|
+
const owner_open_id = getOwnerOpenId();
|
|
254
|
+
if (!owner_open_id) {
|
|
255
|
+
logger.success(`Auto-detected open_id: ${senderId}`);
|
|
256
|
+
saveOwnerOpenId(senderId, profile);
|
|
257
|
+
config.feishu.owner_open_id = senderId;
|
|
258
|
+
}
|
|
259
|
+
else if (senderId !== owner_open_id) {
|
|
260
|
+
logger.warn(`Ignored message from unknown user: ${senderId}`);
|
|
261
|
+
return;
|
|
262
|
+
}
|
|
263
|
+
const chatId = msg.chat_id;
|
|
264
|
+
if (!knownChatId || knownChatId !== chatId) {
|
|
265
|
+
knownChatId = chatId;
|
|
266
|
+
saveChatId(chatId);
|
|
267
|
+
logger.dim(`chat_id saved: ${chatId} (${isGroup ? "group" : "p2p"})`);
|
|
268
|
+
}
|
|
269
|
+
// ── 解析消息内容 ──────────────────────────────────────
|
|
270
|
+
const stripHtml = (s) => s.replace(/<[^>]*>/g, "").trim();
|
|
271
|
+
let text = "";
|
|
272
|
+
let images = [];
|
|
273
|
+
let downloadedFile = null;
|
|
274
|
+
// 文件处理配置(提前定义,供多处使用)
|
|
275
|
+
const fileConfig = config.file;
|
|
276
|
+
const profileKey = profile ?? "default";
|
|
277
|
+
if (msg.message_type === "text") {
|
|
278
|
+
const raw = stripHtml(JSON.parse(msg.content).text ?? "");
|
|
279
|
+
text = isGroup ? stripMentions(raw) : raw;
|
|
280
|
+
}
|
|
281
|
+
else if (msg.message_type === "post") {
|
|
282
|
+
const raw = JSON.parse(msg.content);
|
|
283
|
+
const post = raw.zh_cn ?? raw;
|
|
284
|
+
const title = stripHtml(post.title ?? "");
|
|
285
|
+
const blocks = post.content ?? [];
|
|
286
|
+
// 提取文字内容
|
|
287
|
+
const lines = blocks.map(line => line.map(el => stripHtml(el.text ?? "")).join("").trim()).filter(Boolean);
|
|
288
|
+
const body = lines.join("\n").trim();
|
|
289
|
+
text = isGroup ? stripMentions([title, body].filter(Boolean).join("\n").trim())
|
|
290
|
+
: [title, body].filter(Boolean).join("\n").trim();
|
|
291
|
+
// 提取富文本中的图片
|
|
292
|
+
for (const line of blocks) {
|
|
293
|
+
for (const el of line) {
|
|
294
|
+
if (el.tag === "img" && el.image_key) {
|
|
295
|
+
logger.dim(`downloading image from post: ${el.image_key}`);
|
|
296
|
+
const img = await downloadImage(client, msg.message_id, el.image_key);
|
|
297
|
+
if (img) {
|
|
298
|
+
images.push(img);
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
else if (msg.message_type === "image") {
|
|
305
|
+
const imageKey = JSON.parse(msg.content).image_key;
|
|
306
|
+
if (imageKey) {
|
|
307
|
+
logger.dim(`downloading image: ${imageKey}`);
|
|
308
|
+
const img = await downloadImage(client, msg.message_id, imageKey);
|
|
309
|
+
if (img) {
|
|
310
|
+
images = [img];
|
|
311
|
+
text = config.image_prompt ?? "分析图片,给出回应";
|
|
312
|
+
}
|
|
313
|
+
else {
|
|
314
|
+
await sendText(client, chatId, "❌ 图片下载失败");
|
|
315
|
+
return;
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
else if (msg.message_type === "file") {
|
|
320
|
+
// 文件消息处理
|
|
321
|
+
if (!fileConfig?.enabled) {
|
|
322
|
+
await sendText(client, chatId, "❌ 文件处理功能未启用");
|
|
323
|
+
return;
|
|
324
|
+
}
|
|
325
|
+
const fileContent = JSON.parse(msg.content);
|
|
326
|
+
const fileKey = fileContent.file_key;
|
|
327
|
+
const fileName = fileContent.file_name ?? "unknown";
|
|
328
|
+
const fileSize = fileContent.file_size ?? 0;
|
|
329
|
+
if (!fileKey) {
|
|
330
|
+
await sendText(client, chatId, "❌ 无法获取文件信息");
|
|
331
|
+
return;
|
|
332
|
+
}
|
|
333
|
+
// 检查文件大小
|
|
334
|
+
const sizeLimit = fileConfig.size_limit ?? 30 * 1024 * 1024;
|
|
335
|
+
if (fileSize > sizeLimit) {
|
|
336
|
+
const limitMB = Math.round(sizeLimit / 1024 / 1024);
|
|
337
|
+
const sizeMB = Math.round(fileSize / 1024 / 1024);
|
|
338
|
+
await sendText(client, chatId, `❌ 文件太大(${sizeMB}MB),超过限制(${limitMB}MB)`);
|
|
339
|
+
return;
|
|
340
|
+
}
|
|
341
|
+
// 检查是否在多文件模式
|
|
342
|
+
if (multifile.isActive(profileKey, chatId)) {
|
|
343
|
+
// 多文件模式:下载并缓存
|
|
344
|
+
logger.dim(`[multifile] downloading file: ${fileName}`);
|
|
345
|
+
const tempDir = fileConfig.temp_dir ?? path.join(os.homedir(), ".larkcc", "temp", profileKey);
|
|
346
|
+
const file = await downloadFile(client, msg.message_id, fileKey, tempDir, fileName);
|
|
347
|
+
if (file) {
|
|
348
|
+
multifile.addItem(profileKey, chatId, { type: "file", content: file, timestamp: Date.now() });
|
|
349
|
+
const count = multifile.getItemCount(profileKey, chatId);
|
|
350
|
+
// 添加 reaction 确认
|
|
351
|
+
await client.im.messageReaction.create({
|
|
352
|
+
path: { message_id: msg.message_id },
|
|
353
|
+
data: { reaction_type: { emoji_type: "OK" } },
|
|
354
|
+
}).catch(() => { });
|
|
355
|
+
logger.dim(`[multifile] cached file ${count}: ${fileName}`);
|
|
356
|
+
}
|
|
357
|
+
else {
|
|
358
|
+
await sendText(client, chatId, `❌ 文件下载失败:${fileName}`);
|
|
359
|
+
}
|
|
360
|
+
return; // 不继续处理,等待 /mf done
|
|
361
|
+
}
|
|
362
|
+
// 单文件模式:下载并处理
|
|
363
|
+
logger.dim(`downloading file: ${fileName}`);
|
|
364
|
+
const tempDir = fileConfig.temp_dir ?? path.join(os.homedir(), ".larkcc", "temp", profileKey);
|
|
365
|
+
downloadedFile = await downloadFile(client, msg.message_id, fileKey, tempDir, fileName);
|
|
366
|
+
if (!downloadedFile) {
|
|
367
|
+
await sendText(client, chatId, "❌ 文件下载失败,请重试");
|
|
368
|
+
return;
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
if (!text && images.length === 0 && !downloadedFile)
|
|
372
|
+
return;
|
|
373
|
+
// ── 多文件模式处理 ────────────────────────────────────────
|
|
374
|
+
// 检查多文件模式超时
|
|
375
|
+
if (fileConfig && multifile.isActive(profileKey, chatId)) {
|
|
376
|
+
const timeout = fileConfig.multifile_timeout ?? 300;
|
|
377
|
+
const timeoutItems = multifile.checkTimeout(profileKey, chatId, timeout);
|
|
378
|
+
if (timeoutItems && timeoutItems.length > 0) {
|
|
379
|
+
logger.dim(`[multifile] timeout, auto-processing ${timeoutItems.length} items`);
|
|
380
|
+
// 超时自动处理 - 这里暂时只通知用户,不自动处理
|
|
381
|
+
await sendText(client, chatId, `⏰ 多文件模式已超时,已缓存 ${timeoutItems.length} 个项目。发送 /mf done 开始处理,或 /mf start 重新开始。`);
|
|
382
|
+
return;
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
// 多文件模式下,文字消息作为说明缓存
|
|
386
|
+
if (text && !text.startsWith("/") && multifile.isActive(profileKey, chatId)) {
|
|
387
|
+
multifile.addItem(profileKey, chatId, { type: "text", content: text, timestamp: Date.now() });
|
|
388
|
+
const count = multifile.getItemCount(profileKey, chatId);
|
|
389
|
+
await client.im.messageReaction.create({
|
|
390
|
+
path: { message_id: msg.message_id },
|
|
391
|
+
data: { reaction_type: { emoji_type: "OK" } },
|
|
392
|
+
}).catch(() => { });
|
|
393
|
+
logger.dim(`[multifile] cached text ${count}: ${text.slice(0, 50)}...`);
|
|
394
|
+
return;
|
|
395
|
+
}
|
|
396
|
+
// ── Slash 命令拦截 ────────────────────────────────────
|
|
397
|
+
if (text.startsWith("/")) {
|
|
398
|
+
const cmdText = text.toLowerCase().trim();
|
|
399
|
+
if (cmdText === "/stop" || cmdText === "/cancel") {
|
|
400
|
+
if (processing && currentAbortController) {
|
|
401
|
+
currentAbortController.abort();
|
|
402
|
+
await sendText(client, chatId, "⏹ 已发送中断信号,等待当前步骤完成...");
|
|
403
|
+
}
|
|
404
|
+
else if (processing) {
|
|
405
|
+
processing = false;
|
|
406
|
+
processingStartedAt = 0;
|
|
407
|
+
await sendText(client, chatId, "⏹ 已强制释放,可以发新消息了");
|
|
408
|
+
}
|
|
409
|
+
else {
|
|
410
|
+
await sendText(client, chatId, "没有正在处理的任务");
|
|
411
|
+
}
|
|
412
|
+
return;
|
|
413
|
+
}
|
|
414
|
+
const result = parseCommand(text, cwd, customCommands);
|
|
415
|
+
if (result) {
|
|
416
|
+
// 多文件模式命令处理
|
|
417
|
+
if (result.type === "multifile_start") {
|
|
418
|
+
const wasActive = multifile.isActive(profileKey, chatId);
|
|
419
|
+
if (wasActive) {
|
|
420
|
+
multifile.resetMode(profileKey, chatId);
|
|
421
|
+
await sendText(client, chatId, "📁 多文件模式已重置,之前的缓存已清空,请重新发送文件");
|
|
422
|
+
}
|
|
423
|
+
else {
|
|
424
|
+
multifile.startMode(profileKey, chatId);
|
|
425
|
+
await sendText(client, chatId, "📁 多文件模式已开始,请发送文件和说明文字,完成后发送 /mf done");
|
|
426
|
+
}
|
|
427
|
+
return;
|
|
428
|
+
}
|
|
429
|
+
if (result.type === "multifile_done") {
|
|
430
|
+
if (!multifile.isActive(profileKey, chatId)) {
|
|
431
|
+
await sendText(client, chatId, "❌ 未在多文件模式中,请先发送 /mf start");
|
|
432
|
+
return;
|
|
433
|
+
}
|
|
434
|
+
const items = multifile.endMode(profileKey, chatId);
|
|
435
|
+
if (items.length === 0) {
|
|
436
|
+
await sendText(client, chatId, "❌ 没有缓存的内容,请先发送文件");
|
|
437
|
+
return;
|
|
438
|
+
}
|
|
439
|
+
// 构建多文件 prompt
|
|
440
|
+
const fileItems = items.filter(i => i.type === "file");
|
|
441
|
+
const textItems = items.filter(i => i.type === "text");
|
|
442
|
+
if (fileItems.length === 0) {
|
|
443
|
+
await sendText(client, chatId, "❌ 没有缓存文件,请先发送文件");
|
|
444
|
+
return;
|
|
445
|
+
}
|
|
446
|
+
// 格式化文件列表
|
|
447
|
+
const filesList = fileItems.map((item, idx) => {
|
|
448
|
+
const file = item.content;
|
|
449
|
+
return `${idx + 1}. ${file.filename}(${Math.round(file.size / 1024)}KB,${file.mime_type})\n 路径:${file.filepath}`;
|
|
450
|
+
}).join("\n");
|
|
451
|
+
// 格式化文字说明
|
|
452
|
+
const textContent = textItems.map(i => i.content).join("\n");
|
|
453
|
+
// 使用配置的 multifile_prompt
|
|
454
|
+
const mfPrompt = fileConfig?.multifile_prompt ?? "分析以下 {count} 个文件:\n{files}\n\n用户说明:{text}";
|
|
455
|
+
text = mfPrompt
|
|
456
|
+
.replace("{count}", String(fileItems.length))
|
|
457
|
+
.replace("{files}", filesList)
|
|
458
|
+
.replace("{text}", textContent || "(无文字说明)");
|
|
459
|
+
logger.dim(`[multifile] processing ${fileItems.length} files, ${textItems.length} text items`);
|
|
460
|
+
// 继续执行,进入正常的处理流程
|
|
461
|
+
}
|
|
462
|
+
if (result.type === "exec" || result.type === "help" || result.type === "unknown") {
|
|
463
|
+
await sendText(client, chatId, result.output ?? "");
|
|
464
|
+
return;
|
|
465
|
+
}
|
|
466
|
+
if (result.type === "prompt" && result.prompt) {
|
|
467
|
+
text = result.prompt;
|
|
468
|
+
}
|
|
469
|
+
}
|
|
470
|
+
}
|
|
471
|
+
logger.msg(senderId, `[${isGroup ? "group" : "p2p"}] ${text || "[image]"}`);
|
|
472
|
+
// 超时自动释放
|
|
473
|
+
if (processing && Date.now() - processingStartedAt > PROCESSING_TIMEOUT_MS) {
|
|
474
|
+
logger.warn("Processing timeout, force releasing lock...");
|
|
475
|
+
processing = false;
|
|
476
|
+
processingStartedAt = 0;
|
|
477
|
+
currentAbortController = null;
|
|
478
|
+
}
|
|
479
|
+
if (processing) {
|
|
480
|
+
const elapsed = Math.round((Date.now() - processingStartedAt) / 1000);
|
|
481
|
+
logger.warn("Still processing previous message, skipping...");
|
|
482
|
+
await sendText(client, chatId, `⏳ 上一条消息还在处理中(已${elapsed}秒),发送 /stop 可强制中断`);
|
|
483
|
+
return;
|
|
484
|
+
}
|
|
485
|
+
processing = true;
|
|
486
|
+
processingStartedAt = Date.now();
|
|
487
|
+
currentAbortController = new AbortController();
|
|
488
|
+
let reactionId;
|
|
489
|
+
try {
|
|
490
|
+
const reactionRes = await client.im.messageReaction.create({
|
|
491
|
+
path: { message_id: msg.message_id },
|
|
492
|
+
data: { reaction_type: { emoji_type: "OK" } },
|
|
493
|
+
});
|
|
494
|
+
reactionId = reactionRes.data?.reaction_id;
|
|
495
|
+
}
|
|
496
|
+
catch { }
|
|
497
|
+
try {
|
|
498
|
+
// 处理单文件:修改 prompt 包含文件信息
|
|
499
|
+
let finalPrompt = text;
|
|
500
|
+
if (downloadedFile && !multifile.isActive(profileKey, chatId)) {
|
|
501
|
+
const fp = fileConfig?.prompt ?? "分析文件 {filename}(路径:{filepath},大小:{size},类型:{mime_type})";
|
|
502
|
+
const sizeStr = `${Math.round(downloadedFile.size / 1024)}KB`;
|
|
503
|
+
finalPrompt = fp
|
|
504
|
+
.replace("{filename}", downloadedFile.filename)
|
|
505
|
+
.replace("{filepath}", downloadedFile.filepath)
|
|
506
|
+
.replace("{size}", sizeStr)
|
|
507
|
+
.replace("{mime_type}", downloadedFile.mime_type)
|
|
508
|
+
.replace("{file_key}", downloadedFile.file_key);
|
|
509
|
+
// 如果用户也发送了文字,附加到 prompt 后面
|
|
510
|
+
if (text && !text.startsWith("/")) {
|
|
511
|
+
finalPrompt = `${finalPrompt}\n\n用户说明:${text}`;
|
|
512
|
+
}
|
|
513
|
+
}
|
|
514
|
+
await runAgent(finalPrompt, cwd, config, client, chatId, msg.message_id, images.length > 0 ? images : undefined, currentAbortController?.signal, profile);
|
|
515
|
+
if (reactionId) {
|
|
516
|
+
await client.im.messageReaction.delete({
|
|
517
|
+
path: { message_id: msg.message_id, reaction_id: reactionId },
|
|
518
|
+
}).catch(() => { });
|
|
519
|
+
}
|
|
520
|
+
await client.im.messageReaction.create({
|
|
521
|
+
path: { message_id: msg.message_id },
|
|
522
|
+
data: { reaction_type: { emoji_type: "DONE" } },
|
|
523
|
+
}).catch(() => { });
|
|
524
|
+
}
|
|
525
|
+
catch (err) {
|
|
526
|
+
logger.error(`Agent error: ${String(err)}`);
|
|
527
|
+
await sendText(client, chatId, `❌ 出错了:${String(err)}`);
|
|
528
|
+
if (reactionId) {
|
|
529
|
+
await client.im.messageReaction.delete({
|
|
530
|
+
path: { message_id: msg.message_id, reaction_id: reactionId },
|
|
531
|
+
}).catch(() => { });
|
|
532
|
+
}
|
|
533
|
+
await client.im.messageReaction.create({
|
|
534
|
+
path: { message_id: msg.message_id },
|
|
535
|
+
data: { reaction_type: { emoji_type: "OnIt" } },
|
|
536
|
+
}).catch(() => { });
|
|
537
|
+
}
|
|
538
|
+
finally {
|
|
539
|
+
processing = false;
|
|
540
|
+
processingStartedAt = 0;
|
|
541
|
+
currentAbortController = null;
|
|
542
|
+
}
|
|
543
|
+
},
|
|
544
|
+
}),
|
|
545
|
+
});
|
|
546
|
+
logger.success("Feishu connected! Waiting for messages...");
|
|
547
|
+
logger.dim("Press Ctrl+C to stop\n");
|
|
548
|
+
if (knownChatId) {
|
|
549
|
+
const sessionNote = continueSession ? "(续接上次对话)" : "(新会话)";
|
|
550
|
+
await sendText(client, knownChatId, `✅ larkcc 已连接${profileLabel} ${sessionNote}\n📁 当前项目:\`${cwd}\``);
|
|
551
|
+
}
|
|
552
|
+
else {
|
|
553
|
+
logger.dim("No chat_id yet — send any message to the bot first");
|
|
554
|
+
}
|
|
555
|
+
const shutdown = async () => {
|
|
556
|
+
console.log("");
|
|
557
|
+
logger.info("Stopping larkcc...");
|
|
558
|
+
clearLock(profile);
|
|
559
|
+
if (knownChatId) {
|
|
560
|
+
try {
|
|
561
|
+
await sendText(client, knownChatId, `👋 larkcc 已断开${profileLabel}\n📁 项目:\`${cwd}\``);
|
|
562
|
+
}
|
|
563
|
+
catch { }
|
|
564
|
+
}
|
|
565
|
+
process.exit(0);
|
|
566
|
+
};
|
|
567
|
+
process.on("SIGINT", shutdown);
|
|
568
|
+
process.on("SIGTERM", shutdown);
|
|
569
|
+
}
|
|
570
|
+
//# sourceMappingURL=app.js.map
|
package/dist/app.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"app.js","sourceRoot":"","sources":["../src/app.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
export declare const BUILTIN_EXEC: Record<string, (cwd: string) => string>;
|
|
2
|
+
export declare const BUILTIN_PROMPTS: Record<string, (args: string, cwd: string) => string>;
|
|
3
|
+
export interface CommandResult {
|
|
4
|
+
type: "exec" | "prompt" | "unknown" | "help" | "multifile_start" | "multifile_done";
|
|
5
|
+
output?: string;
|
|
6
|
+
prompt?: string;
|
|
7
|
+
}
|
|
8
|
+
export declare function parseCommand(text: string, cwd: string, customCommands?: Record<string, string>): CommandResult | null;
|