reigncode-app 1.3.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +30 -0
- package/Dockerfile +21 -0
- package/README.md +51 -0
- package/bunfig.toml +3 -0
- package/create-effect-simplification-spec.md +515 -0
- package/e2e/AGENTS.md +226 -0
- package/e2e/actions.ts +1018 -0
- package/e2e/app/home.spec.ts +24 -0
- package/e2e/app/navigation.spec.ts +10 -0
- package/e2e/app/palette.spec.ts +20 -0
- package/e2e/app/server-default.spec.ts +58 -0
- package/e2e/app/session.spec.ts +16 -0
- package/e2e/app/titlebar-history.spec.ts +120 -0
- package/e2e/commands/input-focus.spec.ts +15 -0
- package/e2e/commands/panels.spec.ts +33 -0
- package/e2e/commands/tab-close.spec.ts +32 -0
- package/e2e/files/file-open.spec.ts +31 -0
- package/e2e/files/file-tree.spec.ts +56 -0
- package/e2e/files/file-viewer.spec.ts +156 -0
- package/e2e/fixtures.ts +154 -0
- package/e2e/models/model-picker.spec.ts +48 -0
- package/e2e/models/models-visibility.spec.ts +61 -0
- package/e2e/projects/project-edit.spec.ts +43 -0
- package/e2e/projects/projects-close.spec.ts +54 -0
- package/e2e/projects/projects-switch.spec.ts +116 -0
- package/e2e/projects/workspace-new-session.spec.ts +94 -0
- package/e2e/projects/workspaces.spec.ts +375 -0
- package/e2e/prompt/context.spec.ts +95 -0
- package/e2e/prompt/prompt-async.spec.ts +76 -0
- package/e2e/prompt/prompt-drop-file-uri.spec.ts +22 -0
- package/e2e/prompt/prompt-drop-file.spec.ts +30 -0
- package/e2e/prompt/prompt-history.spec.ts +184 -0
- package/e2e/prompt/prompt-mention.spec.ts +26 -0
- package/e2e/prompt/prompt-multiline.spec.ts +24 -0
- package/e2e/prompt/prompt-shell.spec.ts +62 -0
- package/e2e/prompt/prompt-slash-open.spec.ts +22 -0
- package/e2e/prompt/prompt-slash-share.spec.ts +64 -0
- package/e2e/prompt/prompt-slash-terminal.spec.ts +18 -0
- package/e2e/prompt/prompt.spec.ts +55 -0
- package/e2e/selectors.ts +75 -0
- package/e2e/session/session-child-navigation.spec.ts +37 -0
- package/e2e/session/session-composer-dock.spec.ts +530 -0
- package/e2e/session/session-model-persistence.spec.ts +359 -0
- package/e2e/session/session-review.spec.ts +426 -0
- package/e2e/session/session-undo-redo.spec.ts +233 -0
- package/e2e/session/session.spec.ts +174 -0
- package/e2e/settings/settings-keybinds.spec.ts +389 -0
- package/e2e/settings/settings-models.spec.ts +122 -0
- package/e2e/settings/settings-providers.spec.ts +136 -0
- package/e2e/settings/settings.spec.ts +519 -0
- package/e2e/sidebar/sidebar-popover-actions.spec.ts +118 -0
- package/e2e/sidebar/sidebar-session-links.spec.ts +30 -0
- package/e2e/sidebar/sidebar.spec.ts +40 -0
- package/e2e/status/status-popover.spec.ts +94 -0
- package/e2e/terminal/terminal-init.spec.ts +28 -0
- package/e2e/terminal/terminal-reconnect.spec.ts +46 -0
- package/e2e/terminal/terminal-tabs.spec.ts +168 -0
- package/e2e/terminal/terminal.spec.ts +18 -0
- package/e2e/thinking-level.spec.ts +25 -0
- package/e2e/tsconfig.json +9 -0
- package/e2e/utils.ts +63 -0
- package/happydom.ts +75 -0
- package/index.html +23 -0
- package/package.json +77 -0
- package/playwright.config.ts +45 -0
- package/public/_headers +17 -0
- package/public/oc-theme-preload.js +35 -0
- package/script/e2e-local.ts +180 -0
- package/src/addons/serialize.test.ts +319 -0
- package/src/addons/serialize.ts +634 -0
- package/src/app.tsx +308 -0
- package/src/components/debug-bar.tsx +443 -0
- package/src/components/dialog-connect-provider.tsx +617 -0
- package/src/components/dialog-custom-provider-form.ts +158 -0
- package/src/components/dialog-custom-provider.test.ts +80 -0
- package/src/components/dialog-custom-provider.tsx +329 -0
- package/src/components/dialog-edit-project.tsx +255 -0
- package/src/components/dialog-fork.tsx +108 -0
- package/src/components/dialog-manage-models.tsx +101 -0
- package/src/components/dialog-release-notes.tsx +144 -0
- package/src/components/dialog-select-directory.tsx +392 -0
- package/src/components/dialog-select-file.tsx +466 -0
- package/src/components/dialog-select-mcp.tsx +107 -0
- package/src/components/dialog-select-model-unpaid.tsx +137 -0
- package/src/components/dialog-select-model.tsx +220 -0
- package/src/components/dialog-select-provider.tsx +86 -0
- package/src/components/dialog-select-server.tsx +649 -0
- package/src/components/dialog-settings.tsx +73 -0
- package/src/components/file-tree.test.ts +78 -0
- package/src/components/file-tree.tsx +507 -0
- package/src/components/link.tsx +26 -0
- package/src/components/model-tooltip.tsx +91 -0
- package/src/components/prompt-input/attachments.test.ts +44 -0
- package/src/components/prompt-input/attachments.ts +201 -0
- package/src/components/prompt-input/build-request-parts.test.ts +312 -0
- package/src/components/prompt-input/build-request-parts.ts +175 -0
- package/src/components/prompt-input/context-items.tsx +88 -0
- package/src/components/prompt-input/drag-overlay.tsx +25 -0
- package/src/components/prompt-input/editor-dom.test.ts +99 -0
- package/src/components/prompt-input/editor-dom.ts +148 -0
- package/src/components/prompt-input/files.ts +66 -0
- package/src/components/prompt-input/history.test.ts +153 -0
- package/src/components/prompt-input/history.ts +256 -0
- package/src/components/prompt-input/image-attachments.tsx +58 -0
- package/src/components/prompt-input/paste.ts +24 -0
- package/src/components/prompt-input/placeholder.test.ts +48 -0
- package/src/components/prompt-input/placeholder.ts +15 -0
- package/src/components/prompt-input/slash-popover.tsx +141 -0
- package/src/components/prompt-input/submit.test.ts +346 -0
- package/src/components/prompt-input/submit.ts +579 -0
- package/src/components/prompt-input.tsx +1595 -0
- package/src/components/server/server-row.tsx +130 -0
- package/src/components/session/index.ts +5 -0
- package/src/components/session/session-context-breakdown.test.ts +61 -0
- package/src/components/session/session-context-breakdown.ts +132 -0
- package/src/components/session/session-context-format.ts +20 -0
- package/src/components/session/session-context-metrics.test.ts +101 -0
- package/src/components/session/session-context-metrics.ts +82 -0
- package/src/components/session/session-context-tab.tsx +339 -0
- package/src/components/session/session-header.tsx +486 -0
- package/src/components/session/session-new-view.tsx +91 -0
- package/src/components/session/session-sortable-tab.tsx +70 -0
- package/src/components/session/session-sortable-terminal-tab.tsx +193 -0
- package/src/components/session-context-usage.tsx +122 -0
- package/src/components/settings-general.tsx +585 -0
- package/src/components/settings-keybinds.tsx +453 -0
- package/src/components/settings-list.tsx +5 -0
- package/src/components/settings-models.tsx +137 -0
- package/src/components/settings-providers.tsx +251 -0
- package/src/components/status-popover.tsx +419 -0
- package/src/components/terminal.tsx +653 -0
- package/src/components/titlebar-history.test.ts +63 -0
- package/src/components/titlebar-history.ts +57 -0
- package/src/components/titlebar.tsx +312 -0
- package/src/constants/file-picker.ts +89 -0
- package/src/context/command-keybind.test.ts +69 -0
- package/src/context/command.test.ts +25 -0
- package/src/context/command.tsx +437 -0
- package/src/context/comments.test.ts +186 -0
- package/src/context/comments.tsx +243 -0
- package/src/context/file/content-cache.ts +88 -0
- package/src/context/file/path.test.ts +360 -0
- package/src/context/file/path.ts +151 -0
- package/src/context/file/tree-store.ts +170 -0
- package/src/context/file/types.ts +41 -0
- package/src/context/file/view-cache.ts +146 -0
- package/src/context/file/watcher.test.ts +149 -0
- package/src/context/file/watcher.ts +53 -0
- package/src/context/file-content-eviction-accounting.test.ts +65 -0
- package/src/context/file.tsx +280 -0
- package/src/context/global-sdk.tsx +232 -0
- package/src/context/global-sync/bootstrap.ts +206 -0
- package/src/context/global-sync/child-store.test.ts +38 -0
- package/src/context/global-sync/child-store.ts +281 -0
- package/src/context/global-sync/event-reducer.test.ts +552 -0
- package/src/context/global-sync/event-reducer.ts +359 -0
- package/src/context/global-sync/eviction.ts +28 -0
- package/src/context/global-sync/queue.ts +83 -0
- package/src/context/global-sync/session-cache.test.ts +102 -0
- package/src/context/global-sync/session-cache.ts +62 -0
- package/src/context/global-sync/session-load.ts +25 -0
- package/src/context/global-sync/session-prefetch.test.ts +96 -0
- package/src/context/global-sync/session-prefetch.ts +100 -0
- package/src/context/global-sync/session-trim.test.ts +59 -0
- package/src/context/global-sync/session-trim.ts +56 -0
- package/src/context/global-sync/types.ts +133 -0
- package/src/context/global-sync/utils.ts +25 -0
- package/src/context/global-sync.test.ts +122 -0
- package/src/context/global-sync.tsx +408 -0
- package/src/context/highlights.tsx +233 -0
- package/src/context/language.tsx +248 -0
- package/src/context/layout-scroll.test.ts +64 -0
- package/src/context/layout-scroll.ts +126 -0
- package/src/context/layout.test.ts +69 -0
- package/src/context/layout.tsx +937 -0
- package/src/context/local.tsx +422 -0
- package/src/context/model-variant.test.ts +86 -0
- package/src/context/model-variant.ts +52 -0
- package/src/context/models.tsx +163 -0
- package/src/context/notification.tsx +373 -0
- package/src/context/permission-auto-respond.test.ts +102 -0
- package/src/context/permission-auto-respond.ts +51 -0
- package/src/context/permission.tsx +277 -0
- package/src/context/platform.tsx +99 -0
- package/src/context/prompt.tsx +297 -0
- package/src/context/sdk.tsx +49 -0
- package/src/context/server.tsx +295 -0
- package/src/context/settings.tsx +241 -0
- package/src/context/sync-optimistic.test.ts +123 -0
- package/src/context/sync.tsx +618 -0
- package/src/context/terminal-title.ts +51 -0
- package/src/context/terminal.test.ts +82 -0
- package/src/context/terminal.tsx +437 -0
- package/src/entry.tsx +144 -0
- package/src/env.d.ts +18 -0
- package/src/hooks/use-providers.ts +44 -0
- package/src/i18n/ar.ts +855 -0
- package/src/i18n/br.ts +867 -0
- package/src/i18n/bs.ts +943 -0
- package/src/i18n/da.ts +937 -0
- package/src/i18n/de.ts +879 -0
- package/src/i18n/en.ts +948 -0
- package/src/i18n/es.ts +950 -0
- package/src/i18n/fr.ts +878 -0
- package/src/i18n/ja.ts +861 -0
- package/src/i18n/ko.ts +860 -0
- package/src/i18n/no.ts +944 -0
- package/src/i18n/parity.test.ts +32 -0
- package/src/i18n/pl.ts +865 -0
- package/src/i18n/ru.ts +946 -0
- package/src/i18n/th.ts +933 -0
- package/src/i18n/tr.ts +952 -0
- package/src/i18n/zh.ts +930 -0
- package/src/i18n/zht.ts +925 -0
- package/src/index.css +29 -0
- package/src/index.ts +6 -0
- package/src/pages/directory-layout.tsx +88 -0
- package/src/pages/error.tsx +327 -0
- package/src/pages/home.tsx +131 -0
- package/src/pages/layout/deep-links.ts +50 -0
- package/src/pages/layout/helpers.test.ts +211 -0
- package/src/pages/layout/helpers.ts +98 -0
- package/src/pages/layout/inline-editor.tsx +126 -0
- package/src/pages/layout/sidebar-items.tsx +437 -0
- package/src/pages/layout/sidebar-project.tsx +384 -0
- package/src/pages/layout/sidebar-shell.tsx +125 -0
- package/src/pages/layout/sidebar-workspace.tsx +504 -0
- package/src/pages/layout.tsx +2509 -0
- package/src/pages/session/composer/index.ts +2 -0
- package/src/pages/session/composer/session-composer-region.tsx +255 -0
- package/src/pages/session/composer/session-composer-state.test.ts +128 -0
- package/src/pages/session/composer/session-composer-state.ts +249 -0
- package/src/pages/session/composer/session-followup-dock.tsx +109 -0
- package/src/pages/session/composer/session-permission-dock.tsx +74 -0
- package/src/pages/session/composer/session-question-dock.tsx +449 -0
- package/src/pages/session/composer/session-request-tree.ts +52 -0
- package/src/pages/session/composer/session-revert-dock.tsx +99 -0
- package/src/pages/session/composer/session-todo-dock.tsx +330 -0
- package/src/pages/session/file-tab-scroll.test.ts +40 -0
- package/src/pages/session/file-tab-scroll.ts +67 -0
- package/src/pages/session/file-tabs.tsx +456 -0
- package/src/pages/session/handoff.ts +36 -0
- package/src/pages/session/helpers.test.ts +181 -0
- package/src/pages/session/helpers.ts +198 -0
- package/src/pages/session/message-gesture.test.ts +62 -0
- package/src/pages/session/message-gesture.ts +21 -0
- package/src/pages/session/message-id-from-hash.ts +6 -0
- package/src/pages/session/message-timeline.tsx +1013 -0
- package/src/pages/session/review-tab.tsx +170 -0
- package/src/pages/session/session-layout.ts +20 -0
- package/src/pages/session/session-model-helpers.test.ts +51 -0
- package/src/pages/session/session-model-helpers.ts +16 -0
- package/src/pages/session/session-side-panel.tsx +453 -0
- package/src/pages/session/terminal-label.ts +16 -0
- package/src/pages/session/terminal-panel.test.ts +25 -0
- package/src/pages/session/terminal-panel.tsx +326 -0
- package/src/pages/session/use-session-commands.tsx +495 -0
- package/src/pages/session/use-session-hash-scroll.test.ts +16 -0
- package/src/pages/session/use-session-hash-scroll.ts +197 -0
- package/src/pages/session.tsx +1841 -0
- package/src/sst-env.d.ts +12 -0
- package/src/testing/model-selection.ts +80 -0
- package/src/testing/prompt.ts +56 -0
- package/src/testing/session-composer.ts +84 -0
- package/src/testing/terminal.ts +118 -0
- package/src/theme-preload.test.ts +46 -0
- package/src/utils/agent.ts +23 -0
- package/src/utils/aim.ts +138 -0
- package/src/utils/base64.ts +10 -0
- package/src/utils/comment-note.ts +88 -0
- package/src/utils/id.ts +99 -0
- package/src/utils/notification-click.test.ts +27 -0
- package/src/utils/notification-click.ts +13 -0
- package/src/utils/persist.test.ts +115 -0
- package/src/utils/persist.ts +476 -0
- package/src/utils/prompt.test.ts +44 -0
- package/src/utils/prompt.ts +203 -0
- package/src/utils/runtime-adapters.test.ts +62 -0
- package/src/utils/runtime-adapters.ts +39 -0
- package/src/utils/same.ts +6 -0
- package/src/utils/scoped-cache.test.ts +69 -0
- package/src/utils/scoped-cache.ts +104 -0
- package/src/utils/server-errors.test.ts +131 -0
- package/src/utils/server-errors.ts +80 -0
- package/src/utils/server-health.test.ts +123 -0
- package/src/utils/server-health.ts +91 -0
- package/src/utils/server.ts +22 -0
- package/src/utils/solid-dnd.tsx +49 -0
- package/src/utils/sound.ts +117 -0
- package/src/utils/terminal-writer.test.ts +64 -0
- package/src/utils/terminal-writer.ts +65 -0
- package/src/utils/time.ts +22 -0
- package/src/utils/uuid.test.ts +78 -0
- package/src/utils/uuid.ts +12 -0
- package/src/utils/worktree.test.ts +46 -0
- package/src/utils/worktree.ts +73 -0
- package/sst-env.d.ts +10 -0
- package/tsconfig.json +26 -0
- package/vite.config.ts +15 -0
- package/vite.js +26 -0
|
@@ -0,0 +1,359 @@
|
|
|
1
|
+
import type { Locator, Page } from "@playwright/test"
|
|
2
|
+
import { test, expect } from "../fixtures"
|
|
3
|
+
import {
|
|
4
|
+
openSidebar,
|
|
5
|
+
resolveSlug,
|
|
6
|
+
sessionIDFromUrl,
|
|
7
|
+
setWorkspacesEnabled,
|
|
8
|
+
waitSession,
|
|
9
|
+
waitSessionIdle,
|
|
10
|
+
waitSlug,
|
|
11
|
+
} from "../actions"
|
|
12
|
+
import {
|
|
13
|
+
promptAgentSelector,
|
|
14
|
+
promptModelSelector,
|
|
15
|
+
promptSelector,
|
|
16
|
+
promptVariantSelector,
|
|
17
|
+
workspaceItemSelector,
|
|
18
|
+
workspaceNewSessionSelector,
|
|
19
|
+
} from "../selectors"
|
|
20
|
+
import { createSdk, sessionPath } from "../utils"
|
|
21
|
+
|
|
22
|
+
type Footer = {
|
|
23
|
+
agent: string
|
|
24
|
+
model: string
|
|
25
|
+
variant: string
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
type Probe = {
|
|
29
|
+
dir?: string
|
|
30
|
+
sessionID?: string
|
|
31
|
+
model?: { providerID: string; modelID: string }
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const escape = (value: string) => value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&")
|
|
35
|
+
|
|
36
|
+
const text = async (locator: Locator) => ((await locator.textContent()) ?? "").trim()
|
|
37
|
+
|
|
38
|
+
const modelKey = (state: Probe | null) => (state?.model ? `${state.model.providerID}:${state.model.modelID}` : null)
|
|
39
|
+
|
|
40
|
+
async function probe(page: Page): Promise<Probe | null> {
|
|
41
|
+
return page.evaluate(() => {
|
|
42
|
+
const win = window as Window & {
|
|
43
|
+
__opencode_e2e?: {
|
|
44
|
+
model?: {
|
|
45
|
+
current?: Probe
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
return win.__opencode_e2e?.model?.current ?? null
|
|
50
|
+
})
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
async function read(page: Page): Promise<Footer> {
|
|
54
|
+
return {
|
|
55
|
+
agent: await text(page.locator(`${promptAgentSelector} [data-slot="select-select-trigger-value"]`).first()),
|
|
56
|
+
model: await text(page.locator(`${promptModelSelector} [data-action="prompt-model"] span`).first()),
|
|
57
|
+
variant: await text(page.locator(`${promptVariantSelector} [data-slot="select-select-trigger-value"]`).first()),
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
async function waitFooter(page: Page, expected: Partial<Footer>) {
|
|
62
|
+
let hit: Footer | null = null
|
|
63
|
+
await expect
|
|
64
|
+
.poll(
|
|
65
|
+
async () => {
|
|
66
|
+
const state = await read(page)
|
|
67
|
+
const ok = Object.entries(expected).every(([key, value]) => state[key as keyof Footer] === value)
|
|
68
|
+
if (ok) hit = state
|
|
69
|
+
return ok
|
|
70
|
+
},
|
|
71
|
+
{ timeout: 30_000 },
|
|
72
|
+
)
|
|
73
|
+
.toBe(true)
|
|
74
|
+
if (!hit) throw new Error("Failed to resolve prompt footer state")
|
|
75
|
+
return hit
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
async function waitModel(page: Page, value: string) {
|
|
79
|
+
await expect.poll(() => probe(page).then(modelKey), { timeout: 30_000 }).toBe(value)
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
async function choose(page: Page, root: string, value: string) {
|
|
83
|
+
const select = page.locator(root)
|
|
84
|
+
await expect(select).toBeVisible()
|
|
85
|
+
await select.locator('[data-action], [data-slot="select-select-trigger"]').first().click()
|
|
86
|
+
const item = page
|
|
87
|
+
.locator('[data-slot="select-select-item"]')
|
|
88
|
+
.filter({ hasText: new RegExp(`^\\s*${escape(value)}\\s*$`) })
|
|
89
|
+
.first()
|
|
90
|
+
await expect(item).toBeVisible()
|
|
91
|
+
await item.click()
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
async function variantCount(page: Page) {
|
|
95
|
+
const select = page.locator(promptVariantSelector)
|
|
96
|
+
await expect(select).toBeVisible()
|
|
97
|
+
await select.locator('[data-slot="select-select-trigger"]').click()
|
|
98
|
+
const count = await page.locator('[data-slot="select-select-item"]').count()
|
|
99
|
+
await page.keyboard.press("Escape")
|
|
100
|
+
return count
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
async function agents(page: Page) {
|
|
104
|
+
const select = page.locator(promptAgentSelector)
|
|
105
|
+
await expect(select).toBeVisible()
|
|
106
|
+
await select.locator('[data-action], [data-slot="select-select-trigger"]').first().click()
|
|
107
|
+
const labels = await page.locator('[data-slot="select-select-item-label"]').allTextContents()
|
|
108
|
+
await page.keyboard.press("Escape")
|
|
109
|
+
return labels.map((item) => item.trim()).filter(Boolean)
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
async function ensureVariant(page: Page, directory: string): Promise<Footer> {
|
|
113
|
+
const current = await read(page)
|
|
114
|
+
if ((await variantCount(page)) >= 2) return current
|
|
115
|
+
|
|
116
|
+
const cfg = await createSdk(directory)
|
|
117
|
+
.config.get()
|
|
118
|
+
.then((x) => x.data)
|
|
119
|
+
const visible = new Set(await agents(page))
|
|
120
|
+
const entry = Object.entries(cfg?.agent ?? {}).find((item) => {
|
|
121
|
+
const value = item[1]
|
|
122
|
+
return !!value && typeof value === "object" && "variant" in value && "model" in value && visible.has(item[0])
|
|
123
|
+
})
|
|
124
|
+
const name = entry?.[0]
|
|
125
|
+
test.skip(!name, "no agent with alternate variants available")
|
|
126
|
+
if (!name) return current
|
|
127
|
+
|
|
128
|
+
await choose(page, promptAgentSelector, name)
|
|
129
|
+
await expect.poll(() => variantCount(page), { timeout: 30_000 }).toBeGreaterThanOrEqual(2)
|
|
130
|
+
return waitFooter(page, { agent: name })
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
async function chooseDifferentVariant(page: Page): Promise<Footer> {
|
|
134
|
+
const current = await read(page)
|
|
135
|
+
const select = page.locator(promptVariantSelector)
|
|
136
|
+
await expect(select).toBeVisible()
|
|
137
|
+
await select.locator('[data-slot="select-select-trigger"]').click()
|
|
138
|
+
|
|
139
|
+
const items = page.locator('[data-slot="select-select-item"]')
|
|
140
|
+
const count = await items.count()
|
|
141
|
+
if (count < 2) throw new Error("Current model has no alternate variant to select")
|
|
142
|
+
|
|
143
|
+
for (let i = 0; i < count; i++) {
|
|
144
|
+
const item = items.nth(i)
|
|
145
|
+
const next = await text(item.locator('[data-slot="select-select-item-label"]').first())
|
|
146
|
+
if (!next || next === current.variant) continue
|
|
147
|
+
await item.click()
|
|
148
|
+
return waitFooter(page, { agent: current.agent, model: current.model, variant: next })
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
throw new Error("Failed to choose a different variant")
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
async function chooseOtherModel(page: Page): Promise<Footer> {
|
|
155
|
+
const current = await read(page)
|
|
156
|
+
const button = page.locator(`${promptModelSelector} [data-action="prompt-model"]`)
|
|
157
|
+
await expect(button).toBeVisible()
|
|
158
|
+
await button.click()
|
|
159
|
+
|
|
160
|
+
const dialog = page.getByRole("dialog")
|
|
161
|
+
await expect(dialog).toBeVisible()
|
|
162
|
+
const items = dialog.locator('[data-slot="list-item"]')
|
|
163
|
+
const count = await items.count()
|
|
164
|
+
expect(count).toBeGreaterThan(1)
|
|
165
|
+
|
|
166
|
+
for (let i = 0; i < count; i++) {
|
|
167
|
+
const item = items.nth(i)
|
|
168
|
+
const selected = (await item.getAttribute("data-selected")) === "true"
|
|
169
|
+
if (selected) continue
|
|
170
|
+
await item.click()
|
|
171
|
+
await expect(dialog).toHaveCount(0)
|
|
172
|
+
await expect.poll(async () => (await read(page)).model !== current.model, { timeout: 30_000 }).toBe(true)
|
|
173
|
+
return read(page)
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
throw new Error("Failed to choose a different model")
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
async function goto(page: Page, directory: string, sessionID?: string) {
|
|
180
|
+
await page.goto(sessionPath(directory, sessionID))
|
|
181
|
+
await waitSession(page, { directory, sessionID })
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
async function submit(page: Page, value: string) {
|
|
185
|
+
const prompt = page.locator(promptSelector)
|
|
186
|
+
await expect(prompt).toBeVisible()
|
|
187
|
+
await prompt.click()
|
|
188
|
+
await prompt.fill(value)
|
|
189
|
+
await prompt.press("Enter")
|
|
190
|
+
|
|
191
|
+
await expect.poll(() => sessionIDFromUrl(page.url()) ?? "", { timeout: 30_000 }).not.toBe("")
|
|
192
|
+
const id = sessionIDFromUrl(page.url())
|
|
193
|
+
if (!id) throw new Error(`Failed to resolve session id from ${page.url()}`)
|
|
194
|
+
return id
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
async function waitUser(directory: string, sessionID: string) {
|
|
198
|
+
const sdk = createSdk(directory)
|
|
199
|
+
await expect
|
|
200
|
+
.poll(
|
|
201
|
+
async () => {
|
|
202
|
+
const items = await sdk.session.messages({ sessionID, limit: 20 }).then((x) => x.data ?? [])
|
|
203
|
+
return items.some((item) => item.info.role === "user")
|
|
204
|
+
},
|
|
205
|
+
{ timeout: 30_000 },
|
|
206
|
+
)
|
|
207
|
+
.toBe(true)
|
|
208
|
+
await sdk.session.abort({ sessionID }).catch(() => undefined)
|
|
209
|
+
await waitSessionIdle(sdk, sessionID, 30_000).catch(() => undefined)
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
async function createWorkspace(page: Page, root: string, seen: string[]) {
|
|
213
|
+
await openSidebar(page)
|
|
214
|
+
await page.getByRole("button", { name: "New workspace" }).first().click()
|
|
215
|
+
|
|
216
|
+
const next = await resolveSlug(await waitSlug(page, [root, ...seen]))
|
|
217
|
+
await waitSession(page, { directory: next.directory })
|
|
218
|
+
return next
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
async function waitWorkspace(page: Page, slug: string) {
|
|
222
|
+
await openSidebar(page)
|
|
223
|
+
await expect
|
|
224
|
+
.poll(
|
|
225
|
+
async () => {
|
|
226
|
+
const item = page.locator(workspaceItemSelector(slug)).first()
|
|
227
|
+
try {
|
|
228
|
+
await item.hover({ timeout: 500 })
|
|
229
|
+
return true
|
|
230
|
+
} catch {
|
|
231
|
+
return false
|
|
232
|
+
}
|
|
233
|
+
},
|
|
234
|
+
{ timeout: 60_000 },
|
|
235
|
+
)
|
|
236
|
+
.toBe(true)
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
async function newWorkspaceSession(page: Page, slug: string) {
|
|
240
|
+
await waitWorkspace(page, slug)
|
|
241
|
+
const item = page.locator(workspaceItemSelector(slug)).first()
|
|
242
|
+
await item.hover()
|
|
243
|
+
|
|
244
|
+
const button = page.locator(workspaceNewSessionSelector(slug)).first()
|
|
245
|
+
await expect(button).toBeVisible()
|
|
246
|
+
await button.click({ force: true })
|
|
247
|
+
|
|
248
|
+
const next = await resolveSlug(await waitSlug(page))
|
|
249
|
+
return waitSession(page, { directory: next.directory }).then((item) => item.directory)
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
test("session model and variant restore per session without leaking into new sessions", async ({
|
|
253
|
+
page,
|
|
254
|
+
withProject,
|
|
255
|
+
}) => {
|
|
256
|
+
await page.setViewportSize({ width: 1440, height: 900 })
|
|
257
|
+
|
|
258
|
+
await withProject(async ({ directory, gotoSession, trackSession }) => {
|
|
259
|
+
await gotoSession()
|
|
260
|
+
|
|
261
|
+
await ensureVariant(page, directory)
|
|
262
|
+
const firstState = await chooseDifferentVariant(page)
|
|
263
|
+
const first = await submit(page, `session variant ${Date.now()}`)
|
|
264
|
+
trackSession(first)
|
|
265
|
+
await waitUser(directory, first)
|
|
266
|
+
|
|
267
|
+
await page.reload()
|
|
268
|
+
await waitSession(page, { directory, sessionID: first })
|
|
269
|
+
await waitFooter(page, firstState)
|
|
270
|
+
|
|
271
|
+
await gotoSession()
|
|
272
|
+
const fresh = await ensureVariant(page, directory)
|
|
273
|
+
expect(fresh.variant).not.toBe(firstState.variant)
|
|
274
|
+
|
|
275
|
+
const secondState = await chooseOtherModel(page)
|
|
276
|
+
const second = await submit(page, `session model ${Date.now()}`)
|
|
277
|
+
trackSession(second)
|
|
278
|
+
await waitUser(directory, second)
|
|
279
|
+
|
|
280
|
+
await goto(page, directory, first)
|
|
281
|
+
await waitFooter(page, firstState)
|
|
282
|
+
|
|
283
|
+
await goto(page, directory, second)
|
|
284
|
+
await waitFooter(page, secondState)
|
|
285
|
+
|
|
286
|
+
await gotoSession()
|
|
287
|
+
await waitFooter(page, fresh)
|
|
288
|
+
})
|
|
289
|
+
})
|
|
290
|
+
|
|
291
|
+
test("session model restore across workspaces", async ({ page, withProject }) => {
|
|
292
|
+
await page.setViewportSize({ width: 1440, height: 900 })
|
|
293
|
+
|
|
294
|
+
await withProject(async ({ directory: root, slug, gotoSession, trackDirectory, trackSession }) => {
|
|
295
|
+
await gotoSession()
|
|
296
|
+
|
|
297
|
+
await ensureVariant(page, root)
|
|
298
|
+
const firstState = await chooseDifferentVariant(page)
|
|
299
|
+
const first = await submit(page, `root session ${Date.now()}`)
|
|
300
|
+
trackSession(first, root)
|
|
301
|
+
await waitUser(root, first)
|
|
302
|
+
|
|
303
|
+
await openSidebar(page)
|
|
304
|
+
await setWorkspacesEnabled(page, slug, true)
|
|
305
|
+
|
|
306
|
+
const one = await createWorkspace(page, slug, [])
|
|
307
|
+
const oneDir = await newWorkspaceSession(page, one.slug)
|
|
308
|
+
trackDirectory(oneDir)
|
|
309
|
+
|
|
310
|
+
const secondState = await chooseOtherModel(page)
|
|
311
|
+
const second = await submit(page, `workspace one ${Date.now()}`)
|
|
312
|
+
trackSession(second, oneDir)
|
|
313
|
+
await waitUser(oneDir, second)
|
|
314
|
+
|
|
315
|
+
const two = await createWorkspace(page, slug, [one.slug])
|
|
316
|
+
const twoDir = await newWorkspaceSession(page, two.slug)
|
|
317
|
+
trackDirectory(twoDir)
|
|
318
|
+
|
|
319
|
+
await ensureVariant(page, twoDir)
|
|
320
|
+
const thirdState = await chooseDifferentVariant(page)
|
|
321
|
+
const third = await submit(page, `workspace two ${Date.now()}`)
|
|
322
|
+
trackSession(third, twoDir)
|
|
323
|
+
await waitUser(twoDir, third)
|
|
324
|
+
|
|
325
|
+
await goto(page, root, first)
|
|
326
|
+
await waitFooter(page, firstState)
|
|
327
|
+
|
|
328
|
+
await goto(page, oneDir, second)
|
|
329
|
+
await waitFooter(page, secondState)
|
|
330
|
+
|
|
331
|
+
await goto(page, twoDir, third)
|
|
332
|
+
await waitFooter(page, thirdState)
|
|
333
|
+
|
|
334
|
+
await goto(page, root, first)
|
|
335
|
+
await waitFooter(page, firstState)
|
|
336
|
+
})
|
|
337
|
+
})
|
|
338
|
+
|
|
339
|
+
test("variant preserved when switching agent modes", async ({ page, withProject }) => {
|
|
340
|
+
await page.setViewportSize({ width: 1440, height: 900 })
|
|
341
|
+
|
|
342
|
+
await withProject(async ({ directory, gotoSession }) => {
|
|
343
|
+
await gotoSession()
|
|
344
|
+
|
|
345
|
+
await ensureVariant(page, directory)
|
|
346
|
+
const updated = await chooseDifferentVariant(page)
|
|
347
|
+
|
|
348
|
+
const available = await agents(page)
|
|
349
|
+
const other = available.find((name) => name !== updated.agent)
|
|
350
|
+
test.skip(!other, "only one agent available")
|
|
351
|
+
if (!other) return
|
|
352
|
+
|
|
353
|
+
await choose(page, promptAgentSelector, other)
|
|
354
|
+
await waitFooter(page, { agent: other, variant: updated.variant })
|
|
355
|
+
|
|
356
|
+
await choose(page, promptAgentSelector, updated.agent)
|
|
357
|
+
await waitFooter(page, { agent: updated.agent, variant: updated.variant })
|
|
358
|
+
})
|
|
359
|
+
})
|