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.
- package/README.md +354 -283
- package/dist/archiver-AVNBYCKQ.js +15340 -0
- package/dist/archiver-AVNBYCKQ.js.map +1 -0
- package/dist/audit-logger-OBPR7CRO.js +22 -0
- package/dist/auth-UOX5K2BE.js +18 -0
- package/dist/autonomy-ZXDBDQUJ.js +86 -0
- package/dist/autonomy-ZXDBDQUJ.js.map +1 -0
- package/dist/aws-s3-Q4LLZZPD.js +146 -0
- package/dist/aws-s3-Q4LLZZPD.js.map +1 -0
- package/dist/backup-restore-PZ7CYYB7.js +16 -0
- package/dist/blocks-R3PODY47.js +23 -0
- package/dist/bot-QRARP4UN.js +36 -0
- package/dist/brain-7XLLM3KC.js +56 -0
- package/dist/camera-monitor-M5CYKUU4.js +335 -0
- package/dist/camera-monitor-M5CYKUU4.js.map +1 -0
- package/dist/{charts-MMXM6BWW.js → charts-V7ARZNKF.js} +2 -2
- package/dist/chunk-22VGGA7S.js +330 -0
- package/dist/chunk-22VGGA7S.js.map +1 -0
- package/dist/chunk-35WYTA3C.js +382 -0
- package/dist/chunk-35WYTA3C.js.map +1 -0
- package/dist/chunk-3E2PSU2C.js +146 -0
- package/dist/chunk-3E2PSU2C.js.map +1 -0
- package/dist/{chunk-L3F43VPB.js → chunk-4GLYY4NN.js} +2 -2
- package/dist/{chunk-L3F43VPB.js.map → chunk-4GLYY4NN.js.map} +1 -1
- package/dist/{chunk-L3PDU3XN.js → chunk-4UOE5TUZ.js} +4 -4
- package/dist/{chunk-6SNHU3CY.js → chunk-66OJ3WB4.js} +2 -2
- package/dist/chunk-6KONMXQ6.js +297 -0
- package/dist/chunk-6KONMXQ6.js.map +1 -0
- package/dist/chunk-6PMVAAA7.js +196 -0
- package/dist/chunk-6PMVAAA7.js.map +1 -0
- package/dist/chunk-766ASQWE.js +32620 -0
- package/dist/chunk-766ASQWE.js.map +1 -0
- package/dist/chunk-7WQO5J2M.js +29 -0
- package/dist/chunk-7WQO5J2M.js.map +1 -0
- package/dist/chunk-APHSRMBS.js +148 -0
- package/dist/chunk-APHSRMBS.js.map +1 -0
- package/dist/{chunk-4LVWXUNC.js → chunk-AYUKPTSM.js} +57 -39
- package/dist/chunk-AYUKPTSM.js.map +1 -0
- package/dist/chunk-BIPYADGB.js +84 -0
- package/dist/chunk-BIPYADGB.js.map +1 -0
- package/dist/chunk-BRBWNV65.js +457 -0
- package/dist/chunk-BRBWNV65.js.map +1 -0
- package/dist/chunk-BXZ6EA52.js +382 -0
- package/dist/chunk-BXZ6EA52.js.map +1 -0
- package/dist/chunk-EVE7MIIY.js +290 -0
- package/dist/chunk-EVE7MIIY.js.map +1 -0
- package/dist/chunk-F3TTNID2.js +138 -0
- package/dist/chunk-F3TTNID2.js.map +1 -0
- package/dist/chunk-H5RQOFO2.js +190 -0
- package/dist/chunk-H5RQOFO2.js.map +1 -0
- package/dist/chunk-HN3F4WSW.js +145 -0
- package/dist/chunk-HN3F4WSW.js.map +1 -0
- package/dist/{chunk-6DRDKB45.js → chunk-I6BDYQIG.js} +20 -9
- package/dist/chunk-I6BDYQIG.js.map +1 -0
- package/dist/chunk-IZJMVV7O.js +347 -0
- package/dist/chunk-IZJMVV7O.js.map +1 -0
- package/dist/chunk-KM22GV7G.js +211 -0
- package/dist/chunk-KM22GV7G.js.map +1 -0
- package/dist/chunk-MGFBLVR7.js +103 -0
- package/dist/chunk-MGFBLVR7.js.map +1 -0
- package/dist/chunk-MQJ2ECQT.js +228 -0
- package/dist/chunk-MQJ2ECQT.js.map +1 -0
- package/dist/{chunk-F6QUZQGI.js → chunk-MXAPLSJ5.js} +2 -2
- package/dist/{chunk-GK3E2I7A.js → chunk-NHMBTUMW.js} +2 -2
- package/dist/chunk-NPRTSZIF.js +131 -0
- package/dist/chunk-NPRTSZIF.js.map +1 -0
- package/dist/chunk-O7IH7JTI.js +1898 -0
- package/dist/chunk-O7IH7JTI.js.map +1 -0
- package/dist/chunk-OCVQGBJK.js +293 -0
- package/dist/chunk-OCVQGBJK.js.map +1 -0
- package/dist/chunk-P6QINGFL.js +332 -0
- package/dist/chunk-P6QINGFL.js.map +1 -0
- package/dist/chunk-PHDZKPNE.js +91 -0
- package/dist/chunk-PHDZKPNE.js.map +1 -0
- package/dist/chunk-PLDDJCW6.js +49 -0
- package/dist/chunk-PTGTGXV2.js +164 -0
- package/dist/chunk-PTGTGXV2.js.map +1 -0
- package/dist/chunk-REMIY4U2.js +171 -0
- package/dist/chunk-REMIY4U2.js.map +1 -0
- package/dist/chunk-RZ4YESBG.js +141 -0
- package/dist/chunk-RZ4YESBG.js.map +1 -0
- package/dist/chunk-SAX5MHK4.js +111 -0
- package/dist/chunk-SAX5MHK4.js.map +1 -0
- package/dist/{chunk-GVJVEWHI.js → chunk-SJSUSJ47.js} +2 -2
- package/dist/chunk-SPPMCAKG.js +777 -0
- package/dist/chunk-SPPMCAKG.js.map +1 -0
- package/dist/chunk-SVAPX2XN.js +2441 -0
- package/dist/chunk-SVAPX2XN.js.map +1 -0
- package/dist/chunk-TVEWKIK3.js +452 -0
- package/dist/chunk-TVEWKIK3.js.map +1 -0
- package/dist/{chunk-HH2HBTQM.js → chunk-TYAGMJNV.js} +5 -5
- package/dist/{chunk-JXUP2X7V.js → chunk-VEHFVBLI.js} +2 -2
- package/dist/chunk-VNX5GMTN.js +128 -0
- package/dist/chunk-VNX5GMTN.js.map +1 -0
- package/dist/chunk-VRD5CYRL.js +1568 -0
- package/dist/chunk-VRD5CYRL.js.map +1 -0
- package/dist/chunk-WLUHNG6X.js +122 -0
- package/dist/chunk-WLUHNG6X.js.map +1 -0
- package/dist/chunk-WRAKK6K6.js +265 -0
- package/dist/chunk-WRAKK6K6.js.map +1 -0
- package/dist/chunk-XKYRH4FM.js +681 -0
- package/dist/chunk-XKYRH4FM.js.map +1 -0
- package/dist/{chunk-GUBEEYDW.js → chunk-XMCVRVTF.js} +2 -2
- package/dist/{chunk-GUBEEYDW.js.map → chunk-XMCVRVTF.js.map} +1 -1
- package/dist/chunk-ZLZKF2PM.js +310 -0
- package/dist/chunk-ZLZKF2PM.js.map +1 -0
- package/dist/cli.js +5 -1
- package/dist/cli.js.map +1 -1
- package/dist/client-ZQSFPMOB.js +21 -0
- package/dist/clipboard-manager-TEO2GEDN.js +24 -0
- package/dist/commands/setup.js +3 -3
- package/dist/commands/setup.js.map +1 -1
- package/dist/commands/start.js +3 -3
- package/dist/commands/status.js +2 -2
- package/dist/commands/stop.js +2 -2
- package/dist/commands/utils.js +2 -2
- package/dist/cron-explain-HHQKPD3M.js +16 -0
- package/dist/crypto-4AP47IKC.js +14 -0
- package/dist/crypto-4AP47IKC.js.map +1 -0
- package/dist/databases-37X4CI2Y.js +21 -0
- package/dist/databases-37X4CI2Y.js.map +1 -0
- package/dist/discord-B3HUPGQ6.js +70 -0
- package/dist/discord-B3HUPGQ6.js.map +1 -0
- package/dist/dist-UISMLMFN.js +21847 -0
- package/dist/dist-UISMLMFN.js.map +1 -0
- package/dist/email-K7LO2IPB.js +268 -0
- package/dist/email-K7LO2IPB.js.map +1 -0
- package/dist/enhanced-retrieval-DNLLEM4Z.js +753 -0
- package/dist/enhanced-retrieval-DNLLEM4Z.js.map +1 -0
- package/dist/enrichment-pipeline-MNHNW65K.js +13 -0
- package/dist/enrichment-pipeline-MNHNW65K.js.map +1 -0
- package/dist/entity-resolution-Y3IUWEAT.js +24 -0
- package/dist/entity-resolution-Y3IUWEAT.js.map +1 -0
- package/dist/env-IWXUVTCB.js +12 -0
- package/dist/env-IWXUVTCB.js.map +1 -0
- package/dist/google-workspace-DKWUVNGC.js +169 -0
- package/dist/google-workspace-DKWUVNGC.js.map +1 -0
- package/dist/hash-tool-ULQYD7B5.js +22 -0
- package/dist/hash-tool-ULQYD7B5.js.map +1 -0
- package/dist/heartbeat-monitor-GCISLXI3.js +22 -0
- package/dist/heartbeat-monitor-GCISLXI3.js.map +1 -0
- package/dist/image-generation-OSU7FP6F.js +486 -0
- package/dist/image-generation-OSU7FP6F.js.map +1 -0
- package/dist/imessage-NGA2XF2V.js +35 -0
- package/dist/imessage-NGA2XF2V.js.map +1 -0
- package/dist/inbox-summarizer-NRI4S7IF.js +47 -0
- package/dist/inbox-summarizer-NRI4S7IF.js.map +1 -0
- package/dist/incident-response-C5J7Q6DT.js +244 -0
- package/dist/incident-response-C5J7Q6DT.js.map +1 -0
- package/dist/inventory-manager-352OHXWD.js +24 -0
- package/dist/inventory-manager-352OHXWD.js.map +1 -0
- package/dist/jira-GSGDBMIG.js +199 -0
- package/dist/jira-GSGDBMIG.js.map +1 -0
- package/dist/json-tool-QE2SYHEG.js +26 -0
- package/dist/json-tool-QE2SYHEG.js.map +1 -0
- package/dist/key-rotation-DPHU4ZTB.js +18 -0
- package/dist/key-rotation-DPHU4ZTB.js.map +1 -0
- package/dist/lib.d.ts +603 -11
- package/dist/lib.js +161 -35
- package/dist/lib.js.map +1 -1
- package/dist/mailchimp-KKNF6QJ7.js +152 -0
- package/dist/mailchimp-KKNF6QJ7.js.map +1 -0
- package/dist/matrix-QVHG76I7.js +279 -0
- package/dist/matrix-QVHG76I7.js.map +1 -0
- package/dist/{mcp-LS7Q3Z5W.js → mcp-3JI6W7ZE.js} +3 -3
- package/dist/mcp-3JI6W7ZE.js.map +1 -0
- package/dist/microsoft365-UCBKJHNX.js +164 -0
- package/dist/microsoft365-UCBKJHNX.js.map +1 -0
- package/dist/ocr-AC7NPX33.js +22 -0
- package/dist/ocr-AC7NPX33.js.map +1 -0
- package/dist/ollama-BOAMSPLJ.js +8 -0
- package/dist/ollama-BOAMSPLJ.js.map +1 -0
- package/dist/pages-MI523RB7.js +26 -0
- package/dist/pages-MI523RB7.js.map +1 -0
- package/dist/pair-JDFTERIK.js +24 -0
- package/dist/pair-JDFTERIK.js.map +1 -0
- package/dist/pairing-IFQYCPNS.js +10 -0
- package/dist/pairing-IFQYCPNS.js.map +1 -0
- package/dist/pdf-ALQVOEJR.js +17 -0
- package/dist/pdf-ALQVOEJR.js.map +1 -0
- package/dist/presentations-DSV5IHG5.js +1002 -0
- package/dist/presentations-DSV5IHG5.js.map +1 -0
- package/dist/prometheus-JNT2BD4L.js +10 -0
- package/dist/prometheus-JNT2BD4L.js.map +1 -0
- package/dist/providers-J4LYPHDR.js +19 -0
- package/dist/providers-J4LYPHDR.js.map +1 -0
- package/dist/qr-code-WIX4PB4U.js +16 -0
- package/dist/qr-code-WIX4PB4U.js.map +1 -0
- package/dist/quickbooks-XB4NII2S.js +190 -0
- package/dist/quickbooks-XB4NII2S.js.map +1 -0
- package/dist/regex-tool-W4ABRKGK.js +24 -0
- package/dist/regex-tool-W4ABRKGK.js.map +1 -0
- package/dist/scheduler-VK4WFERV.js +63 -0
- package/dist/scheduler-VK4WFERV.js.map +1 -0
- package/dist/search-BCLBO5E3.js +25 -0
- package/dist/search-BCLBO5E3.js.map +1 -0
- package/dist/sendgrid-RNXCAFKM.js +152 -0
- package/dist/sendgrid-RNXCAFKM.js.map +1 -0
- package/dist/shopify-NCXYJB4R.js +171 -0
- package/dist/shopify-NCXYJB4R.js.map +1 -0
- package/dist/signal-6CGDFYL2.js +35 -0
- package/dist/signal-6CGDFYL2.js.map +1 -0
- package/dist/slack-IZQWIKOH.js +75 -0
- package/dist/slack-IZQWIKOH.js.map +1 -0
- package/dist/sms-M3JIOTCW.js +23 -0
- package/dist/sms-M3JIOTCW.js.map +1 -0
- package/dist/{src-K7GASHRH.js → src-VYUE6LRA.js} +138 -32
- package/dist/src-VYUE6LRA.js.map +1 -0
- package/dist/stocks-XXWBPOCU.js +14 -0
- package/dist/stocks-XXWBPOCU.js.map +1 -0
- package/dist/text-transform-6SGUA5Z4.js +22 -0
- package/dist/text-transform-6SGUA5Z4.js.map +1 -0
- package/dist/tools-2RLEI2N6.js +38 -0
- package/dist/tools-2RLEI2N6.js.map +1 -0
- package/dist/tunnel-IWMXUML4.js +301 -0
- package/dist/tunnel-IWMXUML4.js.map +1 -0
- package/dist/twilio-53GEW5JT.js +139 -0
- package/dist/twilio-53GEW5JT.js.map +1 -0
- package/dist/unit-converter-ZYXMEZOE.js +14 -0
- package/dist/unit-converter-ZYXMEZOE.js.map +1 -0
- package/dist/whatsapp-LFX6YKCM.js +35 -0
- package/dist/whatsapp-LFX6YKCM.js.map +1 -0
- package/dist/word-document-7B6SJMAY.js +902 -0
- package/dist/word-document-7B6SJMAY.js.map +1 -0
- package/dist/xero-QYO66D45.js +162 -0
- package/dist/xero-QYO66D45.js.map +1 -0
- package/dist/zapier-webhook-TBZ5YF2A.js +106 -0
- package/dist/zapier-webhook-TBZ5YF2A.js.map +1 -0
- package/drizzle/0002_mushy_master_mold.sql +140 -0
- package/drizzle/meta/0002_snapshot.json +3637 -0
- package/drizzle/meta/_journal.json +7 -0
- package/package.json +100 -98
- package/dist/bot-KJ26BG56.js +0 -15
- package/dist/chunk-4LVWXUNC.js.map +0 -1
- package/dist/chunk-4TG2IG5K.js +0 -5249
- package/dist/chunk-4TG2IG5K.js.map +0 -1
- package/dist/chunk-6DRDKB45.js.map +0 -1
- package/dist/chunk-CI6Q63MM.js +0 -1613
- package/dist/chunk-CI6Q63MM.js.map +0 -1
- package/dist/chunk-KHNYJY2Z.js +0 -178
- package/dist/chunk-KHNYJY2Z.js.map +0 -1
- package/dist/chunk-NSBPE2FW.js +0 -17
- package/dist/discord-ZOJFTVTB.js +0 -49
- package/dist/imessage-JFRB6EJ7.js +0 -14
- package/dist/scheduler-EZ7CZMCS.js +0 -42
- package/dist/signal-T3MCSULM.js +0 -14
- package/dist/slack-N2M4FHAJ.js +0 -54
- package/dist/src-K7GASHRH.js.map +0 -1
- package/dist/tools-24GZHYRF.js +0 -16
- package/dist/whatsapp-VCRUPAO5.js +0 -14
- /package/dist/{bot-KJ26BG56.js.map → audit-logger-OBPR7CRO.js.map} +0 -0
- /package/dist/{chunk-NSBPE2FW.js.map → auth-UOX5K2BE.js.map} +0 -0
- /package/dist/{discord-ZOJFTVTB.js.map → backup-restore-PZ7CYYB7.js.map} +0 -0
- /package/dist/{imessage-JFRB6EJ7.js.map → blocks-R3PODY47.js.map} +0 -0
- /package/dist/{mcp-LS7Q3Z5W.js.map → bot-QRARP4UN.js.map} +0 -0
- /package/dist/{scheduler-EZ7CZMCS.js.map → brain-7XLLM3KC.js.map} +0 -0
- /package/dist/{charts-MMXM6BWW.js.map → charts-V7ARZNKF.js.map} +0 -0
- /package/dist/{chunk-L3PDU3XN.js.map → chunk-4UOE5TUZ.js.map} +0 -0
- /package/dist/{chunk-6SNHU3CY.js.map → chunk-66OJ3WB4.js.map} +0 -0
- /package/dist/{chunk-F6QUZQGI.js.map → chunk-MXAPLSJ5.js.map} +0 -0
- /package/dist/{chunk-GK3E2I7A.js.map → chunk-NHMBTUMW.js.map} +0 -0
- /package/dist/{signal-T3MCSULM.js.map → chunk-PLDDJCW6.js.map} +0 -0
- /package/dist/{chunk-GVJVEWHI.js.map → chunk-SJSUSJ47.js.map} +0 -0
- /package/dist/{chunk-HH2HBTQM.js.map → chunk-TYAGMJNV.js.map} +0 -0
- /package/dist/{chunk-JXUP2X7V.js.map → chunk-VEHFVBLI.js.map} +0 -0
- /package/dist/{slack-N2M4FHAJ.js.map → client-ZQSFPMOB.js.map} +0 -0
- /package/dist/{tools-24GZHYRF.js.map → clipboard-manager-TEO2GEDN.js.map} +0 -0
- /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":[]}
|