session-composer-dock.spec.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. import { test, expect } from "../fixtures"
  2. import { clearSessionDockSeed, seedSessionQuestion, seedSessionTodos } from "../actions"
  3. import {
  4. permissionDockSelector,
  5. promptSelector,
  6. questionDockSelector,
  7. sessionComposerDockSelector,
  8. sessionTodoDockSelector,
  9. sessionTodoListSelector,
  10. sessionTodoToggleButtonSelector,
  11. } from "../selectors"
  12. type Sdk = Parameters<typeof clearSessionDockSeed>[0]
  13. type PermissionRule = { permission: string; pattern: string; action: "allow" | "deny" | "ask" }
  14. async function withDockSession<T>(
  15. sdk: Sdk,
  16. title: string,
  17. fn: (session: { id: string; title: string }) => Promise<T>,
  18. opts?: { permission?: PermissionRule[] },
  19. ) {
  20. const session = await sdk.session
  21. .create(opts?.permission ? { title, permission: opts.permission } : { title })
  22. .then((r) => r.data)
  23. if (!session?.id) throw new Error("Session create did not return an id")
  24. try {
  25. return await fn(session)
  26. } finally {
  27. await sdk.session.delete({ sessionID: session.id }).catch(() => undefined)
  28. }
  29. }
  30. test.setTimeout(120_000)
  31. async function withDockSeed<T>(sdk: Sdk, sessionID: string, fn: () => Promise<T>) {
  32. try {
  33. return await fn()
  34. } finally {
  35. await clearSessionDockSeed(sdk, sessionID).catch(() => undefined)
  36. }
  37. }
  38. async function clearPermissionDock(page: any, label: RegExp) {
  39. const dock = page.locator(permissionDockSelector)
  40. for (let i = 0; i < 3; i++) {
  41. const count = await dock.count()
  42. if (count === 0) return
  43. await dock.getByRole("button", { name: label }).click()
  44. await page.waitForTimeout(150)
  45. }
  46. }
  47. async function setAutoAccept(page: any, enabled: boolean) {
  48. const button = page.locator('[data-action="prompt-permissions"]').first()
  49. await expect(button).toBeVisible()
  50. const pressed = (await button.getAttribute("aria-pressed")) === "true"
  51. if (pressed === enabled) return
  52. await button.click()
  53. await expect(button).toHaveAttribute("aria-pressed", enabled ? "true" : "false")
  54. }
  55. async function withMockPermission<T>(
  56. page: any,
  57. request: {
  58. id: string
  59. sessionID: string
  60. permission: string
  61. patterns: string[]
  62. metadata?: Record<string, unknown>
  63. always?: string[]
  64. },
  65. opts: { child?: any } | undefined,
  66. fn: () => Promise<T>,
  67. ) {
  68. let pending = [
  69. {
  70. ...request,
  71. always: request.always ?? ["*"],
  72. metadata: request.metadata ?? {},
  73. },
  74. ]
  75. const list = async (route: any) => {
  76. await route.fulfill({
  77. status: 200,
  78. contentType: "application/json",
  79. body: JSON.stringify(pending),
  80. })
  81. }
  82. const reply = async (route: any) => {
  83. const url = new URL(route.request().url())
  84. const id = url.pathname.split("/").pop()
  85. pending = pending.filter((item) => item.id !== id)
  86. await route.fulfill({
  87. status: 200,
  88. contentType: "application/json",
  89. body: JSON.stringify(true),
  90. })
  91. }
  92. await page.route("**/permission", list)
  93. await page.route("**/session/*/permissions/*", reply)
  94. const sessionList = opts?.child
  95. ? async (route: any) => {
  96. const res = await route.fetch()
  97. const json = await res.json()
  98. const list = Array.isArray(json) ? json : Array.isArray(json?.data) ? json.data : undefined
  99. if (Array.isArray(list) && !list.some((item) => item?.id === opts.child?.id)) list.push(opts.child)
  100. await route.fulfill({
  101. status: res.status(),
  102. headers: res.headers(),
  103. contentType: "application/json",
  104. body: JSON.stringify(json),
  105. })
  106. }
  107. : undefined
  108. if (sessionList) await page.route("**/session?*", sessionList)
  109. try {
  110. return await fn()
  111. } finally {
  112. await page.unroute("**/permission", list)
  113. await page.unroute("**/session/*/permissions/*", reply)
  114. if (sessionList) await page.unroute("**/session?*", sessionList)
  115. }
  116. }
  117. test("default dock shows prompt input", async ({ page, sdk, gotoSession }) => {
  118. await withDockSession(sdk, "e2e composer dock default", async (session) => {
  119. await gotoSession(session.id)
  120. await expect(page.locator(sessionComposerDockSelector)).toBeVisible()
  121. await expect(page.locator(promptSelector)).toBeVisible()
  122. await expect(page.locator(questionDockSelector)).toHaveCount(0)
  123. await expect(page.locator(permissionDockSelector)).toHaveCount(0)
  124. await page.locator(promptSelector).click()
  125. await expect(page.locator(promptSelector)).toBeFocused()
  126. })
  127. })
  128. test("blocked question flow unblocks after submit", async ({ page, sdk, gotoSession }) => {
  129. await withDockSession(sdk, "e2e composer dock question", async (session) => {
  130. await withDockSeed(sdk, session.id, async () => {
  131. await gotoSession(session.id)
  132. await seedSessionQuestion(sdk, {
  133. sessionID: session.id,
  134. questions: [
  135. {
  136. header: "Need input",
  137. question: "Pick one option",
  138. options: [
  139. { label: "Continue", description: "Continue now" },
  140. { label: "Stop", description: "Stop here" },
  141. ],
  142. },
  143. ],
  144. })
  145. const dock = page.locator(questionDockSelector)
  146. await expect.poll(() => dock.count(), { timeout: 10_000 }).toBe(1)
  147. await expect(page.locator(promptSelector)).toHaveCount(0)
  148. await dock.locator('[data-slot="question-option"]').first().click()
  149. await dock.getByRole("button", { name: /submit/i }).click()
  150. await expect.poll(() => page.locator(questionDockSelector).count(), { timeout: 10_000 }).toBe(0)
  151. await expect(page.locator(promptSelector)).toBeVisible()
  152. })
  153. })
  154. })
  155. test("blocked permission flow supports allow once", async ({ page, sdk, gotoSession }) => {
  156. await withDockSession(sdk, "e2e composer dock permission once", async (session) => {
  157. await gotoSession(session.id)
  158. await setAutoAccept(page, false)
  159. await withMockPermission(
  160. page,
  161. {
  162. id: "per_e2e_once",
  163. sessionID: session.id,
  164. permission: "bash",
  165. patterns: ["/tmp/opencode-e2e-perm-once"],
  166. metadata: { description: "Need permission for command" },
  167. },
  168. undefined,
  169. async () => {
  170. await page.goto(page.url())
  171. await expect.poll(() => page.locator(permissionDockSelector).count(), { timeout: 10_000 }).toBe(1)
  172. await expect(page.locator(promptSelector)).toHaveCount(0)
  173. await clearPermissionDock(page, /allow once/i)
  174. await page.goto(page.url())
  175. await expect.poll(() => page.locator(permissionDockSelector).count(), { timeout: 10_000 }).toBe(0)
  176. await expect(page.locator(promptSelector)).toBeVisible()
  177. },
  178. )
  179. })
  180. })
  181. test("blocked permission flow supports reject", async ({ page, sdk, gotoSession }) => {
  182. await withDockSession(sdk, "e2e composer dock permission reject", async (session) => {
  183. await gotoSession(session.id)
  184. await setAutoAccept(page, false)
  185. await withMockPermission(
  186. page,
  187. {
  188. id: "per_e2e_reject",
  189. sessionID: session.id,
  190. permission: "bash",
  191. patterns: ["/tmp/opencode-e2e-perm-reject"],
  192. },
  193. undefined,
  194. async () => {
  195. await page.goto(page.url())
  196. await expect.poll(() => page.locator(permissionDockSelector).count(), { timeout: 10_000 }).toBe(1)
  197. await expect(page.locator(promptSelector)).toHaveCount(0)
  198. await clearPermissionDock(page, /deny/i)
  199. await page.goto(page.url())
  200. await expect.poll(() => page.locator(permissionDockSelector).count(), { timeout: 10_000 }).toBe(0)
  201. await expect(page.locator(promptSelector)).toBeVisible()
  202. },
  203. )
  204. })
  205. })
  206. test("blocked permission flow supports allow always", async ({ page, sdk, gotoSession }) => {
  207. await withDockSession(sdk, "e2e composer dock permission always", async (session) => {
  208. await gotoSession(session.id)
  209. await setAutoAccept(page, false)
  210. await withMockPermission(
  211. page,
  212. {
  213. id: "per_e2e_always",
  214. sessionID: session.id,
  215. permission: "bash",
  216. patterns: ["/tmp/opencode-e2e-perm-always"],
  217. metadata: { description: "Need permission for command" },
  218. },
  219. undefined,
  220. async () => {
  221. await page.goto(page.url())
  222. await expect.poll(() => page.locator(permissionDockSelector).count(), { timeout: 10_000 }).toBe(1)
  223. await expect(page.locator(promptSelector)).toHaveCount(0)
  224. await clearPermissionDock(page, /allow always/i)
  225. await page.goto(page.url())
  226. await expect.poll(() => page.locator(permissionDockSelector).count(), { timeout: 10_000 }).toBe(0)
  227. await expect(page.locator(promptSelector)).toBeVisible()
  228. },
  229. )
  230. })
  231. })
  232. test("child session question request blocks parent dock and unblocks after submit", async ({
  233. page,
  234. sdk,
  235. gotoSession,
  236. }) => {
  237. await withDockSession(sdk, "e2e composer dock child question parent", async (session) => {
  238. await gotoSession(session.id)
  239. const child = await sdk.session
  240. .create({
  241. title: "e2e composer dock child question",
  242. parentID: session.id,
  243. })
  244. .then((r) => r.data)
  245. if (!child?.id) throw new Error("Child session create did not return an id")
  246. try {
  247. await withDockSeed(sdk, child.id, async () => {
  248. await seedSessionQuestion(sdk, {
  249. sessionID: child.id,
  250. questions: [
  251. {
  252. header: "Child input",
  253. question: "Pick one child option",
  254. options: [
  255. { label: "Continue", description: "Continue child" },
  256. { label: "Stop", description: "Stop child" },
  257. ],
  258. },
  259. ],
  260. })
  261. const dock = page.locator(questionDockSelector)
  262. await expect.poll(() => dock.count(), { timeout: 10_000 }).toBe(1)
  263. await expect(page.locator(promptSelector)).toHaveCount(0)
  264. await dock.locator('[data-slot="question-option"]').first().click()
  265. await dock.getByRole("button", { name: /submit/i }).click()
  266. await expect.poll(() => page.locator(questionDockSelector).count(), { timeout: 10_000 }).toBe(0)
  267. await expect(page.locator(promptSelector)).toBeVisible()
  268. })
  269. } finally {
  270. await sdk.session.delete({ sessionID: child.id }).catch(() => undefined)
  271. }
  272. })
  273. })
  274. test("child session permission request blocks parent dock and supports allow once", async ({
  275. page,
  276. sdk,
  277. gotoSession,
  278. }) => {
  279. await withDockSession(sdk, "e2e composer dock child permission parent", async (session) => {
  280. await gotoSession(session.id)
  281. await setAutoAccept(page, false)
  282. const child = await sdk.session
  283. .create({
  284. title: "e2e composer dock child permission",
  285. parentID: session.id,
  286. })
  287. .then((r) => r.data)
  288. if (!child?.id) throw new Error("Child session create did not return an id")
  289. try {
  290. await withMockPermission(
  291. page,
  292. {
  293. id: "per_e2e_child",
  294. sessionID: child.id,
  295. permission: "bash",
  296. patterns: ["/tmp/opencode-e2e-perm-child"],
  297. metadata: { description: "Need child permission" },
  298. },
  299. { child },
  300. async () => {
  301. await page.goto(page.url())
  302. const dock = page.locator(permissionDockSelector)
  303. await expect.poll(() => dock.count(), { timeout: 10_000 }).toBe(1)
  304. await expect(page.locator(promptSelector)).toHaveCount(0)
  305. await clearPermissionDock(page, /allow once/i)
  306. await page.goto(page.url())
  307. await expect.poll(() => page.locator(permissionDockSelector).count(), { timeout: 10_000 }).toBe(0)
  308. await expect(page.locator(promptSelector)).toBeVisible()
  309. },
  310. )
  311. } finally {
  312. await sdk.session.delete({ sessionID: child.id }).catch(() => undefined)
  313. }
  314. })
  315. })
  316. test("todo dock transitions and collapse behavior", async ({ page, sdk, gotoSession }) => {
  317. await withDockSession(sdk, "e2e composer dock todo", async (session) => {
  318. await withDockSeed(sdk, session.id, async () => {
  319. await gotoSession(session.id)
  320. await seedSessionTodos(sdk, {
  321. sessionID: session.id,
  322. todos: [
  323. { content: "first task", status: "pending", priority: "high" },
  324. { content: "second task", status: "in_progress", priority: "medium" },
  325. ],
  326. })
  327. await expect.poll(() => page.locator(sessionTodoDockSelector).count(), { timeout: 10_000 }).toBe(1)
  328. await expect(page.locator(sessionTodoListSelector)).toBeVisible()
  329. await page.locator(sessionTodoToggleButtonSelector).click()
  330. await expect(page.locator(sessionTodoListSelector)).toBeHidden()
  331. await page.locator(sessionTodoToggleButtonSelector).click()
  332. await expect(page.locator(sessionTodoListSelector)).toBeVisible()
  333. await seedSessionTodos(sdk, {
  334. sessionID: session.id,
  335. todos: [
  336. { content: "first task", status: "completed", priority: "high" },
  337. { content: "second task", status: "cancelled", priority: "medium" },
  338. ],
  339. })
  340. await expect.poll(() => page.locator(sessionTodoDockSelector).count(), { timeout: 10_000 }).toBe(0)
  341. })
  342. })
  343. })
  344. test("keyboard focus stays off prompt while blocked", async ({ page, sdk, gotoSession }) => {
  345. await withDockSession(sdk, "e2e composer dock keyboard", async (session) => {
  346. await withDockSeed(sdk, session.id, async () => {
  347. await gotoSession(session.id)
  348. await seedSessionQuestion(sdk, {
  349. sessionID: session.id,
  350. questions: [
  351. {
  352. header: "Need input",
  353. question: "Pick one option",
  354. options: [{ label: "Continue", description: "Continue now" }],
  355. },
  356. ],
  357. })
  358. await expect.poll(() => page.locator(questionDockSelector).count(), { timeout: 10_000 }).toBe(1)
  359. await expect(page.locator(promptSelector)).toHaveCount(0)
  360. await page.locator("main").click({ position: { x: 5, y: 5 } })
  361. await page.keyboard.type("abc")
  362. await expect(page.locator(promptSelector)).toHaveCount(0)
  363. })
  364. })
  365. })