@kizenapps/cli 0.4.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 (33) hide show
  1. package/dist/electron/main.js +30 -10
  2. package/dist/electron/main.js.map +1 -1
  3. package/dist/index.js +11 -3
  4. package/dist/index.js.map +1 -1
  5. package/dist/viewer/assets/index-B0d3If0o.js +582 -0
  6. package/dist/viewer/assets/index-B1mMv9H8.css +2 -0
  7. package/dist/viewer/assets/index-BTLnOfRu.css +2 -0
  8. package/dist/viewer/assets/index-B_uo-pdA.js +582 -0
  9. package/dist/viewer/assets/index-Bp6umXGR.js +582 -0
  10. package/dist/viewer/assets/index-Bsja5bOD.js +582 -0
  11. package/dist/viewer/assets/index-CAej9s9R.js +582 -0
  12. package/dist/viewer/assets/index-CCMFzDS-.css +2 -0
  13. package/dist/viewer/assets/index-Ci4aelpQ.js +582 -0
  14. package/dist/viewer/assets/index-CuVHD6vO.css +2 -0
  15. package/dist/viewer/assets/index-DQZ1fN2j.js +582 -0
  16. package/dist/viewer/assets/index-DSeA5UDI.js +582 -0
  17. package/dist/viewer/assets/index-w7OW3o4t.js +582 -0
  18. package/dist/viewer/index.html +2 -2
  19. package/package.json +1 -1
  20. package/dist/viewer/assets/index-BERinSUs.css +0 -2
  21. package/dist/viewer/assets/index-BExFCUNb.js +0 -582
  22. package/dist/viewer/assets/index-BTnaSAbf.js +0 -582
  23. package/dist/viewer/assets/index-BsYWmcLM.css +0 -2
  24. package/dist/viewer/assets/index-C9gGMSH0.js +0 -582
  25. package/dist/viewer/assets/index-CGp_x2sR.js +0 -582
  26. package/dist/viewer/assets/index-D1lc5mUR.js +0 -582
  27. package/dist/viewer/assets/index-DBXDPp8Y.css +0 -2
  28. package/dist/viewer/assets/index-DUFimvi2.js +0 -582
  29. package/dist/viewer/assets/index-DVM12um9.js +0 -582
  30. package/dist/viewer/assets/index-DiwdXOw0.css +0 -2
  31. package/dist/viewer/assets/index-DxGbfW9l.js +0 -582
  32. package/dist/viewer/assets/index-o7N1iv71.js +0 -582
  33. package/dist/viewer/assets/index-wLgWfI8W.js +0 -582
@@ -95,27 +95,43 @@ void app.whenReady().then(() => {
95
95
  win.webContents.openDevTools();
96
96
  });
97
97
  function parseProp(prop) {
98
- if (prop.type === "number") return Number(prop.value);
99
- if (prop.type === "boolean") return prop.value === "true";
100
- if (prop.type === "undefined") return void 0;
101
- if (prop.type === "object" && prop.value === "null") return null;
98
+ if (prop.type === "number") {
99
+ return Number(prop.value);
100
+ }
101
+ if (prop.type === "boolean") {
102
+ return prop.value === "true";
103
+ }
104
+ if (prop.type === "undefined") {
105
+ return void 0;
106
+ }
107
+ if (prop.type === "object" && prop.value === "null") {
108
+ return null;
109
+ }
102
110
  return prop.value ?? `[${prop.type}]`;
103
111
  }
104
112
  function serializeArg(arg) {
105
- if (arg.value !== void 0) return arg.value;
113
+ if (arg.value !== void 0) {
114
+ return arg.value;
115
+ }
106
116
  if (arg.preview) {
107
117
  const { preview } = arg;
108
118
  if (preview.subtype === "array") {
109
119
  return preview.properties.filter((p) => /^\d+$/.test(p.name)).sort((a, b) => parseInt(a.name) - parseInt(b.name)).map(parseProp);
110
120
  }
111
121
  const obj = {};
112
- for (const prop of preview.properties) obj[prop.name] = parseProp(prop);
122
+ for (const prop of preview.properties) {
123
+ obj[prop.name] = parseProp(prop);
124
+ }
113
125
  return obj;
114
126
  }
115
127
  return arg.description ?? `[${arg.type}]`;
116
128
  }
117
129
  const enableRuntime = (sessionId) => {
118
- void win.webContents.debugger.sendCommand("Runtime.enable", { generatePreview: true }, sessionId);
130
+ void win.webContents.debugger.sendCommand(
131
+ "Runtime.enable",
132
+ { generatePreview: true },
133
+ sessionId
134
+ );
119
135
  };
120
136
  win.webContents.debugger.attach("1.3");
121
137
  enableRuntime();
@@ -124,15 +140,19 @@ void app.whenReady().then(() => {
124
140
  waitForDebuggerOnStart: false,
125
141
  flatten: true
126
142
  });
127
- win.webContents.debugger.on("message", (_event, method, params, sessionId) => {
143
+ win.webContents.debugger.on("message", (_event, method, params) => {
128
144
  if (method === "Target.attachedToTarget") {
129
145
  const { sessionId: childSessionId } = params;
130
146
  enableRuntime(childSessionId);
131
147
  return;
132
148
  }
133
- if (method !== "Runtime.consoleAPICalled") return;
149
+ if (method !== "Runtime.consoleAPICalled") {
150
+ return;
151
+ }
134
152
  const { type, args: args2 } = params;
135
- if (args2[0]?.type === "string" && String(args2[0].value).includes("Electron Security Warning")) return;
153
+ if (args2[0]?.type === "string" && String(args2[0].value).includes("Electron Security Warning")) {
154
+ return;
155
+ }
136
156
  const level = type === "warning" ? "warn" : ["log", "warn", "error", "info"].includes(type) ? type : "log";
137
157
  win.webContents.send("console-message", { level, args: args2.map(serializeArg) });
138
158
  });
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/electron/main.ts"],"sourcesContent":["import { app, BrowserWindow, session, nativeImage, Menu, ipcMain } from 'electron';\nimport { join, dirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\n\nconst args = process.argv.slice(2);\nconst port = parseInt(args.find((a) => a.startsWith('--port='))?.slice(7) ?? '3121', 10);\nconst userDataDir = args.find((a) => a.startsWith('--user-data-dir='))?.slice(16);\n\nif (userDataDir) {\n app.setPath('userData', userDataDir);\n}\n\napp.setName('Kizen App Builder');\n\nconst CSP_HEADERS = new Set([\n 'content-security-policy',\n 'content-security-policy-report-only',\n 'x-frame-options',\n]);\n\nconst KIZEN_DOMAINS = ['kizen.dev', 'kizen.com'];\n\nfunction isKizenUrl(url: string): boolean {\n try {\n const { hostname } = new URL(url);\n return KIZEN_DOMAINS.some((d) => hostname === d || hostname.endsWith('.' + d));\n } catch {\n return false;\n }\n}\n\nfunction setupSession(): void {\n // Strip CSP headers and fix SameSite cookies on Kizen responses.\n session.defaultSession.webRequest.onHeadersReceived((details, callback) => {\n const headers: Record<string, string[]> = {};\n\n for (const [key, value] of Object.entries(details.responseHeaders ?? {})) {\n if (!CSP_HEADERS.has(key.toLowerCase())) {\n headers[key] = value;\n }\n }\n\n if (isKizenUrl(details.url) && headers['set-cookie']) {\n headers['set-cookie'] = headers['set-cookie'].map((cookie) => {\n if (!/SameSite=/i.test(cookie)) {\n return cookie + '; SameSite=None; Secure';\n }\n\n return cookie.replace(/SameSite=\\w+/i, 'SameSite=None');\n });\n }\n\n callback({ responseHeaders: headers });\n });\n\n // Periodically re-patch existing Kizen cookies that arrived without SameSite=None.\n // Needed for cookies set before the interceptor was active or via JS document.cookie.\n setInterval(() => {\n void (async () => {\n for (const domain of KIZEN_DOMAINS) {\n try {\n const cookies = await session.defaultSession.cookies.get({ domain });\n for (const cookie of cookies) {\n if (cookie.sameSite !== 'no_restriction') {\n const raw = cookie.domain ?? domain;\n const bare = raw.startsWith('.') ? raw.slice(1) : raw;\n await session.defaultSession.cookies.set({\n url: `https://${bare}`,\n name: cookie.name,\n value: cookie.value,\n domain: raw,\n path: cookie.path ?? '/',\n secure: true,\n ...(cookie.httpOnly !== undefined && { httpOnly: cookie.httpOnly }),\n ...(cookie.expirationDate !== undefined && {\n expirationDate: cookie.expirationDate,\n }),\n sameSite: 'no_restriction',\n });\n }\n }\n } catch {\n /* ignore */\n }\n }\n })();\n }, 2000);\n}\n\nvoid app.whenReady().then(() => {\n Menu.setApplicationMenu(null);\n\n setupSession();\n\n const iconPath = join(dirname(fileURLToPath(import.meta.url)), 'icon.png');\n const icon = nativeImage.createFromPath(iconPath);\n\n if (process.platform === 'darwin' && !icon.isEmpty()) {\n app.dock?.setIcon(icon);\n }\n\n const win = new BrowserWindow({\n width: 1920,\n height: 1080,\n title: 'Kizen App Builder',\n icon,\n webPreferences: {\n nodeIntegration: false,\n contextIsolation: true,\n sandbox: false,\n preload: join(dirname(fileURLToPath(import.meta.url)), 'preload.cjs'),\n },\n });\n\n ipcMain.on('open-devtools', () => {\n win.webContents.openDevTools();\n });\n\n interface PropertyPreview { name: string; type: string; value?: string; subtype?: string }\n interface ObjectPreview { type: string; subtype?: string; overflow: boolean; properties: PropertyPreview[] }\n interface RemoteObject { type: string; subtype?: string; value?: unknown; description?: string; preview?: ObjectPreview }\n interface ConsoleApiCalledParams { type: string; args: RemoteObject[] }\n\n function parseProp(prop: PropertyPreview): unknown {\n if (prop.type === 'number') return Number(prop.value);\n if (prop.type === 'boolean') return prop.value === 'true';\n if (prop.type === 'undefined') return undefined;\n if (prop.type === 'object' && prop.value === 'null') return null;\n return prop.value ?? `[${prop.type}]`;\n }\n\n function serializeArg(arg: RemoteObject): unknown {\n if (arg.value !== undefined) return arg.value;\n if (arg.preview) {\n const { preview } = arg;\n if (preview.subtype === 'array') {\n return preview.properties\n .filter((p) => /^\\d+$/.test(p.name))\n .sort((a, b) => parseInt(a.name) - parseInt(b.name))\n .map(parseProp);\n }\n const obj: Record<string, unknown> = {};\n for (const prop of preview.properties) obj[prop.name] = parseProp(prop);\n return obj;\n }\n return arg.description ?? `[${arg.type}]`;\n }\n\n const enableRuntime = (sessionId?: string): void => {\n void win.webContents.debugger.sendCommand('Runtime.enable', { generatePreview: true }, sessionId);\n };\n\n win.webContents.debugger.attach('1.3');\n enableRuntime();\n void win.webContents.debugger.sendCommand('Target.setAutoAttach', {\n autoAttach: true,\n waitForDebuggerOnStart: false,\n flatten: true,\n });\n\n win.webContents.debugger.on('message', (_event, method, params, sessionId) => {\n if (method === 'Target.attachedToTarget') {\n const { sessionId: childSessionId } = params as { sessionId: string };\n enableRuntime(childSessionId);\n return;\n }\n\n if (method !== 'Runtime.consoleAPICalled') return;\n const { type, args } = params as ConsoleApiCalledParams;\n if (args[0]?.type === 'string' && String(args[0].value).includes('Electron Security Warning')) return;\n const level = type === 'warning' ? 'warn' : (['log', 'warn', 'error', 'info'].includes(type) ? type : 'log');\n win.webContents.send('console-message', { level, args: args.map(serializeArg) });\n });\n\n void win.loadURL(`http://localhost:${String(port)}`);\n});\n\napp.on('window-all-closed', () => {\n app.quit();\n});\n"],"mappings":"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","names":["args"]}
1
+ {"version":3,"sources":["../../src/electron/main.ts"],"sourcesContent":["import { app, BrowserWindow, session, nativeImage, Menu, ipcMain } from 'electron';\nimport { join, dirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\n\nconst args = process.argv.slice(2);\nconst port = parseInt(args.find((a) => a.startsWith('--port='))?.slice(7) ?? '3121', 10);\nconst userDataDir = args.find((a) => a.startsWith('--user-data-dir='))?.slice(16);\n\nif (userDataDir) {\n app.setPath('userData', userDataDir);\n}\n\napp.setName('Kizen App Builder');\n\nconst CSP_HEADERS = new Set([\n 'content-security-policy',\n 'content-security-policy-report-only',\n 'x-frame-options',\n]);\n\nconst KIZEN_DOMAINS = ['kizen.dev', 'kizen.com'];\n\nfunction isKizenUrl(url: string): boolean {\n try {\n const { hostname } = new URL(url);\n return KIZEN_DOMAINS.some((d) => hostname === d || hostname.endsWith('.' + d));\n } catch {\n return false;\n }\n}\n\nfunction setupSession(): void {\n // Strip CSP headers and fix SameSite cookies on Kizen responses.\n session.defaultSession.webRequest.onHeadersReceived((details, callback) => {\n const headers: Record<string, string[]> = {};\n\n for (const [key, value] of Object.entries(details.responseHeaders ?? {})) {\n if (!CSP_HEADERS.has(key.toLowerCase())) {\n headers[key] = value;\n }\n }\n\n if (isKizenUrl(details.url) && headers['set-cookie']) {\n headers['set-cookie'] = headers['set-cookie'].map((cookie) => {\n if (!/SameSite=/i.test(cookie)) {\n return cookie + '; SameSite=None; Secure';\n }\n\n return cookie.replace(/SameSite=\\w+/i, 'SameSite=None');\n });\n }\n\n callback({ responseHeaders: headers });\n });\n\n // Periodically re-patch existing Kizen cookies that arrived without SameSite=None.\n // Needed for cookies set before the interceptor was active or via JS document.cookie.\n setInterval(() => {\n void (async () => {\n for (const domain of KIZEN_DOMAINS) {\n try {\n const cookies = await session.defaultSession.cookies.get({ domain });\n for (const cookie of cookies) {\n if (cookie.sameSite !== 'no_restriction') {\n const raw = cookie.domain ?? domain;\n const bare = raw.startsWith('.') ? raw.slice(1) : raw;\n await session.defaultSession.cookies.set({\n url: `https://${bare}`,\n name: cookie.name,\n value: cookie.value,\n domain: raw,\n path: cookie.path ?? '/',\n secure: true,\n ...(cookie.httpOnly !== undefined && { httpOnly: cookie.httpOnly }),\n ...(cookie.expirationDate !== undefined && {\n expirationDate: cookie.expirationDate,\n }),\n sameSite: 'no_restriction',\n });\n }\n }\n } catch {\n /* ignore */\n }\n }\n })();\n }, 2000);\n}\n\nvoid app.whenReady().then(() => {\n Menu.setApplicationMenu(null);\n\n setupSession();\n\n const iconPath = join(dirname(fileURLToPath(import.meta.url)), 'icon.png');\n const icon = nativeImage.createFromPath(iconPath);\n\n if (process.platform === 'darwin' && !icon.isEmpty()) {\n app.dock?.setIcon(icon);\n }\n\n const win = new BrowserWindow({\n width: 1920,\n height: 1080,\n title: 'Kizen App Builder',\n icon,\n webPreferences: {\n nodeIntegration: false,\n contextIsolation: true,\n sandbox: false,\n preload: join(dirname(fileURLToPath(import.meta.url)), 'preload.cjs'),\n },\n });\n\n ipcMain.on('open-devtools', () => {\n win.webContents.openDevTools();\n });\n\n interface PropertyPreview {\n name: string;\n type: string;\n value?: string;\n subtype?: string;\n }\n interface ObjectPreview {\n type: string;\n subtype?: string;\n overflow: boolean;\n properties: PropertyPreview[];\n }\n interface RemoteObject {\n type: string;\n subtype?: string;\n value?: unknown;\n description?: string;\n preview?: ObjectPreview;\n }\n interface ConsoleApiCalledParams {\n type: string;\n args: RemoteObject[];\n }\n\n function parseProp(prop: PropertyPreview): unknown {\n if (prop.type === 'number') {\n return Number(prop.value);\n }\n if (prop.type === 'boolean') {\n return prop.value === 'true';\n }\n if (prop.type === 'undefined') {\n return undefined;\n }\n if (prop.type === 'object' && prop.value === 'null') {\n return null;\n }\n return prop.value ?? `[${prop.type}]`;\n }\n\n function serializeArg(arg: RemoteObject): unknown {\n if (arg.value !== undefined) {\n return arg.value;\n }\n if (arg.preview) {\n const { preview } = arg;\n if (preview.subtype === 'array') {\n return preview.properties\n .filter((p) => /^\\d+$/.test(p.name))\n .sort((a, b) => parseInt(a.name) - parseInt(b.name))\n .map(parseProp);\n }\n const obj: Record<string, unknown> = {};\n for (const prop of preview.properties) {\n obj[prop.name] = parseProp(prop);\n }\n return obj;\n }\n return arg.description ?? `[${arg.type}]`;\n }\n\n const enableRuntime = (sessionId?: string): void => {\n void win.webContents.debugger.sendCommand(\n 'Runtime.enable',\n { generatePreview: true },\n sessionId,\n );\n };\n\n win.webContents.debugger.attach('1.3');\n enableRuntime();\n void win.webContents.debugger.sendCommand('Target.setAutoAttach', {\n autoAttach: true,\n waitForDebuggerOnStart: false,\n flatten: true,\n });\n\n win.webContents.debugger.on('message', (_event, method, params) => {\n if (method === 'Target.attachedToTarget') {\n const { sessionId: childSessionId } = params as { sessionId: string };\n enableRuntime(childSessionId);\n return;\n }\n\n if (method !== 'Runtime.consoleAPICalled') {\n return;\n }\n const { type, args } = params as ConsoleApiCalledParams;\n if (args[0]?.type === 'string' && String(args[0].value).includes('Electron Security Warning')) {\n return;\n }\n const level =\n type === 'warning' ? 'warn' : ['log', 'warn', 'error', 'info'].includes(type) ? type : 'log';\n win.webContents.send('console-message', { level, args: args.map(serializeArg) });\n });\n\n void win.loadURL(`http://localhost:${String(port)}`);\n});\n\napp.on('window-all-closed', () => {\n app.quit();\n});\n"],"mappings":"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","names":["args"]}
package/dist/index.js CHANGED
@@ -59,10 +59,11 @@ async function readLocalFiles(rootDir) {
59
59
  }
60
60
 
61
61
  // src/lib/runBuild.ts
62
- var serializePlugin = (plugin) => ({
62
+ var serializePlugin = (plugin, allReleaseNotes) => ({
63
63
  ...plugin,
64
64
  thumbnail: plugin.thumbnail ? Buffer.from(plugin.thumbnail).toString("base64") : null,
65
- kznFile: plugin.kznFile ? Buffer.from(plugin.kznFile).toString("base64") : null
65
+ kznFile: plugin.kznFile ? Buffer.from(plugin.kznFile).toString("base64") : null,
66
+ allReleaseNotes
66
67
  });
67
68
  async function runBuild(pluginDir, outputDir, onStep) {
68
69
  await mkdir(outputDir, { recursive: true });
@@ -79,7 +80,14 @@ async function runBuild(pluginDir, outputDir, onStep) {
79
80
  const packaged = packagePlugin(minified, manifests);
80
81
  const deployable = Object.values(packaged).map(transformDeployablePlugin);
81
82
  onStep?.("writing-bundle");
82
- const bundle = deployable.map(serializePlugin);
83
+ const bundle = deployable.map((plugin) => {
84
+ const notesDir = plugin.release_notes_directory?.replace(/\/$/, "");
85
+ const allReleaseNotes = notesDir ? files.filter((f) => f.path.startsWith(notesDir + "/") && f.path.endsWith(".md") && f.content).map((f) => ({
86
+ version: f.path.slice(f.path.lastIndexOf("/") + 1, -3),
87
+ notes: f.content
88
+ })).sort((a, b) => b.version.localeCompare(a.version, void 0, { numeric: true })) : [];
89
+ return serializePlugin(plugin, allReleaseNotes);
90
+ });
83
91
  await writeFile(join2(outputDir, "bundle.json"), JSON.stringify(bundle, null, 2), "utf-8");
84
92
  }
85
93
 
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/commands/build.ts","../src/ui/BuildUI.tsx","../src/lib/runBuild.ts","../src/lib/readFiles.ts","../src/ui/Logo.tsx","../src/lib/gitignore.ts","../src/commands/dev.ts","../src/ui/DevUI.tsx","../src/lib/config.ts","../src/ui/CredentialSetupUI.tsx","../src/lib/credentials.ts"],"sourcesContent":["import { program } from 'commander';\nimport { buildCommand } from './commands/build.js';\nimport { devCommand } from './commands/dev.js';\n\nprogram.name('appbuilder').description('Kizen plugin app builder').version('0.1.0');\n\nbuildCommand(program);\ndevCommand(program);\n\nprogram.parse();\n","import { createElement } from 'react';\nimport { render } from 'ink';\nimport type { Command } from 'commander';\nimport { BuildUI } from '../ui/BuildUI.js';\nimport { ensureGitignore } from '../lib/gitignore.js';\n\nexport function buildCommand(program: Command): void {\n program\n .command('build')\n .description('Bundle the plugin app into .kizenapp directory')\n .action(async () => {\n const pluginDir = process.cwd();\n const outputDir = `${pluginDir}/.kizenapp`;\n ensureGitignore(pluginDir);\n const { waitUntilExit } = render(createElement(BuildUI, { outputDir, pluginDir }));\n await waitUntilExit();\n });\n}\n","import type { FC } from 'react';\nimport { useEffect, useState } from 'react';\nimport { Box, Text, useApp } from 'ink';\nimport { runBuild } from '../lib/runBuild.js';\nimport type { BuildStepName } from '../lib/runBuild.js';\nimport { Logo } from './Logo.js';\n\ntype BuildStep = BuildStepName | 'done' | 'error';\n\ninterface BuildUIProps {\n outputDir: string;\n pluginDir: string;\n}\n\nconst SPINNER_FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];\n\nconst Spinner: FC = () => {\n const [frame, setFrame] = useState(0);\n\n useEffect(() => {\n const id = setInterval(() => {\n setFrame((prev) => (prev + 1) % SPINNER_FRAMES.length);\n }, 80);\n\n return () => {\n clearInterval(id);\n };\n }, []);\n\n return <Text color=\"cyan\">{SPINNER_FRAMES[frame] ?? '⠋'}</Text>;\n};\n\nconst STEPS: BuildStepName[] = [\n 'creating-dir',\n 'reading-files',\n 'minifying',\n 'packaging',\n 'writing-bundle',\n];\n\nconst STEP_LABELS: Record<BuildStepName, string> = {\n 'creating-dir': 'Creating .kizenapp directory',\n 'reading-files': 'Reading plugin files',\n minifying: 'Minifying scripts',\n packaging: 'Packaging plugin',\n 'writing-bundle': 'Writing bundle.json',\n};\n\nexport const BuildUI: FC<BuildUIProps> = ({ outputDir, pluginDir }) => {\n const app = useApp();\n const [step, setStep] = useState<BuildStep>('creating-dir');\n const [errorMessage, setErrorMessage] = useState<string | null>(null);\n\n useEffect(() => {\n void runBuild(pluginDir, outputDir, setStep)\n .then(() => {\n setStep('done');\n app.exit();\n })\n .catch((err: unknown) => {\n const message = err instanceof Error ? err.message : String(err);\n setErrorMessage(message);\n setStep('error');\n app.exit();\n });\n }, [outputDir, pluginDir, app]);\n\n const currentIndex = STEPS.indexOf(step as BuildStepName);\n const isError = step === 'error';\n const isDone = step === 'done';\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2}>\n <Logo />\n\n <Box flexDirection=\"column\" marginTop={1} marginBottom={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n\n <Box flexDirection=\"column\" gap={0}>\n {STEPS.map((s, i) => {\n const isActive = step === s;\n const isStepDone = isDone || currentIndex > i;\n const isFailed = isError && i === currentIndex;\n\n return (\n <Box key={s} gap={1}>\n {isFailed ? (\n <Text color=\"red\">✗ {STEP_LABELS[s]}</Text>\n ) : isStepDone ? (\n <Text color=\"green\">✓ {STEP_LABELS[s]}</Text>\n ) : isActive ? (\n <>\n <Spinner />\n <Text>{STEP_LABELS[s]}</Text>\n </>\n ) : (\n <Text dimColor> {STEP_LABELS[s]}</Text>\n )}\n </Box>\n );\n })}\n\n {errorMessage !== null && (\n <Box marginTop={1} borderStyle=\"single\" borderColor=\"red\" paddingX={1}>\n <Text color=\"red\">{errorMessage}</Text>\n </Box>\n )}\n </Box>\n </Box>\n );\n};\n","import { mkdir, writeFile } from 'node:fs/promises';\nimport { join } from 'node:path';\nimport { minifyFiles, packagePlugin, transformDeployablePlugin } from '@kizenapps/packager';\nimport type { DeployablePlugin } from '@kizenapps/packager';\nimport { readLocalFiles } from './readFiles.js';\n\nexport type BuildStepName =\n | 'creating-dir'\n | 'reading-files'\n | 'minifying'\n | 'packaging'\n | 'writing-bundle';\n\ntype SerializableDeployablePlugin = Omit<DeployablePlugin, 'thumbnail' | 'kznFile'> & {\n thumbnail: string | null;\n kznFile: string | null;\n};\n\nconst serializePlugin = (plugin: DeployablePlugin): SerializableDeployablePlugin => ({\n ...plugin,\n thumbnail: plugin.thumbnail ? Buffer.from(plugin.thumbnail).toString('base64') : null,\n kznFile: plugin.kznFile ? Buffer.from(plugin.kznFile).toString('base64') : null,\n});\n\nexport async function runBuild(\n pluginDir: string,\n outputDir: string,\n onStep?: (step: BuildStepName) => void,\n): Promise<void> {\n await mkdir(outputDir, { recursive: true });\n\n onStep?.('reading-files');\n const files = await readLocalFiles(pluginDir);\n\n onStep?.('minifying');\n const minified = await minifyFiles(files);\n\n onStep?.('packaging');\n const manifestFile = minified.find((f) => f.path === 'kizen.json');\n if (!manifestFile) {\n throw new Error('kizen.json not found in plugin directory.');\n }\n const manifests = JSON.parse(manifestFile.content) as Parameters<typeof packagePlugin>[1];\n const packaged = packagePlugin(minified, manifests);\n const deployable = Object.values(packaged).map(transformDeployablePlugin);\n\n onStep?.('writing-bundle');\n const bundle = deployable.map(serializePlugin);\n await writeFile(join(outputDir, 'bundle.json'), JSON.stringify(bundle, null, 2), 'utf-8');\n}\n","import { readdir, readFile } from 'node:fs/promises';\nimport { join, relative } from 'node:path';\nimport type { FileContent } from '@kizenapps/packager';\n\nconst IMAGE_EXTENSIONS = new Set(['.png', '.svg']);\nconst BINARY_EXTENSIONS = new Set(['.kzn']);\n\nconst SKIP_DIRS = new Set(['node_modules', '.git', '.kizenapp', '.github']);\n\nasync function walk(dir: string, rootDir: string): Promise<string[]> {\n const entries = await readdir(dir, { withFileTypes: true });\n const paths: string[] = [];\n\n for (const entry of entries) {\n if (entry.isDirectory()) {\n if (SKIP_DIRS.has(entry.name)) {\n continue;\n }\n\n paths.push(...(await walk(join(dir, entry.name), rootDir)));\n } else if (entry.isFile()) {\n paths.push(join(dir, entry.name));\n }\n }\n\n return paths;\n}\n\nexport async function readLocalFiles(rootDir: string): Promise<FileContent[]> {\n const absolutePaths = await walk(rootDir, rootDir);\n\n return Promise.all(\n absolutePaths.map(async (absPath): Promise<FileContent> => {\n const relPath = relative(rootDir, absPath).split('\\\\').join('/');\n const dotIndex = relPath.lastIndexOf('.');\n const ext = dotIndex >= 0 ? relPath.slice(dotIndex).toLowerCase() : '';\n\n if (IMAGE_EXTENSIONS.has(ext)) {\n const buf = await readFile(absPath);\n return { path: relPath, content: '', base64Image: buf.toString('base64') };\n }\n\n if (BINARY_EXTENSIONS.has(ext)) {\n const buf = await readFile(absPath);\n return { path: relPath, content: '', binaryData: buf };\n }\n\n const content = await readFile(absPath, 'utf-8');\n return { path: relPath, content };\n }),\n );\n}\n","import type { FC } from 'react';\nimport { Box, Text } from 'ink';\n\nconst LOGO_LINES = [\n ' @@@ ',\n ' @@@@@ @@@@@@@ ',\n ' @@@@ @@@@@@@@@@ ',\n ' @@@@ @@@@@ ',\n ' @@@ @@@@ @@@@@ @@@@@@@ ',\n ' @@@@ @@@@ @@@@ @@@@@@@@@@@@@',\n ' @@@@ @@@@ @@@@@',\n ' @@@@ @@@ @@@@ @@@',\n ' @@@@ @@@@@@ ',\n ' @@@@@@ @@@@ ',\n ' @@@ @@@@@ @@@ @@@@ ',\n ' @@@@@ @@@@ @@@@ ',\n ' @@@@@@@@@@@@@ @@@ @@@ @@@ ',\n ' @@@@@@@@ @@@@ @@@ @@@@ ',\n ' @@@@@ @@@@ ',\n ' @@@@@@@@@@@ @@@@ ',\n ' @@@@@@@ @@@@@ ',\n ' @@@@ ',\n];\n\nexport const Logo: FC = () => (\n <Box flexDirection=\"column\">\n {LOGO_LINES.map((line, i) => (\n <Text key={i} color=\"cyan\">\n {line}\n </Text>\n ))}\n </Box>\n);\n","import * as fs from 'node:fs';\nimport * as path from 'node:path';\n\nexport function ensureGitignore(projectDir: string): void {\n const gitignorePath = path.join(projectDir, '.gitignore');\n const entry = '.kizenapp/';\n\n let contents = '';\n if (fs.existsSync(gitignorePath)) {\n contents = fs.readFileSync(gitignorePath, 'utf8');\n const lines = contents.split('\\n').map((l) => l.trim());\n if (lines.includes(entry) || lines.includes('.kizenapp')) {\n return;\n }\n }\n\n const addition = contents.endsWith('\\n') ? entry + '\\n' : '\\n' + entry + '\\n';\n fs.writeFileSync(gitignorePath, contents + addition);\n}\n","import { createElement } from 'react';\nimport { render } from 'ink';\nimport type { Command } from 'commander';\nimport { DevUI } from '../ui/DevUI.js';\nimport { CredentialSetupUI } from '../ui/CredentialSetupUI.js';\nimport type { CredentialMode, CredentialSetupResult } from '../ui/CredentialSetupUI.js';\nimport { loadCredentialsFromFile, loadGlobalCredentials } from '../lib/credentials.js';\nimport type { Credentials } from '../lib/credentials.js';\nimport { loadConfig, saveConfig } from '../lib/config.js';\nimport { ensureGitignore } from '../lib/gitignore.js';\n\nasync function runSetupUI(initialMode?: CredentialMode): Promise<CredentialSetupResult> {\n return new Promise((resolve) => {\n let unmountFn: (() => void) | null = null;\n const handleComplete = (result: CredentialSetupResult): void => {\n unmountFn?.();\n resolve(result);\n };\n const { unmount } = render(\n createElement(CredentialSetupUI, {\n ...(initialMode !== undefined && { initialMode }),\n onComplete: handleComplete,\n }),\n { exitOnCtrlC: false },\n );\n unmountFn = unmount;\n });\n}\n\nexport function devCommand(program: Command): void {\n program\n .command('dev')\n .description('Start the plugin viewer dev server')\n .option('-p, --port <port>', 'port to listen on', '3121')\n .option('-c, --credentials <path>', 'path to a credentials JSON file')\n .action(async (options: { port: string; credentials?: string }) => {\n const port = parseInt(options.port, 10);\n const pluginDir = process.cwd();\n const outputDir = `${pluginDir}/.kizenapp`;\n ensureGitignore(pluginDir);\n\n let credentials: Credentials | null = null;\n let credentialMode: CredentialMode | undefined;\n\n if (options.credentials) {\n credentials = await loadCredentialsFromFile(options.credentials);\n } else {\n const config = await loadConfig(outputDir);\n\n if (config.credentialMode === 'local') {\n credentialMode = 'local';\n } else if (config.credentialMode === 'global') {\n credentialMode = 'global';\n // Load silently — only show TUI if the file was deleted since last run\n credentials = await loadGlobalCredentials();\n if (!credentials) {\n const result = await runSetupUI('global');\n credentials = result.credentials;\n }\n } else {\n // First run — show full mode selection TUI\n const result = await runSetupUI();\n credentials = result.credentials;\n credentialMode = result.mode;\n await saveConfig(outputDir, { credentialMode: result.mode });\n }\n }\n\n const { waitUntilExit } = render(\n createElement(DevUI, {\n port,\n pluginDir,\n outputDir,\n credentials,\n ...(credentialMode !== undefined && { credentialMode }),\n }),\n { exitOnCtrlC: false },\n );\n await waitUntilExit();\n });\n}\n","import type { FC } from 'react';\nimport { useCallback, useEffect, useRef, useState } from 'react';\nimport { Box, Text, useInput } from 'ink';\nimport { spawn } from 'node:child_process';\nimport { createReadStream, watch } from 'node:fs';\nimport type { FSWatcher } from 'node:fs';\nimport { access, readFile } from 'node:fs/promises';\nimport { createServer } from 'node:http';\nimport type { IncomingMessage, Server, ServerResponse } from 'node:http';\nimport { createRequire } from 'node:module';\nimport { dirname, extname, join } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { type WebSocket, WebSocketServer } from 'ws';\nimport { runBuild } from '../lib/runBuild.js';\nimport type { Credentials } from '../lib/credentials.js';\nimport { saveConfig } from '../lib/config.js';\nimport { CredentialSetupUI } from './CredentialSetupUI.js';\nimport type { CredentialMode, CredentialSetupResult } from './CredentialSetupUI.js';\nimport { Logo } from './Logo.js';\n\ntype ServerStatus = 'starting' | 'running' | 'error';\ntype BuildStatus = 'pending' | 'building' | 'done' | 'error';\n\nconst SKIP_WATCH_PREFIXES = ['.kizenapp', '.git'];\nconst LOG_LIMIT = 50;\nconst LOG_DISPLAY = 8;\n\ninterface DevUIProps {\n port: number;\n pluginDir: string;\n outputDir: string;\n credentials: Credentials | null;\n credentialMode?: CredentialMode;\n}\n\nconst MIME_TYPES: Readonly<Record<string, string>> = {\n '.html': 'text/html; charset=utf-8',\n '.js': 'application/javascript; charset=utf-8',\n '.mjs': 'application/javascript; charset=utf-8',\n '.css': 'text/css; charset=utf-8',\n '.png': 'image/png',\n '.jpg': 'image/jpeg',\n '.svg': 'image/svg+xml',\n '.ico': 'image/x-icon',\n '.json': 'application/json; charset=utf-8',\n '.woff': 'font/woff',\n '.woff2': 'font/woff2',\n '.ttf': 'font/ttf',\n '.map': 'application/json',\n};\n\nconst SPINNER_FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];\n\nconst Spinner: FC = () => {\n const [frame, setFrame] = useState(0);\n\n useEffect(() => {\n const id = setInterval(() => {\n setFrame((prev) => (prev + 1) % SPINNER_FRAMES.length);\n }, 80);\n\n return () => {\n clearInterval(id);\n };\n }, []);\n\n return <Text color=\"cyan\">{SPINNER_FRAMES[frame] ?? '⠋'}</Text>;\n};\n\nfunction getViewerPath(): string {\n const filename = fileURLToPath(import.meta.url);\n return join(dirname(filename), 'viewer');\n}\n\nfunction getElectronMainPath(): string {\n const filename = fileURLToPath(import.meta.url);\n return join(dirname(filename), 'electron', 'main.js');\n}\n\nasync function fileExists(filePath: string): Promise<boolean> {\n try {\n await access(filePath);\n return true;\n } catch {\n return false;\n }\n}\n\nfunction createRequestHandler(\n viewerPath: string,\n createServerLog: (message: string) => void,\n createProxyLog: (message: string) => void,\n credentialsRef: { current: Credentials | null },\n): (req: IncomingMessage, res: ServerResponse) => void {\n return (req, res) => {\n void (async () => {\n const url = req.url ?? '/';\n\n try {\n createServerLog(`Received request: ${url}`);\n\n if (url === '/api/credentials') {\n res.writeHead(200, { 'Content-Type': 'application/json; charset=utf-8' });\n res.end(credentialsRef.current !== null ? JSON.stringify(credentialsRef.current) : '{}');\n return;\n }\n\n if (url === '/api/bundle') {\n const bundlePath = join(process.cwd(), '.kizenapp', 'bundle.json');\n try {\n const content = await readFile(bundlePath, 'utf-8');\n res.writeHead(200, { 'Content-Type': 'application/json; charset=utf-8' });\n res.end(content);\n } catch {\n res.writeHead(200, { 'Content-Type': 'application/json; charset=utf-8' });\n res.end('{}');\n }\n return;\n }\n\n if (url.startsWith('/api/proxy')) {\n const proxyTarget = req.headers['x-proxy-target'];\n if (typeof proxyTarget !== 'string') {\n res.writeHead(400);\n res.end('Missing x-proxy-target header');\n return;\n }\n\n const upstreamPath = url.slice('/api/proxy'.length) || '/';\n const upstreamUrl = `${proxyTarget}${upstreamPath}`;\n\n const chunks: Buffer[] = [];\n for await (const chunk of req) {\n chunks.push(chunk as Buffer);\n }\n\n const body = chunks.length > 0 ? Buffer.concat(chunks) : undefined;\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n const { host, 'x-proxy-target': _drop, ...forwardHeaders } = req.headers;\n\n const resolvedBody = body && body.length > 0 ? body : undefined;\n const upstream = await fetch(upstreamUrl, {\n ...(req.method !== undefined && { method: req.method }),\n headers: forwardHeaders as Record<string, string>,\n ...(resolvedBody !== undefined && { body: resolvedBody }),\n });\n\n createProxyLog(`${req.method ?? 'GET'} ${upstreamPath} → ${String(upstream.status)}`);\n\n // Node fetch auto-decompresses the body, so strip encoding/length headers\n // that describe the compressed wire format — they no longer apply.\n const responseHeaders = Object.fromEntries(upstream.headers);\n delete responseHeaders['content-encoding'];\n delete responseHeaders['content-length'];\n\n res.writeHead(upstream.status, responseHeaders);\n res.end(Buffer.from(await upstream.arrayBuffer()));\n return;\n }\n\n const rawPath = url === '/' ? '/index.html' : url;\n const filePath = join(viewerPath, rawPath);\n const resolvedPath = (await fileExists(filePath))\n ? filePath\n : join(viewerPath, 'index.html');\n const ext = extname(resolvedPath);\n const mimeType = MIME_TYPES[ext] ?? 'application/octet-stream';\n res.writeHead(200, { 'Content-Type': mimeType });\n createReadStream(resolvedPath).pipe(res);\n } catch (err) {\n createProxyLog(\n `Error handling ${url}: ${err instanceof Error ? err.message : String(err)}`,\n );\n if (!res.headersSent) {\n res.writeHead(502);\n res.end('Bad Gateway');\n }\n }\n })();\n };\n}\n\nexport const DevUI: FC<DevUIProps> = ({\n port,\n pluginDir,\n outputDir,\n credentials: initialCredentials,\n}) => {\n const credentialsRef = useRef<Credentials | null>(initialCredentials);\n const [credMode, setCredMode] = useState<'main' | 'editing'>('main');\n\n const [status, setStatus] = useState<ServerStatus>('starting');\n const [errorMessage, setErrorMessage] = useState<string | null>(null);\n const [serverLogHistory, setServerLogHistory] = useState<string[]>([]);\n const [buildLogHistory, setBuildLogHistory] = useState<string[]>([]);\n const [proxyLogHistory, setProxyLogHistory] = useState<string[]>([]);\n const [buildStatus, setBuildStatus] = useState<BuildStatus>('pending');\n const [buildError, setBuildError] = useState<string | null>(null);\n const [lastBuilt, setLastBuilt] = useState<Date | null>(null);\n const [wsClientCount, setWsClientCount] = useState(0);\n\n const buildingRef = useRef(false);\n const electronLaunchedRef = useRef(false);\n const debounceTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);\n const wsClientsRef = useRef<Set<WebSocket>>(new Set());\n const pendingMessagesRef = useRef<string[]>([]);\n\n useInput(\n (input, key) => {\n if (input === 'q' || (key.ctrl && input === 'c')) {\n process.exit(0);\n }\n if (input === 'c') {\n setCredMode('editing');\n }\n if (input === 'd') {\n broadcast({ type: 'open-devtools' });\n }\n },\n { isActive: credMode === 'main' },\n );\n\n useEffect(() => {\n if (status !== 'running' || electronLaunchedRef.current) {\n return;\n }\n\n electronLaunchedRef.current = true;\n\n const _require = createRequire(import.meta.url);\n const electronBin = _require('electron') as string;\n const electronMain = getElectronMainPath();\n const userDataDir = join(outputDir, '.electron');\n\n const proc = spawn(\n electronBin,\n [electronMain, `--port=${String(port)}`, `--user-data-dir=${userDataDir}`],\n {\n stdio: 'ignore',\n },\n );\n proc.unref();\n process.on('exit', () => {\n proc.kill();\n });\n }, [status, port, outputDir]);\n\n const createServerLog = useCallback((message: string): void => {\n setServerLogHistory((h) =>\n [...h, `${new Date().toLocaleTimeString()}: ${message}`].slice(-LOG_LIMIT),\n );\n }, []);\n\n const broadcast = useCallback((msg: object): void => {\n const json = JSON.stringify(msg);\n if (wsClientsRef.current.size === 0) {\n pendingMessagesRef.current = [...pendingMessagesRef.current, json].slice(-LOG_LIMIT);\n return;\n }\n for (const client of wsClientsRef.current) {\n if (client.readyState === 1) {\n client.send(json);\n }\n }\n }, []);\n\n const handleCredentialsDone = useCallback(\n (result: CredentialSetupResult): void => {\n credentialsRef.current = result.credentials;\n setCredMode('main');\n void saveConfig(outputDir, { credentialMode: result.mode });\n broadcast({ type: 'credentials-updated', credentials: result.credentials });\n },\n [outputDir, broadcast],\n );\n\n const createProxyLog = useCallback(\n (message: string): void => {\n setProxyLogHistory((h) =>\n [...h, `${new Date().toLocaleTimeString()}: ${message}`].slice(-LOG_LIMIT),\n );\n broadcast({ type: 'proxy-log', message });\n },\n [broadcast],\n );\n\n const createBuildLog = useCallback(\n (message: string): void => {\n setBuildLogHistory((h) =>\n [...h, `${new Date().toLocaleTimeString()}: ${message}`].slice(-LOG_LIMIT),\n );\n broadcast({ type: 'log', message });\n },\n [broadcast],\n );\n\n const triggerBuild = useCallback(() => {\n if (buildingRef.current) {\n return;\n }\n\n buildingRef.current = true;\n\n setBuildStatus('building');\n setBuildError(null);\n createBuildLog('Build started');\n\n void runBuild(pluginDir, outputDir)\n .then(() => {\n createBuildLog('Build finished');\n setBuildStatus('done');\n setLastBuilt(new Date());\n\n createBuildLog('Notifying viewers to reload');\n for (const client of wsClientsRef.current) {\n if (client.readyState === 1) {\n client.send(JSON.stringify({ type: 'rebuild' }));\n }\n }\n })\n .catch((err: unknown) => {\n const message = err instanceof Error ? err.message : String(err);\n setBuildError(message);\n setBuildStatus('error');\n })\n .finally(() => {\n buildingRef.current = false;\n });\n }, [pluginDir, outputDir, createBuildLog]);\n\n useEffect(() => {\n triggerBuild();\n\n const watcher: FSWatcher = watch(pluginDir, { recursive: true }, (_, filename) => {\n if (!filename) {\n return;\n }\n\n const normalized = filename.replace(/\\\\/g, '/');\n\n if (SKIP_WATCH_PREFIXES.some((prefix) => normalized.startsWith(prefix))) {\n return;\n }\n\n createBuildLog(`File change detected: ${filename}`);\n\n if (debounceTimerRef.current !== null) {\n clearTimeout(debounceTimerRef.current);\n }\n debounceTimerRef.current = setTimeout(triggerBuild, 150);\n });\n\n return () => {\n watcher.close();\n\n if (debounceTimerRef.current !== null) {\n clearTimeout(debounceTimerRef.current);\n }\n };\n }, [pluginDir, triggerBuild, createBuildLog]);\n\n useEffect(() => {\n const viewerPath = getViewerPath();\n\n void fileExists(join(viewerPath, 'index.html')).then((viewerBuilt) => {\n if (!viewerBuilt) {\n setErrorMessage(\"Viewer not built. Run 'pnpm build:viewer' first.\");\n setStatus('error');\n return;\n }\n\n const handler = createRequestHandler(\n viewerPath,\n createServerLog,\n createProxyLog,\n credentialsRef,\n );\n const server: Server = createServer(handler);\n const wss = new WebSocketServer({ server });\n\n wss.on('connection', (ws: WebSocket) => {\n for (const json of pendingMessagesRef.current) {\n if (ws.readyState === 1) {\n ws.send(json);\n }\n }\n pendingMessagesRef.current = [];\n\n createServerLog('Viewer connected for live reload');\n wsClientsRef.current.add(ws);\n setWsClientCount(wsClientsRef.current.size);\n\n ws.on('close', () => {\n wsClientsRef.current.delete(ws);\n setWsClientCount(wsClientsRef.current.size);\n });\n });\n\n server.listen(port, () => {\n setStatus('running');\n createServerLog(`Server started on port ${String(port)}`);\n });\n\n server.on('error', (err: Error) => {\n setErrorMessage(err.message);\n setStatus('error');\n });\n\n return () => {\n wss.close();\n server.close();\n };\n });\n }, [port, createServerLog, createProxyLog]);\n\n if (credMode === 'editing') {\n return (\n <CredentialSetupUI\n onComplete={handleCredentialsDone}\n onCancel={() => {\n setCredMode('main');\n }}\n />\n );\n }\n\n const elapsedSeconds =\n lastBuilt !== null ? Math.round((Date.now() - lastBuilt.getTime()) / 1000) : null;\n const elapsedLabel =\n elapsedSeconds === null\n ? ''\n : elapsedSeconds < 5\n ? ' (just now)'\n : ` (${String(elapsedSeconds)}s ago)`;\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2}>\n {/* Logo + header */}\n <Logo />\n <Box flexDirection=\"column\" marginTop={1} marginBottom={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n\n {/* Build panel */}\n <Box flexDirection=\"column\">\n <Box gap={2} marginBottom={0}>\n <Text bold dimColor>\n Build\n </Text>\n {buildStatus === 'pending' && <Text dimColor>waiting…</Text>}\n {buildStatus === 'building' && (\n <Box gap={1}>\n <Spinner />\n <Text>Building plugin package…</Text>\n </Box>\n )}\n {buildStatus === 'done' && (\n <Box gap={1}>\n <Text color=\"green\">✓ Built</Text>\n {lastBuilt !== null && (\n <Text dimColor>\n at {lastBuilt.toLocaleTimeString()}\n {elapsedLabel}\n </Text>\n )}\n </Box>\n )}\n {buildStatus === 'error' && <Text color=\"red\">✗ {buildError ?? 'unknown error'}</Text>}\n </Box>\n <Box\n flexDirection=\"column\"\n height={LOG_DISPLAY}\n borderStyle=\"single\"\n borderColor=\"gray\"\n overflow=\"hidden\"\n >\n {buildLogHistory.slice(-LOG_DISPLAY).map((log, index) => (\n <Text key={index} dimColor wrap=\"truncate\">\n {log}\n </Text>\n ))}\n </Box>\n </Box>\n\n {/* Server panel */}\n <Box marginTop={1} flexDirection=\"column\">\n <Box gap={2}>\n <Text bold dimColor>\n Server\n </Text>\n {status === 'starting' && <Text dimColor>starting…</Text>}\n {status === 'running' && (\n <Box gap={2}>\n <Text color=\"green\">✓ Running</Text>\n <Text dimColor>\n ● {wsClientCount} viewer{wsClientCount !== 1 ? 's' : ''}\n </Text>\n </Box>\n )}\n {status === 'error' && <Text color=\"red\">✗ {errorMessage ?? 'Server error'}</Text>}\n </Box>\n <Box\n flexDirection=\"column\"\n height={LOG_DISPLAY}\n borderStyle=\"single\"\n borderColor=\"gray\"\n overflow=\"hidden\"\n >\n {serverLogHistory.slice(-LOG_DISPLAY).map((log, index) => (\n <Text key={index} dimColor wrap=\"truncate\">\n {log}\n </Text>\n ))}\n </Box>\n </Box>\n\n {/* Proxy panel */}\n <Box marginTop={1} flexDirection=\"column\">\n <Box gap={2}>\n <Text bold dimColor>\n Proxy\n </Text>\n <Text color=\"green\">✓ Active</Text>\n </Box>\n <Box\n flexDirection=\"column\"\n height={LOG_DISPLAY}\n borderStyle=\"single\"\n borderColor=\"gray\"\n overflow=\"hidden\"\n >\n {proxyLogHistory.length === 0 ? (\n <Text dimColor> No requests yet.</Text>\n ) : (\n proxyLogHistory.slice(-LOG_DISPLAY).map((log, i) => (\n <Text key={i} dimColor wrap=\"truncate\">\n {log}\n </Text>\n ))\n )}\n </Box>\n </Box>\n\n {/* Footer */}\n <Box marginTop={1} gap={1}>\n <Text dimColor>[</Text>\n <Text>q</Text>\n <Text dimColor>] quit [</Text>\n <Text>c</Text>\n <Text dimColor>] credentials [</Text>\n <Text>d</Text>\n <Text dimColor>] devtools</Text>\n </Box>\n </Box>\n );\n};\n","import { mkdir, readFile, writeFile } from 'node:fs/promises';\nimport { join } from 'node:path';\n\nexport interface AppBuilderConfig {\n credentialMode?: 'global' | 'local';\n}\n\nexport async function loadConfig(outputDir: string): Promise<AppBuilderConfig> {\n try {\n const content = await readFile(join(outputDir, 'config.json'), 'utf-8');\n return JSON.parse(content) as AppBuilderConfig;\n } catch {\n return {};\n }\n}\n\nexport async function saveConfig(outputDir: string, config: AppBuilderConfig): Promise<void> {\n await mkdir(outputDir, { recursive: true });\n await writeFile(join(outputDir, 'config.json'), JSON.stringify(config, null, 2), 'utf-8');\n}\n","import type { FC } from 'react';\nimport { useCallback, useEffect, useState } from 'react';\nimport { Box, Text, useInput } from 'ink';\nimport {\n ENVIRONMENTS,\n GLOBAL_CREDENTIALS_PATH,\n loadGlobalCredentials,\n saveGlobalCredentials,\n} from '../lib/credentials.js';\nimport type { Credentials } from '../lib/credentials.js';\nimport { Logo } from './Logo.js';\n\ntype Mode = 'global' | 'local';\n\ntype Phase =\n | { type: 'mode-select'; cursor: 0 | 1 }\n | { type: 'loading' }\n | { type: 'creds-entry'; field: number; values: Partial<Credentials>; envCursor: number }\n | { type: 'saving' }\n | { type: 'done' };\n\nconst FIELDS = ['apiKey', 'userId', 'businessId'] as const;\nconst FIELD_LABELS: Record<string, string> = {\n apiKey: 'API Key',\n userId: 'User ID',\n businessId: 'Business ID',\n};\n\nexport type CredentialMode = 'global' | 'local';\n\nexport interface CredentialSetupResult {\n mode: CredentialMode;\n credentials: Credentials | null;\n}\n\ninterface CredentialSetupUIProps {\n initialMode?: CredentialMode;\n onComplete: (result: CredentialSetupResult) => void;\n onCancel?: () => void;\n}\n\nconst Hint: FC<{ text: string }> = ({ text }) => <Text dimColor>{text}</Text>;\n\nexport const CredentialSetupUI: FC<CredentialSetupUIProps> = ({\n initialMode,\n onComplete,\n onCancel,\n}) => {\n const [phase, setPhase] = useState<Phase>(\n initialMode === 'global' ? { type: 'loading' } : { type: 'mode-select', cursor: 0 },\n );\n const [inputBuffer, setInputBuffer] = useState('');\n const [error, setError] = useState<string | null>(null);\n\n const handleModeChosen = useCallback(\n async (mode: Mode) => {\n if (mode === 'local') {\n onComplete({ mode: 'local', credentials: null });\n return;\n }\n\n setPhase({ type: 'loading' });\n\n const existing = await loadGlobalCredentials();\n const envCursor = existing ? Math.max(0, ENVIRONMENTS.indexOf(existing.environment)) : 0;\n setPhase({\n type: 'creds-entry',\n field: 0,\n values: existing ?? {},\n envCursor,\n });\n },\n [onComplete],\n );\n\n const handleSave = useCallback(\n async (values: Partial<Credentials>, envCursor: number) => {\n const environment = ENVIRONMENTS[envCursor] ?? 'go';\n const credentials: Credentials = {\n apiKey: values.apiKey ?? '',\n userId: values.userId ?? '',\n businessId: values.businessId ?? '',\n environment,\n };\n setPhase({ type: 'saving' });\n try {\n await saveGlobalCredentials(credentials);\n onComplete({ mode: 'global', credentials });\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n setPhase({ type: 'creds-entry', field: 0, values, envCursor });\n }\n },\n [onComplete],\n );\n\n useInput((input, key) => {\n if (key.ctrl && input === 'c') {\n process.exit(0);\n }\n\n if (key.escape) {\n onCancel?.();\n return;\n }\n\n if (phase.type === 'mode-select') {\n if (key.upArrow) {\n setPhase({ type: 'mode-select', cursor: 0 });\n } else if (key.downArrow) {\n setPhase({ type: 'mode-select', cursor: 1 });\n } else if (key.return) {\n const mode: Mode = phase.cursor === 0 ? 'global' : 'local';\n void handleModeChosen(mode);\n }\n return;\n }\n\n if (phase.type === 'creds-entry') {\n const { field, values, envCursor } = phase;\n const isEnvField = field === FIELDS.length;\n\n if (isEnvField) {\n if (key.leftArrow) {\n setPhase({\n ...phase,\n envCursor: (envCursor - 1 + ENVIRONMENTS.length) % ENVIRONMENTS.length,\n });\n } else if (key.rightArrow) {\n setPhase({ ...phase, envCursor: (envCursor + 1) % ENVIRONMENTS.length });\n } else if (key.upArrow) {\n setPhase({ ...phase, field: field - 1 });\n setInputBuffer(values[FIELDS[field - 1] as keyof typeof values] ?? '');\n } else if (key.return) {\n void handleSave(values, envCursor);\n }\n return;\n }\n\n const fieldName = FIELDS[field];\n if (!fieldName) {\n return;\n }\n\n if (key.backspace || key.delete) {\n setInputBuffer((prev) => prev.slice(0, -1));\n } else if (key.upArrow && field > 0) {\n const updatedValues = { ...values, [fieldName]: inputBuffer };\n const prevField = field - 1;\n setPhase({ ...phase, field: prevField, values: updatedValues });\n setInputBuffer(updatedValues[FIELDS[prevField] as keyof typeof updatedValues] ?? '');\n } else if (key.return || key.tab || key.downArrow) {\n // Advance to next field\n const updatedValues = { ...values, [fieldName]: inputBuffer };\n const nextField = field + 1;\n setPhase({ ...phase, field: nextField, values: updatedValues });\n if (nextField < FIELDS.length) {\n setInputBuffer(updatedValues[FIELDS[nextField] as keyof typeof updatedValues] ?? '');\n } else {\n setInputBuffer('');\n }\n } else if (input && !key.ctrl && !key.meta) {\n setInputBuffer((prev) => prev + input);\n }\n }\n });\n\n // Trigger global credential load when initialMode skips mode selection\n useEffect(() => {\n if (initialMode === 'global') {\n void handleModeChosen('global');\n }\n }, []); // eslint-disable-line react-hooks/exhaustive-deps\n\n // Reset input buffer when switching to creds-entry\n useEffect(() => {\n if (phase.type === 'creds-entry' && phase.field === 0) {\n setInputBuffer(phase.values.apiKey ?? '');\n }\n }, [phase.type]); // eslint-disable-line react-hooks/exhaustive-deps\n\n if (phase.type === 'loading' || phase.type === 'saving') {\n return (\n <Box paddingY={1} paddingX={2}>\n <Text dimColor>\n {phase.type === 'loading' ? 'Loading credentials…' : 'Saving credentials…'}\n </Text>\n </Box>\n );\n }\n\n if (phase.type === 'done') {\n return null;\n }\n\n if (phase.type === 'mode-select') {\n const options: { label: string; desc: string; mode: Mode }[] = [\n { label: 'Global', desc: `~/.kizenappbuilder/credentials.json`, mode: 'global' },\n { label: 'Local', desc: 'Enter credentials in the browser dev tools', mode: 'local' },\n ];\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2} gap={1}>\n <Logo />\n <Box flexDirection=\"column\" marginTop={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n <Box flexDirection=\"column\" gap={0}>\n <Text bold>Credential mode</Text>\n {options.map((opt, i) => {\n const selected = phase.cursor === i;\n return (\n <Box key={opt.mode} gap={2}>\n <Text {...(selected && { color: 'cyan' as const })}>{selected ? '❯' : ' '}</Text>\n <Text bold={selected} {...(selected && { color: 'cyan' as const })}>\n {opt.label}\n </Text>\n <Text dimColor>{opt.desc}</Text>\n </Box>\n );\n })}\n </Box>\n <Hint text=\"↑↓ to move · Enter to select · Ctrl+C to quit\" />\n </Box>\n );\n }\n\n // creds-entry phase\n const { field: activeField, values, envCursor } = phase;\n const isEnvField = activeField === FIELDS.length;\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2} gap={1}>\n <Logo />\n <Box flexDirection=\"column\" marginTop={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n <Box flexDirection=\"column\" gap={0}>\n <Text bold>Global credentials</Text>\n <Text dimColor>Saved to: {GLOBAL_CREDENTIALS_PATH}</Text>\n </Box>\n {error && <Text color=\"red\">Error: {error}</Text>}\n <Box flexDirection=\"column\" gap={0}>\n {FIELDS.map((name, i) => {\n const isActive = activeField === i;\n const displayValue = isActive ? inputBuffer : (values[name] ?? '');\n return (\n <Box key={name} gap={2}>\n <Box width={12}>\n <Text bold={isActive} {...(isActive && { color: 'cyan' as const })}>\n {FIELD_LABELS[name]}\n </Text>\n </Box>\n <Text {...(isActive && { color: 'cyan' as const })}>{'>'}</Text>\n <Text>{displayValue}</Text>\n {isActive && <Text color=\"cyan\">{'█'}</Text>}\n </Box>\n );\n })}\n <Box gap={2}>\n <Box width={12}>\n <Text bold={isEnvField} {...(isEnvField && { color: 'cyan' as const })}>\n Environment\n </Text>\n </Box>\n <Text {...(isEnvField && { color: 'cyan' as const })}>{'>'}</Text>\n <Box gap={1}>\n {ENVIRONMENTS.map((env, i) => {\n const isSelected = i === envCursor;\n if (isEnvField) {\n return (\n <Text key={env} bold={isSelected} {...(isSelected && { color: 'cyan' as const })}>\n {isSelected ? `[${env}]` : env}\n </Text>\n );\n }\n return (\n <Text key={env} dimColor={!isSelected} bold={isSelected}>\n {env}\n </Text>\n );\n })}\n </Box>\n </Box>\n </Box>\n {isEnvField ? (\n <Hint text=\"←→ to select · ↑↓ to move · Enter to save · Esc to cancel\" />\n ) : (\n <Hint text=\"↑↓ to move · Backspace to delete · Esc to cancel\" />\n )}\n </Box>\n );\n};\n","import { mkdir, readFile, writeFile } from 'node:fs/promises';\nimport { homedir } from 'node:os';\nimport { dirname, join } from 'node:path';\n\nexport const ENVIRONMENTS = ['go', 'fmo', 'staging', 'integration', 'test1'] as const;\nexport type Environment = (typeof ENVIRONMENTS)[number];\n\nexport interface Credentials {\n apiKey: string;\n userId: string;\n businessId: string;\n environment: Environment;\n}\n\nconst GLOBAL_CREDENTIALS_DIR = join(homedir(), '.kizenappbuilder');\nexport const GLOBAL_CREDENTIALS_PATH = join(GLOBAL_CREDENTIALS_DIR, 'credentials.json');\n\nfunction isValidEnvironment(value: unknown): value is Environment {\n return ENVIRONMENTS.includes(value as Environment);\n}\n\nfunction parseCredentials(raw: unknown): Credentials {\n if (typeof raw !== 'object' || raw === null) {\n throw new Error('Credentials must be a JSON object');\n }\n const obj = raw as Record<string, unknown>;\n const env = obj.environment;\n return {\n apiKey: typeof obj.apiKey === 'string' ? obj.apiKey : '',\n userId: typeof obj.userId === 'string' ? obj.userId : '',\n businessId: typeof obj.businessId === 'string' ? obj.businessId : '',\n environment: isValidEnvironment(env) ? env : 'go',\n };\n}\n\nexport async function loadCredentialsFromFile(filePath: string): Promise<Credentials> {\n const content = await readFile(filePath, 'utf-8');\n return parseCredentials(JSON.parse(content) as unknown);\n}\n\nexport async function loadGlobalCredentials(): Promise<Credentials | null> {\n try {\n return await loadCredentialsFromFile(GLOBAL_CREDENTIALS_PATH);\n } catch {\n return null;\n }\n}\n\nexport async function saveGlobalCredentials(credentials: Credentials): Promise<void> {\n await mkdir(dirname(GLOBAL_CREDENTIALS_PATH), { recursive: true });\n await writeFile(GLOBAL_CREDENTIALS_PATH, JSON.stringify(credentials, null, 2), 'utf-8');\n}\n"],"mappings":"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","names":["Box","Text","join","join","jsx","Text","Box","program","createElement","render","useCallback","useEffect","useState","Box","Text","useInput","readFile","dirname","join","mkdir","readFile","writeFile","join","useEffect","useState","Box","Text","mkdir","readFile","writeFile","join","jsx","jsxs","Text","useState","envCursor","values","isEnvField","useEffect","Box","jsx","jsxs","SPINNER_FRAMES","Spinner","useState","useEffect","Text","join","dirname","readFile","useInput","useCallback","Box","render","createElement","program"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/commands/build.ts","../src/ui/BuildUI.tsx","../src/lib/runBuild.ts","../src/lib/readFiles.ts","../src/ui/Logo.tsx","../src/lib/gitignore.ts","../src/commands/dev.ts","../src/ui/DevUI.tsx","../src/lib/config.ts","../src/ui/CredentialSetupUI.tsx","../src/lib/credentials.ts"],"sourcesContent":["import { program } from 'commander';\nimport { buildCommand } from './commands/build.js';\nimport { devCommand } from './commands/dev.js';\n\nprogram.name('appbuilder').description('Kizen plugin app builder').version('0.1.0');\n\nbuildCommand(program);\ndevCommand(program);\n\nprogram.parse();\n","import { createElement } from 'react';\nimport { render } from 'ink';\nimport type { Command } from 'commander';\nimport { BuildUI } from '../ui/BuildUI.js';\nimport { ensureGitignore } from '../lib/gitignore.js';\n\nexport function buildCommand(program: Command): void {\n program\n .command('build')\n .description('Bundle the plugin app into .kizenapp directory')\n .action(async () => {\n const pluginDir = process.cwd();\n const outputDir = `${pluginDir}/.kizenapp`;\n ensureGitignore(pluginDir);\n const { waitUntilExit } = render(createElement(BuildUI, { outputDir, pluginDir }));\n await waitUntilExit();\n });\n}\n","import type { FC } from 'react';\nimport { useEffect, useState } from 'react';\nimport { Box, Text, useApp } from 'ink';\nimport { runBuild } from '../lib/runBuild.js';\nimport type { BuildStepName } from '../lib/runBuild.js';\nimport { Logo } from './Logo.js';\n\ntype BuildStep = BuildStepName | 'done' | 'error';\n\ninterface BuildUIProps {\n outputDir: string;\n pluginDir: string;\n}\n\nconst SPINNER_FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];\n\nconst Spinner: FC = () => {\n const [frame, setFrame] = useState(0);\n\n useEffect(() => {\n const id = setInterval(() => {\n setFrame((prev) => (prev + 1) % SPINNER_FRAMES.length);\n }, 80);\n\n return () => {\n clearInterval(id);\n };\n }, []);\n\n return <Text color=\"cyan\">{SPINNER_FRAMES[frame] ?? '⠋'}</Text>;\n};\n\nconst STEPS: BuildStepName[] = [\n 'creating-dir',\n 'reading-files',\n 'minifying',\n 'packaging',\n 'writing-bundle',\n];\n\nconst STEP_LABELS: Record<BuildStepName, string> = {\n 'creating-dir': 'Creating .kizenapp directory',\n 'reading-files': 'Reading plugin files',\n minifying: 'Minifying scripts',\n packaging: 'Packaging plugin',\n 'writing-bundle': 'Writing bundle.json',\n};\n\nexport const BuildUI: FC<BuildUIProps> = ({ outputDir, pluginDir }) => {\n const app = useApp();\n const [step, setStep] = useState<BuildStep>('creating-dir');\n const [errorMessage, setErrorMessage] = useState<string | null>(null);\n\n useEffect(() => {\n void runBuild(pluginDir, outputDir, setStep)\n .then(() => {\n setStep('done');\n app.exit();\n })\n .catch((err: unknown) => {\n const message = err instanceof Error ? err.message : String(err);\n setErrorMessage(message);\n setStep('error');\n app.exit();\n });\n }, [outputDir, pluginDir, app]);\n\n const currentIndex = STEPS.indexOf(step as BuildStepName);\n const isError = step === 'error';\n const isDone = step === 'done';\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2}>\n <Logo />\n\n <Box flexDirection=\"column\" marginTop={1} marginBottom={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n\n <Box flexDirection=\"column\" gap={0}>\n {STEPS.map((s, i) => {\n const isActive = step === s;\n const isStepDone = isDone || currentIndex > i;\n const isFailed = isError && i === currentIndex;\n\n return (\n <Box key={s} gap={1}>\n {isFailed ? (\n <Text color=\"red\">✗ {STEP_LABELS[s]}</Text>\n ) : isStepDone ? (\n <Text color=\"green\">✓ {STEP_LABELS[s]}</Text>\n ) : isActive ? (\n <>\n <Spinner />\n <Text>{STEP_LABELS[s]}</Text>\n </>\n ) : (\n <Text dimColor> {STEP_LABELS[s]}</Text>\n )}\n </Box>\n );\n })}\n\n {errorMessage !== null && (\n <Box marginTop={1} borderStyle=\"single\" borderColor=\"red\" paddingX={1}>\n <Text color=\"red\">{errorMessage}</Text>\n </Box>\n )}\n </Box>\n </Box>\n );\n};\n","import { mkdir, writeFile } from 'node:fs/promises';\nimport { join } from 'node:path';\nimport { minifyFiles, packagePlugin, transformDeployablePlugin } from '@kizenapps/packager';\nimport type { DeployablePlugin } from '@kizenapps/packager';\nimport { readLocalFiles } from './readFiles.js';\n\nexport type BuildStepName =\n | 'creating-dir'\n | 'reading-files'\n | 'minifying'\n | 'packaging'\n | 'writing-bundle';\n\ninterface ReleaseNote { version: string; notes: string }\n\ntype SerializableDeployablePlugin = Omit<DeployablePlugin, 'thumbnail' | 'kznFile'> & {\n thumbnail: string | null;\n kznFile: string | null;\n allReleaseNotes: ReleaseNote[];\n};\n\nconst serializePlugin = (\n plugin: DeployablePlugin,\n allReleaseNotes: ReleaseNote[],\n): SerializableDeployablePlugin => ({\n ...plugin,\n thumbnail: plugin.thumbnail ? Buffer.from(plugin.thumbnail).toString('base64') : null,\n kznFile: plugin.kznFile ? Buffer.from(plugin.kznFile).toString('base64') : null,\n allReleaseNotes,\n});\n\nexport async function runBuild(\n pluginDir: string,\n outputDir: string,\n onStep?: (step: BuildStepName) => void,\n): Promise<void> {\n await mkdir(outputDir, { recursive: true });\n\n onStep?.('reading-files');\n const files = await readLocalFiles(pluginDir);\n\n onStep?.('minifying');\n const minified = await minifyFiles(files);\n\n onStep?.('packaging');\n const manifestFile = minified.find((f) => f.path === 'kizen.json');\n if (!manifestFile) {\n throw new Error('kizen.json not found in plugin directory.');\n }\n const manifests = JSON.parse(manifestFile.content) as Parameters<typeof packagePlugin>[1];\n const packaged = packagePlugin(minified, manifests);\n const deployable = Object.values(packaged).map(transformDeployablePlugin);\n\n onStep?.('writing-bundle');\n const bundle = deployable.map((plugin) => {\n const notesDir = plugin.release_notes_directory?.replace(/\\/$/, '');\n const allReleaseNotes: ReleaseNote[] = notesDir\n ? files\n .filter((f) => f.path.startsWith(notesDir + '/') && f.path.endsWith('.md') && f.content)\n .map((f) => ({\n version: f.path.slice(f.path.lastIndexOf('/') + 1, -3),\n notes: f.content,\n }))\n .sort((a, b) => b.version.localeCompare(a.version, undefined, { numeric: true }))\n : [];\n return serializePlugin(plugin, allReleaseNotes);\n });\n await writeFile(join(outputDir, 'bundle.json'), JSON.stringify(bundle, null, 2), 'utf-8');\n}\n","import { readdir, readFile } from 'node:fs/promises';\nimport { join, relative } from 'node:path';\nimport type { FileContent } from '@kizenapps/packager';\n\nconst IMAGE_EXTENSIONS = new Set(['.png', '.svg']);\nconst BINARY_EXTENSIONS = new Set(['.kzn']);\n\nconst SKIP_DIRS = new Set(['node_modules', '.git', '.kizenapp', '.github']);\n\nasync function walk(dir: string, rootDir: string): Promise<string[]> {\n const entries = await readdir(dir, { withFileTypes: true });\n const paths: string[] = [];\n\n for (const entry of entries) {\n if (entry.isDirectory()) {\n if (SKIP_DIRS.has(entry.name)) {\n continue;\n }\n\n paths.push(...(await walk(join(dir, entry.name), rootDir)));\n } else if (entry.isFile()) {\n paths.push(join(dir, entry.name));\n }\n }\n\n return paths;\n}\n\nexport async function readLocalFiles(rootDir: string): Promise<FileContent[]> {\n const absolutePaths = await walk(rootDir, rootDir);\n\n return Promise.all(\n absolutePaths.map(async (absPath): Promise<FileContent> => {\n const relPath = relative(rootDir, absPath).split('\\\\').join('/');\n const dotIndex = relPath.lastIndexOf('.');\n const ext = dotIndex >= 0 ? relPath.slice(dotIndex).toLowerCase() : '';\n\n if (IMAGE_EXTENSIONS.has(ext)) {\n const buf = await readFile(absPath);\n return { path: relPath, content: '', base64Image: buf.toString('base64') };\n }\n\n if (BINARY_EXTENSIONS.has(ext)) {\n const buf = await readFile(absPath);\n return { path: relPath, content: '', binaryData: buf };\n }\n\n const content = await readFile(absPath, 'utf-8');\n return { path: relPath, content };\n }),\n );\n}\n","import type { FC } from 'react';\nimport { Box, Text } from 'ink';\n\nconst LOGO_LINES = [\n ' @@@ ',\n ' @@@@@ @@@@@@@ ',\n ' @@@@ @@@@@@@@@@ ',\n ' @@@@ @@@@@ ',\n ' @@@ @@@@ @@@@@ @@@@@@@ ',\n ' @@@@ @@@@ @@@@ @@@@@@@@@@@@@',\n ' @@@@ @@@@ @@@@@',\n ' @@@@ @@@ @@@@ @@@',\n ' @@@@ @@@@@@ ',\n ' @@@@@@ @@@@ ',\n ' @@@ @@@@@ @@@ @@@@ ',\n ' @@@@@ @@@@ @@@@ ',\n ' @@@@@@@@@@@@@ @@@ @@@ @@@ ',\n ' @@@@@@@@ @@@@ @@@ @@@@ ',\n ' @@@@@ @@@@ ',\n ' @@@@@@@@@@@ @@@@ ',\n ' @@@@@@@ @@@@@ ',\n ' @@@@ ',\n];\n\nexport const Logo: FC = () => (\n <Box flexDirection=\"column\">\n {LOGO_LINES.map((line, i) => (\n <Text key={i} color=\"cyan\">\n {line}\n </Text>\n ))}\n </Box>\n);\n","import * as fs from 'node:fs';\nimport * as path from 'node:path';\n\nexport function ensureGitignore(projectDir: string): void {\n const gitignorePath = path.join(projectDir, '.gitignore');\n const entry = '.kizenapp/';\n\n let contents = '';\n if (fs.existsSync(gitignorePath)) {\n contents = fs.readFileSync(gitignorePath, 'utf8');\n const lines = contents.split('\\n').map((l) => l.trim());\n if (lines.includes(entry) || lines.includes('.kizenapp')) {\n return;\n }\n }\n\n const addition = contents.endsWith('\\n') ? entry + '\\n' : '\\n' + entry + '\\n';\n fs.writeFileSync(gitignorePath, contents + addition);\n}\n","import { createElement } from 'react';\nimport { render } from 'ink';\nimport type { Command } from 'commander';\nimport { DevUI } from '../ui/DevUI.js';\nimport { CredentialSetupUI } from '../ui/CredentialSetupUI.js';\nimport type { CredentialMode, CredentialSetupResult } from '../ui/CredentialSetupUI.js';\nimport { loadCredentialsFromFile, loadGlobalCredentials } from '../lib/credentials.js';\nimport type { Credentials } from '../lib/credentials.js';\nimport { loadConfig, saveConfig } from '../lib/config.js';\nimport { ensureGitignore } from '../lib/gitignore.js';\n\nasync function runSetupUI(initialMode?: CredentialMode): Promise<CredentialSetupResult> {\n return new Promise((resolve) => {\n let unmountFn: (() => void) | null = null;\n const handleComplete = (result: CredentialSetupResult): void => {\n unmountFn?.();\n resolve(result);\n };\n const { unmount } = render(\n createElement(CredentialSetupUI, {\n ...(initialMode !== undefined && { initialMode }),\n onComplete: handleComplete,\n }),\n { exitOnCtrlC: false },\n );\n unmountFn = unmount;\n });\n}\n\nexport function devCommand(program: Command): void {\n program\n .command('dev')\n .description('Start the plugin viewer dev server')\n .option('-p, --port <port>', 'port to listen on', '3121')\n .option('-c, --credentials <path>', 'path to a credentials JSON file')\n .action(async (options: { port: string; credentials?: string }) => {\n const port = parseInt(options.port, 10);\n const pluginDir = process.cwd();\n const outputDir = `${pluginDir}/.kizenapp`;\n ensureGitignore(pluginDir);\n\n let credentials: Credentials | null = null;\n let credentialMode: CredentialMode | undefined;\n\n if (options.credentials) {\n credentials = await loadCredentialsFromFile(options.credentials);\n } else {\n const config = await loadConfig(outputDir);\n\n if (config.credentialMode === 'local') {\n credentialMode = 'local';\n } else if (config.credentialMode === 'global') {\n credentialMode = 'global';\n // Load silently — only show TUI if the file was deleted since last run\n credentials = await loadGlobalCredentials();\n if (!credentials) {\n const result = await runSetupUI('global');\n credentials = result.credentials;\n }\n } else {\n // First run — show full mode selection TUI\n const result = await runSetupUI();\n credentials = result.credentials;\n credentialMode = result.mode;\n await saveConfig(outputDir, { credentialMode: result.mode });\n }\n }\n\n const { waitUntilExit } = render(\n createElement(DevUI, {\n port,\n pluginDir,\n outputDir,\n credentials,\n ...(credentialMode !== undefined && { credentialMode }),\n }),\n { exitOnCtrlC: false },\n );\n await waitUntilExit();\n });\n}\n","import type { FC } from 'react';\nimport { useCallback, useEffect, useRef, useState } from 'react';\nimport { Box, Text, useInput } from 'ink';\nimport { spawn } from 'node:child_process';\nimport { createReadStream, watch } from 'node:fs';\nimport type { FSWatcher } from 'node:fs';\nimport { access, readFile } from 'node:fs/promises';\nimport { createServer } from 'node:http';\nimport type { IncomingMessage, Server, ServerResponse } from 'node:http';\nimport { createRequire } from 'node:module';\nimport { dirname, extname, join } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { type WebSocket, WebSocketServer } from 'ws';\nimport { runBuild } from '../lib/runBuild.js';\nimport type { Credentials } from '../lib/credentials.js';\nimport { saveConfig } from '../lib/config.js';\nimport { CredentialSetupUI } from './CredentialSetupUI.js';\nimport type { CredentialMode, CredentialSetupResult } from './CredentialSetupUI.js';\nimport { Logo } from './Logo.js';\n\ntype ServerStatus = 'starting' | 'running' | 'error';\ntype BuildStatus = 'pending' | 'building' | 'done' | 'error';\n\nconst SKIP_WATCH_PREFIXES = ['.kizenapp', '.git'];\nconst LOG_LIMIT = 50;\nconst LOG_DISPLAY = 8;\n\ninterface DevUIProps {\n port: number;\n pluginDir: string;\n outputDir: string;\n credentials: Credentials | null;\n credentialMode?: CredentialMode;\n}\n\nconst MIME_TYPES: Readonly<Record<string, string>> = {\n '.html': 'text/html; charset=utf-8',\n '.js': 'application/javascript; charset=utf-8',\n '.mjs': 'application/javascript; charset=utf-8',\n '.css': 'text/css; charset=utf-8',\n '.png': 'image/png',\n '.jpg': 'image/jpeg',\n '.svg': 'image/svg+xml',\n '.ico': 'image/x-icon',\n '.json': 'application/json; charset=utf-8',\n '.woff': 'font/woff',\n '.woff2': 'font/woff2',\n '.ttf': 'font/ttf',\n '.map': 'application/json',\n};\n\nconst SPINNER_FRAMES = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏'];\n\nconst Spinner: FC = () => {\n const [frame, setFrame] = useState(0);\n\n useEffect(() => {\n const id = setInterval(() => {\n setFrame((prev) => (prev + 1) % SPINNER_FRAMES.length);\n }, 80);\n\n return () => {\n clearInterval(id);\n };\n }, []);\n\n return <Text color=\"cyan\">{SPINNER_FRAMES[frame] ?? '⠋'}</Text>;\n};\n\nfunction getViewerPath(): string {\n const filename = fileURLToPath(import.meta.url);\n return join(dirname(filename), 'viewer');\n}\n\nfunction getElectronMainPath(): string {\n const filename = fileURLToPath(import.meta.url);\n return join(dirname(filename), 'electron', 'main.js');\n}\n\nasync function fileExists(filePath: string): Promise<boolean> {\n try {\n await access(filePath);\n return true;\n } catch {\n return false;\n }\n}\n\nfunction createRequestHandler(\n viewerPath: string,\n createServerLog: (message: string) => void,\n createProxyLog: (message: string) => void,\n credentialsRef: { current: Credentials | null },\n): (req: IncomingMessage, res: ServerResponse) => void {\n return (req, res) => {\n void (async () => {\n const url = req.url ?? '/';\n\n try {\n createServerLog(`Received request: ${url}`);\n\n if (url === '/api/credentials') {\n res.writeHead(200, { 'Content-Type': 'application/json; charset=utf-8' });\n res.end(credentialsRef.current !== null ? JSON.stringify(credentialsRef.current) : '{}');\n return;\n }\n\n if (url === '/api/bundle') {\n const bundlePath = join(process.cwd(), '.kizenapp', 'bundle.json');\n try {\n const content = await readFile(bundlePath, 'utf-8');\n res.writeHead(200, { 'Content-Type': 'application/json; charset=utf-8' });\n res.end(content);\n } catch {\n res.writeHead(200, { 'Content-Type': 'application/json; charset=utf-8' });\n res.end('{}');\n }\n return;\n }\n\n if (url.startsWith('/api/proxy')) {\n const proxyTarget = req.headers['x-proxy-target'];\n if (typeof proxyTarget !== 'string') {\n res.writeHead(400);\n res.end('Missing x-proxy-target header');\n return;\n }\n\n const upstreamPath = url.slice('/api/proxy'.length) || '/';\n const upstreamUrl = `${proxyTarget}${upstreamPath}`;\n\n const chunks: Buffer[] = [];\n for await (const chunk of req) {\n chunks.push(chunk as Buffer);\n }\n\n const body = chunks.length > 0 ? Buffer.concat(chunks) : undefined;\n\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n const { host, 'x-proxy-target': _drop, ...forwardHeaders } = req.headers;\n\n const resolvedBody = body && body.length > 0 ? body : undefined;\n const upstream = await fetch(upstreamUrl, {\n ...(req.method !== undefined && { method: req.method }),\n headers: forwardHeaders as Record<string, string>,\n ...(resolvedBody !== undefined && { body: resolvedBody }),\n });\n\n createProxyLog(`${req.method ?? 'GET'} ${upstreamPath} → ${String(upstream.status)}`);\n\n // Node fetch auto-decompresses the body, so strip encoding/length headers\n // that describe the compressed wire format — they no longer apply.\n const responseHeaders = Object.fromEntries(upstream.headers);\n delete responseHeaders['content-encoding'];\n delete responseHeaders['content-length'];\n\n res.writeHead(upstream.status, responseHeaders);\n res.end(Buffer.from(await upstream.arrayBuffer()));\n return;\n }\n\n const rawPath = url === '/' ? '/index.html' : url;\n const filePath = join(viewerPath, rawPath);\n const resolvedPath = (await fileExists(filePath))\n ? filePath\n : join(viewerPath, 'index.html');\n const ext = extname(resolvedPath);\n const mimeType = MIME_TYPES[ext] ?? 'application/octet-stream';\n res.writeHead(200, { 'Content-Type': mimeType });\n createReadStream(resolvedPath).pipe(res);\n } catch (err) {\n createProxyLog(\n `Error handling ${url}: ${err instanceof Error ? err.message : String(err)}`,\n );\n if (!res.headersSent) {\n res.writeHead(502);\n res.end('Bad Gateway');\n }\n }\n })();\n };\n}\n\nexport const DevUI: FC<DevUIProps> = ({\n port,\n pluginDir,\n outputDir,\n credentials: initialCredentials,\n}) => {\n const credentialsRef = useRef<Credentials | null>(initialCredentials);\n const [credMode, setCredMode] = useState<'main' | 'editing'>('main');\n\n const [status, setStatus] = useState<ServerStatus>('starting');\n const [errorMessage, setErrorMessage] = useState<string | null>(null);\n const [serverLogHistory, setServerLogHistory] = useState<string[]>([]);\n const [buildLogHistory, setBuildLogHistory] = useState<string[]>([]);\n const [proxyLogHistory, setProxyLogHistory] = useState<string[]>([]);\n const [buildStatus, setBuildStatus] = useState<BuildStatus>('pending');\n const [buildError, setBuildError] = useState<string | null>(null);\n const [lastBuilt, setLastBuilt] = useState<Date | null>(null);\n const [wsClientCount, setWsClientCount] = useState(0);\n\n const buildingRef = useRef(false);\n const electronLaunchedRef = useRef(false);\n const debounceTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);\n const wsClientsRef = useRef<Set<WebSocket>>(new Set());\n const pendingMessagesRef = useRef<string[]>([]);\n\n useInput(\n (input, key) => {\n if (input === 'q' || (key.ctrl && input === 'c')) {\n process.exit(0);\n }\n if (input === 'c') {\n setCredMode('editing');\n }\n if (input === 'd') {\n broadcast({ type: 'open-devtools' });\n }\n },\n { isActive: credMode === 'main' },\n );\n\n useEffect(() => {\n if (status !== 'running' || electronLaunchedRef.current) {\n return;\n }\n\n electronLaunchedRef.current = true;\n\n const _require = createRequire(import.meta.url);\n const electronBin = _require('electron') as string;\n const electronMain = getElectronMainPath();\n const userDataDir = join(outputDir, '.electron');\n\n const proc = spawn(\n electronBin,\n [electronMain, `--port=${String(port)}`, `--user-data-dir=${userDataDir}`],\n {\n stdio: 'ignore',\n },\n );\n proc.unref();\n process.on('exit', () => {\n proc.kill();\n });\n }, [status, port, outputDir]);\n\n const createServerLog = useCallback((message: string): void => {\n setServerLogHistory((h) =>\n [...h, `${new Date().toLocaleTimeString()}: ${message}`].slice(-LOG_LIMIT),\n );\n }, []);\n\n const broadcast = useCallback((msg: object): void => {\n const json = JSON.stringify(msg);\n if (wsClientsRef.current.size === 0) {\n pendingMessagesRef.current = [...pendingMessagesRef.current, json].slice(-LOG_LIMIT);\n return;\n }\n for (const client of wsClientsRef.current) {\n if (client.readyState === 1) {\n client.send(json);\n }\n }\n }, []);\n\n const handleCredentialsDone = useCallback(\n (result: CredentialSetupResult): void => {\n credentialsRef.current = result.credentials;\n setCredMode('main');\n void saveConfig(outputDir, { credentialMode: result.mode });\n broadcast({ type: 'credentials-updated', credentials: result.credentials });\n },\n [outputDir, broadcast],\n );\n\n const createProxyLog = useCallback(\n (message: string): void => {\n setProxyLogHistory((h) =>\n [...h, `${new Date().toLocaleTimeString()}: ${message}`].slice(-LOG_LIMIT),\n );\n broadcast({ type: 'proxy-log', message });\n },\n [broadcast],\n );\n\n const createBuildLog = useCallback(\n (message: string): void => {\n setBuildLogHistory((h) =>\n [...h, `${new Date().toLocaleTimeString()}: ${message}`].slice(-LOG_LIMIT),\n );\n broadcast({ type: 'log', message });\n },\n [broadcast],\n );\n\n const triggerBuild = useCallback(() => {\n if (buildingRef.current) {\n return;\n }\n\n buildingRef.current = true;\n\n setBuildStatus('building');\n setBuildError(null);\n createBuildLog('Build started');\n\n void runBuild(pluginDir, outputDir)\n .then(() => {\n createBuildLog('Build finished');\n setBuildStatus('done');\n setLastBuilt(new Date());\n\n createBuildLog('Notifying viewers to reload');\n for (const client of wsClientsRef.current) {\n if (client.readyState === 1) {\n client.send(JSON.stringify({ type: 'rebuild' }));\n }\n }\n })\n .catch((err: unknown) => {\n const message = err instanceof Error ? err.message : String(err);\n setBuildError(message);\n setBuildStatus('error');\n })\n .finally(() => {\n buildingRef.current = false;\n });\n }, [pluginDir, outputDir, createBuildLog]);\n\n useEffect(() => {\n triggerBuild();\n\n const watcher: FSWatcher = watch(pluginDir, { recursive: true }, (_, filename) => {\n if (!filename) {\n return;\n }\n\n const normalized = filename.replace(/\\\\/g, '/');\n\n if (SKIP_WATCH_PREFIXES.some((prefix) => normalized.startsWith(prefix))) {\n return;\n }\n\n createBuildLog(`File change detected: ${filename}`);\n\n if (debounceTimerRef.current !== null) {\n clearTimeout(debounceTimerRef.current);\n }\n debounceTimerRef.current = setTimeout(triggerBuild, 150);\n });\n\n return () => {\n watcher.close();\n\n if (debounceTimerRef.current !== null) {\n clearTimeout(debounceTimerRef.current);\n }\n };\n }, [pluginDir, triggerBuild, createBuildLog]);\n\n useEffect(() => {\n const viewerPath = getViewerPath();\n\n void fileExists(join(viewerPath, 'index.html')).then((viewerBuilt) => {\n if (!viewerBuilt) {\n setErrorMessage(\"Viewer not built. Run 'pnpm build:viewer' first.\");\n setStatus('error');\n return;\n }\n\n const handler = createRequestHandler(\n viewerPath,\n createServerLog,\n createProxyLog,\n credentialsRef,\n );\n const server: Server = createServer(handler);\n const wss = new WebSocketServer({ server });\n\n wss.on('connection', (ws: WebSocket) => {\n for (const json of pendingMessagesRef.current) {\n if (ws.readyState === 1) {\n ws.send(json);\n }\n }\n pendingMessagesRef.current = [];\n\n createServerLog('Viewer connected for live reload');\n wsClientsRef.current.add(ws);\n setWsClientCount(wsClientsRef.current.size);\n\n ws.on('close', () => {\n wsClientsRef.current.delete(ws);\n setWsClientCount(wsClientsRef.current.size);\n });\n });\n\n server.listen(port, () => {\n setStatus('running');\n createServerLog(`Server started on port ${String(port)}`);\n });\n\n server.on('error', (err: Error) => {\n setErrorMessage(err.message);\n setStatus('error');\n });\n\n return () => {\n wss.close();\n server.close();\n };\n });\n }, [port, createServerLog, createProxyLog]);\n\n if (credMode === 'editing') {\n return (\n <CredentialSetupUI\n onComplete={handleCredentialsDone}\n onCancel={() => {\n setCredMode('main');\n }}\n />\n );\n }\n\n const elapsedSeconds =\n lastBuilt !== null ? Math.round((Date.now() - lastBuilt.getTime()) / 1000) : null;\n const elapsedLabel =\n elapsedSeconds === null\n ? ''\n : elapsedSeconds < 5\n ? ' (just now)'\n : ` (${String(elapsedSeconds)}s ago)`;\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2}>\n {/* Logo + header */}\n <Logo />\n <Box flexDirection=\"column\" marginTop={1} marginBottom={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n\n {/* Build panel */}\n <Box flexDirection=\"column\">\n <Box gap={2} marginBottom={0}>\n <Text bold dimColor>\n Build\n </Text>\n {buildStatus === 'pending' && <Text dimColor>waiting…</Text>}\n {buildStatus === 'building' && (\n <Box gap={1}>\n <Spinner />\n <Text>Building plugin package…</Text>\n </Box>\n )}\n {buildStatus === 'done' && (\n <Box gap={1}>\n <Text color=\"green\">✓ Built</Text>\n {lastBuilt !== null && (\n <Text dimColor>\n at {lastBuilt.toLocaleTimeString()}\n {elapsedLabel}\n </Text>\n )}\n </Box>\n )}\n {buildStatus === 'error' && <Text color=\"red\">✗ {buildError ?? 'unknown error'}</Text>}\n </Box>\n <Box\n flexDirection=\"column\"\n height={LOG_DISPLAY}\n borderStyle=\"single\"\n borderColor=\"gray\"\n overflow=\"hidden\"\n >\n {buildLogHistory.slice(-LOG_DISPLAY).map((log, index) => (\n <Text key={index} dimColor wrap=\"truncate\">\n {log}\n </Text>\n ))}\n </Box>\n </Box>\n\n {/* Server panel */}\n <Box marginTop={1} flexDirection=\"column\">\n <Box gap={2}>\n <Text bold dimColor>\n Server\n </Text>\n {status === 'starting' && <Text dimColor>starting…</Text>}\n {status === 'running' && (\n <Box gap={2}>\n <Text color=\"green\">✓ Running</Text>\n <Text dimColor>\n ● {wsClientCount} viewer{wsClientCount !== 1 ? 's' : ''}\n </Text>\n </Box>\n )}\n {status === 'error' && <Text color=\"red\">✗ {errorMessage ?? 'Server error'}</Text>}\n </Box>\n <Box\n flexDirection=\"column\"\n height={LOG_DISPLAY}\n borderStyle=\"single\"\n borderColor=\"gray\"\n overflow=\"hidden\"\n >\n {serverLogHistory.slice(-LOG_DISPLAY).map((log, index) => (\n <Text key={index} dimColor wrap=\"truncate\">\n {log}\n </Text>\n ))}\n </Box>\n </Box>\n\n {/* Proxy panel */}\n <Box marginTop={1} flexDirection=\"column\">\n <Box gap={2}>\n <Text bold dimColor>\n Proxy\n </Text>\n <Text color=\"green\">✓ Active</Text>\n </Box>\n <Box\n flexDirection=\"column\"\n height={LOG_DISPLAY}\n borderStyle=\"single\"\n borderColor=\"gray\"\n overflow=\"hidden\"\n >\n {proxyLogHistory.length === 0 ? (\n <Text dimColor> No requests yet.</Text>\n ) : (\n proxyLogHistory.slice(-LOG_DISPLAY).map((log, i) => (\n <Text key={i} dimColor wrap=\"truncate\">\n {log}\n </Text>\n ))\n )}\n </Box>\n </Box>\n\n {/* Footer */}\n <Box marginTop={1} gap={1}>\n <Text dimColor>[</Text>\n <Text>q</Text>\n <Text dimColor>] quit [</Text>\n <Text>c</Text>\n <Text dimColor>] credentials [</Text>\n <Text>d</Text>\n <Text dimColor>] devtools</Text>\n </Box>\n </Box>\n );\n};\n","import { mkdir, readFile, writeFile } from 'node:fs/promises';\nimport { join } from 'node:path';\n\nexport interface AppBuilderConfig {\n credentialMode?: 'global' | 'local';\n}\n\nexport async function loadConfig(outputDir: string): Promise<AppBuilderConfig> {\n try {\n const content = await readFile(join(outputDir, 'config.json'), 'utf-8');\n return JSON.parse(content) as AppBuilderConfig;\n } catch {\n return {};\n }\n}\n\nexport async function saveConfig(outputDir: string, config: AppBuilderConfig): Promise<void> {\n await mkdir(outputDir, { recursive: true });\n await writeFile(join(outputDir, 'config.json'), JSON.stringify(config, null, 2), 'utf-8');\n}\n","import type { FC } from 'react';\nimport { useCallback, useEffect, useState } from 'react';\nimport { Box, Text, useInput } from 'ink';\nimport {\n ENVIRONMENTS,\n GLOBAL_CREDENTIALS_PATH,\n loadGlobalCredentials,\n saveGlobalCredentials,\n} from '../lib/credentials.js';\nimport type { Credentials } from '../lib/credentials.js';\nimport { Logo } from './Logo.js';\n\ntype Mode = 'global' | 'local';\n\ntype Phase =\n | { type: 'mode-select'; cursor: 0 | 1 }\n | { type: 'loading' }\n | { type: 'creds-entry'; field: number; values: Partial<Credentials>; envCursor: number }\n | { type: 'saving' }\n | { type: 'done' };\n\nconst FIELDS = ['apiKey', 'userId', 'businessId'] as const;\nconst FIELD_LABELS: Record<string, string> = {\n apiKey: 'API Key',\n userId: 'User ID',\n businessId: 'Business ID',\n};\n\nexport type CredentialMode = 'global' | 'local';\n\nexport interface CredentialSetupResult {\n mode: CredentialMode;\n credentials: Credentials | null;\n}\n\ninterface CredentialSetupUIProps {\n initialMode?: CredentialMode;\n onComplete: (result: CredentialSetupResult) => void;\n onCancel?: () => void;\n}\n\nconst Hint: FC<{ text: string }> = ({ text }) => <Text dimColor>{text}</Text>;\n\nexport const CredentialSetupUI: FC<CredentialSetupUIProps> = ({\n initialMode,\n onComplete,\n onCancel,\n}) => {\n const [phase, setPhase] = useState<Phase>(\n initialMode === 'global' ? { type: 'loading' } : { type: 'mode-select', cursor: 0 },\n );\n const [inputBuffer, setInputBuffer] = useState('');\n const [error, setError] = useState<string | null>(null);\n\n const handleModeChosen = useCallback(\n async (mode: Mode) => {\n if (mode === 'local') {\n onComplete({ mode: 'local', credentials: null });\n return;\n }\n\n setPhase({ type: 'loading' });\n\n const existing = await loadGlobalCredentials();\n const envCursor = existing ? Math.max(0, ENVIRONMENTS.indexOf(existing.environment)) : 0;\n setPhase({\n type: 'creds-entry',\n field: 0,\n values: existing ?? {},\n envCursor,\n });\n },\n [onComplete],\n );\n\n const handleSave = useCallback(\n async (values: Partial<Credentials>, envCursor: number) => {\n const environment = ENVIRONMENTS[envCursor] ?? 'go';\n const credentials: Credentials = {\n apiKey: values.apiKey ?? '',\n userId: values.userId ?? '',\n businessId: values.businessId ?? '',\n environment,\n };\n setPhase({ type: 'saving' });\n try {\n await saveGlobalCredentials(credentials);\n onComplete({ mode: 'global', credentials });\n } catch (err) {\n setError(err instanceof Error ? err.message : String(err));\n setPhase({ type: 'creds-entry', field: 0, values, envCursor });\n }\n },\n [onComplete],\n );\n\n useInput((input, key) => {\n if (key.ctrl && input === 'c') {\n process.exit(0);\n }\n\n if (key.escape) {\n onCancel?.();\n return;\n }\n\n if (phase.type === 'mode-select') {\n if (key.upArrow) {\n setPhase({ type: 'mode-select', cursor: 0 });\n } else if (key.downArrow) {\n setPhase({ type: 'mode-select', cursor: 1 });\n } else if (key.return) {\n const mode: Mode = phase.cursor === 0 ? 'global' : 'local';\n void handleModeChosen(mode);\n }\n return;\n }\n\n if (phase.type === 'creds-entry') {\n const { field, values, envCursor } = phase;\n const isEnvField = field === FIELDS.length;\n\n if (isEnvField) {\n if (key.leftArrow) {\n setPhase({\n ...phase,\n envCursor: (envCursor - 1 + ENVIRONMENTS.length) % ENVIRONMENTS.length,\n });\n } else if (key.rightArrow) {\n setPhase({ ...phase, envCursor: (envCursor + 1) % ENVIRONMENTS.length });\n } else if (key.upArrow) {\n setPhase({ ...phase, field: field - 1 });\n setInputBuffer(values[FIELDS[field - 1] as keyof typeof values] ?? '');\n } else if (key.return) {\n void handleSave(values, envCursor);\n }\n return;\n }\n\n const fieldName = FIELDS[field];\n if (!fieldName) {\n return;\n }\n\n if (key.backspace || key.delete) {\n setInputBuffer((prev) => prev.slice(0, -1));\n } else if (key.upArrow && field > 0) {\n const updatedValues = { ...values, [fieldName]: inputBuffer };\n const prevField = field - 1;\n setPhase({ ...phase, field: prevField, values: updatedValues });\n setInputBuffer(updatedValues[FIELDS[prevField] as keyof typeof updatedValues] ?? '');\n } else if (key.return || key.tab || key.downArrow) {\n // Advance to next field\n const updatedValues = { ...values, [fieldName]: inputBuffer };\n const nextField = field + 1;\n setPhase({ ...phase, field: nextField, values: updatedValues });\n if (nextField < FIELDS.length) {\n setInputBuffer(updatedValues[FIELDS[nextField] as keyof typeof updatedValues] ?? '');\n } else {\n setInputBuffer('');\n }\n } else if (input && !key.ctrl && !key.meta) {\n setInputBuffer((prev) => prev + input);\n }\n }\n });\n\n // Trigger global credential load when initialMode skips mode selection\n useEffect(() => {\n if (initialMode === 'global') {\n void handleModeChosen('global');\n }\n }, []); // eslint-disable-line react-hooks/exhaustive-deps\n\n // Reset input buffer when switching to creds-entry\n useEffect(() => {\n if (phase.type === 'creds-entry' && phase.field === 0) {\n setInputBuffer(phase.values.apiKey ?? '');\n }\n }, [phase.type]); // eslint-disable-line react-hooks/exhaustive-deps\n\n if (phase.type === 'loading' || phase.type === 'saving') {\n return (\n <Box paddingY={1} paddingX={2}>\n <Text dimColor>\n {phase.type === 'loading' ? 'Loading credentials…' : 'Saving credentials…'}\n </Text>\n </Box>\n );\n }\n\n if (phase.type === 'done') {\n return null;\n }\n\n if (phase.type === 'mode-select') {\n const options: { label: string; desc: string; mode: Mode }[] = [\n { label: 'Global', desc: `~/.kizenappbuilder/credentials.json`, mode: 'global' },\n { label: 'Local', desc: 'Enter credentials in the browser dev tools', mode: 'local' },\n ];\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2} gap={1}>\n <Logo />\n <Box flexDirection=\"column\" marginTop={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n <Box flexDirection=\"column\" gap={0}>\n <Text bold>Credential mode</Text>\n {options.map((opt, i) => {\n const selected = phase.cursor === i;\n return (\n <Box key={opt.mode} gap={2}>\n <Text {...(selected && { color: 'cyan' as const })}>{selected ? '❯' : ' '}</Text>\n <Text bold={selected} {...(selected && { color: 'cyan' as const })}>\n {opt.label}\n </Text>\n <Text dimColor>{opt.desc}</Text>\n </Box>\n );\n })}\n </Box>\n <Hint text=\"↑↓ to move · Enter to select · Ctrl+C to quit\" />\n </Box>\n );\n }\n\n // creds-entry phase\n const { field: activeField, values, envCursor } = phase;\n const isEnvField = activeField === FIELDS.length;\n\n return (\n <Box flexDirection=\"column\" paddingY={1} paddingX={2} gap={1}>\n <Logo />\n <Box flexDirection=\"column\" marginTop={1}>\n <Text bold color=\"cyan\">\n Kizen App Builder\n </Text>\n <Text dimColor>{'─'.repeat(24)}</Text>\n </Box>\n <Box flexDirection=\"column\" gap={0}>\n <Text bold>Global credentials</Text>\n <Text dimColor>Saved to: {GLOBAL_CREDENTIALS_PATH}</Text>\n </Box>\n {error && <Text color=\"red\">Error: {error}</Text>}\n <Box flexDirection=\"column\" gap={0}>\n {FIELDS.map((name, i) => {\n const isActive = activeField === i;\n const displayValue = isActive ? inputBuffer : (values[name] ?? '');\n return (\n <Box key={name} gap={2}>\n <Box width={12}>\n <Text bold={isActive} {...(isActive && { color: 'cyan' as const })}>\n {FIELD_LABELS[name]}\n </Text>\n </Box>\n <Text {...(isActive && { color: 'cyan' as const })}>{'>'}</Text>\n <Text>{displayValue}</Text>\n {isActive && <Text color=\"cyan\">{'█'}</Text>}\n </Box>\n );\n })}\n <Box gap={2}>\n <Box width={12}>\n <Text bold={isEnvField} {...(isEnvField && { color: 'cyan' as const })}>\n Environment\n </Text>\n </Box>\n <Text {...(isEnvField && { color: 'cyan' as const })}>{'>'}</Text>\n <Box gap={1}>\n {ENVIRONMENTS.map((env, i) => {\n const isSelected = i === envCursor;\n if (isEnvField) {\n return (\n <Text key={env} bold={isSelected} {...(isSelected && { color: 'cyan' as const })}>\n {isSelected ? `[${env}]` : env}\n </Text>\n );\n }\n return (\n <Text key={env} dimColor={!isSelected} bold={isSelected}>\n {env}\n </Text>\n );\n })}\n </Box>\n </Box>\n </Box>\n {isEnvField ? (\n <Hint text=\"←→ to select · ↑↓ to move · Enter to save · Esc to cancel\" />\n ) : (\n <Hint text=\"↑↓ to move · Backspace to delete · Esc to cancel\" />\n )}\n </Box>\n );\n};\n","import { mkdir, readFile, writeFile } from 'node:fs/promises';\nimport { homedir } from 'node:os';\nimport { dirname, join } from 'node:path';\n\nexport const ENVIRONMENTS = ['go', 'fmo', 'staging', 'integration', 'test1'] as const;\nexport type Environment = (typeof ENVIRONMENTS)[number];\n\nexport interface Credentials {\n apiKey: string;\n userId: string;\n businessId: string;\n environment: Environment;\n}\n\nconst GLOBAL_CREDENTIALS_DIR = join(homedir(), '.kizenappbuilder');\nexport const GLOBAL_CREDENTIALS_PATH = join(GLOBAL_CREDENTIALS_DIR, 'credentials.json');\n\nfunction isValidEnvironment(value: unknown): value is Environment {\n return ENVIRONMENTS.includes(value as Environment);\n}\n\nfunction parseCredentials(raw: unknown): Credentials {\n if (typeof raw !== 'object' || raw === null) {\n throw new Error('Credentials must be a JSON object');\n }\n const obj = raw as Record<string, unknown>;\n const env = obj.environment;\n return {\n apiKey: typeof obj.apiKey === 'string' ? obj.apiKey : '',\n userId: typeof obj.userId === 'string' ? obj.userId : '',\n businessId: typeof obj.businessId === 'string' ? obj.businessId : '',\n environment: isValidEnvironment(env) ? env : 'go',\n };\n}\n\nexport async function loadCredentialsFromFile(filePath: string): Promise<Credentials> {\n const content = await readFile(filePath, 'utf-8');\n return parseCredentials(JSON.parse(content) as unknown);\n}\n\nexport async function loadGlobalCredentials(): Promise<Credentials | null> {\n try {\n return await loadCredentialsFromFile(GLOBAL_CREDENTIALS_PATH);\n } catch {\n return null;\n }\n}\n\nexport async function saveGlobalCredentials(credentials: Credentials): Promise<void> {\n await mkdir(dirname(GLOBAL_CREDENTIALS_PATH), { recursive: true });\n await writeFile(GLOBAL_CREDENTIALS_PATH, JSON.stringify(credentials, null, 2), 'utf-8');\n}\n"],"mappings":"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","names":["Box","Text","join","join","jsx","Text","Box","program","createElement","render","useCallback","useEffect","useState","Box","Text","useInput","readFile","dirname","join","mkdir","readFile","writeFile","join","useEffect","useState","Box","Text","mkdir","readFile","writeFile","join","jsx","jsxs","Text","useState","envCursor","values","isEnvField","useEffect","Box","jsx","jsxs","SPINNER_FRAMES","Spinner","useState","useEffect","Text","join","dirname","readFile","useInput","useCallback","Box","render","createElement","program"]}