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.
Files changed (300) hide show
  1. package/AGENTS.md +30 -0
  2. package/Dockerfile +21 -0
  3. package/README.md +51 -0
  4. package/bunfig.toml +3 -0
  5. package/create-effect-simplification-spec.md +515 -0
  6. package/e2e/AGENTS.md +226 -0
  7. package/e2e/actions.ts +1018 -0
  8. package/e2e/app/home.spec.ts +24 -0
  9. package/e2e/app/navigation.spec.ts +10 -0
  10. package/e2e/app/palette.spec.ts +20 -0
  11. package/e2e/app/server-default.spec.ts +58 -0
  12. package/e2e/app/session.spec.ts +16 -0
  13. package/e2e/app/titlebar-history.spec.ts +120 -0
  14. package/e2e/commands/input-focus.spec.ts +15 -0
  15. package/e2e/commands/panels.spec.ts +33 -0
  16. package/e2e/commands/tab-close.spec.ts +32 -0
  17. package/e2e/files/file-open.spec.ts +31 -0
  18. package/e2e/files/file-tree.spec.ts +56 -0
  19. package/e2e/files/file-viewer.spec.ts +156 -0
  20. package/e2e/fixtures.ts +154 -0
  21. package/e2e/models/model-picker.spec.ts +48 -0
  22. package/e2e/models/models-visibility.spec.ts +61 -0
  23. package/e2e/projects/project-edit.spec.ts +43 -0
  24. package/e2e/projects/projects-close.spec.ts +54 -0
  25. package/e2e/projects/projects-switch.spec.ts +116 -0
  26. package/e2e/projects/workspace-new-session.spec.ts +94 -0
  27. package/e2e/projects/workspaces.spec.ts +375 -0
  28. package/e2e/prompt/context.spec.ts +95 -0
  29. package/e2e/prompt/prompt-async.spec.ts +76 -0
  30. package/e2e/prompt/prompt-drop-file-uri.spec.ts +22 -0
  31. package/e2e/prompt/prompt-drop-file.spec.ts +30 -0
  32. package/e2e/prompt/prompt-history.spec.ts +184 -0
  33. package/e2e/prompt/prompt-mention.spec.ts +26 -0
  34. package/e2e/prompt/prompt-multiline.spec.ts +24 -0
  35. package/e2e/prompt/prompt-shell.spec.ts +62 -0
  36. package/e2e/prompt/prompt-slash-open.spec.ts +22 -0
  37. package/e2e/prompt/prompt-slash-share.spec.ts +64 -0
  38. package/e2e/prompt/prompt-slash-terminal.spec.ts +18 -0
  39. package/e2e/prompt/prompt.spec.ts +55 -0
  40. package/e2e/selectors.ts +75 -0
  41. package/e2e/session/session-child-navigation.spec.ts +37 -0
  42. package/e2e/session/session-composer-dock.spec.ts +530 -0
  43. package/e2e/session/session-model-persistence.spec.ts +359 -0
  44. package/e2e/session/session-review.spec.ts +426 -0
  45. package/e2e/session/session-undo-redo.spec.ts +233 -0
  46. package/e2e/session/session.spec.ts +174 -0
  47. package/e2e/settings/settings-keybinds.spec.ts +389 -0
  48. package/e2e/settings/settings-models.spec.ts +122 -0
  49. package/e2e/settings/settings-providers.spec.ts +136 -0
  50. package/e2e/settings/settings.spec.ts +519 -0
  51. package/e2e/sidebar/sidebar-popover-actions.spec.ts +118 -0
  52. package/e2e/sidebar/sidebar-session-links.spec.ts +30 -0
  53. package/e2e/sidebar/sidebar.spec.ts +40 -0
  54. package/e2e/status/status-popover.spec.ts +94 -0
  55. package/e2e/terminal/terminal-init.spec.ts +28 -0
  56. package/e2e/terminal/terminal-reconnect.spec.ts +46 -0
  57. package/e2e/terminal/terminal-tabs.spec.ts +168 -0
  58. package/e2e/terminal/terminal.spec.ts +18 -0
  59. package/e2e/thinking-level.spec.ts +25 -0
  60. package/e2e/tsconfig.json +9 -0
  61. package/e2e/utils.ts +63 -0
  62. package/happydom.ts +75 -0
  63. package/index.html +23 -0
  64. package/package.json +77 -0
  65. package/playwright.config.ts +45 -0
  66. package/public/_headers +17 -0
  67. package/public/oc-theme-preload.js +35 -0
  68. package/script/e2e-local.ts +180 -0
  69. package/src/addons/serialize.test.ts +319 -0
  70. package/src/addons/serialize.ts +634 -0
  71. package/src/app.tsx +308 -0
  72. package/src/components/debug-bar.tsx +443 -0
  73. package/src/components/dialog-connect-provider.tsx +617 -0
  74. package/src/components/dialog-custom-provider-form.ts +158 -0
  75. package/src/components/dialog-custom-provider.test.ts +80 -0
  76. package/src/components/dialog-custom-provider.tsx +329 -0
  77. package/src/components/dialog-edit-project.tsx +255 -0
  78. package/src/components/dialog-fork.tsx +108 -0
  79. package/src/components/dialog-manage-models.tsx +101 -0
  80. package/src/components/dialog-release-notes.tsx +144 -0
  81. package/src/components/dialog-select-directory.tsx +392 -0
  82. package/src/components/dialog-select-file.tsx +466 -0
  83. package/src/components/dialog-select-mcp.tsx +107 -0
  84. package/src/components/dialog-select-model-unpaid.tsx +137 -0
  85. package/src/components/dialog-select-model.tsx +220 -0
  86. package/src/components/dialog-select-provider.tsx +86 -0
  87. package/src/components/dialog-select-server.tsx +649 -0
  88. package/src/components/dialog-settings.tsx +73 -0
  89. package/src/components/file-tree.test.ts +78 -0
  90. package/src/components/file-tree.tsx +507 -0
  91. package/src/components/link.tsx +26 -0
  92. package/src/components/model-tooltip.tsx +91 -0
  93. package/src/components/prompt-input/attachments.test.ts +44 -0
  94. package/src/components/prompt-input/attachments.ts +201 -0
  95. package/src/components/prompt-input/build-request-parts.test.ts +312 -0
  96. package/src/components/prompt-input/build-request-parts.ts +175 -0
  97. package/src/components/prompt-input/context-items.tsx +88 -0
  98. package/src/components/prompt-input/drag-overlay.tsx +25 -0
  99. package/src/components/prompt-input/editor-dom.test.ts +99 -0
  100. package/src/components/prompt-input/editor-dom.ts +148 -0
  101. package/src/components/prompt-input/files.ts +66 -0
  102. package/src/components/prompt-input/history.test.ts +153 -0
  103. package/src/components/prompt-input/history.ts +256 -0
  104. package/src/components/prompt-input/image-attachments.tsx +58 -0
  105. package/src/components/prompt-input/paste.ts +24 -0
  106. package/src/components/prompt-input/placeholder.test.ts +48 -0
  107. package/src/components/prompt-input/placeholder.ts +15 -0
  108. package/src/components/prompt-input/slash-popover.tsx +141 -0
  109. package/src/components/prompt-input/submit.test.ts +346 -0
  110. package/src/components/prompt-input/submit.ts +579 -0
  111. package/src/components/prompt-input.tsx +1595 -0
  112. package/src/components/server/server-row.tsx +130 -0
  113. package/src/components/session/index.ts +5 -0
  114. package/src/components/session/session-context-breakdown.test.ts +61 -0
  115. package/src/components/session/session-context-breakdown.ts +132 -0
  116. package/src/components/session/session-context-format.ts +20 -0
  117. package/src/components/session/session-context-metrics.test.ts +101 -0
  118. package/src/components/session/session-context-metrics.ts +82 -0
  119. package/src/components/session/session-context-tab.tsx +339 -0
  120. package/src/components/session/session-header.tsx +486 -0
  121. package/src/components/session/session-new-view.tsx +91 -0
  122. package/src/components/session/session-sortable-tab.tsx +70 -0
  123. package/src/components/session/session-sortable-terminal-tab.tsx +193 -0
  124. package/src/components/session-context-usage.tsx +122 -0
  125. package/src/components/settings-general.tsx +585 -0
  126. package/src/components/settings-keybinds.tsx +453 -0
  127. package/src/components/settings-list.tsx +5 -0
  128. package/src/components/settings-models.tsx +137 -0
  129. package/src/components/settings-providers.tsx +251 -0
  130. package/src/components/status-popover.tsx +419 -0
  131. package/src/components/terminal.tsx +653 -0
  132. package/src/components/titlebar-history.test.ts +63 -0
  133. package/src/components/titlebar-history.ts +57 -0
  134. package/src/components/titlebar.tsx +312 -0
  135. package/src/constants/file-picker.ts +89 -0
  136. package/src/context/command-keybind.test.ts +69 -0
  137. package/src/context/command.test.ts +25 -0
  138. package/src/context/command.tsx +437 -0
  139. package/src/context/comments.test.ts +186 -0
  140. package/src/context/comments.tsx +243 -0
  141. package/src/context/file/content-cache.ts +88 -0
  142. package/src/context/file/path.test.ts +360 -0
  143. package/src/context/file/path.ts +151 -0
  144. package/src/context/file/tree-store.ts +170 -0
  145. package/src/context/file/types.ts +41 -0
  146. package/src/context/file/view-cache.ts +146 -0
  147. package/src/context/file/watcher.test.ts +149 -0
  148. package/src/context/file/watcher.ts +53 -0
  149. package/src/context/file-content-eviction-accounting.test.ts +65 -0
  150. package/src/context/file.tsx +280 -0
  151. package/src/context/global-sdk.tsx +232 -0
  152. package/src/context/global-sync/bootstrap.ts +206 -0
  153. package/src/context/global-sync/child-store.test.ts +38 -0
  154. package/src/context/global-sync/child-store.ts +281 -0
  155. package/src/context/global-sync/event-reducer.test.ts +552 -0
  156. package/src/context/global-sync/event-reducer.ts +359 -0
  157. package/src/context/global-sync/eviction.ts +28 -0
  158. package/src/context/global-sync/queue.ts +83 -0
  159. package/src/context/global-sync/session-cache.test.ts +102 -0
  160. package/src/context/global-sync/session-cache.ts +62 -0
  161. package/src/context/global-sync/session-load.ts +25 -0
  162. package/src/context/global-sync/session-prefetch.test.ts +96 -0
  163. package/src/context/global-sync/session-prefetch.ts +100 -0
  164. package/src/context/global-sync/session-trim.test.ts +59 -0
  165. package/src/context/global-sync/session-trim.ts +56 -0
  166. package/src/context/global-sync/types.ts +133 -0
  167. package/src/context/global-sync/utils.ts +25 -0
  168. package/src/context/global-sync.test.ts +122 -0
  169. package/src/context/global-sync.tsx +408 -0
  170. package/src/context/highlights.tsx +233 -0
  171. package/src/context/language.tsx +248 -0
  172. package/src/context/layout-scroll.test.ts +64 -0
  173. package/src/context/layout-scroll.ts +126 -0
  174. package/src/context/layout.test.ts +69 -0
  175. package/src/context/layout.tsx +937 -0
  176. package/src/context/local.tsx +422 -0
  177. package/src/context/model-variant.test.ts +86 -0
  178. package/src/context/model-variant.ts +52 -0
  179. package/src/context/models.tsx +163 -0
  180. package/src/context/notification.tsx +373 -0
  181. package/src/context/permission-auto-respond.test.ts +102 -0
  182. package/src/context/permission-auto-respond.ts +51 -0
  183. package/src/context/permission.tsx +277 -0
  184. package/src/context/platform.tsx +99 -0
  185. package/src/context/prompt.tsx +297 -0
  186. package/src/context/sdk.tsx +49 -0
  187. package/src/context/server.tsx +295 -0
  188. package/src/context/settings.tsx +241 -0
  189. package/src/context/sync-optimistic.test.ts +123 -0
  190. package/src/context/sync.tsx +618 -0
  191. package/src/context/terminal-title.ts +51 -0
  192. package/src/context/terminal.test.ts +82 -0
  193. package/src/context/terminal.tsx +437 -0
  194. package/src/entry.tsx +144 -0
  195. package/src/env.d.ts +18 -0
  196. package/src/hooks/use-providers.ts +44 -0
  197. package/src/i18n/ar.ts +855 -0
  198. package/src/i18n/br.ts +867 -0
  199. package/src/i18n/bs.ts +943 -0
  200. package/src/i18n/da.ts +937 -0
  201. package/src/i18n/de.ts +879 -0
  202. package/src/i18n/en.ts +948 -0
  203. package/src/i18n/es.ts +950 -0
  204. package/src/i18n/fr.ts +878 -0
  205. package/src/i18n/ja.ts +861 -0
  206. package/src/i18n/ko.ts +860 -0
  207. package/src/i18n/no.ts +944 -0
  208. package/src/i18n/parity.test.ts +32 -0
  209. package/src/i18n/pl.ts +865 -0
  210. package/src/i18n/ru.ts +946 -0
  211. package/src/i18n/th.ts +933 -0
  212. package/src/i18n/tr.ts +952 -0
  213. package/src/i18n/zh.ts +930 -0
  214. package/src/i18n/zht.ts +925 -0
  215. package/src/index.css +29 -0
  216. package/src/index.ts +6 -0
  217. package/src/pages/directory-layout.tsx +88 -0
  218. package/src/pages/error.tsx +327 -0
  219. package/src/pages/home.tsx +131 -0
  220. package/src/pages/layout/deep-links.ts +50 -0
  221. package/src/pages/layout/helpers.test.ts +211 -0
  222. package/src/pages/layout/helpers.ts +98 -0
  223. package/src/pages/layout/inline-editor.tsx +126 -0
  224. package/src/pages/layout/sidebar-items.tsx +437 -0
  225. package/src/pages/layout/sidebar-project.tsx +384 -0
  226. package/src/pages/layout/sidebar-shell.tsx +125 -0
  227. package/src/pages/layout/sidebar-workspace.tsx +504 -0
  228. package/src/pages/layout.tsx +2509 -0
  229. package/src/pages/session/composer/index.ts +2 -0
  230. package/src/pages/session/composer/session-composer-region.tsx +255 -0
  231. package/src/pages/session/composer/session-composer-state.test.ts +128 -0
  232. package/src/pages/session/composer/session-composer-state.ts +249 -0
  233. package/src/pages/session/composer/session-followup-dock.tsx +109 -0
  234. package/src/pages/session/composer/session-permission-dock.tsx +74 -0
  235. package/src/pages/session/composer/session-question-dock.tsx +449 -0
  236. package/src/pages/session/composer/session-request-tree.ts +52 -0
  237. package/src/pages/session/composer/session-revert-dock.tsx +99 -0
  238. package/src/pages/session/composer/session-todo-dock.tsx +330 -0
  239. package/src/pages/session/file-tab-scroll.test.ts +40 -0
  240. package/src/pages/session/file-tab-scroll.ts +67 -0
  241. package/src/pages/session/file-tabs.tsx +456 -0
  242. package/src/pages/session/handoff.ts +36 -0
  243. package/src/pages/session/helpers.test.ts +181 -0
  244. package/src/pages/session/helpers.ts +198 -0
  245. package/src/pages/session/message-gesture.test.ts +62 -0
  246. package/src/pages/session/message-gesture.ts +21 -0
  247. package/src/pages/session/message-id-from-hash.ts +6 -0
  248. package/src/pages/session/message-timeline.tsx +1013 -0
  249. package/src/pages/session/review-tab.tsx +170 -0
  250. package/src/pages/session/session-layout.ts +20 -0
  251. package/src/pages/session/session-model-helpers.test.ts +51 -0
  252. package/src/pages/session/session-model-helpers.ts +16 -0
  253. package/src/pages/session/session-side-panel.tsx +453 -0
  254. package/src/pages/session/terminal-label.ts +16 -0
  255. package/src/pages/session/terminal-panel.test.ts +25 -0
  256. package/src/pages/session/terminal-panel.tsx +326 -0
  257. package/src/pages/session/use-session-commands.tsx +495 -0
  258. package/src/pages/session/use-session-hash-scroll.test.ts +16 -0
  259. package/src/pages/session/use-session-hash-scroll.ts +197 -0
  260. package/src/pages/session.tsx +1841 -0
  261. package/src/sst-env.d.ts +12 -0
  262. package/src/testing/model-selection.ts +80 -0
  263. package/src/testing/prompt.ts +56 -0
  264. package/src/testing/session-composer.ts +84 -0
  265. package/src/testing/terminal.ts +118 -0
  266. package/src/theme-preload.test.ts +46 -0
  267. package/src/utils/agent.ts +23 -0
  268. package/src/utils/aim.ts +138 -0
  269. package/src/utils/base64.ts +10 -0
  270. package/src/utils/comment-note.ts +88 -0
  271. package/src/utils/id.ts +99 -0
  272. package/src/utils/notification-click.test.ts +27 -0
  273. package/src/utils/notification-click.ts +13 -0
  274. package/src/utils/persist.test.ts +115 -0
  275. package/src/utils/persist.ts +476 -0
  276. package/src/utils/prompt.test.ts +44 -0
  277. package/src/utils/prompt.ts +203 -0
  278. package/src/utils/runtime-adapters.test.ts +62 -0
  279. package/src/utils/runtime-adapters.ts +39 -0
  280. package/src/utils/same.ts +6 -0
  281. package/src/utils/scoped-cache.test.ts +69 -0
  282. package/src/utils/scoped-cache.ts +104 -0
  283. package/src/utils/server-errors.test.ts +131 -0
  284. package/src/utils/server-errors.ts +80 -0
  285. package/src/utils/server-health.test.ts +123 -0
  286. package/src/utils/server-health.ts +91 -0
  287. package/src/utils/server.ts +22 -0
  288. package/src/utils/solid-dnd.tsx +49 -0
  289. package/src/utils/sound.ts +117 -0
  290. package/src/utils/terminal-writer.test.ts +64 -0
  291. package/src/utils/terminal-writer.ts +65 -0
  292. package/src/utils/time.ts +22 -0
  293. package/src/utils/uuid.test.ts +78 -0
  294. package/src/utils/uuid.ts +12 -0
  295. package/src/utils/worktree.test.ts +46 -0
  296. package/src/utils/worktree.ts +73 -0
  297. package/sst-env.d.ts +10 -0
  298. package/tsconfig.json +26 -0
  299. package/vite.config.ts +15 -0
  300. 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
+ })