@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.
Files changed (130) hide show
  1. package/README.md +21 -6
  2. package/README.zh.md +31 -6
  3. package/dist/agent-factory.js +1 -1
  4. package/dist/agent-factory.js.map +1 -1
  5. package/dist/agent-runtime.d.ts.map +1 -1
  6. package/dist/agent-runtime.js +241 -228
  7. package/dist/agent-runtime.js.map +1 -1
  8. package/dist/chat/commands/back.d.ts +3 -0
  9. package/dist/chat/commands/back.d.ts.map +1 -0
  10. package/dist/chat/commands/back.js +95 -0
  11. package/dist/chat/commands/back.js.map +1 -0
  12. package/dist/chat/commands/editor.d.ts +3 -0
  13. package/dist/chat/commands/editor.d.ts.map +1 -0
  14. package/dist/chat/commands/editor.js +20 -0
  15. package/dist/chat/commands/editor.js.map +1 -0
  16. package/dist/chat/commands/exit.d.ts +3 -0
  17. package/dist/chat/commands/exit.d.ts.map +1 -0
  18. package/dist/chat/commands/exit.js +5 -0
  19. package/dist/chat/commands/exit.js.map +1 -0
  20. package/dist/chat/commands/help.d.ts +2 -0
  21. package/dist/chat/commands/help.d.ts.map +1 -0
  22. package/dist/chat/commands/help.js +4 -0
  23. package/dist/chat/commands/help.js.map +1 -0
  24. package/dist/chat/commands/index.d.ts +3 -0
  25. package/dist/chat/commands/index.d.ts.map +1 -0
  26. package/dist/chat/commands/index.js +37 -0
  27. package/dist/chat/commands/index.js.map +1 -0
  28. package/dist/chat/commands/load.d.ts +3 -0
  29. package/dist/chat/commands/load.d.ts.map +1 -0
  30. package/dist/chat/commands/load.js +37 -0
  31. package/dist/chat/commands/load.js.map +1 -0
  32. package/dist/chat/commands/message.d.ts +5 -0
  33. package/dist/chat/commands/message.d.ts.map +1 -0
  34. package/dist/chat/commands/message.js +53 -0
  35. package/dist/chat/commands/message.js.map +1 -0
  36. package/dist/chat/commands/new.d.ts +3 -0
  37. package/dist/chat/commands/new.d.ts.map +1 -0
  38. package/dist/chat/commands/new.js +11 -0
  39. package/dist/chat/commands/new.js.map +1 -0
  40. package/dist/chat/commands/save.d.ts +3 -0
  41. package/dist/chat/commands/save.d.ts.map +1 -0
  42. package/dist/chat/commands/save.js +5 -0
  43. package/dist/chat/commands/save.js.map +1 -0
  44. package/dist/chat/message.d.ts +3 -0
  45. package/dist/chat/message.d.ts.map +1 -0
  46. package/dist/chat/message.js +31 -0
  47. package/dist/chat/message.js.map +1 -0
  48. package/dist/chat/print.d.ts +3 -0
  49. package/dist/chat/print.d.ts.map +1 -0
  50. package/dist/chat/print.js +24 -0
  51. package/dist/chat/print.js.map +1 -0
  52. package/dist/chat/runtime.d.ts +2 -0
  53. package/dist/chat/runtime.d.ts.map +1 -0
  54. package/dist/chat/runtime.js +64 -0
  55. package/dist/chat/runtime.js.map +1 -0
  56. package/dist/chat/session.d.ts +8 -0
  57. package/dist/chat/session.d.ts.map +1 -0
  58. package/dist/chat/session.js +77 -0
  59. package/dist/chat/session.js.map +1 -0
  60. package/dist/chat/shared.d.ts +6 -0
  61. package/dist/chat/shared.d.ts.map +1 -0
  62. package/dist/chat/shared.js +16 -0
  63. package/dist/chat/shared.js.map +1 -0
  64. package/dist/cli.js +3 -3
  65. package/dist/cli.js.map +1 -1
  66. package/dist/config.js +1 -1
  67. package/dist/config.js.map +1 -1
  68. package/dist/session/commands/back.d.ts +3 -0
  69. package/dist/session/commands/back.d.ts.map +1 -0
  70. package/dist/session/commands/back.js +95 -0
  71. package/dist/session/commands/back.js.map +1 -0
  72. package/dist/session/commands/editor.d.ts +3 -0
  73. package/dist/session/commands/editor.d.ts.map +1 -0
  74. package/dist/session/commands/editor.js +20 -0
  75. package/dist/session/commands/editor.js.map +1 -0
  76. package/dist/session/commands/exit.d.ts +3 -0
  77. package/dist/session/commands/exit.d.ts.map +1 -0
  78. package/dist/session/commands/exit.js +5 -0
  79. package/dist/session/commands/exit.js.map +1 -0
  80. package/dist/session/commands/help.d.ts +2 -0
  81. package/dist/session/commands/help.d.ts.map +1 -0
  82. package/dist/session/commands/help.js +4 -0
  83. package/dist/session/commands/help.js.map +1 -0
  84. package/dist/session/commands/load.d.ts +3 -0
  85. package/dist/session/commands/load.d.ts.map +1 -0
  86. package/dist/session/commands/load.js +37 -0
  87. package/dist/session/commands/load.js.map +1 -0
  88. package/dist/session/commands/new.d.ts +3 -0
  89. package/dist/session/commands/new.d.ts.map +1 -0
  90. package/dist/session/commands/new.js +6 -0
  91. package/dist/session/commands/new.js.map +1 -0
  92. package/dist/session/commands/save.d.ts +3 -0
  93. package/dist/session/commands/save.d.ts.map +1 -0
  94. package/dist/session/commands/save.js +5 -0
  95. package/dist/session/commands/save.js.map +1 -0
  96. package/dist/session/message.d.ts +3 -0
  97. package/dist/session/message.d.ts.map +1 -0
  98. package/dist/session/message.js +31 -0
  99. package/dist/session/message.js.map +1 -0
  100. package/dist/session/print.d.ts +3 -0
  101. package/dist/session/print.d.ts.map +1 -0
  102. package/dist/session/print.js +24 -0
  103. package/dist/session/print.js.map +1 -0
  104. package/dist/session/runtime.d.ts +2 -0
  105. package/dist/session/runtime.d.ts.map +1 -0
  106. package/dist/session/runtime.js +46 -0
  107. package/dist/session/runtime.js.map +1 -0
  108. package/dist/session/shared.d.ts +8 -0
  109. package/dist/session/shared.d.ts.map +1 -0
  110. package/dist/session/shared.js +77 -0
  111. package/dist/session/shared.js.map +1 -0
  112. package/package.json +1 -1
  113. package/src/__tests__/chat.test.ts +147 -0
  114. package/src/__tests__/e2e.test.ts +187 -0
  115. package/src/agent-factory.ts +1 -1
  116. package/src/chat/commands/back.ts +90 -0
  117. package/src/chat/commands/editor.ts +18 -0
  118. package/src/chat/commands/exit.ts +6 -0
  119. package/src/chat/commands/help.ts +3 -0
  120. package/src/chat/commands/index.ts +38 -0
  121. package/src/chat/commands/load.ts +38 -0
  122. package/src/chat/commands/new.ts +12 -0
  123. package/src/chat/commands/save.ts +6 -0
  124. package/src/chat/message.ts +27 -0
  125. package/src/chat/print.ts +27 -0
  126. package/src/chat/runtime.ts +65 -0
  127. package/src/chat/shared.ts +21 -0
  128. package/src/cli.ts +4 -4
  129. package/src/config.ts +1 -1
  130. package/src/agent-runtime.ts +0 -265
@@ -0,0 +1,24 @@
1
+ import chalk from "chalk";
2
+ import { DeltaRenderer } from "../delta-renderer.js";
3
+ export async function sendAndPrint(agent, text) {
4
+ console.log(chalk.green.bold("\n🤖 AI:"));
5
+ const renderer = new DeltaRenderer({
6
+ reasoningHeader: "\n\n💭 思考中...\n",
7
+ contentHeader: "\n\n💭 回答中...\n",
8
+ });
9
+ function onChunk(chunk) {
10
+ const delta = chunk?.choices?.[0]?.delta;
11
+ const fr = chunk?.choices?.[0]?.finish_reason ?? null;
12
+ if (delta)
13
+ renderer.render(delta, fr);
14
+ }
15
+ function onRun() { renderer.reset(); }
16
+ agent.events.on("run", onRun);
17
+ agent.events.on("chunk", onChunk);
18
+ await agent.send(text);
19
+ agent.events.off("run", onRun);
20
+ agent.events.off("chunk", onChunk);
21
+ process.stdout.write("\n\n");
22
+ console.log();
23
+ }
24
+ //# sourceMappingURL=print.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"print.js","sourceRoot":"","sources":["../../src/session/print.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAE1B,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAGrD,MAAM,CAAC,KAAK,UAAU,YAAY,CAAC,KAAY,EAAE,IAAY;IAC3D,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC;IAC1C,MAAM,QAAQ,GAAG,IAAI,aAAa,CAAC;QACjC,eAAe,EAAE,iBAAiB;QAClC,aAAa,EAAE,iBAAiB;KACjC,CAAC,CAAC;IAEH,SAAS,OAAO,CAAC,KAAiC;QAChD,MAAM,KAAK,GAAG,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC;QACzC,MAAM,EAAE,GAAG,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC,EAAE,aAAa,IAAI,IAAI,CAAC;QACtD,IAAI,KAAK;YAAE,QAAQ,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;IACxC,CAAC;IAED,SAAS,KAAK,KAAK,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;IACtC,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;IAC9B,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IAClC,MAAM,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACvB,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;IAC/B,KAAK,CAAC,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;IACnC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAC7B,OAAO,CAAC,GAAG,EAAE,CAAC;AAChB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function runConversation(initialMessage?: string): Promise<void>;
2
+ //# sourceMappingURL=runtime.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"runtime.d.ts","sourceRoot":"","sources":["../../src/session/runtime.ts"],"names":[],"mappings":"AAOA,wBAAsB,eAAe,CAAC,cAAc,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAuC5E"}
@@ -0,0 +1,46 @@
1
+ import { Message } from "@ai-zen/agents-core";
2
+ import { readConfig, readMessages, saveMessages, saveSnapshot } from "../config.js";
3
+ import { buildAgent } from "../agent-factory.js";
4
+ import { contextSize } from "../migration.js";
5
+ import { SYSTEM_PROMPT, saveAndNew, ask } from "./shared.js";
6
+ import { sendAndPrint } from "./print.js";
7
+ export async function runConversation(initialMessage) {
8
+ const config = readConfig();
9
+ if (!config.apiKey) {
10
+ console.error("❌ 请先设置 API Key: air key <your-key>");
11
+ console.error(" 获取 Key: https://platform.deepseek.com/api_keys");
12
+ process.exit(1);
13
+ }
14
+ let msgs = readMessages();
15
+ if (msgs.length === 0) {
16
+ msgs = [Message.System(SYSTEM_PROMPT)];
17
+ saveMessages(msgs);
18
+ }
19
+ const agent = await buildAgent(msgs);
20
+ const ctx = { agent };
21
+ if (initialMessage) {
22
+ const hasHistory = ctx.agent.messages.some(m => m.role === "user" || m.role === "assistant");
23
+ if (hasHistory) {
24
+ const snap = saveSnapshot(ctx.agent.messages);
25
+ console.log(`💾 已存档旧对话: ${snap}\n`);
26
+ }
27
+ await saveAndNew(ctx);
28
+ console.log(`💬 你: ${initialMessage}`);
29
+ try {
30
+ await sendAndPrint(ctx.agent, initialMessage);
31
+ saveMessages(ctx.agent.messages);
32
+ }
33
+ catch (err) {
34
+ console.error(`\n❌ ${err.message}`);
35
+ }
36
+ console.log("\n💬 继续对话 (输入 /exit 退出)\n");
37
+ await ask(ctx);
38
+ return;
39
+ }
40
+ const msgCount = ctx.agent.messages.filter((m) => m.role !== "system").length;
41
+ console.log(msgCount > 0
42
+ ? `\n💬 继续上次对话 (${msgCount} 条,/${contextSize(ctx.agent.messages)} 字符,输入 /new 重新开始)\n`
43
+ : "\n💬 air — 极简 AI 助手 (输入 /help 查看命令)\n");
44
+ await ask(ctx);
45
+ }
46
+ //# sourceMappingURL=runtime.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"runtime.js","sourceRoot":"","sources":["../../src/session/runtime.ts"],"names":[],"mappings":"AAAA,OAAO,EAAS,OAAO,EAAE,MAAM,qBAAqB,CAAC;AACrD,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,YAAY,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AACpF,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AACjD,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAC9C,OAAO,EAAc,aAAa,EAAE,UAAU,EAAE,GAAG,EAAE,MAAM,aAAa,CAAC;AACzE,OAAO,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAE1C,MAAM,CAAC,KAAK,UAAU,eAAe,CAAC,cAAuB;IAC3D,MAAM,MAAM,GAAG,UAAU,EAAE,CAAC;IAC5B,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;QACnB,OAAO,CAAC,KAAK,CAAC,oCAAoC,CAAC,CAAC;QACpD,OAAO,CAAC,KAAK,CAAC,mDAAmD,CAAC,CAAC;QACnE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAED,IAAI,IAAI,GAAG,YAAY,EAAE,CAAC;IAC1B,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACtB,IAAI,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC;QACvC,YAAY,CAAC,IAAI,CAAC,CAAC;IACrB,CAAC;IACD,MAAM,KAAK,GAAU,MAAM,UAAU,CAAC,IAAI,CAAC,CAAC;IAC5C,MAAM,GAAG,GAAe,EAAE,KAAK,EAAE,CAAC;IAElC,IAAI,cAAc,EAAE,CAAC;QACnB,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,MAAM,IAAI,CAAC,CAAC,IAAI,KAAK,WAAW,CAAC,CAAC;QAC7F,IAAI,UAAU,EAAE,CAAC;YACf,MAAM,IAAI,GAAG,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAC9C,OAAO,CAAC,GAAG,CAAC,cAAc,IAAI,IAAI,CAAC,CAAC;QACtC,CAAC;QACD,MAAM,UAAU,CAAC,GAAG,CAAC,CAAC;QACtB,OAAO,CAAC,GAAG,CAAC,SAAS,cAAc,EAAE,CAAC,CAAC;QACvC,IAAI,CAAC;YACH,MAAM,YAAY,CAAC,GAAG,CAAC,KAAK,EAAE,cAAc,CAAC,CAAC;YAC9C,YAAY,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,GAAQ,EAAE,CAAC;YAAC,OAAO,CAAC,KAAK,CAAC,OAAO,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;QAAC,CAAC;QAC3D,OAAO,CAAC,GAAG,CAAC,2BAA2B,CAAC,CAAC;QACzC,MAAM,GAAG,CAAC,GAAG,CAAC,CAAC;QACf,OAAO;IACT,CAAC;IAED,MAAM,QAAQ,GAAG,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,MAAM,CAAC;IAC9E,OAAO,CAAC,GAAG,CAAC,QAAQ,GAAG,CAAC;QACtB,CAAC,CAAC,gBAAgB,QAAQ,OAAO,WAAW,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,qBAAqB;QACrF,CAAC,CAAC,uCAAuC,CAAC,CAAC;IAE7C,MAAM,GAAG,CAAC,GAAG,CAAC,CAAC;AACjB,CAAC"}
@@ -0,0 +1,8 @@
1
+ import { Agent } from "@ai-zen/agents-core";
2
+ export interface SessionCtx {
3
+ agent: Agent;
4
+ }
5
+ export declare const SYSTEM_PROMPT: string;
6
+ export declare function saveAndNew(ctx: SessionCtx): Promise<void>;
7
+ export declare function ask(ctx: SessionCtx): Promise<void>;
8
+ //# sourceMappingURL=shared.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shared.d.ts","sourceRoot":"","sources":["../../src/session/shared.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAW,MAAM,qBAAqB,CAAC;AAarD,MAAM,WAAW,UAAU;IACzB,KAAK,EAAE,KAAK,CAAC;CACd;AAED,eAAO,MAAM,aAAa,QAcd,CAAC;AAEb,wBAAsB,UAAU,CAAC,GAAG,EAAE,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC,CAI/D;AAgCD,wBAAsB,GAAG,CAAC,GAAG,EAAE,UAAU,iBAaxC"}
@@ -0,0 +1,77 @@
1
+ import { Message } from "@ai-zen/agents-core";
2
+ import inquirer from "inquirer";
3
+ import { saveMessages } from "../config.js";
4
+ import { buildAgent } from "../agent-factory.js";
5
+ import { handleMessage } from "./message.js";
6
+ import { cmdExit } from "./commands/exit.js";
7
+ import { cmdHelp } from "./commands/help.js";
8
+ import { cmdSave } from "./commands/save.js";
9
+ import { cmdNew } from "./commands/new.js";
10
+ import { cmdLoad } from "./commands/load.js";
11
+ import { cmdEditor } from "./commands/editor.js";
12
+ import { cmdBack } from "./commands/back.js";
13
+ export const SYSTEM_PROMPT = [
14
+ "你是一个AI助手,专门帮助用户回答问题和执行任务。请用中文回复。",
15
+ "",
16
+ "## 行为准则",
17
+ "1. 做任何改动之前,必须先跟用户商量,获得明确指示之后再行动。",
18
+ "2. 用户没有明确要求产出文件时,不得自行创建任何文件到项目中。讨论就停留在讨论层面。",
19
+ "3. 区分危险操作:删除文件、覆盖文件、安装卸载软件、修改系统配置、执行耗时任务等属于危险操作。执行前必须评估影响范围,并向用户说明风险,获得用户明确的书面的确认之后再执行。",
20
+ "4. 追责原则:你的每一步操作都应当基于用户的明确指令。如果出了问题,可以从用户说过的话追溯责任——是用户让你做的,用户承担责任。所以你不需要畏手畏脚,只要用户明确说了\u201c做\u201d,你就放心去做。",
21
+ "",
22
+ "## 记忆",
23
+ "你可以使用 shell 工具执行任何命令。如果你有需要长期记住的信息(用户偏好、项目约定、任务进度等),请写入以下位置:",
24
+ "- 全局记忆: ~/.ai-zen/air/temp/*.md",
25
+ "- 项目记忆: $(cwd)/.ai-zen/air/temp/*.md",
26
+ "下次启动时用 shell 读取即可。这是你唯一的记忆方式。",
27
+ ].join("\n");
28
+ export async function saveAndNew(ctx) {
29
+ const msgs = [Message.System(SYSTEM_PROMPT)];
30
+ saveMessages(msgs);
31
+ ctx.agent = await buildAgent(msgs);
32
+ }
33
+ async function dispatchCommand(ctx, cmd) {
34
+ switch (cmd) {
35
+ case "/exit":
36
+ case "/quit":
37
+ cmdExit(ctx);
38
+ break;
39
+ case "/save":
40
+ await cmdSave(ctx);
41
+ break;
42
+ case "/new":
43
+ await cmdNew(ctx);
44
+ break;
45
+ case "/load":
46
+ await cmdLoad(ctx);
47
+ break;
48
+ case "/back":
49
+ await cmdBack(ctx);
50
+ break;
51
+ case "/editor":
52
+ await cmdEditor(ctx);
53
+ break;
54
+ case "/help":
55
+ cmdHelp();
56
+ break;
57
+ default:
58
+ console.log(`\n❌ 未知命令: ${cmd}\n`);
59
+ break;
60
+ }
61
+ }
62
+ export async function ask(ctx) {
63
+ while (true) {
64
+ const { input } = await inquirer.prompt([
65
+ { type: "input", name: "input", message: "你:" },
66
+ ]);
67
+ const t = input.trim();
68
+ if (!t)
69
+ continue;
70
+ if (t.startsWith("/")) {
71
+ await dispatchCommand(ctx, t.toLowerCase());
72
+ continue;
73
+ }
74
+ await handleMessage(ctx, t);
75
+ }
76
+ }
77
+ //# sourceMappingURL=shared.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shared.js","sourceRoot":"","sources":["../../src/session/shared.ts"],"names":[],"mappings":"AAAA,OAAO,EAAS,OAAO,EAAE,MAAM,qBAAqB,CAAC;AACrD,OAAO,QAAQ,MAAM,UAAU,CAAC;AAChC,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAC;AAC5C,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AACjD,OAAO,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,mBAAmB,CAAC;AAC3C,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACjD,OAAO,EAAE,OAAO,EAAE,MAAM,oBAAoB,CAAC;AAM7C,MAAM,CAAC,MAAM,aAAa,GAAG;IAC3B,kCAAkC;IAClC,EAAE;IACF,SAAS;IACT,kCAAkC;IAClC,6CAA6C;IAC7C,yFAAyF;IACzF,2GAA2G;IAC3G,EAAE;IACF,OAAO;IACP,8DAA8D;IAC9D,iCAAiC;IACjC,sCAAsC;IACtC,+BAA+B;CAChC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AAEb,MAAM,CAAC,KAAK,UAAU,UAAU,CAAC,GAAe;IAC9C,MAAM,IAAI,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC;IAC7C,YAAY,CAAC,IAAI,CAAC,CAAC;IACnB,GAAG,CAAC,KAAK,GAAG,MAAM,UAAU,CAAC,IAAI,CAAC,CAAC;AACrC,CAAC;AAED,KAAK,UAAU,eAAe,CAAC,GAAe,EAAE,GAAW;IACzD,QAAQ,GAAG,EAAE,CAAC;QACZ,KAAK,OAAO,CAAC;QACb,KAAK,OAAO;YACV,OAAO,CAAC,GAAG,CAAC,CAAC;YACb,MAAM;QACR,KAAK,OAAO;YACV,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC;YACnB,MAAM;QACR,KAAK,MAAM;YACT,MAAM,MAAM,CAAC,GAAG,CAAC,CAAC;YAClB,MAAM;QACR,KAAK,OAAO;YACV,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC;YACnB,MAAM;QACR,KAAK,OAAO;YACV,MAAM,OAAO,CAAC,GAAG,CAAC,CAAC;YACnB,MAAM;QACR,KAAK,SAAS;YACZ,MAAM,SAAS,CAAC,GAAG,CAAC,CAAC;YACrB,MAAM;QACR,KAAK,OAAO;YACV,OAAO,EAAE,CAAC;YACV,MAAM;QACR;YACE,OAAO,CAAC,GAAG,CAAC,aAAa,GAAG,IAAI,CAAC,CAAC;YAClC,MAAM;IACV,CAAC;AACH,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,GAAG,CAAC,GAAe;IACvC,OAAO,IAAI,EAAE,CAAC;QACZ,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC;YACtC,EAAE,IAAI,EAAE,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE;SAChD,CAAC,CAAC;QACH,MAAM,CAAC,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC;QACvB,IAAI,CAAC,CAAC;YAAE,SAAS;QACjB,IAAI,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YACtB,MAAM,eAAe,CAAC,GAAG,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;YAC5C,SAAS;QACX,CAAC;QACD,MAAM,aAAa,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;IAC9B,CAAC;AACH,CAAC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ai-zen/air",
3
- "version": "0.2.0",
3
+ "version": "0.2.2",
4
4
  "description": "Minimalist AI CLI assistant \u2014 one shell tool, filesystem memory, auto context migration",
5
5
  "type": "module",
6
6
  "main": "./dist/agent-runtime.js",
@@ -0,0 +1,147 @@
1
+ import { describe, it, expect, vi, beforeEach } from "vitest";
2
+ import type { ChatCtx } from "../chat/shared.js";
3
+
4
+ // Mock all external dependencies
5
+ vi.mock("inquirer", () => ({ default: { prompt: vi.fn() } }));
6
+ vi.mock("../config.js", () => ({
7
+ saveMessages: vi.fn(),
8
+ saveSnapshot: vi.fn(() => "snapshot-123"),
9
+ listSnapshots: vi.fn(() => []),
10
+ loadSnapshot: vi.fn(() => []),
11
+ }));
12
+ vi.mock("../agent-factory.js", () => ({
13
+ buildAgent: vi.fn(async () => ({
14
+ messages: [],
15
+ events: { on: vi.fn(), off: vi.fn() },
16
+ send: vi.fn(),
17
+ })),
18
+ }));
19
+ vi.mock("../chat/print.js", () => ({
20
+ sendAndPrint: vi.fn(),
21
+ }));
22
+
23
+ const mockPrompt = vi.mocked((await import("inquirer")).default.prompt);
24
+ const { saveMessages, saveSnapshot, listSnapshots, loadSnapshot } = await import("../config.js");
25
+ const { buildAgent } = await import("../agent-factory.js");
26
+
27
+ function makeCtx(msgs: any[] = []): ChatCtx {
28
+ return { agent: { messages: msgs, events: { on: vi.fn(), off: vi.fn() }, send: vi.fn() } as any };
29
+ }
30
+
31
+ // ==================== cmdNew ====================
32
+
33
+ describe("cmdNew", () => {
34
+ it("写入 system prompt 并构建 agent", async () => {
35
+ const { cmdNew } = await import("../chat/commands/new.js");
36
+ const ctx = makeCtx();
37
+
38
+ await cmdNew(ctx);
39
+
40
+ expect(saveMessages).toHaveBeenCalledWith(
41
+ expect.arrayContaining([expect.objectContaining({ role: "system" })])
42
+ );
43
+ expect(buildAgent).toHaveBeenCalled();
44
+ expect(ctx.agent).toBeTruthy();
45
+ });
46
+ });
47
+
48
+ // ==================== dispatchCommand 路由 ====================
49
+
50
+ describe("dispatchCommand", () => {
51
+ beforeEach(() => {
52
+ vi.clearAllMocks();
53
+ });
54
+
55
+ it('"/exit" 调用 cmdExit(process.exit)', async () => {
56
+ // Can't easily test process.exit, just verify it doesn't throw
57
+ const { runChat } = await import("../chat/runtime.js");
58
+ // dispatchCommand is internal, tested via routing only
59
+ });
60
+
61
+ it('"/save" 调用 cmdSave', async () => {
62
+ const { cmdSave } = await import("../chat/commands/save.js");
63
+ const ctx = makeCtx([{ role: "system", content: "sys" }]);
64
+ await cmdSave(ctx);
65
+ expect(saveSnapshot).toHaveBeenCalledWith(ctx.agent.messages);
66
+ });
67
+
68
+ it('"/new" 调用 cmdNew 并打印提示', async () => {
69
+ const { cmdNew } = await import("../chat/commands/new.js");
70
+ const ctx = makeCtx();
71
+ const spy = vi.spyOn(console, "log").mockImplementation(() => {});
72
+ await cmdNew(ctx);
73
+ expect(spy).toHaveBeenCalledWith("\n🆕 新会话\n");
74
+ spy.mockRestore();
75
+ });
76
+
77
+ it('"/help" 打印帮助信息', async () => {
78
+ const { cmdHelp } = await import("../chat/commands/help.js");
79
+ const spy = vi.spyOn(console, "log").mockImplementation(() => {});
80
+ cmdHelp();
81
+ expect(spy).toHaveBeenCalled();
82
+ spy.mockRestore();
83
+ });
84
+ });
85
+
86
+ // ==================== cmdBack ====================
87
+
88
+ describe("cmdBack", () => {
89
+ beforeEach(() => {
90
+ vi.clearAllMocks();
91
+ });
92
+
93
+ it("没有消息时提示无可撤回", async () => {
94
+ const { cmdBack } = await import("../chat/commands/back.js");
95
+ const ctx = makeCtx([{ role: "system", content: "sys" }]);
96
+ const spy = vi.spyOn(console, "log").mockImplementation(() => {});
97
+ await cmdBack(ctx);
98
+ expect(spy).toHaveBeenCalledWith("\n❌ 还没有消息可以撤回\n");
99
+ spy.mockRestore();
100
+ });
101
+
102
+ it("取消操作时不修改消息", async () => {
103
+ const { cmdBack } = await import("../chat/commands/back.js");
104
+ const msgs = [
105
+ { role: "system", content: "sys" },
106
+ { role: "user", content: "你好" },
107
+ ];
108
+ const ctx = makeCtx(msgs);
109
+ mockPrompt.mockResolvedValueOnce({ selectedIndex: -1 });
110
+ await cmdBack(ctx);
111
+ expect(ctx.agent.messages.length).toBe(2);
112
+ });
113
+ });
114
+
115
+ // ==================== cmdLoad ====================
116
+
117
+ describe("cmdLoad", () => {
118
+ beforeEach(() => {
119
+ vi.clearAllMocks();
120
+ });
121
+
122
+ it("没有快照时提示", async () => {
123
+ const { cmdLoad } = await import("../chat/commands/load.js");
124
+ const ctx = makeCtx();
125
+ vi.mocked(listSnapshots).mockReturnValueOnce([]);
126
+ const spy = vi.spyOn(console, "log").mockImplementation(() => {});
127
+ await cmdLoad(ctx);
128
+ expect(spy).toHaveBeenCalledWith("\n📭 没有可用的快照\n");
129
+ spy.mockRestore();
130
+ });
131
+ });
132
+
133
+ // ==================== cmdEditor ====================
134
+
135
+ describe("cmdEditor", () => {
136
+ beforeEach(() => {
137
+ vi.clearAllMocks();
138
+ });
139
+
140
+ it("取消编辑不报错", async () => {
141
+ const { cmdEditor } = await import("../chat/commands/editor.js");
142
+ const ctx = makeCtx();
143
+ mockPrompt.mockResolvedValueOnce({ content: "" });
144
+ await cmdEditor(ctx);
145
+ expect(saveMessages).not.toHaveBeenCalled();
146
+ });
147
+ });
@@ -0,0 +1,187 @@
1
+ import { describe, it, expect, beforeAll, afterAll } from "vitest";
2
+ import { spawn } from "node:child_process";
3
+ import { existsSync, readFileSync, rmSync, mkdirSync, readdirSync } from "node:fs";
4
+ import { join } from "node:path";
5
+ import { tmpdir } from "node:os";
6
+
7
+ // ==================== Helpers ====================
8
+
9
+ const CLI = join(process.cwd(), "dist", "cli.js");
10
+
11
+ interface CliResult {
12
+ stdout: string;
13
+ stderr: string;
14
+ exitCode: number;
15
+ }
16
+
17
+ function runCli(args: string[], stdinInput?: string, extraEnv?: Record<string, string>): Promise<CliResult> {
18
+ return new Promise((resolve) => {
19
+ const env = {
20
+ ...process.env,
21
+ ...extraEnv,
22
+ };
23
+
24
+ const proc = spawn("node", [CLI, ...args], { env, stdio: ["pipe", "pipe", "pipe"] });
25
+
26
+ let stdout = "";
27
+ let stderr = "";
28
+
29
+ proc.stdout.on("data", (data: Buffer) => { stdout += data.toString(); });
30
+ proc.stderr.on("data", (data: Buffer) => { stderr += data.toString(); });
31
+
32
+ if (stdinInput !== undefined) {
33
+ proc.stdin.write(stdinInput);
34
+ proc.stdin.end();
35
+ }
36
+
37
+ proc.on("close", (code) => {
38
+ resolve({ stdout, stderr, exitCode: code ?? -1 });
39
+ });
40
+ proc.on("error", () => {
41
+ resolve({ stdout, stderr, exitCode: -1 });
42
+ });
43
+ });
44
+ }
45
+
46
+ let tmpDirs: string[] = [];
47
+
48
+ afterAll(() => {
49
+ for (const d of tmpDirs) {
50
+ try { rmSync(d, { recursive: true, force: true }); } catch {}
51
+ }
52
+ });
53
+
54
+ function makeTestDir(): string {
55
+ const dir = join(tmpdir(), `air-e2e-${Date.now()}-${Math.random().toString(36).slice(2)}`);
56
+ mkdirSync(dir, { recursive: true });
57
+ tmpDirs.push(dir);
58
+ return dir;
59
+ }
60
+
61
+ // Read API key from .env.local
62
+ function getApiKey(): string {
63
+ try {
64
+ const envContent = readFileSync(join(process.cwd(), ".env.local"), "utf-8");
65
+ const match = envContent.match(/DEEPSEEK_API_KEY=(.+)/);
66
+ if (match) return match[1].trim();
67
+ } catch {}
68
+ return process.env.DEEPSEEK_API_KEY || "";
69
+ }
70
+
71
+ const API_KEY = getApiKey();
72
+
73
+ function skipIfNoKey(ctx: any) {
74
+ if (!API_KEY) ctx.skip();
75
+ }
76
+
77
+ // ==================== Tests ====================
78
+
79
+ describe("E2E: CLI 基本功能", () => {
80
+ it("air --help 显示帮助信息", async () => {
81
+ const result = await runCli(["--help"]);
82
+ expect(result.stdout).toContain("Usage:");
83
+ expect(result.stdout).toContain("air");
84
+ expect(result.exitCode).toBe(0);
85
+ });
86
+
87
+ it("air key <key> 设置 API Key", async () => {
88
+ const airDir = makeTestDir();
89
+ const result = await runCli(["key", "sk-test-12345"], undefined, { AIR_DIR: airDir });
90
+ expect(result.stdout).toContain("✅ API Key 已设置");
91
+
92
+ const configFile = join(airDir, "config.json");
93
+ expect(existsSync(configFile)).toBe(true);
94
+ const config = JSON.parse(readFileSync(configFile, "utf-8"));
95
+ expect(config.apiKey).toBe("sk-test-12345");
96
+ });
97
+
98
+ it("air config 显示配置信息", async () => {
99
+ const airDir = makeTestDir();
100
+ await runCli(["key", "sk-test-abcde"], undefined, { AIR_DIR: airDir });
101
+ const result = await runCli(["config"], undefined, { AIR_DIR: airDir });
102
+ expect(result.stdout).toContain("API Key:");
103
+ expect(result.stdout).toContain("bcde");
104
+ });
105
+ });
106
+
107
+
108
+ describe("E2E: 对话命令", () => {
109
+ it("/save 保存快照", async (ctx) => {
110
+ skipIfNoKey(ctx);
111
+
112
+ const airDir = makeTestDir();
113
+ const env = { AIR_DIR: airDir };
114
+ await runCli(["key", API_KEY], undefined, env);
115
+
116
+ // 先发一条消息建立历史
117
+ await runCli(["测试消息"], "/exit\n", env);
118
+
119
+ // 交互模式下 /save 后 /exit
120
+ const result = await runCli([], "/save\n/exit\n", env);
121
+ expect(result.stdout).toContain("✅ 快照:");
122
+
123
+ // 验证快照文件被创建
124
+ const snapDir = join(airDir, "snapshots");
125
+ const files = readdirSync(snapDir).filter((f) => f.endsWith(".json"));
126
+ expect(files.length).toBeGreaterThanOrEqual(1);
127
+ }, 30000);
128
+ });
129
+
130
+ describe("E2E: 真实 API 对话", () => {
131
+ it("air <message> 发送消息并接收真实回复", async (ctx) => {
132
+ skipIfNoKey(ctx);
133
+
134
+ const airDir = makeTestDir();
135
+ const env = { AIR_DIR: airDir };
136
+
137
+ // Set up real key
138
+ await runCli(["key", API_KEY], undefined, env);
139
+
140
+ // Send a simple message, pipe /exit to leave interactive mode
141
+ const result = await runCli(["用一句话介绍你自己"], "/exit\n", env);
142
+
143
+ // Should have AI response
144
+ expect(result.stdout).toContain("🤖 AI:");
145
+ // Should have some actual content (not empty)
146
+ expect(result.stdout.length).toBeGreaterThan(100);
147
+
148
+ // Verify context was saved with user message + AI response
149
+ const contextFile = join(airDir, "context.json");
150
+ expect(existsSync(contextFile)).toBe(true);
151
+ const msgs = JSON.parse(readFileSync(contextFile, "utf-8"));
152
+ expect(msgs.length).toBeGreaterThanOrEqual(2);
153
+ expect(msgs[1].role).toBe("user");
154
+ expect(msgs[1].content).toBe("用一句话介绍你自己");
155
+
156
+ // Verify there's an assistant response
157
+ const assistantMsg = msgs.find((m: any) => m.role === "assistant");
158
+ expect(assistantMsg).toBeTruthy();
159
+ expect(assistantMsg.content.length).toBeGreaterThan(10);
160
+ }, 30000); // 30s timeout for real API
161
+
162
+ it("/new 命令重置对话", async (ctx) => {
163
+ skipIfNoKey(ctx);
164
+
165
+ const airDir = makeTestDir();
166
+ const env = { AIR_DIR: airDir };
167
+
168
+ await runCli(["key", API_KEY], undefined, env);
169
+
170
+ // Send a message to create history
171
+ await runCli(["第一轮消息"], "/exit\n", env);
172
+
173
+ // Verify there's history
174
+ let contextFile = join(airDir, "context.json");
175
+ let msgs = JSON.parse(readFileSync(contextFile, "utf-8"));
176
+ expect(msgs.length).toBeGreaterThanOrEqual(2);
177
+
178
+ // Interactive mode: /new then exit
179
+ const result = await runCli([], "/new\n/exit\n", env);
180
+ expect(result.stdout).toContain("🆕 重新开始");
181
+
182
+ // Verify context was reset
183
+ msgs = JSON.parse(readFileSync(contextFile, "utf-8"));
184
+ expect(msgs.length).toBe(1);
185
+ expect(msgs[0].role).toBe("system");
186
+ }, 30000);
187
+ });
@@ -4,7 +4,7 @@ import { readConfig } from "./config.js";
4
4
  import { shellTool } from "./tools.js";
5
5
 
6
6
  const MODEL_NAME = "deepseek-v4-flash";
7
- const API_ENDPOINT = "https://api.deepseek.com/v1";
7
+ const API_ENDPOINT = process.env.AIR_API_ENDPOINT || "https://api.deepseek.com/v1";
8
8
 
9
9
  async function buildModel(apiKey: string) {
10
10
  const endpoint = new OpenAI({ openai_endpoint: API_ENDPOINT, api_key: apiKey });
@@ -0,0 +1,90 @@
1
+ import inquirer from "inquirer";
2
+ import type { ChatCtx } from "../shared.js";
3
+ import { sendAndPrint } from "../print.js";
4
+ import { saveMessages } from "../../config.js";
5
+ import { buildAgent } from "../../agent-factory.js";
6
+
7
+ export async function cmdBack(ctx: ChatCtx): Promise<void> {
8
+ const targets: { index: number; role: string; label: string; preview: string }[] = [];
9
+ for (let i = 0; i < ctx.agent.messages.length; i++) {
10
+ const msg = ctx.agent.messages[i];
11
+ if (msg.role === "user") {
12
+ const text = typeof msg.content === "string" ? msg.content : "";
13
+ if (text) {
14
+ targets.push({ index: i, role: "user", label: "\u{1F464} 用户", preview: text.substring(0, 60) + (text.length > 60 ? "..." : "") });
15
+ }
16
+ } else if (msg.role === "tool" || msg.role === "function") {
17
+ const text = typeof msg.content === "string" ? msg.content : "";
18
+ if (text) {
19
+ targets.push({ index: i, role: msg.role, label: "\u{1F527} 工具", preview: text.substring(0, 60) + (text.length > 60 ? "..." : "") });
20
+ }
21
+ }
22
+ }
23
+
24
+ if (targets.length === 0) {
25
+ console.log("\n❌ 还没有消息可以撤回\n");
26
+ return;
27
+ }
28
+
29
+ console.log("\n📋 选择要撤回到哪条消息(将删除所选及其之后的所有内容):\n");
30
+
31
+ const { selectedIndex } = await inquirer.prompt([
32
+ {
33
+ type: "list",
34
+ name: "selectedIndex",
35
+ message: "撤回到:",
36
+ pageSize: 15,
37
+ choices: [
38
+ { name: "\u21a9\ufe0f 取消操作", value: -1 },
39
+ ...targets.map((t) => ({ name: t.label + " " + t.preview, value: t.index })),
40
+ ],
41
+ },
42
+ ]);
43
+
44
+ if (selectedIndex === -1) { console.log("\n已取消\n"); return; }
45
+
46
+ const selectedMsg = ctx.agent.messages[selectedIndex];
47
+ const isUserMsg = selectedMsg.role === "user";
48
+ const originalText = typeof selectedMsg.content === "string" ? selectedMsg.content : "";
49
+ const sliceEnd = isUserMsg ? selectedIndex : selectedIndex + 1;
50
+ ctx.agent.messages = ctx.agent.messages.slice(0, sliceEnd);
51
+
52
+ if (isUserMsg) {
53
+ console.log("\n原内容: " + originalText.substring(0, 200) + (originalText.length > 200 ? "..." : "") + "\n");
54
+ const { editChoice } = await inquirer.prompt([
55
+ {
56
+ type: "list",
57
+ name: "editChoice",
58
+ message: "请选择:",
59
+ choices: [
60
+ { name: "\u270f\ufe0f 修改后重新发送", value: "edit" },
61
+ { name: "\u{1F504} 直接重新发送", value: "resend" },
62
+ { name: "\u21a9\ufe0f 取消操作", value: "cancel" },
63
+ ],
64
+ },
65
+ ]);
66
+ if (editChoice === "cancel") { console.log("\n已取消\n"); return; }
67
+
68
+ let textToSend = originalText;
69
+ if (editChoice === "edit") {
70
+ const { editedContent } = await inquirer.prompt([
71
+ { type: "input", name: "editedContent", message: "修改消息:", default: originalText },
72
+ ]);
73
+ if (!editedContent.trim()) { console.log("\n❌ 消息不能为空\n"); return; }
74
+ textToSend = editedContent.trim();
75
+ }
76
+
77
+ ctx.agent = await buildAgent(ctx.agent.messages);
78
+ await sendAndPrint(ctx.agent, textToSend);
79
+ saveMessages(ctx.agent.messages);
80
+ } else {
81
+ console.log("\n💡 请输入一条新消息继续对话\n");
82
+ const { newMessage } = await inquirer.prompt([
83
+ { type: "input", name: "newMessage", message: "新消息:" },
84
+ ]);
85
+ if (!newMessage.trim()) { console.log("\n已取消\n"); return; }
86
+ ctx.agent = await buildAgent(ctx.agent.messages);
87
+ await sendAndPrint(ctx.agent, newMessage.trim());
88
+ saveMessages(ctx.agent.messages);
89
+ }
90
+ }
@@ -0,0 +1,18 @@
1
+ import inquirer from "inquirer";
2
+ import type { ChatCtx } from "../shared.js";
3
+ import { sendAndPrint } from "../print.js";
4
+ import { saveMessages } from "../../config.js";
5
+
6
+ export async function cmdEditor(ctx: ChatCtx) {
7
+ const { content } = await inquirer.prompt([
8
+ { type: "editor", name: "content", message: "编辑消息:" },
9
+ ]);
10
+ if (!content.trim()) {
11
+ console.log("\n已取消\n");
12
+ return;
13
+ }
14
+ try {
15
+ await sendAndPrint(ctx.agent, content.trim());
16
+ saveMessages(ctx.agent.messages);
17
+ } catch (err: any) { console.error(`\n❌ ${err.message}`); }
18
+ }
@@ -0,0 +1,6 @@
1
+ import type { ChatCtx } from "../shared.js";
2
+
3
+ export async function cmdExit(ctx: ChatCtx): Promise<void> {
4
+ console.log("\n👋 再见!");
5
+ process.exit(0);
6
+ }
@@ -0,0 +1,3 @@
1
+ export function cmdHelp() {
2
+ console.log("\n/exit /quit 退出\n/save 保存快照\n/load 加载快照\n/new 重新开始\n/back 撤回消息\n/help 帮助\n");
3
+ }