| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149 |
- import { base64Decode } from "@opencode-ai/util/encode"
- import type { Page } from "@playwright/test"
- import { test, expect } from "../fixtures"
- import {
- defocus,
- createTestProject,
- cleanupTestProject,
- openSidebar,
- sessionIDFromUrl,
- waitDir,
- waitSlug,
- } from "../actions"
- import { projectSwitchSelector, promptSelector, workspaceItemSelector, workspaceNewSessionSelector } from "../selectors"
- import { dirSlug, resolveDirectory } from "../utils"
- async function workspaces(page: Page, directory: string, enabled: boolean) {
- await page.evaluate(
- ({ directory, enabled }: { directory: string; enabled: boolean }) => {
- const key = "opencode.global.dat:layout"
- const raw = localStorage.getItem(key)
- const data = raw ? JSON.parse(raw) : {}
- const sidebar = data.sidebar && typeof data.sidebar === "object" ? data.sidebar : {}
- const current =
- sidebar.workspaces && typeof sidebar.workspaces === "object" && !Array.isArray(sidebar.workspaces)
- ? sidebar.workspaces
- : {}
- const next = { ...current }
- if (enabled) next[directory] = true
- if (!enabled) delete next[directory]
- localStorage.setItem(
- key,
- JSON.stringify({
- ...data,
- sidebar: {
- ...sidebar,
- workspaces: next,
- },
- }),
- )
- },
- { directory, enabled },
- )
- }
- test("can switch between projects from sidebar", async ({ page, withProject }) => {
- await page.setViewportSize({ width: 1400, height: 800 })
- const other = await createTestProject()
- const otherSlug = dirSlug(other)
- try {
- await withProject(
- async ({ directory }) => {
- await defocus(page)
- const currentSlug = dirSlug(directory)
- const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
- await expect(otherButton).toBeVisible()
- await otherButton.click()
- await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
- const currentButton = page.locator(projectSwitchSelector(currentSlug)).first()
- await expect(currentButton).toBeVisible()
- await currentButton.click()
- await expect(page).toHaveURL(new RegExp(`/${currentSlug}/session`))
- },
- { extra: [other] },
- )
- } finally {
- await cleanupTestProject(other)
- }
- })
- test("switching back to a project opens the latest workspace session", async ({ page, withProject }) => {
- await page.setViewportSize({ width: 1400, height: 800 })
- const other = await createTestProject()
- const otherSlug = dirSlug(other)
- try {
- await withProject(
- async ({ directory, slug, trackSession, trackDirectory }) => {
- await defocus(page)
- await workspaces(page, directory, true)
- await page.reload()
- await expect(page.locator(promptSelector)).toBeVisible()
- await openSidebar(page)
- await expect(page.getByRole("button", { name: "New workspace" }).first()).toBeVisible()
- await page.getByRole("button", { name: "New workspace" }).first().click()
- const raw = await waitSlug(page, [slug])
- const dir = base64Decode(raw)
- if (!dir) throw new Error(`Failed to decode workspace slug: ${raw}`)
- const space = await resolveDirectory(dir)
- const next = dirSlug(space)
- trackDirectory(space)
- await openSidebar(page)
- const item = page.locator(`${workspaceItemSelector(next)}, ${workspaceItemSelector(raw)}`).first()
- await expect(item).toBeVisible()
- await item.hover()
- const btn = page.locator(`${workspaceNewSessionSelector(next)}, ${workspaceNewSessionSelector(raw)}`).first()
- await expect(btn).toBeVisible()
- await btn.click({ force: true })
- await waitSlug(page)
- await waitDir(page, space)
- // Create a session by sending a prompt
- const prompt = page.locator(promptSelector)
- await expect(prompt).toBeVisible()
- await prompt.fill("test")
- await page.keyboard.press("Enter")
- // Wait for the URL to update with the new session ID
- await expect.poll(() => sessionIDFromUrl(page.url()) ?? "", { timeout: 15_000 }).not.toBe("")
- const created = sessionIDFromUrl(page.url())
- if (!created) throw new Error(`Failed to get session ID from url: ${page.url()}`)
- trackSession(created, space)
- await expect(page).toHaveURL(new RegExp(`/${next}/session/${created}(?:[/?#]|$)`))
- await openSidebar(page)
- const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
- await expect(otherButton).toBeVisible()
- await otherButton.click()
- await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
- const rootButton = page.locator(projectSwitchSelector(slug)).first()
- await expect(rootButton).toBeVisible()
- await rootButton.click()
- await waitDir(page, space)
- await expect.poll(() => sessionIDFromUrl(page.url()) ?? "").toBe(created)
- await expect(page).toHaveURL(new RegExp(`/session/${created}(?:[/?#]|$)`))
- },
- { extra: [other] },
- )
- } finally {
- await cleanupTestProject(other)
- }
- })
|