gstack/design/src/cli.ts

422 lines
14 KiB
TypeScript

/**
* gstack design CLI — stateless CLI for AI-powered design generation.
*
* Unlike the browse binary (persistent Chromium daemon), the design binary
* is stateless: each invocation makes API calls and writes files. Session
* state for multi-turn iteration is a JSON file in /tmp.
*
* Flow:
* 1. Parse command + flags from argv
* 2. Resolve auth (~/. gstack/openai.json → OPENAI_API_KEY → guided setup)
* 3. Execute command (API call → write PNG/HTML)
* 4. Print result JSON to stdout
*/
import { COMMANDS } from "./commands";
import { generate } from "./generate";
import { checkCommand } from "./check";
import { compare } from "./compare";
import { variants } from "./variants";
import { iterate } from "./iterate";
import { resolveApiKey, saveApiKey } from "./auth";
import { extractDesignLanguage, updateDesignMd } from "./memory";
import { diffMockups, verifyAgainstMockup } from "./diff";
import { evolve } from "./evolve";
import { generateDesignToCodePrompt } from "./design-to-code";
import { serve } from "./serve";
import { gallery } from "./gallery";
import {
daemonStatus as daemonStatusClient,
ensureDaemon,
publishBoard,
shutdownDaemon,
} from "./daemon-client";
import { spawn as nodeSpawn } from "child_process";
function parseArgs(argv: string[]): {
command: string;
flags: Record<string, string | boolean>;
positionals: string[];
} {
const args = argv.slice(2); // skip bun/node and script path
if (args.length === 0) {
printUsage();
process.exit(0);
}
const command = args[0];
const flags: Record<string, string | boolean> = {};
const positionals: string[] = [];
for (let i = 1; i < args.length; i++) {
const arg = args[i];
if (arg.startsWith("--")) {
const key = arg.slice(2);
const next = args[i + 1];
if (next && !next.startsWith("--")) {
flags[key] = next;
i++;
} else {
flags[key] = true;
}
} else {
positionals.push(arg);
}
}
return { command, flags, positionals };
}
function printUsage(): void {
console.log("gstack design — AI-powered UI mockup generation\n");
console.log("Commands:");
for (const [name, info] of COMMANDS) {
console.log(` ${name.padEnd(12)} ${info.description}`);
console.log(` ${"".padEnd(12)} ${info.usage}`);
}
console.log("\nAuth: ~/.gstack/openai.json, then OPENAI_API_KEY env var");
console.log("If OPENAI_API_KEY matches a current-directory .env file, the source is reported before billing.");
console.log("Setup: $D setup");
}
async function runSetup(): Promise<void> {
const existing = resolveApiKey();
if (existing) {
console.log("Existing API key found. Running smoke test...");
} else {
console.log("No API key found. Please enter your OpenAI API key.");
console.log("Get one at: https://platform.openai.com/api-keys");
console.log("(Needs image generation permissions)\n");
// Read from stdin
process.stdout.write("API key: ");
const reader = Bun.stdin.stream().getReader();
const { value } = await reader.read();
reader.releaseLock();
const key = new TextDecoder().decode(value).trim();
if (!key || !key.startsWith("sk-")) {
console.error("Invalid key. Must start with 'sk-'.");
process.exit(1);
}
saveApiKey(key);
console.log("Key saved to ~/.gstack/openai.json (0600 permissions).");
}
// Smoke test
console.log("\nRunning smoke test (generating a simple image)...");
try {
await generate({
brief: "A simple blue square centered on a white background. Minimal, geometric, clean.",
output: "/tmp/gstack-design-smoke-test.png",
size: "1024x1024",
quality: "low",
});
console.log("\nSmoke test PASSED. Design generation is working.");
} catch (err: any) {
console.error(`\nSmoke test FAILED: ${err.message}`);
console.error("Check your API key and organization verification status.");
process.exit(1);
}
}
async function main(): Promise<void> {
const { command, flags, positionals } = parseArgs(process.argv);
if (!COMMANDS.has(command)) {
console.error(`Unknown command: ${command}`);
printUsage();
process.exit(1);
}
switch (command) {
case "generate":
await generate({
brief: flags.brief as string,
briefFile: flags["brief-file"] as string,
output: (flags.output as string) || "/tmp/gstack-mockup.png",
check: !!flags.check,
retry: flags.retry ? parseInt(flags.retry as string) : 0,
size: flags.size as string,
quality: flags.quality as string,
});
break;
case "check":
await checkCommand(flags.image as string, flags.brief as string);
break;
case "compare": {
// Parse --images as glob or multiple files
const imagesArg = flags.images as string;
const images = await resolveImagePaths(imagesArg);
const outputPath = (flags.output as string) || "/tmp/gstack-design-board.html";
compare({ images, output: outputPath });
// If --serve flag is set, publish the board.
// Default: ensure the persistent daemon is up, POST the board, open
// the browser, exit. The daemon survives the CLI and hosts every
// board the user has published this day at stable URLs.
// --no-daemon: legacy single-process server in serve.ts (kept for
// tests / Windows / explicit debugging).
if (flags.serve) {
if (flags["no-daemon"]) {
await serve({
html: outputPath,
timeout: flags.timeout ? parseInt(flags.timeout as string) : 600,
});
} else {
await publishToDaemon({
html: outputPath,
title: flags.title as string | undefined,
});
}
}
break;
}
case "prompt": {
const promptImage = flags.image as string;
if (!promptImage) {
console.error("--image is required");
process.exit(1);
}
console.error(`Generating implementation prompt from ${promptImage}...`);
const proc2 = Bun.spawn(["git", "rev-parse", "--show-toplevel"]);
const root = (await new Response(proc2.stdout).text()).trim();
const d2c = await generateDesignToCodePrompt(promptImage, root || undefined);
console.log(JSON.stringify(d2c, null, 2));
break;
}
case "setup":
await runSetup();
break;
case "variants":
await variants({
brief: flags.brief as string,
briefFile: flags["brief-file"] as string,
count: flags.count ? parseInt(flags.count as string) : 3,
outputDir: (flags["output-dir"] as string) || "/tmp/gstack-variants/",
size: flags.size as string,
quality: flags.quality as string,
viewports: flags.viewports as string,
});
break;
case "iterate":
await iterate({
session: flags.session as string,
feedback: flags.feedback as string,
output: (flags.output as string) || "/tmp/gstack-iterate.png",
});
break;
case "extract": {
const imagePath = flags.image as string;
if (!imagePath) {
console.error("--image is required");
process.exit(1);
}
console.error(`Extracting design language from ${imagePath}...`);
const extracted = await extractDesignLanguage(imagePath);
const proc = Bun.spawn(["git", "rev-parse", "--show-toplevel"]);
const repoRoot = (await new Response(proc.stdout).text()).trim();
if (repoRoot) {
updateDesignMd(repoRoot, extracted, imagePath);
}
console.log(JSON.stringify(extracted, null, 2));
break;
}
case "diff": {
const before = flags.before as string;
const after = flags.after as string;
if (!before || !after) {
console.error("--before and --after are required");
process.exit(1);
}
console.error(`Comparing ${before} vs ${after}...`);
const diffResult = await diffMockups(before, after);
console.log(JSON.stringify(diffResult, null, 2));
break;
}
case "verify": {
const mockup = flags.mockup as string;
const screenshot = flags.screenshot as string;
if (!mockup || !screenshot) {
console.error("--mockup and --screenshot are required");
process.exit(1);
}
console.error(`Verifying implementation against approved mockup...`);
const verifyResult = await verifyAgainstMockup(mockup, screenshot);
console.error(`Match: ${verifyResult.matchScore}/100 — ${verifyResult.pass ? "PASS" : "FAIL"}`);
console.log(JSON.stringify(verifyResult, null, 2));
break;
}
case "evolve":
await evolve({
screenshot: flags.screenshot as string,
brief: flags.brief as string,
output: (flags.output as string) || "/tmp/gstack-evolved.png",
});
break;
case "gallery":
gallery({
designsDir: flags["designs-dir"] as string,
output: (flags.output as string) || "/tmp/gstack-design-gallery.html",
});
break;
case "serve":
if (flags["no-daemon"]) {
await serve({
html: flags.html as string,
timeout: flags.timeout ? parseInt(flags.timeout as string) : 600,
});
} else {
await publishToDaemon({
html: flags.html as string,
title: flags.title as string | undefined,
});
}
break;
case "daemon": {
// Sub-commands: `$D daemon status` and `$D daemon stop [--force]`.
const sub = positionals[0] || "status";
if (sub === "status") {
const s = await daemonStatusClient();
if (!s.running) {
console.log(JSON.stringify({ running: false }, null, 2));
process.exit(0);
}
console.log(JSON.stringify(s, null, 2));
break;
}
if (sub === "stop") {
const r = await shutdownDaemon({ force: !!flags.force });
if (r.stopped) {
console.log(JSON.stringify({ stopped: true, reason: r.reason }, null, 2));
process.exit(0);
}
console.error(
`Refused to stop daemon: ${r.reason} (activeBoards=${r.activeBoards ?? 0})`,
);
console.error(
`Submit/close active boards first, or pass --force to drop in-memory history.`,
);
process.exit(1);
}
console.error(`Unknown daemon sub-command: ${sub}. Use 'status' or 'stop'.`);
process.exit(2);
}
}
}
/**
* Default `$D compare --serve` path: ensure the persistent daemon is up,
* publish the board, open the browser to its URL, then exit. The daemon
* survives.
*
* Stderr lines (in order):
* - "DAEMON_STARTED port=N version=V" (or "DAEMON_ATTACHED port=N ..."
* if a daemon was already running)
* - "BOARD_PUBLISHED: http://127.0.0.1:N/boards/<id>/"
* - "BOARD_URL: <same url>" (alias for grep-friendliness)
* - "SERVE_STARTED: port=N html=<path>" (legacy back-compat alias for
* any external script that scraped the pre-daemon output — note the
* daemon hosts boards under /boards/<id>/, not /, so scripts that
* ALSO POSTed /api/reload at the parsed port need to switch to
* BOARD_URL + ./api/reload to work end-to-end. Emitting the legacy
* line keeps port-only consumers from breaking outright.)
*/
async function publishToDaemon(opts: { html: string; title?: string }): Promise<void> {
if (!opts.html) {
console.error("--html is required (compare --serve provides --output as the html)");
process.exit(1);
}
const ensured = await ensureDaemon({});
console.error(
`${ensured.spawned ? "DAEMON_STARTED" : "DAEMON_ATTACHED"} port=${ensured.port} version=${ensured.version}`,
);
const result = await publishBoard({
port: ensured.port,
html: opts.html,
title: opts.title,
});
console.error(`BOARD_PUBLISHED: ${result.url}`);
console.error(`BOARD_URL: ${result.url}`);
// Legacy alias so anything still grepping `SERVE_STARTED: port=` gets the
// port. The full back-compat story requires the caller to ALSO learn the
// per-board path; see publishToDaemon docstring above.
console.error(`SERVE_STARTED: port=${ensured.port} html=${opts.html}`);
console.log(JSON.stringify({ id: result.id, url: result.url, sourceDir: result.sourceDir }, null, 2));
openBrowser(result.url);
// Short-lived publisher process exits; daemon keeps serving.
}
/** Open a URL in the default browser. Stays cross-platform with serve.ts. */
function openBrowser(url: string): void {
const platform = process.platform;
let cmd: string;
if (platform === "darwin") cmd = "open";
else if (platform === "linux") cmd = "xdg-open";
else {
console.error(`Open this URL in your browser: ${url}`);
return;
}
try {
const child = nodeSpawn(cmd, [url], { stdio: "ignore", detached: true });
child.unref();
} catch {
console.error(`Open this URL in your browser: ${url}`);
}
}
/**
* Resolve image paths from a glob pattern or comma-separated list.
*/
async function resolveImagePaths(input: string): Promise<string[]> {
if (!input) {
console.error("--images is required. Provide glob pattern or comma-separated paths.");
process.exit(1);
}
// Check if it's a glob pattern
if (input.includes("*")) {
const glob = new Bun.Glob(input);
const paths: string[] = [];
for await (const match of glob.scan({ absolute: true })) {
if (match.endsWith(".png") || match.endsWith(".jpg") || match.endsWith(".jpeg")) {
paths.push(match);
}
}
return paths.sort();
}
// Comma-separated or single path
return input.split(",").map(p => p.trim());
}
// Self-execution shortcut: when invoked with --daemon-mode, this same
// binary runs as the persistent design daemon instead of the CLI. Keeps
// the production install to a single executable; daemon-client.ts spawns
// `<this binary> --daemon-mode` (or `bun run cli.ts --daemon-mode` in dev)
// rather than relying on a separate daemon.ts file at a known path.
if (process.argv.includes("--daemon-mode")) {
const { start } = await import("./daemon");
start();
// start() binds Bun.serve and registers signal handlers; this branch
// never falls through to main(). Process stays alive on the bound port.
} else {
main().catch((err) => {
console.error(err.message || err);
process.exit(1);
});
}