🤖 GENERATED WITH ASSISTANCE OF [OhMyOpenCode](https://github.com/code-yeongyu/oh-my-opencode)
173 lines
5.5 KiB
TypeScript
173 lines
5.5 KiB
TypeScript
import type {
|
|
PreToolUseInput,
|
|
PreToolUseOutput,
|
|
PermissionDecision,
|
|
ClaudeHooksConfig,
|
|
} from "./types"
|
|
import { findMatchingHooks, executeHookCommand, objectToSnakeCase, transformToolName, log } from "../../shared"
|
|
import { DEFAULT_CONFIG } from "./plugin-config"
|
|
import { isHookCommandDisabled, type PluginExtendedConfig } from "./config-loader"
|
|
|
|
export interface PreToolUseContext {
|
|
sessionId: string
|
|
toolName: string
|
|
toolInput: Record<string, unknown>
|
|
cwd: string
|
|
transcriptPath?: string
|
|
toolUseId?: string
|
|
permissionMode?: "default" | "plan" | "acceptEdits" | "bypassPermissions"
|
|
}
|
|
|
|
export interface PreToolUseResult {
|
|
decision: PermissionDecision
|
|
reason?: string
|
|
modifiedInput?: Record<string, unknown>
|
|
elapsedMs?: number
|
|
hookName?: string
|
|
toolName?: string
|
|
inputLines?: string
|
|
// Common output fields (Claude Code spec)
|
|
continue?: boolean
|
|
stopReason?: string
|
|
suppressOutput?: boolean
|
|
systemMessage?: string
|
|
}
|
|
|
|
function buildInputLines(toolInput: Record<string, unknown>): string {
|
|
return Object.entries(toolInput)
|
|
.slice(0, 3)
|
|
.map(([key, val]) => {
|
|
const valStr = String(val).slice(0, 40)
|
|
return ` ${key}: ${valStr}${String(val).length > 40 ? "..." : ""}`
|
|
})
|
|
.join("\n")
|
|
}
|
|
|
|
export async function executePreToolUseHooks(
|
|
ctx: PreToolUseContext,
|
|
config: ClaudeHooksConfig | null,
|
|
extendedConfig?: PluginExtendedConfig | null
|
|
): Promise<PreToolUseResult> {
|
|
if (!config) {
|
|
return { decision: "allow" }
|
|
}
|
|
|
|
const transformedToolName = transformToolName(ctx.toolName)
|
|
const matchers = findMatchingHooks(config, "PreToolUse", transformedToolName)
|
|
if (matchers.length === 0) {
|
|
return { decision: "allow" }
|
|
}
|
|
|
|
const stdinData: PreToolUseInput = {
|
|
session_id: ctx.sessionId,
|
|
transcript_path: ctx.transcriptPath,
|
|
cwd: ctx.cwd,
|
|
permission_mode: ctx.permissionMode ?? "bypassPermissions",
|
|
hook_event_name: "PreToolUse",
|
|
tool_name: transformedToolName,
|
|
tool_input: objectToSnakeCase(ctx.toolInput),
|
|
tool_use_id: ctx.toolUseId,
|
|
hook_source: "opencode-plugin",
|
|
}
|
|
|
|
const startTime = Date.now()
|
|
let firstHookName: string | undefined
|
|
const inputLines = buildInputLines(ctx.toolInput)
|
|
|
|
for (const matcher of matchers) {
|
|
for (const hook of matcher.hooks) {
|
|
if (hook.type !== "command") continue
|
|
|
|
if (isHookCommandDisabled("PreToolUse", hook.command, extendedConfig ?? null)) {
|
|
log("PreToolUse hook command skipped (disabled by config)", { command: hook.command, toolName: ctx.toolName })
|
|
continue
|
|
}
|
|
|
|
const hookName = hook.command.split("/").pop() || hook.command
|
|
if (!firstHookName) firstHookName = hookName
|
|
|
|
const result = await executeHookCommand(
|
|
hook.command,
|
|
JSON.stringify(stdinData),
|
|
ctx.cwd,
|
|
{ forceZsh: DEFAULT_CONFIG.forceZsh, zshPath: DEFAULT_CONFIG.zshPath }
|
|
)
|
|
|
|
if (result.exitCode === 2) {
|
|
return {
|
|
decision: "deny",
|
|
reason: result.stderr || result.stdout || "Hook blocked the operation",
|
|
elapsedMs: Date.now() - startTime,
|
|
hookName: firstHookName,
|
|
toolName: transformedToolName,
|
|
inputLines,
|
|
}
|
|
}
|
|
|
|
if (result.exitCode === 1) {
|
|
return {
|
|
decision: "ask",
|
|
reason: result.stderr || result.stdout,
|
|
elapsedMs: Date.now() - startTime,
|
|
hookName: firstHookName,
|
|
toolName: transformedToolName,
|
|
inputLines,
|
|
}
|
|
}
|
|
|
|
if (result.stdout) {
|
|
try {
|
|
const output = JSON.parse(result.stdout) as PreToolUseOutput
|
|
|
|
// Handle deprecated decision/reason fields (Claude Code backward compat)
|
|
let decision: PermissionDecision | undefined
|
|
let reason: string | undefined
|
|
let modifiedInput: Record<string, unknown> | undefined
|
|
|
|
if (output.hookSpecificOutput?.permissionDecision) {
|
|
decision = output.hookSpecificOutput.permissionDecision
|
|
reason = output.hookSpecificOutput.permissionDecisionReason
|
|
modifiedInput = output.hookSpecificOutput.updatedInput
|
|
} else if (output.decision) {
|
|
// Map deprecated values: approve->allow, block->deny, ask->ask
|
|
const legacyDecision = output.decision
|
|
if (legacyDecision === "approve" || legacyDecision === "allow") {
|
|
decision = "allow"
|
|
} else if (legacyDecision === "block" || legacyDecision === "deny") {
|
|
decision = "deny"
|
|
} else if (legacyDecision === "ask") {
|
|
decision = "ask"
|
|
}
|
|
reason = output.reason
|
|
}
|
|
|
|
// Return if decision is set OR if any common fields are set (fallback to allow)
|
|
const hasCommonFields = output.continue !== undefined ||
|
|
output.stopReason !== undefined ||
|
|
output.suppressOutput !== undefined ||
|
|
output.systemMessage !== undefined
|
|
|
|
if (decision || hasCommonFields) {
|
|
return {
|
|
decision: decision ?? "allow",
|
|
reason,
|
|
modifiedInput,
|
|
elapsedMs: Date.now() - startTime,
|
|
hookName: firstHookName,
|
|
toolName: transformedToolName,
|
|
inputLines,
|
|
continue: output.continue,
|
|
stopReason: output.stopReason,
|
|
suppressOutput: output.suppressOutput,
|
|
systemMessage: output.systemMessage,
|
|
}
|
|
}
|
|
} catch {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return { decision: "allow" }
|
|
}
|