openpencil/server/api/ai/install-agent.ts
Kayshen Xu 6a1891fc6e V0.4.3 (#50)
* chore(ai): update dependencies and enhance logging functionality

- Bump version of `@github/copilot-sdk` and related packages to `0.1.32` and `1.0.7` for improved features and bug fixes.
- Update Discord invite links across multiple README files to the new server.
- Introduce a new logging utility in `server/utils/server-logger.ts` for better server process logging, including automatic log cleanup and directory management.
- Enhance the `connect-agent.ts` and `install-agent.ts` files to improve OpenCode binary resolution and installation commands.
- Refactor `resolve-claude-cli.ts` and `resolve-copilot-cli.ts` to include detailed logging for CLI binary resolution processes.

This update improves dependency management, enhances user experience with updated links, and provides better insights into server operations through logging.

* chore: bump version from 0.4.0 to 0.4.3 in package.json
2026-03-18 21:34:58 +08:00

188 lines
5.2 KiB
TypeScript

import { defineEventHandler, readBody, setResponseHeaders } from 'h3'
import { execSync } from 'node:child_process'
interface InstallBody {
agent: 'claude-code' | 'codex-cli' | 'opencode' | 'copilot'
}
interface InstallResult {
success: boolean
error?: string
command?: string
docsUrl?: string
}
const BINARY_MAP: Record<string, string> = {
'claude-code': 'claude',
'codex-cli': 'codex',
'opencode': 'opencode',
'copilot': 'copilot',
}
function checkBinary(binary: string): boolean {
try {
const cmd = process.platform === 'win32'
? `where ${binary} 2>nul`
: `which ${binary} 2>/dev/null`
return !!execSync(cmd, { encoding: 'utf-8', timeout: 5000 }).trim()
} catch {
return false
}
}
function hasCommand(cmd: string): boolean {
return checkBinary(cmd)
}
function getInstallInfo(agent: string): { command: string; docsUrl: string } {
const isWin = process.platform === 'win32'
const isMac = process.platform === 'darwin'
switch (agent) {
case 'claude-code':
return {
command: 'npm install -g @anthropic-ai/claude-code',
docsUrl: 'https://docs.anthropic.com/en/docs/claude-code',
}
case 'codex-cli':
return {
command: 'npm install -g @openai/codex',
docsUrl: 'https://github.com/openai/codex',
}
case 'opencode':
return {
command: isWin
? 'npm install -g opencode-ai'
: 'curl -fsSL https://opencode.ai/install | bash',
docsUrl: 'https://opencode.ai',
}
case 'copilot':
return {
command: isMac
? 'brew install github/copilot/copilot'
: isWin
? 'winget install GitHub.CopilotCLI'
: 'See documentation',
docsUrl: 'https://docs.github.com/copilot/how-tos/copilot-cli',
}
default:
return { command: '', docsUrl: '' }
}
}
/**
* POST /api/ai/install-agent
* Attempts to auto-install a CLI agent. Returns manual instructions on failure.
*/
export default defineEventHandler(async (event) => {
const body = await readBody<InstallBody>(event)
setResponseHeaders(event, { 'Content-Type': 'application/json' })
if (!body?.agent) {
return { success: false, error: 'Missing agent field' } satisfies InstallResult
}
const binary = BINARY_MAP[body.agent]
if (!binary) {
return { success: false, error: `Unknown agent: ${body.agent}` } satisfies InstallResult
}
// Already installed
if (checkBinary(binary)) {
return { success: true } satisfies InstallResult
}
const info = getInstallInfo(body.agent)
// Try auto-install
const result = await tryAutoInstall(body.agent, binary)
if (result.success) return result
// Return failure with manual instructions
return {
success: false,
error: result.error || 'Auto-install failed',
command: info.command,
docsUrl: info.docsUrl,
} satisfies InstallResult
})
async function tryAutoInstall(agent: string, binary: string): Promise<InstallResult> {
switch (agent) {
case 'claude-code':
return tryNpmInstall('@anthropic-ai/claude-code', binary)
case 'codex-cli':
return tryNpmInstall('@openai/codex', binary)
case 'opencode':
return tryOpenCodeInstall(binary)
case 'copilot':
return tryCopilotInstall(binary)
default:
return { success: false, error: 'Unknown agent' }
}
}
async function tryNpmInstall(pkg: string, binary: string): Promise<InstallResult> {
if (!hasCommand('npm')) {
return { success: false, error: 'npm not found. Install Node.js first.' }
}
try {
const npmBin = process.platform === 'win32' ? 'npm.cmd' : 'npm'
execSync(`${npmBin} install -g ${pkg}`, {
encoding: 'utf-8',
timeout: 180_000,
stdio: 'pipe',
})
} catch (err) {
const stderr = (err as { stderr?: string }).stderr || ''
const msg = stderr.includes('EACCES')
? 'Permission denied. Try running with sudo or fix npm permissions.'
: err instanceof Error ? err.message : 'npm install failed'
return { success: false, error: msg }
}
return checkBinary(binary)
? { success: true }
: { success: false, error: 'Install completed but binary not found in PATH' }
}
async function tryOpenCodeInstall(binary: string): Promise<InstallResult> {
const isWin = process.platform === 'win32'
const cmd = isWin
? 'npm.cmd install -g opencode-ai'
: 'curl -fsSL https://opencode.ai/install | bash'
try {
execSync(cmd, {
encoding: 'utf-8',
timeout: 120_000,
stdio: 'pipe',
})
} catch (err) {
const msg = err instanceof Error ? err.message : 'Install failed'
return { success: false, error: msg }
}
return checkBinary(binary)
? { success: true }
: { success: false, error: 'Install completed but binary not found in PATH' }
}
async function tryCopilotInstall(binary: string): Promise<InstallResult> {
// Try brew on macOS
if (process.platform === 'darwin' && hasCommand('brew')) {
try {
execSync('brew install github/copilot/copilot', {
encoding: 'utf-8',
timeout: 180_000,
stdio: 'pipe',
})
if (checkBinary(binary)) return { success: true }
} catch {
// Fall through to failure
}
}
return { success: false, error: 'Auto-install not available for this platform' }
}