import { test, expect } from "../fixtures" import { composerEvent, type ComposerDriverState, type ComposerProbeState, type ComposerWindow, } from "../../src/testing/session-composer" import { cleanupSession, clearSessionDockSeed, seedSessionQuestion } from "../actions" import { permissionDockSelector, promptSelector, questionDockSelector, sessionComposerDockSelector, sessionTodoToggleButtonSelector, } from "../selectors" type Sdk = Parameters[0] type PermissionRule = { permission: string; pattern: string; action: "allow" | "deny" | "ask" } async function withDockSession( sdk: Sdk, title: string, fn: (session: { id: string; title: string }) => Promise, opts?: { permission?: PermissionRule[] }, ) { const session = await sdk.session .create(opts?.permission ? { title, permission: opts.permission } : { title }) .then((r) => r.data) if (!session?.id) throw new Error("Session create did not return an id") try { return await fn(session) } finally { await cleanupSession({ sdk, sessionID: session.id }) } } test.setTimeout(120_000) async function withDockSeed(sdk: Sdk, sessionID: string, fn: () => Promise) { try { return await fn() } finally { await clearSessionDockSeed(sdk, sessionID).catch(() => undefined) } } async function clearPermissionDock(page: any, label: RegExp) { const dock = page.locator(permissionDockSelector) await expect(dock).toBeVisible() await dock.getByRole("button", { name: label }).click() } async function setAutoAccept(page: any, enabled: boolean) { const button = page.locator('[data-action="prompt-permissions"]').first() await expect(button).toBeVisible() const pressed = (await button.getAttribute("aria-pressed")) === "true" if (pressed === enabled) return await button.click() await expect(button).toHaveAttribute("aria-pressed", enabled ? "true" : "false") } async function expectQuestionBlocked(page: any) { await expect(page.locator(questionDockSelector)).toBeVisible() await expect(page.locator(promptSelector)).toHaveCount(0) } async function expectQuestionOpen(page: any) { await expect(page.locator(questionDockSelector)).toHaveCount(0) await expect(page.locator(promptSelector)).toBeVisible() } async function expectPermissionBlocked(page: any) { await expect(page.locator(permissionDockSelector)).toBeVisible() await expect(page.locator(promptSelector)).toHaveCount(0) } async function expectPermissionOpen(page: any) { await expect(page.locator(permissionDockSelector)).toHaveCount(0) await expect(page.locator(promptSelector)).toBeVisible() } async function todoDock(page: any, sessionID: string) { await page.addInitScript(() => { const win = window as ComposerWindow win.__opencode_e2e = { ...win.__opencode_e2e, composer: { enabled: true, sessions: {}, }, } }) const write = async (driver: ComposerDriverState | undefined) => { await page.evaluate( (input) => { const win = window as ComposerWindow const composer = win.__opencode_e2e?.composer if (!composer?.enabled) throw new Error("Composer e2e driver is not enabled") composer.sessions ??= {} const prev = composer.sessions[input.sessionID] ?? {} if (!input.driver) { if (!prev.probe) { delete composer.sessions[input.sessionID] } else { composer.sessions[input.sessionID] = { probe: prev.probe } } } else { composer.sessions[input.sessionID] = { ...prev, driver: input.driver, } } window.dispatchEvent(new CustomEvent(input.event, { detail: { sessionID: input.sessionID } })) }, { event: composerEvent, sessionID, driver }, ) } const read = () => page.evaluate((sessionID) => { const win = window as ComposerWindow return win.__opencode_e2e?.composer?.sessions?.[sessionID]?.probe ?? null }, sessionID) as Promise const api = { async clear() { await write(undefined) return api }, async open(todos: NonNullable) { await write({ live: true, todos }) return api }, async finish(todos: NonNullable) { await write({ live: false, todos }) return api }, async expectOpen(states: ComposerProbeState["states"]) { await expect.poll(read, { timeout: 10_000 }).toMatchObject({ mounted: true, collapsed: false, hidden: false, count: states.length, states, }) return api }, async expectCollapsed(states: ComposerProbeState["states"]) { await expect.poll(read, { timeout: 10_000 }).toMatchObject({ mounted: true, collapsed: true, hidden: true, count: states.length, states, }) return api }, async expectClosed() { await expect.poll(read, { timeout: 10_000 }).toMatchObject({ mounted: false }) return api }, async collapse() { await page.locator(sessionTodoToggleButtonSelector).click() return api }, async expand() { await page.locator(sessionTodoToggleButtonSelector).click() return api }, } return api } async function withMockPermission( page: any, request: { id: string sessionID: string permission: string patterns: string[] metadata?: Record always?: string[] }, opts: { child?: any } | undefined, fn: (state: { resolved: () => Promise }) => Promise, ) { let pending = [ { ...request, always: request.always ?? ["*"], metadata: request.metadata ?? {}, }, ] const list = async (route: any) => { await route.fulfill({ status: 200, contentType: "application/json", body: JSON.stringify(pending), }) } const reply = async (route: any) => { const url = new URL(route.request().url()) const id = url.pathname.split("/").pop() pending = pending.filter((item) => item.id !== id) await route.fulfill({ status: 200, contentType: "application/json", body: JSON.stringify(true), }) } await page.route("**/permission", list) await page.route("**/session/*/permissions/*", reply) const sessionList = opts?.child ? async (route: any) => { const res = await route.fetch() const json = await res.json() const list = Array.isArray(json) ? json : Array.isArray(json?.data) ? json.data : undefined if (Array.isArray(list) && !list.some((item) => item?.id === opts.child?.id)) list.push(opts.child) await route.fulfill({ status: res.status(), headers: res.headers(), contentType: "application/json", body: JSON.stringify(json), }) } : undefined if (sessionList) await page.route("**/session?*", sessionList) const state = { async resolved() { await expect.poll(() => pending.length, { timeout: 10_000 }).toBe(0) }, } try { return await fn(state) } finally { await page.unroute("**/permission", list) await page.unroute("**/session/*/permissions/*", reply) if (sessionList) await page.unroute("**/session?*", sessionList) } } test("default dock shows prompt input", async ({ page, sdk, gotoSession }) => { await withDockSession(sdk, "e2e composer dock default", async (session) => { await gotoSession(session.id) await expect(page.locator(sessionComposerDockSelector)).toBeVisible() await expect(page.locator(promptSelector)).toBeVisible() await expect(page.locator(questionDockSelector)).toHaveCount(0) await expect(page.locator(permissionDockSelector)).toHaveCount(0) await page.locator(promptSelector).click() await expect(page.locator(promptSelector)).toBeFocused() }) }) test("auto-accept toggle works before first submit", async ({ page, gotoSession }) => { await gotoSession() const button = page.locator('[data-action="prompt-permissions"]').first() await expect(button).toBeVisible() await expect(button).toHaveAttribute("aria-pressed", "false") await setAutoAccept(page, true) await setAutoAccept(page, false) }) test("blocked question flow unblocks after submit", async ({ page, sdk, gotoSession }) => { await withDockSession(sdk, "e2e composer dock question", async (session) => { await withDockSeed(sdk, session.id, async () => { await gotoSession(session.id) await seedSessionQuestion(sdk, { sessionID: session.id, questions: [ { header: "Need input", question: "Pick one option", options: [ { label: "Continue", description: "Continue now" }, { label: "Stop", description: "Stop here" }, ], }, ], }) const dock = page.locator(questionDockSelector) await expectQuestionBlocked(page) await dock.locator('[data-slot="question-option"]').first().click() await dock.getByRole("button", { name: /submit/i }).click() await expectQuestionOpen(page) }) }) }) test("blocked permission flow supports allow once", async ({ page, sdk, gotoSession }) => { await withDockSession(sdk, "e2e composer dock permission once", async (session) => { await gotoSession(session.id) await setAutoAccept(page, false) await withMockPermission( page, { id: "per_e2e_once", sessionID: session.id, permission: "bash", patterns: ["/tmp/opencode-e2e-perm-once"], metadata: { description: "Need permission for command" }, }, undefined, async (state) => { await page.goto(page.url()) await expectPermissionBlocked(page) await clearPermissionDock(page, /allow once/i) await state.resolved() await page.goto(page.url()) await expectPermissionOpen(page) }, ) }) }) test("blocked permission flow supports reject", async ({ page, sdk, gotoSession }) => { await withDockSession(sdk, "e2e composer dock permission reject", async (session) => { await gotoSession(session.id) await setAutoAccept(page, false) await withMockPermission( page, { id: "per_e2e_reject", sessionID: session.id, permission: "bash", patterns: ["/tmp/opencode-e2e-perm-reject"], }, undefined, async (state) => { await page.goto(page.url()) await expectPermissionBlocked(page) await clearPermissionDock(page, /deny/i) await state.resolved() await page.goto(page.url()) await expectPermissionOpen(page) }, ) }) }) test("blocked permission flow supports allow always", async ({ page, sdk, gotoSession }) => { await withDockSession(sdk, "e2e composer dock permission always", async (session) => { await gotoSession(session.id) await setAutoAccept(page, false) await withMockPermission( page, { id: "per_e2e_always", sessionID: session.id, permission: "bash", patterns: ["/tmp/opencode-e2e-perm-always"], metadata: { description: "Need permission for command" }, }, undefined, async (state) => { await page.goto(page.url()) await expectPermissionBlocked(page) await clearPermissionDock(page, /allow always/i) await state.resolved() await page.goto(page.url()) await expectPermissionOpen(page) }, ) }) }) test("child session question request blocks parent dock and unblocks after submit", async ({ page, sdk, gotoSession, }) => { await withDockSession(sdk, "e2e composer dock child question parent", async (session) => { await gotoSession(session.id) const child = await sdk.session .create({ title: "e2e composer dock child question", parentID: session.id, }) .then((r) => r.data) if (!child?.id) throw new Error("Child session create did not return an id") try { await withDockSeed(sdk, child.id, async () => { await seedSessionQuestion(sdk, { sessionID: child.id, questions: [ { header: "Child input", question: "Pick one child option", options: [ { label: "Continue", description: "Continue child" }, { label: "Stop", description: "Stop child" }, ], }, ], }) const dock = page.locator(questionDockSelector) await expectQuestionBlocked(page) await dock.locator('[data-slot="question-option"]').first().click() await dock.getByRole("button", { name: /submit/i }).click() await expectQuestionOpen(page) }) } finally { await cleanupSession({ sdk, sessionID: child.id }) } }) }) test("child session permission request blocks parent dock and supports allow once", async ({ page, sdk, gotoSession, }) => { await withDockSession(sdk, "e2e composer dock child permission parent", async (session) => { await gotoSession(session.id) await setAutoAccept(page, false) const child = await sdk.session .create({ title: "e2e composer dock child permission", parentID: session.id, }) .then((r) => r.data) if (!child?.id) throw new Error("Child session create did not return an id") try { await withMockPermission( page, { id: "per_e2e_child", sessionID: child.id, permission: "bash", patterns: ["/tmp/opencode-e2e-perm-child"], metadata: { description: "Need child permission" }, }, { child }, async (state) => { await page.goto(page.url()) await expectPermissionBlocked(page) await clearPermissionDock(page, /allow once/i) await state.resolved() await page.goto(page.url()) await expectPermissionOpen(page) }, ) } finally { await cleanupSession({ sdk, sessionID: child.id }) } }) }) test("todo dock transitions and collapse behavior", async ({ page, sdk, gotoSession }) => { await withDockSession(sdk, "e2e composer dock todo", async (session) => { const dock = await todoDock(page, session.id) await gotoSession(session.id) await expect(page.locator(sessionComposerDockSelector)).toBeVisible() try { await dock.open([ { content: "first task", status: "pending", priority: "high" }, { content: "second task", status: "in_progress", priority: "medium" }, ]) await dock.expectOpen(["pending", "in_progress"]) await dock.collapse() await dock.expectCollapsed(["pending", "in_progress"]) await dock.expand() await dock.expectOpen(["pending", "in_progress"]) await dock.finish([ { content: "first task", status: "completed", priority: "high" }, { content: "second task", status: "cancelled", priority: "medium" }, ]) await dock.expectClosed() } finally { await dock.clear() } }) }) test("keyboard focus stays off prompt while blocked", async ({ page, sdk, gotoSession }) => { await withDockSession(sdk, "e2e composer dock keyboard", async (session) => { await withDockSeed(sdk, session.id, async () => { await gotoSession(session.id) await seedSessionQuestion(sdk, { sessionID: session.id, questions: [ { header: "Need input", question: "Pick one option", options: [{ label: "Continue", description: "Continue now" }], }, ], }) await expectQuestionBlocked(page) await page.locator("main").click({ position: { x: 5, y: 5 } }) await page.keyboard.type("abc") await expect(page.locator(promptSelector)).toHaveCount(0) }) }) })