- Add thinking_max_chars?: number to BackgroundOutputOptions type - Add thinking_max_chars argument to background_output tool schema - Add formatFullSession option for controlling output format - Add 2 tests for thinking_max_chars functionality
266 lines
7.4 KiB
TypeScript
266 lines
7.4 KiB
TypeScript
import { createBackgroundOutput } from "./tools"
|
|
import type { BackgroundTask } from "../../features/background-agent"
|
|
import type { ToolContext } from "@opencode-ai/plugin/tool"
|
|
import type { BackgroundOutputManager, BackgroundOutputClient } from "./tools"
|
|
|
|
const projectDir = "/Users/yeongyu/local-workspaces/oh-my-opencode"
|
|
|
|
const mockContext: ToolContext = {
|
|
sessionID: "test-session",
|
|
messageID: "test-message",
|
|
agent: "test-agent",
|
|
directory: projectDir,
|
|
worktree: projectDir,
|
|
abort: new AbortController().signal,
|
|
metadata: () => {},
|
|
ask: async () => {},
|
|
}
|
|
|
|
function createMockManager(task: BackgroundTask): BackgroundOutputManager {
|
|
return {
|
|
getTask: (id: string) => (id === task.id ? task : undefined),
|
|
}
|
|
}
|
|
|
|
function createMockClient(messagesBySession: Record<string, BackgroundOutputMessage[]>): BackgroundOutputClient {
|
|
const emptyMessages: BackgroundOutputMessage[] = []
|
|
const client = {
|
|
session: {
|
|
messages: async ({ path }: { path: { id: string } }) => ({
|
|
data: messagesBySession[path.id] ?? emptyMessages,
|
|
}),
|
|
},
|
|
} satisfies BackgroundOutputClient
|
|
return client
|
|
}
|
|
|
|
function createTask(overrides: Partial<BackgroundTask> = {}): BackgroundTask {
|
|
return {
|
|
id: "task-1",
|
|
sessionID: "ses-1",
|
|
parentSessionID: "main-1",
|
|
parentMessageID: "msg-1",
|
|
description: "background task",
|
|
prompt: "do work",
|
|
agent: "test-agent",
|
|
status: "running",
|
|
...overrides,
|
|
}
|
|
}
|
|
|
|
describe("background_output full_session", () => {
|
|
test("includes thinking and tool results when enabled", async () => {
|
|
// #given
|
|
const task = createTask()
|
|
const manager = createMockManager(task)
|
|
const client = createMockClient({
|
|
"ses-1": [
|
|
{
|
|
id: "m1",
|
|
info: { role: "assistant", time: "2026-01-01T00:00:00Z", agent: "test" },
|
|
parts: [
|
|
{ type: "text", text: "hello" },
|
|
{ type: "thinking", thinking: "thinking text" },
|
|
{ type: "tool_result", content: "tool output" },
|
|
],
|
|
},
|
|
{
|
|
id: "m2",
|
|
info: { role: "assistant", time: "2026-01-01T00:00:01Z" },
|
|
parts: [
|
|
{ type: "reasoning", text: "reasoning text" },
|
|
{ type: "text", text: "after" },
|
|
],
|
|
},
|
|
],
|
|
})
|
|
const tool = createBackgroundOutput(manager, client)
|
|
|
|
// #when
|
|
const output = await tool.execute({
|
|
task_id: "task-1",
|
|
full_session: true,
|
|
include_thinking: true,
|
|
include_tool_results: true,
|
|
}, mockContext)
|
|
|
|
// #then
|
|
expect(output).toContain("thinking text")
|
|
expect(output).toContain("reasoning text")
|
|
expect(output).toContain("tool output")
|
|
})
|
|
|
|
test("respects since_message_id exclusive filtering", async () => {
|
|
// #given
|
|
const task = createTask()
|
|
const manager = createMockManager(task)
|
|
const client = createMockClient({
|
|
"ses-1": [
|
|
{
|
|
id: "m1",
|
|
info: { role: "assistant", time: "2026-01-01T00:00:00Z" },
|
|
parts: [{ type: "text", text: "hello" }],
|
|
},
|
|
{
|
|
id: "m2",
|
|
info: { role: "assistant", time: "2026-01-01T00:00:01Z" },
|
|
parts: [{ type: "text", text: "after" }],
|
|
},
|
|
],
|
|
})
|
|
const tool = createBackgroundOutput(manager, client)
|
|
|
|
// #when
|
|
const output = await tool.execute({
|
|
task_id: "task-1",
|
|
full_session: true,
|
|
since_message_id: "m1",
|
|
}, mockContext)
|
|
|
|
// #then
|
|
expect(output.includes("hello")).toBe(false)
|
|
expect(output).toContain("after")
|
|
})
|
|
|
|
test("returns error when since_message_id not found", async () => {
|
|
// #given
|
|
const task = createTask()
|
|
const manager = createMockManager(task)
|
|
const client = createMockClient({
|
|
"ses-1": [
|
|
{
|
|
id: "m1",
|
|
info: { role: "assistant", time: "2026-01-01T00:00:00Z" },
|
|
parts: [{ type: "text", text: "hello" }],
|
|
},
|
|
],
|
|
})
|
|
const tool = createBackgroundOutput(manager, client)
|
|
|
|
// #when
|
|
const output = await tool.execute({
|
|
task_id: "task-1",
|
|
full_session: true,
|
|
since_message_id: "missing",
|
|
}, mockContext)
|
|
|
|
// #then
|
|
expect(output).toContain("since_message_id not found")
|
|
})
|
|
|
|
test("caps message_limit at 100", async () => {
|
|
// #given
|
|
const task = createTask()
|
|
const manager = createMockManager(task)
|
|
const messages = Array.from({ length: 120 }, (_, index) => ({
|
|
id: `m${index}`,
|
|
info: {
|
|
role: "assistant",
|
|
time: new Date(2026, 0, 1, 0, 0, index).toISOString(),
|
|
},
|
|
parts: [{ type: "text", text: `message-${index}` }],
|
|
}))
|
|
const client = createMockClient({ "ses-1": messages })
|
|
const tool = createBackgroundOutput(manager, client)
|
|
|
|
// #when
|
|
const output = await tool.execute({
|
|
task_id: "task-1",
|
|
full_session: true,
|
|
message_limit: 200,
|
|
}, mockContext)
|
|
|
|
// #then
|
|
expect(output).toContain("Returned: 100")
|
|
expect(output).toContain("Has more: true")
|
|
})
|
|
|
|
test("keeps legacy status output when full_session is false", async () => {
|
|
// #given
|
|
const task = createTask({ status: "running" })
|
|
const manager = createMockManager(task)
|
|
const client = createMockClient({})
|
|
const tool = createBackgroundOutput(manager, client)
|
|
|
|
// #when
|
|
const output = await tool.execute({ task_id: "task-1" }, mockContext)
|
|
|
|
// #then
|
|
expect(output).toContain("# Task Status")
|
|
expect(output).toContain("Task ID")
|
|
})
|
|
|
|
test("truncates thinking content to thinking_max_chars", async () => {
|
|
// #given
|
|
const longThinking = "x".repeat(500)
|
|
const task = createTask()
|
|
const manager = createMockManager(task)
|
|
const client = createMockClient({
|
|
"ses-1": [
|
|
{
|
|
id: "m1",
|
|
info: { role: "assistant", time: "2026-01-01T00:00:00Z" },
|
|
parts: [
|
|
{ type: "thinking", thinking: longThinking },
|
|
{ type: "text", text: "hello" },
|
|
],
|
|
},
|
|
],
|
|
})
|
|
const tool = createBackgroundOutput(manager, client)
|
|
|
|
// #when
|
|
const output = await tool.execute({
|
|
task_id: "task-1",
|
|
full_session: true,
|
|
include_thinking: true,
|
|
thinking_max_chars: 100,
|
|
}, mockContext)
|
|
|
|
// #then
|
|
expect(output).toContain("[thinking] " + "x".repeat(100) + "...")
|
|
expect(output).not.toContain("x".repeat(200))
|
|
})
|
|
|
|
test("uses default 2000 chars when thinking_max_chars not provided", async () => {
|
|
// #given
|
|
const longThinking = "y".repeat(2500)
|
|
const task = createTask()
|
|
const manager = createMockManager(task)
|
|
const client = createMockClient({
|
|
"ses-1": [
|
|
{
|
|
id: "m1",
|
|
info: { role: "assistant", time: "2026-01-01T00:00:00Z" },
|
|
parts: [
|
|
{ type: "thinking", thinking: longThinking },
|
|
{ type: "text", text: "hello" },
|
|
],
|
|
},
|
|
],
|
|
})
|
|
const tool = createBackgroundOutput(manager, client)
|
|
|
|
// #when
|
|
const output = await tool.execute({
|
|
task_id: "task-1",
|
|
full_session: true,
|
|
include_thinking: true,
|
|
}, mockContext)
|
|
|
|
// #then
|
|
expect(output).toContain("[thinking] " + "y".repeat(2000) + "...")
|
|
expect(output).not.toContain("y".repeat(2100))
|
|
})
|
|
})
|
|
type BackgroundOutputMessage = {
|
|
id?: string
|
|
info?: { role?: string; time?: string | { created?: number }; agent?: string }
|
|
parts?: Array<{
|
|
type?: string
|
|
text?: string
|
|
thinking?: string
|
|
content?: string | Array<{ type: string; text?: string }>
|
|
}>
|
|
}
|