- Add directory parameter to session API calls (session.get, session.todo,
session.status, session.children)
- Improve agent resolver with display name support via agent-display-names
- Add tool execution visibility in event handlers with running/completed
status output
- Enhance poll-for-completion with main session status checking and
stabilization period handling
- Add normalizeSDKResponse import for consistent response handling
- Update types with Todo, ChildSession, and toast-related interfaces
🤖 Generated with OhMyOpenCode assistance
137 lines
3.7 KiB
TypeScript
137 lines
3.7 KiB
TypeScript
/// <reference types="bun-types" />
|
|
|
|
import { describe, it, expect, spyOn, afterEach } from "bun:test"
|
|
import type { OhMyOpenCodeConfig } from "../../config"
|
|
import { resolveRunAgent, waitForEventProcessorShutdown } from "./runner"
|
|
|
|
const createConfig = (overrides: Partial<OhMyOpenCodeConfig> = {}): OhMyOpenCodeConfig => ({
|
|
...overrides,
|
|
})
|
|
|
|
describe("resolveRunAgent", () => {
|
|
it("uses CLI agent over env and config", () => {
|
|
// given
|
|
const config = createConfig({ default_run_agent: "prometheus" })
|
|
const env = { OPENCODE_DEFAULT_AGENT: "Atlas" }
|
|
|
|
// when
|
|
const agent = resolveRunAgent(
|
|
{ message: "test", agent: "Hephaestus" },
|
|
config,
|
|
env
|
|
)
|
|
|
|
// then
|
|
expect(agent).toBe("Hephaestus (Deep Agent)")
|
|
})
|
|
|
|
it("uses env agent over config", () => {
|
|
// given
|
|
const config = createConfig({ default_run_agent: "prometheus" })
|
|
const env = { OPENCODE_DEFAULT_AGENT: "Atlas" }
|
|
|
|
// when
|
|
const agent = resolveRunAgent({ message: "test" }, config, env)
|
|
|
|
// then
|
|
expect(agent).toBe("Atlas (Plan Executor)")
|
|
})
|
|
|
|
it("uses config agent over default", () => {
|
|
// given
|
|
const config = createConfig({ default_run_agent: "Prometheus" })
|
|
|
|
// when
|
|
const agent = resolveRunAgent({ message: "test" }, config, {})
|
|
|
|
// then
|
|
expect(agent).toBe("Prometheus (Plan Builder)")
|
|
})
|
|
|
|
it("falls back to sisyphus when none set", () => {
|
|
// given
|
|
const config = createConfig()
|
|
|
|
// when
|
|
const agent = resolveRunAgent({ message: "test" }, config, {})
|
|
|
|
// then
|
|
expect(agent).toBe("Sisyphus (Ultraworker)")
|
|
})
|
|
|
|
it("skips disabled sisyphus for next available core agent", () => {
|
|
// given
|
|
const config = createConfig({ disabled_agents: ["sisyphus"] })
|
|
|
|
// when
|
|
const agent = resolveRunAgent({ message: "test" }, config, {})
|
|
|
|
// then
|
|
expect(agent).toBe("Hephaestus (Deep Agent)")
|
|
})
|
|
|
|
it("maps display-name style default_run_agent values to canonical display names", () => {
|
|
// given
|
|
const config = createConfig({ default_run_agent: "Sisyphus (Ultraworker)" })
|
|
|
|
// when
|
|
const agent = resolveRunAgent({ message: "test" }, config, {})
|
|
|
|
// then
|
|
expect(agent).toBe("Sisyphus (Ultraworker)")
|
|
})
|
|
})
|
|
|
|
describe("waitForEventProcessorShutdown", () => {
|
|
let consoleLogSpy: ReturnType<typeof spyOn<typeof console, "log">> | null = null
|
|
|
|
afterEach(() => {
|
|
if (consoleLogSpy) {
|
|
consoleLogSpy.mockRestore()
|
|
consoleLogSpy = null
|
|
}
|
|
})
|
|
|
|
it("returns quickly when event processor completes", async () => {
|
|
//#given
|
|
const eventProcessor = new Promise<void>((resolve) => {
|
|
setTimeout(() => {
|
|
resolve()
|
|
}, 25)
|
|
})
|
|
consoleLogSpy = spyOn(console, "log").mockImplementation(() => {})
|
|
const start = performance.now()
|
|
|
|
//#when
|
|
await waitForEventProcessorShutdown(eventProcessor, 200)
|
|
|
|
//#then
|
|
const elapsed = performance.now() - start
|
|
expect(elapsed).toBeLessThan(200)
|
|
expect(console.log).not.toHaveBeenCalledWith(
|
|
"[run] Event stream did not close within 200ms after abort; continuing shutdown.",
|
|
)
|
|
})
|
|
|
|
it("times out and continues when event processor does not complete", async () => {
|
|
//#given
|
|
const eventProcessor = new Promise<void>(() => {})
|
|
const spy = spyOn(console, "log").mockImplementation(() => {})
|
|
consoleLogSpy = spy
|
|
const timeoutMs = 200
|
|
const start = performance.now()
|
|
|
|
try {
|
|
//#when
|
|
await waitForEventProcessorShutdown(eventProcessor, timeoutMs)
|
|
|
|
//#then
|
|
const elapsed = performance.now() - start
|
|
expect(elapsed).toBeGreaterThanOrEqual(timeoutMs - 10)
|
|
expect(spy.mock.calls.length).toBeGreaterThanOrEqual(1)
|
|
} finally {
|
|
spy.mockRestore()
|
|
}
|
|
})
|
|
})
|