gsd-pi 2.73.0 → 2.73.1-dev.6ddfa43

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 (224) hide show
  1. package/dist/cli.js +0 -47
  2. package/dist/help-text.js +1 -1
  3. package/dist/resources/extensions/claude-code-cli/stream-adapter.js +9 -3
  4. package/dist/resources/extensions/gsd/auto-dispatch.js +5 -3
  5. package/dist/resources/extensions/gsd/auto-model-selection.js +54 -11
  6. package/dist/resources/extensions/gsd/auto-prompts.js +9 -6
  7. package/dist/resources/extensions/gsd/auto-start.js +20 -6
  8. package/dist/resources/extensions/gsd/auto.js +5 -1
  9. package/dist/resources/extensions/gsd/bootstrap/crash-log.js +31 -0
  10. package/dist/resources/extensions/gsd/bootstrap/register-extension.js +18 -7
  11. package/dist/resources/extensions/gsd/bootstrap/system-context.js +6 -1
  12. package/dist/resources/extensions/gsd/crash-recovery.js +51 -0
  13. package/dist/resources/extensions/gsd/gsd-db.js +36 -2
  14. package/dist/resources/extensions/gsd/milestone-actions.js +19 -1
  15. package/dist/resources/extensions/gsd/preferences-models.js +43 -0
  16. package/dist/resources/extensions/gsd/preferences-types.js +1 -0
  17. package/dist/resources/extensions/gsd/preferences-validation.js +22 -0
  18. package/dist/startup-model-validation.js +8 -5
  19. package/dist/web/standalone/.next/BUILD_ID +1 -1
  20. package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
  21. package/dist/web/standalone/.next/build-manifest.json +3 -3
  22. package/dist/web/standalone/.next/prerender-manifest.json +3 -3
  23. package/dist/web/standalone/.next/required-server-files.json +3 -3
  24. package/dist/web/standalone/.next/server/app/_global-error/page.js +3 -3
  25. package/dist/web/standalone/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  26. package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
  27. package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
  28. package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  29. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  30. package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  31. package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  32. package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  33. package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  34. package/dist/web/standalone/.next/server/app/_not-found/page.js +2 -2
  35. package/dist/web/standalone/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  36. package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
  37. package/dist/web/standalone/.next/server/app/_not-found.rsc +3 -3
  38. package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +3 -3
  39. package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  40. package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +3 -3
  41. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  42. package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  43. package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
  44. package/dist/web/standalone/.next/server/app/api/boot/route.js +1 -1
  45. package/dist/web/standalone/.next/server/app/api/boot/route_client-reference-manifest.js +1 -1
  46. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route.js +1 -1
  47. package/dist/web/standalone/.next/server/app/api/bridge-terminal/input/route_client-reference-manifest.js +1 -1
  48. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route.js +1 -1
  49. package/dist/web/standalone/.next/server/app/api/bridge-terminal/resize/route_client-reference-manifest.js +1 -1
  50. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route.js +2 -2
  51. package/dist/web/standalone/.next/server/app/api/bridge-terminal/stream/route_client-reference-manifest.js +1 -1
  52. package/dist/web/standalone/.next/server/app/api/browse-directories/route.js +1 -1
  53. package/dist/web/standalone/.next/server/app/api/browse-directories/route_client-reference-manifest.js +1 -1
  54. package/dist/web/standalone/.next/server/app/api/captures/route.js +1 -1
  55. package/dist/web/standalone/.next/server/app/api/captures/route_client-reference-manifest.js +1 -1
  56. package/dist/web/standalone/.next/server/app/api/cleanup/route.js +1 -1
  57. package/dist/web/standalone/.next/server/app/api/cleanup/route_client-reference-manifest.js +1 -1
  58. package/dist/web/standalone/.next/server/app/api/dev-mode/route.js +1 -1
  59. package/dist/web/standalone/.next/server/app/api/dev-mode/route_client-reference-manifest.js +1 -1
  60. package/dist/web/standalone/.next/server/app/api/doctor/route.js +1 -1
  61. package/dist/web/standalone/.next/server/app/api/doctor/route_client-reference-manifest.js +1 -1
  62. package/dist/web/standalone/.next/server/app/api/experimental/route.js +2 -2
  63. package/dist/web/standalone/.next/server/app/api/experimental/route_client-reference-manifest.js +1 -1
  64. package/dist/web/standalone/.next/server/app/api/export-data/route.js +1 -1
  65. package/dist/web/standalone/.next/server/app/api/export-data/route_client-reference-manifest.js +1 -1
  66. package/dist/web/standalone/.next/server/app/api/files/route.js +1 -1
  67. package/dist/web/standalone/.next/server/app/api/files/route_client-reference-manifest.js +1 -1
  68. package/dist/web/standalone/.next/server/app/api/forensics/route.js +1 -1
  69. package/dist/web/standalone/.next/server/app/api/forensics/route_client-reference-manifest.js +1 -1
  70. package/dist/web/standalone/.next/server/app/api/git/route.js +1 -1
  71. package/dist/web/standalone/.next/server/app/api/git/route_client-reference-manifest.js +1 -1
  72. package/dist/web/standalone/.next/server/app/api/history/route.js +1 -1
  73. package/dist/web/standalone/.next/server/app/api/history/route_client-reference-manifest.js +1 -1
  74. package/dist/web/standalone/.next/server/app/api/hooks/route.js +1 -1
  75. package/dist/web/standalone/.next/server/app/api/hooks/route_client-reference-manifest.js +1 -1
  76. package/dist/web/standalone/.next/server/app/api/inspect/route.js +1 -1
  77. package/dist/web/standalone/.next/server/app/api/inspect/route_client-reference-manifest.js +1 -1
  78. package/dist/web/standalone/.next/server/app/api/knowledge/route.js +1 -1
  79. package/dist/web/standalone/.next/server/app/api/knowledge/route_client-reference-manifest.js +1 -1
  80. package/dist/web/standalone/.next/server/app/api/live-state/route.js +1 -1
  81. package/dist/web/standalone/.next/server/app/api/live-state/route_client-reference-manifest.js +1 -1
  82. package/dist/web/standalone/.next/server/app/api/notifications/route.js +2 -2
  83. package/dist/web/standalone/.next/server/app/api/notifications/route_client-reference-manifest.js +1 -1
  84. package/dist/web/standalone/.next/server/app/api/onboarding/route.js +1 -1
  85. package/dist/web/standalone/.next/server/app/api/onboarding/route_client-reference-manifest.js +1 -1
  86. package/dist/web/standalone/.next/server/app/api/preferences/route.js +1 -1
  87. package/dist/web/standalone/.next/server/app/api/preferences/route_client-reference-manifest.js +1 -1
  88. package/dist/web/standalone/.next/server/app/api/projects/route.js +1 -1
  89. package/dist/web/standalone/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  90. package/dist/web/standalone/.next/server/app/api/recovery/route.js +1 -1
  91. package/dist/web/standalone/.next/server/app/api/recovery/route_client-reference-manifest.js +1 -1
  92. package/dist/web/standalone/.next/server/app/api/remote-questions/route.js +2 -2
  93. package/dist/web/standalone/.next/server/app/api/remote-questions/route_client-reference-manifest.js +1 -1
  94. package/dist/web/standalone/.next/server/app/api/session/browser/route.js +1 -1
  95. package/dist/web/standalone/.next/server/app/api/session/browser/route_client-reference-manifest.js +1 -1
  96. package/dist/web/standalone/.next/server/app/api/session/command/route.js +1 -1
  97. package/dist/web/standalone/.next/server/app/api/session/command/route_client-reference-manifest.js +1 -1
  98. package/dist/web/standalone/.next/server/app/api/session/events/route.js +2 -2
  99. package/dist/web/standalone/.next/server/app/api/session/events/route_client-reference-manifest.js +1 -1
  100. package/dist/web/standalone/.next/server/app/api/session/manage/route.js +1 -1
  101. package/dist/web/standalone/.next/server/app/api/session/manage/route_client-reference-manifest.js +1 -1
  102. package/dist/web/standalone/.next/server/app/api/settings-data/route.js +1 -1
  103. package/dist/web/standalone/.next/server/app/api/settings-data/route_client-reference-manifest.js +1 -1
  104. package/dist/web/standalone/.next/server/app/api/shutdown/route.js +1 -1
  105. package/dist/web/standalone/.next/server/app/api/shutdown/route_client-reference-manifest.js +1 -1
  106. package/dist/web/standalone/.next/server/app/api/skill-health/route.js +1 -1
  107. package/dist/web/standalone/.next/server/app/api/skill-health/route_client-reference-manifest.js +1 -1
  108. package/dist/web/standalone/.next/server/app/api/steer/route.js +1 -1
  109. package/dist/web/standalone/.next/server/app/api/steer/route_client-reference-manifest.js +1 -1
  110. package/dist/web/standalone/.next/server/app/api/switch-root/route.js +1 -1
  111. package/dist/web/standalone/.next/server/app/api/switch-root/route_client-reference-manifest.js +1 -1
  112. package/dist/web/standalone/.next/server/app/api/terminal/input/route.js +2 -2
  113. package/dist/web/standalone/.next/server/app/api/terminal/input/route_client-reference-manifest.js +1 -1
  114. package/dist/web/standalone/.next/server/app/api/terminal/resize/route.js +2 -2
  115. package/dist/web/standalone/.next/server/app/api/terminal/resize/route_client-reference-manifest.js +1 -1
  116. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route.js +2 -2
  117. package/dist/web/standalone/.next/server/app/api/terminal/sessions/route_client-reference-manifest.js +1 -1
  118. package/dist/web/standalone/.next/server/app/api/terminal/stream/route.js +4 -4
  119. package/dist/web/standalone/.next/server/app/api/terminal/stream/route_client-reference-manifest.js +1 -1
  120. package/dist/web/standalone/.next/server/app/api/terminal/upload/route.js +1 -1
  121. package/dist/web/standalone/.next/server/app/api/terminal/upload/route_client-reference-manifest.js +1 -1
  122. package/dist/web/standalone/.next/server/app/api/undo/route.js +1 -1
  123. package/dist/web/standalone/.next/server/app/api/undo/route_client-reference-manifest.js +1 -1
  124. package/dist/web/standalone/.next/server/app/api/update/route.js +1 -1
  125. package/dist/web/standalone/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  126. package/dist/web/standalone/.next/server/app/api/visualizer/route.js +1 -1
  127. package/dist/web/standalone/.next/server/app/api/visualizer/route_client-reference-manifest.js +1 -1
  128. package/dist/web/standalone/.next/server/app/index.html +1 -1
  129. package/dist/web/standalone/.next/server/app/index.rsc +4 -4
  130. package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  131. package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +4 -4
  132. package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
  133. package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +3 -3
  134. package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
  135. package/dist/web/standalone/.next/server/app/page.js +2 -2
  136. package/dist/web/standalone/.next/server/app/page_client-reference-manifest.js +1 -1
  137. package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
  138. package/dist/web/standalone/.next/server/chunks/63.js +3 -3
  139. package/dist/web/standalone/.next/server/chunks/6897.js +1 -1
  140. package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
  141. package/dist/web/standalone/.next/server/middleware-manifest.json +5 -5
  142. package/dist/web/standalone/.next/server/middleware.js +2 -2
  143. package/dist/web/standalone/.next/server/next-font-manifest.js +1 -1
  144. package/dist/web/standalone/.next/server/next-font-manifest.json +1 -1
  145. package/dist/web/standalone/.next/server/pages/404.html +1 -1
  146. package/dist/web/standalone/.next/server/pages/500.html +1 -1
  147. package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
  148. package/dist/web/standalone/.next/static/chunks/app/_not-found/{page-2f24283c162b6ab3.js → page-f2a7482d42a5614b.js} +1 -1
  149. package/dist/web/standalone/.next/static/chunks/app/{layout-9ecfd95f343793f0.js → layout-a16c7a7ecdf0c2cf.js} +1 -1
  150. package/dist/web/standalone/.next/static/chunks/app/page-f1e30ab6bb269149.js +1 -0
  151. package/dist/web/standalone/.next/static/chunks/main-app-fdab67f7802d7832.js +1 -0
  152. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-459824ffb8c323dd.js +1 -0
  153. package/dist/web/standalone/node_modules/node-pty/build/Makefile +2 -2
  154. package/dist/web/standalone/node_modules/node-pty/build/Release/pty.node +0 -0
  155. package/dist/web/standalone/node_modules/node-pty/build/pty.target.mk +14 -14
  156. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api.target.mk +14 -14
  157. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_except.target.mk +14 -14
  158. package/dist/web/standalone/node_modules/node-pty/node-addon-api/node_addon_api_maybe.target.mk +14 -14
  159. package/dist/web/standalone/server.js +1 -1
  160. package/package.json +1 -1
  161. package/packages/pi-ai/dist/index.d.ts +1 -0
  162. package/packages/pi-ai/dist/index.d.ts.map +1 -1
  163. package/packages/pi-ai/dist/index.js +1 -0
  164. package/packages/pi-ai/dist/index.js.map +1 -1
  165. package/packages/pi-ai/src/index.ts +4 -0
  166. package/packages/pi-coding-agent/dist/core/auth-storage.js +1 -1
  167. package/packages/pi-coding-agent/dist/core/auth-storage.js.map +1 -1
  168. package/packages/pi-coding-agent/dist/core/auth-storage.test.js +27 -0
  169. package/packages/pi-coding-agent/dist/core/auth-storage.test.js.map +1 -1
  170. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js +175 -8
  171. package/packages/pi-coding-agent/dist/core/chat-controller-ordering.test.js.map +1 -1
  172. package/packages/pi-coding-agent/dist/core/model-resolver.d.ts.map +1 -1
  173. package/packages/pi-coding-agent/dist/core/model-resolver.js +25 -68
  174. package/packages/pi-coding-agent/dist/core/model-resolver.js.map +1 -1
  175. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts +12 -2
  176. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.d.ts.map +1 -1
  177. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js +51 -26
  178. package/packages/pi-coding-agent/dist/modes/interactive/components/assistant-message.js.map +1 -1
  179. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts +1 -0
  180. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.d.ts.map +1 -1
  181. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js +95 -21
  182. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.js.map +1 -1
  183. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.d.ts +2 -0
  184. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.d.ts.map +1 -0
  185. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.js +63 -0
  186. package/packages/pi-coding-agent/dist/modes/interactive/controllers/chat-controller.test.js.map +1 -0
  187. package/packages/pi-coding-agent/package.json +1 -1
  188. package/packages/pi-coding-agent/src/core/auth-storage.test.ts +38 -0
  189. package/packages/pi-coding-agent/src/core/auth-storage.ts +1 -1
  190. package/packages/pi-coding-agent/src/core/chat-controller-ordering.test.ts +198 -8
  191. package/packages/pi-coding-agent/src/core/model-resolver.ts +26 -70
  192. package/packages/pi-coding-agent/src/modes/interactive/components/assistant-message.ts +62 -26
  193. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.test.ts +71 -0
  194. package/packages/pi-coding-agent/src/modes/interactive/controllers/chat-controller.ts +115 -26
  195. package/pkg/package.json +1 -1
  196. package/src/resources/extensions/claude-code-cli/stream-adapter.ts +12 -4
  197. package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +23 -2
  198. package/src/resources/extensions/gsd/auto-dispatch.ts +5 -0
  199. package/src/resources/extensions/gsd/auto-model-selection.ts +85 -11
  200. package/src/resources/extensions/gsd/auto-prompts.ts +9 -3
  201. package/src/resources/extensions/gsd/auto-start.ts +27 -6
  202. package/src/resources/extensions/gsd/auto.ts +5 -0
  203. package/src/resources/extensions/gsd/bootstrap/crash-log.ts +32 -0
  204. package/src/resources/extensions/gsd/bootstrap/register-extension.ts +19 -7
  205. package/src/resources/extensions/gsd/bootstrap/system-context.ts +8 -1
  206. package/src/resources/extensions/gsd/crash-recovery.ts +59 -0
  207. package/src/resources/extensions/gsd/gsd-db.ts +52 -2
  208. package/src/resources/extensions/gsd/milestone-actions.ts +19 -1
  209. package/src/resources/extensions/gsd/preferences-models.ts +41 -0
  210. package/src/resources/extensions/gsd/preferences-types.ts +12 -0
  211. package/src/resources/extensions/gsd/preferences-validation.ts +23 -0
  212. package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +51 -2
  213. package/src/resources/extensions/gsd/tests/crash-handler-secondary.test.ts +235 -0
  214. package/src/resources/extensions/gsd/tests/flat-rate-routing-guard.test.ts +137 -1
  215. package/src/resources/extensions/gsd/tests/gsd-db.test.ts +59 -1
  216. package/src/resources/extensions/gsd/tests/model-isolation.test.ts +91 -2
  217. package/src/resources/extensions/gsd/tests/park-milestone.test.ts +64 -0
  218. package/src/resources/extensions/gsd/tests/preferences.test.ts +47 -0
  219. package/src/resources/extensions/gsd/tests/subagent-model-dispatch.test.ts +267 -0
  220. package/dist/web/standalone/.next/static/chunks/app/page-7115e62689b5fd84.js +0 -1
  221. package/dist/web/standalone/.next/static/chunks/main-app-d3d4c336195465f9.js +0 -1
  222. package/dist/web/standalone/.next/static/chunks/next/dist/client/components/builtin/global-error-ab5a8926e07ec673.js +0 -1
  223. /package/dist/web/standalone/.next/static/{KSZ2dcC3p4z6lOmUpPpzr → r6AvNu-aMwn4nwqjHqAfw}/_buildManifest.js +0 -0
  224. /package/dist/web/standalone/.next/static/{KSZ2dcC3p4z6lOmUpPpzr → r6AvNu-aMwn4nwqjHqAfw}/_ssgManifest.js +0 -0
@@ -1,6 +1,8 @@
1
1
  /**
2
- * Tests for model config isolation between concurrent instances (#650, #1065)
3
- * and session-scoped model precedence behavior.
2
+ * Tests for model config isolation between concurrent instances (#650, #1065),
3
+ * session-scoped model precedence behavior including manual session override,
4
+ * GSD preferences override of settings.json defaults (#3517), and custom
5
+ * provider precedence over PREFERENCES.md when set via `/gsd model` (#4122).
4
6
  */
5
7
 
6
8
  import { describe, it, beforeEach, afterEach } from "node:test";
@@ -214,3 +216,90 @@ describe("manual session model override precedence", () => {
214
216
  "should be null when no model source is available");
215
217
  });
216
218
  });
219
+
220
+ // ─── Custom provider session model wins over PREFERENCES.md (#4122) ─────────
221
+
222
+ describe("custom provider session model overrides PREFERENCES.md (#4122)", () => {
223
+ // Mirrors the auto-start.ts logic:
224
+ // sessionProviderIsCustom && ctx.model
225
+ // ? ctx.model
226
+ // : (preferredModel ?? ctx.model ?? null)
227
+ function selectStartModel(args: {
228
+ ctxModel: { provider: string; id: string } | null;
229
+ preferredModel: { provider: string; id: string } | undefined;
230
+ sessionProviderIsCustom: boolean;
231
+ }): { provider: string; id: string } | null {
232
+ const { ctxModel, preferredModel, sessionProviderIsCustom } = args;
233
+ if (sessionProviderIsCustom && ctxModel) {
234
+ return { provider: ctxModel.provider, id: ctxModel.id };
235
+ }
236
+ return preferredModel
237
+ ?? (ctxModel ? { provider: ctxModel.provider, id: ctxModel.id } : null);
238
+ }
239
+
240
+ it("custom provider from /gsd model wins over PREFERENCES.md built-in default", () => {
241
+ // User runs `/gsd model ollama/llama3.1:8b`, then `/gsd auto`.
242
+ // PREFERENCES.md still has the project-template claude-code default.
243
+ const ctxModel = { provider: "ollama", id: "llama3.1:8b" };
244
+ const preferredModel = { provider: "claude-code", id: "claude-sonnet-4-6" };
245
+
246
+ const snapshot = selectStartModel({
247
+ ctxModel,
248
+ preferredModel,
249
+ sessionProviderIsCustom: true,
250
+ });
251
+
252
+ assert.equal(snapshot?.provider, "ollama",
253
+ "custom-provider session model must win over PREFERENCES.md");
254
+ assert.equal(snapshot?.id, "llama3.1:8b",
255
+ "custom-provider session model id must be preserved");
256
+ assert.notEqual(snapshot?.provider, "claude-code",
257
+ "claude-code from PREFERENCES.md must NOT be selected when session is custom");
258
+ });
259
+
260
+ it("built-in session provider still defers to PREFERENCES.md (#3517 preserved)", () => {
261
+ // ctx.model is a built-in provider (claude-code) but PREFERENCES.md has
262
+ // an explicit openai-codex preference. PREFERENCES.md should still win.
263
+ const ctxModel = { provider: "claude-code", id: "claude-sonnet-4-6" };
264
+ const preferredModel = { provider: "openai-codex", id: "gpt-5.4" };
265
+
266
+ const snapshot = selectStartModel({
267
+ ctxModel,
268
+ preferredModel,
269
+ sessionProviderIsCustom: false,
270
+ });
271
+
272
+ assert.equal(snapshot?.provider, "openai-codex",
273
+ "PREFERENCES.md must still win when session provider is built-in");
274
+ assert.equal(snapshot?.id, "gpt-5.4");
275
+ });
276
+
277
+ it("custom provider with no PREFERENCES.md still uses ctx.model", () => {
278
+ const ctxModel = { provider: "vllm", id: "qwen2.5-coder:32b" };
279
+
280
+ const snapshot = selectStartModel({
281
+ ctxModel,
282
+ preferredModel: undefined,
283
+ sessionProviderIsCustom: true,
284
+ });
285
+
286
+ assert.equal(snapshot?.provider, "vllm");
287
+ assert.equal(snapshot?.id, "qwen2.5-coder:32b");
288
+ });
289
+
290
+ it("null ctx.model with custom flag falls through to preferredModel", () => {
291
+ // Defensive: sessionProviderIsCustom can only be true if ctx.model exists,
292
+ // but verify the guard works if that invariant is ever broken.
293
+ const preferredModel = { provider: "claude-code", id: "claude-sonnet-4-6" };
294
+
295
+ const snapshot = selectStartModel({
296
+ ctxModel: null,
297
+ preferredModel,
298
+ sessionProviderIsCustom: true,
299
+ });
300
+
301
+ assert.equal(snapshot?.provider, "claude-code",
302
+ "should fall back to preferredModel when ctx.model is null");
303
+ });
304
+ });
305
+
@@ -3,10 +3,22 @@ import assert from 'node:assert/strict';
3
3
  import { mkdtempSync, mkdirSync, rmSync, writeFileSync, existsSync, readFileSync } from 'node:fs';
4
4
  import { join } from 'node:path';
5
5
  import { tmpdir } from 'node:os';
6
+ import { execSync } from 'node:child_process';
6
7
 
7
8
  import { deriveState, invalidateStateCache, getActiveMilestoneId } from '../state.ts';
8
9
  import { clearPathCache } from '../paths.ts';
9
10
  import { parkMilestone, unparkMilestone, discardMilestone, isParked, getParkedReason } from '../milestone-actions.ts';
11
+ import {
12
+ closeDatabase,
13
+ getMilestone,
14
+ getMilestoneSlices,
15
+ getSliceTasks,
16
+ insertMilestone,
17
+ insertSlice,
18
+ insertTask,
19
+ openDatabase,
20
+ } from "../gsd-db.ts";
21
+ import { createWorktree } from "../worktree-manager.ts";
10
22
 
11
23
 
12
24
 
@@ -60,9 +72,29 @@ function createMilestone(base: string, mid: string, opts?: { withRoadmap?: boole
60
72
  }
61
73
 
62
74
  function cleanup(base: string): void {
75
+ try {
76
+ closeDatabase();
77
+ } catch {
78
+ // ignore
79
+ }
63
80
  rmSync(base, { recursive: true, force: true });
64
81
  }
65
82
 
83
+ function run(cmd: string, cwd: string): string {
84
+ return execSync(cmd, { cwd, stdio: ["ignore", "pipe", "pipe"], encoding: "utf-8" }).trim();
85
+ }
86
+
87
+ function initGitRepo(base: string): void {
88
+ writeFileSync(join(base, "README.md"), "# test\n", "utf-8");
89
+ writeFileSync(join(base, ".gsd", "STATE.md"), "# State\n", "utf-8");
90
+ run("git init", base);
91
+ run("git config user.email test@test.com", base);
92
+ run("git config user.name Test", base);
93
+ run("git add .", base);
94
+ run('git commit -m "init"', base);
95
+ run("git branch -M main", base);
96
+ }
97
+
66
98
  function clearCaches(): void {
67
99
  clearPathCache();
68
100
  invalidateStateCache();
@@ -294,6 +326,38 @@ test('discardMilestone updates queue order', () => {
294
326
  }
295
327
  });
296
328
 
329
+ test('discardMilestone removes DB rows, worktree, and milestone branch', () => {
330
+ const base = createFixtureBase();
331
+ try {
332
+ createMilestone(base, 'M001', { withRoadmap: true });
333
+ initGitRepo(base);
334
+ clearCaches();
335
+
336
+ assert.ok(openDatabase(join(base, '.gsd', 'gsd.db')), 'database opens');
337
+ insertMilestone({ id: 'M001', title: 'Discard me', status: 'active' });
338
+ insertSlice({ milestoneId: 'M001', id: 'S01', title: 'Only slice', status: 'pending' });
339
+ insertTask({ milestoneId: 'M001', sliceId: 'S01', id: 'T01', title: 'Only task', status: 'pending' });
340
+
341
+ const wt = createWorktree(base, 'M001', { branch: 'milestone/M001' });
342
+ assert.ok(existsSync(wt.path), 'worktree exists before discard');
343
+ assert.ok(run('git branch', base).includes('milestone/M001'), 'milestone branch exists before discard');
344
+ assert.ok(getMilestone('M001'), 'milestone exists in DB before discard');
345
+ assert.equal(getMilestoneSlices('M001').length, 1, 'slice exists in DB before discard');
346
+ assert.equal(getSliceTasks('M001', 'S01').length, 1, 'task exists in DB before discard');
347
+
348
+ const success = discardMilestone(base, 'M001');
349
+ assert.ok(success, 'discardMilestone returns true');
350
+
351
+ assert.equal(getMilestone('M001'), null, 'milestone row removed from DB');
352
+ assert.equal(getMilestoneSlices('M001').length, 0, 'slice rows removed from DB');
353
+ assert.equal(getSliceTasks('M001', 'S01').length, 0, 'task rows removed from DB');
354
+ assert.ok(!existsSync(wt.path), 'worktree removed after discard');
355
+ assert.ok(!run('git branch', base).includes('milestone/M001'), 'milestone branch removed after discard');
356
+ } finally {
357
+ cleanup(base);
358
+ }
359
+ });
360
+
297
361
  // ─── Test 12: All milestones parked → no active milestone ─────────────
298
362
  test('All milestones parked → no active', async () => {
299
363
  const base = createFixtureBase();
@@ -134,6 +134,53 @@ test("invalid value types produce errors and fall back to undefined", () => {
134
134
  }
135
135
  });
136
136
 
137
+ test("flat_rate_providers: accepts string array", () => {
138
+ const { errors, preferences } = validatePreferences({
139
+ flat_rate_providers: ["my-proxy", "private-cli"],
140
+ });
141
+ assert.equal(errors.length, 0);
142
+ assert.deepEqual(preferences.flat_rate_providers, ["my-proxy", "private-cli"]);
143
+ });
144
+
145
+ test("flat_rate_providers: trims whitespace and drops empty entries", () => {
146
+ const { errors, preferences } = validatePreferences({
147
+ flat_rate_providers: [" my-proxy ", "", " ", "private-cli"],
148
+ });
149
+ assert.equal(errors.length, 0);
150
+ assert.deepEqual(preferences.flat_rate_providers, ["my-proxy", "private-cli"]);
151
+ });
152
+
153
+ test("flat_rate_providers: non-array rejected", () => {
154
+ const { errors } = validatePreferences({
155
+ flat_rate_providers: "my-proxy" as any,
156
+ });
157
+ assert.ok(
158
+ errors.some(e => e.includes("flat_rate_providers")),
159
+ "should error on non-array value",
160
+ );
161
+ });
162
+
163
+ test("flat_rate_providers: non-string elements rejected", () => {
164
+ const { errors } = validatePreferences({
165
+ flat_rate_providers: ["ok", 123 as any, "also-ok"],
166
+ });
167
+ assert.ok(
168
+ errors.some(e => e.includes("flat_rate_providers")),
169
+ "should error when array contains non-strings",
170
+ );
171
+ });
172
+
173
+ test("flat_rate_providers is a recognized preference key (no warning)", () => {
174
+ const { warnings } = validatePreferences({
175
+ flat_rate_providers: ["my-proxy"],
176
+ });
177
+ assert.equal(
178
+ warnings.filter(w => w.includes("flat_rate_providers")).length,
179
+ 0,
180
+ "flat_rate_providers must be in KNOWN_PREFERENCE_KEYS",
181
+ );
182
+ });
183
+
137
184
  test("valid values pass through correctly", () => {
138
185
  const { preferences: p1 } = validatePreferences({ budget_enforcement: "halt" });
139
186
  assert.equal(p1.budget_enforcement, "halt");
@@ -0,0 +1,267 @@
1
+ /**
2
+ * Regression tests for subagent model preference wiring.
3
+ *
4
+ * Fixes: subagent_model config in reactive_execution was validated and stored
5
+ * but never passed through to subagent dispatch instruction strings, so the
6
+ * executing agent autonomously chose "sonnet" instead of the configured model.
7
+ *
8
+ * Issue: gsd-build/gsd-2#4078
9
+ */
10
+
11
+ import test from "node:test";
12
+ import assert from "node:assert/strict";
13
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync, readFileSync } from "node:fs";
14
+ import { join, dirname } from "node:path";
15
+ import { tmpdir } from "node:os";
16
+ import { fileURLToPath } from "node:url";
17
+ import { validatePreferences } from "../preferences-validation.ts";
18
+
19
+ const __dirname = dirname(fileURLToPath(import.meta.url));
20
+ const promptsSrc = readFileSync(join(__dirname, "..", "auto-prompts.ts"), "utf-8");
21
+ const dispatchSrc = readFileSync(join(__dirname, "..", "auto-dispatch.ts"), "utf-8");
22
+
23
+ // ─── Preference Validation ────────────────────────────────────────────────
24
+
25
+ test("reactive_execution: subagent_model is preserved in validated preferences", () => {
26
+ const result = validatePreferences({
27
+ reactive_execution: {
28
+ enabled: true,
29
+ max_parallel: 2,
30
+ isolation_mode: "same-tree",
31
+ subagent_model: "claude-opus-4-6",
32
+ },
33
+ });
34
+ assert.equal(result.errors.length, 0);
35
+ assert.equal(
36
+ result.preferences.reactive_execution?.subagent_model,
37
+ "claude-opus-4-6",
38
+ "subagent_model should be preserved through validation",
39
+ );
40
+ });
41
+
42
+ test("reactive_execution: subagent_model rejects empty string", () => {
43
+ const result = validatePreferences({
44
+ reactive_execution: {
45
+ enabled: true,
46
+ max_parallel: 2,
47
+ isolation_mode: "same-tree",
48
+ subagent_model: "",
49
+ } as any,
50
+ });
51
+ assert.ok(
52
+ result.errors.some((e) => e.includes("subagent_model")),
53
+ "empty subagent_model should produce a validation error",
54
+ );
55
+ });
56
+
57
+ // ─── Structural: Prompt Builders Accept subagentModel ────────────────────
58
+
59
+ test("buildReactiveExecutePrompt: accepts subagentModel parameter", () => {
60
+ const fnStart = promptsSrc.indexOf("export async function buildReactiveExecutePrompt");
61
+ assert.ok(fnStart !== -1, "buildReactiveExecutePrompt should be exported");
62
+ const signature = promptsSrc.slice(fnStart, fnStart + 300);
63
+ assert.ok(
64
+ signature.includes("subagentModel"),
65
+ "buildReactiveExecutePrompt should accept a subagentModel parameter",
66
+ );
67
+ });
68
+
69
+ test("buildParallelResearchSlicesPrompt: accepts subagentModel parameter", () => {
70
+ const fnStart = promptsSrc.indexOf("export async function buildParallelResearchSlicesPrompt");
71
+ assert.ok(fnStart !== -1, "buildParallelResearchSlicesPrompt should be exported");
72
+ const signature = promptsSrc.slice(fnStart, fnStart + 300);
73
+ assert.ok(
74
+ signature.includes("subagentModel"),
75
+ "buildParallelResearchSlicesPrompt should accept a subagentModel parameter",
76
+ );
77
+ });
78
+
79
+ test("buildGateEvaluatePrompt: accepts subagentModel parameter", () => {
80
+ const fnStart = promptsSrc.indexOf("export async function buildGateEvaluatePrompt");
81
+ assert.ok(fnStart !== -1, "buildGateEvaluatePrompt should be exported");
82
+ const signature = promptsSrc.slice(fnStart, fnStart + 300);
83
+ assert.ok(
84
+ signature.includes("subagentModel"),
85
+ "buildGateEvaluatePrompt should accept a subagentModel parameter",
86
+ );
87
+ });
88
+
89
+ // ─── Structural: Instruction Strings Inject Model ────────────────────────
90
+
91
+ test("buildReactiveExecutePrompt: instruction string uses subagentModel when set", () => {
92
+ const fnStart = promptsSrc.indexOf("export async function buildReactiveExecutePrompt");
93
+ const fnEnd = promptsSrc.indexOf("\nexport async function", fnStart + 1);
94
+ const fnBody = promptsSrc.slice(fnStart, fnEnd);
95
+ assert.ok(
96
+ fnBody.includes("subagentModel"),
97
+ "buildReactiveExecutePrompt body should reference subagentModel",
98
+ );
99
+ // The instruction line must be dynamic (not a plain string literal)
100
+ assert.ok(
101
+ !fnBody.includes('"Use this as the prompt for a `subagent` call:"'),
102
+ "instruction should not be a plain static string — model must be injectable",
103
+ );
104
+ });
105
+
106
+ test("buildParallelResearchSlicesPrompt: instruction string uses subagentModel when set", () => {
107
+ const fnStart = promptsSrc.indexOf("export async function buildParallelResearchSlicesPrompt");
108
+ const fnEnd = promptsSrc.indexOf("\nexport async function", fnStart + 1);
109
+ const fnBody = promptsSrc.slice(fnStart, fnEnd);
110
+ assert.ok(
111
+ fnBody.includes("subagentModel"),
112
+ "buildParallelResearchSlicesPrompt body should reference subagentModel",
113
+ );
114
+ });
115
+
116
+ test("buildGateEvaluatePrompt: instruction string uses subagentModel when set", () => {
117
+ const fnStart = promptsSrc.indexOf("export async function buildGateEvaluatePrompt");
118
+ const fnEnd = promptsSrc.indexOf("\nexport async function", fnStart + 1);
119
+ const fnBody = promptsSrc.slice(fnStart, fnEnd);
120
+ assert.ok(
121
+ fnBody.includes("subagentModel"),
122
+ "buildGateEvaluatePrompt body should reference subagentModel",
123
+ );
124
+ });
125
+
126
+ // ─── Structural: Dispatch Wires Model to Prompt Builders ─────────────────
127
+
128
+ test("auto-dispatch: passes model to buildReactiveExecutePrompt", () => {
129
+ // Find the reactive-execute dispatch rule
130
+ const ruleStart = dispatchSrc.indexOf("reactive-execute (parallel dispatch)");
131
+ assert.ok(ruleStart !== -1, "reactive-execute dispatch rule should exist");
132
+ const ruleBlock = dispatchSrc.slice(ruleStart, ruleStart + 1000);
133
+ assert.ok(
134
+ ruleBlock.includes("subagent_model") || ruleBlock.includes("subagentModel"),
135
+ "reactive-execute rule should resolve and pass the subagent model",
136
+ );
137
+ });
138
+
139
+ test("auto-dispatch: passes model to buildParallelResearchSlicesPrompt", () => {
140
+ const callIdx = dispatchSrc.indexOf("buildParallelResearchSlicesPrompt(");
141
+ assert.ok(callIdx !== -1, "buildParallelResearchSlicesPrompt call should exist");
142
+ // The call site should pass a model argument (not just 4 args)
143
+ const callSite = dispatchSrc.slice(callIdx, callIdx + 300);
144
+ assert.ok(
145
+ callSite.includes("subagentModel") || callSite.includes("resolveModelWithFallbacksForUnit"),
146
+ "buildParallelResearchSlicesPrompt call should include model argument",
147
+ );
148
+ });
149
+
150
+ test("auto-dispatch: passes model to buildGateEvaluatePrompt", () => {
151
+ const callIdx = dispatchSrc.indexOf("buildGateEvaluatePrompt(");
152
+ assert.ok(callIdx !== -1, "buildGateEvaluatePrompt call should exist");
153
+ const callSite = dispatchSrc.slice(callIdx, callIdx + 300);
154
+ assert.ok(
155
+ callSite.includes("subagentModel") || callSite.includes("resolveModelWithFallbacksForUnit"),
156
+ "buildGateEvaluatePrompt call should include model argument",
157
+ );
158
+ });
159
+
160
+ // ─── Integration: Prompt Output Contains Model String ────────────────────
161
+
162
+ test("buildReactiveExecutePrompt: output contains model string when subagentModel provided", async (t) => {
163
+ const { buildReactiveExecutePrompt } = await import("../auto-prompts.ts");
164
+ const repo = mkdtempSync(join(tmpdir(), "gsd-subagent-model-reactive-"));
165
+ t.after(() => rmSync(repo, { recursive: true, force: true }));
166
+
167
+ const gsd = join(repo, ".gsd", "milestones", "M001", "slices", "S01");
168
+ mkdirSync(join(gsd, "tasks"), { recursive: true });
169
+
170
+ writeFileSync(
171
+ join(gsd, "S01-PLAN.md"),
172
+ [
173
+ "# S01: Test Slice",
174
+ "",
175
+ "**Goal:** Verify model injection",
176
+ "**Demo:** Model appears in subagent prompt",
177
+ "",
178
+ "## Tasks",
179
+ "",
180
+ "- [ ] **T01: Task One** `est:15m`",
181
+ " Do something.",
182
+ "",
183
+ ].join("\n"),
184
+ );
185
+
186
+ writeFileSync(
187
+ join(gsd, "tasks", "T01-PLAN.md"),
188
+ [
189
+ "# T01: Task One",
190
+ "",
191
+ "## Description",
192
+ "Do something.",
193
+ "",
194
+ "## Inputs",
195
+ "",
196
+ "- `src/config.json` — Config",
197
+ "",
198
+ "## Expected Output",
199
+ "",
200
+ "- `src/out.ts` — Result",
201
+ ].join("\n"),
202
+ );
203
+
204
+ const prompt = await buildReactiveExecutePrompt(
205
+ "M001", "Test Milestone", "S01", "Test Slice",
206
+ ["T01"], repo, "claude-opus-4-6",
207
+ );
208
+
209
+ assert.ok(
210
+ prompt.includes('model: "claude-opus-4-6"'),
211
+ `Prompt should contain model instruction. Got:\n${prompt.slice(0, 500)}`,
212
+ );
213
+ });
214
+
215
+ test("buildReactiveExecutePrompt: no model instruction when subagentModel omitted", async (t) => {
216
+ const { buildReactiveExecutePrompt } = await import("../auto-prompts.ts");
217
+ const repo = mkdtempSync(join(tmpdir(), "gsd-subagent-model-none-"));
218
+ t.after(() => rmSync(repo, { recursive: true, force: true }));
219
+
220
+ const gsd = join(repo, ".gsd", "milestones", "M001", "slices", "S01");
221
+ mkdirSync(join(gsd, "tasks"), { recursive: true });
222
+
223
+ writeFileSync(
224
+ join(gsd, "S01-PLAN.md"),
225
+ [
226
+ "# S01: Test Slice",
227
+ "",
228
+ "**Goal:** Verify no model when omitted",
229
+ "**Demo:** No model string",
230
+ "",
231
+ "## Tasks",
232
+ "",
233
+ "- [ ] **T01: Task One** `est:15m`",
234
+ " Do something.",
235
+ "",
236
+ ].join("\n"),
237
+ );
238
+
239
+ writeFileSync(
240
+ join(gsd, "tasks", "T01-PLAN.md"),
241
+ [
242
+ "# T01: Task One",
243
+ "",
244
+ "## Description",
245
+ "Do something.",
246
+ "",
247
+ "## Inputs",
248
+ "",
249
+ "- `src/config.json` — Config",
250
+ "",
251
+ "## Expected Output",
252
+ "",
253
+ "- `src/out.ts` — Result",
254
+ ].join("\n"),
255
+ );
256
+
257
+ const prompt = await buildReactiveExecutePrompt(
258
+ "M001", "Test Milestone", "S01", "Test Slice",
259
+ ["T01"], repo,
260
+ // no subagentModel
261
+ );
262
+
263
+ assert.ok(
264
+ !prompt.includes('with model:'),
265
+ "Prompt should not contain model instruction when subagentModel is omitted",
266
+ );
267
+ });
@@ -1 +0,0 @@
1
- (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[8974],{5214:(e,t,n)=>{"use strict";Object.defineProperty(t,"__esModule",{value:!0}),Object.defineProperty(t,"workAsyncStorage",{enumerable:!0,get:function(){return r.workAsyncStorageInstance}});let r=n(17828)},15726:(e,t,n)=>{Promise.resolve().then(n.bind(n,66919))},17828:(e,t,n)=>{"use strict";Object.defineProperty(t,"__esModule",{value:!0}),Object.defineProperty(t,"workAsyncStorageInstance",{enumerable:!0,get:function(){return r}});let r=(0,n(64054).createAsyncLocalStorage)()},21957:(e,t,n)=>{"use strict";function r({moduleIds:e}){return null}Object.defineProperty(t,"__esModule",{value:!0}),Object.defineProperty(t,"PreloadChunks",{enumerable:!0,get:function(){return r}}),n(95155),n(47650),n(5214),n(2451),n(53887)},37206:(e,t,n)=>{"use strict";n.d(t,{default:()=>u.a});var r=n(75707),u=n.n(r)},41112:(e,t,n)=>{"use strict";function r({reason:e,children:t}){return t}Object.defineProperty(t,"__esModule",{value:!0}),Object.defineProperty(t,"BailoutToCSR",{enumerable:!0,get:function(){return r}}),n(1980)},64054:(e,t)=>{"use strict";Object.defineProperty(t,"__esModule",{value:!0});var n={bindSnapshot:function(){return s},createAsyncLocalStorage:function(){return a},createSnapshot:function(){return i}};for(var r in n)Object.defineProperty(t,r,{enumerable:!0,get:n[r]});let u=Object.defineProperty(Error("Invariant: AsyncLocalStorage accessed in runtime where it is not available"),"__NEXT_ERROR_CODE",{value:"E504",enumerable:!1,configurable:!0});class l{disable(){throw u}getStore(){}run(){throw u}exit(){throw u}enterWith(){throw u}static bind(e){return e}}let o="u">typeof globalThis&&globalThis.AsyncLocalStorage;function a(){return o?new o:new l}function s(e){return o?o.bind(e):l.bind(e)}function i(){return o?o.snapshot():function(e,...t){return e(...t)}}},66919:(e,t,n)=>{"use strict";n.r(t),n.d(t,{default:()=>l});var r=n(95155);let u=(0,n(37206).default)(()=>Promise.all([n.e(1838),n.e(6079),n.e(4986),n.e(2008),n.e(2826)]).then(n.bind(n,62826)).then(e=>e.GSDAppShell),{loadableGenerated:{webpack:()=>[62826]},ssr:!1,loading:()=>(0,r.jsx)("div",{className:"flex h-screen items-center justify-center bg-background text-sm text-muted-foreground",children:"Loading workspace…"})});function l(){return(0,r.jsx)(u,{})}},68635:(e,t,n)=>{"use strict";Object.defineProperty(t,"__esModule",{value:!0}),Object.defineProperty(t,"default",{enumerable:!0,get:function(){return s}});let r=n(95155),u=n(12115),l=n(41112);function o(e){return{default:e&&"default"in e?e.default:e}}n(21957);let a={loader:()=>Promise.resolve(o(()=>null)),loading:null,ssr:!0},s=function(e){let t={...a,...e},n=(0,u.lazy)(()=>t.loader().then(o)),s=t.loading;function i(e){let o=s?(0,r.jsx)(s,{isLoading:!0,pastDelay:!0,error:null}):null,a=!t.ssr||!!t.loading,i=a?u.Suspense:u.Fragment,c=t.ssr?(0,r.jsxs)(r.Fragment,{children:[null,(0,r.jsx)(n,{...e})]}):(0,r.jsx)(l.BailoutToCSR,{reason:"next/dynamic",children:(0,r.jsx)(n,{...e})});return(0,r.jsx)(i,{...a?{fallback:o}:{},children:c})}return i.displayName="LoadableComponent",i}},75707:(e,t,n)=>{"use strict";Object.defineProperty(t,"__esModule",{value:!0}),Object.defineProperty(t,"default",{enumerable:!0,get:function(){return u}});let r=n(73623)._(n(68635));function u(e,t){let n={};"function"==typeof e&&(n.loader=e);let u={...n,...t};return(0,r.default)({...u,modules:u.loadableGenerated?.modules})}("function"==typeof t.default||"object"==typeof t.default&&null!==t.default)&&void 0===t.default.__esModule&&(Object.defineProperty(t.default,"__esModule",{value:!0}),Object.assign(t.default,t),e.exports=t.default)}},e=>{e.O(0,[8441,3794,7358],()=>e(e.s=15726)),_N_E=e.O()}]);
@@ -1 +0,0 @@
1
- (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[7358],{2852:(e,s,n)=>{Promise.resolve().then(n.t.bind(n,27123,23)),Promise.resolve().then(n.t.bind(n,61304,23)),Promise.resolve().then(n.t.bind(n,78616,23)),Promise.resolve().then(n.t.bind(n,64777,23)),Promise.resolve().then(n.t.bind(n,57121,23)),Promise.resolve().then(n.t.bind(n,74581,23)),Promise.resolve().then(n.t.bind(n,90484,23)),Promise.resolve().then(n.bind(n,86869))},19393:()=>{}},e=>{var s=s=>e(e.s=s);e.O(0,[8441,3794],()=>(s(83861),s(2852))),_N_E=e.O()}]);
@@ -1 +0,0 @@
1
- (self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[9337],{52560:(e,s,_)=>{Promise.resolve().then(_.t.bind(_,27123,23))}},e=>{e.O(0,[8441,3794,7358],()=>e(e.s=52560)),_N_E=e.O()}]);