- Use os.tmpdir() instead of hardcoded /tmp for cross-platform temp files - Use os.homedir() with USERPROFILE fallback for Windows home directory - Disable forceZsh on Windows (zsh not available by default) 🤖 GENERATED WITH ASSISTANCE OF [OhMyOpenCode](https://github.com/code-yeongyu/oh-my-opencode)
209 lines
4.6 KiB
TypeScript
209 lines
4.6 KiB
TypeScript
import { spawn } from "child_process"
|
|
import { exec } from "child_process"
|
|
import { promisify } from "util"
|
|
import { existsSync } from "fs"
|
|
import { homedir } from "os"
|
|
|
|
const DEFAULT_ZSH_PATHS = ["/bin/zsh", "/usr/bin/zsh", "/usr/local/bin/zsh"]
|
|
|
|
function getHomeDir(): string {
|
|
return process.env.HOME || process.env.USERPROFILE || homedir()
|
|
}
|
|
|
|
function findZshPath(customZshPath?: string): string | null {
|
|
if (customZshPath && existsSync(customZshPath)) {
|
|
return customZshPath
|
|
}
|
|
for (const path of DEFAULT_ZSH_PATHS) {
|
|
if (existsSync(path)) {
|
|
return path
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
|
|
const execAsync = promisify(exec)
|
|
|
|
export interface CommandResult {
|
|
exitCode: number
|
|
stdout?: string
|
|
stderr?: string
|
|
}
|
|
|
|
export interface ExecuteHookOptions {
|
|
forceZsh?: boolean
|
|
zshPath?: string
|
|
}
|
|
|
|
/**
|
|
* Execute a hook command with stdin input
|
|
*/
|
|
export async function executeHookCommand(
|
|
command: string,
|
|
stdin: string,
|
|
cwd: string,
|
|
options?: ExecuteHookOptions
|
|
): Promise<CommandResult> {
|
|
const home = getHomeDir()
|
|
|
|
let expandedCommand = command
|
|
.replace(/^~(?=\/|$)/g, home)
|
|
.replace(/\s~(?=\/)/g, ` ${home}`)
|
|
.replace(/\$CLAUDE_PROJECT_DIR/g, cwd)
|
|
.replace(/\$\{CLAUDE_PROJECT_DIR\}/g, cwd)
|
|
|
|
let finalCommand = expandedCommand
|
|
|
|
if (options?.forceZsh) {
|
|
const zshPath = options.zshPath || findZshPath()
|
|
if (zshPath) {
|
|
const escapedCommand = expandedCommand.replace(/'/g, "'\\''")
|
|
finalCommand = `${zshPath} -lc '${escapedCommand}'`
|
|
}
|
|
}
|
|
|
|
return new Promise((resolve) => {
|
|
const proc = spawn(finalCommand, {
|
|
cwd,
|
|
shell: true,
|
|
env: { ...process.env, HOME: home, CLAUDE_PROJECT_DIR: cwd },
|
|
})
|
|
|
|
let stdout = ""
|
|
let stderr = ""
|
|
|
|
proc.stdout?.on("data", (data) => {
|
|
stdout += data.toString()
|
|
})
|
|
|
|
proc.stderr?.on("data", (data) => {
|
|
stderr += data.toString()
|
|
})
|
|
|
|
proc.stdin?.write(stdin)
|
|
proc.stdin?.end()
|
|
|
|
proc.on("close", (code) => {
|
|
resolve({
|
|
exitCode: code ?? 0,
|
|
stdout: stdout.trim(),
|
|
stderr: stderr.trim(),
|
|
})
|
|
})
|
|
|
|
proc.on("error", (err) => {
|
|
resolve({
|
|
exitCode: 1,
|
|
stderr: err.message,
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Execute a simple command and return output
|
|
*/
|
|
export async function executeCommand(command: string): Promise<string> {
|
|
try {
|
|
const { stdout, stderr } = await execAsync(command)
|
|
|
|
const out = stdout?.toString().trim() ?? ""
|
|
const err = stderr?.toString().trim() ?? ""
|
|
|
|
if (err) {
|
|
if (out) {
|
|
return `${out}\n[stderr: ${err}]`
|
|
}
|
|
return `[stderr: ${err}]`
|
|
}
|
|
|
|
return out
|
|
} catch (error: unknown) {
|
|
const e = error as { stdout?: Buffer; stderr?: Buffer; message?: string }
|
|
const stdout = e?.stdout?.toString().trim() ?? ""
|
|
const stderr = e?.stderr?.toString().trim() ?? ""
|
|
const errMsg = stderr || e?.message || String(error)
|
|
|
|
if (stdout) {
|
|
return `${stdout}\n[stderr: ${errMsg}]`
|
|
}
|
|
return `[stderr: ${errMsg}]`
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Find and execute embedded commands in text (!`command`)
|
|
*/
|
|
interface CommandMatch {
|
|
fullMatch: string
|
|
command: string
|
|
start: number
|
|
end: number
|
|
}
|
|
|
|
const COMMAND_PATTERN = /!`([^`]+)`/g
|
|
|
|
function findCommands(text: string): CommandMatch[] {
|
|
const matches: CommandMatch[] = []
|
|
let match: RegExpExecArray | null
|
|
|
|
COMMAND_PATTERN.lastIndex = 0
|
|
|
|
while ((match = COMMAND_PATTERN.exec(text)) !== null) {
|
|
matches.push({
|
|
fullMatch: match[0],
|
|
command: match[1],
|
|
start: match.index,
|
|
end: match.index + match[0].length,
|
|
})
|
|
}
|
|
|
|
return matches
|
|
}
|
|
|
|
/**
|
|
* Resolve embedded commands in text recursively
|
|
*/
|
|
export async function resolveCommandsInText(
|
|
text: string,
|
|
depth: number = 0,
|
|
maxDepth: number = 3
|
|
): Promise<string> {
|
|
if (depth >= maxDepth) {
|
|
return text
|
|
}
|
|
|
|
const matches = findCommands(text)
|
|
if (matches.length === 0) {
|
|
return text
|
|
}
|
|
|
|
const tasks = matches.map((m) => executeCommand(m.command))
|
|
const results = await Promise.allSettled(tasks)
|
|
|
|
const replacements = new Map<string, string>()
|
|
|
|
matches.forEach((match, idx) => {
|
|
const result = results[idx]
|
|
if (result.status === "rejected") {
|
|
replacements.set(
|
|
match.fullMatch,
|
|
`[error: ${result.reason instanceof Error ? result.reason.message : String(result.reason)}]`
|
|
)
|
|
} else {
|
|
replacements.set(match.fullMatch, result.value)
|
|
}
|
|
})
|
|
|
|
let resolved = text
|
|
for (const [pattern, replacement] of replacements.entries()) {
|
|
resolved = resolved.split(pattern).join(replacement)
|
|
}
|
|
|
|
if (findCommands(resolved).length > 0) {
|
|
return resolveCommandsInText(resolved, depth + 1, maxDepth)
|
|
}
|
|
|
|
return resolved
|
|
}
|