- Split 25+ index.ts files into hook.ts + extracted modules - Rename all catch-all utils.ts/helpers.ts to domain-specific names - Split src/tools/lsp/ into ~15 focused modules - Split src/tools/delegate-task/ into ~18 focused modules - Separate shared types from implementation - 155 files changed, 60+ new files created - All typecheck clean, 61 tests pass
146 lines
4.2 KiB
TypeScript
146 lines
4.2 KiB
TypeScript
import {
|
|
detectSlashCommand,
|
|
extractPromptText,
|
|
findSlashCommandPartIndex,
|
|
} from "./detector"
|
|
import { executeSlashCommand, type ExecutorOptions } from "./executor"
|
|
import { log } from "../../shared"
|
|
import {
|
|
AUTO_SLASH_COMMAND_TAG_CLOSE,
|
|
AUTO_SLASH_COMMAND_TAG_OPEN,
|
|
} from "./constants"
|
|
import type {
|
|
AutoSlashCommandHookInput,
|
|
AutoSlashCommandHookOutput,
|
|
CommandExecuteBeforeInput,
|
|
CommandExecuteBeforeOutput,
|
|
} from "./types"
|
|
import type { LoadedSkill } from "../../features/opencode-skill-loader"
|
|
|
|
const sessionProcessedCommands = new Set<string>()
|
|
const sessionProcessedCommandExecutions = new Set<string>()
|
|
|
|
export interface AutoSlashCommandHookOptions {
|
|
skills?: LoadedSkill[]
|
|
}
|
|
|
|
export function createAutoSlashCommandHook(options?: AutoSlashCommandHookOptions) {
|
|
const executorOptions: ExecutorOptions = {
|
|
skills: options?.skills,
|
|
}
|
|
|
|
return {
|
|
"chat.message": async (
|
|
input: AutoSlashCommandHookInput,
|
|
output: AutoSlashCommandHookOutput
|
|
): Promise<void> => {
|
|
const promptText = extractPromptText(output.parts)
|
|
|
|
// Debug logging to diagnose slash command issues
|
|
if (promptText.startsWith("/")) {
|
|
log(`[auto-slash-command] chat.message hook received slash command`, {
|
|
sessionID: input.sessionID,
|
|
promptText: promptText.slice(0, 100),
|
|
})
|
|
}
|
|
|
|
if (
|
|
promptText.includes(AUTO_SLASH_COMMAND_TAG_OPEN) ||
|
|
promptText.includes(AUTO_SLASH_COMMAND_TAG_CLOSE)
|
|
) {
|
|
return
|
|
}
|
|
|
|
const parsed = detectSlashCommand(promptText)
|
|
|
|
if (!parsed) {
|
|
return
|
|
}
|
|
|
|
const commandKey = `${input.sessionID}:${input.messageID}:${parsed.command}`
|
|
if (sessionProcessedCommands.has(commandKey)) {
|
|
return
|
|
}
|
|
sessionProcessedCommands.add(commandKey)
|
|
|
|
log(`[auto-slash-command] Detected: /${parsed.command}`, {
|
|
sessionID: input.sessionID,
|
|
args: parsed.args,
|
|
})
|
|
|
|
const result = await executeSlashCommand(parsed, executorOptions)
|
|
|
|
const idx = findSlashCommandPartIndex(output.parts)
|
|
if (idx < 0) {
|
|
return
|
|
}
|
|
|
|
if (!result.success || !result.replacementText) {
|
|
log(`[auto-slash-command] Command not found, skipping`, {
|
|
sessionID: input.sessionID,
|
|
command: parsed.command,
|
|
error: result.error,
|
|
})
|
|
return
|
|
}
|
|
|
|
const taggedContent = `${AUTO_SLASH_COMMAND_TAG_OPEN}\n${result.replacementText}\n${AUTO_SLASH_COMMAND_TAG_CLOSE}`
|
|
output.parts[idx].text = taggedContent
|
|
|
|
log(`[auto-slash-command] Replaced message with command template`, {
|
|
sessionID: input.sessionID,
|
|
command: parsed.command,
|
|
})
|
|
},
|
|
|
|
"command.execute.before": async (
|
|
input: CommandExecuteBeforeInput,
|
|
output: CommandExecuteBeforeOutput
|
|
): Promise<void> => {
|
|
const commandKey = `${input.sessionID}:${input.command}:${Date.now()}`
|
|
if (sessionProcessedCommandExecutions.has(commandKey)) {
|
|
return
|
|
}
|
|
|
|
log(`[auto-slash-command] command.execute.before received`, {
|
|
sessionID: input.sessionID,
|
|
command: input.command,
|
|
arguments: input.arguments,
|
|
})
|
|
|
|
const parsed = {
|
|
command: input.command,
|
|
args: input.arguments || "",
|
|
raw: `/${input.command}${input.arguments ? " " + input.arguments : ""}`,
|
|
}
|
|
|
|
const result = await executeSlashCommand(parsed, executorOptions)
|
|
|
|
if (!result.success || !result.replacementText) {
|
|
log(`[auto-slash-command] command.execute.before - command not found in our executor`, {
|
|
sessionID: input.sessionID,
|
|
command: input.command,
|
|
error: result.error,
|
|
})
|
|
return
|
|
}
|
|
|
|
sessionProcessedCommandExecutions.add(commandKey)
|
|
|
|
const taggedContent = `${AUTO_SLASH_COMMAND_TAG_OPEN}\n${result.replacementText}\n${AUTO_SLASH_COMMAND_TAG_CLOSE}`
|
|
|
|
const idx = findSlashCommandPartIndex(output.parts)
|
|
if (idx >= 0) {
|
|
output.parts[idx].text = taggedContent
|
|
} else {
|
|
output.parts.unshift({ type: "text", text: taggedContent })
|
|
}
|
|
|
|
log(`[auto-slash-command] command.execute.before - injected template`, {
|
|
sessionID: input.sessionID,
|
|
command: input.command,
|
|
})
|
|
},
|
|
}
|
|
}
|