@ai-zen/air 0.2.0 → 0.2.2
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.md +21 -6
- package/README.zh.md +31 -6
- package/dist/agent-factory.js +1 -1
- package/dist/agent-factory.js.map +1 -1
- package/dist/agent-runtime.d.ts.map +1 -1
- package/dist/agent-runtime.js +241 -228
- package/dist/agent-runtime.js.map +1 -1
- package/dist/chat/commands/back.d.ts +3 -0
- package/dist/chat/commands/back.d.ts.map +1 -0
- package/dist/chat/commands/back.js +95 -0
- package/dist/chat/commands/back.js.map +1 -0
- package/dist/chat/commands/editor.d.ts +3 -0
- package/dist/chat/commands/editor.d.ts.map +1 -0
- package/dist/chat/commands/editor.js +20 -0
- package/dist/chat/commands/editor.js.map +1 -0
- package/dist/chat/commands/exit.d.ts +3 -0
- package/dist/chat/commands/exit.d.ts.map +1 -0
- package/dist/chat/commands/exit.js +5 -0
- package/dist/chat/commands/exit.js.map +1 -0
- package/dist/chat/commands/help.d.ts +2 -0
- package/dist/chat/commands/help.d.ts.map +1 -0
- package/dist/chat/commands/help.js +4 -0
- package/dist/chat/commands/help.js.map +1 -0
- package/dist/chat/commands/index.d.ts +3 -0
- package/dist/chat/commands/index.d.ts.map +1 -0
- package/dist/chat/commands/index.js +37 -0
- package/dist/chat/commands/index.js.map +1 -0
- package/dist/chat/commands/load.d.ts +3 -0
- package/dist/chat/commands/load.d.ts.map +1 -0
- package/dist/chat/commands/load.js +37 -0
- package/dist/chat/commands/load.js.map +1 -0
- package/dist/chat/commands/message.d.ts +5 -0
- package/dist/chat/commands/message.d.ts.map +1 -0
- package/dist/chat/commands/message.js +53 -0
- package/dist/chat/commands/message.js.map +1 -0
- package/dist/chat/commands/new.d.ts +3 -0
- package/dist/chat/commands/new.d.ts.map +1 -0
- package/dist/chat/commands/new.js +11 -0
- package/dist/chat/commands/new.js.map +1 -0
- package/dist/chat/commands/save.d.ts +3 -0
- package/dist/chat/commands/save.d.ts.map +1 -0
- package/dist/chat/commands/save.js +5 -0
- package/dist/chat/commands/save.js.map +1 -0
- package/dist/chat/message.d.ts +3 -0
- package/dist/chat/message.d.ts.map +1 -0
- package/dist/chat/message.js +31 -0
- package/dist/chat/message.js.map +1 -0
- package/dist/chat/print.d.ts +3 -0
- package/dist/chat/print.d.ts.map +1 -0
- package/dist/chat/print.js +24 -0
- package/dist/chat/print.js.map +1 -0
- package/dist/chat/runtime.d.ts +2 -0
- package/dist/chat/runtime.d.ts.map +1 -0
- package/dist/chat/runtime.js +64 -0
- package/dist/chat/runtime.js.map +1 -0
- package/dist/chat/session.d.ts +8 -0
- package/dist/chat/session.d.ts.map +1 -0
- package/dist/chat/session.js +77 -0
- package/dist/chat/session.js.map +1 -0
- package/dist/chat/shared.d.ts +6 -0
- package/dist/chat/shared.d.ts.map +1 -0
- package/dist/chat/shared.js +16 -0
- package/dist/chat/shared.js.map +1 -0
- package/dist/cli.js +3 -3
- package/dist/cli.js.map +1 -1
- package/dist/config.js +1 -1
- package/dist/config.js.map +1 -1
- package/dist/session/commands/back.d.ts +3 -0
- package/dist/session/commands/back.d.ts.map +1 -0
- package/dist/session/commands/back.js +95 -0
- package/dist/session/commands/back.js.map +1 -0
- package/dist/session/commands/editor.d.ts +3 -0
- package/dist/session/commands/editor.d.ts.map +1 -0
- package/dist/session/commands/editor.js +20 -0
- package/dist/session/commands/editor.js.map +1 -0
- package/dist/session/commands/exit.d.ts +3 -0
- package/dist/session/commands/exit.d.ts.map +1 -0
- package/dist/session/commands/exit.js +5 -0
- package/dist/session/commands/exit.js.map +1 -0
- package/dist/session/commands/help.d.ts +2 -0
- package/dist/session/commands/help.d.ts.map +1 -0
- package/dist/session/commands/help.js +4 -0
- package/dist/session/commands/help.js.map +1 -0
- package/dist/session/commands/load.d.ts +3 -0
- package/dist/session/commands/load.d.ts.map +1 -0
- package/dist/session/commands/load.js +37 -0
- package/dist/session/commands/load.js.map +1 -0
- package/dist/session/commands/new.d.ts +3 -0
- package/dist/session/commands/new.d.ts.map +1 -0
- package/dist/session/commands/new.js +6 -0
- package/dist/session/commands/new.js.map +1 -0
- package/dist/session/commands/save.d.ts +3 -0
- package/dist/session/commands/save.d.ts.map +1 -0
- package/dist/session/commands/save.js +5 -0
- package/dist/session/commands/save.js.map +1 -0
- package/dist/session/message.d.ts +3 -0
- package/dist/session/message.d.ts.map +1 -0
- package/dist/session/message.js +31 -0
- package/dist/session/message.js.map +1 -0
- package/dist/session/print.d.ts +3 -0
- package/dist/session/print.d.ts.map +1 -0
- package/dist/session/print.js +24 -0
- package/dist/session/print.js.map +1 -0
- package/dist/session/runtime.d.ts +2 -0
- package/dist/session/runtime.d.ts.map +1 -0
- package/dist/session/runtime.js +46 -0
- package/dist/session/runtime.js.map +1 -0
- package/dist/session/shared.d.ts +8 -0
- package/dist/session/shared.d.ts.map +1 -0
- package/dist/session/shared.js +77 -0
- package/dist/session/shared.js.map +1 -0
- package/package.json +1 -1
- package/src/__tests__/chat.test.ts +147 -0
- package/src/__tests__/e2e.test.ts +187 -0
- package/src/agent-factory.ts +1 -1
- package/src/chat/commands/back.ts +90 -0
- package/src/chat/commands/editor.ts +18 -0
- package/src/chat/commands/exit.ts +6 -0
- package/src/chat/commands/help.ts +3 -0
- package/src/chat/commands/index.ts +38 -0
- package/src/chat/commands/load.ts +38 -0
- package/src/chat/commands/new.ts +12 -0
- package/src/chat/commands/save.ts +6 -0
- package/src/chat/message.ts +27 -0
- package/src/chat/print.ts +27 -0
- package/src/chat/runtime.ts +65 -0
- package/src/chat/shared.ts +21 -0
- package/src/cli.ts +4 -4
- package/src/config.ts +1 -1
- package/src/agent-runtime.ts +0 -265
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import type { ChatCtx } from "../shared.js";
|
|
2
|
+
import { cmdExit } from "./exit.js";
|
|
3
|
+
import { cmdHelp } from "./help.js";
|
|
4
|
+
import { cmdSave } from "./save.js";
|
|
5
|
+
import { cmdNew } from "./new.js";
|
|
6
|
+
import { cmdLoad } from "./load.js";
|
|
7
|
+
import { cmdEditor } from "./editor.js";
|
|
8
|
+
import { cmdBack } from "./back.js";
|
|
9
|
+
|
|
10
|
+
export async function dispatchCommand(ctx: ChatCtx, cmd: string): Promise<void> {
|
|
11
|
+
switch (cmd) {
|
|
12
|
+
case "/exit":
|
|
13
|
+
case "/quit":
|
|
14
|
+
cmdExit(ctx);
|
|
15
|
+
break;
|
|
16
|
+
case "/save":
|
|
17
|
+
await cmdSave(ctx);
|
|
18
|
+
break;
|
|
19
|
+
case "/new":
|
|
20
|
+
await cmdNew(ctx);
|
|
21
|
+
break;
|
|
22
|
+
case "/load":
|
|
23
|
+
await cmdLoad(ctx);
|
|
24
|
+
break;
|
|
25
|
+
case "/back":
|
|
26
|
+
await cmdBack(ctx);
|
|
27
|
+
break;
|
|
28
|
+
case "/editor":
|
|
29
|
+
await cmdEditor(ctx);
|
|
30
|
+
break;
|
|
31
|
+
case "/help":
|
|
32
|
+
cmdHelp();
|
|
33
|
+
break;
|
|
34
|
+
default:
|
|
35
|
+
console.log(`\n❌ 未知命令: ${cmd}\n`);
|
|
36
|
+
break;
|
|
37
|
+
}
|
|
38
|
+
}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { Message } from "@ai-zen/agents-core";
|
|
2
|
+
import inquirer from "inquirer";
|
|
3
|
+
import type { ChatCtx } from "../shared.js";
|
|
4
|
+
import { SYSTEM_PROMPT } from "../shared.js";
|
|
5
|
+
import { listSnapshots, loadSnapshot, saveMessages } from "../../config.js";
|
|
6
|
+
import { buildAgent } from "../../agent-factory.js";
|
|
7
|
+
|
|
8
|
+
export async function cmdLoad(ctx: ChatCtx): Promise<void> {
|
|
9
|
+
const snapshots = listSnapshots();
|
|
10
|
+
if (snapshots.length === 0) {
|
|
11
|
+
console.log("\n📭 没有可用的快照\n");
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
const { selectedName } = await inquirer.prompt([
|
|
15
|
+
{
|
|
16
|
+
type: "list",
|
|
17
|
+
name: "selectedName",
|
|
18
|
+
message: "选择要加载的快照:",
|
|
19
|
+
pageSize: 15,
|
|
20
|
+
choices: [
|
|
21
|
+
{ name: "\u21a9\ufe0f 取消操作", value: "" },
|
|
22
|
+
...snapshots.map((s) => ({ name: s.date, value: s.name })),
|
|
23
|
+
],
|
|
24
|
+
},
|
|
25
|
+
]);
|
|
26
|
+
if (!selectedName) {
|
|
27
|
+
console.log("\n已取消\n");
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
let loaded = loadSnapshot(selectedName);
|
|
31
|
+
if (loaded.length === 0) {
|
|
32
|
+
loaded = [Message.System(SYSTEM_PROMPT)];
|
|
33
|
+
}
|
|
34
|
+
saveMessages(loaded);
|
|
35
|
+
const snap = snapshots.find((s) => s.name === selectedName);
|
|
36
|
+
console.log("\n✅ 已加载快照: " + (snap ? snap.date : selectedName) + "\n");
|
|
37
|
+
ctx.agent = await buildAgent(loaded);
|
|
38
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import type { ChatCtx } from "../shared.js";
|
|
2
|
+
import { Message } from "@ai-zen/agents-core";
|
|
3
|
+
import { saveMessages } from "../../config.js";
|
|
4
|
+
import { buildAgent } from "../../agent-factory.js";
|
|
5
|
+
import { SYSTEM_PROMPT } from "../shared.js";
|
|
6
|
+
|
|
7
|
+
export async function cmdNew(ctx: ChatCtx): Promise<void> {
|
|
8
|
+
const msgs = [Message.System(SYSTEM_PROMPT)];
|
|
9
|
+
saveMessages(msgs);
|
|
10
|
+
ctx.agent = await buildAgent(msgs);
|
|
11
|
+
console.log("\n🆕 新会话\n");
|
|
12
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { Message } from "@ai-zen/agents-core";
|
|
2
|
+
import type { ChatCtx } from "./shared.js";
|
|
3
|
+
import { SYSTEM_PROMPT } from "./shared.js";
|
|
4
|
+
import { sendAndPrint } from "./print.js";
|
|
5
|
+
import { saveMessages, saveSnapshot } from "../config.js";
|
|
6
|
+
import { buildAgent } from "../agent-factory.js";
|
|
7
|
+
import { contextSize, shouldMigrate, generateMigrationDoc, MAX_CONTEXT_CHARS } from "../migration.js";
|
|
8
|
+
|
|
9
|
+
export async function handleMessage(ctx: ChatCtx, text: string): Promise<void> {
|
|
10
|
+
try {
|
|
11
|
+
await sendAndPrint(ctx.agent, text);
|
|
12
|
+
saveMessages(ctx.agent.messages);
|
|
13
|
+
|
|
14
|
+
if (shouldMigrate(ctx.agent.messages)) {
|
|
15
|
+
console.log(`🔄 上下文 ${contextSize(ctx.agent.messages)}/${MAX_CONTEXT_CHARS},准备迁移...`);
|
|
16
|
+
try {
|
|
17
|
+
const snap = saveSnapshot(ctx.agent.messages);
|
|
18
|
+
console.log(` 💾 快照: ${snap}`);
|
|
19
|
+
const summary = await generateMigrationDoc(ctx.agent.messages);
|
|
20
|
+
const msgs = [Message.System(SYSTEM_PROMPT), Message.User(summary)];
|
|
21
|
+
saveMessages(msgs);
|
|
22
|
+
console.log("✅ 迁移完成\n");
|
|
23
|
+
ctx.agent = await buildAgent(msgs);
|
|
24
|
+
} catch (err: any) { console.error(`❌ 迁移失败: ${err.message}\n`); }
|
|
25
|
+
}
|
|
26
|
+
} catch (err: any) { console.error(`\n❌ ${err.message}`); }
|
|
27
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import chalk from "chalk";
|
|
2
|
+
import { Agent } from "@ai-zen/agents-core";
|
|
3
|
+
import { DeltaRenderer } from "../delta-renderer.js";
|
|
4
|
+
import type { AgentNS } from "@ai-zen/agents-core";
|
|
5
|
+
|
|
6
|
+
export async function sendAndPrint(agent: Agent, text: string): Promise<void> {
|
|
7
|
+
console.log(chalk.green.bold("\n🤖 AI:"));
|
|
8
|
+
const renderer = new DeltaRenderer({
|
|
9
|
+
reasoningHeader: "\n\n💭 思考中...\n",
|
|
10
|
+
contentHeader: "\n\n💭 回答中...\n",
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
function onChunk(chunk: AgentNS.StreamResponseData) {
|
|
14
|
+
const delta = chunk?.choices?.[0]?.delta;
|
|
15
|
+
const fr = chunk?.choices?.[0]?.finish_reason ?? null;
|
|
16
|
+
if (delta) renderer.render(delta, fr);
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
function onRun() { renderer.reset(); }
|
|
20
|
+
agent.events.on("run", onRun);
|
|
21
|
+
agent.events.on("chunk", onChunk);
|
|
22
|
+
await agent.send(text);
|
|
23
|
+
agent.events.off("run", onRun);
|
|
24
|
+
agent.events.off("chunk", onChunk);
|
|
25
|
+
process.stdout.write("\n\n");
|
|
26
|
+
console.log();
|
|
27
|
+
}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import { Agent, Message } from "@ai-zen/agents-core";
|
|
2
|
+
import inquirer from "inquirer";
|
|
3
|
+
import { readConfig, readMessages, saveMessages, saveSnapshot } from "../config.js";
|
|
4
|
+
import { buildAgent } from "../agent-factory.js";
|
|
5
|
+
import { contextSize } from "../migration.js";
|
|
6
|
+
import { ChatCtx, SYSTEM_PROMPT } from "./shared.js";
|
|
7
|
+
import { dispatchCommand } from "./commands/index.js";
|
|
8
|
+
import { handleMessage } from "./message.js";
|
|
9
|
+
import { sendAndPrint } from "./print.js";
|
|
10
|
+
|
|
11
|
+
async function chatLoop(ctx: ChatCtx) {
|
|
12
|
+
while (true) {
|
|
13
|
+
const { input } = await inquirer.prompt([
|
|
14
|
+
{ type: "input", name: "input", message: "你:" },
|
|
15
|
+
]);
|
|
16
|
+
const t = input.trim();
|
|
17
|
+
if (!t) continue;
|
|
18
|
+
if (t.startsWith("/")) {
|
|
19
|
+
await dispatchCommand(ctx, t.toLowerCase());
|
|
20
|
+
continue;
|
|
21
|
+
}
|
|
22
|
+
await handleMessage(ctx, t);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export async function runChat(initialMessage?: string): Promise<void> {
|
|
27
|
+
const config = readConfig();
|
|
28
|
+
if (!config.apiKey) {
|
|
29
|
+
console.error("❌ 请先设置 API Key: air key <your-key>");
|
|
30
|
+
console.error(" 获取 Key: https://platform.deepseek.com/api_keys");
|
|
31
|
+
process.exit(1);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
let msgs = readMessages();
|
|
35
|
+
if (msgs.length === 0) {
|
|
36
|
+
msgs = [Message.System(SYSTEM_PROMPT)];
|
|
37
|
+
saveMessages(msgs);
|
|
38
|
+
}
|
|
39
|
+
const agent: Agent = await buildAgent(msgs);
|
|
40
|
+
const ctx: ChatCtx = { agent };
|
|
41
|
+
|
|
42
|
+
if (initialMessage) {
|
|
43
|
+
const hasHistory = ctx.agent.messages.some(m => m.role === "user" || m.role === "assistant");
|
|
44
|
+
if (hasHistory) {
|
|
45
|
+
const snap = saveSnapshot(ctx.agent.messages);
|
|
46
|
+
console.log(`💾 已存档旧对话: ${snap}\n`);
|
|
47
|
+
}
|
|
48
|
+
await dispatchCommand(ctx, "/new");
|
|
49
|
+
console.log(`💬 你: ${initialMessage}`);
|
|
50
|
+
try {
|
|
51
|
+
await sendAndPrint(ctx.agent, initialMessage);
|
|
52
|
+
saveMessages(ctx.agent.messages);
|
|
53
|
+
} catch (err: any) { console.error(`\n❌ ${err.message}`); }
|
|
54
|
+
console.log("\n💬 继续对话 (输入 /exit 退出)\n");
|
|
55
|
+
await chatLoop(ctx);
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
const msgCount = ctx.agent.messages.filter((m) => m.role !== "system").length;
|
|
60
|
+
console.log(msgCount > 0
|
|
61
|
+
? `\n💬 继续上次对话 (${msgCount} 条,/${contextSize(ctx.agent.messages)} 字符,输入 /new 重新开始)\n`
|
|
62
|
+
: "\n💬 air — 极简 AI 助手 (输入 /help 查看命令)\n");
|
|
63
|
+
|
|
64
|
+
await chatLoop(ctx);
|
|
65
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { Agent } from "@ai-zen/agents-core";
|
|
2
|
+
|
|
3
|
+
export interface ChatCtx {
|
|
4
|
+
agent: Agent;
|
|
5
|
+
}
|
|
6
|
+
|
|
7
|
+
export const SYSTEM_PROMPT = [
|
|
8
|
+
"你是一个AI助手,专门帮助用户回答问题和执行任务。请用中文回复。",
|
|
9
|
+
"",
|
|
10
|
+
"## 行为准则",
|
|
11
|
+
"1. 做任何改动之前,必须先跟用户商量,获得明确指示之后再行动。",
|
|
12
|
+
"2. 用户没有明确要求产出文件时,不得自行创建任何文件到项目中。讨论就停留在讨论层面。",
|
|
13
|
+
"3. 区分危险操作:删除文件、覆盖文件、安装卸载软件、修改系统配置、执行耗时任务等属于危险操作。执行前必须评估影响范围,并向用户说明风险,获得用户明确的书面的确认之后再执行。",
|
|
14
|
+
"4. 追责原则:你的每一步操作都应当基于用户的明确指令。如果出了问题,可以从用户说过的话追溯责任——是用户让你做的,用户承担责任。所以你不需要畏手畏脚,只要用户明确说了\u201c做\u201d,你就放心去做。",
|
|
15
|
+
"",
|
|
16
|
+
"## 记忆",
|
|
17
|
+
"你可以使用 shell 工具执行任何命令。如果你有需要长期记住的信息(用户偏好、项目约定、任务进度等),请写入以下位置:",
|
|
18
|
+
"- 全局记忆: ~/.ai-zen/air/temp/*.md",
|
|
19
|
+
"- 项目记忆: $(cwd)/.ai-zen/air/temp/*.md",
|
|
20
|
+
"下次启动时用 shell 读取即可。这是你唯一的记忆方式。",
|
|
21
|
+
].join("\n");
|
package/src/cli.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import { readFileSync } from "node:fs";
|
|
3
3
|
import { Command } from "commander";
|
|
4
|
-
import {
|
|
4
|
+
import { runChat } from "./chat/runtime.js";
|
|
5
5
|
import { installHook, uninstallHook } from "./hook.js";
|
|
6
6
|
import { readConfig, saveConfig } from "./config.js";
|
|
7
7
|
|
|
@@ -49,15 +49,15 @@ program
|
|
|
49
49
|
});
|
|
50
50
|
|
|
51
51
|
program
|
|
52
|
-
.argument("[message]", "要发送的消息(不传则进入交互模式)")
|
|
53
|
-
.action(async (message?: string) => {
|
|
52
|
+
.argument("[message...]", "要发送的消息(不传则进入交互模式)")
|
|
53
|
+
.action(async (message?: string[]) => {
|
|
54
54
|
const config = readConfig();
|
|
55
55
|
if (!config.apiKey) {
|
|
56
56
|
console.error("❌ 请先设置 API Key: air key <your-key>");
|
|
57
57
|
console.error(" 获取 Key: https://platform.deepseek.com/api_keys");
|
|
58
58
|
process.exit(1);
|
|
59
59
|
}
|
|
60
|
-
await
|
|
60
|
+
await runChat(message?.join(" "));
|
|
61
61
|
});
|
|
62
62
|
|
|
63
63
|
program.parse(process.argv);
|
package/src/config.ts
CHANGED
|
@@ -10,7 +10,7 @@ export interface Config {
|
|
|
10
10
|
|
|
11
11
|
// ==================== 路径 ====================
|
|
12
12
|
|
|
13
|
-
const AIR_DIR = () => join(homedir(), ".ai-zen", "air");
|
|
13
|
+
const AIR_DIR = () => process.env.AIR_DIR || join(homedir(), ".ai-zen", "air");
|
|
14
14
|
const CONFIG_FILE = () => join(AIR_DIR(), "config.json");
|
|
15
15
|
const CONTEXT_FILE = () => join(AIR_DIR(), "context.json");
|
|
16
16
|
const SNAPSHOTS_DIR = () => join(AIR_DIR(), "snapshots");
|
package/src/agent-runtime.ts
DELETED
|
@@ -1,265 +0,0 @@
|
|
|
1
|
-
import chalk from "chalk";
|
|
2
|
-
import { Agent, Message } from "@ai-zen/agents-core";
|
|
3
|
-
import { DeltaRenderer } from "./delta-renderer.js";
|
|
4
|
-
import type { AgentNS } from "@ai-zen/agents-core";
|
|
5
|
-
import inquirer from "inquirer";
|
|
6
|
-
import { readConfig, readMessages, saveMessages, clearMessages, saveSnapshot, listSnapshots, loadSnapshot } from "./config.js";
|
|
7
|
-
import { buildAgent } from "./agent-factory.js";
|
|
8
|
-
import { contextSize, shouldMigrate, generateMigrationDoc, MAX_CONTEXT_CHARS } from "./migration.js";
|
|
9
|
-
|
|
10
|
-
const SYSTEM_PROMPT = [
|
|
11
|
-
"你是一个AI助手,专门帮助用户回答问题和执行任务。请用中文回复。",
|
|
12
|
-
"",
|
|
13
|
-
"## 行为准则",
|
|
14
|
-
"1. 做任何改动之前,必须先跟用户商量,获得明确指示之后再行动。",
|
|
15
|
-
"2. 用户没有明确要求产出文件时,不得自行创建任何文件到项目中。讨论就停留在讨论层面。",
|
|
16
|
-
"3. 区分危险操作:删除文件、覆盖文件、安装卸载软件、修改系统配置、执行耗时任务等属于危险操作。执行前必须评估影响范围,并向用户说明风险,获得用户明确的书面的确认之后再执行。",
|
|
17
|
-
"4. 追责原则:你的每一步操作都应当基于用户的明确指令。如果出了问题,可以从用户说过的话追溯责任——是用户让你做的,用户承担责任。所以你不需要畏手畏脚,只要用户明确说了\u201c做\u201d,你就放心去做。",
|
|
18
|
-
"",
|
|
19
|
-
"## 记忆",
|
|
20
|
-
"你可以使用 shell 工具执行任何命令。如果你有需要长期记住的信息(用户偏好、项目约定、任务进度等),请写入以下位置:",
|
|
21
|
-
"- 全局记忆: ~/.ai-zen/air/temp/*.md",
|
|
22
|
-
"- 项目记忆: $(cwd)/.ai-zen/air/temp/*.md",
|
|
23
|
-
"下次启动时用 shell 读取即可。这是你唯一的记忆方式。",
|
|
24
|
-
].join("\n");
|
|
25
|
-
|
|
26
|
-
// 流式发送
|
|
27
|
-
|
|
28
|
-
async function sendAndPrint(agent: Agent, text: string): Promise<void> {
|
|
29
|
-
const renderer = new DeltaRenderer({
|
|
30
|
-
reasoningHeader: "\n\n💭 思考中...\n",
|
|
31
|
-
contentHeader: "\n\n💭 回答中...\n",
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
function onChunk(chunk: AgentNS.StreamResponseData) {
|
|
35
|
-
const delta = chunk?.choices?.[0]?.delta;
|
|
36
|
-
const fr = chunk?.choices?.[0]?.finish_reason ?? null;
|
|
37
|
-
if (delta) renderer.render(delta, fr);
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
function onRun() { renderer.reset(); }
|
|
41
|
-
agent.events.on("run", onRun);
|
|
42
|
-
agent.events.on("chunk", onChunk);
|
|
43
|
-
await agent.send(text);
|
|
44
|
-
agent.events.off("run", onRun);
|
|
45
|
-
agent.events.off("chunk", onChunk);
|
|
46
|
-
process.stdout.write("\n\n");
|
|
47
|
-
console.log();
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
async function sendAndSave(agent: Agent, text: string): Promise<void> {
|
|
51
|
-
console.log(chalk.green.bold("\n🤖 AI:"));
|
|
52
|
-
await sendAndPrint(agent, text);
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
// 对话循环
|
|
56
|
-
|
|
57
|
-
export async function runConversation(initialMessage?: string): Promise<void> {
|
|
58
|
-
const config = readConfig();
|
|
59
|
-
if (!config.apiKey) {
|
|
60
|
-
console.error("❌ 请先设置 API Key: air key <your-key>");
|
|
61
|
-
console.error(" 获取 Key: https://platform.deepseek.com/api_keys");
|
|
62
|
-
process.exit(1);
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
let msgs = readMessages();
|
|
66
|
-
if (msgs.length === 0) {
|
|
67
|
-
msgs = [Message.System(SYSTEM_PROMPT)];
|
|
68
|
-
saveMessages(msgs);
|
|
69
|
-
}
|
|
70
|
-
let agent = await buildAgent(msgs);
|
|
71
|
-
|
|
72
|
-
if (initialMessage) {
|
|
73
|
-
// 有旧对话则先存档,再开新对话
|
|
74
|
-
const hasHistory = agent.messages.some(m => m.role === "user" || m.role === "assistant");
|
|
75
|
-
if (hasHistory) {
|
|
76
|
-
const snap = saveSnapshot(agent.messages);
|
|
77
|
-
console.log(`💾 已存档旧对话: ${snap}\n`);
|
|
78
|
-
}
|
|
79
|
-
msgs = [Message.System(SYSTEM_PROMPT)];
|
|
80
|
-
saveMessages(msgs);
|
|
81
|
-
agent = await buildAgent(msgs);
|
|
82
|
-
console.log(`💬 你: ${initialMessage}`);
|
|
83
|
-
try {
|
|
84
|
-
await sendAndSave(agent, initialMessage);
|
|
85
|
-
saveMessages(agent.messages);
|
|
86
|
-
console.log("💾 已保存\n");
|
|
87
|
-
} catch (err: any) { console.error(`\n❌ ${err.message}`); }
|
|
88
|
-
return;
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
const msgCount = agent.messages.filter((m) => m.role !== "system").length;
|
|
92
|
-
console.log(msgCount > 0
|
|
93
|
-
? `\n💬 继续上次对话 (${msgCount} 条,/${contextSize(agent.messages)} 字符,输入 /new 重新开始)\n`
|
|
94
|
-
: "\n💬 air — 极简 AI 助手 (输入 /help 查看命令)\n");
|
|
95
|
-
|
|
96
|
-
async function ask() {
|
|
97
|
-
const { input } = await inquirer.prompt([
|
|
98
|
-
{ type: "input", name: "input", message: "你:" },
|
|
99
|
-
]);
|
|
100
|
-
const t = input.trim();
|
|
101
|
-
if (!t) { await ask(); return; }
|
|
102
|
-
|
|
103
|
-
if (t.startsWith("/")) {
|
|
104
|
-
const c = t.toLowerCase();
|
|
105
|
-
if (c === "/exit" || c === "/quit") { console.log("\n👋 再见!"); process.exit(0); }
|
|
106
|
-
if (c === "/save") { console.log(`\n✅ 快照: ${saveSnapshot(agent.messages)}\n`); await ask(); return; }
|
|
107
|
-
if (c === "/new") { clearMessages(); const msgs = [Message.System(SYSTEM_PROMPT)]; saveMessages(msgs); agent = await buildAgent(msgs); console.log("\n🆕 重新开始\n"); await ask(); return; }
|
|
108
|
-
if (c === "/load") {
|
|
109
|
-
const snapshots = listSnapshots();
|
|
110
|
-
if (snapshots.length === 0) {
|
|
111
|
-
console.log("\n📭 没有可用的快照\n");
|
|
112
|
-
await ask(); return;
|
|
113
|
-
}
|
|
114
|
-
const { selectedName } = await inquirer.prompt([
|
|
115
|
-
{
|
|
116
|
-
type: "list",
|
|
117
|
-
name: "selectedName",
|
|
118
|
-
message: "选择要加载的快照:",
|
|
119
|
-
pageSize: 15,
|
|
120
|
-
choices: [
|
|
121
|
-
{ name: "↩️ 取消操作", value: "" },
|
|
122
|
-
...snapshots.map((s) => ({ name: s.date, value: s.name })),
|
|
123
|
-
],
|
|
124
|
-
},
|
|
125
|
-
]);
|
|
126
|
-
if (!selectedName) {
|
|
127
|
-
console.log("\n已取消\n");
|
|
128
|
-
await ask(); return;
|
|
129
|
-
}
|
|
130
|
-
let msgs = loadSnapshot(selectedName);
|
|
131
|
-
if (msgs.length === 0) {
|
|
132
|
-
msgs = [Message.System(SYSTEM_PROMPT)];
|
|
133
|
-
}
|
|
134
|
-
saveMessages(msgs);
|
|
135
|
-
agent = await buildAgent(msgs);
|
|
136
|
-
const snap = snapshots.find((s) => s.name === selectedName);
|
|
137
|
-
console.log("\n✅ 已加载快照: " + (snap ? snap.date : selectedName) + "\n");
|
|
138
|
-
await ask(); return;
|
|
139
|
-
}
|
|
140
|
-
if (c === "/back") {
|
|
141
|
-
await handleBack();
|
|
142
|
-
await ask(); return;
|
|
143
|
-
}
|
|
144
|
-
if (c === "/editor") {
|
|
145
|
-
const { content } = await inquirer.prompt([
|
|
146
|
-
{ type: "editor", name: "content", message: "编辑消息:" },
|
|
147
|
-
]);
|
|
148
|
-
if (!content.trim()) { console.log("\n已取消\n"); await ask(); return; }
|
|
149
|
-
try {
|
|
150
|
-
await sendAndSave(agent, content.trim());
|
|
151
|
-
saveMessages(agent.messages);
|
|
152
|
-
} catch (err: any) { console.error(`\n❌ ${err.message}`); }
|
|
153
|
-
await ask(); return;
|
|
154
|
-
}
|
|
155
|
-
if (c === "/help") { console.log("\n/exit /quit 退出\n/save 保存快照\n/load 加载快照\n/new 重新开始\n/back 撤回消息\n/help 帮助\n"); await ask(); return; }
|
|
156
|
-
console.log(`\n❌ 未知命令: ${t}\n`); await ask(); return; }
|
|
157
|
-
|
|
158
|
-
try {
|
|
159
|
-
await sendAndSave(agent, t);
|
|
160
|
-
saveMessages(agent.messages);
|
|
161
|
-
|
|
162
|
-
if (shouldMigrate(agent.messages)) {
|
|
163
|
-
console.log(`🔄 上下文 ${contextSize(agent.messages)}/${MAX_CONTEXT_CHARS},准备迁移...`);
|
|
164
|
-
try {
|
|
165
|
-
const snap = saveSnapshot(agent.messages);
|
|
166
|
-
console.log(` 💾 快照: ${snap}`);
|
|
167
|
-
const summary = await generateMigrationDoc(agent.messages);
|
|
168
|
-
const msgs = [Message.System(SYSTEM_PROMPT), Message.User(summary)];
|
|
169
|
-
saveMessages(msgs);
|
|
170
|
-
agent = await buildAgent(msgs);
|
|
171
|
-
console.log("✅ 迁移完成\n");
|
|
172
|
-
} catch (err: any) { console.error(`❌ 迁移失败: ${err.message}\n`); }
|
|
173
|
-
}
|
|
174
|
-
} catch (err: any) { console.error(`\n❌ ${err.message}`); }
|
|
175
|
-
await ask();
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
async function handleBack(): Promise<void> {
|
|
179
|
-
const targets: { index: number; role: string; label: string; preview: string }[] = [];
|
|
180
|
-
for (let i = 0; i < agent.messages.length; i++) {
|
|
181
|
-
const msg = agent.messages[i];
|
|
182
|
-
if (msg.role === "user") {
|
|
183
|
-
const text = typeof msg.content === "string" ? msg.content : "";
|
|
184
|
-
if (text) {
|
|
185
|
-
targets.push({ index: i, role: "user", label: "👤 用户", preview: text.substring(0, 60) + (text.length > 60 ? "..." : "") });
|
|
186
|
-
}
|
|
187
|
-
} else if (msg.role === "tool" || msg.role === "function") {
|
|
188
|
-
const text = typeof msg.content === "string" ? msg.content : "";
|
|
189
|
-
if (text) {
|
|
190
|
-
targets.push({ index: i, role: msg.role, label: "🔧 工具", preview: text.substring(0, 60) + (text.length > 60 ? "..." : "") });
|
|
191
|
-
}
|
|
192
|
-
}
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
if (targets.length === 0) {
|
|
196
|
-
console.log("\n❌ 还没有消息可以撤回\n");
|
|
197
|
-
return;
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
console.log("\n📋 选择要撤回到哪条消息(将删除所选及其之后的所有内容):\n");
|
|
201
|
-
|
|
202
|
-
const { selectedIndex } = await inquirer.prompt([
|
|
203
|
-
{
|
|
204
|
-
type: "list",
|
|
205
|
-
name: "selectedIndex",
|
|
206
|
-
message: "撤回到:",
|
|
207
|
-
pageSize: 15,
|
|
208
|
-
choices: [
|
|
209
|
-
{ name: "↩️ 取消操作", value: -1 },
|
|
210
|
-
...targets.map((t) => ({ name: t.label + " " + t.preview, value: t.index })),
|
|
211
|
-
],
|
|
212
|
-
},
|
|
213
|
-
]);
|
|
214
|
-
|
|
215
|
-
if (selectedIndex === -1) { console.log("\n已取消\n"); return; }
|
|
216
|
-
|
|
217
|
-
const selectedMsg = agent.messages[selectedIndex];
|
|
218
|
-
const isUserMsg = selectedMsg.role === "user";
|
|
219
|
-
const originalText = typeof selectedMsg.content === "string" ? selectedMsg.content : "";
|
|
220
|
-
const sliceEnd = isUserMsg ? selectedIndex : selectedIndex + 1;
|
|
221
|
-
agent.messages = agent.messages.slice(0, sliceEnd);
|
|
222
|
-
|
|
223
|
-
if (isUserMsg) {
|
|
224
|
-
console.log("\n原内容: " + originalText.substring(0, 200) + (originalText.length > 200 ? "..." : "") + "\n");
|
|
225
|
-
const { editChoice } = await inquirer.prompt([
|
|
226
|
-
{
|
|
227
|
-
type: "list",
|
|
228
|
-
name: "editChoice",
|
|
229
|
-
message: "请选择:",
|
|
230
|
-
choices: [
|
|
231
|
-
{ name: "✏️ 修改后重新发送", value: "edit" },
|
|
232
|
-
{ name: "🔄 直接重新发送", value: "resend" },
|
|
233
|
-
{ name: "↩️ 取消操作", value: "cancel" },
|
|
234
|
-
],
|
|
235
|
-
},
|
|
236
|
-
]);
|
|
237
|
-
if (editChoice === "cancel") { console.log("\n已取消\n"); return; }
|
|
238
|
-
|
|
239
|
-
let textToSend = originalText;
|
|
240
|
-
if (editChoice === "edit") {
|
|
241
|
-
const { editedContent } = await inquirer.prompt([
|
|
242
|
-
{ type: "input", name: "editedContent", message: "修改消息:", default: originalText },
|
|
243
|
-
]);
|
|
244
|
-
if (!editedContent.trim()) { console.log("\n❌ 消息不能为空\n"); return; }
|
|
245
|
-
textToSend = editedContent.trim();
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
agent = await buildAgent(agent.messages);
|
|
249
|
-
await sendAndSave(agent, textToSend);
|
|
250
|
-
saveMessages(agent.messages);
|
|
251
|
-
} else {
|
|
252
|
-
console.log("\n💡 请输入一条新消息继续对话\n");
|
|
253
|
-
const { newMessage } = await inquirer.prompt([
|
|
254
|
-
{ type: "input", name: "newMessage", message: "新消息:" },
|
|
255
|
-
]);
|
|
256
|
-
if (!newMessage.trim()) { console.log("\n已取消\n"); return; }
|
|
257
|
-
agent = await buildAgent(agent.messages);
|
|
258
|
-
await sendAndSave(agent, newMessage.trim());
|
|
259
|
-
saveMessages(agent.messages);
|
|
260
|
-
}
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
await ask();
|
|
264
|
-
process.on("SIGINT", () => { console.log("\n\n👋 再见!"); process.exit(0); });
|
|
265
|
-
}
|