minutework 0.1.39 → 0.1.41

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 (166) hide show
  1. package/EXTERNAL_ALPHA.md +17 -1
  2. package/README.md +21 -1
  3. package/assets/claude-local/skills/README.md +6 -0
  4. package/assets/claude-local/skills/ai-capability-defaults/SKILL.md +3 -0
  5. package/assets/claude-local/skills/app-pack-authoring/SKILL.md +15 -0
  6. package/assets/claude-local/skills/generated-workspace-architecture/SKILL.md +25 -9
  7. package/assets/claude-local/skills/integration-broker-and-connectors/SKILL.md +122 -0
  8. package/assets/claude-local/skills/layering-and-import-modes/SKILL.md +4 -0
  9. package/assets/claude-local/skills/project-overview-and-strategy/SKILL.md +6 -0
  10. package/assets/claude-local/skills/shell-architecture/SKILL.md +15 -0
  11. package/assets/claude-local/skills/vuilder-public-site-authoring/SKILL.md +10 -4
  12. package/assets/claude-local/skills/vuilder-workspace-architecture/SKILL.md +78 -0
  13. package/assets/templates/vuilder-public-site/.env.example +11 -0
  14. package/assets/templates/vuilder-public-site/README.md +15 -0
  15. package/assets/templates/vuilder-public-site/eslint.config.mjs +6 -0
  16. package/assets/templates/vuilder-public-site/next-env.d.ts +4 -0
  17. package/assets/templates/vuilder-public-site/next.config.mjs +28 -0
  18. package/assets/templates/vuilder-public-site/package.json +39 -0
  19. package/assets/templates/vuilder-public-site/postcss.config.mjs +5 -0
  20. package/assets/templates/vuilder-public-site/src/app/api/onboarding/start/route.ts +19 -0
  21. package/assets/templates/vuilder-public-site/src/app/blog/[slug]/page.tsx +25 -0
  22. package/assets/templates/vuilder-public-site/src/app/blog/page.tsx +26 -0
  23. package/assets/templates/vuilder-public-site/src/app/globals.css +103 -0
  24. package/assets/templates/vuilder-public-site/src/app/layout.tsx +18 -0
  25. package/assets/templates/vuilder-public-site/src/app/onboarding/[[...step]]/page.tsx +39 -0
  26. package/assets/templates/vuilder-public-site/src/app/page.tsx +43 -0
  27. package/assets/templates/vuilder-public-site/src/lib/env.server.ts +31 -0
  28. package/assets/templates/vuilder-public-site/src/lib/platform.server.ts +47 -0
  29. package/assets/templates/vuilder-public-site/src/lib/public-dj.server.ts +86 -0
  30. package/assets/templates/vuilder-public-site/src/lib/public-dj.test.ts +8 -0
  31. package/assets/templates/vuilder-public-site/src/lib/routes.test.ts +13 -0
  32. package/assets/templates/vuilder-public-site/src/lib/routes.ts +12 -0
  33. package/assets/templates/vuilder-public-site/template.json +21 -0
  34. package/assets/templates/vuilder-public-site/tools/template/validate-template.mjs +44 -0
  35. package/assets/templates/vuilder-public-site/tsconfig.json +23 -0
  36. package/assets/templates/vuilder-public-site/vitest.config.ts +13 -0
  37. package/assets/templates/vuilder-shell/.env.example +8 -0
  38. package/assets/templates/vuilder-shell/.storybook/main.ts +19 -0
  39. package/assets/templates/vuilder-shell/.storybook/preview.tsx +38 -0
  40. package/assets/templates/vuilder-shell/README.md +49 -0
  41. package/assets/templates/vuilder-shell/components.json +21 -0
  42. package/assets/templates/vuilder-shell/eslint.config.mjs +41 -0
  43. package/assets/templates/vuilder-shell/next-env.d.ts +6 -0
  44. package/assets/templates/vuilder-shell/next.config.mjs +33 -0
  45. package/assets/templates/vuilder-shell/package.json +61 -0
  46. package/assets/templates/vuilder-shell/postcss.config.mjs +8 -0
  47. package/assets/templates/vuilder-shell/public/.gitkeep +1 -0
  48. package/assets/templates/vuilder-shell/src/app/api/auth/accept-shell-session/route.test.ts +105 -0
  49. package/assets/templates/vuilder-shell/src/app/api/auth/accept-shell-session/route.ts +63 -0
  50. package/assets/templates/vuilder-shell/src/app/api/auth/logout/route.test.ts +63 -0
  51. package/assets/templates/vuilder-shell/src/app/api/auth/logout/route.ts +24 -0
  52. package/assets/templates/vuilder-shell/src/app/api/auth/session/route.test.ts +70 -0
  53. package/assets/templates/vuilder-shell/src/app/api/auth/session/route.ts +27 -0
  54. package/assets/templates/vuilder-shell/src/app/app/layout.tsx +17 -0
  55. package/assets/templates/vuilder-shell/src/app/app/page.tsx +30 -0
  56. package/assets/templates/vuilder-shell/src/app/blog/[slug]/page.tsx +15 -0
  57. package/assets/templates/vuilder-shell/src/app/blog/page.tsx +15 -0
  58. package/assets/templates/vuilder-shell/src/app/docs/[...slug]/page.tsx +15 -0
  59. package/assets/templates/vuilder-shell/src/app/docs/page.tsx +15 -0
  60. package/assets/templates/vuilder-shell/src/app/globals.css +70 -0
  61. package/assets/templates/vuilder-shell/src/app/layout.tsx +69 -0
  62. package/assets/templates/vuilder-shell/src/app/login/route.test.ts +33 -0
  63. package/assets/templates/vuilder-shell/src/app/login/route.ts +21 -0
  64. package/assets/templates/vuilder-shell/src/app/page.tsx +16 -0
  65. package/assets/templates/vuilder-shell/src/app/pricing/page.tsx +15 -0
  66. package/assets/templates/vuilder-shell/src/app/providers.tsx +25 -0
  67. package/assets/templates/vuilder-shell/src/app/robots.ts +21 -0
  68. package/assets/templates/vuilder-shell/src/app/sitemap.ts +33 -0
  69. package/assets/templates/vuilder-shell/src/app/w/[workspace_slug]/connect/page.tsx +31 -0
  70. package/assets/templates/vuilder-shell/src/app/w/[workspace_slug]/page.tsx +54 -0
  71. package/assets/templates/vuilder-shell/src/components/ui/button.tsx +59 -0
  72. package/assets/templates/vuilder-shell/src/components/ui/input.tsx +21 -0
  73. package/assets/templates/vuilder-shell/src/design-system/docs/governance.mdx +26 -0
  74. package/assets/templates/vuilder-shell/src/design-system/patterns/panel-frame.stories.tsx +48 -0
  75. package/assets/templates/vuilder-shell/src/design-system/patterns/panel-frame.tsx +26 -0
  76. package/assets/templates/vuilder-shell/src/design-system/patterns/status-badge.stories.tsx +26 -0
  77. package/assets/templates/vuilder-shell/src/design-system/patterns/status-badge.tsx +35 -0
  78. package/assets/templates/vuilder-shell/src/design-system/patterns/theme-mode-toggle.stories.tsx +21 -0
  79. package/assets/templates/vuilder-shell/src/design-system/patterns/theme-mode-toggle.tsx +75 -0
  80. package/assets/templates/vuilder-shell/src/design-system/primitives/button.stories.tsx +37 -0
  81. package/assets/templates/vuilder-shell/src/design-system/primitives/button.ts +1 -0
  82. package/assets/templates/vuilder-shell/src/design-system/primitives/input.stories.tsx +26 -0
  83. package/assets/templates/vuilder-shell/src/design-system/primitives/input.ts +1 -0
  84. package/assets/templates/vuilder-shell/src/design-system/recipes/chrome.ts +28 -0
  85. package/assets/templates/vuilder-shell/src/design-system/tokens/foundation.css +31 -0
  86. package/assets/templates/vuilder-shell/src/design-system/tokens/index.css +3 -0
  87. package/assets/templates/vuilder-shell/src/design-system/tokens/manifest.json +85 -0
  88. package/assets/templates/vuilder-shell/src/design-system/tokens/manifest.ts +87 -0
  89. package/assets/templates/vuilder-shell/src/design-system/tokens/semantic.css +105 -0
  90. package/assets/templates/vuilder-shell/src/design-system/tokens/theme.css +59 -0
  91. package/assets/templates/vuilder-shell/src/design-system/tokens/tokens.stories.tsx +71 -0
  92. package/assets/templates/vuilder-shell/src/features/dashboard/components/tenant-dashboard.tsx +134 -0
  93. package/assets/templates/vuilder-shell/src/features/public-shell/components/static-public-page.tsx +58 -0
  94. package/assets/templates/vuilder-shell/src/features/shell/components/authenticated-app-layout-shell.tsx +84 -0
  95. package/assets/templates/vuilder-shell/src/features/shell/components/private-app-shell.tsx +22 -0
  96. package/assets/templates/vuilder-shell/src/features/vuilder-shell/components/vuilder-connect-screen.tsx +89 -0
  97. package/assets/templates/vuilder-shell/src/features/vuilder-shell/components/vuilder-workspace-screen.tsx +49 -0
  98. package/assets/templates/vuilder-shell/src/lib/app-routes.test.ts +37 -0
  99. package/assets/templates/vuilder-shell/src/lib/app-routes.ts +86 -0
  100. package/assets/templates/vuilder-shell/src/lib/auth-routes.server.test.ts +26 -0
  101. package/assets/templates/vuilder-shell/src/lib/auth-routes.server.ts +53 -0
  102. package/assets/templates/vuilder-shell/src/lib/http/same-origin.test.ts +23 -0
  103. package/assets/templates/vuilder-shell/src/lib/http/same-origin.ts +18 -0
  104. package/assets/templates/vuilder-shell/src/lib/platform/client.server.test.ts +201 -0
  105. package/assets/templates/vuilder-shell/src/lib/platform/client.server.ts +540 -0
  106. package/assets/templates/vuilder-shell/src/lib/platform/contracts.ts +190 -0
  107. package/assets/templates/vuilder-shell/src/lib/platform/endpoints.server.ts +29 -0
  108. package/assets/templates/vuilder-shell/src/lib/platform/env.server.ts +82 -0
  109. package/assets/templates/vuilder-shell/src/lib/platform/route-response.ts +33 -0
  110. package/assets/templates/vuilder-shell/src/lib/platform/session.server.ts +145 -0
  111. package/assets/templates/vuilder-shell/src/lib/public-site.test.ts +20 -0
  112. package/assets/templates/vuilder-shell/src/lib/public-site.ts +48 -0
  113. package/assets/templates/vuilder-shell/src/lib/theme-config.ts +10 -0
  114. package/assets/templates/vuilder-shell/src/lib/theme.tsx +159 -0
  115. package/assets/templates/vuilder-shell/src/lib/utils.ts +6 -0
  116. package/assets/templates/vuilder-shell/template.json +28 -0
  117. package/assets/templates/vuilder-shell/template.schema.json +171 -0
  118. package/assets/templates/vuilder-shell/test/server-only-stub.ts +1 -0
  119. package/assets/templates/vuilder-shell/tools/design-system/build-token-manifest.mjs +3 -0
  120. package/assets/templates/vuilder-shell/tools/design-system/check-imports.mjs +9 -0
  121. package/assets/templates/vuilder-shell/tools/design-system/check-stories.mjs +9 -0
  122. package/assets/templates/vuilder-shell/tools/design-system/check-values.mjs +9 -0
  123. package/assets/templates/vuilder-shell/tools/design-system/checks.mjs +238 -0
  124. package/assets/templates/vuilder-shell/tools/design-system/eslint-plugin-design-system.mjs +184 -0
  125. package/assets/templates/vuilder-shell/tools/design-system/playwright.config.mjs +34 -0
  126. package/assets/templates/vuilder-shell/tools/design-system/run-checks.mjs +22 -0
  127. package/assets/templates/vuilder-shell/tools/design-system/shared.mjs +166 -0
  128. package/assets/templates/vuilder-shell/tools/design-system/visual.spec.ts +41 -0
  129. package/assets/templates/vuilder-shell/tools/template/validate-route-contract.mjs +373 -0
  130. package/assets/templates/vuilder-shell/tools/template/validate-template.mjs +45 -0
  131. package/assets/templates/vuilder-shell/tools/template/with-public-site-fixture.mjs +45 -0
  132. package/assets/templates/vuilder-shell/tsconfig.json +42 -0
  133. package/assets/templates/vuilder-shell/vitest.config.ts +23 -0
  134. package/dist/auth.js +66 -14
  135. package/dist/auth.js.map +1 -1
  136. package/dist/deploy-state.d.ts +1 -0
  137. package/dist/deploy-state.js.map +1 -1
  138. package/dist/deploy.js +18 -4
  139. package/dist/deploy.js.map +1 -1
  140. package/dist/developer-client.d.ts +1 -1
  141. package/dist/index.js +12 -2
  142. package/dist/index.js.map +1 -1
  143. package/dist/init-prompt.js +21 -13
  144. package/dist/init-prompt.js.map +1 -1
  145. package/dist/init.d.ts +3 -1
  146. package/dist/init.js +103 -12
  147. package/dist/init.js.map +1 -1
  148. package/dist/orchestrator-context.js +17 -5
  149. package/dist/orchestrator-context.js.map +1 -1
  150. package/dist/orchestrator-state.d.ts +2 -2
  151. package/dist/orchestrator-state.js.map +1 -1
  152. package/dist/publish.js +12 -2
  153. package/dist/publish.js.map +1 -1
  154. package/dist/state.d.ts +2 -0
  155. package/dist/state.js +9 -0
  156. package/dist/state.js.map +1 -1
  157. package/package.json +2 -2
  158. package/vendor/workspace-mcp/context.d.ts +3 -1
  159. package/vendor/workspace-mcp/context.js +134 -21
  160. package/vendor/workspace-mcp/context.js.map +1 -1
  161. package/vendor/workspace-mcp/types.d.ts +72 -7
  162. package/vendor/workspace-mcp/types.js +8 -4
  163. package/vendor/workspace-mcp/types.js.map +1 -1
  164. package/assets/templates/fastapi-sidecar/poetry.lock +0 -757
  165. package/assets/templates/next-tenant-app/package-lock.json +0 -9682
  166. package/assets/templates/next-tenant-app/pnpm-lock.yaml +0 -6062
@@ -0,0 +1,49 @@
1
+ # Vuilder Shell Template
2
+
3
+ `vuilder-shell` is the branded customer tenant shell for Vuilder-owned verticals.
4
+ It is generated next to `vuilder-app` in a Vuilder workspace.
5
+
6
+ The shell opens after the customer has signed in through central SSO and the
7
+ platform has completed the `customer_vertical_signup` intent. The platform owns
8
+ identity, tenant runtime provisioning, and app-pack install authority. This
9
+ template owns customer-visible navigation, workspace status, and vertical
10
+ product UI.
11
+
12
+ ## Route Shape
13
+
14
+ - public route at `/`
15
+ - platform member auth handoff at `/login`, which redirects to central SSO with
16
+ a shell return URL
17
+ - branded tenant setup/status route at `/w/[workspace_slug]/connect`
18
+ - branded tenant workspace route at `/w/[workspace_slug]`
19
+ - compatibility authenticated app route at `/app`
20
+
21
+ ## Auth Profile
22
+
23
+ This template uses central platform SSO plus a tenant-bound shell session minted
24
+ by the platform after membership is established. `/login` sets a short-lived
25
+ host-only shell handoff state cookie before redirecting to SSO. SSO returns to
26
+ `/api/auth/accept-shell-session` with a short-lived one-use code and the echoed
27
+ state; the shell validates the state, exchanges the code server-side, sets a
28
+ host-only `mw_vuilder_shell_session` cookie, and redirects to the clean
29
+ workspace URL. Raw platform session and CSRF cookies stay on the SSO origin; the
30
+ shell never builds Django session or CSRF headers. The shell does not render a
31
+ password form and does not expose a local credential-login BFF; `/login` is only
32
+ an SSO redirect handoff. Local `/api/auth/session` and `/api/auth/logout` are
33
+ thin shell-token helpers; logout best-effort revokes the tenant-bound shell
34
+ token before clearing cookies. Browser React never stores platform credentials,
35
+ runtime keys, provisioning secrets, package refs, or install coordinates.
36
+
37
+ For production handoff from `auth.*` to a branded shell on a sibling subdomain,
38
+ configure `MW_AUTH_BASE_URL` and `MW_PUBLIC_BASE_URL`. Do not configure a shared
39
+ parent-domain shell cookie; the shell session cookie is host-only on the branded
40
+ shell origin.
41
+
42
+ `vuilder-shell` is not a `tenant-app` and must not use `@minutework/web-auth`
43
+ or same-origin `/_mw` customer routes.
44
+
45
+ ## Vuilder Boundary
46
+
47
+ Vuilders may customize UI, route layout, visual design, tenant settings,
48
+ runtime-backed data views, agents, workflows, and branding. They must not move
49
+ provisioning decisions into React state or browser-submitted payloads.
@@ -0,0 +1,21 @@
1
+ {
2
+ "$schema": "https://ui.shadcn.com/schema.json",
3
+ "style": "new-york",
4
+ "rsc": true,
5
+ "tsx": true,
6
+ "tailwind": {
7
+ "config": "",
8
+ "css": "src/app/globals.css",
9
+ "baseColor": "neutral",
10
+ "cssVariables": true,
11
+ "prefix": ""
12
+ },
13
+ "aliases": {
14
+ "components": "@/design-system",
15
+ "utils": "@/lib/utils",
16
+ "ui": "@/design-system/primitives",
17
+ "lib": "@/lib",
18
+ "hooks": "@/hooks"
19
+ },
20
+ "iconLibrary": "lucide"
21
+ }
@@ -0,0 +1,41 @@
1
+ import nextCoreWebVitals from "eslint-config-next/core-web-vitals";
2
+ import nextTypeScript from "eslint-config-next/typescript";
3
+
4
+ import designSystemPlugin from "./tools/design-system/eslint-plugin-design-system.mjs";
5
+
6
+ const eslintConfig = [
7
+ ...nextCoreWebVitals,
8
+ ...nextTypeScript,
9
+ {
10
+ ignores: [
11
+ ".next/**",
12
+ "node_modules/**",
13
+ "next-env.d.ts",
14
+ "storybook-static/**",
15
+ "src/design-system/tokens/manifest.ts",
16
+ ],
17
+ },
18
+ {
19
+ files: ["src/app/**/*.{ts,tsx}", "src/features/**/*.{ts,tsx}"],
20
+ plugins: {
21
+ "design-system": designSystemPlugin,
22
+ },
23
+ rules: {
24
+ "design-system/no-design-system-bypass-imports": "error",
25
+ "design-system/no-cva-outside-design-system": "error",
26
+ "design-system/no-raw-design-values": "error",
27
+ "design-system/no-bespoke-visual-stacks-in-features": "error",
28
+ },
29
+ },
30
+ {
31
+ files: ["src/design-system/**/*.{ts,tsx}"],
32
+ plugins: {
33
+ "design-system": designSystemPlugin,
34
+ },
35
+ rules: {
36
+ "design-system/no-raw-design-values": "error",
37
+ },
38
+ },
39
+ ];
40
+
41
+ export default eslintConfig;
@@ -0,0 +1,6 @@
1
+ /// <reference types="next" />
2
+ /// <reference types="next/image-types/global" />
3
+ import "./.next/types/routes.d.ts";
4
+
5
+ // NOTE: This file should not be edited
6
+ // see https://nextjs.org/docs/app/api-reference/config/typescript for more information.
@@ -0,0 +1,33 @@
1
+ import path from "node:path";
2
+ import { fileURLToPath } from "node:url";
3
+ import { existsSync } from "node:fs";
4
+
5
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
6
+
7
+ function resolveTurbopackRoot(startDirectory) {
8
+ let directory = startDirectory;
9
+
10
+ while (true) {
11
+ if (existsSync(path.join(directory, "pnpm-workspace.yaml"))) {
12
+ return directory;
13
+ }
14
+
15
+ const parent = path.dirname(directory);
16
+ if (parent === directory) {
17
+ return startDirectory;
18
+ }
19
+ directory = parent;
20
+ }
21
+ }
22
+
23
+ const nextConfig = {
24
+ reactStrictMode: true,
25
+ typedRoutes: true,
26
+ // Prefer the nearest workspace root so scaffolded tenant-app packages inside a
27
+ // monorepo and the in-repo template both resolve dependencies correctly.
28
+ turbopack: {
29
+ root: resolveTurbopackRoot(__dirname),
30
+ },
31
+ };
32
+
33
+ export default nextConfig;
@@ -0,0 +1,61 @@
1
+ {
2
+ "name": "vuilder-shell",
3
+ "version": "0.1.0",
4
+ "private": true,
5
+ "packageManager": "pnpm@9.0.0",
6
+ "engines": {
7
+ "node": ">=20.9.0"
8
+ },
9
+ "scripts": {
10
+ "dev": "next dev --hostname 127.0.0.1 --port 3301",
11
+ "build": "next build",
12
+ "build:validate": "node tools/template/with-public-site-fixture.mjs next build",
13
+ "start": "next start -p 3301",
14
+ "lint": "eslint .",
15
+ "test": "vitest run",
16
+ "typegen": "next typegen",
17
+ "template:validate": "node tools/template/validate-template.mjs",
18
+ "template:route-contract": "node tools/template/validate-route-contract.mjs",
19
+ "typecheck": "tsc --noEmit",
20
+ "validate": "pnpm template:validate && pnpm template:route-contract && pnpm typegen && pnpm design-system:tokens && pnpm design-system:check && pnpm typecheck && pnpm lint && pnpm test && pnpm build:validate && pnpm build-storybook:validate",
21
+ "design-system:tokens": "node tools/design-system/build-token-manifest.mjs",
22
+ "design-system:check": "node tools/design-system/run-checks.mjs",
23
+ "design-system:check:staged": "node tools/design-system/run-checks.mjs --staged",
24
+ "storybook": "storybook dev -p 6006",
25
+ "build-storybook": "storybook build",
26
+ "build-storybook:validate": "node tools/template/with-public-site-fixture.mjs storybook build",
27
+ "design-system:visual": "playwright test -c tools/design-system/playwright.config.mjs"
28
+ },
29
+ "dependencies": {
30
+ "@radix-ui/react-slot": "^1.2.4",
31
+ "class-variance-authority": "^0.7.1",
32
+ "clsx": "^2.1.1",
33
+ "lucide-react": "^0.564.0",
34
+ "next": "^16.2.0",
35
+ "react": "^19.2.0",
36
+ "react-dom": "^19.2.0",
37
+ "server-only": "^0.0.1",
38
+ "tailwind-merge": "^3.3.1",
39
+ "tw-animate-css": "^1.3.3",
40
+ "zod": "^3.24.1"
41
+ },
42
+ "devDependencies": {
43
+ "@playwright/test": "^1.58.2",
44
+ "@storybook/addon-a11y": "^10.2.19",
45
+ "@storybook/addon-docs": "^10.2.19",
46
+ "@storybook/nextjs-vite": "^10.2.19",
47
+ "@tailwindcss/postcss": "^4.2.0",
48
+ "@types/node": "^22.0.0",
49
+ "@types/react": "^19.0.0",
50
+ "@types/react-dom": "^19.0.0",
51
+ "ajv": "^8.17.1",
52
+ "eslint": "^9.39.1",
53
+ "eslint-config-next": "^16.2.0",
54
+ "http-server": "^14.1.1",
55
+ "postcss": "^8.5.6",
56
+ "storybook": "^10.2.19",
57
+ "tailwindcss": "^4.2.0",
58
+ "typescript": "^5.6.0",
59
+ "vitest": "^3.2.4"
60
+ }
61
+ }
@@ -0,0 +1,8 @@
1
+ /** @type {import('postcss-load-config').Config} */
2
+ const config = {
3
+ plugins: {
4
+ "@tailwindcss/postcss": {},
5
+ },
6
+ };
7
+
8
+ export default config;
@@ -0,0 +1,105 @@
1
+ import { afterEach, describe, expect, it, vi } from "vitest";
2
+
3
+ const mocks = vi.hoisted(() => ({
4
+ applyVuilderShellSessionToResponse: vi.fn(),
5
+ clearVuilderShellHandoffStateFromResponse: vi.fn(),
6
+ exchangeVuilderShellSessionHandoffCode: vi.fn(),
7
+ readVuilderShellHandoffState: vi.fn(),
8
+ }));
9
+
10
+ vi.mock("@/lib/platform/client.server", () => ({
11
+ exchangeVuilderShellSessionHandoffCode:
12
+ mocks.exchangeVuilderShellSessionHandoffCode,
13
+ }));
14
+
15
+ vi.mock("@/lib/platform/session.server", () => ({
16
+ applyVuilderShellSessionToResponse: mocks.applyVuilderShellSessionToResponse,
17
+ clearVuilderShellHandoffStateFromResponse:
18
+ mocks.clearVuilderShellHandoffStateFromResponse,
19
+ readVuilderShellHandoffState: mocks.readVuilderShellHandoffState,
20
+ }));
21
+
22
+ import { GET } from "./route";
23
+
24
+ describe("GET /api/auth/accept-shell-session", () => {
25
+ afterEach(() => {
26
+ vi.clearAllMocks();
27
+ });
28
+
29
+ it("exchanges the handoff code, sets the shell token, and redirects locally", async () => {
30
+ mocks.readVuilderShellHandoffState.mockResolvedValue("state-1");
31
+ mocks.exchangeVuilderShellSessionHandoffCode.mockResolvedValue({
32
+ data: {
33
+ expires_at: "2026-06-09T12:00:00Z",
34
+ shell_session_token: "shell-token-1",
35
+ },
36
+ });
37
+
38
+ const response = await GET(
39
+ new Request(
40
+ "https://shell.example.com/api/auth/accept-shell-session?code=code-1&state=state-1&return_to=%2Fw%2Ffleet-alpha%2Fconnect",
41
+ ),
42
+ );
43
+
44
+ expect(response.status).toBe(307);
45
+ expect(response.headers.get("location")).toBe(
46
+ "https://shell.example.com/w/fleet-alpha/connect",
47
+ );
48
+ expect(response.headers.get("cache-control")).toBe("no-store");
49
+ expect(response.headers.get("referrer-policy")).toBe("no-referrer");
50
+ expect(mocks.exchangeVuilderShellSessionHandoffCode).toHaveBeenCalledWith(
51
+ "code-1",
52
+ "state-1",
53
+ );
54
+ expect(mocks.applyVuilderShellSessionToResponse).toHaveBeenCalledWith(
55
+ expect.any(Response),
56
+ "shell-token-1",
57
+ "2026-06-09T12:00:00Z",
58
+ );
59
+ });
60
+
61
+ it("rejects external return paths", async () => {
62
+ mocks.readVuilderShellHandoffState.mockResolvedValue("state-1");
63
+ mocks.exchangeVuilderShellSessionHandoffCode.mockResolvedValue({
64
+ data: {
65
+ expires_at: "2026-06-09T12:00:00Z",
66
+ shell_session_token: "shell-token-1",
67
+ },
68
+ });
69
+
70
+ const response = await GET(
71
+ new Request(
72
+ "https://shell.example.com/api/auth/accept-shell-session?code=code-1&state=state-1&return_to=https%3A%2F%2Fevil.example.com%2F",
73
+ ),
74
+ );
75
+
76
+ expect(response.headers.get("location")).toBe("https://shell.example.com/app");
77
+ });
78
+
79
+ it("fails closed when the shell state does not match", async () => {
80
+ mocks.readVuilderShellHandoffState.mockResolvedValue("state-1");
81
+
82
+ const response = await GET(
83
+ new Request(
84
+ "https://shell.example.com/api/auth/accept-shell-session?code=code-1&state=attacker-state&return_to=%2Fw%2Ffleet-alpha%2Fconnect",
85
+ {
86
+ headers: {
87
+ cookie: "mw_vuilder_shell_session=existing-shell-token",
88
+ },
89
+ },
90
+ ),
91
+ );
92
+
93
+ expect(response.headers.get("location")).toBe(
94
+ "https://shell.example.com/w/fleet-alpha/connect",
95
+ );
96
+ expect(mocks.exchangeVuilderShellSessionHandoffCode).not.toHaveBeenCalled();
97
+ expect(mocks.applyVuilderShellSessionToResponse).not.toHaveBeenCalled();
98
+ expect(mocks.clearVuilderShellHandoffStateFromResponse).toHaveBeenCalledWith(
99
+ expect.any(Response),
100
+ );
101
+ expect(response.headers.get("set-cookie") ?? "").not.toContain(
102
+ "mw_vuilder_shell_session",
103
+ );
104
+ });
105
+ });
@@ -0,0 +1,63 @@
1
+ import { NextResponse } from "next/server";
2
+
3
+ import { appRoutes } from "@/lib/app-routes";
4
+ import { exchangeVuilderShellSessionHandoffCode } from "@/lib/platform/client.server";
5
+ import {
6
+ applyVuilderShellSessionToResponse,
7
+ clearVuilderShellHandoffStateFromResponse,
8
+ readVuilderShellHandoffState,
9
+ } from "@/lib/platform/session.server";
10
+
11
+ function normalizeLocalReturnTo(value?: string | null) {
12
+ const candidate = String(value ?? "").trim();
13
+ if (!candidate || !candidate.startsWith("/") || candidate.startsWith("//")) {
14
+ return appRoutes.appHome;
15
+ }
16
+
17
+ try {
18
+ const parsed = new URL(candidate, "http://localhost");
19
+ if (parsed.origin !== "http://localhost" || !parsed.pathname.startsWith("/")) {
20
+ return appRoutes.appHome;
21
+ }
22
+ return `${parsed.pathname}${parsed.search}${parsed.hash}`;
23
+ } catch {
24
+ return appRoutes.appHome;
25
+ }
26
+ }
27
+
28
+ export async function GET(request: Request) {
29
+ const url = new URL(request.url);
30
+ const shellSessionHandoffCode = url.searchParams.get("code")?.trim() ?? "";
31
+ const shellHandoffState = url.searchParams.get("state")?.trim() ?? "";
32
+ const expectedShellHandoffState = await readVuilderShellHandoffState();
33
+ const returnTo = normalizeLocalReturnTo(url.searchParams.get("return_to"));
34
+ const response = NextResponse.redirect(new URL(returnTo, url.origin));
35
+ response.headers.set("Cache-Control", "no-store");
36
+ response.headers.set("Referrer-Policy", "no-referrer");
37
+ clearVuilderShellHandoffStateFromResponse(response);
38
+
39
+ if (
40
+ !shellSessionHandoffCode ||
41
+ !shellHandoffState ||
42
+ !expectedShellHandoffState ||
43
+ shellHandoffState !== expectedShellHandoffState
44
+ ) {
45
+ return response;
46
+ }
47
+
48
+ try {
49
+ const result = await exchangeVuilderShellSessionHandoffCode(
50
+ shellSessionHandoffCode,
51
+ expectedShellHandoffState,
52
+ );
53
+ applyVuilderShellSessionToResponse(
54
+ response,
55
+ result.data.shell_session_token,
56
+ result.data.expires_at,
57
+ );
58
+ } catch {
59
+ return response;
60
+ }
61
+
62
+ return response;
63
+ }
@@ -0,0 +1,63 @@
1
+ import { afterEach, describe, expect, it, vi } from "vitest";
2
+
3
+ const mocks = vi.hoisted(() => ({
4
+ clearPlatformSessionFromResponse: vi.fn(),
5
+ readPlatformAuthState: vi.fn(),
6
+ revokeVuilderShellSession: vi.fn(),
7
+ }));
8
+
9
+ vi.mock("@/lib/platform/client.server", () => ({
10
+ revokeVuilderShellSession: mocks.revokeVuilderShellSession,
11
+ }));
12
+
13
+ vi.mock("@/lib/platform/session.server", () => ({
14
+ clearPlatformSessionFromResponse: mocks.clearPlatformSessionFromResponse,
15
+ readPlatformAuthState: mocks.readPlatformAuthState,
16
+ }));
17
+
18
+ import { POST } from "./route";
19
+
20
+ describe("POST /api/auth/logout", () => {
21
+ afterEach(() => {
22
+ vi.clearAllMocks();
23
+ });
24
+
25
+ it("rejects cross-origin submissions before clearing shared cookies", async () => {
26
+ const response = await POST(
27
+ new Request("https://shell.example.com/api/auth/logout", {
28
+ method: "POST",
29
+ headers: { origin: "https://evil.example.com" },
30
+ }),
31
+ );
32
+
33
+ expect(response.status).toBe(403);
34
+ expect(mocks.readPlatformAuthState).not.toHaveBeenCalled();
35
+ expect(mocks.revokeVuilderShellSession).not.toHaveBeenCalled();
36
+ expect(mocks.clearPlatformSessionFromResponse).not.toHaveBeenCalled();
37
+ });
38
+
39
+ it("revokes and clears shell session cookies for same-origin submissions", async () => {
40
+ const authState = { shellSessionToken: "shell-token-1" };
41
+ mocks.readPlatformAuthState.mockResolvedValue(authState);
42
+ mocks.revokeVuilderShellSession.mockResolvedValue(undefined);
43
+
44
+ const response = await POST(
45
+ new Request("https://shell.example.com/api/auth/logout", {
46
+ method: "POST",
47
+ headers: { origin: "https://shell.example.com" },
48
+ }),
49
+ );
50
+
51
+ expect(response.status).toBe(204);
52
+ expect(response.headers.get("Cache-Control")).toBe("no-store");
53
+ expect(mocks.revokeVuilderShellSession).toHaveBeenCalledWith(authState);
54
+ expect(mocks.clearPlatformSessionFromResponse).toHaveBeenCalledWith(
55
+ expect.any(Response),
56
+ );
57
+ expect(
58
+ mocks.revokeVuilderShellSession.mock.invocationCallOrder[0],
59
+ ).toBeLessThan(
60
+ mocks.clearPlatformSessionFromResponse.mock.invocationCallOrder[0],
61
+ );
62
+ });
63
+ });
@@ -0,0 +1,24 @@
1
+ import { NextResponse } from "next/server";
2
+
3
+ import { hasTrustedBrowserOrigin } from "@/lib/http/same-origin";
4
+ import { revokeVuilderShellSession } from "@/lib/platform/client.server";
5
+ import {
6
+ clearPlatformSessionFromResponse,
7
+ readPlatformAuthState,
8
+ } from "@/lib/platform/session.server";
9
+
10
+ export async function POST(request: Request) {
11
+ if (!hasTrustedBrowserOrigin(request)) {
12
+ return NextResponse.json(
13
+ { detail: "Same-origin browser submission required." },
14
+ { status: 403 },
15
+ );
16
+ }
17
+
18
+ const authState = await readPlatformAuthState();
19
+ await revokeVuilderShellSession(authState).catch(() => null);
20
+ const response = new NextResponse(null, { status: 204 });
21
+ response.headers.set("Cache-Control", "no-store");
22
+ clearPlatformSessionFromResponse(response);
23
+ return response;
24
+ }
@@ -0,0 +1,70 @@
1
+ import { afterEach, describe, expect, it, vi } from "vitest";
2
+
3
+ const mocks = vi.hoisted(() => ({
4
+ loadCurrentSession: vi.fn(),
5
+ readPlatformAuthState: vi.fn(),
6
+ syncPlatformAuthStateToResponse: vi.fn(),
7
+ toPlatformErrorResponse: vi.fn(),
8
+ }));
9
+
10
+ vi.mock("@/lib/platform/client.server", () => ({
11
+ loadCurrentSession: mocks.loadCurrentSession,
12
+ }));
13
+
14
+ vi.mock("@/lib/platform/route-response", () => ({
15
+ toPlatformErrorResponse: mocks.toPlatformErrorResponse,
16
+ }));
17
+
18
+ vi.mock("@/lib/platform/session.server", () => ({
19
+ readPlatformAuthState: mocks.readPlatformAuthState,
20
+ syncPlatformAuthStateToResponse: mocks.syncPlatformAuthStateToResponse,
21
+ }));
22
+
23
+ import { GET } from "./route";
24
+
25
+ describe("GET /api/auth/session", () => {
26
+ afterEach(() => {
27
+ vi.clearAllMocks();
28
+ });
29
+
30
+ it("returns the current session with no-store cache headers", async () => {
31
+ const authState = { shellSessionToken: "shell-token-1" };
32
+ const nextAuthState = { shellSessionToken: "shell-token-2" };
33
+ mocks.readPlatformAuthState.mockResolvedValue(authState);
34
+ mocks.loadCurrentSession.mockResolvedValue({
35
+ authState: nextAuthState,
36
+ data: { authenticated: true },
37
+ });
38
+
39
+ const response = await GET();
40
+
41
+ expect(response.status).toBe(200);
42
+ expect(response.headers.get("Cache-Control")).toBe("no-store");
43
+ await expect(response.json()).resolves.toEqual({ authenticated: true });
44
+ expect(mocks.syncPlatformAuthStateToResponse).toHaveBeenCalledWith(
45
+ expect.any(Response),
46
+ authState,
47
+ nextAuthState,
48
+ );
49
+ });
50
+
51
+ it("returns platform errors with no-store cache headers", async () => {
52
+ const authState = { shellSessionToken: "shell-token-1" };
53
+ const error = new Error("platform failed");
54
+ mocks.readPlatformAuthState.mockResolvedValue(authState);
55
+ mocks.loadCurrentSession.mockRejectedValue(error);
56
+ mocks.toPlatformErrorResponse.mockReturnValue(
57
+ Response.json({ detail: "Platform error" }, { status: 503 }),
58
+ );
59
+
60
+ const response = await GET();
61
+
62
+ expect(response.status).toBe(503);
63
+ expect(response.headers.get("Cache-Control")).toBe("no-store");
64
+ expect(mocks.toPlatformErrorResponse).toHaveBeenCalledWith(
65
+ error,
66
+ "Unable to load the current platform session.",
67
+ authState,
68
+ );
69
+ });
70
+ });
@@ -0,0 +1,27 @@
1
+ import { NextResponse } from "next/server";
2
+
3
+ import { loadCurrentSession } from "@/lib/platform/client.server";
4
+ import { toPlatformErrorResponse } from "@/lib/platform/route-response";
5
+ import {
6
+ readPlatformAuthState,
7
+ syncPlatformAuthStateToResponse,
8
+ } from "@/lib/platform/session.server";
9
+
10
+ export async function GET() {
11
+ const authState = await readPlatformAuthState();
12
+ try {
13
+ const result = await loadCurrentSession(authState);
14
+ const response = NextResponse.json(result.data, { status: 200 });
15
+ response.headers.set("Cache-Control", "no-store");
16
+ syncPlatformAuthStateToResponse(response, authState, result.authState);
17
+ return response;
18
+ } catch (error) {
19
+ const response = toPlatformErrorResponse(
20
+ error,
21
+ "Unable to load the current platform session.",
22
+ authState,
23
+ );
24
+ response.headers.set("Cache-Control", "no-store");
25
+ return response;
26
+ }
27
+ }
@@ -0,0 +1,17 @@
1
+ import type { Metadata } from "next";
2
+ import type { ReactNode } from "react";
3
+
4
+ export const metadata: Metadata = {
5
+ robots: {
6
+ index: false,
7
+ follow: false,
8
+ },
9
+ };
10
+
11
+ export default function AuthenticatedAppLayout({
12
+ children,
13
+ }: {
14
+ children: ReactNode;
15
+ }) {
16
+ return children;
17
+ }
@@ -0,0 +1,30 @@
1
+ import { notFound, redirect } from "next/navigation";
2
+
3
+ import { appRoutes } from "@/lib/app-routes";
4
+ import { loadCurrentWorkspaceShellSession } from "@/lib/platform/client.server";
5
+ import { readPlatformAuthState } from "@/lib/platform/session.server";
6
+
7
+ export const metadata = {
8
+ title: "Workspace",
9
+ };
10
+
11
+ export default async function AppHomePage() {
12
+ const authState = await readPlatformAuthState();
13
+ const shellSession = (await loadCurrentWorkspaceShellSession(authState)).data;
14
+ const { membership, session, workspaceCanOpen, workspaceCanView } =
15
+ shellSession;
16
+
17
+ if (!session.authenticated) {
18
+ redirect(appRoutes.login);
19
+ }
20
+
21
+ if (!membership || !workspaceCanView) {
22
+ notFound();
23
+ }
24
+
25
+ if (!workspaceCanOpen) {
26
+ redirect(appRoutes.workspaceConnect(membership.workspace_slug));
27
+ }
28
+
29
+ redirect(appRoutes.workspaceShell(membership.workspace_slug));
30
+ }
@@ -0,0 +1,15 @@
1
+ import { StaticPublicPage } from "@/features/public-shell/components/static-public-page";
2
+
3
+ export const metadata = {
4
+ title: "Blog",
5
+ };
6
+
7
+ export default function BlogPostPage() {
8
+ return (
9
+ <StaticPublicPage
10
+ eyebrow="Blog"
11
+ title="Update"
12
+ body="A customer-facing update."
13
+ />
14
+ );
15
+ }
@@ -0,0 +1,15 @@
1
+ import { StaticPublicPage } from "@/features/public-shell/components/static-public-page";
2
+
3
+ export const metadata = {
4
+ title: "Blog",
5
+ };
6
+
7
+ export default function BlogIndexPage() {
8
+ return (
9
+ <StaticPublicPage
10
+ eyebrow="Blog"
11
+ title="Updates"
12
+ body="News, releases, and customer notes."
13
+ />
14
+ );
15
+ }