@knotpad/app 0.1.4 → 0.1.6

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 (340) hide show
  1. package/app/(app)/calendar/page.tsx +57 -0
  2. package/app/(app)/error.tsx +35 -0
  3. package/app/(app)/graph/page.tsx +32 -0
  4. package/app/(app)/guide/page.tsx +21 -0
  5. package/app/(app)/kanban/loading.tsx +24 -0
  6. package/app/(app)/kanban/page.tsx +59 -0
  7. package/app/(app)/layout.tsx +122 -0
  8. package/app/(app)/list/loading.tsx +21 -0
  9. package/app/(app)/list/page.tsx +137 -0
  10. package/app/(app)/loading.tsx +18 -0
  11. package/app/(app)/notes/[noteId]/page.tsx +84 -0
  12. package/app/(app)/notes/layout.tsx +30 -0
  13. package/app/(app)/notes/page.tsx +39 -0
  14. package/app/(app)/page.tsx +5 -0
  15. package/app/(app)/settings/agent-token/page.tsx +59 -0
  16. package/app/(app)/settings/backup/page.tsx +49 -0
  17. package/app/(app)/settings/billing/page.tsx +53 -0
  18. package/app/(app)/settings/calendar/page.tsx +41 -0
  19. package/app/(app)/settings/layout.test.tsx +39 -0
  20. package/app/(app)/settings/layout.tsx +71 -0
  21. package/app/(app)/settings/page.tsx +4 -0
  22. package/app/(app)/settings/security/page.tsx +43 -0
  23. package/app/(app)/settings/team/page.tsx +74 -0
  24. package/app/(app)/settings/workspace/page.tsx +27 -0
  25. package/app/(app)/tasks/[taskId]/page.tsx +79 -0
  26. package/app/(auth)/forgot-password/page.tsx +106 -0
  27. package/app/(auth)/guest/page.tsx +56 -0
  28. package/app/(auth)/layout.tsx +13 -0
  29. package/app/(auth)/login/page.tsx +14 -0
  30. package/app/(auth)/register/page.tsx +193 -0
  31. package/app/(auth)/reset-password/page.tsx +138 -0
  32. package/app/api/account/claim/route.tsx +135 -0
  33. package/app/api/admin/backfill-encryption/route.tsx +43 -0
  34. package/app/api/admin/license/route.tsx +42 -0
  35. package/app/api/auth/2fa/route.tsx +148 -0
  36. package/app/api/auth/[...nextauth]/route.tsx +3 -0
  37. package/app/api/auth/change-password/route.tsx +61 -0
  38. package/app/api/auth/check-2fa/route.tsx +19 -0
  39. package/app/api/auth/forgot-password/route.tsx +65 -0
  40. package/app/api/auth/reset-password/route.tsx +52 -0
  41. package/app/api/auth/verify-2fa/route.tsx +88 -0
  42. package/app/api/backup/download/db/route.ts +29 -0
  43. package/app/api/backup/download/notes/route.ts +25 -0
  44. package/app/api/backup/settings/route.ts +92 -0
  45. package/app/api/billing/checkout/route.tsx +81 -0
  46. package/app/api/billing/migrate/route.tsx +163 -0
  47. package/app/api/billing/portal/route.tsx +24 -0
  48. package/app/api/billing/setup-intent/route.tsx +55 -0
  49. package/app/api/billing/status/route.tsx +36 -0
  50. package/app/api/billing/subscribe/route.tsx +85 -0
  51. package/app/api/billing/webhook/route.tsx +199 -0
  52. package/app/api/calendar-feeds/[feedId]/route.tsx +67 -0
  53. package/app/api/calendar-feeds/[feedId]/sync/route.tsx +37 -0
  54. package/app/api/calendar-feeds/events/route.tsx +82 -0
  55. package/app/api/calendar-feeds/route.tsx +52 -0
  56. package/app/api/calendar-feeds/sync-all/route.tsx +34 -0
  57. package/app/api/cron/calendar-feeds/route.tsx +31 -0
  58. package/app/api/cron/stale-tasks/route.tsx +51 -0
  59. package/app/api/cron/sync/route.tsx +34 -0
  60. package/app/api/devices/[deviceId]/route.tsx +25 -0
  61. package/app/api/devices/route.tsx +41 -0
  62. package/app/api/export/route.tsx +40 -0
  63. package/app/api/feedback/route.tsx +54 -0
  64. package/app/api/folders/[folderId]/route.tsx +51 -0
  65. package/app/api/folders/route.tsx +37 -0
  66. package/app/api/graph/route.tsx +242 -0
  67. package/app/api/guest/route.tsx +58 -0
  68. package/app/api/health/route.tsx +10 -0
  69. package/app/api/holidays/countries/route.tsx +14 -0
  70. package/app/api/holidays/route.tsx +49 -0
  71. package/app/api/holidays/states/route.tsx +21 -0
  72. package/app/api/invites/[token]/route.tsx +131 -0
  73. package/app/api/invites/route.tsx +74 -0
  74. package/app/api/mcp/generate-token/route.tsx +55 -0
  75. package/app/api/mcp/revoke-token/[tokenId]/route.tsx +30 -0
  76. package/app/api/mcp/update-alias/[tokenId]/route.tsx +22 -0
  77. package/app/api/notes/[noteId]/export/route.tsx +45 -0
  78. package/app/api/notes/[noteId]/route.tsx +360 -0
  79. package/app/api/notes/route.tsx +112 -0
  80. package/app/api/notifications/route.tsx +44 -0
  81. package/app/api/register/route.tsx +67 -0
  82. package/app/api/restore/route.tsx +148 -0
  83. package/app/api/sync/conflicts/[conflictId]/route.tsx +134 -0
  84. package/app/api/sync/conflicts/route.tsx +48 -0
  85. package/app/api/sync/status/route.tsx +49 -0
  86. package/app/api/sync/trigger/route.tsx +15 -0
  87. package/app/api/tasks/[taskId]/detail/route.tsx +68 -0
  88. package/app/api/tasks/[taskId]/route.tsx +259 -0
  89. package/app/api/tasks/bulk/route.tsx +133 -0
  90. package/app/api/tasks/route.tsx +36 -0
  91. package/app/api/workspace/active/route.tsx +39 -0
  92. package/app/api/workspace/create-team/route.tsx +42 -0
  93. package/app/api/workspace/kanban-statuses/route.tsx +71 -0
  94. package/app/api/workspace/members/[memberId]/route.tsx +69 -0
  95. package/app/api/workspace/route.tsx +24 -0
  96. package/app/download/page.tsx +170 -0
  97. package/app/favicon.ico +0 -0
  98. package/app/generated/prisma/client.d.ts +1 -0
  99. package/app/generated/prisma/client.js +5 -0
  100. package/app/generated/prisma/default.d.ts +1 -0
  101. package/app/generated/prisma/default.js +5 -0
  102. package/app/generated/prisma/edge.d.ts +1 -0
  103. package/app/generated/prisma/edge.js +497 -0
  104. package/app/generated/prisma/index-browser.js +523 -0
  105. package/app/generated/prisma/index.d.ts +46376 -0
  106. package/app/generated/prisma/index.js +497 -0
  107. package/app/generated/prisma/package.json +144 -0
  108. package/app/generated/prisma/query_compiler_fast_bg.js +2 -0
  109. package/app/generated/prisma/query_compiler_fast_bg.wasm +0 -0
  110. package/app/generated/prisma/query_compiler_fast_bg.wasm-base64.js +2 -0
  111. package/app/generated/prisma/runtime/client.d.ts +3386 -0
  112. package/app/generated/prisma/runtime/client.js +86 -0
  113. package/app/generated/prisma/runtime/index-browser.d.ts +90 -0
  114. package/app/generated/prisma/runtime/index-browser.js +6 -0
  115. package/app/generated/prisma/runtime/wasm-compiler-edge.js +76 -0
  116. package/app/generated/prisma/schema.prisma +456 -0
  117. package/app/generated/prisma/wasm-edge-light-loader.mjs +5 -0
  118. package/app/generated/prisma/wasm-worker-loader.mjs +5 -0
  119. package/app/globals.css +54 -0
  120. package/app/invite/[token]/page.tsx +52 -0
  121. package/app/layout.tsx +90 -0
  122. package/app/mcp/route.tsx +430 -0
  123. package/app/opengraph-image.tsx +120 -0
  124. package/app/page.tsx +398 -0
  125. package/app/privacy/page.tsx +69 -0
  126. package/app/robots.tsx +25 -0
  127. package/app/sitemap.tsx +36 -0
  128. package/app/terms/page.tsx +69 -0
  129. package/app/upgrade/page.tsx +75 -0
  130. package/auth.config.ts +33 -0
  131. package/auth.ts +79 -0
  132. package/bin/brief.js +224 -0
  133. package/components/auth/login-form.tsx +302 -0
  134. package/components/auth/password-checklist.tsx +31 -0
  135. package/components/auth/password-input.tsx +36 -0
  136. package/components/auth/switch-account-button.test.tsx +22 -0
  137. package/components/auth/switch-account-button.tsx +19 -0
  138. package/components/auth/two-factor-input.tsx +116 -0
  139. package/components/billing/billing-dashboard.tsx +265 -0
  140. package/components/billing/card-form.tsx +210 -0
  141. package/components/billing/claim-account-form.tsx +99 -0
  142. package/components/branding/app-logo.test.tsx +20 -0
  143. package/components/branding/app-logo.tsx +25 -0
  144. package/components/calendar/calendar-agenda.tsx +150 -0
  145. package/components/calendar/calendar-drag.test.tsx +177 -0
  146. package/components/calendar/calendar-grid.tsx +357 -0
  147. package/components/calendar/calendar-hooks.test.tsx +27 -0
  148. package/components/calendar/calendar-hooks.ts +351 -0
  149. package/components/calendar/calendar-toolbar.test.tsx +68 -0
  150. package/components/calendar/calendar-toolbar.tsx +291 -0
  151. package/components/calendar/calendar-types.ts +148 -0
  152. package/components/calendar/calendar-view.test.tsx +295 -0
  153. package/components/calendar/calendar-view.tsx +307 -0
  154. package/components/calendar/day-detail-popover.tsx +174 -0
  155. package/components/calendar/task-chip.tsx +86 -0
  156. package/components/command/command-palette.test.tsx +33 -0
  157. package/components/command/command-palette.tsx +310 -0
  158. package/components/download-cta.tsx +87 -0
  159. package/components/feedback/feedback-popup.tsx +207 -0
  160. package/components/graph/graph-draw.ts +337 -0
  161. package/components/graph/graph-overlays.tsx +160 -0
  162. package/components/graph/graph-page.test.tsx +131 -0
  163. package/components/graph/graph-page.tsx +263 -0
  164. package/components/graph/graph-types.ts +47 -0
  165. package/components/graph/graph-view.tsx +322 -0
  166. package/components/guide/guide-view.tsx +522 -0
  167. package/components/kanban/kanban-board.test.tsx +128 -0
  168. package/components/kanban/kanban-board.tsx +361 -0
  169. package/components/kanban/kanban-card-menu.tsx +102 -0
  170. package/components/kanban/kanban-card.tsx +227 -0
  171. package/components/kanban/kanban-column.tsx +49 -0
  172. package/components/kanban/kanban-status-context.tsx +28 -0
  173. package/components/landing/calendar-sandbox.test.tsx +15 -0
  174. package/components/landing/calendar-sandbox.tsx +107 -0
  175. package/components/landing/graph-sandbox.test.tsx +27 -0
  176. package/components/landing/graph-sandbox.tsx +80 -0
  177. package/components/landing/kanban-sandbox.test.tsx +24 -0
  178. package/components/landing/kanban-sandbox.tsx +101 -0
  179. package/components/landing/landing-showcase.test.tsx +21 -0
  180. package/components/landing/landing-showcase.tsx +54 -0
  181. package/components/landing/list-sandbox.tsx +86 -0
  182. package/components/landing/mock-workspace.ts +168 -0
  183. package/components/landing/notes-sandbox.test.tsx +14 -0
  184. package/components/landing/notes-sandbox.tsx +88 -0
  185. package/components/layout/app-shell.tsx +83 -0
  186. package/components/layout/backup-scheduler.tsx +122 -0
  187. package/components/layout/bottom-nav.tsx +43 -0
  188. package/components/layout/icon-bar.test.tsx +29 -0
  189. package/components/layout/icon-bar.tsx +118 -0
  190. package/components/layout/mobile-top-bar.tsx +68 -0
  191. package/components/layout/notes-panel-folder.tsx +127 -0
  192. package/components/layout/notes-panel-note-item.tsx +140 -0
  193. package/components/layout/notes-panel-task-tab.tsx +63 -0
  194. package/components/layout/notes-panel-types.ts +44 -0
  195. package/components/layout/notes-panel.tsx +476 -0
  196. package/components/layout/notification-bell.tsx +251 -0
  197. package/components/layout/paywall-screen.tsx +41 -0
  198. package/components/layout/pro-banner.tsx +76 -0
  199. package/components/layout/sw-register.tsx +27 -0
  200. package/components/layout/workspace-switcher.tsx +90 -0
  201. package/components/notes/mobile-bottom-sheet.tsx +99 -0
  202. package/components/notes/note-editor-context-menu.tsx +47 -0
  203. package/components/notes/note-editor-dom.ts +33 -0
  204. package/components/notes/note-editor-dropdowns.tsx +484 -0
  205. package/components/notes/note-editor-hooks.ts +692 -0
  206. package/components/notes/note-editor-keyboard.ts +305 -0
  207. package/components/notes/note-editor-overlay.tsx +90 -0
  208. package/components/notes/note-editor.test.tsx +372 -0
  209. package/components/notes/note-editor.tsx +662 -0
  210. package/components/notes/note-preview-pane.tsx +156 -0
  211. package/components/notes/note-tabs.tsx +120 -0
  212. package/components/notes/note-types.tsx +157 -0
  213. package/components/settings/accept-invite.tsx +108 -0
  214. package/components/settings/agent-token-settings.tsx +369 -0
  215. package/components/settings/backup-restore-settings.test.tsx +25 -0
  216. package/components/settings/backup-restore-settings.tsx +327 -0
  217. package/components/settings/calendar-feeds-settings.tsx +489 -0
  218. package/components/settings/calendar-general-settings.tsx +174 -0
  219. package/components/settings/confirm-danger-action.test.tsx +215 -0
  220. package/components/settings/confirm-danger-action.tsx +65 -0
  221. package/components/settings/security-settings.tsx +252 -0
  222. package/components/settings/settings-guidance.test.tsx +98 -0
  223. package/components/settings/team-settings.tsx +319 -0
  224. package/components/settings/two-factor-auth.tsx +296 -0
  225. package/components/settings/workspace-settings-client.tsx +363 -0
  226. package/components/settings/workspace-settings-form.tsx +73 -0
  227. package/components/sync/conflict-viewer.tsx +247 -0
  228. package/components/sync/sync-indicator.tsx +171 -0
  229. package/components/tasks/snippet-thread.tsx +119 -0
  230. package/components/tasks/status-dot.tsx +47 -0
  231. package/components/tasks/task-badge.tsx +43 -0
  232. package/components/tasks/task-detail.test.tsx +187 -0
  233. package/components/tasks/task-detail.tsx +458 -0
  234. package/components/tasks/task-list-filters.test.tsx +75 -0
  235. package/components/tasks/task-list-filters.tsx +163 -0
  236. package/components/tasks/task-list-types.ts +20 -0
  237. package/components/tasks/task-list.test.tsx +175 -0
  238. package/components/tasks/task-list.tsx +481 -0
  239. package/components/tasks/task-row.tsx +85 -0
  240. package/components/tasks/task-table-row.tsx +259 -0
  241. package/components/ui/skeleton.tsx +3 -0
  242. package/components/ui/toast.test.tsx +42 -0
  243. package/components/ui/toast.tsx +70 -0
  244. package/instrumentation.tsx +23 -0
  245. package/lib/api-error.ts +50 -0
  246. package/lib/backup/backup-runner.test.ts +32 -0
  247. package/lib/backup/backup-runner.ts +19 -0
  248. package/lib/backup/backup-schedule.test.ts +23 -0
  249. package/lib/backup/backup-schedule.ts +55 -0
  250. package/lib/backup/backup-settings.test.ts +30 -0
  251. package/lib/backup/backup-settings.ts +27 -0
  252. package/lib/backup/export-notes-zip.test.ts +26 -0
  253. package/lib/backup/export-notes-zip.ts +82 -0
  254. package/lib/backup/export-workspace-backup.test.ts +17 -0
  255. package/lib/backup/export-workspace-backup.ts +77 -0
  256. package/lib/backup/restore-workspace-from-export.test.ts +18 -0
  257. package/lib/backup/restore-workspace-from-export.ts +183 -0
  258. package/lib/backup/types.ts +14 -0
  259. package/lib/brand-icons.ts +1 -0
  260. package/lib/calendar-feed-crypto.ts +38 -0
  261. package/lib/calendar-feed.ts +239 -0
  262. package/lib/client/online-status.ts +47 -0
  263. package/lib/conflict-resolver.test.ts +57 -0
  264. package/lib/conflict-resolver.ts +240 -0
  265. package/lib/db-init.ts +79 -0
  266. package/lib/email.ts +159 -0
  267. package/lib/encryption.test.ts +41 -0
  268. package/lib/encryption.ts +98 -0
  269. package/lib/extract-snippet.test.ts +123 -0
  270. package/lib/extract-snippet.ts +69 -0
  271. package/lib/kanban-status.ts +55 -0
  272. package/lib/license.ts +21 -0
  273. package/lib/limits.ts +31 -0
  274. package/lib/mcp-auth.test.ts +58 -0
  275. package/lib/mcp-auth.ts +65 -0
  276. package/lib/mcp-contract.test.ts +25 -0
  277. package/lib/mcp-contract.ts +210 -0
  278. package/lib/mcp-handler.ts +31 -0
  279. package/lib/mcp-url.test.ts +12 -0
  280. package/lib/mcp-url.ts +7 -0
  281. package/lib/mentions.test.ts +45 -0
  282. package/lib/mentions.ts +73 -0
  283. package/lib/note-crypto.ts +108 -0
  284. package/lib/note-sync.ts +201 -0
  285. package/lib/note-title.ts +93 -0
  286. package/lib/prisma.ts +193 -0
  287. package/lib/pro-flush.ts +292 -0
  288. package/lib/rate-limit.ts +57 -0
  289. package/lib/stripe.ts +38 -0
  290. package/lib/sync-worker.ts +388 -0
  291. package/lib/task-parser.test.ts +91 -0
  292. package/lib/task-parser.ts +81 -0
  293. package/lib/task-utils.ts +52 -0
  294. package/lib/use-is-electron.ts +19 -0
  295. package/lib/use-is-mobile.ts +22 -0
  296. package/lib/validation/calendar-feed.ts +31 -0
  297. package/lib/validation/note.ts +27 -0
  298. package/lib/validation/task.ts +26 -0
  299. package/lib/view-preferences.test.ts +54 -0
  300. package/lib/view-preferences.ts +28 -0
  301. package/lib/workspace.ts +66 -0
  302. package/next.config.ts +21 -0
  303. package/package.json +49 -3
  304. package/postcss.config.mjs +7 -0
  305. package/prisma/migrations/20260519021916_init/migration.sql +388 -0
  306. package/prisma/migrations/20260519061113_drop_sync_password/migration.sql +8 -0
  307. package/prisma/migrations/20260520065016_add_task_start_date/migration.sql +2 -0
  308. package/prisma/migrations/20260529010600_remove_encryption_fields/migration.sql +12 -0
  309. package/prisma/migrations/20260529020000_restore_encryption_salt/migration.sql +3 -0
  310. package/prisma/migrations/20260529030000_add_folders/migration.sql +17 -0
  311. package/prisma/migrations/20260605000000_deferred_fixes/migration.sql +31 -0
  312. package/prisma/migrations/20260605020806_add_pending_sync_to_note_and_task/migration.sql +5 -0
  313. package/prisma/migrations/20260605063634_add_stripe_webhook_event_sync_lock/migration.sql +14 -0
  314. package/prisma/migrations/20260605100000_add_prod_indexes/migration.sql +26 -0
  315. package/prisma/migrations/20260608081404_add_kanban_statuses/migration.sql +23 -0
  316. package/prisma/migrations/20260611032723_add_calendar_feeds/migration.sql +43 -0
  317. package/prisma/migrations/20260611040000_add_calendar_feed_color/migration.sql +2 -0
  318. package/prisma/migrations/20260611050000_add_task_priority/migration.sql +14 -0
  319. package/prisma/migrations/20260612060000_add_critical_priority/migration.sql +2 -0
  320. package/prisma/migrations/20260613090000_add_backup_settings/migration.sql +25 -0
  321. package/prisma/migrations/20260614160000_add_feedback/migration.sql +20 -0
  322. package/prisma/migrations/20260614210000_add_2fa/migration.sql +4 -0
  323. package/prisma/migrations/migration_lock.toml +3 -0
  324. package/prisma/schema.prisma +457 -0
  325. package/public/Logo_icon.svg +1 -0
  326. package/public/file.svg +1 -0
  327. package/public/globe.svg +1 -0
  328. package/public/icon-192.png +0 -0
  329. package/public/icon-512.png +0 -0
  330. package/public/icon.svg +4 -0
  331. package/public/icon_dark.svg +1 -0
  332. package/public/knotpad_icon.svg +1 -0
  333. package/public/knotpad_logo_full.svg +1 -0
  334. package/public/manifest.json +14 -0
  335. package/public/next.svg +1 -0
  336. package/public/sw.js +137 -0
  337. package/public/vercel.svg +1 -0
  338. package/public/window.svg +1 -0
  339. package/tsconfig.json +35 -0
  340. package/brief.js +0 -311
@@ -0,0 +1,27 @@
1
+ import { z } from "zod";
2
+ import { MAX_CONTENT_LEN, MAX_TITLE_LEN } from "@/lib/limits";
3
+
4
+ export const createNoteSchema = z.object({
5
+ title: z
6
+ .string()
7
+ .max(MAX_TITLE_LEN, `title must be ≤ ${MAX_TITLE_LEN} chars`)
8
+ .optional(),
9
+ content: z
10
+ .string()
11
+ .max(MAX_CONTENT_LEN, `content exceeds ${MAX_CONTENT_LEN} char limit`)
12
+ .nullish(),
13
+ folderId: z.string().nullish(),
14
+ });
15
+
16
+ export const updateNoteSchema = z.object({
17
+ title: z
18
+ .string()
19
+ .max(MAX_TITLE_LEN, `title must be ≤ ${MAX_TITLE_LEN} chars`)
20
+ .optional(),
21
+ content: z
22
+ .string()
23
+ .max(MAX_CONTENT_LEN, `content exceeds ${MAX_CONTENT_LEN} char limit`)
24
+ .optional(),
25
+ baseVersion: z.number().int().optional(),
26
+ folderId: z.string().nullable().optional(),
27
+ });
@@ -0,0 +1,26 @@
1
+ import { z } from "zod";
2
+
3
+ // ISO date string, or null to clear, or omitted to leave unchanged.
4
+ const dateField = z.string().nullable().optional();
5
+
6
+ export const updateTaskSchema = z
7
+ .object({
8
+ // Status keys are workspace-specific (custom kanban columns), so the exact
9
+ // allowed set is validated in the handler against getKanbanStatuses().
10
+ status: z.string().optional(),
11
+ priority: z.enum(["LOW", "MEDIUM", "HIGH", "CRITICAL"]).optional(),
12
+ startDate: dateField,
13
+ dueDate: dateField,
14
+ assigneeType: z.enum(["HUMAN", "AGENT"]).optional(),
15
+ assigneeId: z.string().nullable().optional(),
16
+ })
17
+ .refine(
18
+ (d) =>
19
+ d.status !== undefined ||
20
+ d.priority !== undefined ||
21
+ d.startDate !== undefined ||
22
+ d.dueDate !== undefined ||
23
+ d.assigneeType !== undefined ||
24
+ d.assigneeId !== undefined,
25
+ { message: "status, priority, startDate, dueDate, assigneeType, or assigneeId required" }
26
+ );
@@ -0,0 +1,54 @@
1
+ import { beforeEach, describe, expect, it, vi } from "vitest";
2
+ import { loadPreference, savePreference } from "@/lib/view-preferences";
3
+
4
+ describe("view preferences", () => {
5
+ beforeEach(() => {
6
+ vi.unstubAllGlobals();
7
+ });
8
+
9
+ it("reads a saved boolean preference", () => {
10
+ const getItem = vi.fn().mockReturnValue("true");
11
+
12
+ vi.stubGlobal("window", {
13
+ localStorage: {
14
+ getItem,
15
+ setItem: vi.fn(),
16
+ },
17
+ });
18
+
19
+ expect(loadPreference("brief:test", false)).toBe(true);
20
+ expect(getItem).toHaveBeenCalledWith("brief:test");
21
+ });
22
+
23
+ it("falls back when storage contains invalid JSON", () => {
24
+ vi.stubGlobal("window", {
25
+ localStorage: {
26
+ getItem: vi.fn().mockReturnValue("{"),
27
+ setItem: vi.fn(),
28
+ },
29
+ });
30
+
31
+ expect(loadPreference("brief:test", "fallback")).toBe("fallback");
32
+ });
33
+
34
+ it("falls back when storage is unavailable", () => {
35
+ vi.stubGlobal("window", undefined);
36
+
37
+ expect(loadPreference("brief:test", "fallback")).toBe("fallback");
38
+ });
39
+
40
+ it("saves a serialized preference", () => {
41
+ const setItem = vi.fn();
42
+
43
+ vi.stubGlobal("window", {
44
+ localStorage: {
45
+ getItem: vi.fn(),
46
+ setItem,
47
+ },
48
+ });
49
+
50
+ savePreference("brief:test", { open: true });
51
+
52
+ expect(setItem).toHaveBeenCalledWith("brief:test", JSON.stringify({ open: true }));
53
+ });
54
+ });
@@ -0,0 +1,28 @@
1
+ export function loadPreference<T>(key: string, fallback: T): T {
2
+ if (typeof window === "undefined") {
3
+ return fallback;
4
+ }
5
+
6
+ try {
7
+ const raw = window.localStorage.getItem(key);
8
+ if (raw === null) {
9
+ return fallback;
10
+ }
11
+
12
+ return JSON.parse(raw) as T;
13
+ } catch {
14
+ return fallback;
15
+ }
16
+ }
17
+
18
+ export function savePreference<T>(key: string, value: T): void {
19
+ if (typeof window === "undefined") {
20
+ return;
21
+ }
22
+
23
+ try {
24
+ window.localStorage.setItem(key, JSON.stringify(value));
25
+ } catch {
26
+ // Preferences should never block the UI.
27
+ }
28
+ }
@@ -0,0 +1,66 @@
1
+ /**
2
+ * Active-workspace resolution.
3
+ *
4
+ * A user can belong to multiple workspaces (a Personal one + Team ones). The
5
+ * "active" workspace is remembered in the `brief_ws` cookie; everything in the
6
+ * main app (notes, the board/list/calendar/graph views, and the tasks reached
7
+ * from notes) resolves through here so switching is consistent.
8
+ *
9
+ * Falls back to the user's first workspace when the cookie is absent or points
10
+ * at a workspace they're no longer a member of.
11
+ */
12
+
13
+ import { cookies } from "next/headers";
14
+ import { prisma } from "@/lib/prisma";
15
+ import { isWorkspacePro } from "@/lib/license";
16
+
17
+ export const ACTIVE_WS_COOKIE = "brief_ws";
18
+
19
+ const IS_CLOUD = process.env.IS_CLOUD === "true";
20
+
21
+ /**
22
+ * On IS_CLOUD=true, only workspaces where isWorkspacePro() is true may be
23
+ * resolved as "active" — a FREE Personal workspace must never be selectable
24
+ * on web/PWA, even if it's the user's only or first-joined workspace.
25
+ */
26
+ export async function getActiveWorkspaceId(userId: string): Promise<string | null> {
27
+ const store = await cookies();
28
+ const preferred = store.get(ACTIVE_WS_COOKIE)?.value;
29
+ if (preferred) {
30
+ const m = await prisma.workspaceMember.findFirst({
31
+ where: { userId, workspaceId: preferred, revokedAt: null },
32
+ select: { workspaceId: true, workspace: { select: { planType: true, licenseType: true } } },
33
+ });
34
+ if (m && (!IS_CLOUD || isWorkspacePro(m.workspace))) return m.workspaceId;
35
+ }
36
+ const members = await prisma.workspaceMember.findMany({
37
+ where: { userId, revokedAt: null },
38
+ orderBy: { joinedAt: "asc" },
39
+ select: { workspaceId: true, workspace: { select: { planType: true, licenseType: true } } },
40
+ });
41
+ const first = IS_CLOUD ? members.find((m) => isWorkspacePro(m.workspace)) : members[0];
42
+ return first?.workspaceId ?? null;
43
+ }
44
+
45
+ export type UserWorkspace = {
46
+ id: string;
47
+ name: string;
48
+ type: "PERSONAL" | "TEAM";
49
+ role: "OWNER" | "ADMIN" | "MEMBER";
50
+ isPro: boolean;
51
+ };
52
+
53
+ export async function listUserWorkspaces(userId: string): Promise<UserWorkspace[]> {
54
+ const members = await prisma.workspaceMember.findMany({
55
+ where: { userId, revokedAt: null },
56
+ include: { workspace: { select: { id: true, name: true, type: true, planType: true, licenseType: true } } },
57
+ orderBy: { joinedAt: "asc" },
58
+ });
59
+ return members.map((m) => ({
60
+ id: m.workspace.id,
61
+ name: m.workspace.name,
62
+ type: m.workspace.type as "PERSONAL" | "TEAM",
63
+ role: m.role as "OWNER" | "ADMIN" | "MEMBER",
64
+ isPro: isWorkspacePro(m.workspace),
65
+ }));
66
+ }
package/next.config.ts ADDED
@@ -0,0 +1,21 @@
1
+ import type { NextConfig } from "next";
2
+ import path from "path";
3
+
4
+ const nextConfig: NextConfig = {
5
+ // Remove .ts from pageExtensions so that middleware.ts and proxy.ts are not
6
+ // auto-detected as middleware/proxy files by Next.js 16. Route handlers use
7
+ // .tsx (copied from .ts) to remain detectable.
8
+ pageExtensions: ["tsx", "jsx", "js"],
9
+ // PGlite and its adapter use WASM and native bindings — Turbopack cannot
10
+ // bundle them. Marking them external emits runtime require() calls instead,
11
+ // which are only reached inside initLocalPrisma() (guarded by IS_CLOUD check)
12
+ // so they're never executed in cloud mode even if the packages aren't installed.
13
+ serverExternalPackages: ["@electric-sql/pglite", "pglite-prisma-adapter", "node-ical"],
14
+ turbopack: {
15
+ // Fix for git worktree: point to this project's directory as the turbopack root
16
+ // so it doesn't infer the parent repo as the workspace root.
17
+ root: path.resolve(__dirname),
18
+ },
19
+ };
20
+
21
+ export default nextConfig;
package/package.json CHANGED
@@ -1,15 +1,61 @@
1
1
  {
2
2
  "name": "@knotpad/app",
3
- "version": "0.1.4",
3
+ "version": "0.1.6",
4
4
  "description": "Note-first project management with AI agent task routing",
5
5
  "bin": {
6
- "knotpad": "brief.js"
6
+ "knotpad": "bin/brief.js"
7
7
  },
8
8
  "files": [
9
9
  "package.json",
10
- "brief.js"
10
+ "next.config.ts",
11
+ "app",
12
+ "components",
13
+ "lib",
14
+ "prisma",
15
+ "public",
16
+ "bin",
17
+ "auth.ts",
18
+ "auth.config.ts",
19
+ "middleware.ts",
20
+ "tailwind.config.ts",
21
+ "tsconfig.json",
22
+ "postcss.config.mjs",
23
+ "instrumentation.tsx"
11
24
  ],
12
25
  "publishConfig": {
13
26
  "access": "public"
27
+ },
28
+ "scripts": {
29
+ "postinstall": "prisma generate"
30
+ },
31
+ "dependencies": {
32
+ "@auth/prisma-adapter": "^2.11.2",
33
+ "@dnd-kit/core": "^6.3.1",
34
+ "@dnd-kit/sortable": "^10.0.0",
35
+ "@dnd-kit/utilities": "^3.2.2",
36
+ "@electric-sql/pglite": "^0.4.1",
37
+ "@neondatabase/serverless": "^1.1.0",
38
+ "@prisma/adapter-neon": "^7.8.0",
39
+ "@prisma/adapter-pg": "^7.8.0",
40
+ "@stripe/react-stripe-js": "^6.3.0",
41
+ "@stripe/stripe-js": "^9.5.0",
42
+ "bcryptjs": "^3.0.3",
43
+ "d3": "^7.9.0",
44
+ "date-holidays": "^3.30.2",
45
+ "dotenv": "^17.4.2",
46
+ "fflate": "^0.8.3",
47
+ "jose": "^6.2.3",
48
+ "lucide-react": "^1.16.0",
49
+ "next": "16.2.6",
50
+ "next-auth": "^5.0.0-beta.31",
51
+ "node-ical": "^0.26.1",
52
+ "pg": "^8.20.0",
53
+ "pglite-prisma-adapter": "^0.7.2",
54
+ "react": "19.2.4",
55
+ "react-dom": "19.2.4",
56
+ "react-markdown": "^10.1.0",
57
+ "remark-gfm": "^4.0.1",
58
+ "stripe": "^22.1.1",
59
+ "zod": "^4.4.3"
14
60
  }
15
61
  }
@@ -0,0 +1,7 @@
1
+ const config = {
2
+ plugins: {
3
+ "@tailwindcss/postcss": {},
4
+ },
5
+ };
6
+
7
+ export default config;
@@ -0,0 +1,388 @@
1
+ -- CreateEnum
2
+ CREATE TYPE "Role" AS ENUM ('OWNER', 'ADMIN', 'MEMBER');
3
+
4
+ -- CreateEnum
5
+ CREATE TYPE "TaskStatus" AS ENUM ('OPEN', 'CLAIMED', 'IN_PROGRESS', 'REVIEW', 'DONE');
6
+
7
+ -- CreateEnum
8
+ CREATE TYPE "AssigneeType" AS ENUM ('HUMAN', 'AGENT');
9
+
10
+ -- CreateEnum
11
+ CREATE TYPE "WorkspaceType" AS ENUM ('PERSONAL', 'TEAM');
12
+
13
+ -- CreateEnum
14
+ CREATE TYPE "PlanType" AS ENUM ('FREE', 'PERSONAL_PRO', 'TEAM_PRO');
15
+
16
+ -- CreateEnum
17
+ CREATE TYPE "LicenseType" AS ENUM ('STANDARD', 'COMPLIMENTARY');
18
+
19
+ -- CreateTable
20
+ CREATE TABLE "User" (
21
+ "id" TEXT NOT NULL,
22
+ "email" TEXT,
23
+ "name" TEXT,
24
+ "passwordHash" TEXT,
25
+ "image" TEXT,
26
+ "isPro" BOOLEAN NOT NULL DEFAULT false,
27
+ "role" "Role" NOT NULL DEFAULT 'MEMBER',
28
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
29
+
30
+ CONSTRAINT "User_pkey" PRIMARY KEY ("id")
31
+ );
32
+
33
+ -- CreateTable
34
+ CREATE TABLE "Workspace" (
35
+ "id" TEXT NOT NULL,
36
+ "name" TEXT NOT NULL,
37
+ "slug" TEXT NOT NULL,
38
+ "type" "WorkspaceType" NOT NULL DEFAULT 'PERSONAL',
39
+ "planType" "PlanType" NOT NULL DEFAULT 'FREE',
40
+ "licenseType" "LicenseType" NOT NULL DEFAULT 'STANDARD',
41
+ "isCloud" BOOLEAN NOT NULL DEFAULT false,
42
+ "isPro" BOOLEAN NOT NULL DEFAULT false,
43
+ "seatCount" INTEGER NOT NULL DEFAULT 1,
44
+ "stripeId" TEXT,
45
+ "stripeSubId" TEXT,
46
+ "encryptionSalt" TEXT,
47
+ "syncPasswordSet" BOOLEAN NOT NULL DEFAULT false,
48
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
49
+
50
+ CONSTRAINT "Workspace_pkey" PRIMARY KEY ("id")
51
+ );
52
+
53
+ -- CreateTable
54
+ CREATE TABLE "WorkspaceMember" (
55
+ "id" TEXT NOT NULL,
56
+ "userId" TEXT NOT NULL,
57
+ "workspaceId" TEXT NOT NULL,
58
+ "role" "Role" NOT NULL DEFAULT 'MEMBER',
59
+ "mcpAlias" TEXT,
60
+ "joinedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
61
+ "revokedAt" TIMESTAMP(3),
62
+
63
+ CONSTRAINT "WorkspaceMember_pkey" PRIMARY KEY ("id")
64
+ );
65
+
66
+ -- CreateTable
67
+ CREATE TABLE "Note" (
68
+ "id" TEXT NOT NULL,
69
+ "title" TEXT NOT NULL,
70
+ "content" TEXT NOT NULL,
71
+ "workspaceId" TEXT NOT NULL,
72
+ "folderId" TEXT,
73
+ "isLocked" BOOLEAN NOT NULL DEFAULT false,
74
+ "cloudOnly" BOOLEAN NOT NULL DEFAULT false,
75
+ "version" INTEGER NOT NULL DEFAULT 0,
76
+ "deviceId" TEXT,
77
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
78
+ "updatedAt" TIMESTAMP(3) NOT NULL,
79
+
80
+ CONSTRAINT "Note_pkey" PRIMARY KEY ("id")
81
+ );
82
+
83
+ -- CreateTable
84
+ CREATE TABLE "Task" (
85
+ "id" TEXT NOT NULL,
86
+ "title" TEXT NOT NULL,
87
+ "status" "TaskStatus" NOT NULL DEFAULT 'OPEN',
88
+ "noteId" TEXT NOT NULL,
89
+ "workspaceId" TEXT NOT NULL,
90
+ "assigneeId" TEXT,
91
+ "assigneeType" "AssigneeType" NOT NULL DEFAULT 'HUMAN',
92
+ "claimedBy" TEXT,
93
+ "claimedByAlias" TEXT,
94
+ "claimedAt" TIMESTAMP(3),
95
+ "lastHeartbeat" TIMESTAMP(3),
96
+ "fileRefs" TEXT[],
97
+ "dueDate" TIMESTAMP(3),
98
+ "syncLocal" BOOLEAN NOT NULL DEFAULT true,
99
+ "version" INTEGER NOT NULL DEFAULT 0,
100
+ "deviceId" TEXT,
101
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
102
+ "updatedAt" TIMESTAMP(3) NOT NULL,
103
+
104
+ CONSTRAINT "Task_pkey" PRIMARY KEY ("id")
105
+ );
106
+
107
+ -- CreateTable
108
+ CREATE TABLE "TaskReference" (
109
+ "id" TEXT NOT NULL,
110
+ "taskId" TEXT NOT NULL,
111
+ "noteId" TEXT NOT NULL,
112
+ "snippet" TEXT NOT NULL,
113
+
114
+ CONSTRAINT "TaskReference_pkey" PRIMARY KEY ("id")
115
+ );
116
+
117
+ -- CreateTable
118
+ CREATE TABLE "McpToken" (
119
+ "id" TEXT NOT NULL,
120
+ "token" TEXT NOT NULL,
121
+ "userId" TEXT NOT NULL,
122
+ "workspaceId" TEXT NOT NULL,
123
+ "alias" TEXT,
124
+ "lastUsed" TIMESTAMP(3),
125
+ "revokedAt" TIMESTAMP(3),
126
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
127
+
128
+ CONSTRAINT "McpToken_pkey" PRIMARY KEY ("id")
129
+ );
130
+
131
+ -- CreateTable
132
+ CREATE TABLE "DeviceSession" (
133
+ "id" TEXT NOT NULL,
134
+ "userId" TEXT NOT NULL,
135
+ "deviceName" TEXT NOT NULL,
136
+ "deviceId" TEXT NOT NULL,
137
+ "lastActive" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
138
+ "revokedAt" TIMESTAMP(3),
139
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
140
+
141
+ CONSTRAINT "DeviceSession_pkey" PRIMARY KEY ("id")
142
+ );
143
+
144
+ -- CreateTable
145
+ CREATE TABLE "AuditLog" (
146
+ "id" TEXT NOT NULL,
147
+ "taskId" TEXT NOT NULL,
148
+ "userId" TEXT,
149
+ "action" TEXT NOT NULL,
150
+ "detail" TEXT,
151
+ "deviceId" TEXT,
152
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
153
+
154
+ CONSTRAINT "AuditLog_pkey" PRIMARY KEY ("id")
155
+ );
156
+
157
+ -- CreateTable
158
+ CREATE TABLE "SyncState" (
159
+ "id" TEXT NOT NULL,
160
+ "workspaceId" TEXT NOT NULL,
161
+ "lastSyncedAt" TIMESTAMP(3),
162
+ "localVersion" INTEGER NOT NULL DEFAULT 0,
163
+ "cloudVersion" INTEGER NOT NULL DEFAULT 0,
164
+ "keyRotatedAt" TIMESTAMP(3),
165
+ "noteSnapshots" TEXT,
166
+
167
+ CONSTRAINT "SyncState_pkey" PRIMARY KEY ("id")
168
+ );
169
+
170
+ -- CreateTable
171
+ CREATE TABLE "ConflictLog" (
172
+ "id" TEXT NOT NULL,
173
+ "workspaceId" TEXT NOT NULL,
174
+ "entityType" TEXT NOT NULL,
175
+ "entityId" TEXT NOT NULL,
176
+ "localValue" TEXT NOT NULL,
177
+ "cloudValue" TEXT NOT NULL,
178
+ "resolvedBy" TEXT,
179
+ "resolvedAt" TIMESTAMP(3),
180
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
181
+
182
+ CONSTRAINT "ConflictLog_pkey" PRIMARY KEY ("id")
183
+ );
184
+
185
+ -- CreateTable
186
+ CREATE TABLE "Tombstone" (
187
+ "id" TEXT NOT NULL,
188
+ "workspaceId" TEXT NOT NULL,
189
+ "entityType" TEXT NOT NULL,
190
+ "entityId" TEXT NOT NULL,
191
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
192
+
193
+ CONSTRAINT "Tombstone_pkey" PRIMARY KEY ("id")
194
+ );
195
+
196
+ -- CreateTable
197
+ CREATE TABLE "Account" (
198
+ "id" TEXT NOT NULL,
199
+ "userId" TEXT NOT NULL,
200
+ "type" TEXT NOT NULL,
201
+ "provider" TEXT NOT NULL,
202
+ "providerAccountId" TEXT NOT NULL,
203
+ "refresh_token" TEXT,
204
+ "access_token" TEXT,
205
+ "expires_at" INTEGER,
206
+ "token_type" TEXT,
207
+ "scope" TEXT,
208
+ "id_token" TEXT,
209
+ "session_state" TEXT,
210
+
211
+ CONSTRAINT "Account_pkey" PRIMARY KEY ("id")
212
+ );
213
+
214
+ -- CreateTable
215
+ CREATE TABLE "Session" (
216
+ "id" TEXT NOT NULL,
217
+ "sessionToken" TEXT NOT NULL,
218
+ "userId" TEXT NOT NULL,
219
+ "expires" TIMESTAMP(3) NOT NULL,
220
+
221
+ CONSTRAINT "Session_pkey" PRIMARY KEY ("id")
222
+ );
223
+
224
+ -- CreateTable
225
+ CREATE TABLE "VerificationToken" (
226
+ "identifier" TEXT NOT NULL,
227
+ "token" TEXT NOT NULL,
228
+ "expires" TIMESTAMP(3) NOT NULL
229
+ );
230
+
231
+ -- CreateTable
232
+ CREATE TABLE "InviteToken" (
233
+ "id" TEXT NOT NULL,
234
+ "token" TEXT NOT NULL,
235
+ "email" TEXT NOT NULL,
236
+ "workspaceId" TEXT NOT NULL,
237
+ "role" "Role" NOT NULL DEFAULT 'MEMBER',
238
+ "invitedById" TEXT NOT NULL,
239
+ "expiresAt" TIMESTAMP(3) NOT NULL,
240
+ "acceptedAt" TIMESTAMP(3),
241
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
242
+
243
+ CONSTRAINT "InviteToken_pkey" PRIMARY KEY ("id")
244
+ );
245
+
246
+ -- CreateTable
247
+ CREATE TABLE "Notification" (
248
+ "id" TEXT NOT NULL,
249
+ "userId" TEXT NOT NULL,
250
+ "type" TEXT NOT NULL,
251
+ "title" TEXT NOT NULL,
252
+ "body" TEXT,
253
+ "read" BOOLEAN NOT NULL DEFAULT false,
254
+ "taskId" TEXT,
255
+ "noteId" TEXT,
256
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
257
+
258
+ CONSTRAINT "Notification_pkey" PRIMARY KEY ("id")
259
+ );
260
+
261
+ -- CreateIndex
262
+ CREATE UNIQUE INDEX "User_email_key" ON "User"("email");
263
+
264
+ -- CreateIndex
265
+ CREATE UNIQUE INDEX "Workspace_slug_key" ON "Workspace"("slug");
266
+
267
+ -- CreateIndex
268
+ CREATE INDEX "WorkspaceMember_userId_idx" ON "WorkspaceMember"("userId");
269
+
270
+ -- CreateIndex
271
+ CREATE INDEX "WorkspaceMember_workspaceId_idx" ON "WorkspaceMember"("workspaceId");
272
+
273
+ -- CreateIndex
274
+ CREATE UNIQUE INDEX "WorkspaceMember_userId_workspaceId_key" ON "WorkspaceMember"("userId", "workspaceId");
275
+
276
+ -- CreateIndex
277
+ CREATE INDEX "Note_workspaceId_idx" ON "Note"("workspaceId");
278
+
279
+ -- CreateIndex
280
+ CREATE INDEX "Note_workspaceId_updatedAt_idx" ON "Note"("workspaceId", "updatedAt");
281
+
282
+ -- CreateIndex
283
+ CREATE INDEX "Task_workspaceId_idx" ON "Task"("workspaceId");
284
+
285
+ -- CreateIndex
286
+ CREATE INDEX "Task_noteId_idx" ON "Task"("noteId");
287
+
288
+ -- CreateIndex
289
+ CREATE INDEX "Task_workspaceId_status_idx" ON "Task"("workspaceId", "status");
290
+
291
+ -- CreateIndex
292
+ CREATE INDEX "Task_workspaceId_assigneeType_status_idx" ON "Task"("workspaceId", "assigneeType", "status");
293
+
294
+ -- CreateIndex
295
+ CREATE UNIQUE INDEX "McpToken_token_key" ON "McpToken"("token");
296
+
297
+ -- CreateIndex
298
+ CREATE UNIQUE INDEX "DeviceSession_deviceId_key" ON "DeviceSession"("deviceId");
299
+
300
+ -- CreateIndex
301
+ CREATE INDEX "AuditLog_taskId_idx" ON "AuditLog"("taskId");
302
+
303
+ -- CreateIndex
304
+ CREATE UNIQUE INDEX "SyncState_workspaceId_key" ON "SyncState"("workspaceId");
305
+
306
+ -- CreateIndex
307
+ CREATE INDEX "ConflictLog_workspaceId_resolvedAt_idx" ON "ConflictLog"("workspaceId", "resolvedAt");
308
+
309
+ -- CreateIndex
310
+ CREATE INDEX "Tombstone_workspaceId_entityType_entityId_idx" ON "Tombstone"("workspaceId", "entityType", "entityId");
311
+
312
+ -- CreateIndex
313
+ CREATE UNIQUE INDEX "Account_provider_providerAccountId_key" ON "Account"("provider", "providerAccountId");
314
+
315
+ -- CreateIndex
316
+ CREATE UNIQUE INDEX "Session_sessionToken_key" ON "Session"("sessionToken");
317
+
318
+ -- CreateIndex
319
+ CREATE UNIQUE INDEX "VerificationToken_token_key" ON "VerificationToken"("token");
320
+
321
+ -- CreateIndex
322
+ CREATE UNIQUE INDEX "VerificationToken_identifier_token_key" ON "VerificationToken"("identifier", "token");
323
+
324
+ -- CreateIndex
325
+ CREATE UNIQUE INDEX "InviteToken_token_key" ON "InviteToken"("token");
326
+
327
+ -- AddForeignKey
328
+ ALTER TABLE "WorkspaceMember" ADD CONSTRAINT "WorkspaceMember_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
329
+
330
+ -- AddForeignKey
331
+ ALTER TABLE "WorkspaceMember" ADD CONSTRAINT "WorkspaceMember_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
332
+
333
+ -- AddForeignKey
334
+ ALTER TABLE "Note" ADD CONSTRAINT "Note_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
335
+
336
+ -- AddForeignKey
337
+ ALTER TABLE "Task" ADD CONSTRAINT "Task_noteId_fkey" FOREIGN KEY ("noteId") REFERENCES "Note"("id") ON DELETE CASCADE ON UPDATE CASCADE;
338
+
339
+ -- AddForeignKey
340
+ ALTER TABLE "Task" ADD CONSTRAINT "Task_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
341
+
342
+ -- AddForeignKey
343
+ ALTER TABLE "Task" ADD CONSTRAINT "Task_assigneeId_fkey" FOREIGN KEY ("assigneeId") REFERENCES "User"("id") ON DELETE SET NULL ON UPDATE CASCADE;
344
+
345
+ -- AddForeignKey
346
+ ALTER TABLE "TaskReference" ADD CONSTRAINT "TaskReference_taskId_fkey" FOREIGN KEY ("taskId") REFERENCES "Task"("id") ON DELETE CASCADE ON UPDATE CASCADE;
347
+
348
+ -- AddForeignKey
349
+ ALTER TABLE "TaskReference" ADD CONSTRAINT "TaskReference_noteId_fkey" FOREIGN KEY ("noteId") REFERENCES "Note"("id") ON DELETE CASCADE ON UPDATE CASCADE;
350
+
351
+ -- AddForeignKey
352
+ ALTER TABLE "McpToken" ADD CONSTRAINT "McpToken_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
353
+
354
+ -- AddForeignKey
355
+ ALTER TABLE "McpToken" ADD CONSTRAINT "McpToken_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
356
+
357
+ -- AddForeignKey
358
+ ALTER TABLE "DeviceSession" ADD CONSTRAINT "DeviceSession_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
359
+
360
+ -- AddForeignKey
361
+ ALTER TABLE "AuditLog" ADD CONSTRAINT "AuditLog_taskId_fkey" FOREIGN KEY ("taskId") REFERENCES "Task"("id") ON DELETE CASCADE ON UPDATE CASCADE;
362
+
363
+ -- AddForeignKey
364
+ ALTER TABLE "AuditLog" ADD CONSTRAINT "AuditLog_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE SET NULL ON UPDATE CASCADE;
365
+
366
+ -- AddForeignKey
367
+ ALTER TABLE "SyncState" ADD CONSTRAINT "SyncState_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
368
+
369
+ -- AddForeignKey
370
+ ALTER TABLE "ConflictLog" ADD CONSTRAINT "ConflictLog_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
371
+
372
+ -- AddForeignKey
373
+ ALTER TABLE "Tombstone" ADD CONSTRAINT "Tombstone_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
374
+
375
+ -- AddForeignKey
376
+ ALTER TABLE "Account" ADD CONSTRAINT "Account_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
377
+
378
+ -- AddForeignKey
379
+ ALTER TABLE "Session" ADD CONSTRAINT "Session_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
380
+
381
+ -- AddForeignKey
382
+ ALTER TABLE "InviteToken" ADD CONSTRAINT "InviteToken_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
383
+
384
+ -- AddForeignKey
385
+ ALTER TABLE "InviteToken" ADD CONSTRAINT "InviteToken_invitedById_fkey" FOREIGN KEY ("invitedById") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
386
+
387
+ -- AddForeignKey
388
+ ALTER TABLE "Notification" ADD CONSTRAINT "Notification_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;