projects-switch.spec.ts 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. import { base64Decode } from "@opencode-ai/util/encode"
  2. import { test, expect } from "../fixtures"
  3. import {
  4. defocus,
  5. createTestProject,
  6. cleanupTestProject,
  7. openSidebar,
  8. setWorkspacesEnabled,
  9. sessionIDFromUrl,
  10. } from "../actions"
  11. import { projectSwitchSelector, promptSelector, workspaceItemSelector, workspaceNewSessionSelector } from "../selectors"
  12. import { createSdk, dirSlug, sessionPath } from "../utils"
  13. function slugFromUrl(url: string) {
  14. return /\/([^/]+)\/session(?:\/|$)/.exec(url)?.[1] ?? ""
  15. }
  16. test("can switch between projects from sidebar", async ({ page, withProject }) => {
  17. await page.setViewportSize({ width: 1400, height: 800 })
  18. const other = await createTestProject()
  19. const otherSlug = dirSlug(other)
  20. try {
  21. await withProject(
  22. async ({ directory }) => {
  23. await defocus(page)
  24. const currentSlug = dirSlug(directory)
  25. const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
  26. await expect(otherButton).toBeVisible()
  27. await otherButton.click()
  28. await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
  29. const currentButton = page.locator(projectSwitchSelector(currentSlug)).first()
  30. await expect(currentButton).toBeVisible()
  31. await currentButton.click()
  32. await expect(page).toHaveURL(new RegExp(`/${currentSlug}/session`))
  33. },
  34. { extra: [other] },
  35. )
  36. } finally {
  37. await cleanupTestProject(other)
  38. }
  39. })
  40. test("switching back to a project opens the latest workspace session", async ({ page, withProject }) => {
  41. await page.setViewportSize({ width: 1400, height: 800 })
  42. const other = await createTestProject()
  43. const otherSlug = dirSlug(other)
  44. let rootDir: string | undefined
  45. let workspaceDir: string | undefined
  46. let sessionID: string | undefined
  47. try {
  48. await withProject(
  49. async ({ directory, slug }) => {
  50. rootDir = directory
  51. await defocus(page)
  52. await openSidebar(page)
  53. await setWorkspacesEnabled(page, slug, true)
  54. await page.getByRole("button", { name: "New workspace" }).first().click()
  55. await expect
  56. .poll(
  57. () => {
  58. const next = slugFromUrl(page.url())
  59. if (!next) return ""
  60. if (next === slug) return ""
  61. return next
  62. },
  63. { timeout: 45_000 },
  64. )
  65. .not.toBe("")
  66. const workspaceSlug = slugFromUrl(page.url())
  67. workspaceDir = base64Decode(workspaceSlug)
  68. if (!workspaceDir) throw new Error(`Failed to decode workspace slug: ${workspaceSlug}`)
  69. await openSidebar(page)
  70. const workspace = page.locator(workspaceItemSelector(workspaceSlug)).first()
  71. await expect(workspace).toBeVisible()
  72. await workspace.hover()
  73. const newSession = page.locator(workspaceNewSessionSelector(workspaceSlug)).first()
  74. await expect(newSession).toBeVisible()
  75. await newSession.click({ force: true })
  76. await expect(page).toHaveURL(new RegExp(`/${workspaceSlug}/session(?:[/?#]|$)`))
  77. // Create a session by sending a prompt
  78. const prompt = page.locator(promptSelector)
  79. await expect(prompt).toBeVisible()
  80. await prompt.fill("test")
  81. await page.keyboard.press("Enter")
  82. // Wait for the URL to update with the new session ID
  83. await expect.poll(() => sessionIDFromUrl(page.url()) ?? "", { timeout: 15_000 }).not.toBe("")
  84. const created = sessionIDFromUrl(page.url())
  85. if (!created) throw new Error(`Failed to get session ID from url: ${page.url()}`)
  86. sessionID = created
  87. await expect(page).toHaveURL(new RegExp(`/${workspaceSlug}/session/${created}(?:[/?#]|$)`))
  88. await openSidebar(page)
  89. const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
  90. await expect(otherButton).toBeVisible()
  91. await otherButton.click()
  92. await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
  93. const rootButton = page.locator(projectSwitchSelector(slug)).first()
  94. await expect(rootButton).toBeVisible()
  95. await rootButton.click()
  96. await expect.poll(() => sessionIDFromUrl(page.url()) ?? "").toBe(created)
  97. await expect(page).toHaveURL(new RegExp(`/session/${created}(?:[/?#]|$)`))
  98. },
  99. { extra: [other] },
  100. )
  101. } finally {
  102. if (sessionID) {
  103. const id = sessionID
  104. const dirs = [rootDir, workspaceDir].filter((x): x is string => !!x)
  105. await Promise.all(
  106. dirs.map((directory) =>
  107. createSdk(directory)
  108. .session.delete({ sessionID: id })
  109. .catch(() => undefined),
  110. ),
  111. )
  112. }
  113. if (workspaceDir) {
  114. await cleanupTestProject(workspaceDir)
  115. }
  116. await cleanupTestProject(other)
  117. }
  118. })