kaizenai 0.3.0 → 0.5.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.
Files changed (74) hide show
  1. package/README.md +0 -4
  2. package/bin/kaizen +16 -5
  3. package/dist/client/app-icon-180.png +0 -0
  4. package/dist/client/app-icon-192.png +0 -0
  5. package/dist/client/app-icon-512.png +0 -0
  6. package/dist/client/app-icon-96.png +0 -0
  7. package/dist/client/apple-touch-icon.png +0 -0
  8. package/dist/client/assets/index-C4Zm475L.js +638 -0
  9. package/dist/client/assets/index-CxKis6wK.css +32 -0
  10. package/dist/client/favicon.png +0 -0
  11. package/dist/client/index.html +17 -10
  12. package/dist/client/manifest-dark.webmanifest +3 -3
  13. package/dist/client/manifest.webmanifest +3 -3
  14. package/dist/client/pwa-192.png +0 -0
  15. package/dist/client/pwa-512.png +0 -0
  16. package/dist/server/cli-supervisor.js +306 -0
  17. package/dist/server/cli.js +12636 -0
  18. package/package.json +7 -10
  19. package/dist/client/assets/index-BBs80KD-.js +0 -623
  20. package/dist/client/assets/index-CkCgyLNq.css +0 -32
  21. package/src/server/acp-shared.ts +0 -315
  22. package/src/server/agent.ts +0 -1159
  23. package/src/server/attachments.ts +0 -133
  24. package/src/server/backgrounds.ts +0 -74
  25. package/src/server/cli-runtime.ts +0 -375
  26. package/src/server/cli-supervisor.ts +0 -97
  27. package/src/server/cli.ts +0 -68
  28. package/src/server/codex-app-server-protocol.ts +0 -453
  29. package/src/server/codex-app-server.ts +0 -1350
  30. package/src/server/cursor-acp.ts +0 -819
  31. package/src/server/discovery.ts +0 -322
  32. package/src/server/event-store.ts +0 -1470
  33. package/src/server/events.ts +0 -252
  34. package/src/server/external-open.ts +0 -272
  35. package/src/server/gemini-acp.ts +0 -844
  36. package/src/server/gemini-cli.ts +0 -525
  37. package/src/server/generate-title.ts +0 -36
  38. package/src/server/git-manager.ts +0 -79
  39. package/src/server/git-repository.ts +0 -101
  40. package/src/server/harness-types.ts +0 -20
  41. package/src/server/keybindings.ts +0 -177
  42. package/src/server/machine-name.ts +0 -22
  43. package/src/server/paths.ts +0 -112
  44. package/src/server/process-utils.ts +0 -22
  45. package/src/server/project-icon.ts +0 -352
  46. package/src/server/project-metadata.ts +0 -10
  47. package/src/server/provider-catalog.ts +0 -85
  48. package/src/server/provider-settings.ts +0 -155
  49. package/src/server/quick-response.ts +0 -153
  50. package/src/server/read-models.ts +0 -275
  51. package/src/server/recovery.ts +0 -507
  52. package/src/server/restart.ts +0 -56
  53. package/src/server/server.ts +0 -244
  54. package/src/server/terminal-manager.ts +0 -350
  55. package/src/server/theme-settings.ts +0 -179
  56. package/src/server/update-manager.ts +0 -230
  57. package/src/server/usage/base-provider-usage.ts +0 -57
  58. package/src/server/usage/claude-usage.ts +0 -558
  59. package/src/server/usage/codex-usage.ts +0 -144
  60. package/src/server/usage/cursor-browser.ts +0 -120
  61. package/src/server/usage/cursor-cookies.ts +0 -390
  62. package/src/server/usage/cursor-usage.ts +0 -490
  63. package/src/server/usage/gemini-usage.ts +0 -24
  64. package/src/server/usage/provider-usage.ts +0 -61
  65. package/src/server/usage/test-helpers.ts +0 -9
  66. package/src/server/usage/types.ts +0 -54
  67. package/src/server/usage/utils.ts +0 -325
  68. package/src/server/ws-router.ts +0 -742
  69. package/src/shared/branding.ts +0 -83
  70. package/src/shared/dev-ports.ts +0 -43
  71. package/src/shared/ports.ts +0 -2
  72. package/src/shared/protocol.ts +0 -156
  73. package/src/shared/tools.ts +0 -251
  74. package/src/shared/types.ts +0 -1040
@@ -0,0 +1,306 @@
1
+ // @bun
2
+ // src/server/cli-supervisor.ts
3
+ import process from "process";
4
+ import { spawn } from "child_process";
5
+ import { fileURLToPath } from "url";
6
+ // package.json
7
+ var package_default = {
8
+ name: "kaizenai",
9
+ type: "module",
10
+ version: "0.5.0",
11
+ description: "Local web UI for coding agents",
12
+ keywords: [
13
+ "claude",
14
+ "claude-code",
15
+ "ai",
16
+ "chat",
17
+ "ui",
18
+ "agent",
19
+ "anthropic",
20
+ "bun",
21
+ "react"
22
+ ],
23
+ repository: {
24
+ type: "git",
25
+ url: "git+https://github.com/jayleaton/kaizen.git"
26
+ },
27
+ homepage: "https://github.com/jayleaton/kaizen#readme",
28
+ bugs: {
29
+ url: "https://github.com/jayleaton/kaizen/issues"
30
+ },
31
+ bin: {
32
+ kaizen: "bin/kaizen"
33
+ },
34
+ publishConfig: {
35
+ access: "public"
36
+ },
37
+ packageManager: "bun@1.3.5",
38
+ files: [
39
+ "bin/",
40
+ "dist/server/",
41
+ "dist/client/"
42
+ ],
43
+ engines: {
44
+ bun: ">=1.3.5"
45
+ },
46
+ scripts: {
47
+ build: "vite build && bun run ./scripts/build-server.ts",
48
+ check: "tsc --noEmit && bun run build",
49
+ dev: "bun run ./scripts/dev.ts",
50
+ "dev:client": "vite --host 0.0.0.0 --port 5174",
51
+ "dev:server": "bun run ./scripts/dev-server.ts --no-open --port 5175",
52
+ fmt: "prettier package.json README.md --check",
53
+ "install:dev": "bun install && bun run build && bun install -g .",
54
+ lint: "tsc --noEmit",
55
+ start: "bun run ./dist/server/cli.js",
56
+ test: "bun test",
57
+ typecheck: "tsc --noEmit",
58
+ prepublishOnly: "bun run build"
59
+ },
60
+ dependencies: {
61
+ "@anthropic-ai/claude-agent-sdk": "^0.2.39",
62
+ "@radix-ui/react-context-menu": "^2.2.16",
63
+ "@radix-ui/react-select": "^2.2.6",
64
+ "@xterm/addon-fit": "^0.11.0",
65
+ "@xterm/addon-serialize": "^0.14.0",
66
+ "@xterm/addon-web-links": "^0.12.0",
67
+ "@xterm/headless": "^6.0.0",
68
+ "@xterm/xterm": "^6.0.0",
69
+ "default-shell": "^2.2.0",
70
+ "puppeteer-core": "^24.40.0",
71
+ "react-resizable-panels": "^4.7.3",
72
+ sonner: "^2.0.7"
73
+ },
74
+ devDependencies: {
75
+ "@dnd-kit/core": "^6.3.1",
76
+ "@dnd-kit/sortable": "^10.0.0",
77
+ "@dnd-kit/utilities": "^3.2.2",
78
+ "@radix-ui/react-dialog": "^1.1.15",
79
+ "@radix-ui/react-popover": "^1.1.15",
80
+ "@radix-ui/react-tooltip": "^1.2.8",
81
+ "@tailwindcss/postcss": "^4.1.18",
82
+ "@tailwindcss/typography": "^0.5.19",
83
+ "@kaizen-core/themes": "workspace:*",
84
+ "@types/bun": "latest",
85
+ "@types/node": "^24.10.1",
86
+ "@types/react": "19.2.7",
87
+ "@types/react-dom": "19.2.3",
88
+ "@vitejs/plugin-react": "5.1.1",
89
+ autoprefixer: "^10.4.23",
90
+ "class-variance-authority": "^0.7.1",
91
+ clsx: "^2.1.1",
92
+ "lucide-react": "^0.562.0",
93
+ react: "19.2.1",
94
+ "react-dom": "19.2.1",
95
+ "react-markdown": "^10.1.0",
96
+ "react-router-dom": "^7.12.0",
97
+ prettier: "^3.6.2",
98
+ "remark-gfm": "^4.0.1",
99
+ "tailwind-merge": "^3.4.0",
100
+ tailwindcss: "^4.1.18",
101
+ typescript: "5.8.3",
102
+ vite: "^6.0.0",
103
+ zustand: "^5.0.10"
104
+ }
105
+ };
106
+
107
+ // src/shared/branding.ts
108
+ var APP_NAME = "Kaizen";
109
+ var CLI_COMMAND = "kaizen";
110
+ var DATA_ROOT_NAME = ".kaizen";
111
+ var DEV_DATA_ROOT_NAME = ".kaizen-dev";
112
+ var PACKAGE_NAME = "kaizenai";
113
+ var PROJECT_METADATA_DIR_NAME = ".kaizen";
114
+ var RUNTIME_PROFILE_ENV_VAR = "KAIZEN_RUNTIME_PROFILE";
115
+ var DISABLE_SELF_UPDATE_ENV_VAR = "KAIZEN_DISABLE_SELF_UPDATE";
116
+ var SDK_CLIENT_APP = `kaizen/${package_default.version}`;
117
+ var LOG_PREFIX = "[kaizen]";
118
+ var DEFAULT_NEW_PROJECT_ROOT = `~/${APP_NAME}`;
119
+ function getRuntimeEnv() {
120
+ const candidate = globalThis;
121
+ return candidate.process?.env;
122
+ }
123
+ function getRuntimeProfile(env = getRuntimeEnv()) {
124
+ const profile = env?.[RUNTIME_PROFILE_ENV_VAR];
125
+ return profile?.trim().toLowerCase() === "dev" ? "dev" : "prod";
126
+ }
127
+ function getDataRootName(env = getRuntimeEnv()) {
128
+ return getRuntimeProfile(env) === "dev" ? DEV_DATA_ROOT_NAME : DATA_ROOT_NAME;
129
+ }
130
+ function getDataRootDir(homeDir, env = getRuntimeEnv()) {
131
+ return `${homeDir}/${getDataRootName(env)}`;
132
+ }
133
+ function getDataRootDirDisplay(env = getRuntimeEnv()) {
134
+ return `~/${getDataRootName(env)}`;
135
+ }
136
+ function getDataDir(homeDir, env = getRuntimeEnv()) {
137
+ return `${getDataRootDir(homeDir, env)}/data`;
138
+ }
139
+ function getDataDirDisplay(env = getRuntimeEnv()) {
140
+ return `${getDataRootDirDisplay(env)}/data`;
141
+ }
142
+ function getKeybindingsFilePath(homeDir, env = getRuntimeEnv()) {
143
+ return `${getDataRootDir(homeDir, env)}/keybindings.json`;
144
+ }
145
+ function getThemeSettingsFilePath(homeDir, env = getRuntimeEnv()) {
146
+ return `${getDataRootDir(homeDir, env)}/theme.json`;
147
+ }
148
+ function getThemesDirPath(homeDir, env = getRuntimeEnv()) {
149
+ return `${getDataRootDir(homeDir, env)}/themes`;
150
+ }
151
+ function getThemesTemplateFilePath(homeDir, env = getRuntimeEnv()) {
152
+ return `${getThemesDirPath(homeDir, env)}/custom-theme.example.json`;
153
+ }
154
+ function getThemesReadmeFilePath(homeDir, env = getRuntimeEnv()) {
155
+ return `${getThemesDirPath(homeDir, env)}/THEME_README.md`;
156
+ }
157
+ function getProviderSettingsFilePath(homeDir, env = getRuntimeEnv()) {
158
+ return `${getDataRootDir(homeDir, env)}/providers.json`;
159
+ }
160
+
161
+ // src/server/restart.ts
162
+ var CLI_CHILD_MODE_ENV_VAR = "KAIZEN_CLI_MODE";
163
+ var CLI_CHILD_MODE = "child";
164
+ var CLI_STARTUP_UPDATE_RESTART_EXIT_CODE = 75;
165
+ var CLI_UI_UPDATE_RESTART_EXIT_CODE = 76;
166
+ var CLI_CHILD_COMMAND_ENV_VAR = "KAIZEN_CLI_CHILD_COMMAND";
167
+ var CLI_CHILD_ARGS_ENV_VAR = "KAIZEN_CLI_CHILD_ARGS";
168
+ var CLI_SUPPRESS_OPEN_ONCE_ENV_VAR = "KAIZEN_SUPPRESS_OPEN_ONCE";
169
+ var CLI_SKIP_STARTUP_UPDATE_ONCE_ENV_VAR = "KAIZEN_SKIP_STARTUP_UPDATE_ONCE";
170
+ var CLI_WEB_ONLY_MODE_ENV_VAR = "KAIZEN_WEB_ONLY_MODE";
171
+ var MAX_UPDATE_RESTART_ATTEMPTS = 5;
172
+ function shouldRestartCliProcess(code, signal) {
173
+ return signal === null && (code === CLI_STARTUP_UPDATE_RESTART_EXIT_CODE || code === CLI_UI_UPDATE_RESTART_EXIT_CODE);
174
+ }
175
+ function isStartupUpdateRestart(code, signal) {
176
+ return signal === null && code === CLI_STARTUP_UPDATE_RESTART_EXIT_CODE;
177
+ }
178
+ function isUiUpdateRestart(code, signal) {
179
+ return signal === null && code === CLI_UI_UPDATE_RESTART_EXIT_CODE;
180
+ }
181
+ function shouldSkipStartupUpdateOnce(value) {
182
+ return value === "1";
183
+ }
184
+ function shouldRunWebOnlyMode(value) {
185
+ return value === "1";
186
+ }
187
+ function isUpdateRestart(code, signal) {
188
+ return isStartupUpdateRestart(code, signal) || isUiUpdateRestart(code, signal);
189
+ }
190
+ function getNextUpdateRestartAttempt(currentAttempt, code, signal) {
191
+ if (!isUpdateRestart(code, signal)) {
192
+ return 0;
193
+ }
194
+ return currentAttempt + 1;
195
+ }
196
+ function shouldStartWebOnlyFallback(updateRestartAttempt) {
197
+ return updateRestartAttempt >= MAX_UPDATE_RESTART_ATTEMPTS;
198
+ }
199
+ function parseChildArgsEnv(value) {
200
+ if (!value)
201
+ return [];
202
+ try {
203
+ const parsed = JSON.parse(value);
204
+ if (!Array.isArray(parsed) || !parsed.every((entry) => typeof entry === "string")) {
205
+ throw new Error("child args must be an array of strings");
206
+ }
207
+ return parsed;
208
+ } catch (error) {
209
+ const message = error instanceof Error ? error.message : String(error);
210
+ throw new Error(`Invalid ${CLI_CHILD_ARGS_ENV_VAR}: ${message}`);
211
+ }
212
+ }
213
+
214
+ // src/server/cli-supervisor.ts
215
+ function getChildProcessSpec(env = process.env) {
216
+ const command = env[CLI_CHILD_COMMAND_ENV_VAR] || CLI_COMMAND;
217
+ const args = parseChildArgsEnv(env[CLI_CHILD_ARGS_ENV_VAR]);
218
+ return { command, args };
219
+ }
220
+ function getFallbackChildProcessSpec() {
221
+ return {
222
+ command: process.execPath,
223
+ args: [fileURLToPath(new URL("./cli.ts", import.meta.url))]
224
+ };
225
+ }
226
+ function createChildEnv(env, options) {
227
+ return {
228
+ ...env,
229
+ [CLI_CHILD_MODE_ENV_VAR]: CLI_CHILD_MODE,
230
+ ...options.suppressOpen ? { [CLI_SUPPRESS_OPEN_ONCE_ENV_VAR]: "1" } : {},
231
+ ...options.skipStartupUpdate ? { [CLI_SKIP_STARTUP_UPDATE_ONCE_ENV_VAR]: "1" } : {},
232
+ ...options.webOnlyMode ? { [CLI_WEB_ONLY_MODE_ENV_VAR]: "1" } : {}
233
+ };
234
+ }
235
+ function spawnChild(argv, childProcess, options) {
236
+ const suppressOpenThisChild = suppressOpenOnNextChild;
237
+ const skipStartupUpdateThisChild = skipStartupUpdateOnNextChild;
238
+ suppressOpenOnNextChild = false;
239
+ skipStartupUpdateOnNextChild = false;
240
+ return new Promise((resolve, reject) => {
241
+ const child = spawn(childProcess.command, [...childProcess.args, ...argv], {
242
+ stdio: "inherit",
243
+ env: createChildEnv(process.env, {
244
+ suppressOpen: options.suppressOpen || suppressOpenThisChild,
245
+ skipStartupUpdate: options.skipStartupUpdate || skipStartupUpdateThisChild,
246
+ webOnlyMode: options.webOnlyMode
247
+ })
248
+ });
249
+ const forwardSignal = (signal) => {
250
+ if (child.exitCode !== null)
251
+ return;
252
+ child.kill(signal);
253
+ };
254
+ const onSigint = () => {
255
+ forwardSignal("SIGINT");
256
+ };
257
+ const onSigterm = () => {
258
+ forwardSignal("SIGTERM");
259
+ };
260
+ process.on("SIGINT", onSigint);
261
+ process.on("SIGTERM", onSigterm);
262
+ child.once("error", (error) => {
263
+ process.off("SIGINT", onSigint);
264
+ process.off("SIGTERM", onSigterm);
265
+ reject(error);
266
+ });
267
+ child.once("exit", (code, signal) => {
268
+ process.off("SIGINT", onSigint);
269
+ process.off("SIGTERM", onSigterm);
270
+ resolve({ code, signal });
271
+ });
272
+ });
273
+ }
274
+ var argv = process.argv.slice(2);
275
+ var suppressOpenOnNextChild = false;
276
+ var skipStartupUpdateOnNextChild = false;
277
+ var updateRestartAttempt = 0;
278
+ var childProcess = getChildProcessSpec();
279
+ while (true) {
280
+ const result = await spawnChild(argv, childProcess, {
281
+ suppressOpen: false,
282
+ skipStartupUpdate: false,
283
+ webOnlyMode: shouldRunWebOnlyMode(process.env[CLI_WEB_ONLY_MODE_ENV_VAR])
284
+ });
285
+ updateRestartAttempt = getNextUpdateRestartAttempt(updateRestartAttempt, result.code, result.signal);
286
+ if (shouldRestartCliProcess(result.code, result.signal)) {
287
+ if (shouldStartWebOnlyFallback(updateRestartAttempt)) {
288
+ console.error(`${LOG_PREFIX} update restart failed ${String(updateRestartAttempt)} times; starting web-only fallback`);
289
+ childProcess = getFallbackChildProcessSpec();
290
+ process.env[CLI_WEB_ONLY_MODE_ENV_VAR] = "1";
291
+ suppressOpenOnNextChild = isUiUpdateRestart(result.code, result.signal);
292
+ skipStartupUpdateOnNextChild = true;
293
+ updateRestartAttempt = 0;
294
+ continue;
295
+ }
296
+ suppressOpenOnNextChild = isUiUpdateRestart(result.code, result.signal);
297
+ skipStartupUpdateOnNextChild = isStartupUpdateRestart(result.code, result.signal);
298
+ console.log(`${LOG_PREFIX} supervisor restarting ${CLI_COMMAND} in the same terminal session`);
299
+ continue;
300
+ }
301
+ process.exit(result.code ?? (result.signal ? 1 : 0));
302
+ }
303
+ export {
304
+ getChildProcessSpec,
305
+ createChildEnv
306
+ };