@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
@@ -0,0 +1,210 @@
1
+ import { describe, expect, it } from 'vitest'
2
+ import {
3
+ CONTENT_KIND_REGISTRY,
4
+ defineContentType,
5
+ lookupContentType,
6
+ pipelineKind,
7
+ stageKind,
8
+ templateKind,
9
+ templateStepKind,
10
+ statusFlowKind,
11
+ statusKind,
12
+ configKvKind
13
+ } from '../content-kinds/index'
14
+ import type { ContentTypeDefinition, ContentTypeKey } from '../content-kinds/types'
15
+
16
+ // ---------------------------------------------------------------------------
17
+ // CONTENT_KIND_REGISTRY shape (D8: static object literal)
18
+ // ---------------------------------------------------------------------------
19
+
20
+ describe('CONTENT_KIND_REGISTRY', () => {
21
+ it('contains all 7 expected keys', () => {
22
+ const expectedKeys: ContentTypeKey[] = [
23
+ 'schema:pipeline',
24
+ 'schema:stage',
25
+ 'schema:template',
26
+ 'schema:template-step',
27
+ 'schema:status-flow',
28
+ 'schema:status',
29
+ 'config:kv'
30
+ ]
31
+ for (const key of expectedKeys) {
32
+ expect(CONTENT_KIND_REGISTRY).toHaveProperty(key)
33
+ }
34
+ })
35
+
36
+ it('maps schema:pipeline to the pipelineKind definition', () => {
37
+ expect(CONTENT_KIND_REGISTRY['schema:pipeline']).toBe(pipelineKind)
38
+ })
39
+
40
+ it('maps schema:stage to the stageKind definition', () => {
41
+ expect(CONTENT_KIND_REGISTRY['schema:stage']).toBe(stageKind)
42
+ })
43
+
44
+ it('maps schema:template to the templateKind definition', () => {
45
+ expect(CONTENT_KIND_REGISTRY['schema:template']).toBe(templateKind)
46
+ })
47
+
48
+ it('maps schema:template-step to the templateStepKind definition', () => {
49
+ expect(CONTENT_KIND_REGISTRY['schema:template-step']).toBe(templateStepKind)
50
+ })
51
+
52
+ it('maps schema:status-flow to the statusFlowKind definition', () => {
53
+ expect(CONTENT_KIND_REGISTRY['schema:status-flow']).toBe(statusFlowKind)
54
+ })
55
+
56
+ it('maps schema:status to the statusKind definition', () => {
57
+ expect(CONTENT_KIND_REGISTRY['schema:status']).toBe(statusKind)
58
+ })
59
+
60
+ it('maps config:kv to the configKvKind definition', () => {
61
+ expect(CONTENT_KIND_REGISTRY['config:kv']).toBe(configKvKind)
62
+ })
63
+
64
+ it('has exactly 7 keys (no extra unintended entries)', () => {
65
+ expect(Object.keys(CONTENT_KIND_REGISTRY)).toHaveLength(7)
66
+ })
67
+
68
+ it('every entry has kind, type, payloadSchema, label, and description fields', () => {
69
+ for (const [key, def] of Object.entries(CONTENT_KIND_REGISTRY)) {
70
+ expect(def).toHaveProperty('kind')
71
+ expect(def).toHaveProperty('type')
72
+ expect(def).toHaveProperty('payloadSchema')
73
+ expect(typeof def.kind).toBe('string')
74
+ expect(typeof def.type).toBe('string')
75
+ expect(def.label).toBeTruthy()
76
+ expect(def.description).toBeTruthy()
77
+ // key must equal kind:type
78
+ expect(key).toBe(`${def.kind}:${def.type}`)
79
+ }
80
+ })
81
+ })
82
+
83
+ // ---------------------------------------------------------------------------
84
+ // parentTypes constraint
85
+ // ---------------------------------------------------------------------------
86
+
87
+ describe('parentTypes constraints', () => {
88
+ it('schema:pipeline has empty parentTypes (top-level, no parent required)', () => {
89
+ expect(pipelineKind.parentTypes).toEqual([])
90
+ })
91
+
92
+ it('schema:stage parentTypes is ["schema:pipeline"]', () => {
93
+ expect(stageKind.parentTypes).toEqual(['schema:pipeline'])
94
+ })
95
+
96
+ it('schema:template has empty parentTypes', () => {
97
+ expect(templateKind.parentTypes).toEqual([])
98
+ })
99
+
100
+ it('schema:template-step parentTypes is ["schema:template"]', () => {
101
+ expect(templateStepKind.parentTypes).toEqual(['schema:template'])
102
+ })
103
+
104
+ it('schema:status-flow has empty parentTypes', () => {
105
+ expect(statusFlowKind.parentTypes).toEqual([])
106
+ })
107
+
108
+ it('schema:status parentTypes is ["schema:status-flow"]', () => {
109
+ expect(statusKind.parentTypes).toEqual(['schema:status-flow'])
110
+ })
111
+
112
+ it('config:kv has empty parentTypes', () => {
113
+ expect(configKvKind.parentTypes).toEqual([])
114
+ })
115
+ })
116
+
117
+ // ---------------------------------------------------------------------------
118
+ // lookupContentType
119
+ // ---------------------------------------------------------------------------
120
+
121
+ describe('lookupContentType', () => {
122
+ it('returns the correct definition for schema:pipeline', () => {
123
+ const def = lookupContentType('schema', 'pipeline')
124
+ expect(def).toBe(pipelineKind)
125
+ })
126
+
127
+ it('returns the correct definition for schema:stage', () => {
128
+ const def = lookupContentType('schema', 'stage')
129
+ expect(def).toBe(stageKind)
130
+ })
131
+
132
+ it('returns the correct definition for config:kv', () => {
133
+ const def = lookupContentType('config', 'kv')
134
+ expect(def).toBe(configKvKind)
135
+ })
136
+
137
+ it('returns undefined for an unregistered kind (D2: not an error)', () => {
138
+ const def = lookupContentType('tenant', 'custom-thing')
139
+ expect(def).toBeUndefined()
140
+ })
141
+
142
+ it('returns undefined for unknown kind with valid type', () => {
143
+ const def = lookupContentType('unknown', 'pipeline')
144
+ expect(def).toBeUndefined()
145
+ })
146
+
147
+ it('returns undefined for valid kind with unknown type', () => {
148
+ const def = lookupContentType('schema', 'not-a-real-type')
149
+ expect(def).toBeUndefined()
150
+ })
151
+
152
+ it('returns undefined for empty kind string', () => {
153
+ const def = lookupContentType('', 'pipeline')
154
+ expect(def).toBeUndefined()
155
+ })
156
+
157
+ it('returns undefined for empty type string', () => {
158
+ const def = lookupContentType('schema', '')
159
+ expect(def).toBeUndefined()
160
+ })
161
+ })
162
+
163
+ // ---------------------------------------------------------------------------
164
+ // defineContentType (identity factory, L16)
165
+ // ---------------------------------------------------------------------------
166
+
167
+ describe('defineContentType', () => {
168
+ it('is an identity function — returns the same object reference', () => {
169
+ const def: ContentTypeDefinition<{ name: string }> = {
170
+ kind: 'tenant',
171
+ type: 'custom',
172
+ payloadSchema: { safeParse: () => ({ success: true, data: { name: 'x' } }) } as unknown as ContentTypeDefinition<{
173
+ name: string
174
+ }>['payloadSchema'],
175
+ label: 'Custom',
176
+ description: 'A tenant-defined type.',
177
+ parentTypes: []
178
+ }
179
+ const result = defineContentType(def)
180
+ expect(result).toBe(def)
181
+ })
182
+
183
+ it('does NOT mutate the input definition', () => {
184
+ const def: ContentTypeDefinition<unknown> = {
185
+ kind: 'schema',
186
+ type: 'pipeline',
187
+ payloadSchema: {
188
+ safeParse: () => ({ success: true, data: {} })
189
+ } as unknown as ContentTypeDefinition<unknown>['payloadSchema'],
190
+ parentTypes: []
191
+ }
192
+ const original = { ...def }
193
+ defineContentType(def)
194
+ expect(def).toEqual(original)
195
+ })
196
+
197
+ it('does NOT register the definition globally (no side-effects on CONTENT_KIND_REGISTRY)', () => {
198
+ const tenantKey = 'tenant:my-new-type'
199
+ defineContentType({
200
+ kind: 'tenant',
201
+ type: 'my-new-type',
202
+ payloadSchema: {
203
+ safeParse: () => ({ success: true, data: {} })
204
+ } as unknown as ContentTypeDefinition<unknown>['payloadSchema'],
205
+ parentTypes: []
206
+ })
207
+ // After calling defineContentType, the registry must remain unchanged (D8: static)
208
+ expect(CONTENT_KIND_REGISTRY).not.toHaveProperty(tenantKey)
209
+ })
210
+ })
@@ -1,194 +1,168 @@
1
- import { describe, expect, it } from 'vitest'
2
- import { DEFAULT_ORGANIZATION_MODEL } from '../defaults'
3
- import { DEFAULT_ORGANIZATION_MODEL_BRANDING } from '../domains/branding'
4
- import { OrganizationModelBrandingSchema } from '../domains/branding'
5
- import { DEFAULT_ORGANIZATION_MODEL_SALES } from '../domains/sales'
6
- import { OrganizationModelSalesSchema } from '../domains/sales'
7
- import { DEFAULT_ORGANIZATION_MODEL_PROJECTS } from '../domains/projects'
8
- import { OrganizationModelProjectsSchema } from '../domains/projects'
9
- import { DEFAULT_ORGANIZATION_MODEL_PROSPECTING } from '../domains/prospecting'
10
- import { OrganizationModelProspectingSchema } from '../domains/prospecting'
11
- import { DEFAULT_ORGANIZATION_MODEL_NAVIGATION } from '../domains/navigation'
12
- import { OrganizationModelNavigationSchema } from '../domains/navigation'
13
- import { DEFAULT_ORGANIZATION_MODEL_OPERATIONS } from '../domains/operations'
14
- import { OperationsDomainSchema } from '../domains/operations'
15
- import { DEFAULT_ORGANIZATION_MODEL_STATUSES } from '../domains/statuses'
16
- import { StatusesDomainSchema } from '../domains/statuses'
17
- import { DEFAULT_ORGANIZATION_MODEL_SYSTEMS, SystemsDomainSchema } from '../domains/systems'
18
- import { resolveOrganizationModel } from '../resolve'
19
- import { OrganizationModelSchema } from '../schema'
20
-
21
- // All DEFAULT_ORGANIZATION_MODEL* constants and the schemas used to validate them.
22
- // The composite constant lives in defaults.ts; domain sub-constants live in their
23
- // respective domain files. All are covered here for roundtrip integrity.
24
- const domainCases = [
25
- {
26
- name: 'DEFAULT_ORGANIZATION_MODEL_BRANDING',
27
- constant: DEFAULT_ORGANIZATION_MODEL_BRANDING,
28
- schema: OrganizationModelBrandingSchema
29
- },
30
- {
31
- name: 'DEFAULT_ORGANIZATION_MODEL_SALES',
32
- constant: DEFAULT_ORGANIZATION_MODEL_SALES,
33
- schema: OrganizationModelSalesSchema
34
- },
35
- {
36
- name: 'DEFAULT_ORGANIZATION_MODEL_PROJECTS',
37
- constant: DEFAULT_ORGANIZATION_MODEL_PROJECTS,
38
- schema: OrganizationModelProjectsSchema
39
- },
40
- {
41
- name: 'DEFAULT_ORGANIZATION_MODEL_PROSPECTING',
42
- constant: DEFAULT_ORGANIZATION_MODEL_PROSPECTING,
43
- schema: OrganizationModelProspectingSchema
44
- },
45
- {
46
- name: 'DEFAULT_ORGANIZATION_MODEL_NAVIGATION',
47
- constant: DEFAULT_ORGANIZATION_MODEL_NAVIGATION,
48
- schema: OrganizationModelNavigationSchema
49
- },
50
- {
51
- name: 'DEFAULT_ORGANIZATION_MODEL_STATUSES',
52
- constant: DEFAULT_ORGANIZATION_MODEL_STATUSES,
53
- schema: StatusesDomainSchema
54
- },
55
- {
56
- name: 'DEFAULT_ORGANIZATION_MODEL_OPERATIONS',
57
- constant: DEFAULT_ORGANIZATION_MODEL_OPERATIONS,
58
- schema: OperationsDomainSchema
59
- },
60
- {
61
- name: 'DEFAULT_ORGANIZATION_MODEL_SYSTEMS',
62
- constant: DEFAULT_ORGANIZATION_MODEL_SYSTEMS,
63
- schema: SystemsDomainSchema
64
- }
65
- ] as const
66
-
67
- describe('organization-model defaults', () => {
68
- describe('DEFAULT_ORGANIZATION_MODEL (composite)', () => {
69
- it('passes OrganizationModelSchema.parse without throwing', () => {
70
- expect(() => OrganizationModelSchema.parse(DEFAULT_ORGANIZATION_MODEL)).not.toThrow()
71
- })
72
-
73
- it('resolveOrganizationModel(DEFAULT_ORGANIZATION_MODEL) returns a schema-valid model', () => {
74
- const result = resolveOrganizationModel(DEFAULT_ORGANIZATION_MODEL)
75
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
76
- })
77
-
78
- it('resolveOrganizationModel with no override equals resolveOrganizationModel with DEFAULT_ORGANIZATION_MODEL', () => {
79
- const fromUndefined = resolveOrganizationModel(undefined)
80
- const fromDefault = resolveOrganizationModel(DEFAULT_ORGANIZATION_MODEL)
81
- expect(fromDefault).toEqual(fromUndefined)
82
- })
83
-
84
- it('keeps Command View enabled but development-only in the default Knowledge navigation', () => {
85
- const result = resolveOrganizationModel(undefined)
86
- const commandViewFeature = result.features.find((feature) => feature.id === 'knowledge.command-view')
87
- const commandViewSurface = DEFAULT_ORGANIZATION_MODEL_NAVIGATION.surfaces.find(
88
- (surface) => surface.id === 'knowledge.command-view'
89
- )
90
-
91
- expect(commandViewFeature?.enabled).toBe(true)
92
- expect(commandViewFeature?.devOnly).toBe(true)
93
- expect(commandViewSurface?.enabled).toBe(true)
94
- expect(commandViewSurface?.devOnly).toBe(true)
95
- })
96
- })
97
-
98
- describe.each(domainCases)('$name (domain sub-constant)', ({ name: _name, constant, schema }) => {
99
- it('passes its domain schema parse without throwing', () => {
100
- expect(() => schema.parse(constant)).not.toThrow()
101
- })
102
- })
103
-
104
- describe('DEFAULT_ORGANIZATION_MODEL_BRANDING via resolveOrganizationModel', () => {
105
- it('resolving with branding override produces a schema-valid model', () => {
106
- const result = resolveOrganizationModel({ branding: DEFAULT_ORGANIZATION_MODEL_BRANDING })
107
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
108
- })
109
- })
110
-
111
- describe('DEFAULT_ORGANIZATION_MODEL_SALES via resolveOrganizationModel', () => {
112
- it('resolving with sales override produces a schema-valid model', () => {
113
- const result = resolveOrganizationModel({ sales: DEFAULT_ORGANIZATION_MODEL_SALES })
114
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
115
- })
116
- })
117
-
118
- describe('DEFAULT_ORGANIZATION_MODEL_PROJECTS via resolveOrganizationModel', () => {
119
- it('resolving with projects override produces a schema-valid model', () => {
120
- const result = resolveOrganizationModel({ projects: DEFAULT_ORGANIZATION_MODEL_PROJECTS })
121
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
122
- })
123
- })
124
-
125
- describe('DEFAULT_ORGANIZATION_MODEL_PROSPECTING via resolveOrganizationModel', () => {
126
- it('resolving with prospecting override produces a schema-valid model', () => {
127
- const result = resolveOrganizationModel({ prospecting: DEFAULT_ORGANIZATION_MODEL_PROSPECTING })
128
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
129
- })
130
- })
131
-
132
- describe('DEFAULT_ORGANIZATION_MODEL_NAVIGATION via resolveOrganizationModel', () => {
133
- it('resolving with navigation override produces a schema-valid model', () => {
134
- const result = resolveOrganizationModel({ navigation: DEFAULT_ORGANIZATION_MODEL_NAVIGATION })
135
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
136
- })
137
- })
138
-
139
- describe('DEFAULT_ORGANIZATION_MODEL_OPERATIONS via resolveOrganizationModel', () => {
140
- it('resolving with operations override produces a schema-valid model', () => {
141
- const result = resolveOrganizationModel({ operations: DEFAULT_ORGANIZATION_MODEL_OPERATIONS })
142
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
143
- })
144
-
145
- it('default operations seed covers all 5 entity categories', () => {
146
- expect(DEFAULT_ORGANIZATION_MODEL_OPERATIONS.entries).toHaveLength(5)
147
- })
148
-
149
- it('every entry has a non-empty id, label, and semanticClass', () => {
150
- for (const entry of DEFAULT_ORGANIZATION_MODEL_OPERATIONS.entries) {
151
- expect(entry.id.length).toBeGreaterThan(0)
152
- expect(entry.label.length).toBeGreaterThan(0)
153
- expect(entry.semanticClass.length).toBeGreaterThan(0)
154
- }
155
- })
156
-
157
- it('all entry ids are unique', () => {
158
- const ids = DEFAULT_ORGANIZATION_MODEL_OPERATIONS.entries.map((e) => e.id)
159
- const uniqueIds = new Set(ids)
160
- expect(uniqueIds.size).toBe(ids.length)
161
- })
162
- })
163
-
164
- describe('DEFAULT_ORGANIZATION_MODEL_STATUSES via resolveOrganizationModel', () => {
165
- it('resolving with statuses override produces a schema-valid model', () => {
166
- const result = resolveOrganizationModel({ statuses: DEFAULT_ORGANIZATION_MODEL_STATUSES })
167
- expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
168
- })
169
-
170
- it('default statuses seed covers all 9 delivery.task values', () => {
171
- const taskEntries = DEFAULT_ORGANIZATION_MODEL_STATUSES.entries.filter((e) => e.semanticClass === 'delivery.task')
172
- expect(taskEntries).toHaveLength(9)
173
- })
174
-
175
- it('default statuses seed covers all 5 queue values', () => {
176
- const queueEntries = DEFAULT_ORGANIZATION_MODEL_STATUSES.entries.filter((e) => e.semanticClass === 'queue')
177
- expect(queueEntries).toHaveLength(5)
178
- })
179
-
180
- it('every entry has a non-empty id, label, and semanticClass', () => {
181
- for (const entry of DEFAULT_ORGANIZATION_MODEL_STATUSES.entries) {
182
- expect(entry.id.length).toBeGreaterThan(0)
183
- expect(entry.label.length).toBeGreaterThan(0)
184
- expect(entry.semanticClass.length).toBeGreaterThan(0)
185
- }
186
- })
187
-
188
- it('all entry ids are unique', () => {
189
- const ids = DEFAULT_ORGANIZATION_MODEL_STATUSES.entries.map((e) => e.id)
190
- const uniqueIds = new Set(ids)
191
- expect(uniqueIds.size).toBe(ids.length)
192
- })
193
- })
194
- })
1
+ import { describe, expect, it } from 'vitest'
2
+ import { DEFAULT_ORGANIZATION_MODEL } from '../defaults'
3
+ import { DEFAULT_ORGANIZATION_MODEL_BRANDING, OrganizationModelBrandingSchema } from '../domains/branding'
4
+ import { DEFAULT_ORGANIZATION_MODEL_ENTITIES, EntitiesDomainSchema } from '../domains/entities'
5
+ import { DEFAULT_ORGANIZATION_MODEL_STATUSES, StatusesDomainSchema } from '../domains/statuses'
6
+ import { DEFAULT_ORGANIZATION_MODEL_SYSTEMS, SystemsDomainSchema } from '../domains/systems'
7
+ import { resolveOrganizationModel } from '../resolve'
8
+ import { OrganizationModelSchema } from '../schema'
9
+ import { DEFAULT_ORGANIZATION_MODEL_KNOWLEDGE } from '../defaults'
10
+
11
+ // Phase 4 (D8): sales, prospecting, projects, navigation top-level OM fields removed.
12
+ // DEFAULT_ORGANIZATION_MODEL_SALES / OrganizationModelSalesSchema,
13
+ // DEFAULT_ORGANIZATION_MODEL_PROJECTS / OrganizationModelProjectsSchema,
14
+ // DEFAULT_ORGANIZATION_MODEL_PROSPECTING / OrganizationModelProspectingSchema,
15
+ // DEFAULT_ORGANIZATION_MODEL_NAVIGATION / OrganizationModelNavigationSchema
16
+ // are no longer exported from their respective domain files.
17
+ // Tests that depended on these exports are skipped with explicit reason comments.
18
+
19
+ // Only the domain constants and schemas that still exist are covered here.
20
+ const domainCases = [
21
+ {
22
+ name: 'DEFAULT_ORGANIZATION_MODEL_BRANDING',
23
+ constant: DEFAULT_ORGANIZATION_MODEL_BRANDING,
24
+ schema: OrganizationModelBrandingSchema
25
+ },
26
+ {
27
+ name: 'DEFAULT_ORGANIZATION_MODEL_ENTITIES',
28
+ constant: DEFAULT_ORGANIZATION_MODEL_ENTITIES,
29
+ schema: EntitiesDomainSchema
30
+ },
31
+ {
32
+ name: 'DEFAULT_ORGANIZATION_MODEL_STATUSES',
33
+ constant: DEFAULT_ORGANIZATION_MODEL_STATUSES,
34
+ schema: StatusesDomainSchema
35
+ },
36
+ {
37
+ name: 'DEFAULT_ORGANIZATION_MODEL_SYSTEMS',
38
+ constant: DEFAULT_ORGANIZATION_MODEL_SYSTEMS,
39
+ schema: SystemsDomainSchema
40
+ }
41
+ ] as const
42
+
43
+ describe('organization-model defaults', () => {
44
+ describe('DEFAULT_ORGANIZATION_MODEL (composite)', () => {
45
+ it('passes OrganizationModelSchema.parse without throwing', () => {
46
+ expect(() => OrganizationModelSchema.parse(DEFAULT_ORGANIZATION_MODEL)).not.toThrow()
47
+ })
48
+
49
+ it('resolveOrganizationModel(DEFAULT_ORGANIZATION_MODEL) returns a schema-valid model', () => {
50
+ const result = resolveOrganizationModel(DEFAULT_ORGANIZATION_MODEL)
51
+ expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
52
+ })
53
+
54
+ it('resolveOrganizationModel with no override equals resolveOrganizationModel with DEFAULT_ORGANIZATION_MODEL', () => {
55
+ const fromUndefined = resolveOrganizationModel(undefined)
56
+ const fromDefault = resolveOrganizationModel(DEFAULT_ORGANIZATION_MODEL)
57
+ expect(fromDefault).toEqual(fromUndefined)
58
+ })
59
+
60
+ // Phase 4 (D8): model.navigation removed. The knowledge.command-view surface tracking
61
+ // previously used DEFAULT_ORGANIZATION_MODEL_NAVIGATION.surfaces — skipped since that
62
+ // constant is deleted. The system itself still exists; only the navigation surface
63
+ // reference is gone from the top-level schema.
64
+ it.skip('keeps Command View enabled but development-only in the default Knowledge navigation (deferred — Phase 4: navigation domain removed)', () => {
65
+ // Previously checked DEFAULT_ORGANIZATION_MODEL_NAVIGATION.surfaces.
66
+ // The command-view system still exists in DEFAULT_ORGANIZATION_MODEL.systems.
67
+ // Re-enable with a top-level surfaces Record fixture if that assertion is needed.
68
+ })
69
+ })
70
+
71
+ describe.each(domainCases)('$name (domain sub-constant)', ({ name: _name, constant, schema }) => {
72
+ it('passes its domain schema parse without throwing', () => {
73
+ expect(() => schema.parse(constant)).not.toThrow()
74
+ })
75
+ })
76
+
77
+ // Phase 4 (D8): OrganizationModelSalesSchema, DEFAULT_ORGANIZATION_MODEL_SALES removed.
78
+ describe.skip('DEFAULT_ORGANIZATION_MODEL_SALES via resolveOrganizationModel (deferred Phase 4: sales domain removed)', () => {
79
+ it('resolving with sales override produces a schema-valid model', () => {
80
+ // Previously: resolveOrganizationModel({ sales: DEFAULT_ORGANIZATION_MODEL_SALES })
81
+ // Sales data now lives in system.content. Use getAllPipelines(model) from migration-helpers.
82
+ })
83
+ })
84
+
85
+ // Phase 4 (D8): OrganizationModelProjectsSchema, DEFAULT_ORGANIZATION_MODEL_PROJECTS removed.
86
+ describe.skip('DEFAULT_ORGANIZATION_MODEL_PROJECTS via resolveOrganizationModel (deferred Phase 4: projects domain removed)', () => {
87
+ it('resolving with projects override produces a schema-valid model', () => {
88
+ // Previously: resolveOrganizationModel({ projects: DEFAULT_ORGANIZATION_MODEL_PROJECTS })
89
+ // Project statuses now live in system.content. Use getAllProjectStatuses(model) from migration-helpers.
90
+ })
91
+ })
92
+
93
+ // Phase 4 (D8): OrganizationModelProspectingSchema, DEFAULT_ORGANIZATION_MODEL_PROSPECTING removed.
94
+ describe.skip('DEFAULT_ORGANIZATION_MODEL_PROSPECTING via resolveOrganizationModel (deferred — Phase 4: prospecting domain removed)', () => {
95
+ it('resolving with prospecting override produces a schema-valid model', () => {
96
+ // Previously: resolveOrganizationModel({ prospecting: DEFAULT_ORGANIZATION_MODEL_PROSPECTING })
97
+ // Prospecting stages/templates now live in system.content. Use getAllProspectingStages() / getAllBuildTemplates().
98
+ })
99
+ })
100
+
101
+ // Phase 4 (D8): OrganizationModelNavigationSchema, DEFAULT_ORGANIZATION_MODEL_NAVIGATION removed.
102
+ describe.skip('DEFAULT_ORGANIZATION_MODEL_NAVIGATION via resolveOrganizationModel (deferred — Phase 4: navigation domain removed)', () => {
103
+ it('resolving with navigation override produces a schema-valid model', () => {
104
+ // Previously: resolveOrganizationModel({ navigation: DEFAULT_ORGANIZATION_MODEL_NAVIGATION })
105
+ // Surfaces and navigation groups are now top-level Records on OrganizationModelSchemaBase.
106
+ })
107
+ })
108
+
109
+ describe('DEFAULT_ORGANIZATION_MODEL_BRANDING via resolveOrganizationModel', () => {
110
+ it('resolving with branding override produces a schema-valid model', () => {
111
+ const result = resolveOrganizationModel({ branding: DEFAULT_ORGANIZATION_MODEL_BRANDING })
112
+ expect(() => OrganizationModelSchema.parse(result)).not.toThrow()
113
+ })
114
+ })
115
+
116
+ describe('DEFAULT_ORGANIZATION_MODEL_STATUSES via resolveOrganizationModel', () => {
117
+ // Phase 4 (D1): statuses top-level field removed from OM schema.
118
+ // DEFAULT_ORGANIZATION_MODEL_STATUSES is still exported from domains/statuses.ts
119
+ // (it's used by the UI statuses domain as a semantic registry, not OM-schema-level data).
120
+ // The resolveOrganizationModel override with { statuses: ... } was silently accepted
121
+ // before Phase 4 because statuses was an extra field; now statuses is gone from OM entirely.
122
+ // We keep the data-integrity checks below; the resolveOrganizationModel override test is skipped.
123
+ it.skip('resolving with statuses override produces a schema-valid model (deferred — Phase 4: statuses field removed from OM schema)', () => {
124
+ // Previously: resolveOrganizationModel({ statuses: DEFAULT_ORGANIZATION_MODEL_STATUSES })
125
+ // Status data now lives in system.content via (schema:status-flow) + (schema:status) content nodes.
126
+ })
127
+
128
+ it('default statuses seed covers all 9 delivery.task values', () => {
129
+ const taskEntries = Object.values(DEFAULT_ORGANIZATION_MODEL_STATUSES).filter(
130
+ (e) => e.semanticClass === 'delivery.task'
131
+ )
132
+ expect(taskEntries).toHaveLength(9)
133
+ })
134
+
135
+ it('default statuses seed covers all 5 queue values', () => {
136
+ const queueEntries = Object.values(DEFAULT_ORGANIZATION_MODEL_STATUSES).filter((e) => e.semanticClass === 'queue')
137
+ expect(queueEntries).toHaveLength(5)
138
+ })
139
+
140
+ it('every entry has a non-empty id, label, and semanticClass', () => {
141
+ for (const entry of Object.values(DEFAULT_ORGANIZATION_MODEL_STATUSES)) {
142
+ expect(entry.id.length).toBeGreaterThan(0)
143
+ expect(entry.label.length).toBeGreaterThan(0)
144
+ expect(entry.semanticClass.length).toBeGreaterThan(0)
145
+ }
146
+ })
147
+
148
+ it('all entry ids are unique', () => {
149
+ const ids = Object.values(DEFAULT_ORGANIZATION_MODEL_STATUSES).map((e) => e.id)
150
+ const uniqueIds = new Set(ids)
151
+ expect(uniqueIds.size).toBe(ids.length)
152
+ })
153
+ })
154
+
155
+ describe('DEFAULT_ORGANIZATION_MODEL_KNOWLEDGE — flat Record (Phase 4 D3)', () => {
156
+ it('is an empty object (no pre-seeded nodes in generic base)', () => {
157
+ expect(DEFAULT_ORGANIZATION_MODEL_KNOWLEDGE).toEqual({})
158
+ })
159
+
160
+ it('does not have a .nodes array (old wrapper shape removed)', () => {
161
+ expect(DEFAULT_ORGANIZATION_MODEL_KNOWLEDGE).not.toHaveProperty('nodes')
162
+ })
163
+
164
+ it('does not have a .version field (moved to domainMetadata.knowledge)', () => {
165
+ expect(DEFAULT_ORGANIZATION_MODEL_KNOWLEDGE).not.toHaveProperty('version')
166
+ })
167
+ })
168
+ })
@@ -0,0 +1,78 @@
1
+ import { describe, expect, it } from 'vitest'
2
+ import {
3
+ ActionsDomainSchema,
4
+ ActionInvocationSchema,
5
+ CRM_ACTION_ENTRIES,
6
+ DEFAULT_ORGANIZATION_MODEL_ACTIONS,
7
+ LEAD_GEN_ACTION_ENTRIES
8
+ } from '../../domains/actions'
9
+
10
+ describe('actions domain', () => {
11
+ it('defaults lead-gen action entries with invocation contracts', () => {
12
+ const domain = ActionsDomainSchema.parse(LEAD_GEN_ACTION_ENTRIES)
13
+
14
+ expect(Object.keys(domain)).toHaveLength(13)
15
+ expect(domain['lead-gen.company.source']).toMatchObject({
16
+ id: 'lead-gen.company.source',
17
+ scope: { domain: 'sales' },
18
+ lifecycle: 'active'
19
+ })
20
+ expect(Object.values(domain).every((action) => action.invocations.length > 0)).toBe(true)
21
+ })
22
+
23
+ it('defaults CRM action entries for deal actions', () => {
24
+ const domain = ActionsDomainSchema.parse(CRM_ACTION_ENTRIES)
25
+
26
+ expect(Object.keys(domain)).toHaveLength(9)
27
+ expect(domain.mark_no_show).toBeUndefined()
28
+ expect(
29
+ Object.values(domain).every((action) => action.affects?.includes('crm.deal') && action.resourceId !== undefined)
30
+ ).toBe(true)
31
+ })
32
+
33
+ it('defaults all organization-model actions', () => {
34
+ const domain = ActionsDomainSchema.parse(DEFAULT_ORGANIZATION_MODEL_ACTIONS)
35
+
36
+ expect(Object.keys(domain)).toHaveLength(22)
37
+ })
38
+
39
+ it('accepts entity references in affects', () => {
40
+ expect(() =>
41
+ ActionsDomainSchema.parse({
42
+ 'crm.deal.update': {
43
+ id: 'crm.deal.update',
44
+ order: 10,
45
+ label: 'Update deal',
46
+ affects: ['crm.deal']
47
+ }
48
+ })
49
+ ).not.toThrow()
50
+ })
51
+
52
+ it('accepts every invocation kind', () => {
53
+ expect(() =>
54
+ ActionInvocationSchema.parse({
55
+ kind: 'slash-command',
56
+ command: '/lead-gen source',
57
+ toolFactory: 'lead-gen-source'
58
+ })
59
+ ).not.toThrow()
60
+ expect(() => ActionInvocationSchema.parse({ kind: 'mcp-tool', server: 'lead-gen', name: 'source' })).not.toThrow()
61
+ expect(() =>
62
+ ActionInvocationSchema.parse({
63
+ kind: 'api-endpoint',
64
+ method: 'POST',
65
+ path: '/api/prospecting/source',
66
+ requestSchema: 'source-request',
67
+ responseSchema: 'source-response'
68
+ })
69
+ ).not.toThrow()
70
+ expect(() =>
71
+ ActionInvocationSchema.parse({ kind: 'script-execution', resourceId: 'scripts.lead-gen-source' })
72
+ ).not.toThrow()
73
+ })
74
+
75
+ it('rejects malformed slash commands', () => {
76
+ expect(() => ActionInvocationSchema.parse({ kind: 'slash-command', command: 'lead-gen source' })).toThrow()
77
+ })
78
+ })