放弃 Python,Kimi 用 TS + Node.js 重写了一个 Kimi Code

最近开发者圈传得最凶的一句话,是「kimi-cli 用 Python 是彻底的失败,立刻重构为 TS」。配的图是 Kimi 把自家命令行工具从 Python 推翻、重写成 TypeScript 的截图,不少人转发时都带着一句「连 Python 都保不住了」。

作为一个天天和 Node、TypeScript 打交道的人,我对那句狠话兴趣不大,更想知道一件事:它到底换成了什么技术栈?我把新仓库从头到尾翻了一遍,连 package.json 都对着看了一遍。结论比那句口号有意思得多,里面还藏着一个不少人都传错的细节。

先把两件事拆开:一个梗,一个真重写

网上流传的那张截图,其实把两件事并到了一起。

第一件是那条狠话的出处。它不是月之暗面发的官方公告,而是 kimi-cli 仓库里一个 PR(#1707)的描述标题。这个 PR 有几个容易被忽略的点:

  • 作者是普通社区账号 :GitHub 上的关联身份是 NONE,不是月之暗面的人。

  • 开 PR 的日期是 4 月 1 日:愚人节当天提交,标题那句话本身就带着玩梗的味道。

  • 到现在还挂着没合并:状态一直是 Open,没进主分支。

但这个 PR 也不全是玩笑,作者是真把活干了,三百来个文件、几万行代码,还附了一组 SWE-bench 对比。用的技术栈是 Bun + React Ink,全套押在 Bun 上。

第二件才是真正落地的产品。月之暗面 5 月 22 日新建了 kimi-code 仓库,5 月底一口气连发了 0.2 到 0.4 几个版本。老的 kimi-cli(Python 版,攒了八千七百多 star)README 顶部已经挂上提示:项目接下来由 kimi-code 接手,装新版会自动把配置和会话搬过去,Python 版逐步停更。

两件事时间差了一个多月,技术栈也不是一套。把社区那个愚人节 PR 的方案安到官方头上,是这次传播里最大的误会。

扒开 package.json:这是一套很「Node」的栈

先看一眼仓库根目录就有底了:pnpm-lock.yamlpnpm-workspace.yaml.nvmrc 一应俱全,右下角 Languages 写着 TypeScript 97.4%。一个标准的 pnpm + Node 的 TS monorepo,跟那个愚人节 PR 押的 Bun 不是一条路。

直接看官方 kimi-code 里 CLI 应用的依赖,前端和 Node 开发者会觉得每一个都眼熟:

go 复制代码
"dependencies": {
"@earendil-works/pi-tui": "^0.74.0",
"commander": "^13.1.0",
"zod": "^4.3.6",
"chalk": "^5.4.1",
"cli-highlight": "^2.1.11",
"smol-toml": "^1.6.1",
"@mariozechner/clipboard": "^0.3.2",
"semver": "^7.7.4"
}

对照 Python 版那一套,这就是一次很标准的同类替换:命令行解析从 Typer 换成 commander,数据校验从 Pydantic 换成 zod,配置文件用 smol-toml 读 TOML,终端高亮交给 cli-highlightchalk。没有什么生僻库,全是当下 TS 项目的常规选型。

工程链路这边,几乎就是 2026 年 TS 工程的标准答案:

  • 包管理 :pnpm 10 的 monorepo,apps/ 放 CLI,packages/ 放可复用模块。

  • 构建 :用 tsdown 打包,TypeScript 直接上到了 6.0。

  • 静态检查oxlint,而且开了 type-aware 模式,配合 oxlint-tsgolint 做类型相关的检查。

  • 测试vitest,覆盖率走 v8。

  • 版本与发布changesets 管版本号和 changelog,simple-git-hooks + lint-staged 卡提交,sherifpublintarethetypeswrong 做 monorepo 的依赖和打包体检。

换句话说,平时写 TS 库的人,把这套配置原样搬到自己项目里都不违和。一个大模型团队的旗舰命令行工具,骨架和我们平时发的 npm 包没有本质区别。

终端界面用的是 pi-tui,不是 React Ink

这是传错的重灾区,得专门说一句。

先放一张 kimi-code 跑起来的样子,欢迎面板、会话信息、对话区、底部状态栏,能看到模型是 kimi-code/kimi-for-coding,状态栏右下角还实时显示 context 占用,长会话里这块信息挺实用。

那个愚人节 PR 用的是 React Ink ,也就是用 React 那套 reconciler 来渲染终端 UI。但官方 kimi-code 用的是 **@earendil-works/pi-tui**,README 的致谢里专门感谢了 pi-tui 的作者。

pi-tui 来自 earendil-works 的 pi-mono 仓库,是一个专门给终端 Agent 做的 TUI 库。它和 React Ink 是两条不同的路:Ink 胜在能直接复用 React 的心智模型和生态,但渲染链路更重;pi-tui 更像是为「长时间、跑命令、刷大量输出」的 Agent 会话量身做的轻量方案。

社区里也有人点出,与其各家从零造 harness,不如直接复用 pi 这一套底座,把精力留给模型本身。Kimi 这次选 pi-tui,多少印证了这个方向。所以官方的终端层不是 Ink,看到「Kimi 基于 React Ink 重写」的说法,可以直接划掉了。

「无需 Node 环境」,却是用 Node 写的

kimi-code 的 README 里有句很拧巴的话:装的时候「不需要 Node.js」,但仓库的 engines 又写着 node >=24.15。这俩怎么同时成立?

答案藏在它的 native 构建脚本里:

go 复制代码
"build": "tsdown",
"build:native:sea": "node scripts/native/build.mjs --profile=local",
"build:native:release": "node scripts/native/build.mjs --profile=release",
"package:native": "node scripts/native/package.mjs"

它先用 tsdown 把整个 CLI 打成一个 bundle,再走 Node SEA(Single Executable Applications,单可执行文件)把运行时和代码一起塞进一个二进制。最终用户下载到的是一个独立可执行程序,机器上不用预装 Node、不用配 PATH,也不会和全局模块打架。开发者需要 Node 24 和 pnpm,是因为那是构建这台「打包机」的环境,跟用户拿到的成品是两码事。

这其实是 CLI 工具一直头疼的老问题:怎么让用户「装一个命令行软件」而不是「先装一个语言运行时」。Bun 的答案是 bun build --compile,Claude Code 走的就是这条路,把代码连同 Bun 运行时一起编译成原生二进制,跑起来彻底脱离 Node。Node 这边的答案,是这两年逐渐成熟的 SEA。Kimi 押了后者。

benchmark 其实没涨,重写图的是工程

很多人下意识觉得,重写一遍是为了让 Agent 更聪明。但那个 PR 自己附的 SWE-bench Verified 数据(都跑 kimi-k2.5 模型,500 道题)说的是另一回事:

指标 kimi-cli(Python) kimi-cli-ts(TypeScript)
Resolved 325 / 496(65.5%) 317 / 500(63.4%)

两边基本打平,Python 版甚至还略高一点。换语言并没有让解题能力变强。这次重写真正图的,是分发体验、冷启动速度、终端交互这些工程层面的事,跟模型能力没关系。把这点说清楚,免得有人误以为「TS 写的 Agent 就更聪明」。

写在最后

把镜头拉远看,这两年顺手的终端编程 Agent,真正收敛的是语言:清一色用 TypeScript 写核心,配一套打磨过的 TUI。分叉的反而是运行时,Claude Code 走 Bun,Gemini CLI 还是 Node 加一个 JS bundle,Kimi 这次选了 Node + SEA。Bun 这条路 Claude Code 已经验证可行,Kimi 却挑了更稳的那条,考虑到 Bun 如今归了 Anthropic,这个选择就更耐人寻味。

对前端和 Node 开发者来说,这是个挺实在的信号。commanderzod、一个终端 UI 库、pnpm monorepo、SEA(或 bun --compile)单文件打包,这些你本来就会的东西,凑齐了就是一个 Agent harness 的雏形。挡在前面的从来不是语言,而是模型,以及工具调用、上下文管理、审批这些 harness 逻辑要怎么设计。

一年前聊「AI 基础设施」,默认是 Python 的主场。现在终端这一层,已经基本是 TypeScript 的地盘,至于底下跑 Node 还是 Bun,反倒成了各家自己的工程选择。那句愚人节的玩笑话,话糙,但它点破的趋势是真的。

相关推荐
还是鼠鼠1 小时前
AI掘金头条新闻系统 (Toutiao News)-获取用户信息
后端·python·mysql·fastapi·web
Cloud_Shy6181 小时前
解读《Effective Python 3rd Edition》:从练气到老魔
开发语言·python
SunnyDays10111 小时前
Python 操作 Excel 超链接:添加网页、文件、工作表和图片链接
python·excel
雨辰AI1 小时前
MySQL 迁移至达梦 DM9 完整改造指南|99% SQL 零改动
java·开发语言·数据库·sql·mysql·政务
弹简特2 小时前
【Java项目-轻聊】05-AI赋能设计接口文档
java·开发语言
li星野2 小时前
RAG优化系列:HyDE(假设文档嵌入)——让LLM先写答案再检索
python·学习
AI行业学习2 小时前
.NET Framework 3.5 SP1 完整离线包(2029.5.29)
开发语言·windows·.net
知识分享小能手2 小时前
Flask入门学习教程,从入门到精通,Flask智能租房——用户中心知识点详解(9)
python·学习·flask
cany10002 小时前
C++ -- 队列std::queue
开发语言·c++