@leviyuan/lodestar 0.8.2 → 0.9.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.md CHANGED
@@ -53,6 +53,7 @@ lodestar-setup
53
53
  | `clear` / `cl` | 用状态卡展示杀进程并开新 thread(等价 `/clear`)|
54
54
  | `compact` / `cm` | 主动触发当前 thread 的上下文压缩,完成后状态卡收束 |
55
55
  | `model` / `md` | 展示可用 Codex 模型面板,先选模型再选 reasoning effort,并按群持久化 |
56
+ | `task` | 打开项目任务清单面板,启用飞书任务清单自动化(预览版) |
56
57
 
57
58
  **并发 worktree 群**
58
59
 
@@ -69,6 +70,22 @@ lodestar-setup
69
70
 
70
71
  ## 🎁 附加能力
71
72
 
73
+ ### 📋 飞书任务清单自动化(预览版)
74
+
75
+ 在项目群发 `task`,卡片点 `启用`,夜航星会创建并绑定一个 `<project>[lodestar]` 飞书任务清单。清单分组固定为:
76
+
77
+ | 分组 | 用途 |
78
+ | --- | --- |
79
+ | `设计中` | 默认分组;放需求、想法或待讨论任务,Codex 和 agy 会各写一条规划评论。 |
80
+ | `[AI]待执行` | 放已经准备交给 AI 实现的任务;worker 会挑一个进入执行。 |
81
+ | `[AI]执行中` | 当前自动执行中的任务;Codex 在本地 worktree 修改代码并生成本地审查请求。 |
82
+ | `[AI]待审核` | agy 审核后的任务;人工确认后勾选完成,触发 Codex 本地合并。 |
83
+ | `已完成` | 合并确认后的任务。 |
84
+
85
+ 这个能力目前是预览版。worker 启动后会定时扫描绑定清单,一次只推进一个任务。所有规划、执行、审核、合并结果都会写回任务评论区;失败不会静默吞掉,会在评论或日志里暴露原因。
86
+
87
+ `task` 面板里的 `删` 会二次确认,确认后删除整个清单和清单内任务。这个能力需要飞书应用开通任务清单/任务/评论相关权限;缺权限时面板会显示 Open API 返回的失败原因和缺失 scope。
88
+
72
89
  ### 🔔 HTTP 通知端点
73
90
 
74
91
  本机任何脚本一行 curl 就能往群里推一张 markdown 卡片(info / warn / error 三档染色):
@@ -1,13 +1,13 @@
1
1
  #!/usr/bin/env node
2
- import{execSync as A,spawn as b}from"node:child_process";import{existsSync as g,mkdirSync as B,writeFileSync as _}from"node:fs";import{createInterface as j}from"node:readline/promises";import{delimiter as m,dirname as x,join as w}from"node:path";import{fileURLToPath as D}from"node:url";import{homedir as S}from"node:os";import{join as Q}from"node:path";var f=S(),L=process.platform==="win32";function G($,q,J){if($)return $;if(q)return Q(q,"lodestar");return J}function v(){if(L)return Q(process.env.APPDATA??Q(f,"AppData","Roaming"),"Lodestar");return Q(f,".config","lodestar")}function T(){if(L)return Q(process.env.LOCALAPPDATA??Q(f,"AppData","Local"),"Lodestar");return Q(f,".local","share","lodestar")}var N=G(process.env.LODESTAR_CONFIG_DIR,process.env.XDG_CONFIG_HOME,v()),Y=G(process.env.LODESTAR_DATA_DIR,process.env.XDG_DATA_HOME,T()),W=process.env.LODESTAR_CONFIG??Q(N,"config.toml"),d=Q(Y,"daemon.pid"),o=Q(Y,"daemon.log"),i=Q(Y,"session-chat-map.json"),n=Q(Y,"session-resume-map.json"),e=Q(Y,"session-model-map.json"),C=Q(Y,"alive-on-shutdown.json"),zz=Q(Y,"inbox"),$z=Q(Y,"debug.sock"),qz=Q(Y,"debug-context.json");var z={reset:"\x1B[0m",bold:"\x1B[1m",dim:"\x1B[2m",green:"\x1B[32m",yellow:"\x1B[33m",cyan:"\x1B[36m",red:"\x1B[31m"},Z=j({input:process.stdin,output:process.stdout});async function V($,q={}){while(!0){let J=q.default?`${z.dim} [${q.default}]${z.reset}`:"",K=(await Z.question(`${z.cyan}? ${z.reset}${$}${J}
3
- ${z.green}>${z.reset} `)).trim();if(!K&&q.default!==void 0)return q.default;if(!K&&q.required){console.log(`${z.red}必填,请重新输入${z.reset}`);continue}return K}}function h($){let q="═".repeat(58);console.log(`
2
+ import{execSync as B,spawn as b}from"node:child_process";import{existsSync as g,mkdirSync as A,writeFileSync as _}from"node:fs";import{createInterface as j}from"node:readline/promises";import{delimiter as m,dirname as x,join as w}from"node:path";import{fileURLToPath as h}from"node:url";import{homedir as S}from"node:os";import{join as Q}from"node:path";var f=S(),L=process.platform==="win32";function G($,q,J){if($)return $;if(q)return Q(q,"lodestar");return J}function v(){if(L)return Q(process.env.APPDATA??Q(f,"AppData","Roaming"),"Lodestar");return Q(f,".config","lodestar")}function T(){if(L)return Q(process.env.LOCALAPPDATA??Q(f,"AppData","Local"),"Lodestar");return Q(f,".local","share","lodestar")}var N=G(process.env.LODESTAR_CONFIG_DIR,process.env.XDG_CONFIG_HOME,v()),Y=G(process.env.LODESTAR_DATA_DIR,process.env.XDG_DATA_HOME,T()),W=process.env.LODESTAR_CONFIG??Q(N,"config.toml"),d=Q(Y,"daemon.pid"),o=Q(Y,"daemon.log"),i=Q(Y,"session-chat-map.json"),n=Q(Y,"session-resume-map.json"),e=Q(Y,"session-model-map.json"),C=Q(Y,"tasklist-map.json"),zz=Q(Y,"alive-on-shutdown.json"),$z=Q(Y,"inbox"),qz=Q(Y,"debug.sock"),Jz=Q(Y,"debug-context.json");var z={reset:"\x1B[0m",bold:"\x1B[1m",dim:"\x1B[2m",green:"\x1B[32m",yellow:"\x1B[33m",cyan:"\x1B[36m",red:"\x1B[31m"},Z=j({input:process.stdin,output:process.stdout});async function V($,q={}){while(!0){let J=q.default?`${z.dim} [${q.default}]${z.reset}`:"",K=(await Z.question(`${z.cyan}? ${z.reset}${$}${J}
3
+ ${z.green}>${z.reset} `)).trim();if(!K&&q.default!==void 0)return q.default;if(!K&&q.required){console.log(`${z.red}必填,请重新输入${z.reset}`);continue}return K}}function D($){let q="═".repeat(58);console.log(`
4
4
  ${z.bold}${z.cyan}╔${q}╗${z.reset}`),console.log(`${z.bold}${z.cyan}║ ${$.padEnd(54)} ║${z.reset}`),console.log(`${z.bold}${z.cyan}╚${q}╝${z.reset}
5
5
  `)}function M($,q,J){console.log(`
6
6
  ${z.bold}${z.yellow}[${$}/${q}] ${J}${z.reset}
7
- `)}function P($){return $.replace(/\\/g,"\\\\").replace(/"/g,"\\\"")}function E($){let q=process.env.PATH??"";if(!q)return null;let J=process.platform==="win32"?[`${$}.cmd`,`${$}.bat`,`${$}.exe`,$]:[$];for(let K of q.split(m)){if(!K)continue;for(let H of J){let X=w(K,H);if(g(X))return X}}return null}async function I(){let $=process.platform==="win32"?"npm.cmd":"npm";return new Promise((q)=>{let J=b($,["install","-g","@openai/codex@latest"],{stdio:"inherit",shell:process.platform==="win32"});J.on("exit",(K)=>q(K===0)),J.on("error",()=>q(!1))})}function k($){try{let q=A(`"${$}" login status 2>&1`,{timeout:1e4}).toString();return/Logged in using ChatGPT/i.test(q)}catch{return!1}}async function c($){return new Promise((q)=>{let J=b($,["login"],{stdio:"inherit",shell:process.platform==="win32"});J.on("exit",(K)=>q(K===0)),J.on("error",()=>q(!1))})}function l($){try{if(process.platform==="win32")b(process.env.ComSpec??"cmd.exe",["/c","start",'""',$],{detached:!0,stdio:"ignore",windowsHide:!0}).unref();else if(process.platform==="darwin")b("open",[$],{detached:!0,stdio:"ignore"}).unref();else b("xdg-open",[$],{detached:!0,stdio:"ignore"}).unref()}catch{}}async function p($,q){try{let K=await(await fetch("https://open.feishu.cn/open-apis/auth/v3/tenant_access_token/internal",{method:"POST",headers:{"content-type":"application/json"},body:JSON.stringify({app_id:$,app_secret:q})})).json();if(K.tenant_access_token)return{ok:!0};return{ok:!1,error:`飞书拒绝: code=${K.code} msg=${K.msg??"(no msg)"}`}}catch(J){return{ok:!1,error:`网络错误: ${J?.message??String(J)}`}}}function r(){try{let $=x(D(import.meta.url)),q=w($,"lodestar.js");if(!g(q))return{error:`找不到 daemon bundle: ${q}`};let J=b(process.execPath,[q],{detached:!0,stdio:"ignore",windowsHide:!0});return J.unref(),{pid:J.pid}}catch($){return{error:$?.message??String($)}}}async function O(){if(!process.stdin.isTTY)console.error(`${z.red}lodestar-setup: stdin 不是 TTY,无法交互式输入。${z.reset}`),console.error("请直接在 cmd / PowerShell / Terminal 里跑,不要 pipe 或重定向 stdin。"),process.exit(1);if(g(W)){if(console.log(`${z.yellow}发现已有配置: ${W}${z.reset}`),(await V("覆盖? (y/N)",{default:"n"})).toLowerCase()!=="y"){console.log("已取消"),Z.close();return}}h("Lodestar 安装向导"),console.log("Lodestar 把 Feishu (飞书) 群聊接到 Codex。"),console.log("每个群对应一个项目目录, Codex 在那里跑、能读写文件。"),console.log(),console.log("本向导依次做 4 件事:"),console.log(` ${z.dim}1) 确保 Codex CLI 已装好${z.reset}`),console.log(` ${z.dim}2) 确认 ChatGPT 登录${z.reset}`),console.log(` ${z.dim}3) Feishu 自建应用 (含权限 / 事件 / 发版 + 凭据测试)${z.reset}`),console.log(` ${z.dim}4) 工作目录, 自动启动 daemon${z.reset}`),console.log(),await Z.question(`${z.dim}按 Enter 开始 (Ctrl+C 退出)...${z.reset}`),M(1,4,"准备 Codex CLI");let $=E("codex");if($)console.log(`${z.green}✓ codex CLI 已就位${z.reset}: ${z.dim}${$}${z.reset}`);else{if(console.log(`${z.yellow}未在 PATH 找到 codex CLI, 自动安装...${z.reset}`),console.log(`${z.dim}运行: npm install -g @openai/codex@latest${z.reset}`),console.log(),!await I())console.error(`
7
+ `)}function P($){return $.replace(/\\/g,"\\\\").replace(/"/g,"\\\"")}function E($){let q=process.env.PATH??"";if(!q)return null;let J=process.platform==="win32"?[`${$}.cmd`,`${$}.bat`,`${$}.exe`,$]:[$];for(let K of q.split(m)){if(!K)continue;for(let H of J){let X=w(K,H);if(g(X))return X}}return null}async function I(){let $=process.platform==="win32"?"npm.cmd":"npm";return new Promise((q)=>{let J=b($,["install","-g","@openai/codex@latest"],{stdio:"inherit",shell:process.platform==="win32"});J.on("exit",(K)=>q(K===0)),J.on("error",()=>q(!1))})}function k($){try{let q=B(`"${$}" login status 2>&1`,{timeout:1e4}).toString();return/Logged in using ChatGPT/i.test(q)}catch{return!1}}async function c($){return new Promise((q)=>{let J=b($,["login"],{stdio:"inherit",shell:process.platform==="win32"});J.on("exit",(K)=>q(K===0)),J.on("error",()=>q(!1))})}function l($){try{if(process.platform==="win32")b(process.env.ComSpec??"cmd.exe",["/c","start",'""',$],{detached:!0,stdio:"ignore",windowsHide:!0}).unref();else if(process.platform==="darwin")b("open",[$],{detached:!0,stdio:"ignore"}).unref();else b("xdg-open",[$],{detached:!0,stdio:"ignore"}).unref()}catch{}}async function p($,q){try{let K=await(await fetch("https://open.feishu.cn/open-apis/auth/v3/tenant_access_token/internal",{method:"POST",headers:{"content-type":"application/json"},body:JSON.stringify({app_id:$,app_secret:q})})).json();if(K.tenant_access_token)return{ok:!0};return{ok:!1,error:`飞书拒绝: code=${K.code} msg=${K.msg??"(no msg)"}`}}catch(J){return{ok:!1,error:`网络错误: ${J?.message??String(J)}`}}}function r(){try{let $=x(h(import.meta.url)),q=w($,"lodestar.js");if(!g(q))return{error:`找不到 daemon bundle: ${q}`};let J=b(process.execPath,[q],{detached:!0,stdio:"ignore",windowsHide:!0});return J.unref(),{pid:J.pid}}catch($){return{error:$?.message??String($)}}}async function O(){if(!process.stdin.isTTY)console.error(`${z.red}lodestar-setup: stdin 不是 TTY,无法交互式输入。${z.reset}`),console.error("请直接在 cmd / PowerShell / Terminal 里跑,不要 pipe 或重定向 stdin。"),process.exit(1);if(g(W)){if(console.log(`${z.yellow}发现已有配置: ${W}${z.reset}`),(await V("覆盖? (y/N)",{default:"n"})).toLowerCase()!=="y"){console.log("已取消"),Z.close();return}}D("Lodestar 安装向导"),console.log("Lodestar 把 Feishu (飞书) 群聊接到 Codex。"),console.log("每个群对应一个项目目录, Codex 在那里跑、能读写文件。"),console.log(),console.log("本向导依次做 4 件事:"),console.log(` ${z.dim}1) 确保 Codex CLI 已装好${z.reset}`),console.log(` ${z.dim}2) 确认 ChatGPT 登录${z.reset}`),console.log(` ${z.dim}3) Feishu 自建应用 (含权限 / 事件 / 发版 + 凭据测试)${z.reset}`),console.log(` ${z.dim}4) 工作目录, 自动启动 daemon${z.reset}`),console.log(),await Z.question(`${z.dim}按 Enter 开始 (Ctrl+C 退出)...${z.reset}`),M(1,4,"准备 Codex CLI");let $=E("codex");if($)console.log(`${z.green}✓ codex CLI 已就位${z.reset}: ${z.dim}${$}${z.reset}`);else{if(console.log(`${z.yellow}未在 PATH 找到 codex CLI, 自动安装...${z.reset}`),console.log(`${z.dim}运行: npm install -g @openai/codex@latest${z.reset}`),console.log(),!await I())console.error(`
8
8
  ${z.red}安装失败。${z.reset}`),console.error("请手动运行后再开向导:"),console.error(` ${z.cyan}npm install -g @openai/codex@latest${z.reset}`),console.error(` ${z.cyan}lodestar-setup${z.reset}`),Z.close(),process.exit(1);$=E("codex"),console.log(`${z.green}✓ 安装完成${z.reset}: ${z.dim}${$??"(应该装好了, 但 PATH 找不到 — 重开终端再试)"}${z.reset}`)}if(M(2,4,"ChatGPT 登录"),console.log("Lodestar 使用 Codex 的 ChatGPT 登录态。请确保 `codex login status` 显示 ChatGPT 登录。"),console.log(),$&&k($))console.log(`${z.green}✓ Codex 已登录 ChatGPT${z.reset}`);else{if(console.log(`${z.yellow}Codex 尚未登录 ChatGPT。现在启动 \`codex login\`。${z.reset}`),!($?await c($):!1)||!$||!k($))console.error(`
9
- ${z.red}Codex ChatGPT 登录未完成。${z.reset}`),console.error(`请手动运行 ${z.cyan}codex login${z.reset} 后再开向导。`),Z.close(),process.exit(1);console.log(`${z.green}✓ Codex 已登录 ChatGPT${z.reset}`)}M(3,4,"Feishu 自建应用");let q="https://open.feishu.cn/app";console.log("打开飞书开放平台 (浏览器):"),console.log(` ${z.cyan}${q}${z.reset}`),l(q),console.log(`${z.dim}(如果浏览器没自动开, 复制上面 URL 粘到浏览器)${z.reset}`),console.log(),console.log(`${z.bold}详细操作步骤:${z.reset}`),console.log(),console.log(` ${z.bold}① 创建应用${z.reset}`),console.log(` 点 "创建企业自建应用", 填名字 (如 ${z.dim}Lodestar${z.reset}), logo 随意。`),console.log(),console.log(` ${z.bold}② 添加机器人能力${z.reset}`),console.log(` 左侧菜单 "${z.cyan}添加应用能力${z.reset}" → 找到 "机器人" → 点 "${z.bold}添加${z.reset}" 按钮。`),console.log(),console.log(` ${z.bold}③ 申请权限 (左侧 "${z.cyan}权限管理${z.reset}" → "${z.bold}开通权限${z.reset}")${z.reset}`),console.log(` ${z.yellow}缺一个都会让 daemon 启动后默默丢消息, 一定要全开。${z.reset}`),console.log(` ${z.dim}消息类:${z.reset}`),console.log(` • ${z.bold}im:message:send_as_bot${z.reset} ${z.dim}# 以机器人身份发消息${z.reset}`),console.log(` • ${z.bold}im:message${z.reset} ${z.dim}# 接收/操作消息 (核心)${z.reset}`),console.log(` • ${z.bold}im:chat${z.reset} ${z.dim}# 读/写群信息 (匹配群名 ↔ 项目目录)${z.reset}`),console.log(` • ${z.bold}im:chat:create${z.reset} ${z.dim}# wt: 自动创建 worktree 群${z.reset}`),console.log(` • ${z.bold}im:chat:delete${z.reset} ${z.dim}# wt: 解散 worktree 群${z.reset}`),console.log(` • ${z.bold}im:chat.members:read${z.reset} ${z.dim}# wt: 判断发起人是否已在群内${z.reset}`),console.log(` • ${z.bold}im:chat.members:write_only${z.reset} ${z.dim}# wt: 把发起人拉进已有 worktree 群${z.reset}`),console.log(` • ${z.bold}im:resource${z.reset} ${z.dim}# 上传/下载附件 (图文双向)${z.reset}`),console.log(` • ${z.bold}im:message.urgent${z.reset} ${z.dim}# 加急推送 (锁屏通知 / Ask)${z.reset}`),console.log(` • ${z.bold}im:message.group_msg${z.reset} ${z.red}# 敏感: 接收群里所有消息${z.reset}`),console.log(` ${z.dim}└ 关键: 没它机器人只收 @ 自己的消息, 拿不到群里其他对话, 一定要开${z.reset}`),console.log(` ${z.dim}└ 敏感权限要走审批: 申请时填用途, 个人开发者通常秒过${z.reset}`),console.log(` • ${z.bold}im:message.group_at_msg:readonly${z.reset} ${z.dim}# 读 @ 机器人消息 (兜底)${z.reset}`),console.log(` ${z.dim}卡片类 (Card Kit):${z.reset}`),console.log(` • ${z.bold}cardkit:card:read${z.reset} ${z.dim}# 读卡片状态${z.reset}`),console.log(` • ${z.bold}cardkit:card:write${z.reset} ${z.dim}# 创建/更新卡片 (流式渲染核心)${z.reset}`),console.log(),console.log(` ${z.bold}④ 订阅事件 (左侧 "${z.cyan}事件与回调${z.reset}", 拆两个子页:)${z.reset}`),console.log(` ${z.dim}a)${z.reset} ${z.bold}事件配置${z.reset} 页:`),console.log(` ${z.yellow}• "订阅方式" → 选 "长连接" → 点保存${z.reset}`),console.log(` • 添加事件: ${z.bold}im.message.receive_v1${z.reset} ${z.dim}# 收群消息${z.reset}`),console.log(` ${z.dim}b)${z.reset} ${z.bold}回调配置${z.reset} 页:`),console.log(` ${z.yellow}• "订阅方式" → 选 "长连接" → 点保存${z.reset}`),console.log(` • 添加事件: ${z.bold}card.action.trigger${z.reset} ${z.dim}# 卡片按钮点击回调${z.reset}`),console.log(),console.log(` ${z.bold}⑤ 发布版本${z.reset}`),console.log(` 页面顶部 "${z.bold}创建版本${z.reset}" → 滚到底点 "${z.bold}保存${z.reset}" → 弹框点 "${z.bold}发布${z.reset}"。`),console.log(` ${z.yellow}没发版的应用收不到任何事件 — 这步九成新手会忘!${z.reset}`),console.log(),console.log(` ${z.bold}⑥ 拿凭据${z.reset}`),console.log(` 左侧 "凭证与基础信息" → 顶部 ${z.bold}App ID${z.reset} (${z.dim}cli_...${z.reset}) 和 ${z.bold}App Secret${z.reset}, 待会粘到下面。`),console.log(),console.log(` ${z.bold}⑦ 把机器人拉进群${z.reset}`),console.log(" 想用的飞书群 → 群设置 → 群机器人 → 添加机器人 → 选你的应用。"),console.log(` ${z.yellow}群名要等于 projects_root 下的项目目录名${z.reset} (下一步设, 默认是用户主目录)。`),console.log();let J="",K="";while(!0){J=await V("App ID (以 cli_ 开头)",{required:!0}),K=await V("App Secret",{required:!0}),console.log(`${z.dim}测试中... (调 tenant_access_token endpoint)${z.reset}`);let y=await p(J,K);if(y.ok){console.log(`${z.green}✓ Feishu 凭据测试通过${z.reset}`);break}if(console.log(`${z.red}✗ 测试失败:${z.reset} ${y.error}`),console.log(`${z.dim}最常见原因: app_id / app_secret 抄错, 或应用还没 "发布上线" (步骤 ⑤)。${z.reset}`),console.log(),(await V("重新填? (Y/n)",{default:"y"})).toLowerCase()==="n")console.log(`${z.yellow}已取消, 配置未写盘。${z.reset}`),Z.close(),process.exit(1)}M(4,4,"工作目录 + 启动"),console.log("每个 Feishu 群对应 projects_root 下同名的目录。"),console.log();let H=process.platform==="win32"?process.env.USERPROFILE??"C:\\Users\\Default":process.env.HOME??"/root",X=await V("projects_root",{default:H});B(N,{recursive:!0});let F=["# Lodestar config — generated by `lodestar-setup`","# Edit by hand or re-run setup to overwrite.","","[feishu]",`app_id = "${P(J)}"`,`app_secret = "${P(K)}"`,"","[runtime]",`projects_root = "${P(X)}"`,""];_(W,F.join(`
9
+ ${z.red}Codex ChatGPT 登录未完成。${z.reset}`),console.error(`请手动运行 ${z.cyan}codex login${z.reset} 后再开向导。`),Z.close(),process.exit(1);console.log(`${z.green}✓ Codex 已登录 ChatGPT${z.reset}`)}M(3,4,"Feishu 自建应用");let q="https://open.feishu.cn/app";console.log("打开飞书开放平台 (浏览器):"),console.log(` ${z.cyan}${q}${z.reset}`),l(q),console.log(`${z.dim}(如果浏览器没自动开, 复制上面 URL 粘到浏览器)${z.reset}`),console.log(),console.log(`${z.bold}详细操作步骤:${z.reset}`),console.log(),console.log(` ${z.bold}① 创建应用${z.reset}`),console.log(` 点 "创建企业自建应用", 填名字 (如 ${z.dim}Lodestar${z.reset}), logo 随意。`),console.log(),console.log(` ${z.bold}② 添加机器人能力${z.reset}`),console.log(` 左侧菜单 "${z.cyan}添加应用能力${z.reset}" → 找到 "机器人" → 点 "${z.bold}添加${z.reset}" 按钮。`),console.log(),console.log(` ${z.bold}③ 申请权限 (左侧 "${z.cyan}权限管理${z.reset}" → "${z.bold}开通权限${z.reset}")${z.reset}`),console.log(` ${z.yellow}缺一个都会让 daemon 启动后默默丢消息, 一定要全开。${z.reset}`),console.log(` ${z.dim}消息类:${z.reset}`),console.log(` • ${z.bold}im:message:send_as_bot${z.reset} ${z.dim}# 以机器人身份发消息${z.reset}`),console.log(` • ${z.bold}im:message${z.reset} ${z.dim}# 接收/操作消息 (核心)${z.reset}`),console.log(` • ${z.bold}im:chat${z.reset} ${z.dim}# 读/写群信息 (匹配群名 ↔ 项目目录)${z.reset}`),console.log(` • ${z.bold}im:chat:create${z.reset} ${z.dim}# wt: 自动创建 worktree 群${z.reset}`),console.log(` • ${z.bold}im:chat:delete${z.reset} ${z.dim}# wt: 解散 worktree 群${z.reset}`),console.log(` • ${z.bold}im:chat.members:read${z.reset} ${z.dim}# wt: 判断发起人是否已在群内${z.reset}`),console.log(` • ${z.bold}im:chat.members:write_only${z.reset} ${z.dim}# wt: 把发起人拉进已有 worktree 群${z.reset}`),console.log(` • ${z.bold}im:resource${z.reset} ${z.dim}# 上传/下载附件 (图文双向)${z.reset}`),console.log(` • ${z.bold}im:message.urgent${z.reset} ${z.dim}# 加急推送 (锁屏通知 / Ask)${z.reset}`),console.log(` • ${z.bold}im:message.group_msg${z.reset} ${z.red}# 敏感: 接收群里所有消息${z.reset}`),console.log(` ${z.dim}└ 关键: 没它机器人只收 @ 自己的消息, 拿不到群里其他对话, 一定要开${z.reset}`),console.log(` ${z.dim}└ 敏感权限要走审批: 申请时填用途, 个人开发者通常秒过${z.reset}`),console.log(` • ${z.bold}im:message.group_at_msg:readonly${z.reset} ${z.dim}# 读 @ 机器人消息 (兜底)${z.reset}`),console.log(` ${z.dim}卡片类 (Card Kit):${z.reset}`),console.log(` • ${z.bold}cardkit:card:read${z.reset} ${z.dim}# 读卡片状态${z.reset}`),console.log(` • ${z.bold}cardkit:card:write${z.reset} ${z.dim}# 创建/更新卡片 (流式渲染核心)${z.reset}`),console.log(),console.log(` ${z.bold}④ 订阅事件 (左侧 "${z.cyan}事件与回调${z.reset}", 拆两个子页:)${z.reset}`),console.log(` ${z.dim}a)${z.reset} ${z.bold}事件配置${z.reset} 页:`),console.log(` ${z.yellow}• "订阅方式" → 选 "长连接" → 点保存${z.reset}`),console.log(` • 添加事件: ${z.bold}im.message.receive_v1${z.reset} ${z.dim}# 收群消息${z.reset}`),console.log(` ${z.dim}b)${z.reset} ${z.bold}回调配置${z.reset} 页:`),console.log(` ${z.yellow}• "订阅方式" → 选 "长连接" → 点保存${z.reset}`),console.log(` • 添加事件: ${z.bold}card.action.trigger${z.reset} ${z.dim}# 卡片按钮点击回调${z.reset}`),console.log(),console.log(` ${z.bold}⑤ 发布版本${z.reset}`),console.log(` 页面顶部 "${z.bold}创建版本${z.reset}" → 滚到底点 "${z.bold}保存${z.reset}" → 弹框点 "${z.bold}发布${z.reset}"。`),console.log(` ${z.yellow}没发版的应用收不到任何事件 — 这步九成新手会忘!${z.reset}`),console.log(),console.log(` ${z.bold}⑥ 拿凭据${z.reset}`),console.log(` 左侧 "凭证与基础信息" → 顶部 ${z.bold}App ID${z.reset} (${z.dim}cli_...${z.reset}) 和 ${z.bold}App Secret${z.reset}, 待会粘到下面。`),console.log(),console.log(` ${z.bold}⑦ 把机器人拉进群${z.reset}`),console.log(" 想用的飞书群 → 群设置 → 群机器人 → 添加机器人 → 选你的应用。"),console.log(` ${z.yellow}群名要等于 projects_root 下的项目目录名${z.reset} (下一步设, 默认是用户主目录)。`),console.log();let J="",K="";while(!0){J=await V("App ID (以 cli_ 开头)",{required:!0}),K=await V("App Secret",{required:!0}),console.log(`${z.dim}测试中... (调 tenant_access_token endpoint)${z.reset}`);let y=await p(J,K);if(y.ok){console.log(`${z.green}✓ Feishu 凭据测试通过${z.reset}`);break}if(console.log(`${z.red}✗ 测试失败:${z.reset} ${y.error}`),console.log(`${z.dim}最常见原因: app_id / app_secret 抄错, 或应用还没 "发布上线" (步骤 ⑤)。${z.reset}`),console.log(),(await V("重新填? (Y/n)",{default:"y"})).toLowerCase()==="n")console.log(`${z.yellow}已取消, 配置未写盘。${z.reset}`),Z.close(),process.exit(1)}M(4,4,"工作目录 + 启动"),console.log("每个 Feishu 群对应 projects_root 下同名的目录。"),console.log();let H=process.platform==="win32"?process.env.USERPROFILE??"C:\\Users\\Default":process.env.HOME??"/root",X=await V("projects_root",{default:H});A(N,{recursive:!0});let F=["# Lodestar config — generated by `lodestar-setup`","# Edit by hand or re-run setup to overwrite.","","[feishu]",`app_id = "${P(J)}"`,`app_secret = "${P(K)}"`,"","[runtime]",`projects_root = "${P(X)}"`,""];_(W,F.join(`
10
10
  `),{mode:384}),console.log(`
11
11
  ${z.green}${z.bold}✓ 配置已写入${z.reset}`),console.log(` ${z.cyan}${W}${z.reset}`),console.log(`
12
- ${z.bold}启动 daemon...${z.reset}`);let U=r(),R=process.platform==="win32"?"\\":"/",u=process.platform==="win32"?`${process.env.LOCALAPPDATA??"%LOCALAPPDATA%"}\\Lodestar\\daemon.log`:`${process.env.HOME??"~"}/.local/share/lodestar/daemon.log`;if(U.pid)console.log(`${z.green}✓ daemon 已在后台启动${z.reset} (pid ${U.pid})`),console.log(),console.log(`${z.bold}最后一步: 在 Feishu 验证${z.reset}`),console.log(" ① 把机器人拉进任意飞书群"),console.log(` ② 群名 = ${z.cyan}${X}${R}<群名>${z.reset} 下的目录名 (新群第一条消息会自动建)`),console.log(" ③ 在群里发任意一条消息, Codex 上线"),console.log(),console.log("日志:"),console.log(` ${z.cyan}${u}${z.reset}`),console.log(),console.log(`${z.dim}若长期跑后台, 参考 README "7×24 守护" 一节配 systemd / Windows 后台托管。${z.reset}`);else console.log(`${z.yellow}启动失败: ${U.error}${z.reset}`),console.log(`手动运行: ${z.cyan}lodestar-daemon${z.reset}`);console.log(),Z.close()}O().catch(($)=>{console.error(`
12
+ ${z.bold}启动 daemon...${z.reset}`);let U=r(),u=process.platform==="win32"?"\\":"/",R=process.platform==="win32"?`${process.env.LOCALAPPDATA??"%LOCALAPPDATA%"}\\Lodestar\\daemon.log`:`${process.env.HOME??"~"}/.local/share/lodestar/daemon.log`;if(U.pid)console.log(`${z.green}✓ daemon 已在后台启动${z.reset} (pid ${U.pid})`),console.log(),console.log(`${z.bold}最后一步: 在 Feishu 验证${z.reset}`),console.log(" ① 把机器人拉进任意飞书群"),console.log(` ② 群名 = ${z.cyan}${X}${u}<群名>${z.reset} 下的目录名 (新群第一条消息会自动建)`),console.log(" ③ 在群里发任意一条消息, Codex 上线"),console.log(),console.log("日志:"),console.log(` ${z.cyan}${R}${z.reset}`),console.log(),console.log(`${z.dim}若长期跑后台, 参考 README "7×24 守护" 一节配 systemd / Windows 后台托管。${z.reset}`);else console.log(`${z.yellow}启动失败: ${U.error}${z.reset}`),console.log(`手动运行: ${z.cyan}lodestar-daemon${z.reset}`);console.log(),Z.close()}O().catch(($)=>{console.error(`
13
13
  lodestar-setup: ${$?.message??$}`),process.exit(1)});
@@ -1,3 +1,3 @@
1
1
  #!/usr/bin/env node
2
- import{execSync as f}from"node:child_process";import{existsSync as U,readFileSync as y}from"node:fs";import{homedir as W}from"node:os";import{join as K}from"node:path";var X=W(),B=process.platform==="win32";function b(z,J,Y){if(z)return z;if(J)return K(J,"lodestar");return Y}function H(){if(B)return K(process.env.APPDATA??K(X,"AppData","Roaming"),"Lodestar");return K(X,".config","lodestar")}function V(){if(B)return K(process.env.LOCALAPPDATA??K(X,"AppData","Local"),"Lodestar");return K(X,".local","share","lodestar")}var w=b(process.env.LODESTAR_CONFIG_DIR,process.env.XDG_CONFIG_HOME,H()),Q=b(process.env.LODESTAR_DATA_DIR,process.env.XDG_DATA_HOME,V()),O=process.env.LODESTAR_CONFIG??K(w,"config.toml"),Z=K(Q,"daemon.pid"),g=K(Q,"daemon.log"),x=K(Q,"session-chat-map.json"),S=K(Q,"session-resume-map.json"),E=K(Q,"session-model-map.json"),v=K(Q,"alive-on-shutdown.json"),h=K(Q,"inbox"),k=K(Q,"debug.sock"),A=K(Q,"debug-context.json");import{execSync as G}from"node:child_process";import{readFileSync as L,writeFileSync as C}from"node:fs";function M(z){try{if(process.platform==="linux")return L(`/proc/${z}/cmdline`,"utf8").replace(/\0/g," ").trim();if(process.platform==="darwin")return G(`ps -p ${z} -o args=`,{encoding:"utf8",timeout:2000}).trim();if(process.platform==="win32")return G(`powershell -NoProfile -Command "(Get-CimInstance Win32_Process -Filter 'ProcessId=${z}').CommandLine"`,{encoding:"utf8",timeout:5000}).trim()||null;return null}catch{return null}}function N(z,J){if(!Number.isFinite(z)||z<=0||!J)return!1;let Y=M(z);if(!Y)return!1;if(process.platform==="win32")return Y.toLowerCase().includes(J.toLowerCase());return Y.includes(J)}var q={reset:"\x1B[0m",bold:"\x1B[1m",green:"\x1B[32m",yellow:"\x1B[33m",red:"\x1B[31m",dim:"\x1B[2m"};function P(z){return new Promise((J)=>setTimeout(J,z))}async function R(){if(!U(Z)){console.log(`${q.yellow}Lodestar daemon 未运行${q.reset} ${q.dim}(${Z} 不存在)${q.reset}`);return}let z=y(Z,"utf8").split(`
3
- `),J=parseInt((z[0]??"").trim(),10),Y=(z[1]??"").trim();if(!Number.isFinite(J)||J<=0)console.error(`${q.red}PID 文件格式坏:${q.reset} ${Z}`),process.exit(1);if(Y&&!N(J,Y)){console.log(`${q.yellow}PID ${J} 上没有 daemon (stale 文件)${q.reset}`),console.log(`${q.dim}手删 ${Z} 后再试${q.reset}`);return}console.log(`${q.bold}停止 daemon${q.reset} (pid ${J})...`);try{if(process.platform==="win32")f(`taskkill /PID ${J} /T /F`,{stdio:"ignore"});else process.kill(J,"SIGTERM")}catch($){console.error(`${q.red}发信号失败:${q.reset} ${$?.message??$}`),process.exit(1)}for(let $=0;$<50;$++){if(!U(Z)){console.log(`${q.green}✓ daemon 已停${q.reset}`);return}await P(100)}console.log(`${q.yellow}已发 SIGTERM, 但 5s 内 daemon 没清掉 ${Z}${q.reset}`),console.log(`${q.dim}它可能还在收尾 (落盘 alive marker / 关 WS); 下次启动 pid-guard 会自动判别。${q.reset}`)}R().catch((z)=>{console.error(`${q.red}lodestar-stop:${q.reset} ${z?.message??z}`),process.exit(1)});
2
+ import{execSync as f}from"node:child_process";import{existsSync as U,readFileSync as y}from"node:fs";import{homedir as W}from"node:os";import{join as J}from"node:path";var X=W(),B=process.platform==="win32";function b(z,K,Y){if(z)return z;if(K)return J(K,"lodestar");return Y}function H(){if(B)return J(process.env.APPDATA??J(X,"AppData","Roaming"),"Lodestar");return J(X,".config","lodestar")}function V(){if(B)return J(process.env.LOCALAPPDATA??J(X,"AppData","Local"),"Lodestar");return J(X,".local","share","lodestar")}var w=b(process.env.LODESTAR_CONFIG_DIR,process.env.XDG_CONFIG_HOME,H()),Q=b(process.env.LODESTAR_DATA_DIR,process.env.XDG_DATA_HOME,V()),T=process.env.LODESTAR_CONFIG??J(w,"config.toml"),Z=J(Q,"daemon.pid"),g=J(Q,"daemon.log"),x=J(Q,"session-chat-map.json"),S=J(Q,"session-resume-map.json"),E=J(Q,"session-model-map.json"),v=J(Q,"tasklist-map.json"),h=J(Q,"alive-on-shutdown.json"),k=J(Q,"inbox"),F=J(Q,"debug.sock"),j=J(Q,"debug-context.json");import{execSync as G}from"node:child_process";import{readFileSync as L,writeFileSync as _}from"node:fs";function M(z){try{if(process.platform==="linux")return L(`/proc/${z}/cmdline`,"utf8").replace(/\0/g," ").trim();if(process.platform==="darwin")return G(`ps -p ${z} -o args=`,{encoding:"utf8",timeout:2000}).trim();if(process.platform==="win32")return G(`powershell -NoProfile -Command "(Get-CimInstance Win32_Process -Filter 'ProcessId=${z}').CommandLine"`,{encoding:"utf8",timeout:5000}).trim()||null;return null}catch{return null}}function N(z,K){if(!Number.isFinite(z)||z<=0||!K)return!1;let Y=M(z);if(!Y)return!1;if(process.platform==="win32")return Y.toLowerCase().includes(K.toLowerCase());return Y.includes(K)}var q={reset:"\x1B[0m",bold:"\x1B[1m",green:"\x1B[32m",yellow:"\x1B[33m",red:"\x1B[31m",dim:"\x1B[2m"};function P(z){return new Promise((K)=>setTimeout(K,z))}async function u(){if(!U(Z)){console.log(`${q.yellow}Lodestar daemon 未运行${q.reset} ${q.dim}(${Z} 不存在)${q.reset}`);return}let z=y(Z,"utf8").split(`
3
+ `),K=parseInt((z[0]??"").trim(),10),Y=(z[1]??"").trim();if(!Number.isFinite(K)||K<=0)console.error(`${q.red}PID 文件格式坏:${q.reset} ${Z}`),process.exit(1);if(Y&&!N(K,Y)){console.log(`${q.yellow}PID ${K} 上没有 daemon (stale 文件)${q.reset}`),console.log(`${q.dim}手删 ${Z} 后再试${q.reset}`);return}console.log(`${q.bold}停止 daemon${q.reset} (pid ${K})...`);try{if(process.platform==="win32")f(`taskkill /PID ${K} /T /F`,{stdio:"ignore"});else process.kill(K,"SIGTERM")}catch($){console.error(`${q.red}发信号失败:${q.reset} ${$?.message??$}`),process.exit(1)}for(let $=0;$<50;$++){if(!U(Z)){console.log(`${q.green}✓ daemon 已停${q.reset}`);return}await P(100)}console.log(`${q.yellow}已发 SIGTERM, 但 5s 内 daemon 没清掉 ${Z}${q.reset}`),console.log(`${q.dim}它可能还在收尾 (落盘 alive marker / 关 WS); 下次启动 pid-guard 会自动判别。${q.reset}`)}u().catch((z)=>{console.error(`${q.red}lodestar-stop:${q.reset} ${z?.message??z}`),process.exit(1)});
@@ -1,5 +1,5 @@
1
1
  #!/usr/bin/env node
2
- import{spawn as b}from"node:child_process";import{existsSync as w}from"node:fs";import{homedir as p}from"node:os";import{join as o}from"node:path";var r=p(),c=process.platform==="win32";function n(e,l,m){if(e)return e;if(l)return o(l,"lodestar");return m}function y(){if(c)return o(process.env.APPDATA??o(r,"AppData","Roaming"),"Lodestar");return o(r,".config","lodestar")}function d(){if(c)return o(process.env.LOCALAPPDATA??o(r,"AppData","Local"),"Lodestar");return o(r,".local","share","lodestar")}var x=n(process.env.LODESTAR_CONFIG_DIR,process.env.XDG_CONFIG_HOME,y()),t=n(process.env.LODESTAR_DATA_DIR,process.env.XDG_DATA_HOME,d()),C=process.env.LODESTAR_CONFIG??o(x,"config.toml"),a=o(t,"daemon.pid"),P=o(t,"daemon.log"),q=o(t,"session-chat-map.json"),z=o(t,"session-resume-map.json"),J=o(t,"session-model-map.json"),K=o(t,"alive-on-shutdown.json"),Q=o(t,"inbox"),Y=o(t,"debug.sock"),Z=o(t,"debug-context.json");var s={reset:"\x1B[0m",bold:"\x1B[1m",cyan:"\x1B[36m",green:"\x1B[32m",yellow:"\x1B[33m",red:"\x1B[31m",dim:"\x1B[2m"};function g(){let e=process.platform==="win32"?"npm.cmd":"npm";return new Promise((l)=>{let m=b(e,["install","-g","@leviyuan/lodestar@latest","@openai/codex@latest"],{stdio:"inherit",shell:process.platform==="win32"});m.on("exit",($)=>l($??1)),m.on("error",($)=>{console.error(`${s.red}spawn npm 失败:${s.reset} ${$.message}`),l(1)})})}async function u(){console.log(`${s.bold}更新 Lodestar + Codex CLI${s.reset}`),console.log(`${s.dim}npm i -g @leviyuan/lodestar@latest @openai/codex@latest${s.reset}
2
+ import{spawn as b}from"node:child_process";import{existsSync as w}from"node:fs";import{homedir as p}from"node:os";import{join as o}from"node:path";var r=p(),c=process.platform==="win32";function n(e,l,m){if(e)return e;if(l)return o(l,"lodestar");return m}function y(){if(c)return o(process.env.APPDATA??o(r,"AppData","Roaming"),"Lodestar");return o(r,".config","lodestar")}function d(){if(c)return o(process.env.LOCALAPPDATA??o(r,"AppData","Local"),"Lodestar");return o(r,".local","share","lodestar")}var x=n(process.env.LODESTAR_CONFIG_DIR,process.env.XDG_CONFIG_HOME,y()),t=n(process.env.LODESTAR_DATA_DIR,process.env.XDG_DATA_HOME,d()),C=process.env.LODESTAR_CONFIG??o(x,"config.toml"),a=o(t,"daemon.pid"),P=o(t,"daemon.log"),q=o(t,"session-chat-map.json"),z=o(t,"session-resume-map.json"),J=o(t,"session-model-map.json"),K=o(t,"tasklist-map.json"),Q=o(t,"alive-on-shutdown.json"),Y=o(t,"inbox"),Z=o(t,"debug.sock"),W=o(t,"debug-context.json");var s={reset:"\x1B[0m",bold:"\x1B[1m",cyan:"\x1B[36m",green:"\x1B[32m",yellow:"\x1B[33m",red:"\x1B[31m",dim:"\x1B[2m"};function g(){let e=process.platform==="win32"?"npm.cmd":"npm";return new Promise((l)=>{let m=b(e,["install","-g","@leviyuan/lodestar@latest","@openai/codex@latest"],{stdio:"inherit",shell:process.platform==="win32"});m.on("exit",($)=>l($??1)),m.on("error",($)=>{console.error(`${s.red}spawn npm 失败:${s.reset} ${$.message}`),l(1)})})}async function u(){console.log(`${s.bold}更新 Lodestar + Codex CLI${s.reset}`),console.log(`${s.dim}npm i -g @leviyuan/lodestar@latest @openai/codex@latest${s.reset}
3
3
  `);let e=await g();if(e!==0)console.error(`
4
4
  ${s.red}更新失败 (npm exit ${e})${s.reset}`),process.exit(e);if(console.log(`
5
5
  ${s.green}✓ 更新完成${s.reset}`),w(a))console.log(),console.log(`${s.yellow}检测到 daemon 仍在跑老版本进程, 用新版本需要重启:${s.reset}`),console.log(` ${s.dim}# systemd / Windows 后台托管的:${s.reset}`),console.log(` ${s.cyan}systemctl --user restart lodestar${s.reset} ${s.dim}# Linux/macOS${s.reset}`),console.log(` ${s.dim}# 或手动重启:${s.reset}`),console.log(` ${s.cyan}lodestar-stop && lodestar-daemon${s.reset}`)}u().catch((e)=>{console.error(`${s.red}lodestar-update:${s.reset} ${e?.message??e}`),process.exit(1)});