@knotpad/app 0.1.5 → 0.1.7

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 (337) 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 +229 -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 +54 -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_dark.svg +1 -0
  329. package/public/knotpad_icon.svg +1 -0
  330. package/public/knotpad_logo_full.svg +1 -0
  331. package/public/manifest.json +14 -0
  332. package/public/next.svg +1 -0
  333. package/public/sw.js +137 -0
  334. package/public/vercel.svg +1 -0
  335. package/public/window.svg +1 -0
  336. package/tsconfig.json +35 -0
  337. package/brief.js +0 -311
@@ -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;
@@ -0,0 +1,8 @@
1
+ /*
2
+ Warnings:
3
+
4
+ - You are about to drop the column `syncPasswordSet` on the `Workspace` table. All the data in the column will be lost.
5
+
6
+ */
7
+ -- AlterTable
8
+ ALTER TABLE "Workspace" DROP COLUMN "syncPasswordSet";
@@ -0,0 +1,2 @@
1
+ -- AlterTable
2
+ ALTER TABLE "Task" ADD COLUMN "startDate" TIMESTAMP(3);
@@ -0,0 +1,12 @@
1
+ /*
2
+ Warnings:
3
+
4
+ - You are about to drop the column `keyRotatedAt` on the `SyncState` table. All the data in the column will be lost.
5
+ - You are about to drop the column `encryptionSalt` on the `Workspace` table. All the data in the column will be lost.
6
+
7
+ */
8
+ -- AlterTable
9
+ ALTER TABLE "SyncState" DROP COLUMN "keyRotatedAt";
10
+
11
+ -- AlterTable
12
+ ALTER TABLE "Workspace" DROP COLUMN "encryptionSalt";
@@ -0,0 +1,3 @@
1
+ -- Restore per-workspace salt for at-rest note encryption.
2
+ -- (Reverses the encryptionSalt drop in 20260529010600_remove_encryption_fields.)
3
+ ALTER TABLE "Workspace" ADD COLUMN "encryptionSalt" TEXT;
@@ -0,0 +1,17 @@
1
+ -- Folders for organizing notes within a workspace.
2
+ CREATE TABLE "Folder" (
3
+ "id" TEXT NOT NULL,
4
+ "name" TEXT NOT NULL,
5
+ "workspaceId" TEXT NOT NULL,
6
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
7
+ CONSTRAINT "Folder_pkey" PRIMARY KEY ("id")
8
+ );
9
+
10
+ CREATE INDEX "Folder_workspaceId_idx" ON "Folder"("workspaceId");
11
+ CREATE INDEX "Note_folderId_idx" ON "Note"("folderId");
12
+
13
+ ALTER TABLE "Folder" ADD CONSTRAINT "Folder_workspaceId_fkey"
14
+ FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
15
+
16
+ ALTER TABLE "Note" ADD CONSTRAINT "Note_folderId_fkey"
17
+ FOREIGN KEY ("folderId") REFERENCES "Folder"("id") ON DELETE SET NULL ON UPDATE CASCADE;
@@ -0,0 +1,31 @@
1
+ -- Migration: deferred_fixes
2
+ -- 1. SyncState: snapshotVersion for optimistic concurrency on the snapshot blob
3
+ -- 2. PasswordResetToken: password reset flow
4
+ -- 3. ConflictLog: partial unique index to prevent duplicate active conflicts
5
+
6
+ -- 1. SyncState.snapshotVersion
7
+ ALTER TABLE "SyncState" ADD COLUMN "snapshotVersion" INTEGER NOT NULL DEFAULT 0;
8
+
9
+ -- 2. PasswordResetToken table
10
+ CREATE TABLE "PasswordResetToken" (
11
+ "id" TEXT NOT NULL,
12
+ "token" TEXT NOT NULL,
13
+ "userId" TEXT NOT NULL,
14
+ "expiresAt" TIMESTAMP(3) NOT NULL,
15
+ "usedAt" TIMESTAMP(3),
16
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
17
+
18
+ CONSTRAINT "PasswordResetToken_pkey" PRIMARY KEY ("id")
19
+ );
20
+
21
+ CREATE UNIQUE INDEX "PasswordResetToken_token_key" ON "PasswordResetToken"("token");
22
+
23
+ ALTER TABLE "PasswordResetToken"
24
+ ADD CONSTRAINT "PasswordResetToken_userId_fkey"
25
+ FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
26
+
27
+ -- 3. ConflictLog: prevent duplicate unresolved conflicts for the same entity.
28
+ -- Partial unique index (Prisma schema doesn't support partial indexes natively).
29
+ CREATE UNIQUE INDEX IF NOT EXISTS "ConflictLog_active_unique"
30
+ ON "ConflictLog" ("workspaceId", "entityId")
31
+ WHERE "resolvedAt" IS NULL;
@@ -0,0 +1,5 @@
1
+ -- AlterTable
2
+ ALTER TABLE "Note" ADD COLUMN "pendingSync" BOOLEAN NOT NULL DEFAULT false;
3
+
4
+ -- AlterTable
5
+ ALTER TABLE "Task" ADD COLUMN "pendingSync" BOOLEAN NOT NULL DEFAULT false;
@@ -0,0 +1,14 @@
1
+ -- AlterTable
2
+ ALTER TABLE "SyncState" ADD COLUMN "syncLockedAt" TIMESTAMP(3);
3
+
4
+ -- CreateTable
5
+ CREATE TABLE "StripeWebhookEvent" (
6
+ "id" TEXT NOT NULL,
7
+ "type" TEXT NOT NULL,
8
+ "processedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
9
+
10
+ CONSTRAINT "StripeWebhookEvent_pkey" PRIMARY KEY ("id")
11
+ );
12
+
13
+ -- CreateIndex
14
+ CREATE INDEX "StripeWebhookEvent_processedAt_idx" ON "StripeWebhookEvent"("processedAt");
@@ -0,0 +1,26 @@
1
+ -- Notification: index on userId (every notification query was a full table scan)
2
+ CREATE INDEX IF NOT EXISTS "Notification_userId_idx" ON "Notification"("userId");
3
+ CREATE INDEX IF NOT EXISTS "Notification_userId_read_idx" ON "Notification"("userId", "read");
4
+
5
+ -- Task: index on claimedBy (heartbeat + stale-task cron both filter by this)
6
+ CREATE INDEX IF NOT EXISTS "Task_claimedBy_idx" ON "Task"("claimedBy");
7
+
8
+ -- TaskReference: indexes on both FK columns (were completely missing)
9
+ CREATE INDEX IF NOT EXISTS "TaskReference_taskId_idx" ON "TaskReference"("taskId");
10
+ CREATE INDEX IF NOT EXISTS "TaskReference_noteId_idx" ON "TaskReference"("noteId");
11
+
12
+ -- McpToken: index on userId and workspaceId (token settings page queries)
13
+ CREATE INDEX IF NOT EXISTS "McpToken_userId_idx" ON "McpToken"("userId");
14
+ CREATE INDEX IF NOT EXISTS "McpToken_workspaceId_idx" ON "McpToken"("workspaceId");
15
+
16
+ -- ConflictLog: unique constraint to prevent duplicate conflict entries per entity
17
+ -- Deduplicate first in case any duplicates exist from before this migration.
18
+ DELETE FROM "ConflictLog" c1
19
+ USING "ConflictLog" c2
20
+ WHERE c1."id" > c2."id"
21
+ AND c1."workspaceId" = c2."workspaceId"
22
+ AND c1."entityType" = c2."entityType"
23
+ AND c1."entityId" = c2."entityId";
24
+
25
+ CREATE UNIQUE INDEX IF NOT EXISTS "ConflictLog_workspaceId_entityType_entityId_key"
26
+ ON "ConflictLog"("workspaceId", "entityType", "entityId");
@@ -0,0 +1,23 @@
1
+ -- CreateTable
2
+ CREATE TABLE "KanbanStatus" (
3
+ "id" TEXT NOT NULL,
4
+ "workspaceId" TEXT NOT NULL,
5
+ "key" TEXT NOT NULL,
6
+ "label" TEXT NOT NULL,
7
+ "color" TEXT NOT NULL,
8
+ "order" INTEGER NOT NULL DEFAULT 0,
9
+ "isVisible" BOOLEAN NOT NULL DEFAULT true,
10
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
11
+ "updatedAt" TIMESTAMP(3) NOT NULL,
12
+
13
+ CONSTRAINT "KanbanStatus_pkey" PRIMARY KEY ("id")
14
+ );
15
+
16
+ -- CreateIndex
17
+ CREATE INDEX "KanbanStatus_workspaceId_idx" ON "KanbanStatus"("workspaceId");
18
+
19
+ -- CreateIndex
20
+ CREATE UNIQUE INDEX "KanbanStatus_workspaceId_key_key" ON "KanbanStatus"("workspaceId", "key");
21
+
22
+ -- AddForeignKey
23
+ ALTER TABLE "KanbanStatus" ADD CONSTRAINT "KanbanStatus_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,43 @@
1
+ -- CreateTable
2
+ CREATE TABLE "CalendarFeed" (
3
+ "id" TEXT NOT NULL,
4
+ "userId" TEXT NOT NULL,
5
+ "label" TEXT NOT NULL,
6
+ "encryptedUrl" TEXT NOT NULL,
7
+ "urlSalt" TEXT NOT NULL,
8
+ "enabled" BOOLEAN NOT NULL DEFAULT true,
9
+ "lastFetchedAt" TIMESTAMP(3),
10
+ "lastError" TEXT,
11
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
12
+ "updatedAt" TIMESTAMP(3) NOT NULL,
13
+
14
+ CONSTRAINT "CalendarFeed_pkey" PRIMARY KEY ("id")
15
+ );
16
+
17
+ -- CreateTable
18
+ CREATE TABLE "CalendarFeedEvent" (
19
+ "id" TEXT NOT NULL,
20
+ "feedId" TEXT NOT NULL,
21
+ "uid" TEXT NOT NULL,
22
+ "title" TEXT NOT NULL,
23
+ "start" TIMESTAMP(3) NOT NULL,
24
+ "end" TIMESTAMP(3) NOT NULL,
25
+ "allDay" BOOLEAN NOT NULL DEFAULT false,
26
+
27
+ CONSTRAINT "CalendarFeedEvent_pkey" PRIMARY KEY ("id")
28
+ );
29
+
30
+ -- CreateIndex
31
+ CREATE INDEX "CalendarFeed_userId_idx" ON "CalendarFeed"("userId");
32
+
33
+ -- CreateIndex
34
+ CREATE INDEX "CalendarFeedEvent_feedId_start_idx" ON "CalendarFeedEvent"("feedId", "start");
35
+
36
+ -- CreateIndex
37
+ CREATE UNIQUE INDEX "CalendarFeedEvent_feedId_uid_key" ON "CalendarFeedEvent"("feedId", "uid");
38
+
39
+ -- AddForeignKey
40
+ ALTER TABLE "CalendarFeed" ADD CONSTRAINT "CalendarFeed_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
41
+
42
+ -- AddForeignKey
43
+ ALTER TABLE "CalendarFeedEvent" ADD CONSTRAINT "CalendarFeedEvent_feedId_fkey" FOREIGN KEY ("feedId") REFERENCES "CalendarFeed"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,2 @@
1
+ -- AlterTable
2
+ ALTER TABLE "CalendarFeed" ADD COLUMN "color" TEXT NOT NULL DEFAULT 'sky';
@@ -0,0 +1,14 @@
1
+ -- CreateEnum
2
+ CREATE TYPE "Priority" AS ENUM ('HIGH', 'MEDIUM', 'LOW');
3
+
4
+ -- AlterTable
5
+ ALTER TABLE "Task" ADD COLUMN "priority" "Priority" NOT NULL DEFAULT 'MEDIUM';
6
+
7
+ -- CreateIndex
8
+ CREATE INDEX "Task_workspaceId_priority_idx" ON "Task"("workspaceId", "priority");
9
+
10
+ -- DropIndex
11
+ DROP INDEX "Task_workspaceId_priority_idx";
12
+
13
+ -- CreateIndex
14
+ CREATE INDEX "Task_workspaceId_priority_idx" ON "Task"("workspaceId", "priority");
@@ -0,0 +1,2 @@
1
+ -- AlterEnum
2
+ ALTER TYPE "Priority" ADD VALUE 'CRITICAL';
@@ -0,0 +1,25 @@
1
+ -- CreateEnum
2
+ CREATE TYPE "BackupCadence" AS ENUM ('DAILY', 'WEEKLY', 'MONTHLY');
3
+
4
+ -- CreateTable
5
+ CREATE TABLE "WorkspaceBackupSettings" (
6
+ "id" TEXT NOT NULL,
7
+ "workspaceId" TEXT NOT NULL,
8
+ "scheduleEnabled" BOOLEAN NOT NULL DEFAULT false,
9
+ "scheduleCadence" "BackupCadence" NOT NULL DEFAULT 'WEEKLY',
10
+ "destinationPath" TEXT,
11
+ "includeMarkdownZip" BOOLEAN NOT NULL DEFAULT false,
12
+ "lastBackupAt" TIMESTAMP(3),
13
+ "lastBackupStatus" TEXT NOT NULL DEFAULT 'idle',
14
+ "lastBackupError" TEXT,
15
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
16
+ "updatedAt" TIMESTAMP(3) NOT NULL,
17
+
18
+ CONSTRAINT "WorkspaceBackupSettings_pkey" PRIMARY KEY ("id")
19
+ );
20
+
21
+ -- CreateIndex
22
+ CREATE UNIQUE INDEX "WorkspaceBackupSettings_workspaceId_key" ON "WorkspaceBackupSettings"("workspaceId");
23
+
24
+ -- AddForeignKey
25
+ ALTER TABLE "WorkspaceBackupSettings" ADD CONSTRAINT "WorkspaceBackupSettings_workspaceId_fkey" FOREIGN KEY ("workspaceId") REFERENCES "Workspace"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,20 @@
1
+ -- CreateTable
2
+ CREATE TABLE "Feedback" (
3
+ "id" TEXT NOT NULL,
4
+ "userId" TEXT NOT NULL,
5
+ "type" TEXT NOT NULL,
6
+ "subject" TEXT NOT NULL,
7
+ "message" TEXT NOT NULL,
8
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
9
+
10
+ CONSTRAINT "Feedback_pkey" PRIMARY KEY ("id")
11
+ );
12
+
13
+ -- CreateIndex
14
+ CREATE INDEX "Feedback_userId_idx" ON "Feedback"("userId");
15
+
16
+ -- CreateIndex
17
+ CREATE INDEX "Feedback_createdAt_idx" ON "Feedback"("createdAt");
18
+
19
+ -- AddForeignKey
20
+ ALTER TABLE "Feedback" ADD CONSTRAINT "Feedback_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
@@ -0,0 +1,4 @@
1
+ -- Add 2FA fields to User table
2
+ ALTER TABLE "User" ADD COLUMN "twoFactorSecret" TEXT;
3
+ ALTER TABLE "User" ADD COLUMN "twoFactorEnabled" BOOLEAN NOT NULL DEFAULT false;
4
+ ALTER TABLE "User" ADD COLUMN "twoFactorVerified" BOOLEAN NOT NULL DEFAULT false;
@@ -0,0 +1,3 @@
1
+ # Please do not edit this file manually
2
+ # It should be added in your version-control system (e.g., Git)
3
+ provider = "postgresql"