@openbuilder/cli 0.50.21 → 0.50.26
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +164 -943
- package/dist/chunks/{auto-update-Cr7SnKwX.js → auto-update-Dq2PFvjt.js} +12 -46
- package/dist/chunks/auto-update-Dq2PFvjt.js.map +1 -0
- package/dist/chunks/{cli-auth-rqwB_Zwt.js → cli-auth-ChCnxlFl.js} +2 -2
- package/dist/chunks/cli-auth-ChCnxlFl.js.map +1 -0
- package/dist/chunks/{init-DDtrxZ0M.js → init-BsQ3dhwf.js} +3 -2
- package/dist/chunks/{init-DDtrxZ0M.js.map → init-BsQ3dhwf.js.map} +1 -1
- package/dist/chunks/{init-tui-tZtLISA4.js → init-tui-Dvk6Ndvl.js} +5 -4
- package/dist/chunks/{init-tui-tZtLISA4.js.map → init-tui-Dvk6Ndvl.js.map} +1 -1
- package/dist/chunks/{login-DFgTXEGt.js → login-BhtodVsj.js} +2 -2
- package/dist/chunks/{login-DFgTXEGt.js.map → login-BhtodVsj.js.map} +1 -1
- package/dist/chunks/{logout-DwddishT.js → logout-CDDASeuQ.js} +2 -2
- package/dist/chunks/{logout-DwddishT.js.map → logout-CDDASeuQ.js.map} +1 -1
- package/dist/chunks/{main-tui-BrfgfCDw.js → main-tui-C-BqvY5r.js} +6 -5
- package/dist/chunks/{main-tui-BrfgfCDw.js.map → main-tui-C-BqvY5r.js.map} +1 -1
- package/dist/chunks/{run-LjsGm3tI.js → run-BbtmU1EJ.js} +26 -3
- package/dist/chunks/run-BbtmU1EJ.js.map +1 -0
- package/dist/cli/index.js +7 -7
- package/dist/index.js +47 -1
- package/dist/index.js.map +1 -1
- package/package.json +8 -6
- package/dist/chunks/auto-update-Cr7SnKwX.js.map +0 -1
- package/dist/chunks/cli-auth-rqwB_Zwt.js.map +0 -1
- package/dist/chunks/run-LjsGm3tI.js.map +0 -1
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@openbuilder/cli",
|
|
3
|
-
"version": "0.50.
|
|
4
|
-
"description": "OpenBuilder CLI - AI-powered
|
|
3
|
+
"version": "0.50.26",
|
|
4
|
+
"description": "OpenBuilder CLI - Build AI-powered apps with Claude. Just run: npx @openbuilder/cli runner",
|
|
5
5
|
"private": false,
|
|
6
6
|
"type": "module",
|
|
7
7
|
"main": "dist/index.js",
|
|
@@ -23,11 +23,13 @@
|
|
|
23
23
|
},
|
|
24
24
|
"keywords": [
|
|
25
25
|
"openbuilder",
|
|
26
|
-
"runner",
|
|
27
|
-
"cli",
|
|
28
26
|
"ai",
|
|
29
|
-
"
|
|
30
|
-
"
|
|
27
|
+
"claude",
|
|
28
|
+
"code-generation",
|
|
29
|
+
"app-builder",
|
|
30
|
+
"cli",
|
|
31
|
+
"runner",
|
|
32
|
+
"vibe-coding"
|
|
31
33
|
],
|
|
32
34
|
"author": "Cody De Arkland",
|
|
33
35
|
"repository": {
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"auto-update-Cr7SnKwX.js","sources":["../../src/cli/utils/auto-update.ts"],"sourcesContent":["/**\n * Auto-update utility for OpenBuilder CLI\n * \n * Checks GitHub Releases for newer versions and automatically\n * updates both:\n * 1. The CLI itself (globally installed npm package)\n * 2. The app/monorepo (local installation that runs the web app)\n */\n\nimport { execSync, spawnSync } from 'node:child_process';\nimport { existsSync, readFileSync, writeFileSync, mkdirSync, unlinkSync } from 'node:fs';\nimport { join } from 'node:path';\nimport { homedir } from 'node:os';\nimport pc from 'picocolors';\nimport { configManager } from './config-manager.js';\n\n// GitHub API endpoint for releases\nconst GITHUB_RELEASES_URL = 'https://api.github.com/repos/codyde/openbuilder/releases/latest';\n\n// Install command for CLI\nconst INSTALL_COMMAND = 'curl -fsSL https://openbuilder.sh/install | bash';\n\n// Cache settings\nconst CACHE_DIR = join(homedir(), '.config', 'openbuilder');\nconst CACHE_FILE = join(CACHE_DIR, 'update-cache.json');\nconst CACHE_TTL_MS = 60 * 60 * 1000; // 1 hour\n\ninterface UpdateCache {\n lastCheck: number;\n latestVersion: string;\n // Track if we need to upgrade the app after CLI update\n pendingAppUpgrade?: boolean;\n}\n\n/**\n * Read the update cache file\n */\nfunction readUpdateCache(): UpdateCache | null {\n try {\n if (existsSync(CACHE_FILE)) {\n const content = readFileSync(CACHE_FILE, 'utf-8');\n return JSON.parse(content);\n }\n } catch {\n // Cache read failed, that's fine\n }\n return null;\n}\n\n/**\n * Write to the update cache file\n */\nfunction saveUpdateCache(cache: UpdateCache): void {\n try {\n if (!existsSync(CACHE_DIR)) {\n mkdirSync(CACHE_DIR, { recursive: true });\n }\n writeFileSync(CACHE_FILE, JSON.stringify(cache, null, 2));\n } catch {\n // Cache write failed, that's fine\n }\n}\n\n/**\n * Fetch the latest release version from GitHub\n * Returns version string (e.g., \"0.28.0\") or null on failure\n */\nasync function fetchLatestVersion(): Promise<string | null> {\n try {\n const controller = new AbortController();\n const timeout = setTimeout(() => controller.abort(), 5000); // 5s timeout\n\n const response = await fetch(GITHUB_RELEASES_URL, {\n headers: {\n 'Accept': 'application/vnd.github.v3+json',\n 'User-Agent': 'OpenBuilder-CLI-AutoUpdate',\n },\n signal: controller.signal,\n });\n\n clearTimeout(timeout);\n\n if (!response.ok) {\n return null;\n }\n\n const data = await response.json() as { tag_name: string };\n // Remove 'v' prefix if present (e.g., \"v0.28.0\" -> \"0.28.0\")\n return data.tag_name.replace(/^v/, '');\n } catch {\n // Network error, timeout, or parse error\n return null;\n }\n}\n\n/**\n * Compare two semver versions\n * Returns true if version1 < version2 (i.e., version2 is newer)\n */\nfunction isNewerVersion(current: string, latest: string): boolean {\n const parseVersion = (v: string) => {\n const parts = v.replace(/^v/, '').split('.').map(n => parseInt(n, 10) || 0);\n return { major: parts[0] || 0, minor: parts[1] || 0, patch: parts[2] || 0 };\n };\n\n const c = parseVersion(current);\n const l = parseVersion(latest);\n\n if (l.major > c.major) return true;\n if (l.major < c.major) return false;\n if (l.minor > c.minor) return true;\n if (l.minor < c.minor) return false;\n return l.patch > c.patch;\n}\n\n/**\n * Perform the CLI update by running the install script\n */\nfunction performCLIUpdate(): boolean {\n try {\n // Run the install command with quiet mode to avoid banner spam\n execSync(INSTALL_COMMAND, {\n stdio: 'inherit',\n shell: '/bin/bash',\n env: {\n ...process.env,\n OPENBUILDER_QUIET_INSTALL: '1', // Suppress banner in installer\n },\n });\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Check if the app/monorepo needs upgrading and perform the upgrade\n * This updates the local OpenBuilder installation that runs the web app\n */\nfunction performAppUpgrade(): boolean {\n const config = configManager.get();\n const monorepoPath = config.monorepoPath;\n\n // No monorepo configured, skip app upgrade\n if (!monorepoPath || !existsSync(monorepoPath)) {\n return true; // Not an error, just nothing to upgrade\n }\n\n console.log();\n console.log(` ${pc.cyan('⬆')} ${pc.bold('Upgrading app installation...')}`);\n console.log(` ${pc.dim(monorepoPath)}`);\n console.log();\n\n try {\n // Run openbuilder upgrade --force to upgrade the monorepo\n // Use --force to skip prompts since we're in auto-update mode\n const result = spawnSync('openbuilder', ['upgrade', '--force'], {\n stdio: 'inherit',\n env: {\n ...process.env,\n OPENBUILDER_SKIP_UPDATE_CHECK: '1', // Don't re-check for CLI updates\n },\n shell: true,\n });\n\n return result.status === 0;\n } catch {\n return false;\n }\n}\n\n/**\n * Re-launch the CLI with original arguments after update\n */\nfunction relaunchCLI(): void {\n const args = process.argv.slice(2);\n \n try {\n // Get the actual path to openbuilder to avoid shell hash caching issues\n // This ensures we run the newly installed version, not a cached path\n let openbuilderPath = 'openbuilder';\n try {\n // Use 'command -v' to get the actual path, bypassing shell hash\n openbuilderPath = execSync('command -v openbuilder', { encoding: 'utf-8' }).trim();\n } catch {\n // Fallback to just 'openbuilder' if command -v fails\n }\n \n // Use spawnSync with the explicit path to ensure we get the new version\n const result = spawnSync(openbuilderPath, args, {\n stdio: 'inherit',\n env: { \n ...process.env, \n OPENBUILDER_SKIP_UPDATE_CHECK: '1', // Prevent update loop\n OPENBUILDER_SKIP_BANNER: '1', // Suppress banner after restart (already shown before update)\n },\n // Don't use shell: true to avoid shell hash caching\n });\n \n // Exit with the same code as the relaunched process\n process.exit(result.status ?? 0);\n } catch {\n // If relaunch fails, just exit - user can run command again\n process.exit(0);\n }\n}\n\n/**\n * Main auto-update check function\n * Call this early in CLI startup\n * \n * Updates both:\n * 1. The CLI itself (globally installed)\n * 2. The app/monorepo (if configured)\n * \n * @param currentVersion - Current CLI version from package.json\n * @returns true if update was performed and CLI should exit\n */\nexport async function checkAndAutoUpdate(currentVersion: string): Promise<boolean> {\n // Skip if update check is disabled via env var\n if (process.env.OPENBUILDER_NO_UPDATE === '1' || process.env.OPENBUILDER_SKIP_UPDATE_CHECK === '1') {\n // But check if we have a pending app upgrade from a previous CLI update\n const cache = readUpdateCache();\n if (cache?.pendingAppUpgrade) {\n console.log();\n console.log(` ${pc.cyan('⬆')} ${pc.bold('Completing app upgrade...')}`);\n \n const appSuccess = performAppUpgrade();\n \n if (appSuccess) {\n // Clear the pending flag\n saveUpdateCache({ ...cache, pendingAppUpgrade: false });\n console.log(` ${pc.green('✓')} ${pc.bold('App upgrade complete!')}`);\n } else {\n console.log(` ${pc.yellow('⚠')} ${pc.dim('App upgrade failed. Run manually:')} ${pc.cyan('openbuilder upgrade')}`);\n }\n console.log();\n }\n return false;\n }\n\n // Skip if disabled in config\n const config = configManager.get();\n if (config.autoUpdate === false) {\n return false;\n }\n\n let latestVersion: string | null = null;\n\n // Check cache first to avoid hitting GitHub API too often\n const cache = readUpdateCache();\n const now = Date.now();\n\n if (cache && (now - cache.lastCheck) < CACHE_TTL_MS) {\n // Use cached version\n latestVersion = cache.latestVersion;\n } else {\n // Fetch from GitHub (with timeout)\n latestVersion = await fetchLatestVersion();\n \n if (latestVersion) {\n // Update cache\n saveUpdateCache({\n lastCheck: now,\n latestVersion,\n });\n } else if (cache) {\n // Fetch failed, use stale cache\n latestVersion = cache.latestVersion;\n }\n }\n\n // If we couldn't determine latest version, skip update\n if (!latestVersion) {\n return false;\n }\n\n // Check if update is needed\n if (!isNewerVersion(currentVersion, latestVersion)) {\n return false;\n }\n\n // Update available! Show message and perform update\n console.log();\n console.log(` ${pc.cyan('⬆')} ${pc.bold('Update available:')} ${pc.dim(currentVersion)} → ${pc.green(latestVersion)}`);\n console.log();\n\n // Step 1: Update the CLI\n console.log(` ${pc.dim('Step 1/2:')} Updating CLI...`);\n const cliSuccess = performCLIUpdate();\n\n if (!cliSuccess) {\n // CLI update failed, continue with current version\n console.log();\n console.log(` ${pc.yellow('⚠')} ${pc.dim('CLI update failed. Continuing with current version.')}`);\n console.log(` ${pc.dim('You can manually update with:')} ${pc.cyan('curl -fsSL https://openbuilder.sh/install | bash')}`);\n console.log();\n return false;\n }\n\n console.log(` ${pc.green('✓')} CLI updated to ${pc.green(latestVersion)}`);\n \n // Step 2: Check if we need to upgrade the app\n const monorepoPath = config.monorepoPath;\n const hasMonorepo = monorepoPath && existsSync(monorepoPath);\n\n if (hasMonorepo) {\n // Mark that we need to upgrade the app after CLI restart\n // The new CLI version will handle the app upgrade\n saveUpdateCache({\n lastCheck: now,\n latestVersion,\n pendingAppUpgrade: true,\n });\n \n console.log();\n console.log(` ${pc.dim('Step 2/2:')} App upgrade will continue after restart...`);\n }\n\n console.log();\n console.log(` ${pc.green('✓')} ${pc.bold('CLI update complete!')} Restarting...`);\n console.log();\n \n // Relaunch CLI with original args\n // The new CLI will pick up the pendingAppUpgrade flag and complete step 2\n relaunchCLI();\n \n return true; // CLI will exit via relaunchCLI\n}\n\n/**\n * Check for updates without auto-updating (for TUI modes)\n * Returns version info that can be displayed in the TUI\n * \n * @param currentVersion - Current CLI version from package.json\n * @returns Update info or null if no update available\n */\nexport async function checkForUpdate(currentVersion: string): Promise<{\n currentVersion: string;\n latestVersion: string;\n updateAvailable: boolean;\n} | null> {\n // Skip if update check is disabled via env var\n if (process.env.OPENBUILDER_NO_UPDATE === '1' || process.env.OPENBUILDER_SKIP_UPDATE_CHECK === '1') {\n return null;\n }\n\n // Skip if disabled in config\n const config = configManager.get();\n if (config.autoUpdate === false) {\n return null;\n }\n\n let latestVersion: string | null = null;\n\n // Check cache first to avoid hitting GitHub API too often\n const cache = readUpdateCache();\n const now = Date.now();\n\n if (cache && (now - cache.lastCheck) < CACHE_TTL_MS) {\n // Use cached version\n latestVersion = cache.latestVersion;\n } else {\n // Fetch from GitHub (with timeout)\n latestVersion = await fetchLatestVersion();\n \n if (latestVersion) {\n // Update cache\n saveUpdateCache({\n lastCheck: now,\n latestVersion,\n pendingAppUpgrade: cache?.pendingAppUpgrade,\n });\n } else if (cache) {\n // Fetch failed, use stale cache\n latestVersion = cache.latestVersion;\n }\n }\n\n // If we couldn't determine latest version, return null\n if (!latestVersion) {\n return null;\n }\n\n const updateAvailable = isNewerVersion(currentVersion, latestVersion);\n \n return {\n currentVersion,\n latestVersion,\n updateAvailable,\n };\n}\n\n/**\n * Clear the update cache (useful for testing or forcing a fresh check)\n */\nexport function clearUpdateCache(): void {\n try {\n if (existsSync(CACHE_FILE)) {\n unlinkSync(CACHE_FILE);\n }\n } catch {\n // Ignore errors\n }\n}\n\n/**\n * Get the current update status (for debugging/status commands)\n */\nexport function getUpdateStatus(): { \n cacheFile: string;\n cache: UpdateCache | null;\n monorepoPath: string | undefined;\n hasMonorepo: boolean;\n} {\n const config = configManager.get();\n const monorepoPath = config.monorepoPath;\n \n return {\n cacheFile: CACHE_FILE,\n cache: readUpdateCache(),\n monorepoPath,\n hasMonorepo: !!(monorepoPath && existsSync(monorepoPath)),\n };\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"cli-auth-rqwB_Zwt.js","sources":["../../src/cli/utils/cli-auth.ts"],"sourcesContent":["import http from 'node:http';\nimport { URL } from 'node:url';\nimport { hostname, platform } from 'node:os';\nimport { exec } from 'node:child_process';\nimport chalk from 'chalk';\nimport { configManager } from './config-manager.js';\n\n/**\n * Open a URL in the default browser (cross-platform)\n */\nasync function openBrowser(url: string): Promise<void> {\n const os = platform();\n \n return new Promise((resolve, reject) => {\n let command: string;\n \n switch (os) {\n case 'darwin':\n command = `open \"${url}\"`;\n break;\n case 'win32':\n command = `start \"\" \"${url}\"`;\n break;\n default: // Linux and others\n command = `xdg-open \"${url}\"`;\n }\n \n exec(command, (error) => {\n if (error) {\n reject(error);\n } else {\n resolve();\n }\n });\n });\n}\n\ninterface AuthResult {\n success: boolean;\n token?: string;\n error?: string;\n}\n\ninterface CLIAuthOptions {\n apiUrl?: string;\n deviceName?: string;\n silent?: boolean;\n}\n\n/**\n * Find an available port for the callback server\n */\nasync function findAvailablePort(startPort: number = 9876, endPort: number = 9999): Promise<number> {\n for (let port = startPort; port <= endPort; port++) {\n const isAvailable = await new Promise<boolean>((resolve) => {\n const server = http.createServer();\n server.once('error', () => resolve(false));\n server.once('listening', () => {\n server.close();\n resolve(true);\n });\n server.listen(port, '127.0.0.1');\n });\n \n if (isAvailable) {\n return port;\n }\n }\n \n throw new Error('No available ports found for OAuth callback server');\n}\n\n/**\n * Get a device name for the runner key\n */\nfunction getDeviceName(): string {\n const host = hostname();\n const date = new Date().toLocaleDateString();\n return `CLI - ${host} - ${date}`;\n}\n\n/**\n * Start a local HTTP server to receive the OAuth callback\n */\nfunction startCallbackServer(port: number): Promise<AuthResult> {\n return new Promise((resolve) => {\n // Track active connections so we can forcefully close them\n const connections = new Set<import('net').Socket>();\n \n const server = http.createServer((req, res) => {\n const url = new URL(req.url || '/', `http://localhost:${port}`);\n \n if (url.pathname === '/callback') {\n const token = url.searchParams.get('token');\n const status = url.searchParams.get('status');\n const error = url.searchParams.get('error');\n \n // Send response to browser\n res.writeHead(200, { 'Content-Type': 'text/html' });\n \n if (status === 'success' && token) {\n res.end(`\n <!DOCTYPE html>\n <html>\n <head>\n <title>Authentication Successful</title>\n <style>\n body {\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n display: flex;\n justify-content: center;\n align-items: center;\n min-height: 100vh;\n margin: 0;\n background: #09090b;\n color: white;\n }\n .container {\n text-align: center;\n padding: 40px;\n }\n .success {\n color: #22c55e;\n font-size: 48px;\n margin-bottom: 20px;\n }\n h1 { margin: 0 0 10px; }\n p { color: #a1a1aa; }\n </style>\n </head>\n <body>\n <div class=\"container\">\n <div class=\"success\">✓</div>\n <h1>Authentication Successful!</h1>\n <p>You can close this window and return to your terminal.</p>\n </div>\n </body>\n </html>\n `);\n \n // Close server and resolve - destroy all connections to ensure clean exit\n server.close();\n connections.forEach(conn => conn.destroy());\n resolve({ success: true, token });\n } else {\n res.end(`\n <!DOCTYPE html>\n <html>\n <head>\n <title>Authentication Failed</title>\n <style>\n body {\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n display: flex;\n justify-content: center;\n align-items: center;\n min-height: 100vh;\n margin: 0;\n background: #09090b;\n color: white;\n }\n .container {\n text-align: center;\n padding: 40px;\n }\n .error {\n color: #ef4444;\n font-size: 48px;\n margin-bottom: 20px;\n }\n h1 { margin: 0 0 10px; }\n p { color: #a1a1aa; }\n </style>\n </head>\n <body>\n <div class=\"container\">\n <div class=\"error\">✗</div>\n <h1>Authentication Failed</h1>\n <p>${error || 'An error occurred during authentication.'}</p>\n <p>Please close this window and try again.</p>\n </div>\n </body>\n </html>\n `);\n \n server.close();\n connections.forEach(conn => conn.destroy());\n resolve({ success: false, error: error || 'Authentication failed' });\n }\n } else {\n // Handle other paths\n res.writeHead(404, { 'Content-Type': 'text/plain' });\n res.end('Not found');\n }\n });\n \n // Timeout after 5 minutes\n const timeout = setTimeout(() => {\n server.close();\n connections.forEach(conn => conn.destroy());\n resolve({ success: false, error: 'Authentication timed out' });\n }, 5 * 60 * 1000);\n \n server.on('close', () => {\n clearTimeout(timeout);\n });\n \n // Handle server errors (e.g., port became unavailable due to race condition)\n server.on('error', (err: NodeJS.ErrnoException) => {\n clearTimeout(timeout);\n if (err.code === 'EADDRINUSE') {\n resolve({ success: false, error: `Port ${port} is no longer available. Please try again.` });\n } else {\n resolve({ success: false, error: `Server error: ${err.message}` });\n }\n });\n \n // Track connections for cleanup\n server.on('connection', (conn) => {\n connections.add(conn);\n conn.on('close', () => connections.delete(conn));\n });\n \n server.listen(port, '127.0.0.1');\n });\n}\n\n/**\n * Perform CLI OAuth authentication flow\n * \n * 1. Find an available port\n * 2. Start local callback server\n * 3. Request auth session from API\n * 4. Open browser to auth page\n * 5. Wait for callback\n * 6. Return token\n */\nexport async function performOAuthLogin(options: CLIAuthOptions = {}): Promise<AuthResult> {\n const apiUrl = options.apiUrl || 'https://openbuilder.sh';\n const deviceName = options.deviceName || getDeviceName();\n const silent = options.silent || false;\n \n try {\n // Find available port\n if (!silent) {\n console.log(chalk.dim('Finding available port for callback...'));\n }\n const port = await findAvailablePort();\n \n // Start callback server before requesting auth (so it's ready when browser redirects)\n const callbackPromise = startCallbackServer(port);\n \n // Request auth session from API\n if (!silent) {\n console.log(chalk.dim('Initiating authentication...'));\n }\n \n const response = await fetch(`${apiUrl}/api/auth/cli/start`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n callbackPort: port,\n callbackHost: 'localhost',\n deviceName,\n }),\n });\n \n if (!response.ok) {\n const data = await response.json();\n return { success: false, error: data.error || 'Failed to start authentication' };\n }\n \n const { authUrl } = await response.json();\n \n // Open browser\n if (!silent) {\n console.log('');\n console.log(chalk.cyan('Opening browser for authentication...'));\n console.log(chalk.dim(`If the browser doesn't open, visit:`));\n console.log(chalk.underline(authUrl));\n console.log('');\n }\n \n try {\n await openBrowser(authUrl);\n } catch {\n // Browser failed to open - user will need to copy/paste URL\n if (!silent) {\n console.log(chalk.yellow('Could not open browser automatically.'));\n }\n }\n \n if (!silent) {\n console.log(chalk.yellow('Waiting for authentication...'));\n console.log(chalk.dim('(Press Ctrl+C to cancel)'));\n }\n \n // Wait for callback\n const result = await callbackPromise;\n \n return result;\n } catch (error) {\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Authentication failed',\n };\n }\n}\n\n/**\n * Check if we have a valid stored token\n */\nexport function hasStoredToken(): boolean {\n const token = getStoredToken();\n return token !== null && token.startsWith('sv_');\n}\n\n/**\n * Get the stored runner token\n */\nexport function getStoredToken(): string | null {\n const config = configManager.get();\n \n // Check for runner token in config\n if (config.server?.secret && config.server.secret.startsWith('sv_')) {\n return config.server.secret;\n }\n \n // Legacy: check broker config\n if (config.broker?.secret && config.broker.secret.startsWith('sv_')) {\n return config.broker.secret;\n }\n \n return null;\n}\n\n/**\n * Store the runner token in config\n */\nexport function storeToken(token: string, apiUrl?: string): void {\n // Determine the WebSocket URL from the API URL\n let wsUrl = 'wss://openbuilder.sh/ws/runner';\n if (apiUrl) {\n const url = new URL(apiUrl);\n const protocol = url.protocol === 'https:' ? 'wss:' : 'ws:';\n wsUrl = `${protocol}//${url.host}/ws/runner`;\n }\n \n // Store in the server config\n configManager.set('server', {\n wsUrl,\n secret: token,\n });\n \n // Also set the apiUrl if provided\n if (apiUrl) {\n configManager.set('apiUrl', apiUrl);\n }\n}\n\n/**\n * Clear the stored token\n */\nexport function clearToken(): void {\n const config = configManager.get();\n \n // Clear server secret\n if (config.server) {\n configManager.set('server', {\n ...config.server,\n secret: '',\n });\n }\n \n // Clear legacy broker secret\n if (config.broker) {\n configManager.set('broker', {\n ...config.broker,\n secret: '',\n });\n }\n}\n\n/**\n * Validate the stored token against the server\n */\nexport async function validateToken(token: string, apiUrl: string): Promise<boolean> {\n try {\n // Try to fetch user's runner keys - this will fail if token is invalid\n const response = await fetch(`${apiUrl}/api/runner-keys`, {\n headers: {\n Authorization: `Bearer ${token}`,\n },\n });\n \n return response.ok;\n } catch {\n return false;\n }\n}\n"],"names":["http"],"mappings":"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"}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"run-LjsGm3tI.js","sources":["../../src/cli/tui/components/LogPanel.tsx","../../src/cli/tui/components/StatusBar.tsx","../../src/cli/tui/components/FullLogView.tsx","../../src/cli/tui/components/CopyMenu.tsx","../../src/cli/tui/screens/RunnerDashboard.tsx","../../src/cli/commands/run.ts"],"sourcesContent":["/**\n * LogPanel - Right panel showing scrollable log entries\n * Takes up 80% of width, shows:\n * - Timestamped log entries\n * - Tool calls with truncated args\n * - Scrollable with keyboard navigation\n */\n\nimport React, { useState, useEffect, useRef } from 'react';\nimport { Box, Text, useInput } from 'ink';\nimport { colors } from '../theme.js';\nimport type { LogEntry } from '../../../lib/logging/types.js';\n\ninterface LogPanelProps {\n entries: LogEntry[];\n isVerbose: boolean;\n width: number;\n height: number;\n isFocused: boolean;\n}\n\nexport function LogPanel({ entries, isVerbose, width, height, isFocused }: LogPanelProps) {\n const [scrollOffset, setScrollOffset] = useState(0);\n const [autoScroll, setAutoScroll] = useState(true);\n\n // Filter entries based on verbose mode\n const visibleEntries = isVerbose \n ? entries \n : entries.filter(e => !e.verbose);\n\n // Available height for log lines (subtract 2 for border, 1 for header)\n const visibleLines = Math.max(1, height - 3);\n\n // Auto-scroll when new entries arrive\n useEffect(() => {\n if (autoScroll && visibleEntries.length > 0) {\n const maxScroll = Math.max(0, visibleEntries.length - visibleLines);\n setScrollOffset(maxScroll);\n }\n }, [visibleEntries.length, autoScroll, visibleLines]);\n\n // Handle keyboard navigation\n useInput((input, key) => {\n if (!isFocused) return;\n\n if (key.upArrow) {\n setAutoScroll(false);\n setScrollOffset(prev => Math.max(0, prev - 1));\n } else if (key.downArrow) {\n const maxScroll = Math.max(0, visibleEntries.length - visibleLines);\n setScrollOffset(prev => Math.min(maxScroll, prev + 1));\n // Re-enable auto-scroll if we're at the bottom\n if (scrollOffset >= maxScroll - 1) {\n setAutoScroll(true);\n }\n } else if (key.pageUp) {\n setAutoScroll(false);\n setScrollOffset(prev => Math.max(0, prev - visibleLines));\n } else if (key.pageDown) {\n const maxScroll = Math.max(0, visibleEntries.length - visibleLines);\n setScrollOffset(prev => Math.min(maxScroll, prev + visibleLines));\n }\n });\n\n // Get visible slice of entries\n const displayedEntries = visibleEntries.slice(scrollOffset, scrollOffset + visibleLines);\n\n // Format time\n const formatTime = (timestamp: number): string => {\n return new Date(timestamp).toLocaleTimeString('en-US', {\n hour12: false,\n hour: '2-digit',\n minute: '2-digit',\n second: '2-digit',\n });\n };\n\n return (\n <Box\n flexDirection=\"column\"\n width={width}\n height={height}\n borderStyle=\"single\"\n borderColor={isFocused ? colors.cyan : colors.darkGray}\n paddingX={1}\n >\n {/* Header */}\n <Box justifyContent=\"space-between\" marginBottom={0}>\n <Text color={colors.cyan} bold>LOGS</Text>\n <Text color={colors.dimGray}>\n [verbose: {isVerbose ? 'on' : 'off'}]\n </Text>\n </Box>\n\n {/* Log entries */}\n <Box flexDirection=\"column\" flexGrow={1}>\n {displayedEntries.length === 0 ? (\n <Box justifyContent=\"center\" alignItems=\"center\" flexGrow={1}>\n <Text color={colors.dimGray}>Waiting for logs...</Text>\n </Box>\n ) : (\n displayedEntries.map((entry, index) => (\n <LogEntryRow \n key={entry.id} \n entry={entry} \n maxWidth={width - 4}\n />\n ))\n )}\n </Box>\n\n {/* Scroll indicator */}\n {visibleEntries.length > visibleLines && (\n <Box justifyContent=\"flex-end\">\n <Text color={colors.dimGray}>\n {scrollOffset + 1}-{Math.min(scrollOffset + visibleLines, visibleEntries.length)}/{visibleEntries.length}\n {autoScroll ? ' (auto)' : ''}\n </Text>\n </Box>\n )}\n </Box>\n );\n}\n\n// Individual log entry row\nfunction LogEntryRow({ entry, maxWidth }: { entry: LogEntry; maxWidth: number }) {\n const time = new Date(entry.timestamp).toLocaleTimeString('en-US', {\n hour12: false,\n hour: '2-digit',\n minute: '2-digit',\n second: '2-digit',\n });\n\n const levelColors = {\n debug: colors.dimGray,\n info: colors.cyan,\n success: colors.success,\n warn: colors.warning,\n error: colors.error,\n };\n\n const levelIcons = {\n debug: ' ',\n info: '●',\n success: '✓',\n warn: '⚠',\n error: '✗',\n };\n\n // Tool calls get special formatting\n if (entry.toolName) {\n const argsText = entry.toolArgs ? ` ${entry.toolArgs}` : '';\n const fullText = `${entry.toolName}${argsText}`;\n const truncated = fullText.length > maxWidth - 12 \n ? fullText.substring(0, maxWidth - 15) + '...'\n : fullText;\n\n return (\n <Box>\n <Text color={colors.dimGray}>{time}</Text>\n <Text color={colors.cyan}> 🔧 </Text>\n <Text color={colors.white}>{entry.toolName}</Text>\n {entry.toolArgs && <Text color={colors.gray}> {entry.toolArgs}</Text>}\n </Box>\n );\n }\n\n // Regular log entries\n const color = levelColors[entry.level];\n const icon = levelIcons[entry.level];\n \n // Truncate message if needed\n const availableWidth = maxWidth - 12; // time + space + icon + space\n const truncatedMessage = entry.message.length > availableWidth\n ? entry.message.substring(0, availableWidth - 3) + '...'\n : entry.message;\n\n return (\n <Box>\n <Text color={colors.dimGray}>{time}</Text>\n <Text color={color}> {icon} </Text>\n <Text color={color}>{truncatedMessage}</Text>\n </Box>\n );\n}\n","/**\n * StatusBar - Bottom bar showing connection status and keyboard shortcuts\n */\n\nimport React from 'react';\nimport { Box, Text } from 'ink';\nimport { colors } from '../theme.js';\nimport { getVersionInfo } from '../../utils/version-info.js';\n\ninterface StatusBarProps {\n isConnected: boolean;\n isVerbose: boolean;\n buildCount?: number;\n currentBuildIndex?: number;\n view: 'dashboard' | 'fullLog' | 'copyMenu';\n}\n\nexport function StatusBar({\n isConnected,\n isVerbose,\n buildCount = 0,\n currentBuildIndex = 0,\n view,\n}: StatusBarProps) {\n // Get version info\n const versionInfo = getVersionInfo();\n \n // Connection indicator\n const connectionIndicator = (\n <Box marginRight={2}>\n <Text color={isConnected ? colors.success : colors.error}>\n {isConnected ? '●' : '○'}\n </Text>\n <Text color={colors.gray}>\n {' '}{isConnected ? 'Connected' : 'Disconnected'}\n </Text>\n </Box>\n );\n\n // Build count indicator (only show if multiple builds)\n const buildIndicator = buildCount > 1 ? (\n <Box marginRight={2}>\n <Text color={colors.gray}>\n Build {currentBuildIndex + 1}/{buildCount}\n </Text>\n </Box>\n ) : null;\n\n // Shortcuts based on current view\n const shortcuts = view === 'dashboard' ? (\n <Box>\n <Shortcut letter=\"q\" label=\"quit\" />\n <Shortcut letter=\"v\" label={`verbose: ${isVerbose ? 'on' : 'off'}`} />\n <Shortcut letter=\"c\" label=\"copy\" />\n <Shortcut letter=\"t\" label=\"text view\" />\n {buildCount > 1 && <Shortcut letter=\"n/p\" label=\"switch build\" />}\n <Shortcut letter=\"↑↓\" label=\"scroll\" />\n </Box>\n ) : view === 'fullLog' ? (\n <Box>\n <Shortcut letter=\"t\" label=\"dashboard\" />\n <Shortcut letter=\"c\" label=\"copy\" />\n <Shortcut letter=\"/\" label=\"search\" />\n <Shortcut letter=\"f\" label=\"filter\" />\n <Shortcut letter=\"↑↓\" label=\"scroll\" />\n <Shortcut letter=\"PgUp/Dn\" label=\"page\" />\n </Box>\n ) : (\n <Box>\n <Shortcut letter=\"Esc\" label=\"cancel\" />\n </Box>\n );\n\n // Version display\n const versionDisplay = (\n <Box marginLeft={2}>\n <Text color={colors.dimGray}>\n {versionInfo.display}\n </Text>\n </Box>\n );\n\n return (\n <Box\n borderStyle=\"single\"\n borderColor={colors.darkGray}\n paddingX={1}\n justifyContent=\"space-between\"\n >\n <Box>\n {connectionIndicator}\n {buildIndicator}\n </Box>\n <Box>\n {shortcuts}\n {versionDisplay}\n </Box>\n </Box>\n );\n}\n\n// Helper component for shortcuts\nfunction Shortcut({ letter, label }: { letter: string; label: string }) {\n return (\n <Box marginRight={2}>\n <Text color={colors.dimGray}>[</Text>\n <Text color={colors.cyan}>{letter}</Text>\n <Text color={colors.dimGray}>]</Text>\n <Text color={colors.gray}>{label}</Text>\n </Box>\n );\n}\n","/**\n * FullLogView - Full-screen log view with search and filter\n */\n\nimport React, { useState, useEffect } from 'react';\nimport { Box, Text, useInput, useStdout } from 'ink';\nimport TextInput from 'ink-text-input';\nimport { colors } from '../theme.js';\nimport type { LogEntry, LogLevel } from '../../../lib/logging/types.js';\n\ntype FilterMode = 'all' | 'errors' | 'tools' | 'verbose';\n\ninterface FullLogViewProps {\n entries: LogEntry[];\n onBack: () => void;\n onCopy: () => void;\n}\n\nexport function FullLogView({ entries, onBack, onCopy }: FullLogViewProps) {\n const { stdout } = useStdout();\n const terminalHeight = stdout?.rows || 24;\n const terminalWidth = stdout?.columns || 80;\n\n const [searchQuery, setSearchQuery] = useState('');\n const [isSearching, setIsSearching] = useState(false);\n const [filterMode, setFilterMode] = useState<FilterMode>('all');\n const [scrollOffset, setScrollOffset] = useState(0);\n const [searchMode, setSearchMode] = useState<'filter' | 'highlight'>('highlight');\n\n // Available height for log lines\n const headerHeight = 3;\n const footerHeight = 2;\n const visibleLines = Math.max(1, terminalHeight - headerHeight - footerHeight);\n\n // Filter and search entries\n const processedEntries = entries.filter(entry => {\n // Apply filter mode\n if (filterMode === 'errors' && entry.level !== 'error' && entry.level !== 'warn') {\n return false;\n }\n if (filterMode === 'tools' && !entry.toolName) {\n return false;\n }\n if (filterMode !== 'verbose' && entry.verbose) {\n return false;\n }\n\n // Apply search filter (if in filter mode)\n if (searchQuery && searchMode === 'filter') {\n const query = searchQuery.toLowerCase();\n const messageMatch = entry.message.toLowerCase().includes(query);\n const toolMatch = entry.toolName?.toLowerCase().includes(query);\n const argsMatch = entry.toolArgs?.toLowerCase().includes(query);\n return messageMatch || toolMatch || argsMatch;\n }\n\n return true;\n });\n\n // Calculate max scroll\n const maxScroll = Math.max(0, processedEntries.length - visibleLines);\n\n // Get visible entries\n const visibleEntries = processedEntries.slice(scrollOffset, scrollOffset + visibleLines);\n\n // Handle keyboard input\n useInput((input, key) => {\n if (isSearching) {\n if (key.escape || key.return) {\n setIsSearching(false);\n }\n return;\n }\n\n if (input === 't') {\n onBack();\n } else if (input === 'c') {\n onCopy();\n } else if (input === '/') {\n setIsSearching(true);\n } else if (input === 'f') {\n // Cycle through filter modes\n const modes: FilterMode[] = ['all', 'errors', 'tools', 'verbose'];\n const currentIndex = modes.indexOf(filterMode);\n setFilterMode(modes[(currentIndex + 1) % modes.length]);\n setScrollOffset(0);\n } else if (input === 'm') {\n // Toggle search mode\n setSearchMode(prev => prev === 'filter' ? 'highlight' : 'filter');\n } else if (key.upArrow) {\n setScrollOffset(prev => Math.max(0, prev - 1));\n } else if (key.downArrow) {\n setScrollOffset(prev => Math.min(maxScroll, prev + 1));\n } else if (key.pageUp) {\n setScrollOffset(prev => Math.max(0, prev - visibleLines));\n } else if (key.pageDown) {\n setScrollOffset(prev => Math.min(maxScroll, prev + visibleLines));\n } else if (key.escape) {\n if (searchQuery) {\n setSearchQuery('');\n } else {\n onBack();\n }\n }\n });\n\n // Format time\n const formatTime = (timestamp: number): string => {\n return new Date(timestamp).toLocaleTimeString('en-US', {\n hour12: false,\n hour: '2-digit',\n minute: '2-digit',\n second: '2-digit',\n });\n };\n\n // Check if text matches search query\n const highlightSearch = (text: string): React.ReactNode => {\n if (!searchQuery || searchMode !== 'highlight') {\n return text;\n }\n\n const query = searchQuery.toLowerCase();\n const index = text.toLowerCase().indexOf(query);\n if (index === -1) {\n return text;\n }\n\n return (\n <>\n {text.slice(0, index)}\n <Text backgroundColor={colors.warning} color=\"black\">\n {text.slice(index, index + searchQuery.length)}\n </Text>\n {text.slice(index + searchQuery.length)}\n </>\n );\n };\n\n return (\n <Box flexDirection=\"column\" height={terminalHeight}>\n {/* Header with search */}\n <Box\n borderStyle=\"single\"\n borderColor={colors.darkGray}\n paddingX={1}\n justifyContent=\"space-between\"\n >\n <Text color={colors.cyan} bold>LOGS</Text>\n <Box>\n <Text color={colors.dimGray}>Search: </Text>\n {isSearching ? (\n <Box borderStyle=\"round\" borderColor={colors.cyan} paddingX={1}>\n <TextInput\n value={searchQuery}\n onChange={setSearchQuery}\n placeholder=\"type to search...\"\n />\n </Box>\n ) : (\n <Text color={searchQuery ? colors.white : colors.dimGray}>\n [{searchQuery || 'none'}] ({searchMode})\n </Text>\n )}\n <Text color={colors.dimGray}> [/]</Text>\n </Box>\n </Box>\n\n {/* Log content */}\n <Box\n flexDirection=\"column\"\n flexGrow={1}\n borderStyle=\"single\"\n borderColor={colors.darkGray}\n borderTop={false}\n borderBottom={false}\n paddingX={1}\n >\n {visibleEntries.map((entry, index) => {\n const time = formatTime(entry.timestamp);\n const levelColors = {\n debug: colors.dimGray,\n info: colors.cyan,\n success: colors.success,\n warn: colors.warning,\n error: colors.error,\n };\n const levelIcons = {\n debug: ' ',\n info: '●',\n success: '✓',\n warn: '⚠',\n error: '✗',\n };\n\n if (entry.toolName) {\n return (\n <Box key={entry.id}>\n <Text color={colors.dimGray}>{time}</Text>\n <Text color={colors.cyan}> 🔧 </Text>\n <Text color={colors.white}>{highlightSearch(entry.toolName)}</Text>\n {entry.toolArgs && (\n <Text color={colors.gray}> {highlightSearch(entry.toolArgs)}</Text>\n )}\n </Box>\n );\n }\n\n return (\n <Box key={entry.id}>\n <Text color={colors.dimGray}>{time}</Text>\n <Text color={levelColors[entry.level]}> {levelIcons[entry.level]} </Text>\n <Text color={levelColors[entry.level]}>{highlightSearch(entry.message)}</Text>\n </Box>\n );\n })}\n </Box>\n\n {/* Footer with shortcuts and scroll position */}\n <Box\n borderStyle=\"single\"\n borderColor={colors.darkGray}\n paddingX={1}\n justifyContent=\"space-between\"\n >\n <Box>\n <Shortcut letter=\"t\" label=\"dashboard\" />\n <Shortcut letter=\"c\" label=\"copy\" />\n <Shortcut letter=\"/\" label=\"search\" />\n <Shortcut letter=\"f\" label={`filter: ${filterMode}`} />\n <Shortcut letter=\"m\" label={`mode: ${searchMode}`} />\n </Box>\n <Text color={colors.dimGray}>\n {scrollOffset + 1}-{Math.min(scrollOffset + visibleLines, processedEntries.length)}/{processedEntries.length}\n </Text>\n </Box>\n </Box>\n );\n}\n\nfunction Shortcut({ letter, label }: { letter: string; label: string }) {\n return (\n <Box marginRight={2}>\n <Text color={colors.dimGray}>[</Text>\n <Text color={colors.cyan}>{letter}</Text>\n <Text color={colors.dimGray}>]</Text>\n <Text color={colors.gray}>{label}</Text>\n </Box>\n );\n}\n","/**\n * CopyMenu - Modal for copying logs to clipboard\n */\n\nimport React from 'react';\nimport { Box, Text, useInput } from 'ink';\nimport { colors } from '../theme.js';\n\ninterface CopyMenuProps {\n onSelect: (option: 'visible' | 'last50' | 'last100' | 'all' | 'range') => void;\n onCancel: () => void;\n visibleCount: number;\n totalCount: number;\n}\n\nexport function CopyMenu({ onSelect, onCancel, visibleCount, totalCount }: CopyMenuProps) {\n useInput((input, key) => {\n if (key.escape) {\n onCancel();\n } else if (input === '1') {\n onSelect('visible');\n } else if (input === '2') {\n onSelect('last50');\n } else if (input === '3') {\n onSelect('last100');\n } else if (input === '4') {\n onSelect('all');\n } else if (input === '5') {\n onSelect('range');\n }\n });\n\n return (\n <Box\n flexDirection=\"column\"\n borderStyle=\"round\"\n borderColor={colors.cyan}\n paddingX={2}\n paddingY={1}\n >\n <Box marginBottom={1}>\n <Text color={colors.cyan} bold>Copy Logs</Text>\n </Box>\n\n <Box flexDirection=\"column\" marginBottom={1}>\n <CopyOption number=\"1\" label={`Copy visible (${visibleCount} lines)`} />\n <CopyOption number=\"2\" label=\"Copy last 50 lines\" />\n <CopyOption number=\"3\" label=\"Copy last 100 lines\" />\n <CopyOption number=\"4\" label={`Copy all from file (${totalCount} lines)`} />\n <CopyOption number=\"5\" label=\"Copy range...\" />\n </Box>\n\n <Box>\n <Text color={colors.dimGray}>[Esc] Cancel</Text>\n </Box>\n </Box>\n );\n}\n\nfunction CopyOption({ number, label }: { number: string; label: string }) {\n return (\n <Box>\n <Text color={colors.dimGray}>[</Text>\n <Text color={colors.cyan}>{number}</Text>\n <Text color={colors.dimGray}>]</Text>\n <Text color={colors.white}> {label}</Text>\n </Box>\n );\n}\n","/**\n * RunnerDashboard - Main TUI container for the runner\n * \n * Layout:\n * - 20% left: BuildPanel (current build + todos) - hidden when no build\n * - 80% right: LogPanel (scrollable logs)\n * - Bottom: StatusBar (connection, shortcuts)\n * \n * Views:\n * - dashboard: Main split view\n * - fullLog: Full-screen log view with search\n * - copyMenu: Copy options modal overlay\n */\n\nimport React, { useState, useEffect, useCallback } from 'react';\nimport { Box, Text, useInput, useStdout, useApp } from 'ink';\nimport { colors } from '../theme.js';\nimport { BuildPanel } from '../components/BuildPanel.js';\nimport { LogPanel } from '../components/LogPanel.js';\nimport { StatusBar } from '../components/StatusBar.js';\nimport { FullLogView } from '../components/FullLogView.js';\nimport { CopyMenu } from '../components/CopyMenu.js';\nimport { useBuildState, useLogEntries } from '../hooks/useBuildState.js';\nimport { Banner } from '../components/Banner.js';\nimport type { BuildInfo, TodoItem, LogEntry } from '../../../lib/logging/types.js';\nimport { getLogBuffer } from '../../../lib/logging/log-buffer.js';\n\ntype ViewMode = 'dashboard' | 'fullLog' | 'copyMenu';\n\nexport interface RunnerDashboardProps {\n // Initial config (passed from runner)\n config: {\n runnerId: string;\n serverUrl: string;\n workspace: string;\n apiUrl?: string;\n };\n // Event handlers\n onQuit?: () => void;\n}\n\nexport function RunnerDashboard({ config, onQuit }: RunnerDashboardProps) {\n const { exit } = useApp();\n const { stdout } = useStdout();\n const terminalHeight = stdout?.rows || 24;\n const terminalWidth = stdout?.columns || 80;\n\n // State management\n const [buildState, buildActions] = useBuildState();\n const logEntries = useLogEntries(100);\n const [view, setView] = useState<ViewMode>('dashboard');\n\n // Handle keyboard input\n useInput((input, key) => {\n // Global quit handler\n if (input === 'q' && view !== 'copyMenu') {\n if (onQuit) {\n onQuit();\n }\n exit();\n return;\n }\n\n // Copy menu is modal - handle separately\n if (view === 'copyMenu') {\n return; // CopyMenu handles its own input\n }\n\n // Dashboard shortcuts\n if (view === 'dashboard') {\n if (input === 'v') {\n buildActions.toggleVerbose();\n } else if (input === 'c') {\n setView('copyMenu');\n } else if (input === 't') {\n setView('fullLog');\n } else if (input === 'n') {\n buildActions.nextBuild();\n } else if (input === 'p') {\n buildActions.prevBuild();\n }\n }\n\n // Full log view shortcuts\n if (view === 'fullLog') {\n if (input === 't') {\n setView('dashboard');\n } else if (input === 'c') {\n setView('copyMenu');\n }\n }\n });\n\n // Handle copy action\n const handleCopy = useCallback(async (option: 'visible' | 'last50' | 'last100' | 'all' | 'range') => {\n try {\n const buffer = getLogBuffer();\n let entriesToCopy: LogEntry[] = [];\n\n switch (option) {\n case 'visible':\n entriesToCopy = logEntries.slice(-20); // Approximate visible count\n break;\n case 'last50':\n entriesToCopy = buffer.getRecent(50);\n break;\n case 'last100':\n entriesToCopy = buffer.getRecent(100);\n break;\n case 'all':\n entriesToCopy = buffer.readFromFile();\n break;\n case 'range':\n // TODO: Implement range selection\n entriesToCopy = buffer.getRecent(100);\n break;\n }\n\n const text = buffer.toText(entriesToCopy);\n \n // Copy to clipboard using pbcopy on macOS\n const { spawn } = await import('child_process');\n const pbcopy = spawn('pbcopy');\n pbcopy.stdin.write(text);\n pbcopy.stdin.end();\n \n // TODO: Show success message\n } catch (error) {\n console.error('Failed to copy to clipboard:', error);\n }\n\n setView('dashboard');\n }, [logEntries]);\n\n // Calculate panel dimensions\n const bannerHeight = 7; // ASCII art banner\n const headerHeight = 3; // Config/status line\n const statusBarHeight = 3;\n const contentHeight = Math.max(1, terminalHeight - bannerHeight - headerHeight - statusBarHeight);\n \n // 20/80 split\n const buildPanelWidth = Math.floor(terminalWidth * 0.2);\n const logPanelWidth = terminalWidth - buildPanelWidth;\n\n // Show build panel only when there's an active build\n const showBuildPanel = buildState.currentBuild !== null;\n\n // Check for available update (set by auto-update check in index.ts)\n const updateAvailable = process.env.OPENBUILDER_UPDATE_AVAILABLE;\n\n return (\n <Box flexDirection=\"column\" height={terminalHeight} width={terminalWidth}>\n {/* Banner */}\n <Banner />\n\n {/* Update notification banner */}\n {updateAvailable && (\n <Box justifyContent=\"center\" paddingY={0}>\n <Text color={colors.cyan}>⬆ Update available: </Text>\n <Text color={colors.success}>{updateAvailable}</Text>\n <Text color={colors.dimGray}> — Run </Text>\n <Text color={colors.cyan}>openbuilder upgrade</Text>\n <Text color={colors.dimGray}> to update</Text>\n </Box>\n )}\n\n {/* Header bar with connection status */}\n <Box\n borderStyle=\"single\"\n borderColor={colors.darkGray}\n paddingX={1}\n justifyContent=\"space-between\"\n >\n <Text color={colors.dimGray}>\n Runner: <Text color={colors.cyan}>{config.runnerId}</Text> • Server: <Text color={colors.cyan}>{config.serverUrl.replace(/^wss?:\\/\\//, '')}</Text>\n </Text>\n <Box>\n <Text color={buildState.isConnected ? colors.success : colors.error}>\n {buildState.isConnected ? '●' : '○'}\n </Text>\n <Text color={colors.gray}>\n {' '}{buildState.isConnected ? 'Connected' : 'Disconnected'}\n </Text>\n </Box>\n </Box>\n\n {/* Main content */}\n {view === 'dashboard' && (\n <Box flexGrow={1} height={contentHeight}>\n {showBuildPanel && (\n <BuildPanel\n build={buildState.currentBuild}\n width={buildPanelWidth}\n height={contentHeight}\n />\n )}\n <LogPanel\n entries={logEntries}\n isVerbose={buildState.isVerbose}\n width={showBuildPanel ? logPanelWidth : terminalWidth}\n height={contentHeight}\n isFocused={true}\n />\n </Box>\n )}\n\n {view === 'fullLog' && (\n <FullLogView\n entries={logEntries}\n onBack={() => setView('dashboard')}\n onCopy={() => setView('copyMenu')}\n />\n )}\n\n {/* Copy menu overlay */}\n {view === 'copyMenu' && (\n <Box\n position=\"absolute\"\n flexDirection=\"column\"\n justifyContent=\"center\"\n alignItems=\"center\"\n width={terminalWidth}\n height={terminalHeight}\n >\n <CopyMenu\n onSelect={handleCopy}\n onCancel={() => setView('dashboard')}\n visibleCount={Math.min(20, logEntries.length)}\n totalCount={getLogBuffer().readFromFile().length}\n />\n </Box>\n )}\n\n {/* Status bar (only in dashboard view) */}\n {view === 'dashboard' && (\n <StatusBar\n isConnected={buildState.isConnected}\n isVerbose={buildState.isVerbose}\n buildCount={buildState.builds.length}\n currentBuildIndex={buildState.currentBuildIndex}\n view={view}\n />\n )}\n </Box>\n );\n}\n\n// Export for external use\nexport { useBuildState, useLogEntries };\n","import chalk from 'chalk';\nimport { homedir, userInfo } from 'node:os';\nimport { join } from 'node:path';\nimport { render } from 'ink';\nimport React from 'react';\nimport { logger } from '../utils/logger.js';\nimport { configManager } from '../utils/config-manager.js';\nimport { startRunner } from '../../index.js';\nimport { RunnerDashboard } from '../tui/screens/RunnerDashboard.js';\nimport { initRunnerLogger } from '../../lib/logging/index.js';\nimport { setFileLoggerTuiMode } from '../../lib/file-logger.js';\nimport { \n hasStoredToken, \n getStoredToken, \n performOAuthLogin, \n storeToken \n} from '../utils/cli-auth.js';\n\n// Default public OpenBuilder instance\nconst DEFAULT_URL = 'https://openbuilder.up.railway.app';\nconst DEFAULT_WORKSPACE = join(homedir(), 'openbuilder-workspace');\n\n/**\n * Normalize URL by adding protocol if missing\n * Uses http:// for localhost, https:// for everything else\n */\nfunction normalizeUrl(url: string): string {\n if (!url) return url;\n\n // If protocol already present, return as-is\n if (url.match(/^https?:\\/\\//i)) {\n return url;\n }\n\n // For localhost or 127.0.0.1, use http://\n if (url.match(/^(localhost|127\\.0\\.0\\.1)(:|\\/|$)/i)) {\n return `http://${url}`;\n }\n\n // For everything else, use https://\n return `https://${url}`;\n}\n\n/**\n * Derive WebSocket URL from a base HTTP/HTTPS URL\n * Converts https://example.com to wss://example.com/ws/runner\n */\nfunction deriveWsUrl(baseUrl: string): string {\n const normalized = normalizeUrl(baseUrl);\n const wsProtocol = normalized.startsWith('https://') ? 'wss://' : 'ws://';\n const hostPath = normalized.replace(/^https?:\\/\\//, '');\n // Remove trailing slash if present\n const cleanHostPath = hostPath.replace(/\\/$/, '');\n return `${wsProtocol}${cleanHostPath}/ws/runner`;\n}\n\n/**\n * Get the current system username\n */\nfunction getSystemUsername(): string {\n try {\n return userInfo().username;\n } catch {\n // Fallback if userInfo() fails\n return process.env.USER || process.env.USERNAME || 'runner';\n }\n}\n\ninterface RunOptions {\n broker?: string; // Legacy: WebSocket URL override (deprecated, use --url instead)\n url?: string;\n workspace?: string;\n runnerId?: string;\n secret?: string;\n verbose?: boolean;\n local?: boolean; // Enable local mode (bypasses authentication)\n noTui?: boolean; // Disable TUI dashboard\n}\n\n/**\n * Check if we should use TUI\n */\nfunction shouldUseTUI(options: RunOptions): boolean {\n // Explicit flag\n if (options.noTui) return false;\n\n // CI/CD environments\n if (process.env.CI === '1' || process.env.CI === 'true') return false;\n\n // Not a TTY\n if (!process.stdout.isTTY) return false;\n\n // Explicit env var to disable\n if (process.env.NO_TUI === '1') return false;\n\n return true;\n}\n\nexport async function runCommand(options: RunOptions) {\n // Set local mode environment variable if requested\n if (options.local) {\n process.env.OPENBUILDER_LOCAL_MODE = 'true';\n logger.info(chalk.yellow('Local mode enabled - authentication bypassed'));\n }\n\n const useTUI = shouldUseTUI(options);\n\n // Build runner options from CLI flags or smart defaults\n // NOTE: For the `runner` command, we intentionally ignore local config values\n // and default to the public OpenBuilder instance. This command is specifically\n // for connecting to remote servers, not local development.\n // Users can still override with CLI flags if needed.\n \n // Resolve API URL: CLI flag > default public instance (ignore config)\n const apiUrl = normalizeUrl(options.url || DEFAULT_URL);\n \n // Resolve WebSocket URL: CLI broker flag > derive from API URL (ignore config)\n const wsUrl = options.broker || deriveWsUrl(apiUrl);\n \n // Resolve workspace: CLI flag > config > default ~/openbuilder-workspace\n // (workspace from config is fine since it's user's preference for where projects go)\n const config = configManager.get();\n const workspace = options.workspace || config.workspace || DEFAULT_WORKSPACE;\n \n // Resolve runner ID: CLI flag > system username (ignore config 'local' default)\n const runnerId = options.runnerId || getSystemUsername();\n \n // Resolve secret: CLI flag > config (required)\n // Only use config secret if it looks like a valid token (starts with sv_)\n // This prevents the default 'dev-secret' from being used in runner mode\n const configSecret = configManager.getSecret();\n const sharedSecret = options.secret || (configSecret?.startsWith('sv_') ? configSecret : undefined);\n\n const runnerOptions = {\n wsUrl,\n apiUrl,\n sharedSecret,\n runnerId,\n workspace,\n verbose: options.verbose,\n tuiMode: useTUI,\n };\n\n // Validate required options - secret is required\n // If not provided, try to use stored OAuth token or trigger OAuth flow\n if (!runnerOptions.sharedSecret) {\n // Check if we have a stored OAuth token\n if (hasStoredToken()) {\n const storedToken = getStoredToken();\n if (storedToken) {\n runnerOptions.sharedSecret = storedToken;\n logger.info(`Using stored runner token: ${chalk.cyan(storedToken.substring(0, 12) + '...')}`);\n }\n }\n \n // If still no secret and not in local mode, trigger OAuth flow\n if (!runnerOptions.sharedSecret && !options.local) {\n logger.info('No runner token found. Starting OAuth authentication...');\n logger.info('');\n \n const result = await performOAuthLogin({\n apiUrl: runnerOptions.apiUrl,\n silent: false,\n });\n \n if (result.success && result.token) {\n storeToken(result.token, runnerOptions.apiUrl);\n runnerOptions.sharedSecret = result.token;\n logger.log('');\n logger.success('Authentication successful!');\n logger.info(`Token: ${chalk.cyan(result.token.substring(0, 12) + '...')}`);\n logger.log('');\n } else {\n logger.error(result.error || 'Authentication failed');\n logger.info('');\n logger.info('You can also provide a token manually:');\n logger.info(` ${chalk.cyan('openbuilder runner --secret <your-secret>')}`);\n logger.info('');\n logger.info('Or login first:');\n logger.info(` ${chalk.cyan('openbuilder login')}`);\n process.exit(1);\n }\n }\n \n // Final check - if still no secret (and not local mode)\n if (!runnerOptions.sharedSecret && !options.local) {\n logger.error('Shared secret is required');\n logger.info('');\n logger.info('Get a runner key from your OpenBuilder dashboard, or provide via:');\n logger.info(` ${chalk.cyan('openbuilder runner --secret <your-secret>')}`);\n logger.info('');\n logger.info('Or login with OAuth:');\n logger.info(` ${chalk.cyan('openbuilder login')}`);\n process.exit(1);\n }\n }\n\n // ========================================\n // PLAIN TEXT MODE (--no-tui)\n // ========================================\n if (!useTUI) {\n // Display startup info\n logger.section('Starting OpenBuilder Runner');\n logger.info(`Server: ${chalk.cyan(runnerOptions.wsUrl)}`);\n logger.info(`API URL: ${chalk.cyan(runnerOptions.apiUrl)}`);\n logger.info(`Runner ID: ${chalk.cyan(runnerOptions.runnerId)}`);\n logger.info(`Workspace: ${chalk.cyan(runnerOptions.workspace)}`);\n logger.log('');\n\n if (options.verbose) {\n logger.debug('Verbose logging enabled');\n logger.debug(`Full options: ${JSON.stringify(runnerOptions, null, 2)}`);\n }\n\n try {\n // Start the runner (runs indefinitely)\n await startRunner(runnerOptions);\n } catch (error) {\n logger.error('Failed to start runner:');\n logger.error(error instanceof Error ? error.message : 'Unknown error');\n if (error instanceof Error && error.stack) {\n logger.debug(error.stack);\n }\n process.exit(1);\n }\n return;\n }\n\n // ========================================\n // TUI MODE (default)\n // ========================================\n \n // Initialize the logger BEFORE rendering TUI so the TUI can subscribe to events\n // This must happen before startRunner() which would create its own logger\n initRunnerLogger({\n verbose: options.verbose || false,\n tuiMode: true,\n });\n \n // Enable TUI mode in file-logger to suppress terminal output\n setFileLoggerTuiMode(true);\n \n // Track runner cleanup function\n let runnerCleanupFn: (() => Promise<void>) | undefined;\n\n // Clear screen and enter alternate buffer for clean TUI\n process.stdout.write('\\x1b[?1049h'); // Enter alternate screen\n process.stdout.write('\\x1b[2J\\x1b[H'); // Clear and home\n\n // Ensure stdin is in raw mode for keyboard input\n if (process.stdin.setRawMode) {\n process.stdin.setRawMode(true);\n }\n process.stdin.resume();\n\n // Handle quit from TUI\n const handleQuit = async () => {\n // Exit alternate screen buffer\n process.stdout.write('\\x1b[?1049l');\n \n console.log('\\n' + chalk.yellow('Shutting down runner...'));\n \n if (runnerCleanupFn) {\n try {\n await runnerCleanupFn();\n console.log(chalk.green('✓') + ' Runner stopped');\n } catch (e) {\n console.error(chalk.red('✗') + ' Error stopping runner:', e);\n }\n }\n \n process.exit(0);\n };\n\n // Handle SIGINT (Ctrl+C)\n process.on('SIGINT', handleQuit);\n\n // Render the TUI dashboard\n const { waitUntilExit, clear } = render(\n React.createElement(RunnerDashboard, {\n config: {\n runnerId,\n serverUrl: wsUrl,\n workspace,\n apiUrl,\n },\n onQuit: handleQuit,\n }),\n {\n stdin: process.stdin,\n stdout: process.stdout,\n stderr: process.stderr,\n exitOnCtrlC: false, // We handle this ourselves\n patchConsole: false, // We use our own logging\n }\n );\n\n try {\n // Start the runner and get cleanup function\n runnerCleanupFn = await startRunner(runnerOptions);\n\n // Wait for TUI to exit (user pressed 'q')\n await waitUntilExit();\n\n // Clean up\n clear();\n await handleQuit();\n } catch (error) {\n clear();\n process.stdout.write('\\x1b[?1049l'); // Exit alternate screen\n \n logger.error('Failed to start runner:');\n logger.error(error instanceof Error ? error.message : 'Unknown error');\n if (error instanceof Error && error.stack) {\n logger.debug(error.stack);\n }\n process.exit(1);\n }\n}\n"],"names":["useState","useEffect","_jsxs","_jsx","Shortcut","useCallback"],"mappings":"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"}
|