workspace-new-session.spec.ts 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. import { base64Decode } from "@opencode-ai/util/encode"
  2. import type { Page } from "@playwright/test"
  3. import { test, expect } from "../fixtures"
  4. import { cleanupTestProject, openSidebar, sessionIDFromUrl, setWorkspacesEnabled } from "../actions"
  5. import { promptSelector, workspaceItemSelector, workspaceNewSessionSelector } from "../selectors"
  6. import { createSdk } from "../utils"
  7. function slugFromUrl(url: string) {
  8. return /\/([^/]+)\/session(?:\/|$)/.exec(url)?.[1] ?? ""
  9. }
  10. async function waitWorkspaceReady(page: Page, slug: string) {
  11. await openSidebar(page)
  12. await expect
  13. .poll(
  14. async () => {
  15. const item = page.locator(workspaceItemSelector(slug)).first()
  16. try {
  17. await item.hover({ timeout: 500 })
  18. return true
  19. } catch {
  20. return false
  21. }
  22. },
  23. { timeout: 60_000 },
  24. )
  25. .toBe(true)
  26. }
  27. async function createWorkspace(page: Page, root: string, seen: string[]) {
  28. await openSidebar(page)
  29. await page.getByRole("button", { name: "New workspace" }).first().click()
  30. await expect
  31. .poll(
  32. () => {
  33. const slug = slugFromUrl(page.url())
  34. if (!slug) return ""
  35. if (slug === root) return ""
  36. if (seen.includes(slug)) return ""
  37. return slug
  38. },
  39. { timeout: 45_000 },
  40. )
  41. .not.toBe("")
  42. const slug = slugFromUrl(page.url())
  43. const directory = base64Decode(slug)
  44. if (!directory) throw new Error(`Failed to decode workspace slug: ${slug}`)
  45. return { slug, directory }
  46. }
  47. async function openWorkspaceNewSession(page: Page, slug: string) {
  48. await waitWorkspaceReady(page, slug)
  49. const item = page.locator(workspaceItemSelector(slug)).first()
  50. await item.hover()
  51. const button = page.locator(workspaceNewSessionSelector(slug)).first()
  52. await expect(button).toBeVisible()
  53. await button.click({ force: true })
  54. await expect.poll(() => slugFromUrl(page.url())).toBe(slug)
  55. await expect(page).toHaveURL(new RegExp(`/${slug}/session(?:[/?#]|$)`))
  56. }
  57. async function createSessionFromWorkspace(page: Page, slug: string, text: string) {
  58. await openWorkspaceNewSession(page, slug)
  59. const prompt = page.locator(promptSelector)
  60. await expect(prompt).toBeVisible()
  61. await prompt.click()
  62. await page.keyboard.type(text)
  63. await page.keyboard.press("Enter")
  64. await expect.poll(() => slugFromUrl(page.url())).toBe(slug)
  65. await expect(page).toHaveURL(new RegExp(`/${slug}/session/[^/?#]+`), { timeout: 30_000 })
  66. const sessionID = sessionIDFromUrl(page.url())
  67. if (!sessionID) throw new Error(`Failed to parse session id from url: ${page.url()}`)
  68. return sessionID
  69. }
  70. async function sessionDirectory(directory: string, sessionID: string) {
  71. const info = await createSdk(directory)
  72. .session.get({ sessionID })
  73. .then((x) => x.data)
  74. .catch(() => undefined)
  75. if (!info) return ""
  76. return info.directory
  77. }
  78. test("new sessions from sidebar workspace actions stay in selected workspace", async ({ page, withProject }) => {
  79. await page.setViewportSize({ width: 1400, height: 800 })
  80. await withProject(async ({ directory, slug: root }) => {
  81. const workspaces = [] as { slug: string; directory: string }[]
  82. const sessions = [] as string[]
  83. try {
  84. await openSidebar(page)
  85. await setWorkspacesEnabled(page, root, true)
  86. const first = await createWorkspace(page, root, [])
  87. workspaces.push(first)
  88. await waitWorkspaceReady(page, first.slug)
  89. const second = await createWorkspace(page, root, [first.slug])
  90. workspaces.push(second)
  91. await waitWorkspaceReady(page, second.slug)
  92. const firstSession = await createSessionFromWorkspace(page, first.slug, `workspace one ${Date.now()}`)
  93. sessions.push(firstSession)
  94. const secondSession = await createSessionFromWorkspace(page, second.slug, `workspace two ${Date.now()}`)
  95. sessions.push(secondSession)
  96. const thirdSession = await createSessionFromWorkspace(page, first.slug, `workspace one again ${Date.now()}`)
  97. sessions.push(thirdSession)
  98. await expect.poll(() => sessionDirectory(first.directory, firstSession)).toBe(first.directory)
  99. await expect.poll(() => sessionDirectory(second.directory, secondSession)).toBe(second.directory)
  100. await expect.poll(() => sessionDirectory(first.directory, thirdSession)).toBe(first.directory)
  101. } finally {
  102. const dirs = [directory, ...workspaces.map((workspace) => workspace.directory)]
  103. await Promise.all(
  104. sessions.map((sessionID) =>
  105. Promise.all(
  106. dirs.map((dir) =>
  107. createSdk(dir)
  108. .session.delete({ sessionID })
  109. .catch(() => undefined),
  110. ),
  111. ),
  112. ),
  113. )
  114. await Promise.all(workspaces.map((workspace) => cleanupTestProject(workspace.directory)))
  115. }
  116. })
  117. })