mirror of
https://gitea.toothfairyai.com/ToothFairyAI/tf_code.git
synced 2026-03-29 21:33:54 +00:00
150 lines
5.1 KiB
TypeScript
150 lines
5.1 KiB
TypeScript
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)
|
|
}
|
|
})
|