workspace-new-session.spec.ts 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  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 expect(prompt).toBeEditable()
  62. await prompt.click()
  63. await expect(prompt).toBeFocused()
  64. await prompt.fill(text)
  65. await expect.poll(async () => ((await prompt.textContent()) ?? "").trim()).toContain(text)
  66. await prompt.press("Enter")
  67. await expect.poll(() => slugFromUrl(page.url())).toBe(slug)
  68. await expect.poll(() => sessionIDFromUrl(page.url()) ?? "", { timeout: 30_000 }).not.toBe("")
  69. const sessionID = sessionIDFromUrl(page.url())
  70. if (!sessionID) throw new Error(`Failed to parse session id from url: ${page.url()}`)
  71. await expect(page).toHaveURL(new RegExp(`/${slug}/session/${sessionID}(?:[/?#]|$)`))
  72. return sessionID
  73. }
  74. async function sessionDirectory(directory: string, sessionID: string) {
  75. const info = await createSdk(directory)
  76. .session.get({ sessionID })
  77. .then((x) => x.data)
  78. .catch(() => undefined)
  79. if (!info) return ""
  80. return info.directory
  81. }
  82. test("new sessions from sidebar workspace actions stay in selected workspace", async ({ page, withProject }) => {
  83. await page.setViewportSize({ width: 1400, height: 800 })
  84. await withProject(async ({ directory, slug: root }) => {
  85. const workspaces = [] as { slug: string; directory: string }[]
  86. const sessions = [] as string[]
  87. try {
  88. await openSidebar(page)
  89. await setWorkspacesEnabled(page, root, true)
  90. const first = await createWorkspace(page, root, [])
  91. workspaces.push(first)
  92. await waitWorkspaceReady(page, first.slug)
  93. const second = await createWorkspace(page, root, [first.slug])
  94. workspaces.push(second)
  95. await waitWorkspaceReady(page, second.slug)
  96. const firstSession = await createSessionFromWorkspace(page, first.slug, `workspace one ${Date.now()}`)
  97. sessions.push(firstSession)
  98. const secondSession = await createSessionFromWorkspace(page, second.slug, `workspace two ${Date.now()}`)
  99. sessions.push(secondSession)
  100. const thirdSession = await createSessionFromWorkspace(page, first.slug, `workspace one again ${Date.now()}`)
  101. sessions.push(thirdSession)
  102. await expect.poll(() => sessionDirectory(first.directory, firstSession)).toBe(first.directory)
  103. await expect.poll(() => sessionDirectory(second.directory, secondSession)).toBe(second.directory)
  104. await expect.poll(() => sessionDirectory(first.directory, thirdSession)).toBe(first.directory)
  105. } finally {
  106. const dirs = [directory, ...workspaces.map((workspace) => workspace.directory)]
  107. await Promise.all(
  108. sessions.map((sessionID) =>
  109. Promise.all(
  110. dirs.map((dir) =>
  111. createSdk(dir)
  112. .session.delete({ sessionID })
  113. .catch(() => undefined),
  114. ),
  115. ),
  116. ),
  117. )
  118. await Promise.all(workspaces.map((workspace) => cleanupTestProject(workspace.directory)))
  119. }
  120. })
  121. })