@elevasis/core 0.22.0 → 0.24.0

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 (244) hide show
  1. package/dist/index.d.ts +3214 -2501
  2. package/dist/index.js +3112 -1222
  3. package/dist/knowledge/index.d.ts +1108 -1264
  4. package/dist/knowledge/index.js +112 -9
  5. package/dist/organization-model/index.d.ts +3214 -2501
  6. package/dist/organization-model/index.js +3112 -1222
  7. package/dist/test-utils/index.d.ts +985 -1103
  8. package/dist/test-utils/index.js +2464 -1165
  9. package/package.json +5 -5
  10. package/src/README.md +14 -14
  11. package/src/__tests__/publish.test.ts +24 -24
  12. package/src/__tests__/template-core-compatibility.test.ts +9 -80
  13. package/src/_gen/__tests__/__snapshots__/contracts.md.snap +2389 -2121
  14. package/src/_gen/__tests__/scaffold-contracts.test.ts +30 -30
  15. package/src/auth/multi-tenancy/credentials/__tests__/encryption.test.ts +217 -217
  16. package/src/auth/multi-tenancy/credentials/server/encryption.ts +69 -69
  17. package/src/auth/multi-tenancy/credentials/server/kek-loader.ts +37 -37
  18. package/src/auth/multi-tenancy/index.ts +26 -26
  19. package/src/auth/multi-tenancy/invitations/api-schemas.ts +104 -104
  20. package/src/auth/multi-tenancy/memberships/api-schemas.ts +143 -143
  21. package/src/auth/multi-tenancy/memberships/index.ts +26 -26
  22. package/src/auth/multi-tenancy/memberships/membership.ts +130 -130
  23. package/src/auth/multi-tenancy/organizations/__tests__/api-schemas.test.ts +194 -194
  24. package/src/auth/multi-tenancy/organizations/api-schemas.ts +136 -136
  25. package/src/auth/multi-tenancy/permissions.test.ts +42 -42
  26. package/src/auth/multi-tenancy/permissions.ts +123 -123
  27. package/src/auth/multi-tenancy/role-management/api-schemas.ts +78 -78
  28. package/src/auth/multi-tenancy/role-management/index.ts +16 -16
  29. package/src/auth/multi-tenancy/theme-presets.ts +45 -45
  30. package/src/auth/multi-tenancy/types.ts +57 -57
  31. package/src/auth/multi-tenancy/users/api-schemas.ts +165 -165
  32. package/src/business/README.md +2 -2
  33. package/src/business/acquisition/activity-events.test.ts +250 -250
  34. package/src/business/acquisition/activity-events.ts +93 -93
  35. package/src/business/acquisition/api-schemas.test.ts +1883 -1843
  36. package/src/business/acquisition/api-schemas.ts +1493 -1500
  37. package/src/business/acquisition/build-templates.test.ts +240 -240
  38. package/src/business/acquisition/build-templates.ts +83 -41
  39. package/src/business/acquisition/crm-next-action.test.ts +262 -262
  40. package/src/business/acquisition/crm-next-action.ts +220 -220
  41. package/src/business/acquisition/crm-priority.test.ts +216 -216
  42. package/src/business/acquisition/crm-priority.ts +349 -349
  43. package/src/business/acquisition/crm-state-actions.test.ts +153 -151
  44. package/src/business/acquisition/deal-ownership.test.ts +351 -351
  45. package/src/business/acquisition/deal-ownership.ts +120 -120
  46. package/src/business/acquisition/derive-actions.test.ts +129 -104
  47. package/src/business/acquisition/derive-actions.ts +74 -84
  48. package/src/business/acquisition/index.ts +171 -170
  49. package/src/business/acquisition/ontology-validation.ts +309 -0
  50. package/src/business/acquisition/stateful.ts +30 -30
  51. package/src/business/acquisition/types.ts +396 -392
  52. package/src/business/clients/api-schemas.test.ts +115 -115
  53. package/src/business/clients/api-schemas.ts +158 -158
  54. package/src/business/clients/index.ts +1 -1
  55. package/src/business/crm/api-schemas.ts +40 -40
  56. package/src/business/crm/index.ts +1 -1
  57. package/src/business/deals/api-schemas.ts +87 -87
  58. package/src/business/deals/index.ts +1 -1
  59. package/src/business/index.ts +5 -5
  60. package/src/business/projects/types.ts +144 -144
  61. package/src/commands/queue/types/task.ts +15 -15
  62. package/src/execution/core/runner-types.ts +61 -61
  63. package/src/execution/core/sse-executions.ts +7 -7
  64. package/src/execution/engine/__tests__/fixtures/test-agents.ts +10 -10
  65. package/src/execution/engine/agent/core/__tests__/agent.test.ts +16 -16
  66. package/src/execution/engine/agent/core/__tests__/error-passthrough.test.ts +4 -4
  67. package/src/execution/engine/agent/core/types.ts +25 -25
  68. package/src/execution/engine/agent/index.ts +6 -6
  69. package/src/execution/engine/agent/reasoning/__tests__/request-builder.test.ts +24 -24
  70. package/src/execution/engine/index.ts +443 -443
  71. package/src/execution/engine/tools/integration/server/adapters/apify/__tests__/apify-run-actor.integration.test.ts +298 -298
  72. package/src/execution/engine/tools/integration/server/adapters/apify/apify-adapter.test.ts +55 -55
  73. package/src/execution/engine/tools/integration/server/adapters/apify/apify-adapter.ts +107 -107
  74. package/src/execution/engine/tools/integration/server/adapters/apollo/apollo-adapter.test.ts +48 -48
  75. package/src/execution/engine/tools/integration/server/adapters/apollo/apollo-adapter.ts +99 -99
  76. package/src/execution/engine/tools/integration/server/adapters/apollo/index.ts +1 -1
  77. package/src/execution/engine/tools/integration/server/adapters/attio/__tests__/attio-crud.integration.test.ts +363 -363
  78. package/src/execution/engine/tools/integration/server/adapters/attio/fetch/get-record/index.test.ts +162 -162
  79. package/src/execution/engine/tools/integration/server/adapters/attio/fetch/list-records/index.test.ts +316 -316
  80. package/src/execution/engine/tools/integration/server/adapters/clickup/clickup-adapter.test.ts +18 -18
  81. package/src/execution/engine/tools/integration/server/adapters/clickup/clickup-adapter.ts +194 -194
  82. package/src/execution/engine/tools/integration/server/adapters/clickup/index.ts +7 -7
  83. package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-adapter.ts +204 -204
  84. package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-tools.ts +105 -105
  85. package/src/execution/engine/tools/integration/server/adapters/google-calendar/google-calendar-adapter.ts +428 -428
  86. package/src/execution/engine/tools/integration/server/adapters/google-calendar/index.ts +2 -2
  87. package/src/execution/engine/tools/integration/server/adapters/google-sheets/__tests__/google-sheets.integration.test.ts +261 -261
  88. package/src/execution/engine/tools/integration/server/adapters/instantly/instantly-tools.ts +1474 -1474
  89. package/src/execution/engine/tools/integration/server/adapters/millionverifier/millionverifier-tools.ts +103 -103
  90. package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.test.ts +88 -88
  91. package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.ts +141 -141
  92. package/src/execution/engine/tools/integration/server/adapters/resend/fetch/utils/types.ts +76 -76
  93. package/src/execution/engine/tools/integration/server/adapters/signature-api/signature-api-tools.ts +182 -182
  94. package/src/execution/engine/tools/integration/server/adapters/stripe/stripe-tools.ts +310 -310
  95. package/src/execution/engine/tools/integration/service.test.ts +239 -239
  96. package/src/execution/engine/tools/integration/service.ts +172 -172
  97. package/src/execution/engine/tools/integration/tool.ts +255 -255
  98. package/src/execution/engine/tools/lead-service-types.ts +1005 -1005
  99. package/src/execution/engine/tools/messages.ts +43 -43
  100. package/src/execution/engine/tools/platform/acquisition/company-tools.ts +7 -7
  101. package/src/execution/engine/tools/platform/acquisition/contact-tools.ts +6 -6
  102. package/src/execution/engine/tools/platform/acquisition/list-tools.ts +6 -6
  103. package/src/execution/engine/tools/platform/acquisition/types.ts +280 -280
  104. package/src/execution/engine/tools/platform/email/types.ts +97 -97
  105. package/src/execution/engine/tools/registry.ts +704 -704
  106. package/src/execution/engine/tools/tool-maps.ts +831 -831
  107. package/src/execution/engine/tools/types.ts +234 -234
  108. package/src/execution/engine/workflow/types.ts +202 -202
  109. package/src/execution/external/__tests__/api-schemas.test.ts +127 -127
  110. package/src/execution/external/api-schemas.ts +40 -40
  111. package/src/execution/external/index.ts +1 -1
  112. package/src/index.ts +18 -18
  113. package/src/integrations/credentials/__tests__/api-schemas.test.ts +420 -420
  114. package/src/integrations/credentials/api-schemas.ts +146 -146
  115. package/src/integrations/credentials/schemas.ts +200 -200
  116. package/src/integrations/oauth/__tests__/provider-registry.test.ts +7 -7
  117. package/src/integrations/oauth/provider-registry.ts +74 -74
  118. package/src/integrations/oauth/server/credentials.ts +43 -43
  119. package/src/integrations/webhook-endpoints/__tests__/api-schemas.test.ts +327 -327
  120. package/src/integrations/webhook-endpoints/api-schemas.ts +103 -103
  121. package/src/integrations/webhook-endpoints/types.ts +58 -58
  122. package/src/knowledge/README.md +33 -32
  123. package/src/knowledge/__tests__/queries.test.ts +633 -541
  124. package/src/knowledge/format.ts +100 -99
  125. package/src/knowledge/index.ts +5 -5
  126. package/src/knowledge/published.ts +5 -5
  127. package/src/knowledge/queries.ts +274 -222
  128. package/src/operations/activities/api-schemas.ts +80 -80
  129. package/src/operations/activities/types.ts +64 -64
  130. package/src/organization-model/README.md +149 -109
  131. package/src/organization-model/__tests__/content-kinds-registry.test.ts +210 -0
  132. package/src/organization-model/__tests__/defaults.test.ts +168 -194
  133. package/src/organization-model/__tests__/domains/actions.test.ts +78 -0
  134. package/src/organization-model/__tests__/domains/customers.test.ts +48 -44
  135. package/src/organization-model/__tests__/domains/entities.test.ts +56 -0
  136. package/src/organization-model/__tests__/domains/goals.test.ts +110 -96
  137. package/src/organization-model/__tests__/domains/identity.test.ts +4 -3
  138. package/src/organization-model/__tests__/domains/navigation.test.ts +222 -166
  139. package/src/organization-model/__tests__/domains/offerings.test.ts +83 -88
  140. package/src/organization-model/__tests__/domains/policies.test.ts +323 -0
  141. package/src/organization-model/__tests__/domains/resource-mappings.test.ts +30 -30
  142. package/src/organization-model/__tests__/domains/resources.test.ts +396 -175
  143. package/src/organization-model/__tests__/domains/roles.test.ts +463 -402
  144. package/src/organization-model/__tests__/domains/statuses.test.ts +13 -10
  145. package/src/organization-model/__tests__/domains/systems.test.ts +209 -193
  146. package/src/organization-model/__tests__/flatten-additive-merge.test.ts +362 -0
  147. package/src/organization-model/__tests__/foundation.test.ts +47 -75
  148. package/src/organization-model/__tests__/get-resources-for-system.test.ts +144 -0
  149. package/src/organization-model/__tests__/graph.test.ts +1336 -149
  150. package/src/organization-model/__tests__/icons.test.ts +10 -1
  151. package/src/organization-model/__tests__/knowledge.test.ts +418 -61
  152. package/src/organization-model/__tests__/lookup-helpers.test.ts +438 -0
  153. package/src/organization-model/__tests__/migration-helpers.test.ts +591 -0
  154. package/src/organization-model/__tests__/prospecting-ssot.test.ts +103 -94
  155. package/src/organization-model/__tests__/recursive-system-schema.test.ts +549 -0
  156. package/src/organization-model/__tests__/resolve.test.ts +303 -42
  157. package/src/organization-model/__tests__/schema.test.ts +863 -153
  158. package/src/organization-model/__tests__/surface-projection.test.ts +284 -174
  159. package/src/organization-model/catalogs/lead-gen.ts +144 -0
  160. package/src/organization-model/content-kinds/config.ts +36 -0
  161. package/src/organization-model/content-kinds/index.ts +78 -0
  162. package/src/organization-model/content-kinds/pipeline.ts +68 -0
  163. package/src/organization-model/content-kinds/registry.ts +44 -0
  164. package/src/organization-model/content-kinds/status.ts +71 -0
  165. package/src/organization-model/content-kinds/template.ts +83 -0
  166. package/src/organization-model/content-kinds/types.ts +117 -0
  167. package/src/organization-model/contracts.ts +27 -17
  168. package/src/organization-model/defaults.ts +489 -107
  169. package/src/organization-model/domains/actions.ts +333 -0
  170. package/src/organization-model/domains/customers.ts +10 -7
  171. package/src/organization-model/domains/entities.ts +144 -0
  172. package/src/organization-model/domains/goals.ts +9 -6
  173. package/src/organization-model/domains/knowledge.ts +128 -54
  174. package/src/organization-model/domains/navigation.ts +139 -416
  175. package/src/organization-model/domains/offerings.ts +15 -10
  176. package/src/organization-model/domains/policies.ts +102 -0
  177. package/src/organization-model/domains/projects.ts +6 -40
  178. package/src/organization-model/domains/prospecting.ts +395 -514
  179. package/src/organization-model/domains/resources.ts +173 -81
  180. package/src/organization-model/domains/roles.ts +96 -93
  181. package/src/organization-model/domains/sales.test.ts +218 -218
  182. package/src/organization-model/domains/sales.ts +380 -589
  183. package/src/organization-model/domains/shared.ts +8 -8
  184. package/src/organization-model/domains/statuses.ts +298 -89
  185. package/src/organization-model/domains/systems.ts +240 -38
  186. package/src/organization-model/foundation.ts +35 -48
  187. package/src/organization-model/graph/build.ts +1035 -279
  188. package/src/organization-model/graph/index.ts +4 -4
  189. package/src/organization-model/graph/link.ts +10 -10
  190. package/src/organization-model/graph/schema.ts +77 -56
  191. package/src/organization-model/graph/types.ts +75 -56
  192. package/src/organization-model/helpers.ts +312 -59
  193. package/src/organization-model/icons.ts +78 -66
  194. package/src/organization-model/index.ts +129 -16
  195. package/src/organization-model/migration-helpers.ts +252 -0
  196. package/src/organization-model/ontology.ts +661 -0
  197. package/src/organization-model/organization-graph.mdx +110 -89
  198. package/src/organization-model/organization-model.mdx +226 -171
  199. package/src/organization-model/published.ts +295 -139
  200. package/src/organization-model/resolve.ts +139 -21
  201. package/src/organization-model/schema.ts +841 -301
  202. package/src/organization-model/surface-projection.ts +212 -218
  203. package/src/organization-model/types.ts +181 -90
  204. package/src/platform/api/types.ts +38 -38
  205. package/src/platform/constants/versions.ts +3 -3
  206. package/src/platform/index.ts +23 -23
  207. package/src/platform/registry/__tests__/command-view.test.ts +5 -7
  208. package/src/platform/registry/__tests__/resource-link.test.ts +35 -30
  209. package/src/platform/registry/__tests__/resource-registry.integration.test.ts +17 -32
  210. package/src/platform/registry/__tests__/resource-registry.nested-systems.test.ts +245 -0
  211. package/src/platform/registry/__tests__/resource-registry.test.ts +2053 -2051
  212. package/src/platform/registry/__tests__/validation.test.ts +1347 -1343
  213. package/src/platform/registry/command-view.ts +10 -10
  214. package/src/platform/registry/index.ts +103 -103
  215. package/src/platform/registry/resource-link.ts +32 -32
  216. package/src/platform/registry/resource-registry.ts +890 -878
  217. package/src/platform/registry/serialization.ts +295 -295
  218. package/src/platform/registry/serialized-types.ts +166 -166
  219. package/src/platform/registry/stats-types.ts +68 -68
  220. package/src/platform/registry/types.ts +425 -425
  221. package/src/platform/registry/validation.ts +745 -743
  222. package/src/platform/utils/__tests__/validation.test.ts +1084 -1084
  223. package/src/platform/utils/validation.ts +425 -425
  224. package/src/projects/api-schemas.test.ts +39 -39
  225. package/src/projects/api-schemas.ts +291 -291
  226. package/src/reference/_generated/contracts.md +2389 -2121
  227. package/src/reference/glossary.md +76 -76
  228. package/src/scaffold-registry/__tests__/index.test.ts +206 -206
  229. package/src/scaffold-registry/__tests__/schema.test.ts +166 -166
  230. package/src/scaffold-registry/index.ts +392 -392
  231. package/src/scaffold-registry/schema.ts +243 -243
  232. package/src/server.ts +289 -289
  233. package/src/supabase/database.types.ts +3153 -3093
  234. package/src/test-utils/README.md +37 -37
  235. package/src/test-utils/entities.ts +108 -108
  236. package/src/test-utils/fixtures/memberships.ts +82 -82
  237. package/src/test-utils/index.ts +12 -12
  238. package/src/test-utils/organization-model.ts +65 -65
  239. package/src/test-utils/published.ts +6 -6
  240. package/src/test-utils/rls/RLSTestContext.ts +588 -588
  241. package/src/test-utils/test-utils.test.ts +44 -49
  242. package/src/organization-model/__tests__/domains/operations.test.ts +0 -203
  243. package/src/organization-model/domains/features.ts +0 -31
  244. package/src/organization-model/domains/operations.ts +0 -85
@@ -9,26 +9,26 @@
9
9
  * if the output changes without an intentional snapshot update.
10
10
  */
11
11
 
12
- import { readFileSync } from 'node:fs'
13
- import { resolve } from 'node:path'
14
- import { describe, it, expect } from 'vitest'
12
+ import { readFileSync } from 'node:fs'
13
+ import { resolve } from 'node:path'
14
+ import { describe, it, expect } from 'vitest'
15
15
 
16
16
  /** Monorepo root relative to packages/core/src/_gen/__tests__/ */
17
- const ROOT = resolve(import.meta.dirname, '..', '..', '..', '..', '..')
18
-
19
- const OUTPUT_PATH = resolve(ROOT, 'packages/core/src/reference/_generated/contracts.md')
20
- const SNAPSHOT_PATH = resolve(import.meta.dirname, '__snapshots__', 'contracts.md.snap')
21
-
22
- function normalizeSnapshotContent(content: string) {
23
- return content
24
- .replace(/\r\n/g, '\n')
25
- .replace(
26
- /<!-- Auto-generated on \d{4}-\d{2}-\d{2} by scripts\/monorepo\/generate-scaffold-contracts\.js -->/,
27
- '<!-- Auto-generated on YYYY-MM-DD by scripts/monorepo/generate-scaffold-contracts.js -->'
28
- )
29
- }
17
+ const ROOT = resolve(import.meta.dirname, '..', '..', '..', '..', '..')
30
18
 
31
- describe('scaffold-contracts generator', () => {
19
+ const OUTPUT_PATH = resolve(ROOT, 'packages/core/src/reference/_generated/contracts.md')
20
+ const SNAPSHOT_PATH = resolve(import.meta.dirname, '__snapshots__', 'contracts.md.snap')
21
+
22
+ function normalizeSnapshotContent(content: string) {
23
+ return content
24
+ .replace(/\r\n/g, '\n')
25
+ .replace(
26
+ /<!-- Auto-generated on \d{4}-\d{2}-\d{2} by scripts\/monorepo\/generate-scaffold-contracts\.js -->/,
27
+ '<!-- Auto-generated on YYYY-MM-DD by scripts/monorepo/generate-scaffold-contracts.js -->'
28
+ )
29
+ }
30
+
31
+ describe('scaffold-contracts generator', () => {
32
32
  it('output file exists and has content', () => {
33
33
  // The generator must have been run (either manually or by CI gen step).
34
34
  // This test validates the committed artifact — it does NOT re-run the generator
@@ -46,19 +46,19 @@ describe('scaffold-contracts generator', () => {
46
46
  expect(content.length).toBeGreaterThan(0)
47
47
  })
48
48
 
49
- it('output file matches stored snapshot', () => {
50
- let content: string
51
- try {
52
- content = readFileSync(OUTPUT_PATH, 'utf8')
49
+ it('output file matches stored snapshot', () => {
50
+ let content: string
51
+ try {
52
+ content = readFileSync(OUTPUT_PATH, 'utf8')
53
53
  } catch {
54
54
  throw new Error(
55
55
  `Generated file not found: ${OUTPUT_PATH}\n` +
56
- `Run "pnpm scaffold:generate" first to produce the artifact before snapshotting.`
57
- )
58
- }
59
-
60
- const snapshot = readFileSync(SNAPSHOT_PATH, 'utf8')
61
-
62
- expect(normalizeSnapshotContent(content)).toBe(normalizeSnapshotContent(snapshot))
63
- })
64
- })
56
+ `Run "pnpm scaffold:generate" first to produce the artifact before snapshotting.`
57
+ )
58
+ }
59
+
60
+ const snapshot = readFileSync(SNAPSHOT_PATH, 'utf8')
61
+
62
+ expect(normalizeSnapshotContent(content)).toBe(normalizeSnapshotContent(snapshot))
63
+ })
64
+ })
@@ -1,217 +1,217 @@
1
- import crypto from 'crypto'
2
- import { describe, it, expect, beforeAll } from 'vitest'
3
- import { encryptCredential, decryptCredential, setKek, CURRENT_KEY_ID } from '../server/encryption'
4
-
5
- beforeAll(() => {
6
- setKek(CURRENT_KEY_ID, crypto.randomBytes(32))
7
- })
8
-
9
- describe('Credential Encryption', () => {
10
- describe('encryptCredential', () => {
11
- it('encrypts plaintext to base64-encoded JSON string', () => {
12
- const plaintext = 'my-secret-api-key'
13
- const encrypted = encryptCredential(plaintext)
14
-
15
- // Should be a valid JSON string
16
- expect(() => JSON.parse(encrypted)).not.toThrow()
17
-
18
- const parsed = JSON.parse(encrypted)
19
- expect(parsed).toHaveProperty('iv')
20
- expect(parsed).toHaveProperty('authTag')
21
- expect(parsed).toHaveProperty('data')
22
- })
23
-
24
- it('produces different ciphertext for same input (random IV)', () => {
25
- const plaintext = 'same-secret'
26
-
27
- const encrypted1 = encryptCredential(plaintext)
28
- const encrypted2 = encryptCredential(plaintext)
29
-
30
- // Different IVs mean different ciphertext
31
- expect(encrypted1).not.toBe(encrypted2)
32
-
33
- // But both decrypt to same plaintext
34
- expect(decryptCredential(encrypted1)).toBe(plaintext)
35
- expect(decryptCredential(encrypted2)).toBe(plaintext)
36
- })
37
-
38
- it('handles empty string', () => {
39
- const plaintext = ''
40
- const encrypted = encryptCredential(plaintext)
41
- const decrypted = decryptCredential(encrypted)
42
-
43
- expect(decrypted).toBe('')
44
- })
45
-
46
- it('handles long strings', () => {
47
- const plaintext = 'x'.repeat(10000)
48
- const encrypted = encryptCredential(plaintext)
49
- const decrypted = decryptCredential(encrypted)
50
-
51
- expect(decrypted).toBe(plaintext)
52
- })
53
-
54
- it('handles special characters and unicode', () => {
55
- const plaintext = 'Hello 世界! 🔐 Special chars: \n\t\r"\'\\`'
56
- const encrypted = encryptCredential(plaintext)
57
- const decrypted = decryptCredential(encrypted)
58
-
59
- expect(decrypted).toBe(plaintext)
60
- })
61
-
62
- it('validates key is configured at module load', () => {
63
- // We can't test missing key dynamically since MASTER_KEY is set at module load
64
- // Instead, verify the key is set and encryption works
65
- const encrypted = encryptCredential('test')
66
- expect(encrypted).toBeDefined()
67
- })
68
- })
69
-
70
- describe('decryptCredential', () => {
71
- it('decrypts ciphertext back to original plaintext', () => {
72
- const plaintext = 'my-secret-api-key'
73
- const encrypted = encryptCredential(plaintext)
74
- const decrypted = decryptCredential(encrypted)
75
-
76
- expect(decrypted).toBe(plaintext)
77
- })
78
-
79
- it('successfully decrypts with configured key', () => {
80
- const encrypted = encryptCredential('test')
81
- const decrypted = decryptCredential(encrypted)
82
-
83
- expect(decrypted).toBe('test')
84
- })
85
-
86
- it('throws error on malformed JSON', () => {
87
- expect(() => decryptCredential('not-json')).toThrow()
88
- })
89
-
90
- it('throws error on missing fields', () => {
91
- const invalidData = JSON.stringify({ iv: 'abc', authTag: 'def' }) // Missing 'data'
92
-
93
- expect(() => decryptCredential(invalidData)).toThrow()
94
- })
95
-
96
- it('throws error on tampered ciphertext', () => {
97
- const encrypted = encryptCredential('secret')
98
- const parsed = JSON.parse(encrypted)
99
-
100
- // Tamper with the data
101
- parsed.data = 'tampered-data'
102
- const tampered = JSON.stringify(parsed)
103
-
104
- expect(() => decryptCredential(tampered)).toThrow()
105
- })
106
-
107
- it('throws error on tampered auth tag', () => {
108
- const encrypted = encryptCredential('secret')
109
- const parsed = JSON.parse(encrypted)
110
-
111
- // Tamper with the auth tag
112
- parsed.authTag = 'AAAAAAAAAAAAAAAAAAAAAA=='
113
- const tampered = JSON.stringify(parsed)
114
-
115
- expect(() => decryptCredential(tampered)).toThrow()
116
- })
117
-
118
- it('detects tampering via auth tag (GCM mode)', () => {
119
- const encrypted = encryptCredential('secret')
120
- const parsed = JSON.parse(encrypted)
121
-
122
- // Tamper with just one byte of data
123
- const dataBuffer = Buffer.from(parsed.data, 'base64')
124
- dataBuffer[0] ^= 0x01 // Flip one bit
125
- parsed.data = dataBuffer.toString('base64')
126
-
127
- const tampered = JSON.stringify(parsed)
128
-
129
- // GCM should detect the tampering and throw
130
- expect(() => decryptCredential(tampered)).toThrow()
131
- })
132
-
133
- it('throws error on invalid base64 encoding', () => {
134
- const invalidData = JSON.stringify({
135
- iv: 'not-valid-base64!!!',
136
- authTag: 'AAAA',
137
- data: 'BBBB'
138
- })
139
-
140
- expect(() => decryptCredential(invalidData)).toThrow()
141
- })
142
- })
143
-
144
- describe('Round-trip encryption/decryption', () => {
145
- it('preserves API key format', () => {
146
- const apiKey = 'sk_test_1234567890abcdefghijklmnopqrstuvwxyz'
147
- const encrypted = encryptCredential(apiKey)
148
- const decrypted = decryptCredential(encrypted)
149
-
150
- expect(decrypted).toBe(apiKey)
151
- })
152
-
153
- it('preserves database connection strings', () => {
154
- const connString = 'postgresql://user:password@localhost:5432/database?sslmode=require'
155
- const encrypted = encryptCredential(connString)
156
- const decrypted = decryptCredential(encrypted)
157
-
158
- expect(decrypted).toBe(connString)
159
- })
160
-
161
- it('preserves JSON credentials', () => {
162
- const jsonCreds = JSON.stringify({
163
- type: 'service_account',
164
- project_id: 'my-project',
165
- private_key: '-----BEGIN PRIVATE KEY-----\nMIIE...\n-----END PRIVATE KEY-----\n'
166
- })
167
-
168
- const encrypted = encryptCredential(jsonCreds)
169
- const decrypted = decryptCredential(encrypted)
170
-
171
- expect(decrypted).toBe(jsonCreds)
172
- expect(JSON.parse(decrypted)).toEqual(JSON.parse(jsonCreds))
173
- })
174
- })
175
-
176
- describe('Security properties', () => {
177
- it('uses AES-256-GCM (authenticated encryption)', () => {
178
- const encrypted = encryptCredential('test')
179
- const parsed = JSON.parse(encrypted)
180
-
181
- // Auth tag should be present (GCM mode)
182
- expect(parsed.authTag).toBeDefined()
183
- expect(parsed.authTag.length).toBeGreaterThan(0)
184
- })
185
-
186
- it('uses random IV for each encryption', () => {
187
- const ivs = new Set()
188
-
189
- for (let i = 0; i < 100; i++) {
190
- const encrypted = encryptCredential('test')
191
- const parsed = JSON.parse(encrypted)
192
- ivs.add(parsed.iv)
193
- }
194
-
195
- // All IVs should be unique
196
- expect(ivs.size).toBe(100)
197
- })
198
-
199
- it('IV is 16 bytes (128 bits)', () => {
200
- const encrypted = encryptCredential('test')
201
- const parsed = JSON.parse(encrypted)
202
-
203
- // Base64-encoded 16 bytes = 24 characters
204
- const ivBuffer = Buffer.from(parsed.iv, 'base64')
205
- expect(ivBuffer.length).toBe(16)
206
- })
207
-
208
- it('auth tag is 16 bytes (128 bits)', () => {
209
- const encrypted = encryptCredential('test')
210
- const parsed = JSON.parse(encrypted)
211
-
212
- // Base64-encoded 16 bytes = 24 characters
213
- const authTagBuffer = Buffer.from(parsed.authTag, 'base64')
214
- expect(authTagBuffer.length).toBe(16)
215
- })
216
- })
217
- })
1
+ import crypto from 'crypto'
2
+ import { describe, it, expect, beforeAll } from 'vitest'
3
+ import { encryptCredential, decryptCredential, setKek, CURRENT_KEY_ID } from '../server/encryption'
4
+
5
+ beforeAll(() => {
6
+ setKek(CURRENT_KEY_ID, crypto.randomBytes(32))
7
+ })
8
+
9
+ describe('Credential Encryption', () => {
10
+ describe('encryptCredential', () => {
11
+ it('encrypts plaintext to base64-encoded JSON string', () => {
12
+ const plaintext = 'my-secret-api-key'
13
+ const encrypted = encryptCredential(plaintext)
14
+
15
+ // Should be a valid JSON string
16
+ expect(() => JSON.parse(encrypted)).not.toThrow()
17
+
18
+ const parsed = JSON.parse(encrypted)
19
+ expect(parsed).toHaveProperty('iv')
20
+ expect(parsed).toHaveProperty('authTag')
21
+ expect(parsed).toHaveProperty('data')
22
+ })
23
+
24
+ it('produces different ciphertext for same input (random IV)', () => {
25
+ const plaintext = 'same-secret'
26
+
27
+ const encrypted1 = encryptCredential(plaintext)
28
+ const encrypted2 = encryptCredential(plaintext)
29
+
30
+ // Different IVs mean different ciphertext
31
+ expect(encrypted1).not.toBe(encrypted2)
32
+
33
+ // But both decrypt to same plaintext
34
+ expect(decryptCredential(encrypted1)).toBe(plaintext)
35
+ expect(decryptCredential(encrypted2)).toBe(plaintext)
36
+ })
37
+
38
+ it('handles empty string', () => {
39
+ const plaintext = ''
40
+ const encrypted = encryptCredential(plaintext)
41
+ const decrypted = decryptCredential(encrypted)
42
+
43
+ expect(decrypted).toBe('')
44
+ })
45
+
46
+ it('handles long strings', () => {
47
+ const plaintext = 'x'.repeat(10000)
48
+ const encrypted = encryptCredential(plaintext)
49
+ const decrypted = decryptCredential(encrypted)
50
+
51
+ expect(decrypted).toBe(plaintext)
52
+ })
53
+
54
+ it('handles special characters and unicode', () => {
55
+ const plaintext = 'Hello 世界! 🔐 Special chars: \n\t\r"\'\\`'
56
+ const encrypted = encryptCredential(plaintext)
57
+ const decrypted = decryptCredential(encrypted)
58
+
59
+ expect(decrypted).toBe(plaintext)
60
+ })
61
+
62
+ it('validates key is configured at module load', () => {
63
+ // We can't test missing key dynamically since MASTER_KEY is set at module load
64
+ // Instead, verify the key is set and encryption works
65
+ const encrypted = encryptCredential('test')
66
+ expect(encrypted).toBeDefined()
67
+ })
68
+ })
69
+
70
+ describe('decryptCredential', () => {
71
+ it('decrypts ciphertext back to original plaintext', () => {
72
+ const plaintext = 'my-secret-api-key'
73
+ const encrypted = encryptCredential(plaintext)
74
+ const decrypted = decryptCredential(encrypted)
75
+
76
+ expect(decrypted).toBe(plaintext)
77
+ })
78
+
79
+ it('successfully decrypts with configured key', () => {
80
+ const encrypted = encryptCredential('test')
81
+ const decrypted = decryptCredential(encrypted)
82
+
83
+ expect(decrypted).toBe('test')
84
+ })
85
+
86
+ it('throws error on malformed JSON', () => {
87
+ expect(() => decryptCredential('not-json')).toThrow()
88
+ })
89
+
90
+ it('throws error on missing fields', () => {
91
+ const invalidData = JSON.stringify({ iv: 'abc', authTag: 'def' }) // Missing 'data'
92
+
93
+ expect(() => decryptCredential(invalidData)).toThrow()
94
+ })
95
+
96
+ it('throws error on tampered ciphertext', () => {
97
+ const encrypted = encryptCredential('secret')
98
+ const parsed = JSON.parse(encrypted)
99
+
100
+ // Tamper with the data
101
+ parsed.data = 'tampered-data'
102
+ const tampered = JSON.stringify(parsed)
103
+
104
+ expect(() => decryptCredential(tampered)).toThrow()
105
+ })
106
+
107
+ it('throws error on tampered auth tag', () => {
108
+ const encrypted = encryptCredential('secret')
109
+ const parsed = JSON.parse(encrypted)
110
+
111
+ // Tamper with the auth tag
112
+ parsed.authTag = 'AAAAAAAAAAAAAAAAAAAAAA=='
113
+ const tampered = JSON.stringify(parsed)
114
+
115
+ expect(() => decryptCredential(tampered)).toThrow()
116
+ })
117
+
118
+ it('detects tampering via auth tag (GCM mode)', () => {
119
+ const encrypted = encryptCredential('secret')
120
+ const parsed = JSON.parse(encrypted)
121
+
122
+ // Tamper with just one byte of data
123
+ const dataBuffer = Buffer.from(parsed.data, 'base64')
124
+ dataBuffer[0] ^= 0x01 // Flip one bit
125
+ parsed.data = dataBuffer.toString('base64')
126
+
127
+ const tampered = JSON.stringify(parsed)
128
+
129
+ // GCM should detect the tampering and throw
130
+ expect(() => decryptCredential(tampered)).toThrow()
131
+ })
132
+
133
+ it('throws error on invalid base64 encoding', () => {
134
+ const invalidData = JSON.stringify({
135
+ iv: 'not-valid-base64!!!',
136
+ authTag: 'AAAA',
137
+ data: 'BBBB'
138
+ })
139
+
140
+ expect(() => decryptCredential(invalidData)).toThrow()
141
+ })
142
+ })
143
+
144
+ describe('Round-trip encryption/decryption', () => {
145
+ it('preserves API key format', () => {
146
+ const apiKey = 'sk_test_1234567890abcdefghijklmnopqrstuvwxyz'
147
+ const encrypted = encryptCredential(apiKey)
148
+ const decrypted = decryptCredential(encrypted)
149
+
150
+ expect(decrypted).toBe(apiKey)
151
+ })
152
+
153
+ it('preserves database connection strings', () => {
154
+ const connString = 'postgresql://user:password@localhost:5432/database?sslmode=require'
155
+ const encrypted = encryptCredential(connString)
156
+ const decrypted = decryptCredential(encrypted)
157
+
158
+ expect(decrypted).toBe(connString)
159
+ })
160
+
161
+ it('preserves JSON credentials', () => {
162
+ const jsonCreds = JSON.stringify({
163
+ type: 'service_account',
164
+ project_id: 'my-project',
165
+ private_key: '-----BEGIN PRIVATE KEY-----\nMIIE...\n-----END PRIVATE KEY-----\n'
166
+ })
167
+
168
+ const encrypted = encryptCredential(jsonCreds)
169
+ const decrypted = decryptCredential(encrypted)
170
+
171
+ expect(decrypted).toBe(jsonCreds)
172
+ expect(JSON.parse(decrypted)).toEqual(JSON.parse(jsonCreds))
173
+ })
174
+ })
175
+
176
+ describe('Security properties', () => {
177
+ it('uses AES-256-GCM (authenticated encryption)', () => {
178
+ const encrypted = encryptCredential('test')
179
+ const parsed = JSON.parse(encrypted)
180
+
181
+ // Auth tag should be present (GCM mode)
182
+ expect(parsed.authTag).toBeDefined()
183
+ expect(parsed.authTag.length).toBeGreaterThan(0)
184
+ })
185
+
186
+ it('uses random IV for each encryption', () => {
187
+ const ivs = new Set()
188
+
189
+ for (let i = 0; i < 100; i++) {
190
+ const encrypted = encryptCredential('test')
191
+ const parsed = JSON.parse(encrypted)
192
+ ivs.add(parsed.iv)
193
+ }
194
+
195
+ // All IVs should be unique
196
+ expect(ivs.size).toBe(100)
197
+ })
198
+
199
+ it('IV is 16 bytes (128 bits)', () => {
200
+ const encrypted = encryptCredential('test')
201
+ const parsed = JSON.parse(encrypted)
202
+
203
+ // Base64-encoded 16 bytes = 24 characters
204
+ const ivBuffer = Buffer.from(parsed.iv, 'base64')
205
+ expect(ivBuffer.length).toBe(16)
206
+ })
207
+
208
+ it('auth tag is 16 bytes (128 bits)', () => {
209
+ const encrypted = encryptCredential('test')
210
+ const parsed = JSON.parse(encrypted)
211
+
212
+ // Base64-encoded 16 bytes = 24 characters
213
+ const authTagBuffer = Buffer.from(parsed.authTag, 'base64')
214
+ expect(authTagBuffer.length).toBe(16)
215
+ })
216
+ })
217
+ })
@@ -1,69 +1,69 @@
1
- import crypto from 'crypto'
2
-
3
- const ALGORITHM = 'aes-256-gcm'
4
-
5
- // keyId stamped on all newly encrypted ciphertexts.
6
- export const CURRENT_KEY_ID = 'platform-v1'
7
-
8
- // Implicit keyId for pre-Vault ciphertext rows that lack a keyId field. All
9
- // known rows were restamped with CURRENT_KEY_ID during the re-encryption
10
- // migration; this constant remains only so any unexpected legacy blob produces
11
- // a clear "KEK not loaded" error rather than a silent default-key decrypt.
12
- export const LEGACY_KEY_ID = 'platform-v0-legacy'
13
-
14
- interface EncryptedData {
15
- iv: string
16
- authTag: string
17
- data: string
18
- keyId?: string
19
- }
20
-
21
- const kekMap = new Map<string, Buffer>()
22
-
23
- export function setKek(keyId: string, key: Buffer): void {
24
- if (key.length !== 32) {
25
- throw new Error(`KEK must be 32 bytes (256 bits); got ${key.length}`)
26
- }
27
- kekMap.set(keyId, key)
28
- }
29
-
30
- export function clearKeks(): void {
31
- kekMap.clear()
32
- }
33
-
34
- function resolveKek(keyId: string): Buffer | undefined {
35
- return kekMap.get(keyId)
36
- }
37
-
38
- export function encryptCredential(plaintext: string): string {
39
- const key = resolveKek(CURRENT_KEY_ID)
40
- if (!key) {
41
- throw new Error(`Encryption KEK '${CURRENT_KEY_ID}' not loaded; call setKek() at boot`)
42
- }
43
-
44
- const iv = crypto.randomBytes(16)
45
- const cipher = crypto.createCipheriv(ALGORITHM, key, iv)
46
- const encrypted = Buffer.concat([cipher.update(plaintext, 'utf8'), cipher.final()])
47
- const authTag = cipher.getAuthTag()
48
-
49
- return JSON.stringify({
50
- iv: iv.toString('base64'),
51
- authTag: authTag.toString('base64'),
52
- data: encrypted.toString('base64'),
53
- keyId: CURRENT_KEY_ID
54
- })
55
- }
56
-
57
- export function decryptCredential(encrypted: string): string {
58
- const { iv, authTag, data, keyId } = JSON.parse(encrypted) as EncryptedData
59
- const resolvedKeyId = keyId ?? LEGACY_KEY_ID
60
- const key = resolveKek(resolvedKeyId)
61
- if (!key) {
62
- throw new Error(`Decryption KEK '${resolvedKeyId}' not loaded`)
63
- }
64
-
65
- const decipher = crypto.createDecipheriv(ALGORITHM, key, Buffer.from(iv, 'base64'))
66
- decipher.setAuthTag(Buffer.from(authTag, 'base64'))
67
-
68
- return Buffer.concat([decipher.update(Buffer.from(data, 'base64')), decipher.final()]).toString('utf8')
69
- }
1
+ import crypto from 'crypto'
2
+
3
+ const ALGORITHM = 'aes-256-gcm'
4
+
5
+ // keyId stamped on all newly encrypted ciphertexts.
6
+ export const CURRENT_KEY_ID = 'platform-v1'
7
+
8
+ // Implicit keyId for pre-Vault ciphertext rows that lack a keyId field. All
9
+ // known rows were restamped with CURRENT_KEY_ID during the re-encryption
10
+ // migration; this constant remains only so any unexpected legacy blob produces
11
+ // a clear "KEK not loaded" error rather than a silent default-key decrypt.
12
+ export const LEGACY_KEY_ID = 'platform-v0-legacy'
13
+
14
+ interface EncryptedData {
15
+ iv: string
16
+ authTag: string
17
+ data: string
18
+ keyId?: string
19
+ }
20
+
21
+ const kekMap = new Map<string, Buffer>()
22
+
23
+ export function setKek(keyId: string, key: Buffer): void {
24
+ if (key.length !== 32) {
25
+ throw new Error(`KEK must be 32 bytes (256 bits); got ${key.length}`)
26
+ }
27
+ kekMap.set(keyId, key)
28
+ }
29
+
30
+ export function clearKeks(): void {
31
+ kekMap.clear()
32
+ }
33
+
34
+ function resolveKek(keyId: string): Buffer | undefined {
35
+ return kekMap.get(keyId)
36
+ }
37
+
38
+ export function encryptCredential(plaintext: string): string {
39
+ const key = resolveKek(CURRENT_KEY_ID)
40
+ if (!key) {
41
+ throw new Error(`Encryption KEK '${CURRENT_KEY_ID}' not loaded; call setKek() at boot`)
42
+ }
43
+
44
+ const iv = crypto.randomBytes(16)
45
+ const cipher = crypto.createCipheriv(ALGORITHM, key, iv)
46
+ const encrypted = Buffer.concat([cipher.update(plaintext, 'utf8'), cipher.final()])
47
+ const authTag = cipher.getAuthTag()
48
+
49
+ return JSON.stringify({
50
+ iv: iv.toString('base64'),
51
+ authTag: authTag.toString('base64'),
52
+ data: encrypted.toString('base64'),
53
+ keyId: CURRENT_KEY_ID
54
+ })
55
+ }
56
+
57
+ export function decryptCredential(encrypted: string): string {
58
+ const { iv, authTag, data, keyId } = JSON.parse(encrypted) as EncryptedData
59
+ const resolvedKeyId = keyId ?? LEGACY_KEY_ID
60
+ const key = resolveKek(resolvedKeyId)
61
+ if (!key) {
62
+ throw new Error(`Decryption KEK '${resolvedKeyId}' not loaded`)
63
+ }
64
+
65
+ const decipher = crypto.createDecipheriv(ALGORITHM, key, Buffer.from(iv, 'base64'))
66
+ decipher.setAuthTag(Buffer.from(authTag, 'base64'))
67
+
68
+ return Buffer.concat([decipher.update(Buffer.from(data, 'base64')), decipher.final()]).toString('utf8')
69
+ }