opensentinel 2.1.1 → 3.1.1

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 (268) hide show
  1. package/README.md +354 -283
  2. package/dist/archiver-AVNBYCKQ.js +15340 -0
  3. package/dist/archiver-AVNBYCKQ.js.map +1 -0
  4. package/dist/audit-logger-OBPR7CRO.js +22 -0
  5. package/dist/auth-UOX5K2BE.js +18 -0
  6. package/dist/autonomy-ZXDBDQUJ.js +86 -0
  7. package/dist/autonomy-ZXDBDQUJ.js.map +1 -0
  8. package/dist/aws-s3-Q4LLZZPD.js +146 -0
  9. package/dist/aws-s3-Q4LLZZPD.js.map +1 -0
  10. package/dist/backup-restore-PZ7CYYB7.js +16 -0
  11. package/dist/blocks-R3PODY47.js +23 -0
  12. package/dist/bot-QRARP4UN.js +36 -0
  13. package/dist/brain-7XLLM3KC.js +56 -0
  14. package/dist/camera-monitor-M5CYKUU4.js +335 -0
  15. package/dist/camera-monitor-M5CYKUU4.js.map +1 -0
  16. package/dist/{charts-MMXM6BWW.js → charts-V7ARZNKF.js} +2 -2
  17. package/dist/chunk-22VGGA7S.js +330 -0
  18. package/dist/chunk-22VGGA7S.js.map +1 -0
  19. package/dist/chunk-35WYTA3C.js +382 -0
  20. package/dist/chunk-35WYTA3C.js.map +1 -0
  21. package/dist/chunk-3E2PSU2C.js +146 -0
  22. package/dist/chunk-3E2PSU2C.js.map +1 -0
  23. package/dist/{chunk-L3F43VPB.js → chunk-4GLYY4NN.js} +2 -2
  24. package/dist/{chunk-L3F43VPB.js.map → chunk-4GLYY4NN.js.map} +1 -1
  25. package/dist/{chunk-L3PDU3XN.js → chunk-4UOE5TUZ.js} +4 -4
  26. package/dist/{chunk-6SNHU3CY.js → chunk-66OJ3WB4.js} +2 -2
  27. package/dist/chunk-6KONMXQ6.js +297 -0
  28. package/dist/chunk-6KONMXQ6.js.map +1 -0
  29. package/dist/chunk-6PMVAAA7.js +196 -0
  30. package/dist/chunk-6PMVAAA7.js.map +1 -0
  31. package/dist/chunk-766ASQWE.js +32620 -0
  32. package/dist/chunk-766ASQWE.js.map +1 -0
  33. package/dist/chunk-7WQO5J2M.js +29 -0
  34. package/dist/chunk-7WQO5J2M.js.map +1 -0
  35. package/dist/chunk-APHSRMBS.js +148 -0
  36. package/dist/chunk-APHSRMBS.js.map +1 -0
  37. package/dist/{chunk-4LVWXUNC.js → chunk-AYUKPTSM.js} +57 -39
  38. package/dist/chunk-AYUKPTSM.js.map +1 -0
  39. package/dist/chunk-BIPYADGB.js +84 -0
  40. package/dist/chunk-BIPYADGB.js.map +1 -0
  41. package/dist/chunk-BRBWNV65.js +457 -0
  42. package/dist/chunk-BRBWNV65.js.map +1 -0
  43. package/dist/chunk-BXZ6EA52.js +382 -0
  44. package/dist/chunk-BXZ6EA52.js.map +1 -0
  45. package/dist/chunk-EVE7MIIY.js +290 -0
  46. package/dist/chunk-EVE7MIIY.js.map +1 -0
  47. package/dist/chunk-F3TTNID2.js +138 -0
  48. package/dist/chunk-F3TTNID2.js.map +1 -0
  49. package/dist/chunk-H5RQOFO2.js +190 -0
  50. package/dist/chunk-H5RQOFO2.js.map +1 -0
  51. package/dist/chunk-HN3F4WSW.js +145 -0
  52. package/dist/chunk-HN3F4WSW.js.map +1 -0
  53. package/dist/{chunk-6DRDKB45.js → chunk-I6BDYQIG.js} +20 -9
  54. package/dist/chunk-I6BDYQIG.js.map +1 -0
  55. package/dist/chunk-IZJMVV7O.js +347 -0
  56. package/dist/chunk-IZJMVV7O.js.map +1 -0
  57. package/dist/chunk-KM22GV7G.js +211 -0
  58. package/dist/chunk-KM22GV7G.js.map +1 -0
  59. package/dist/chunk-MGFBLVR7.js +103 -0
  60. package/dist/chunk-MGFBLVR7.js.map +1 -0
  61. package/dist/chunk-MQJ2ECQT.js +228 -0
  62. package/dist/chunk-MQJ2ECQT.js.map +1 -0
  63. package/dist/{chunk-F6QUZQGI.js → chunk-MXAPLSJ5.js} +2 -2
  64. package/dist/{chunk-GK3E2I7A.js → chunk-NHMBTUMW.js} +2 -2
  65. package/dist/chunk-NPRTSZIF.js +131 -0
  66. package/dist/chunk-NPRTSZIF.js.map +1 -0
  67. package/dist/chunk-O7IH7JTI.js +1898 -0
  68. package/dist/chunk-O7IH7JTI.js.map +1 -0
  69. package/dist/chunk-OCVQGBJK.js +293 -0
  70. package/dist/chunk-OCVQGBJK.js.map +1 -0
  71. package/dist/chunk-P6QINGFL.js +332 -0
  72. package/dist/chunk-P6QINGFL.js.map +1 -0
  73. package/dist/chunk-PHDZKPNE.js +91 -0
  74. package/dist/chunk-PHDZKPNE.js.map +1 -0
  75. package/dist/chunk-PLDDJCW6.js +49 -0
  76. package/dist/chunk-PTGTGXV2.js +164 -0
  77. package/dist/chunk-PTGTGXV2.js.map +1 -0
  78. package/dist/chunk-REMIY4U2.js +171 -0
  79. package/dist/chunk-REMIY4U2.js.map +1 -0
  80. package/dist/chunk-RZ4YESBG.js +141 -0
  81. package/dist/chunk-RZ4YESBG.js.map +1 -0
  82. package/dist/chunk-SAX5MHK4.js +111 -0
  83. package/dist/chunk-SAX5MHK4.js.map +1 -0
  84. package/dist/{chunk-GVJVEWHI.js → chunk-SJSUSJ47.js} +2 -2
  85. package/dist/chunk-SPPMCAKG.js +777 -0
  86. package/dist/chunk-SPPMCAKG.js.map +1 -0
  87. package/dist/chunk-SVAPX2XN.js +2441 -0
  88. package/dist/chunk-SVAPX2XN.js.map +1 -0
  89. package/dist/chunk-TVEWKIK3.js +452 -0
  90. package/dist/chunk-TVEWKIK3.js.map +1 -0
  91. package/dist/{chunk-HH2HBTQM.js → chunk-TYAGMJNV.js} +5 -5
  92. package/dist/{chunk-JXUP2X7V.js → chunk-VEHFVBLI.js} +2 -2
  93. package/dist/chunk-VNX5GMTN.js +128 -0
  94. package/dist/chunk-VNX5GMTN.js.map +1 -0
  95. package/dist/chunk-VRD5CYRL.js +1568 -0
  96. package/dist/chunk-VRD5CYRL.js.map +1 -0
  97. package/dist/chunk-WLUHNG6X.js +122 -0
  98. package/dist/chunk-WLUHNG6X.js.map +1 -0
  99. package/dist/chunk-WRAKK6K6.js +265 -0
  100. package/dist/chunk-WRAKK6K6.js.map +1 -0
  101. package/dist/chunk-XKYRH4FM.js +681 -0
  102. package/dist/chunk-XKYRH4FM.js.map +1 -0
  103. package/dist/{chunk-GUBEEYDW.js → chunk-XMCVRVTF.js} +2 -2
  104. package/dist/{chunk-GUBEEYDW.js.map → chunk-XMCVRVTF.js.map} +1 -1
  105. package/dist/chunk-ZLZKF2PM.js +310 -0
  106. package/dist/chunk-ZLZKF2PM.js.map +1 -0
  107. package/dist/cli.js +5 -1
  108. package/dist/cli.js.map +1 -1
  109. package/dist/client-ZQSFPMOB.js +21 -0
  110. package/dist/clipboard-manager-TEO2GEDN.js +24 -0
  111. package/dist/commands/setup.js +3 -3
  112. package/dist/commands/setup.js.map +1 -1
  113. package/dist/commands/start.js +3 -3
  114. package/dist/commands/status.js +2 -2
  115. package/dist/commands/stop.js +2 -2
  116. package/dist/commands/utils.js +2 -2
  117. package/dist/cron-explain-HHQKPD3M.js +16 -0
  118. package/dist/crypto-4AP47IKC.js +14 -0
  119. package/dist/crypto-4AP47IKC.js.map +1 -0
  120. package/dist/databases-37X4CI2Y.js +21 -0
  121. package/dist/databases-37X4CI2Y.js.map +1 -0
  122. package/dist/discord-B3HUPGQ6.js +70 -0
  123. package/dist/discord-B3HUPGQ6.js.map +1 -0
  124. package/dist/dist-UISMLMFN.js +21847 -0
  125. package/dist/dist-UISMLMFN.js.map +1 -0
  126. package/dist/email-K7LO2IPB.js +268 -0
  127. package/dist/email-K7LO2IPB.js.map +1 -0
  128. package/dist/enhanced-retrieval-DNLLEM4Z.js +753 -0
  129. package/dist/enhanced-retrieval-DNLLEM4Z.js.map +1 -0
  130. package/dist/enrichment-pipeline-MNHNW65K.js +13 -0
  131. package/dist/enrichment-pipeline-MNHNW65K.js.map +1 -0
  132. package/dist/entity-resolution-Y3IUWEAT.js +24 -0
  133. package/dist/entity-resolution-Y3IUWEAT.js.map +1 -0
  134. package/dist/env-IWXUVTCB.js +12 -0
  135. package/dist/env-IWXUVTCB.js.map +1 -0
  136. package/dist/google-workspace-DKWUVNGC.js +169 -0
  137. package/dist/google-workspace-DKWUVNGC.js.map +1 -0
  138. package/dist/hash-tool-ULQYD7B5.js +22 -0
  139. package/dist/hash-tool-ULQYD7B5.js.map +1 -0
  140. package/dist/heartbeat-monitor-GCISLXI3.js +22 -0
  141. package/dist/heartbeat-monitor-GCISLXI3.js.map +1 -0
  142. package/dist/image-generation-OSU7FP6F.js +486 -0
  143. package/dist/image-generation-OSU7FP6F.js.map +1 -0
  144. package/dist/imessage-NGA2XF2V.js +35 -0
  145. package/dist/imessage-NGA2XF2V.js.map +1 -0
  146. package/dist/inbox-summarizer-NRI4S7IF.js +47 -0
  147. package/dist/inbox-summarizer-NRI4S7IF.js.map +1 -0
  148. package/dist/incident-response-C5J7Q6DT.js +244 -0
  149. package/dist/incident-response-C5J7Q6DT.js.map +1 -0
  150. package/dist/inventory-manager-352OHXWD.js +24 -0
  151. package/dist/inventory-manager-352OHXWD.js.map +1 -0
  152. package/dist/jira-GSGDBMIG.js +199 -0
  153. package/dist/jira-GSGDBMIG.js.map +1 -0
  154. package/dist/json-tool-QE2SYHEG.js +26 -0
  155. package/dist/json-tool-QE2SYHEG.js.map +1 -0
  156. package/dist/key-rotation-DPHU4ZTB.js +18 -0
  157. package/dist/key-rotation-DPHU4ZTB.js.map +1 -0
  158. package/dist/lib.d.ts +603 -11
  159. package/dist/lib.js +161 -35
  160. package/dist/lib.js.map +1 -1
  161. package/dist/mailchimp-KKNF6QJ7.js +152 -0
  162. package/dist/mailchimp-KKNF6QJ7.js.map +1 -0
  163. package/dist/matrix-QVHG76I7.js +279 -0
  164. package/dist/matrix-QVHG76I7.js.map +1 -0
  165. package/dist/{mcp-LS7Q3Z5W.js → mcp-3JI6W7ZE.js} +3 -3
  166. package/dist/mcp-3JI6W7ZE.js.map +1 -0
  167. package/dist/microsoft365-UCBKJHNX.js +164 -0
  168. package/dist/microsoft365-UCBKJHNX.js.map +1 -0
  169. package/dist/ocr-AC7NPX33.js +22 -0
  170. package/dist/ocr-AC7NPX33.js.map +1 -0
  171. package/dist/ollama-BOAMSPLJ.js +8 -0
  172. package/dist/ollama-BOAMSPLJ.js.map +1 -0
  173. package/dist/pages-MI523RB7.js +26 -0
  174. package/dist/pages-MI523RB7.js.map +1 -0
  175. package/dist/pair-JDFTERIK.js +24 -0
  176. package/dist/pair-JDFTERIK.js.map +1 -0
  177. package/dist/pairing-IFQYCPNS.js +10 -0
  178. package/dist/pairing-IFQYCPNS.js.map +1 -0
  179. package/dist/pdf-ALQVOEJR.js +17 -0
  180. package/dist/pdf-ALQVOEJR.js.map +1 -0
  181. package/dist/presentations-DSV5IHG5.js +1002 -0
  182. package/dist/presentations-DSV5IHG5.js.map +1 -0
  183. package/dist/prometheus-JNT2BD4L.js +10 -0
  184. package/dist/prometheus-JNT2BD4L.js.map +1 -0
  185. package/dist/providers-J4LYPHDR.js +19 -0
  186. package/dist/providers-J4LYPHDR.js.map +1 -0
  187. package/dist/qr-code-WIX4PB4U.js +16 -0
  188. package/dist/qr-code-WIX4PB4U.js.map +1 -0
  189. package/dist/quickbooks-XB4NII2S.js +190 -0
  190. package/dist/quickbooks-XB4NII2S.js.map +1 -0
  191. package/dist/regex-tool-W4ABRKGK.js +24 -0
  192. package/dist/regex-tool-W4ABRKGK.js.map +1 -0
  193. package/dist/scheduler-VK4WFERV.js +63 -0
  194. package/dist/scheduler-VK4WFERV.js.map +1 -0
  195. package/dist/search-BCLBO5E3.js +25 -0
  196. package/dist/search-BCLBO5E3.js.map +1 -0
  197. package/dist/sendgrid-RNXCAFKM.js +152 -0
  198. package/dist/sendgrid-RNXCAFKM.js.map +1 -0
  199. package/dist/shopify-NCXYJB4R.js +171 -0
  200. package/dist/shopify-NCXYJB4R.js.map +1 -0
  201. package/dist/signal-6CGDFYL2.js +35 -0
  202. package/dist/signal-6CGDFYL2.js.map +1 -0
  203. package/dist/slack-IZQWIKOH.js +75 -0
  204. package/dist/slack-IZQWIKOH.js.map +1 -0
  205. package/dist/sms-M3JIOTCW.js +23 -0
  206. package/dist/sms-M3JIOTCW.js.map +1 -0
  207. package/dist/{src-K7GASHRH.js → src-VYUE6LRA.js} +138 -32
  208. package/dist/src-VYUE6LRA.js.map +1 -0
  209. package/dist/stocks-XXWBPOCU.js +14 -0
  210. package/dist/stocks-XXWBPOCU.js.map +1 -0
  211. package/dist/text-transform-6SGUA5Z4.js +22 -0
  212. package/dist/text-transform-6SGUA5Z4.js.map +1 -0
  213. package/dist/tools-2RLEI2N6.js +38 -0
  214. package/dist/tools-2RLEI2N6.js.map +1 -0
  215. package/dist/tunnel-IWMXUML4.js +301 -0
  216. package/dist/tunnel-IWMXUML4.js.map +1 -0
  217. package/dist/twilio-53GEW5JT.js +139 -0
  218. package/dist/twilio-53GEW5JT.js.map +1 -0
  219. package/dist/unit-converter-ZYXMEZOE.js +14 -0
  220. package/dist/unit-converter-ZYXMEZOE.js.map +1 -0
  221. package/dist/whatsapp-LFX6YKCM.js +35 -0
  222. package/dist/whatsapp-LFX6YKCM.js.map +1 -0
  223. package/dist/word-document-7B6SJMAY.js +902 -0
  224. package/dist/word-document-7B6SJMAY.js.map +1 -0
  225. package/dist/xero-QYO66D45.js +162 -0
  226. package/dist/xero-QYO66D45.js.map +1 -0
  227. package/dist/zapier-webhook-TBZ5YF2A.js +106 -0
  228. package/dist/zapier-webhook-TBZ5YF2A.js.map +1 -0
  229. package/drizzle/0002_mushy_master_mold.sql +140 -0
  230. package/drizzle/meta/0002_snapshot.json +3637 -0
  231. package/drizzle/meta/_journal.json +7 -0
  232. package/package.json +100 -98
  233. package/dist/bot-KJ26BG56.js +0 -15
  234. package/dist/chunk-4LVWXUNC.js.map +0 -1
  235. package/dist/chunk-4TG2IG5K.js +0 -5249
  236. package/dist/chunk-4TG2IG5K.js.map +0 -1
  237. package/dist/chunk-6DRDKB45.js.map +0 -1
  238. package/dist/chunk-CI6Q63MM.js +0 -1613
  239. package/dist/chunk-CI6Q63MM.js.map +0 -1
  240. package/dist/chunk-KHNYJY2Z.js +0 -178
  241. package/dist/chunk-KHNYJY2Z.js.map +0 -1
  242. package/dist/chunk-NSBPE2FW.js +0 -17
  243. package/dist/discord-ZOJFTVTB.js +0 -49
  244. package/dist/imessage-JFRB6EJ7.js +0 -14
  245. package/dist/scheduler-EZ7CZMCS.js +0 -42
  246. package/dist/signal-T3MCSULM.js +0 -14
  247. package/dist/slack-N2M4FHAJ.js +0 -54
  248. package/dist/src-K7GASHRH.js.map +0 -1
  249. package/dist/tools-24GZHYRF.js +0 -16
  250. package/dist/whatsapp-VCRUPAO5.js +0 -14
  251. /package/dist/{bot-KJ26BG56.js.map → audit-logger-OBPR7CRO.js.map} +0 -0
  252. /package/dist/{chunk-NSBPE2FW.js.map → auth-UOX5K2BE.js.map} +0 -0
  253. /package/dist/{discord-ZOJFTVTB.js.map → backup-restore-PZ7CYYB7.js.map} +0 -0
  254. /package/dist/{imessage-JFRB6EJ7.js.map → blocks-R3PODY47.js.map} +0 -0
  255. /package/dist/{mcp-LS7Q3Z5W.js.map → bot-QRARP4UN.js.map} +0 -0
  256. /package/dist/{scheduler-EZ7CZMCS.js.map → brain-7XLLM3KC.js.map} +0 -0
  257. /package/dist/{charts-MMXM6BWW.js.map → charts-V7ARZNKF.js.map} +0 -0
  258. /package/dist/{chunk-L3PDU3XN.js.map → chunk-4UOE5TUZ.js.map} +0 -0
  259. /package/dist/{chunk-6SNHU3CY.js.map → chunk-66OJ3WB4.js.map} +0 -0
  260. /package/dist/{chunk-F6QUZQGI.js.map → chunk-MXAPLSJ5.js.map} +0 -0
  261. /package/dist/{chunk-GK3E2I7A.js.map → chunk-NHMBTUMW.js.map} +0 -0
  262. /package/dist/{signal-T3MCSULM.js.map → chunk-PLDDJCW6.js.map} +0 -0
  263. /package/dist/{chunk-GVJVEWHI.js.map → chunk-SJSUSJ47.js.map} +0 -0
  264. /package/dist/{chunk-HH2HBTQM.js.map → chunk-TYAGMJNV.js.map} +0 -0
  265. /package/dist/{chunk-JXUP2X7V.js.map → chunk-VEHFVBLI.js.map} +0 -0
  266. /package/dist/{slack-N2M4FHAJ.js.map → client-ZQSFPMOB.js.map} +0 -0
  267. /package/dist/{tools-24GZHYRF.js.map → clipboard-manager-TEO2GEDN.js.map} +0 -0
  268. /package/dist/{whatsapp-VCRUPAO5.js.map → cron-explain-HHQKPD3M.js.map} +0 -0
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/core/security/key-rotation.ts"],"sourcesContent":["/**\r\n * Key Rotation Module\r\n * Rotates encryption keys and re-encrypts all protected data\r\n */\r\n\r\nimport { randomBytes, createCipheriv, createDecipheriv, scryptSync } from \"crypto\";\r\n\r\nexport interface KeyRotationResult {\r\n success: boolean;\r\n rotatedItems?: number;\r\n newKeyHash?: string;\r\n error?: string;\r\n}\r\n\r\nexport interface KeyRotationOptions {\r\n newKey?: string;\r\n rotateEncryption?: boolean;\r\n rotateAudit?: boolean;\r\n dryRun?: boolean;\r\n}\r\n\r\n// Generate a new encryption key\r\nexport function generateEncryptionKey(): string {\r\n return randomBytes(32).toString(\"hex\");\r\n}\r\n\r\n// Hash a key for safe logging (never log the actual key)\r\nexport function hashKeyForLog(key: string): string {\r\n const hash = scryptSync(key, \"key-id\", 16);\r\n return hash.toString(\"hex\").slice(0, 16) + \"...\";\r\n}\r\n\r\n// Re-encrypt a single value from old key to new key\r\nexport function reEncryptValue(\r\n encryptedData: string,\r\n oldKey: string,\r\n newKey: string\r\n): string {\r\n // Decrypt with old key\r\n const parts = encryptedData.split(\":\");\r\n if (parts.length !== 3) {\r\n throw new Error(\"Invalid encrypted data format (expected iv:tag:ciphertext)\");\r\n }\r\n\r\n const [ivHex, tagHex, ciphertext] = parts;\r\n const oldKeyBuffer = scryptSync(oldKey, \"opensentinel-encryption\", 32);\r\n const iv = Buffer.from(ivHex, \"hex\");\r\n const tag = Buffer.from(tagHex, \"hex\");\r\n\r\n const decipher = createDecipheriv(\"aes-256-gcm\", oldKeyBuffer, iv);\r\n decipher.setAuthTag(tag);\r\n let decrypted = decipher.update(ciphertext, \"hex\", \"utf8\");\r\n decrypted += decipher.final(\"utf8\");\r\n\r\n // Re-encrypt with new key\r\n const newKeyBuffer = scryptSync(newKey, \"opensentinel-encryption\", 32);\r\n const newIv = randomBytes(16);\r\n const cipher = createCipheriv(\"aes-256-gcm\", newKeyBuffer, newIv);\r\n let encrypted = cipher.update(decrypted, \"utf8\", \"hex\");\r\n encrypted += cipher.final(\"hex\");\r\n const newTag = cipher.getAuthTag();\r\n\r\n return `${newIv.toString(\"hex\")}:${newTag.toString(\"hex\")}:${encrypted}`;\r\n}\r\n\r\n// Rotate encryption keys for all encrypted data in the database\r\nexport async function rotateEncryptionKeys(\r\n options: KeyRotationOptions = {}\r\n): Promise<KeyRotationResult> {\r\n const currentKey = process.env.ENCRYPTION_MASTER_KEY;\r\n if (!currentKey) {\r\n return { success: false, error: \"ENCRYPTION_MASTER_KEY not set\" };\r\n }\r\n\r\n const newKey = options.newKey || generateEncryptionKey();\r\n let rotatedItems = 0;\r\n\r\n try {\r\n if (options.dryRun) {\r\n return {\r\n success: true,\r\n rotatedItems: 0,\r\n newKeyHash: hashKeyForLog(newKey),\r\n error: \"Dry run - no changes made. Set the new key as ENCRYPTION_MASTER_KEY and restart.\",\r\n };\r\n }\r\n\r\n // In a real rotation, we'd query all encrypted columns from DB\r\n // and re-encrypt each one. This requires a database connection.\r\n // The pattern for each table:\r\n //\r\n // 1. SELECT id, encrypted_column FROM table WHERE encrypted_column IS NOT NULL\r\n // 2. For each row: reEncryptValue(encrypted_column, currentKey, newKey)\r\n // 3. UPDATE table SET encrypted_column = newValue WHERE id = row.id\r\n //\r\n // Tables with encrypted data:\r\n // - memories (content when encrypted)\r\n // - vault entries (all values)\r\n // - two_factor_auth (secret)\r\n\r\n // For now, we expose the rotation primitives and return the new key\r\n // The operator must update ENCRYPTION_MASTER_KEY in .env and restart\r\n\r\n return {\r\n success: true,\r\n rotatedItems,\r\n newKeyHash: hashKeyForLog(newKey),\r\n };\r\n } catch (error) {\r\n return {\r\n success: false,\r\n error: error instanceof Error ? error.message : String(error),\r\n };\r\n }\r\n}\r\n\r\n// Get rotation status\r\nexport function getRotationInfo(): {\r\n hasEncryptionKey: boolean;\r\n hasAuditKey: boolean;\r\n keyHashPrefix: string | null;\r\n} {\r\n const encKey = process.env.ENCRYPTION_MASTER_KEY;\r\n const auditKey = process.env.AUDIT_SIGNING_KEY;\r\n\r\n return {\r\n hasEncryptionKey: !!encKey,\r\n hasAuditKey: !!auditKey,\r\n keyHashPrefix: encKey ? hashKeyForLog(encKey) : null,\r\n };\r\n}\r\n\r\nexport default {\r\n generateEncryptionKey,\r\n hashKeyForLog,\r\n reEncryptValue,\r\n rotateEncryptionKeys,\r\n getRotationInfo,\r\n};\r\n"],"mappings":";AAKA,SAAS,aAAa,gBAAgB,kBAAkB,kBAAkB;AAiBnE,SAAS,wBAAgC;AAC9C,SAAO,YAAY,EAAE,EAAE,SAAS,KAAK;AACvC;AAGO,SAAS,cAAc,KAAqB;AACjD,QAAM,OAAO,WAAW,KAAK,UAAU,EAAE;AACzC,SAAO,KAAK,SAAS,KAAK,EAAE,MAAM,GAAG,EAAE,IAAI;AAC7C;AAGO,SAAS,eACd,eACA,QACA,QACQ;AAER,QAAM,QAAQ,cAAc,MAAM,GAAG;AACrC,MAAI,MAAM,WAAW,GAAG;AACtB,UAAM,IAAI,MAAM,4DAA4D;AAAA,EAC9E;AAEA,QAAM,CAAC,OAAO,QAAQ,UAAU,IAAI;AACpC,QAAM,eAAe,WAAW,QAAQ,2BAA2B,EAAE;AACrE,QAAM,KAAK,OAAO,KAAK,OAAO,KAAK;AACnC,QAAM,MAAM,OAAO,KAAK,QAAQ,KAAK;AAErC,QAAM,WAAW,iBAAiB,eAAe,cAAc,EAAE;AACjE,WAAS,WAAW,GAAG;AACvB,MAAI,YAAY,SAAS,OAAO,YAAY,OAAO,MAAM;AACzD,eAAa,SAAS,MAAM,MAAM;AAGlC,QAAM,eAAe,WAAW,QAAQ,2BAA2B,EAAE;AACrE,QAAM,QAAQ,YAAY,EAAE;AAC5B,QAAM,SAAS,eAAe,eAAe,cAAc,KAAK;AAChE,MAAI,YAAY,OAAO,OAAO,WAAW,QAAQ,KAAK;AACtD,eAAa,OAAO,MAAM,KAAK;AAC/B,QAAM,SAAS,OAAO,WAAW;AAEjC,SAAO,GAAG,MAAM,SAAS,KAAK,CAAC,IAAI,OAAO,SAAS,KAAK,CAAC,IAAI,SAAS;AACxE;AAGA,eAAsB,qBACpB,UAA8B,CAAC,GACH;AAC5B,QAAM,aAAa,QAAQ,IAAI;AAC/B,MAAI,CAAC,YAAY;AACf,WAAO,EAAE,SAAS,OAAO,OAAO,gCAAgC;AAAA,EAClE;AAEA,QAAM,SAAS,QAAQ,UAAU,sBAAsB;AACvD,MAAI,eAAe;AAEnB,MAAI;AACF,QAAI,QAAQ,QAAQ;AAClB,aAAO;AAAA,QACL,SAAS;AAAA,QACT,cAAc;AAAA,QACd,YAAY,cAAc,MAAM;AAAA,QAChC,OAAO;AAAA,MACT;AAAA,IACF;AAkBA,WAAO;AAAA,MACL,SAAS;AAAA,MACT;AAAA,MACA,YAAY,cAAc,MAAM;AAAA,IAClC;AAAA,EACF,SAAS,OAAO;AACd,WAAO;AAAA,MACL,SAAS;AAAA,MACT,OAAO,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAAA,IAC9D;AAAA,EACF;AACF;AAGO,SAAS,kBAId;AACA,QAAM,SAAS,QAAQ,IAAI;AAC3B,QAAM,WAAW,QAAQ,IAAI;AAE7B,SAAO;AAAA,IACL,kBAAkB,CAAC,CAAC;AAAA,IACpB,aAAa,CAAC,CAAC;AAAA,IACf,eAAe,SAAS,cAAc,MAAM,IAAI;AAAA,EAClD;AACF;AAEA,IAAO,uBAAQ;AAAA,EACb;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF;","names":[]}
@@ -0,0 +1,457 @@
1
+ import {
2
+ getNotionClient,
3
+ getRootPageId,
4
+ init_client
5
+ } from "./chunk-VRD5CYRL.js";
6
+
7
+ // src/integrations/notion/databases.ts
8
+ init_client();
9
+ function buildFilter(condition) {
10
+ const { property, type, operator, value } = condition;
11
+ let propertyType = type;
12
+ if (!propertyType) {
13
+ if ([
14
+ "contains",
15
+ "does_not_contain",
16
+ "starts_with",
17
+ "ends_with",
18
+ "equals",
19
+ "does_not_equal"
20
+ ].includes(operator)) {
21
+ propertyType = "rich_text";
22
+ } else if ([
23
+ "greater_than",
24
+ "less_than",
25
+ "greater_than_or_equal_to",
26
+ "less_than_or_equal_to"
27
+ ].includes(operator)) {
28
+ propertyType = "number";
29
+ } else if ([
30
+ "before",
31
+ "after",
32
+ "on_or_before",
33
+ "on_or_after",
34
+ "past_week",
35
+ "past_month",
36
+ "past_year",
37
+ "next_week",
38
+ "next_month",
39
+ "next_year"
40
+ ].includes(operator)) {
41
+ propertyType = "date";
42
+ } else if (["is_empty", "is_not_empty"].includes(operator)) {
43
+ propertyType = "rich_text";
44
+ }
45
+ }
46
+ const filter = { property };
47
+ switch (propertyType) {
48
+ case "title":
49
+ case "rich_text":
50
+ filter[propertyType || "rich_text"] = { [operator]: value ?? true };
51
+ break;
52
+ case "number":
53
+ filter.number = { [operator]: value };
54
+ break;
55
+ case "checkbox":
56
+ filter.checkbox = { [operator]: value };
57
+ break;
58
+ case "select":
59
+ filter.select = { [operator]: value };
60
+ break;
61
+ case "multi_select":
62
+ filter.multi_select = { [operator]: value };
63
+ break;
64
+ case "date":
65
+ if ([
66
+ "past_week",
67
+ "past_month",
68
+ "past_year",
69
+ "next_week",
70
+ "next_month",
71
+ "next_year"
72
+ ].includes(operator)) {
73
+ filter.date = { [operator]: {} };
74
+ } else {
75
+ filter.date = {
76
+ [operator]: value instanceof Date ? value.toISOString() : value
77
+ };
78
+ }
79
+ break;
80
+ case "url":
81
+ case "email":
82
+ case "phone_number":
83
+ filter[propertyType] = { [operator]: value };
84
+ break;
85
+ case "status":
86
+ filter.status = { [operator]: value };
87
+ break;
88
+ default:
89
+ filter.rich_text = { [operator]: value ?? true };
90
+ }
91
+ return filter;
92
+ }
93
+ async function queryDatabase(databaseId, options = {}) {
94
+ const notion = getNotionClient();
95
+ const params = {
96
+ database_id: databaseId
97
+ };
98
+ if (options.filter) {
99
+ if (Array.isArray(options.filter)) {
100
+ if (options.filter.length === 1) {
101
+ params.filter = buildFilter(options.filter[0]);
102
+ } else {
103
+ const filters = options.filter.map(buildFilter);
104
+ params.filter = options.filterOperator === "or" ? { or: filters } : { and: filters };
105
+ }
106
+ } else {
107
+ params.filter = buildFilter(options.filter);
108
+ }
109
+ }
110
+ if (options.sorts) {
111
+ params.sorts = options.sorts.map((sort) => {
112
+ if (sort.timestamp) {
113
+ return { timestamp: sort.timestamp, direction: sort.direction };
114
+ }
115
+ return { property: sort.property, direction: sort.direction };
116
+ });
117
+ }
118
+ if (options.startCursor) {
119
+ params.start_cursor = options.startCursor;
120
+ }
121
+ if (options.pageSize) {
122
+ params.page_size = options.pageSize;
123
+ }
124
+ const response = await (notion.dataSources?.query?.(params) ?? notion.databases.query(params));
125
+ const results = response.results.map((page) => {
126
+ const p = page;
127
+ return {
128
+ id: p.id,
129
+ url: p.url,
130
+ createdTime: p.created_time,
131
+ lastEditedTime: p.last_edited_time,
132
+ archived: p.archived,
133
+ properties: extractPropertyValues(p.properties)
134
+ };
135
+ });
136
+ return {
137
+ results,
138
+ hasMore: response.has_more,
139
+ nextCursor: response.next_cursor
140
+ };
141
+ }
142
+ async function queryAllDatabaseEntries(databaseId, options = {}) {
143
+ const allResults = [];
144
+ let cursor;
145
+ do {
146
+ const response = await queryDatabase(databaseId, {
147
+ ...options,
148
+ startCursor: cursor,
149
+ pageSize: 100
150
+ });
151
+ allResults.push(...response.results);
152
+ cursor = response.nextCursor ?? void 0;
153
+ } while (cursor);
154
+ return allResults;
155
+ }
156
+ async function getDatabase(databaseId) {
157
+ const notion = getNotionClient();
158
+ const response = await notion.databases.retrieve({
159
+ database_id: databaseId
160
+ });
161
+ const properties = {};
162
+ for (const [name, prop] of Object.entries(response.properties || {})) {
163
+ const p = prop;
164
+ properties[name] = {
165
+ type: p.type,
166
+ name: p.name
167
+ };
168
+ }
169
+ return {
170
+ id: response.id,
171
+ title: response.title.map((t) => t.plain_text).join(""),
172
+ description: response.description.map((t) => t.plain_text).join(""),
173
+ url: response.url,
174
+ createdTime: response.created_time,
175
+ lastEditedTime: response.last_edited_time,
176
+ isInline: response.is_inline,
177
+ properties
178
+ };
179
+ }
180
+ async function createDatabaseEntry(options) {
181
+ const notion = getNotionClient();
182
+ const params = {
183
+ parent: { database_id: options.databaseId },
184
+ properties: buildDatabaseProperties(options.properties)
185
+ };
186
+ if (options.icon) {
187
+ if (options.icon.startsWith("http")) {
188
+ params.icon = { type: "external", external: { url: options.icon } };
189
+ } else {
190
+ params.icon = { type: "emoji", emoji: options.icon };
191
+ }
192
+ }
193
+ if (options.cover) {
194
+ params.cover = { type: "external", external: { url: options.cover } };
195
+ }
196
+ if (options.content) {
197
+ const { markdownToBlocks, createBlockObject } = await import("./blocks-R3PODY47.js");
198
+ const blocks = markdownToBlocks(options.content);
199
+ params.children = blocks.map(createBlockObject);
200
+ }
201
+ const response = await notion.pages.create(params);
202
+ return {
203
+ id: response.id,
204
+ url: response.url,
205
+ createdTime: response.created_time,
206
+ lastEditedTime: response.last_edited_time,
207
+ archived: response.archived,
208
+ properties: extractPropertyValues(response.properties)
209
+ };
210
+ }
211
+ async function updateDatabaseEntry(pageId, properties) {
212
+ const notion = getNotionClient();
213
+ const response = await notion.pages.update({
214
+ page_id: pageId,
215
+ properties: buildDatabaseProperties(properties)
216
+ });
217
+ return {
218
+ id: response.id,
219
+ url: response.url,
220
+ createdTime: response.created_time,
221
+ lastEditedTime: response.last_edited_time,
222
+ archived: response.archived,
223
+ properties: extractPropertyValues(response.properties)
224
+ };
225
+ }
226
+ async function archiveDatabaseEntry(pageId) {
227
+ const notion = getNotionClient();
228
+ await notion.pages.update({
229
+ page_id: pageId,
230
+ archived: true
231
+ });
232
+ }
233
+ async function createDatabase(options) {
234
+ const notion = getNotionClient();
235
+ const parentId = options.parentPageId || getRootPageId();
236
+ if (!parentId) {
237
+ throw new Error("No parent page specified and no root page configured");
238
+ }
239
+ const properties = {};
240
+ for (const [name, config] of Object.entries(options.properties)) {
241
+ properties[name] = buildPropertySchema(config.type, config.options);
242
+ }
243
+ if (!Object.values(properties).some((p) => p.type === "title")) {
244
+ properties["Name"] = { title: {} };
245
+ }
246
+ const response = await notion.databases.create({
247
+ parent: { type: "page_id", page_id: parentId },
248
+ title: [{ text: { content: options.title } }],
249
+ properties,
250
+ is_inline: options.isInline ?? false
251
+ });
252
+ const resultProperties = {};
253
+ for (const [name, prop] of Object.entries(response.properties || {})) {
254
+ const p = prop;
255
+ resultProperties[name] = {
256
+ type: p.type,
257
+ name: p.name
258
+ };
259
+ }
260
+ return {
261
+ id: response.id,
262
+ title: response.title.map((t) => t.plain_text).join(""),
263
+ description: "",
264
+ url: response.url,
265
+ createdTime: response.created_time,
266
+ lastEditedTime: response.last_edited_time,
267
+ isInline: response.is_inline,
268
+ properties: resultProperties
269
+ };
270
+ }
271
+ function buildPropertySchema(type, options) {
272
+ switch (type) {
273
+ case "title":
274
+ return { title: {} };
275
+ case "rich_text":
276
+ return { rich_text: {} };
277
+ case "number":
278
+ return { number: { format: options?.format || "number" } };
279
+ case "select":
280
+ return {
281
+ select: {
282
+ options: options?.options?.map((opt) => ({ name: opt })) || []
283
+ }
284
+ };
285
+ case "multi_select":
286
+ return {
287
+ multi_select: {
288
+ options: options?.options?.map((opt) => ({ name: opt })) || []
289
+ }
290
+ };
291
+ case "date":
292
+ return { date: {} };
293
+ case "checkbox":
294
+ return { checkbox: {} };
295
+ case "url":
296
+ return { url: {} };
297
+ case "email":
298
+ return { email: {} };
299
+ case "phone_number":
300
+ return { phone_number: {} };
301
+ case "status":
302
+ return {
303
+ status: {
304
+ options: options?.options?.map((opt) => ({ name: opt })) || [
305
+ { name: "Not started" },
306
+ { name: "In progress" },
307
+ { name: "Done" }
308
+ ]
309
+ }
310
+ };
311
+ default:
312
+ return { rich_text: {} };
313
+ }
314
+ }
315
+ function buildDatabaseProperties(properties) {
316
+ const result = {};
317
+ for (const [key, value] of Object.entries(properties)) {
318
+ if (value === void 0) continue;
319
+ if (typeof value === "object" && value !== null && value._type) {
320
+ const { _type, ...rest } = value;
321
+ result[key] = buildTypedProperty(_type, rest);
322
+ continue;
323
+ }
324
+ if (typeof value === "string") {
325
+ if (key.toLowerCase() === "name" || key.toLowerCase() === "title") {
326
+ result[key] = { title: [{ text: { content: value } }] };
327
+ } else if (value.startsWith("http://") || value.startsWith("https://")) {
328
+ result[key] = { url: value };
329
+ } else if (value.includes("@") && value.includes(".")) {
330
+ result[key] = { email: value };
331
+ } else {
332
+ result[key] = { rich_text: [{ text: { content: value } }] };
333
+ }
334
+ } else if (typeof value === "number") {
335
+ result[key] = { number: value };
336
+ } else if (typeof value === "boolean") {
337
+ result[key] = { checkbox: value };
338
+ } else if (value instanceof Date) {
339
+ result[key] = { date: { start: value.toISOString() } };
340
+ } else if (Array.isArray(value)) {
341
+ result[key] = { multi_select: value.map((v) => ({ name: String(v) })) };
342
+ }
343
+ }
344
+ return result;
345
+ }
346
+ function buildTypedProperty(type, value) {
347
+ switch (type) {
348
+ case "title":
349
+ return { title: [{ text: { content: value.content || value } }] };
350
+ case "rich_text":
351
+ return { rich_text: [{ text: { content: value.content || value } }] };
352
+ case "number":
353
+ return { number: value.value ?? value };
354
+ case "select":
355
+ return { select: { name: value.name || value } };
356
+ case "multi_select":
357
+ const options = Array.isArray(value) ? value : value.options || [value];
358
+ return { multi_select: options.map((v) => ({ name: v.name || v })) };
359
+ case "date":
360
+ return {
361
+ date: {
362
+ start: value.start instanceof Date ? value.start.toISOString() : value.start || value,
363
+ end: value.end ? value.end instanceof Date ? value.end.toISOString() : value.end : void 0
364
+ }
365
+ };
366
+ case "checkbox":
367
+ return { checkbox: value.checked ?? value };
368
+ case "url":
369
+ return { url: value.url || value };
370
+ case "email":
371
+ return { email: value.email || value };
372
+ case "phone_number":
373
+ return { phone_number: value.phone || value };
374
+ case "status":
375
+ return { status: { name: value.name || value } };
376
+ default:
377
+ return { rich_text: [{ text: { content: String(value) } }] };
378
+ }
379
+ }
380
+ function extractPropertyValues(properties) {
381
+ const result = {};
382
+ for (const [name, prop] of Object.entries(properties)) {
383
+ switch (prop.type) {
384
+ case "title":
385
+ result[name] = prop.title?.map((t) => t.plain_text).join("") || "";
386
+ break;
387
+ case "rich_text":
388
+ result[name] = prop.rich_text?.map((t) => t.plain_text).join("") || "";
389
+ break;
390
+ case "number":
391
+ result[name] = prop.number;
392
+ break;
393
+ case "select":
394
+ result[name] = prop.select?.name || null;
395
+ break;
396
+ case "multi_select":
397
+ result[name] = prop.multi_select?.map((s) => s.name) || [];
398
+ break;
399
+ case "date":
400
+ result[name] = prop.date ? { start: prop.date.start, end: prop.date.end } : null;
401
+ break;
402
+ case "checkbox":
403
+ result[name] = prop.checkbox;
404
+ break;
405
+ case "url":
406
+ result[name] = prop.url;
407
+ break;
408
+ case "email":
409
+ result[name] = prop.email;
410
+ break;
411
+ case "phone_number":
412
+ result[name] = prop.phone_number;
413
+ break;
414
+ case "formula":
415
+ result[name] = prop.formula?.[prop.formula.type];
416
+ break;
417
+ case "relation":
418
+ result[name] = prop.relation?.map((r) => r.id) || [];
419
+ break;
420
+ case "rollup":
421
+ result[name] = prop.rollup?.[prop.rollup.type];
422
+ break;
423
+ case "created_time":
424
+ result[name] = prop.created_time;
425
+ break;
426
+ case "created_by":
427
+ result[name] = prop.created_by?.id;
428
+ break;
429
+ case "last_edited_time":
430
+ result[name] = prop.last_edited_time;
431
+ break;
432
+ case "last_edited_by":
433
+ result[name] = prop.last_edited_by?.id;
434
+ break;
435
+ case "files":
436
+ result[name] = prop.files?.map((f) => f.file?.url || f.external?.url) || [];
437
+ break;
438
+ case "status":
439
+ result[name] = prop.status?.name || null;
440
+ break;
441
+ default:
442
+ result[name] = prop;
443
+ }
444
+ }
445
+ return result;
446
+ }
447
+
448
+ export {
449
+ queryDatabase,
450
+ queryAllDatabaseEntries,
451
+ getDatabase,
452
+ createDatabaseEntry,
453
+ updateDatabaseEntry,
454
+ archiveDatabaseEntry,
455
+ createDatabase
456
+ };
457
+ //# sourceMappingURL=chunk-BRBWNV65.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/integrations/notion/databases.ts"],"sourcesContent":["import { getNotionClient, getRootPageId } from \"./client\";\n\n/**\n * Database operations for Notion\n * Handles querying databases and creating entries\n */\n\n// Type definitions for API responses\ninterface PageObjectResponse {\n id: string;\n url: string;\n created_time: string;\n last_edited_time: string;\n archived: boolean;\n properties: Record<string, any>;\n}\n\ninterface QueryDatabaseParams {\n database_id: string;\n filter?: any;\n sorts?: any[];\n start_cursor?: string;\n page_size?: number;\n}\n\nexport type PropertyType =\n | \"title\"\n | \"rich_text\"\n | \"number\"\n | \"select\"\n | \"multi_select\"\n | \"date\"\n | \"checkbox\"\n | \"url\"\n | \"email\"\n | \"phone_number\"\n | \"formula\"\n | \"relation\"\n | \"rollup\"\n | \"created_time\"\n | \"created_by\"\n | \"last_edited_time\"\n | \"last_edited_by\"\n | \"files\"\n | \"status\";\n\nexport type FilterOperator =\n | \"equals\"\n | \"does_not_equal\"\n | \"contains\"\n | \"does_not_contain\"\n | \"starts_with\"\n | \"ends_with\"\n | \"is_empty\"\n | \"is_not_empty\"\n | \"greater_than\"\n | \"less_than\"\n | \"greater_than_or_equal_to\"\n | \"less_than_or_equal_to\"\n | \"before\"\n | \"after\"\n | \"on_or_before\"\n | \"on_or_after\"\n | \"past_week\"\n | \"past_month\"\n | \"past_year\"\n | \"next_week\"\n | \"next_month\"\n | \"next_year\";\n\nexport interface FilterCondition {\n property: string;\n type?: PropertyType;\n operator: FilterOperator;\n value?: string | number | boolean | Date;\n}\n\nexport interface SortOption {\n property?: string;\n timestamp?: \"created_time\" | \"last_edited_time\";\n direction: \"ascending\" | \"descending\";\n}\n\nexport interface QueryOptions {\n filter?: FilterCondition | FilterCondition[];\n filterOperator?: \"and\" | \"or\";\n sorts?: SortOption[];\n startCursor?: string;\n pageSize?: number;\n}\n\nexport interface DatabaseEntry {\n id: string;\n url: string;\n createdTime: string;\n lastEditedTime: string;\n archived: boolean;\n properties: Record<string, any>;\n}\n\nexport interface DatabaseInfo {\n id: string;\n title: string;\n description: string;\n url: string;\n createdTime: string;\n lastEditedTime: string;\n isInline: boolean;\n properties: Record<string, { type: PropertyType; name: string }>;\n}\n\nexport interface CreateEntryOptions {\n databaseId: string;\n properties: Record<string, any>;\n content?: string;\n icon?: string;\n cover?: string;\n}\n\n/**\n * Convert filter condition to Notion API filter format\n */\nfunction buildFilter(condition: FilterCondition): any {\n const { property, type, operator, value } = condition;\n\n // Determine property type from operator if not specified\n let propertyType = type;\n if (!propertyType) {\n if (\n [\n \"contains\",\n \"does_not_contain\",\n \"starts_with\",\n \"ends_with\",\n \"equals\",\n \"does_not_equal\",\n ].includes(operator)\n ) {\n propertyType = \"rich_text\";\n } else if (\n [\n \"greater_than\",\n \"less_than\",\n \"greater_than_or_equal_to\",\n \"less_than_or_equal_to\",\n ].includes(operator)\n ) {\n propertyType = \"number\";\n } else if (\n [\n \"before\",\n \"after\",\n \"on_or_before\",\n \"on_or_after\",\n \"past_week\",\n \"past_month\",\n \"past_year\",\n \"next_week\",\n \"next_month\",\n \"next_year\",\n ].includes(operator)\n ) {\n propertyType = \"date\";\n } else if ([\"is_empty\", \"is_not_empty\"].includes(operator)) {\n propertyType = \"rich_text\";\n }\n }\n\n const filter: any = { property };\n\n switch (propertyType) {\n case \"title\":\n case \"rich_text\":\n filter[propertyType || \"rich_text\"] = { [operator]: value ?? true };\n break;\n\n case \"number\":\n filter.number = { [operator]: value };\n break;\n\n case \"checkbox\":\n filter.checkbox = { [operator]: value };\n break;\n\n case \"select\":\n filter.select = { [operator]: value };\n break;\n\n case \"multi_select\":\n filter.multi_select = { [operator]: value };\n break;\n\n case \"date\":\n if (\n [\n \"past_week\",\n \"past_month\",\n \"past_year\",\n \"next_week\",\n \"next_month\",\n \"next_year\",\n ].includes(operator)\n ) {\n filter.date = { [operator]: {} };\n } else {\n filter.date = {\n [operator]: value instanceof Date ? value.toISOString() : value,\n };\n }\n break;\n\n case \"url\":\n case \"email\":\n case \"phone_number\":\n filter[propertyType] = { [operator]: value };\n break;\n\n case \"status\":\n filter.status = { [operator]: value };\n break;\n\n default:\n filter.rich_text = { [operator]: value ?? true };\n }\n\n return filter;\n}\n\n/**\n * Query a Notion database\n */\nexport async function queryDatabase(\n databaseId: string,\n options: QueryOptions = {}\n): Promise<{\n results: DatabaseEntry[];\n hasMore: boolean;\n nextCursor: string | null;\n}> {\n const notion = getNotionClient();\n\n const params: QueryDatabaseParams = {\n database_id: databaseId,\n };\n\n // Build filter\n if (options.filter) {\n if (Array.isArray(options.filter)) {\n if (options.filter.length === 1) {\n params.filter = buildFilter(options.filter[0]);\n } else {\n const filters = options.filter.map(buildFilter);\n params.filter =\n options.filterOperator === \"or\"\n ? { or: filters as any }\n : { and: filters as any };\n }\n } else {\n params.filter = buildFilter(options.filter);\n }\n }\n\n // Build sorts\n if (options.sorts) {\n params.sorts = options.sorts.map((sort) => {\n if (sort.timestamp) {\n return { timestamp: sort.timestamp, direction: sort.direction };\n }\n return { property: sort.property!, direction: sort.direction };\n });\n }\n\n // Pagination\n if (options.startCursor) {\n params.start_cursor = options.startCursor;\n }\n if (options.pageSize) {\n params.page_size = options.pageSize;\n }\n\n // Use dataSources.query for newer SDK versions, fallback to databases.query\n const response = await ((notion as any).dataSources?.query?.(params) ??\n (notion as any).databases.query(params));\n\n const results: DatabaseEntry[] = response.results.map((page) => {\n const p = page as PageObjectResponse;\n return {\n id: p.id,\n url: p.url,\n createdTime: p.created_time,\n lastEditedTime: p.last_edited_time,\n archived: p.archived,\n properties: extractPropertyValues(p.properties),\n };\n });\n\n return {\n results,\n hasMore: response.has_more,\n nextCursor: response.next_cursor,\n };\n}\n\n/**\n * Query all results from a database (handles pagination)\n */\nexport async function queryAllDatabaseEntries(\n databaseId: string,\n options: Omit<QueryOptions, \"startCursor\" | \"pageSize\"> = {}\n): Promise<DatabaseEntry[]> {\n const allResults: DatabaseEntry[] = [];\n let cursor: string | undefined;\n\n do {\n const response = await queryDatabase(databaseId, {\n ...options,\n startCursor: cursor,\n pageSize: 100,\n });\n allResults.push(...response.results);\n cursor = response.nextCursor ?? undefined;\n } while (cursor);\n\n return allResults;\n}\n\n/**\n * Get database information\n */\nexport async function getDatabase(databaseId: string): Promise<DatabaseInfo> {\n const notion = getNotionClient();\n\n const response = await notion.databases.retrieve({\n database_id: databaseId,\n }) as any;\n\n const properties: Record<string, { type: PropertyType; name: string }> = {};\n for (const [name, prop] of Object.entries(response.properties || {})) {\n const p = prop as any;\n properties[name] = {\n type: p.type as PropertyType,\n name: p.name,\n };\n }\n\n return {\n id: response.id,\n title: response.title.map((t) => t.plain_text).join(\"\"),\n description: response.description.map((t) => t.plain_text).join(\"\"),\n url: response.url,\n createdTime: response.created_time,\n lastEditedTime: response.last_edited_time,\n isInline: response.is_inline,\n properties,\n };\n}\n\n/**\n * Create an entry in a database\n */\nexport async function createDatabaseEntry(\n options: CreateEntryOptions\n): Promise<DatabaseEntry> {\n const notion = getNotionClient();\n\n const params: any = {\n parent: { database_id: options.databaseId },\n properties: buildDatabaseProperties(options.properties),\n };\n\n // Add icon\n if (options.icon) {\n if (options.icon.startsWith(\"http\")) {\n params.icon = { type: \"external\", external: { url: options.icon } };\n } else {\n params.icon = { type: \"emoji\", emoji: options.icon };\n }\n }\n\n // Add cover\n if (options.cover) {\n params.cover = { type: \"external\", external: { url: options.cover } };\n }\n\n // Add content as children\n if (options.content) {\n const { markdownToBlocks, createBlockObject } = await import(\"./blocks\");\n const blocks = markdownToBlocks(options.content);\n params.children = blocks.map(createBlockObject);\n }\n\n const response = (await notion.pages.create(params)) as PageObjectResponse;\n\n return {\n id: response.id,\n url: response.url,\n createdTime: response.created_time,\n lastEditedTime: response.last_edited_time,\n archived: response.archived,\n properties: extractPropertyValues(response.properties),\n };\n}\n\n/**\n * Update a database entry\n */\nexport async function updateDatabaseEntry(\n pageId: string,\n properties: Record<string, any>\n): Promise<DatabaseEntry> {\n const notion = getNotionClient();\n\n const response = (await notion.pages.update({\n page_id: pageId,\n properties: buildDatabaseProperties(properties),\n })) as PageObjectResponse;\n\n return {\n id: response.id,\n url: response.url,\n createdTime: response.created_time,\n lastEditedTime: response.last_edited_time,\n archived: response.archived,\n properties: extractPropertyValues(response.properties),\n };\n}\n\n/**\n * Archive a database entry\n */\nexport async function archiveDatabaseEntry(pageId: string): Promise<void> {\n const notion = getNotionClient();\n\n await notion.pages.update({\n page_id: pageId,\n archived: true,\n });\n}\n\n/**\n * Create a new database\n */\nexport async function createDatabase(options: {\n parentPageId?: string;\n title: string;\n properties: Record<string, { type: PropertyType; options?: any }>;\n isInline?: boolean;\n}): Promise<DatabaseInfo> {\n const notion = getNotionClient();\n\n const parentId = options.parentPageId || getRootPageId();\n if (!parentId) {\n throw new Error(\"No parent page specified and no root page configured\");\n }\n\n const properties: Record<string, any> = {};\n\n for (const [name, config] of Object.entries(options.properties)) {\n properties[name] = buildPropertySchema(config.type, config.options);\n }\n\n // Ensure there's a title property\n if (!Object.values(properties).some((p) => p.type === \"title\")) {\n properties[\"Name\"] = { title: {} };\n }\n\n const response = (await notion.databases.create({\n parent: { type: \"page_id\", page_id: parentId },\n title: [{ text: { content: options.title } }],\n properties,\n is_inline: options.isInline ?? false,\n } as any)) as any;\n\n const resultProperties: Record<string, { type: PropertyType; name: string }> =\n {};\n for (const [name, prop] of Object.entries(response.properties || {})) {\n const p = prop as any;\n resultProperties[name] = {\n type: p.type as PropertyType,\n name: p.name,\n };\n }\n\n return {\n id: response.id,\n title: response.title.map((t) => t.plain_text).join(\"\"),\n description: \"\",\n url: response.url,\n createdTime: response.created_time,\n lastEditedTime: response.last_edited_time,\n isInline: response.is_inline,\n properties: resultProperties,\n };\n}\n\n/**\n * Build property schema for database creation\n */\nfunction buildPropertySchema(\n type: PropertyType,\n options?: any\n): Record<string, any> {\n switch (type) {\n case \"title\":\n return { title: {} };\n\n case \"rich_text\":\n return { rich_text: {} };\n\n case \"number\":\n return { number: { format: options?.format || \"number\" } };\n\n case \"select\":\n return {\n select: {\n options: options?.options?.map((opt: string) => ({ name: opt })) || [],\n },\n };\n\n case \"multi_select\":\n return {\n multi_select: {\n options: options?.options?.map((opt: string) => ({ name: opt })) || [],\n },\n };\n\n case \"date\":\n return { date: {} };\n\n case \"checkbox\":\n return { checkbox: {} };\n\n case \"url\":\n return { url: {} };\n\n case \"email\":\n return { email: {} };\n\n case \"phone_number\":\n return { phone_number: {} };\n\n case \"status\":\n return {\n status: {\n options:\n options?.options?.map((opt: string) => ({ name: opt })) || [\n { name: \"Not started\" },\n { name: \"In progress\" },\n { name: \"Done\" },\n ],\n },\n };\n\n default:\n return { rich_text: {} };\n }\n}\n\n/**\n * Build database properties for entry creation/update\n */\nfunction buildDatabaseProperties(\n properties: Record<string, any>\n): Record<string, any> {\n const result: Record<string, any> = {};\n\n for (const [key, value] of Object.entries(properties)) {\n if (value === undefined) continue;\n\n // Handle explicit type specification\n if (typeof value === \"object\" && value !== null && value._type) {\n const { _type, ...rest } = value;\n result[key] = buildTypedProperty(_type, rest);\n continue;\n }\n\n // Auto-detect type\n if (typeof value === \"string\") {\n // Check if it looks like a title property\n if (key.toLowerCase() === \"name\" || key.toLowerCase() === \"title\") {\n result[key] = { title: [{ text: { content: value } }] };\n } else if (value.startsWith(\"http://\") || value.startsWith(\"https://\")) {\n result[key] = { url: value };\n } else if (value.includes(\"@\") && value.includes(\".\")) {\n result[key] = { email: value };\n } else {\n result[key] = { rich_text: [{ text: { content: value } }] };\n }\n } else if (typeof value === \"number\") {\n result[key] = { number: value };\n } else if (typeof value === \"boolean\") {\n result[key] = { checkbox: value };\n } else if (value instanceof Date) {\n result[key] = { date: { start: value.toISOString() } };\n } else if (Array.isArray(value)) {\n // Multi-select\n result[key] = { multi_select: value.map((v) => ({ name: String(v) })) };\n }\n }\n\n return result;\n}\n\n/**\n * Build typed property value\n */\nfunction buildTypedProperty(\n type: PropertyType,\n value: any\n): Record<string, any> {\n switch (type) {\n case \"title\":\n return { title: [{ text: { content: value.content || value } }] };\n\n case \"rich_text\":\n return { rich_text: [{ text: { content: value.content || value } }] };\n\n case \"number\":\n return { number: value.value ?? value };\n\n case \"select\":\n return { select: { name: value.name || value } };\n\n case \"multi_select\":\n const options = Array.isArray(value) ? value : value.options || [value];\n return { multi_select: options.map((v: any) => ({ name: v.name || v })) };\n\n case \"date\":\n return {\n date: {\n start:\n value.start instanceof Date\n ? value.start.toISOString()\n : value.start || value,\n end: value.end\n ? value.end instanceof Date\n ? value.end.toISOString()\n : value.end\n : undefined,\n },\n };\n\n case \"checkbox\":\n return { checkbox: value.checked ?? value };\n\n case \"url\":\n return { url: value.url || value };\n\n case \"email\":\n return { email: value.email || value };\n\n case \"phone_number\":\n return { phone_number: value.phone || value };\n\n case \"status\":\n return { status: { name: value.name || value } };\n\n default:\n return { rich_text: [{ text: { content: String(value) } }] };\n }\n}\n\n/**\n * Extract simplified property values from Notion response\n */\nfunction extractPropertyValues(\n properties: Record<string, any>\n): Record<string, any> {\n const result: Record<string, any> = {};\n\n for (const [name, prop] of Object.entries(properties)) {\n switch (prop.type) {\n case \"title\":\n result[name] = prop.title?.map((t: any) => t.plain_text).join(\"\") || \"\";\n break;\n\n case \"rich_text\":\n result[name] =\n prop.rich_text?.map((t: any) => t.plain_text).join(\"\") || \"\";\n break;\n\n case \"number\":\n result[name] = prop.number;\n break;\n\n case \"select\":\n result[name] = prop.select?.name || null;\n break;\n\n case \"multi_select\":\n result[name] = prop.multi_select?.map((s: any) => s.name) || [];\n break;\n\n case \"date\":\n result[name] = prop.date\n ? { start: prop.date.start, end: prop.date.end }\n : null;\n break;\n\n case \"checkbox\":\n result[name] = prop.checkbox;\n break;\n\n case \"url\":\n result[name] = prop.url;\n break;\n\n case \"email\":\n result[name] = prop.email;\n break;\n\n case \"phone_number\":\n result[name] = prop.phone_number;\n break;\n\n case \"formula\":\n result[name] = prop.formula?.[prop.formula.type];\n break;\n\n case \"relation\":\n result[name] = prop.relation?.map((r: any) => r.id) || [];\n break;\n\n case \"rollup\":\n result[name] = prop.rollup?.[prop.rollup.type];\n break;\n\n case \"created_time\":\n result[name] = prop.created_time;\n break;\n\n case \"created_by\":\n result[name] = prop.created_by?.id;\n break;\n\n case \"last_edited_time\":\n result[name] = prop.last_edited_time;\n break;\n\n case \"last_edited_by\":\n result[name] = prop.last_edited_by?.id;\n break;\n\n case \"files\":\n result[name] =\n prop.files?.map((f: any) => f.file?.url || f.external?.url) || [];\n break;\n\n case \"status\":\n result[name] = prop.status?.name || null;\n break;\n\n default:\n result[name] = prop;\n }\n }\n\n return result;\n}\n"],"mappings":"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","names":[]}