Files
oh-my-openagent/src/agents/sisyphus-junior/index.test.ts
YeonGyu-Kim fecc488848 fix(sisyphus-junior): disambiguate blocked delegation tool from allowed task management tools
When task_system is enabled, the prompt said 'task tool: BLOCKED' which
LLMs interpreted as blocking task_create/task_update/task_list/task_get
too. Now the constraints section explicitly separates 'task (agent
delegation tool): BLOCKED' from 'task_create, task_update, ...: ALLOWED'
so Junior no longer refuses to use task management tools.
2026-02-10 11:41:44 +09:00

437 lines
13 KiB
TypeScript

import { describe, expect, test } from "bun:test"
import {
createSisyphusJuniorAgentWithOverrides,
SISYPHUS_JUNIOR_DEFAULTS,
getSisyphusJuniorPromptSource,
buildSisyphusJuniorPrompt,
} from "./index"
describe("createSisyphusJuniorAgentWithOverrides", () => {
describe("honored fields", () => {
test("applies model override", () => {
// given
const override = { model: "openai/gpt-5.2" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.model).toBe("openai/gpt-5.2")
})
test("applies temperature override", () => {
// given
const override = { temperature: 0.5 }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.temperature).toBe(0.5)
})
test("applies top_p override", () => {
// given
const override = { top_p: 0.9 }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.top_p).toBe(0.9)
})
test("applies description override", () => {
// given
const override = { description: "Custom description" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.description).toBe("Custom description")
})
test("applies color override", () => {
// given
const override = { color: "#FF0000" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.color).toBe("#FF0000")
})
test("appends prompt_append to base prompt", () => {
// given
const override = { prompt_append: "Extra instructions here" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.prompt).toContain("You work ALONE")
expect(result.prompt).toContain("Extra instructions here")
})
})
describe("defaults", () => {
test("uses default model when no override", () => {
// given
const override = {}
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.model).toBe(SISYPHUS_JUNIOR_DEFAULTS.model)
})
test("uses default temperature when no override", () => {
// given
const override = {}
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.temperature).toBe(SISYPHUS_JUNIOR_DEFAULTS.temperature)
})
})
describe("disable semantics", () => {
test("disable: true causes override block to be ignored", () => {
// given
const override = {
disable: true,
model: "openai/gpt-5.2",
temperature: 0.9,
}
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then - defaults should be used, not the overrides
expect(result.model).toBe(SISYPHUS_JUNIOR_DEFAULTS.model)
expect(result.temperature).toBe(SISYPHUS_JUNIOR_DEFAULTS.temperature)
})
})
describe("constrained fields", () => {
test("mode is forced to subagent", () => {
// given
const override = { mode: "primary" as const }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.mode).toBe("subagent")
})
test("prompt override is ignored (discipline text preserved)", () => {
// given
const override = { prompt: "Completely new prompt that replaces everything" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.prompt).toContain("You work ALONE")
expect(result.prompt).not.toBe("Completely new prompt that replaces everything")
})
})
describe("tool safety (task blocked, call_omo_agent allowed)", () => {
test("task remains blocked, call_omo_agent is allowed via tools format", () => {
// given
const override = {
tools: {
task: true,
call_omo_agent: true,
read: true,
},
}
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
const tools = result.tools as Record<string, boolean> | undefined
const permission = result.permission as Record<string, string> | undefined
if (tools) {
expect(tools.task).toBe(false)
// call_omo_agent is NOW ALLOWED for subagents to spawn explore/librarian
expect(tools.call_omo_agent).toBe(true)
expect(tools.read).toBe(true)
}
if (permission) {
expect(permission.task).toBe("deny")
// call_omo_agent is NOW ALLOWED for subagents to spawn explore/librarian
expect(permission.call_omo_agent).toBe("allow")
}
})
test("task remains blocked when using permission format override", () => {
// given
const override = {
permission: {
task: "allow",
call_omo_agent: "allow",
read: "allow",
},
} as { permission: Record<string, string> }
// when
const result = createSisyphusJuniorAgentWithOverrides(override as Parameters<typeof createSisyphusJuniorAgentWithOverrides>[0])
// then - task blocked, but call_omo_agent allowed for explore/librarian spawning
const tools = result.tools as Record<string, boolean> | undefined
const permission = result.permission as Record<string, string> | undefined
if (tools) {
expect(tools.task).toBe(false)
expect(tools.call_omo_agent).toBe(true)
}
if (permission) {
expect(permission.task).toBe("deny")
expect(permission.call_omo_agent).toBe("allow")
}
})
})
describe("useTaskSystem integration", () => {
test("useTaskSystem=true produces Task_Discipline prompt for Claude", () => {
//#given
const override = { model: "anthropic/claude-sonnet-4-5" }
//#when
const result = createSisyphusJuniorAgentWithOverrides(override, undefined, true)
//#then
expect(result.prompt).toContain("TaskCreate")
expect(result.prompt).toContain("TaskUpdate")
expect(result.prompt).not.toContain("todowrite")
})
test("useTaskSystem=true produces task_discipline_spec prompt for GPT", () => {
//#given
const override = { model: "openai/gpt-5.2" }
//#when
const result = createSisyphusJuniorAgentWithOverrides(override, undefined, true)
//#then
expect(result.prompt).toContain("<task_discipline_spec>")
expect(result.prompt).toContain("TaskCreate")
expect(result.prompt).not.toContain("<todo_discipline_spec>")
})
test("useTaskSystem=false (default) produces Todo_Discipline prompt", () => {
//#given
const override = {}
//#when
const result = createSisyphusJuniorAgentWithOverrides(override)
//#then
expect(result.prompt).toContain("todowrite")
expect(result.prompt).not.toContain("TaskCreate")
})
test("useTaskSystem=true explicitly lists task management tools as ALLOWED for Claude", () => {
//#given
const override = { model: "anthropic/claude-sonnet-4-5" }
//#when
const result = createSisyphusJuniorAgentWithOverrides(override, undefined, true)
//#then - prompt must disambiguate: delegation tool blocked, management tools allowed
expect(result.prompt).toContain("task_create")
expect(result.prompt).toContain("task_update")
expect(result.prompt).toContain("task_list")
expect(result.prompt).toContain("task_get")
expect(result.prompt).toContain("agent delegation tool")
})
test("useTaskSystem=true explicitly lists task management tools as ALLOWED for GPT", () => {
//#given
const override = { model: "openai/gpt-5.2" }
//#when
const result = createSisyphusJuniorAgentWithOverrides(override, undefined, true)
//#then - prompt must disambiguate: delegation tool blocked, management tools allowed
expect(result.prompt).toContain("task_create")
expect(result.prompt).toContain("task_update")
expect(result.prompt).toContain("task_list")
expect(result.prompt).toContain("task_get")
expect(result.prompt).toContain("Agent delegation tool")
})
test("useTaskSystem=false does NOT list task management tools in constraints", () => {
//#given - Claude model without task system
const override = { model: "anthropic/claude-sonnet-4-5" }
//#when
const result = createSisyphusJuniorAgentWithOverrides(override, undefined, false)
//#then - no task management tool references in constraints section
expect(result.prompt).not.toContain("task_create")
expect(result.prompt).not.toContain("task_update")
})
})
describe("prompt composition", () => {
test("base prompt contains discipline constraints", () => {
// given
const override = {}
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.prompt).toContain("Sisyphus-Junior")
expect(result.prompt).toContain("You work ALONE")
})
test("Claude model uses default prompt with BLOCKED ACTIONS section", () => {
// given
const override = { model: "anthropic/claude-sonnet-4-5" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.prompt).toContain("BLOCKED ACTIONS")
expect(result.prompt).not.toContain("<blocked_actions>")
})
test("GPT model uses GPT-optimized prompt with blocked_actions section", () => {
// given
const override = { model: "openai/gpt-5.2" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
expect(result.prompt).toContain("<blocked_actions>")
expect(result.prompt).toContain("<output_verbosity_spec>")
expect(result.prompt).toContain("<scope_and_design_constraints>")
})
test("prompt_append is added after base prompt", () => {
// given
const override = { prompt_append: "CUSTOM_MARKER_FOR_TEST" }
// when
const result = createSisyphusJuniorAgentWithOverrides(override)
// then
const baseEndIndex = result.prompt!.indexOf("Dense > verbose.")
const appendIndex = result.prompt!.indexOf("CUSTOM_MARKER_FOR_TEST")
expect(baseEndIndex).not.toBe(-1)
expect(appendIndex).toBeGreaterThan(baseEndIndex)
})
})
})
describe("getSisyphusJuniorPromptSource", () => {
test("returns 'gpt' for OpenAI models", () => {
// given
const model = "openai/gpt-5.2"
// when
const source = getSisyphusJuniorPromptSource(model)
// then
expect(source).toBe("gpt")
})
test("returns 'gpt' for GitHub Copilot GPT models", () => {
// given
const model = "github-copilot/gpt-4o"
// when
const source = getSisyphusJuniorPromptSource(model)
// then
expect(source).toBe("gpt")
})
test("returns 'default' for Claude models", () => {
// given
const model = "anthropic/claude-sonnet-4-5"
// when
const source = getSisyphusJuniorPromptSource(model)
// then
expect(source).toBe("default")
})
test("returns 'default' for undefined model", () => {
// given
const model = undefined
// when
const source = getSisyphusJuniorPromptSource(model)
// then
expect(source).toBe("default")
})
})
describe("buildSisyphusJuniorPrompt", () => {
test("GPT model prompt contains GPT-5.2 specific sections", () => {
// given
const model = "openai/gpt-5.2"
// when
const prompt = buildSisyphusJuniorPrompt(model, false)
// then
expect(prompt).toContain("<identity>")
expect(prompt).toContain("<output_verbosity_spec>")
expect(prompt).toContain("<scope_and_design_constraints>")
expect(prompt).toContain("<tool_usage_rules>")
})
test("Claude model prompt contains Claude-specific sections", () => {
// given
const model = "anthropic/claude-sonnet-4-5"
// when
const prompt = buildSisyphusJuniorPrompt(model, false)
// then
expect(prompt).toContain("<Role>")
expect(prompt).toContain("<Critical_Constraints>")
expect(prompt).toContain("BLOCKED ACTIONS")
})
test("useTaskSystem=true includes Task_Discipline for GPT", () => {
// given
const model = "openai/gpt-5.2"
// when
const prompt = buildSisyphusJuniorPrompt(model, true)
// then
expect(prompt).toContain("<task_discipline_spec>")
expect(prompt).toContain("TaskCreate")
})
test("useTaskSystem=false includes Todo_Discipline for Claude", () => {
// given
const model = "anthropic/claude-sonnet-4-5"
// when
const prompt = buildSisyphusJuniorPrompt(model, false)
// then
expect(prompt).toContain("<Todo_Discipline>")
expect(prompt).toContain("todowrite")
})
})