- Add builtin commands to command discovery with 'builtin' scope - Improve extractPromptText to prioritize slash command parts - Add findSlashCommandPartIndex helper for locating slash commands - Add CommandExecuteBefore hook support
333 lines
11 KiB
TypeScript
333 lines
11 KiB
TypeScript
import { describe, expect, it, beforeEach, mock, spyOn } from "bun:test"
|
|
import type {
|
|
AutoSlashCommandHookInput,
|
|
AutoSlashCommandHookOutput,
|
|
CommandExecuteBeforeInput,
|
|
CommandExecuteBeforeOutput,
|
|
} from "./types"
|
|
|
|
// Import real shared module to avoid mock leaking to other test files
|
|
import * as shared from "../../shared"
|
|
|
|
// Spy on log instead of mocking the entire module
|
|
const logMock = spyOn(shared, "log").mockImplementation(() => {})
|
|
|
|
|
|
|
|
const { createAutoSlashCommandHook } = await import("./index")
|
|
|
|
function createMockInput(sessionID: string, messageID?: string): AutoSlashCommandHookInput {
|
|
return {
|
|
sessionID,
|
|
messageID: messageID ?? `msg-${Date.now()}-${Math.random()}`,
|
|
agent: "test-agent",
|
|
model: { providerID: "anthropic", modelID: "claude-sonnet-4-5" },
|
|
}
|
|
}
|
|
|
|
function createMockOutput(text: string): AutoSlashCommandHookOutput {
|
|
return {
|
|
message: {
|
|
agent: "test-agent",
|
|
model: { providerID: "anthropic", modelID: "claude-sonnet-4-5" },
|
|
path: { cwd: "/test", root: "/test" },
|
|
tools: {},
|
|
},
|
|
parts: [{ type: "text", text }],
|
|
}
|
|
}
|
|
|
|
describe("createAutoSlashCommandHook", () => {
|
|
beforeEach(() => {
|
|
logMock.mockClear()
|
|
})
|
|
|
|
describe("slash command replacement", () => {
|
|
it("should not modify message when command not found", async () => {
|
|
// given a slash command that doesn't exist
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-notfound-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("/nonexistent-command args")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should NOT modify the message (feature inactive when command not found)
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
|
|
it("should not modify message for unknown command (feature inactive)", async () => {
|
|
// given unknown slash command
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-tags-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("/some-command")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should NOT modify (command not found = feature inactive)
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
|
|
it("should not modify for unknown command (no prepending)", async () => {
|
|
// given unknown slash command
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-replace-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("/test-cmd some args")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify (feature inactive for unknown commands)
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
})
|
|
|
|
describe("no slash command", () => {
|
|
it("should do nothing for regular text", async () => {
|
|
// given regular text without slash
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-regular-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("Just regular text")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
|
|
it("should do nothing for slash in middle of text", async () => {
|
|
// given slash in middle
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-middle-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("Please run /commit later")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not detect (not at start)
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
})
|
|
|
|
describe("excluded commands", () => {
|
|
it("should NOT trigger for ralph-loop command", async () => {
|
|
// given ralph-loop command
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-ralph-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("/ralph-loop do something")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify (excluded command)
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
|
|
it("should NOT trigger for cancel-ralph command", async () => {
|
|
// given cancel-ralph command
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-cancel-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("/cancel-ralph")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
})
|
|
|
|
describe("already processed", () => {
|
|
it("should skip if auto-slash-command tags already present", async () => {
|
|
// given text with existing tags
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-existing-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput(
|
|
"<auto-slash-command>/commit</auto-slash-command>"
|
|
)
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
})
|
|
|
|
describe("code blocks", () => {
|
|
it("should NOT detect command inside code block", async () => {
|
|
// given command inside code block
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-codeblock-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("```\n/commit\n```")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not detect
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
})
|
|
|
|
describe("edge cases", () => {
|
|
it("should handle empty text", async () => {
|
|
// given empty text
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-empty-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("")
|
|
|
|
// when hook is called
|
|
// then should not throw
|
|
await expect(hook["chat.message"](input, output)).resolves.toBeUndefined()
|
|
})
|
|
|
|
it("should handle just slash", async () => {
|
|
// given just slash
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-slash-only-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput("/")
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
|
|
it("should handle command with special characters in args (not found = no modification)", async () => {
|
|
// given command with special characters that doesn't exist
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-special-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output = createMockOutput('/execute "test & stuff <tag>"')
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify (command not found = feature inactive)
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
|
|
it("should handle multiple text parts (unknown command = no modification)", async () => {
|
|
// given multiple text parts with unknown command
|
|
const hook = createAutoSlashCommandHook()
|
|
const sessionID = `test-session-multi-${Date.now()}`
|
|
const input = createMockInput(sessionID)
|
|
const output: AutoSlashCommandHookOutput = {
|
|
message: {},
|
|
parts: [
|
|
{ type: "text", text: "/truly-nonexistent-xyz-cmd " },
|
|
{ type: "text", text: "some args" },
|
|
],
|
|
}
|
|
const originalText = output.parts[0].text
|
|
|
|
// when hook is called
|
|
await hook["chat.message"](input, output)
|
|
|
|
// then should not modify (command not found = feature inactive)
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
})
|
|
|
|
describe("command.execute.before hook", () => {
|
|
function createCommandInput(command: string, args: string = ""): CommandExecuteBeforeInput {
|
|
return {
|
|
command,
|
|
sessionID: `test-session-cmd-${Date.now()}-${Math.random()}`,
|
|
arguments: args,
|
|
}
|
|
}
|
|
|
|
function createCommandOutput(text?: string): CommandExecuteBeforeOutput {
|
|
return {
|
|
parts: text ? [{ type: "text", text }] : [],
|
|
}
|
|
}
|
|
|
|
it("should not modify output for unknown command", async () => {
|
|
//#given
|
|
const hook = createAutoSlashCommandHook()
|
|
const input = createCommandInput("nonexistent-command-xyz")
|
|
const output = createCommandOutput("original text")
|
|
const originalText = output.parts[0].text
|
|
|
|
//#when
|
|
await hook["command.execute.before"](input, output)
|
|
|
|
//#then
|
|
expect(output.parts[0].text).toBe(originalText)
|
|
})
|
|
|
|
it("should add text part when parts array is empty and command is unknown", async () => {
|
|
//#given
|
|
const hook = createAutoSlashCommandHook()
|
|
const input = createCommandInput("nonexistent-command-abc")
|
|
const output = createCommandOutput()
|
|
|
|
//#when
|
|
await hook["command.execute.before"](input, output)
|
|
|
|
//#then
|
|
expect(output.parts.length).toBe(0)
|
|
})
|
|
|
|
it("should inject template for known builtin commands like ralph-loop", async () => {
|
|
//#given
|
|
const hook = createAutoSlashCommandHook()
|
|
const input = createCommandInput("ralph-loop")
|
|
const output = createCommandOutput("original")
|
|
|
|
//#when
|
|
await hook["command.execute.before"](input, output)
|
|
|
|
//#then
|
|
expect(output.parts[0].text).toContain("<auto-slash-command>")
|
|
expect(output.parts[0].text).toContain("/ralph-loop Command")
|
|
})
|
|
|
|
it("should pass command arguments correctly", async () => {
|
|
//#given
|
|
const hook = createAutoSlashCommandHook()
|
|
const input = createCommandInput("some-command", "arg1 arg2 arg3")
|
|
const output = createCommandOutput("original")
|
|
|
|
//#when
|
|
await hook["command.execute.before"](input, output)
|
|
|
|
//#then
|
|
expect(logMock).toHaveBeenCalledWith(
|
|
"[auto-slash-command] command.execute.before received",
|
|
expect.objectContaining({
|
|
command: "some-command",
|
|
arguments: "arg1 arg2 arg3",
|
|
})
|
|
)
|
|
})
|
|
})
|
|
})
|