projects-switch.spec.ts 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  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. const created = await createSdk(workspaceDir)
  78. .session.create()
  79. .then((x) => x.data?.id)
  80. if (!created) throw new Error(`Failed to create session for workspace: ${workspaceDir}`)
  81. sessionID = created
  82. await page.goto(sessionPath(workspaceDir, created))
  83. await expect(page.locator(promptSelector)).toBeVisible()
  84. await expect(page).toHaveURL(new RegExp(`/${workspaceSlug}/session/${created}(?:[/?#]|$)`))
  85. await openSidebar(page)
  86. const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
  87. await expect(otherButton).toBeVisible()
  88. await otherButton.click()
  89. await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
  90. const rootButton = page.locator(projectSwitchSelector(slug)).first()
  91. await expect(rootButton).toBeVisible()
  92. await rootButton.click()
  93. await expect.poll(() => sessionIDFromUrl(page.url()) ?? "").toBe(created)
  94. await expect(page).toHaveURL(new RegExp(`/session/${created}(?:[/?#]|$)`))
  95. },
  96. { extra: [other] },
  97. )
  98. } finally {
  99. if (sessionID) {
  100. const id = sessionID
  101. const dirs = [rootDir, workspaceDir].filter((x): x is string => !!x)
  102. await Promise.all(
  103. dirs.map((directory) =>
  104. createSdk(directory)
  105. .session.delete({ sessionID: id })
  106. .catch(() => undefined),
  107. ),
  108. )
  109. }
  110. if (workspaceDir) {
  111. await cleanupTestProject(workspaceDir)
  112. }
  113. await cleanupTestProject(other)
  114. }
  115. })