import { describe, expect, it, beforeEach, mock, spyOn } from "bun:test" import type { AutoSlashCommandHookInput, AutoSlashCommandHookOutput, } 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( "/commit" ) 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 "') 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) }) }) })