projects-switch.spec.ts 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. import { base64Decode } from "@opencode-ai/util/encode"
  2. import type { Page } from "@playwright/test"
  3. import { test, expect } from "../fixtures"
  4. import {
  5. defocus,
  6. createTestProject,
  7. cleanupTestProject,
  8. openSidebar,
  9. sessionIDFromUrl,
  10. waitDir,
  11. waitSlug,
  12. } from "../actions"
  13. import { projectSwitchSelector, promptSelector, workspaceItemSelector, workspaceNewSessionSelector } from "../selectors"
  14. import { dirSlug, resolveDirectory } from "../utils"
  15. async function workspaces(page: Page, directory: string, enabled: boolean) {
  16. await page.evaluate(
  17. ({ directory, enabled }: { directory: string; enabled: boolean }) => {
  18. const key = "opencode.global.dat:layout"
  19. const raw = localStorage.getItem(key)
  20. const data = raw ? JSON.parse(raw) : {}
  21. const sidebar = data.sidebar && typeof data.sidebar === "object" ? data.sidebar : {}
  22. const current =
  23. sidebar.workspaces && typeof sidebar.workspaces === "object" && !Array.isArray(sidebar.workspaces)
  24. ? sidebar.workspaces
  25. : {}
  26. const next = { ...current }
  27. if (enabled) next[directory] = true
  28. if (!enabled) delete next[directory]
  29. localStorage.setItem(
  30. key,
  31. JSON.stringify({
  32. ...data,
  33. sidebar: {
  34. ...sidebar,
  35. workspaces: next,
  36. },
  37. }),
  38. )
  39. },
  40. { directory, enabled },
  41. )
  42. }
  43. test("can switch between projects from sidebar", async ({ page, withProject }) => {
  44. await page.setViewportSize({ width: 1400, height: 800 })
  45. const other = await createTestProject()
  46. const otherSlug = dirSlug(other)
  47. try {
  48. await withProject(
  49. async ({ directory }) => {
  50. await defocus(page)
  51. const currentSlug = dirSlug(directory)
  52. const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
  53. await expect(otherButton).toBeVisible()
  54. await otherButton.click()
  55. await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
  56. const currentButton = page.locator(projectSwitchSelector(currentSlug)).first()
  57. await expect(currentButton).toBeVisible()
  58. await currentButton.click()
  59. await expect(page).toHaveURL(new RegExp(`/${currentSlug}/session`))
  60. },
  61. { extra: [other] },
  62. )
  63. } finally {
  64. await cleanupTestProject(other)
  65. }
  66. })
  67. test("switching back to a project opens the latest workspace session", async ({ page, withProject }) => {
  68. await page.setViewportSize({ width: 1400, height: 800 })
  69. const other = await createTestProject()
  70. const otherSlug = dirSlug(other)
  71. try {
  72. await withProject(
  73. async ({ directory, slug, trackSession, trackDirectory }) => {
  74. await defocus(page)
  75. await workspaces(page, directory, true)
  76. await page.reload()
  77. await expect(page.locator(promptSelector)).toBeVisible()
  78. await openSidebar(page)
  79. await expect(page.getByRole("button", { name: "New workspace" }).first()).toBeVisible()
  80. await page.getByRole("button", { name: "New workspace" }).first().click()
  81. const raw = await waitSlug(page, [slug])
  82. const dir = base64Decode(raw)
  83. if (!dir) throw new Error(`Failed to decode workspace slug: ${raw}`)
  84. const space = await resolveDirectory(dir)
  85. const next = dirSlug(space)
  86. trackDirectory(space)
  87. await openSidebar(page)
  88. const item = page.locator(`${workspaceItemSelector(next)}, ${workspaceItemSelector(raw)}`).first()
  89. await expect(item).toBeVisible()
  90. await item.hover()
  91. const btn = page.locator(`${workspaceNewSessionSelector(next)}, ${workspaceNewSessionSelector(raw)}`).first()
  92. await expect(btn).toBeVisible()
  93. await btn.click({ force: true })
  94. await waitSlug(page)
  95. await waitDir(page, space)
  96. // Create a session by sending a prompt
  97. const prompt = page.locator(promptSelector)
  98. await expect(prompt).toBeVisible()
  99. await prompt.fill("test")
  100. await page.keyboard.press("Enter")
  101. // Wait for the URL to update with the new session ID
  102. await expect.poll(() => sessionIDFromUrl(page.url()) ?? "", { timeout: 15_000 }).not.toBe("")
  103. const created = sessionIDFromUrl(page.url())
  104. if (!created) throw new Error(`Failed to get session ID from url: ${page.url()}`)
  105. trackSession(created, space)
  106. await expect(page).toHaveURL(new RegExp(`/${next}/session/${created}(?:[/?#]|$)`))
  107. await openSidebar(page)
  108. const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
  109. await expect(otherButton).toBeVisible()
  110. await otherButton.click()
  111. await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
  112. const rootButton = page.locator(projectSwitchSelector(slug)).first()
  113. await expect(rootButton).toBeVisible()
  114. await rootButton.click()
  115. await waitDir(page, space)
  116. await expect.poll(() => sessionIDFromUrl(page.url()) ?? "").toBe(created)
  117. await expect(page).toHaveURL(new RegExp(`/session/${created}(?:[/?#]|$)`))
  118. },
  119. { extra: [other] },
  120. )
  121. } finally {
  122. await cleanupTestProject(other)
  123. }
  124. })