@elevasis/core 0.23.0 → 0.24.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (243) hide show
  1. package/dist/index.d.ts +4343 -2690
  2. package/dist/index.js +1101 -156
  3. package/dist/knowledge/index.d.ts +574 -210
  4. package/dist/knowledge/index.js +104 -1
  5. package/dist/organization-model/index.d.ts +4343 -2690
  6. package/dist/organization-model/index.js +1101 -156
  7. package/dist/test-utils/index.d.ts +483 -109
  8. package/dist/test-utils/index.js +904 -144
  9. package/package.json +3 -3
  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 -12
  13. package/src/_gen/__tests__/__snapshots__/contracts.md.snap +2137 -2093
  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 +1492 -1497
  37. package/src/business/acquisition/build-templates.test.ts +240 -240
  38. package/src/business/acquisition/build-templates.ts +98 -98
  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 -153
  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 -396
  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 +195 -197
  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 +32 -32
  123. package/src/knowledge/__tests__/queries.test.ts +626 -535
  124. package/src/knowledge/format.ts +99 -99
  125. package/src/knowledge/index.ts +5 -5
  126. package/src/knowledge/published.ts +5 -5
  127. package/src/knowledge/queries.ts +269 -218
  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 -149
  131. package/src/organization-model/__tests__/content-kinds-registry.test.ts +210 -210
  132. package/src/organization-model/__tests__/defaults.test.ts +168 -168
  133. package/src/organization-model/__tests__/domains/actions.test.ts +78 -56
  134. package/src/organization-model/__tests__/domains/customers.test.ts +299 -299
  135. package/src/organization-model/__tests__/domains/entities.test.ts +56 -56
  136. package/src/organization-model/__tests__/domains/goals.test.ts +493 -493
  137. package/src/organization-model/__tests__/domains/identity.test.ts +280 -280
  138. package/src/organization-model/__tests__/domains/navigation.test.ts +268 -268
  139. package/src/organization-model/__tests__/domains/offerings.test.ts +414 -414
  140. package/src/organization-model/__tests__/domains/policies.test.ts +323 -323
  141. package/src/organization-model/__tests__/domains/resource-mappings.test.ts +293 -293
  142. package/src/organization-model/__tests__/domains/resources.test.ts +387 -277
  143. package/src/organization-model/__tests__/domains/roles.test.ts +463 -463
  144. package/src/organization-model/__tests__/domains/statuses.test.ts +246 -246
  145. package/src/organization-model/__tests__/domains/systems.test.ts +209 -209
  146. package/src/organization-model/__tests__/domains/topology.test.ts +188 -0
  147. package/src/organization-model/__tests__/flatten-additive-merge.test.ts +362 -361
  148. package/src/organization-model/__tests__/foundation.test.ts +77 -77
  149. package/src/organization-model/__tests__/get-resources-for-system.test.ts +144 -144
  150. package/src/organization-model/__tests__/graph.test.ts +1312 -862
  151. package/src/organization-model/__tests__/icons.test.ts +10 -1
  152. package/src/organization-model/__tests__/knowledge.test.ts +251 -15
  153. package/src/organization-model/__tests__/lookup-helpers.test.ts +438 -438
  154. package/src/organization-model/__tests__/migration-helpers.test.ts +591 -591
  155. package/src/organization-model/__tests__/prospecting-ssot.test.ts +103 -103
  156. package/src/organization-model/__tests__/recursive-system-schema.test.ts +535 -506
  157. package/src/organization-model/__tests__/resolve.test.ts +274 -164
  158. package/src/organization-model/__tests__/schema.test.ts +844 -301
  159. package/src/organization-model/__tests__/surface-projection.test.ts +284 -284
  160. package/src/organization-model/catalogs/lead-gen.ts +144 -144
  161. package/src/organization-model/content-kinds/config.ts +36 -36
  162. package/src/organization-model/content-kinds/index.ts +76 -72
  163. package/src/organization-model/content-kinds/pipeline.ts +68 -68
  164. package/src/organization-model/content-kinds/registry.ts +44 -44
  165. package/src/organization-model/content-kinds/status.ts +71 -71
  166. package/src/organization-model/content-kinds/template.ts +83 -83
  167. package/src/organization-model/content-kinds/types.ts +117 -117
  168. package/src/organization-model/contracts.ts +27 -27
  169. package/src/organization-model/defaults.ts +42 -50
  170. package/src/organization-model/domains/actions.ts +333 -239
  171. package/src/organization-model/domains/customers.ts +78 -78
  172. package/src/organization-model/domains/entities.ts +144 -144
  173. package/src/organization-model/domains/goals.ts +83 -83
  174. package/src/organization-model/domains/knowledge.ts +117 -101
  175. package/src/organization-model/domains/navigation.ts +139 -139
  176. package/src/organization-model/domains/offerings.ts +71 -71
  177. package/src/organization-model/domains/policies.ts +102 -102
  178. package/src/organization-model/domains/projects.ts +14 -14
  179. package/src/organization-model/domains/prospecting.ts +395 -395
  180. package/src/organization-model/domains/resources.ts +202 -124
  181. package/src/organization-model/domains/roles.ts +96 -96
  182. package/src/organization-model/domains/sales.test.ts +218 -218
  183. package/src/organization-model/domains/sales.ts +380 -380
  184. package/src/organization-model/domains/shared.ts +63 -63
  185. package/src/organization-model/domains/statuses.ts +339 -339
  186. package/src/organization-model/domains/systems.ts +217 -172
  187. package/src/organization-model/domains/topology.ts +261 -0
  188. package/src/organization-model/foundation.ts +75 -75
  189. package/src/organization-model/graph/build.ts +1043 -867
  190. package/src/organization-model/graph/index.ts +4 -4
  191. package/src/organization-model/graph/link.ts +10 -10
  192. package/src/organization-model/graph/schema.ts +75 -68
  193. package/src/organization-model/graph/types.ts +71 -64
  194. package/src/organization-model/helpers.ts +289 -241
  195. package/src/organization-model/icons.ts +78 -66
  196. package/src/organization-model/index.ts +128 -125
  197. package/src/organization-model/migration-helpers.ts +247 -244
  198. package/src/organization-model/ontology.ts +658 -0
  199. package/src/organization-model/organization-graph.mdx +110 -90
  200. package/src/organization-model/organization-model.mdx +225 -213
  201. package/src/organization-model/published.ts +299 -222
  202. package/src/organization-model/resolve.ts +146 -91
  203. package/src/organization-model/schema.ts +818 -659
  204. package/src/organization-model/surface-projection.ts +212 -212
  205. package/src/organization-model/types.ts +179 -155
  206. package/src/platform/api/types.ts +38 -38
  207. package/src/platform/constants/versions.ts +3 -3
  208. package/src/platform/index.ts +23 -23
  209. package/src/platform/registry/__tests__/command-view.test.ts +10 -10
  210. package/src/platform/registry/__tests__/resource-link.test.ts +35 -35
  211. package/src/platform/registry/__tests__/resource-registry.integration.test.ts +20 -20
  212. package/src/platform/registry/__tests__/resource-registry.nested-systems.test.ts +245 -245
  213. package/src/platform/registry/__tests__/resource-registry.test.ts +2053 -2053
  214. package/src/platform/registry/__tests__/validation.test.ts +1444 -1259
  215. package/src/platform/registry/command-view.ts +10 -10
  216. package/src/platform/registry/index.ts +103 -103
  217. package/src/platform/registry/resource-link.ts +32 -32
  218. package/src/platform/registry/resource-registry.ts +886 -886
  219. package/src/platform/registry/serialization.ts +295 -295
  220. package/src/platform/registry/serialized-types.ts +166 -166
  221. package/src/platform/registry/stats-types.ts +68 -68
  222. package/src/platform/registry/types.ts +425 -425
  223. package/src/platform/registry/validation.ts +876 -684
  224. package/src/platform/utils/__tests__/validation.test.ts +1084 -1084
  225. package/src/platform/utils/validation.ts +425 -425
  226. package/src/projects/api-schemas.test.ts +39 -39
  227. package/src/projects/api-schemas.ts +291 -291
  228. package/src/reference/_generated/contracts.md +2136 -2093
  229. package/src/reference/glossary.md +76 -76
  230. package/src/scaffold-registry/__tests__/index.test.ts +206 -206
  231. package/src/scaffold-registry/__tests__/schema.test.ts +166 -166
  232. package/src/scaffold-registry/index.ts +392 -392
  233. package/src/scaffold-registry/schema.ts +243 -243
  234. package/src/server.ts +289 -289
  235. package/src/supabase/database.types.ts +3 -0
  236. package/src/test-utils/README.md +37 -37
  237. package/src/test-utils/entities.ts +108 -108
  238. package/src/test-utils/fixtures/memberships.ts +82 -82
  239. package/src/test-utils/index.ts +12 -12
  240. package/src/test-utils/organization-model.ts +65 -65
  241. package/src/test-utils/published.ts +6 -6
  242. package/src/test-utils/rls/RLSTestContext.ts +588 -588
  243. package/src/test-utils/test-utils.test.ts +44 -44
@@ -1,361 +1,362 @@
1
- import { describe, expect, it } from 'vitest'
2
- import { DEFAULT_ORGANIZATION_MODEL } from '../defaults'
3
- import { listDomain } from '../helpers'
4
- import { resolveOrganizationModel } from '../resolve'
5
- import type { DeepPartial, OrganizationModel } from '../types'
6
-
7
- // ---------------------------------------------------------------------------
8
- // Step 5 of om-flatten-single-array-domains: lock in additive-by-id merge
9
- // across all 9 flattened domains.
10
- //
11
- // Before the flatten, every domain was `{ field: Entry[] }` and deepMerge's
12
- // array branch (`Array.isArray(base) -> return override`) caused tenant
13
- // overrides to wholesale-replace the array. The systems domain needed a
14
- // special-case mergeSystemsById helper (sidebar-regression-fix.mdx OD1) to
15
- // produce additive behavior.
16
- //
17
- // After the flatten, every domain is `Record<string, Entry>` keyed by id, so
18
- // deepMerge's object branch (`isPlainObject(base) && isPlainObject(override)
19
- // -> deepMerge per key`) gives additive-by-id semantics for free.
20
- //
21
- // These tests pin that semantic in place. If anyone re-introduces array shape
22
- // on these domains, or special-cases merge logic, these tests will surface it.
23
- // ---------------------------------------------------------------------------
24
-
25
- describe('flatten: deepMerge produces additive-by-id behavior across all 9 flattened domains', () => {
26
- // Phase 4 (D1): 'statuses' top-level field removed from OrganizationModel.
27
- // Status data moved into system.content as (schema:status-flow) / (schema:status) nodes.
28
- // The remaining 8 domains are all Record<string, Entry> after the flatten.
29
- const FLATTENED_DOMAINS = [
30
- 'systems',
31
- 'roles',
32
- 'goals',
33
- 'customers',
34
- 'actions',
35
- 'offerings',
36
- 'resources',
37
- 'policies'
38
- ] as const
39
-
40
- it.each(FLATTENED_DOMAINS)('%s domain shape is Record<string, Entry>, not an array wrapper', (domain) => {
41
- const value = DEFAULT_ORGANIZATION_MODEL[domain]
42
- expect(Array.isArray(value)).toBe(false)
43
- expect(typeof value).toBe('object')
44
- expect(value).not.toBeNull()
45
- })
46
-
47
- it('adds a tenant system to the merged record without dropping defaults', () => {
48
- const baseSize = Object.keys(DEFAULT_ORGANIZATION_MODEL.systems).length
49
-
50
- const model = resolveOrganizationModel({
51
- systems: {
52
- 'tenant.workspace': {
53
- id: 'tenant.workspace',
54
- order: 10000,
55
- label: 'Tenant Workspace',
56
- lifecycle: 'active',
57
- path: '/tenant'
58
- }
59
- }
60
- })
61
-
62
- expect(Object.keys(model.systems)).toHaveLength(baseSize + 1)
63
- expect(model.systems['tenant.workspace']?.label).toBe('Tenant Workspace')
64
- // Default systems survive
65
- expect(model.systems['dashboard']).toBeDefined()
66
- expect(model.systems['operations']).toBeDefined()
67
- })
68
-
69
- it('merges a tenant role additively without clobbering default roles', () => {
70
- const baseSize = Object.keys(DEFAULT_ORGANIZATION_MODEL.roles).length
71
-
72
- const model = resolveOrganizationModel({
73
- roles: {
74
- 'role.tenant-lead': {
75
- id: 'role.tenant-lead',
76
- order: 10000,
77
- title: 'Tenant Lead'
78
- }
79
- }
80
- })
81
-
82
- expect(Object.keys(model.roles)).toHaveLength(baseSize + 1)
83
- expect(model.roles['role.tenant-lead']?.title).toBe('Tenant Lead')
84
- })
85
-
86
- it('merges a tenant action additively while preserving the default 13 lead-gen actions', () => {
87
- const baseSize = Object.keys(DEFAULT_ORGANIZATION_MODEL.actions).length
88
- expect(baseSize).toBe(13)
89
-
90
- const model = resolveOrganizationModel({
91
- actions: {
92
- 'tenant.custom-action': {
93
- id: 'tenant.custom-action',
94
- order: 10000,
95
- label: 'Custom Action',
96
- scope: 'global',
97
- resourceId: 'tenant-workflow'
98
- }
99
- }
100
- })
101
-
102
- expect(Object.keys(model.actions)).toHaveLength(baseSize + 1)
103
- expect(model.actions['tenant.custom-action']?.label).toBe('Custom Action')
104
- expect(model.actions['lead-gen.company.source']).toBeDefined()
105
- })
106
-
107
- it('overrides an existing system by id (last write wins on the same map key)', () => {
108
- const model = resolveOrganizationModel({
109
- systems: {
110
- dashboard: {
111
- id: 'dashboard',
112
- order: 10,
113
- label: 'Tenant Dashboard',
114
- lifecycle: 'active',
115
- path: '/'
116
- }
117
- }
118
- })
119
-
120
- expect(model.systems['dashboard']?.label).toBe('Tenant Dashboard')
121
- // Other defaults remain
122
- expect(model.systems['operations']).toBeDefined()
123
- })
124
-
125
- it('merges nested fields within a single entry deeply (not replacing the whole entry)', () => {
126
- // Overriding only `label` on a default system preserves its other fields.
127
- const defaultDashboard = DEFAULT_ORGANIZATION_MODEL.systems['dashboard']
128
- expect(defaultDashboard).toBeDefined()
129
-
130
- const model = resolveOrganizationModel({
131
- systems: {
132
- dashboard: {
133
- id: 'dashboard',
134
- order: defaultDashboard!.order,
135
- label: 'Renamed'
136
- }
137
- }
138
- })
139
-
140
- const merged = model.systems['dashboard']
141
- expect(merged?.label).toBe('Renamed')
142
- // Path from defaults survives because object merge is deep per-key.
143
- expect(merged?.path).toBe(defaultDashboard!.path)
144
- })
145
-
146
- it('confirms no special-case mergeSystemsById helper exists in resolve.ts', async () => {
147
- // The OD1 special case from sidebar-regression-fix.mdx is superseded by
148
- // this flatten. If anyone reintroduces it under any of these names,
149
- // surface that drift.
150
- const source = await import('node:fs/promises').then((fs) =>
151
- fs.readFile(new URL('../resolve.ts', import.meta.url), 'utf-8')
152
- )
153
-
154
- expect(source).not.toMatch(/mergeSystemsById/)
155
- expect(source).not.toMatch(/mergeById/)
156
- expect(source).not.toMatch(/mergeRecordById/)
157
- })
158
-
159
- it('multiple flattened domains receive independent additive merges in one call', () => {
160
- const baseSystems = Object.keys(DEFAULT_ORGANIZATION_MODEL.systems).length
161
- const baseRoles = Object.keys(DEFAULT_ORGANIZATION_MODEL.roles).length
162
- const baseActions = Object.keys(DEFAULT_ORGANIZATION_MODEL.actions).length
163
-
164
- const model = resolveOrganizationModel({
165
- systems: {
166
- 'tenant.a': { id: 'tenant.a', order: 10000, label: 'A', lifecycle: 'active', path: '/a' }
167
- },
168
- roles: {
169
- 'role.a': { id: 'role.a', order: 10000, title: 'A' }
170
- },
171
- actions: {
172
- 'tenant.a': {
173
- id: 'tenant.a',
174
- order: 10000,
175
- label: 'A Action',
176
- scope: 'global',
177
- resourceId: 'a-workflow'
178
- }
179
- }
180
- })
181
-
182
- expect(Object.keys(model.systems)).toHaveLength(baseSystems + 1)
183
- expect(Object.keys(model.roles)).toHaveLength(baseRoles + 1)
184
- expect(Object.keys(model.actions)).toHaveLength(baseActions + 1)
185
- })
186
-
187
- // ---------------------------------------------------------------------------
188
- // Override merge edge cases
189
- // ---------------------------------------------------------------------------
190
-
191
- it('order override on existing entry takes precedence and re-sorts the listed domain', () => {
192
- const override: DeepPartial<OrganizationModel> = {
193
- systems: {
194
- dashboard: {
195
- id: 'dashboard',
196
- order: 1000,
197
- label: 'Dashboard',
198
- lifecycle: 'active',
199
- path: '/'
200
- }
201
- }
202
- }
203
-
204
- const resolved = resolveOrganizationModel(override)
205
-
206
- expect(resolved.systems['dashboard']!.order).toBe(1000)
207
-
208
- const listed = listDomain(resolved.systems)
209
- const dashboardIndex = listed.findIndex((s) => s.id === 'dashboard')
210
- const systemsBefore = listed.filter((s) => s.order < 1000)
211
- expect(dashboardIndex).toBeGreaterThan(systemsBefore.length - 1)
212
- // All entries with order < 1000 must appear before dashboard in the sorted list
213
- systemsBefore.forEach((s) => {
214
- expect(listed.indexOf(s)).toBeLessThan(dashboardIndex)
215
- })
216
- })
217
-
218
- it('override entry omitting order preserves the base order value', () => {
219
- const baseOrder = DEFAULT_ORGANIZATION_MODEL.systems['dashboard']!.order
220
-
221
- const override: DeepPartial<OrganizationModel> = {
222
- systems: {
223
- dashboard: {
224
- id: 'dashboard',
225
- label: 'Renamed Dashboard'
226
- }
227
- }
228
- }
229
-
230
- const resolved = resolveOrganizationModel(override)
231
-
232
- expect(resolved.systems['dashboard']!.order).toBe(baseOrder)
233
- expect(resolved.systems['dashboard']!.label).toBe('Renamed Dashboard')
234
- })
235
-
236
- it('throws when system map key does not match the entry id', () => {
237
- const badOverride: DeepPartial<OrganizationModel> = {
238
- systems: {
239
- 'wrong-key': {
240
- id: 'different-id',
241
- order: 10,
242
- label: 'Mismatch',
243
- lifecycle: 'active',
244
- path: '/mismatch'
245
- }
246
- }
247
- }
248
-
249
- expect(() => resolveOrganizationModel(badOverride)).toThrow(/map key/)
250
- })
251
-
252
- it('replaces nested arrays rather than merging them', () => {
253
- // sales.lead-gen has a top-level `actions` array field on the entry.
254
- // deepMerge hits Array.isArray(existing) => returns override value directly.
255
- const singleRef = [{ actionId: 'lead-gen.company.source', intent: 'exposes' as const }]
256
-
257
- const override: DeepPartial<OrganizationModel> = {
258
- systems: {
259
- 'sales.lead-gen': {
260
- id: 'sales.lead-gen',
261
- order: DEFAULT_ORGANIZATION_MODEL.systems['sales.lead-gen']!.order,
262
- label: 'Lead Gen',
263
- actions: singleRef
264
- }
265
- }
266
- }
267
-
268
- const resolved = resolveOrganizationModel(override)
269
- const entry = resolved.systems['sales.lead-gen']!
270
-
271
- expect(entry.actions).toHaveLength(1)
272
- expect(entry.actions).toEqual(singleRef)
273
- })
274
-
275
- it('empty systems override produces same systems domain as no override', () => {
276
- const withEmpty = resolveOrganizationModel({ systems: {} })
277
- const withNone = resolveOrganizationModel()
278
-
279
- expect(withEmpty.systems).toEqual(withNone.systems)
280
- })
281
-
282
- it('field set to undefined in override is skipped and retains the base value', () => {
283
- const basePath = DEFAULT_ORGANIZATION_MODEL.systems['dashboard']!.path
284
-
285
- const override: DeepPartial<OrganizationModel> = {
286
- systems: {
287
- dashboard: {
288
- id: 'dashboard',
289
- order: DEFAULT_ORGANIZATION_MODEL.systems['dashboard']!.order,
290
- label: 'Dashboard',
291
- path: undefined
292
- }
293
- }
294
- }
295
-
296
- const resolved = resolveOrganizationModel(override)
297
-
298
- expect(resolved.systems['dashboard']!.path).toBe(basePath)
299
- })
300
-
301
- it('simultaneous override of systems, roles, and actions merges each domain independently without cross-contamination', () => {
302
- const baseSystems = Object.keys(DEFAULT_ORGANIZATION_MODEL.systems).length
303
- const baseRoles = Object.keys(DEFAULT_ORGANIZATION_MODEL.roles).length
304
- const baseActions = Object.keys(DEFAULT_ORGANIZATION_MODEL.actions).length
305
- const baseCustomers = DEFAULT_ORGANIZATION_MODEL.customers
306
- const baseGoals = DEFAULT_ORGANIZATION_MODEL.goals
307
-
308
- const model = resolveOrganizationModel({
309
- systems: {
310
- 'multi.sys': { id: 'multi.sys', order: 50000, label: 'Multi Sys', lifecycle: 'active', path: '/multi-sys' }
311
- },
312
- roles: {
313
- 'multi.role': { id: 'multi.role', order: 50000, title: 'Multi Role' }
314
- },
315
- actions: {
316
- 'multi.action': {
317
- id: 'multi.action',
318
- order: 50000,
319
- label: 'Multi Action',
320
- scope: 'global',
321
- resourceId: 'multi-workflow'
322
- }
323
- }
324
- })
325
-
326
- expect(Object.keys(model.systems)).toHaveLength(baseSystems + 1)
327
- expect(model.systems['multi.sys']).toBeDefined()
328
-
329
- expect(Object.keys(model.roles)).toHaveLength(baseRoles + 1)
330
- expect(model.roles['multi.role']).toBeDefined()
331
-
332
- expect(Object.keys(model.actions)).toHaveLength(baseActions + 1)
333
- expect(model.actions['multi.action']).toBeDefined()
334
-
335
- expect(model.customers).toEqual(baseCustomers)
336
- expect(model.goals).toEqual(baseGoals)
337
- })
338
-
339
- it('override sets top-level array field on entry to single-element array replacing the original', () => {
340
- // The `invocations` field on an action entry is a top-level array.
341
- // deepMerge sees Array.isArray(existing) and returns override directly.
342
- const replacementInvocations = [{ kind: 'slash-command' as const, command: '/custom-source' }]
343
-
344
- const override: DeepPartial<OrganizationModel> = {
345
- actions: {
346
- 'lead-gen.company.source': {
347
- id: 'lead-gen.company.source',
348
- order: DEFAULT_ORGANIZATION_MODEL.actions['lead-gen.company.source']!.order,
349
- label: 'Source companies',
350
- invocations: replacementInvocations
351
- }
352
- }
353
- }
354
-
355
- const resolved = resolveOrganizationModel(override)
356
- const entry = resolved.actions['lead-gen.company.source']!
357
-
358
- expect(entry.invocations).toHaveLength(1)
359
- expect(entry.invocations).toEqual(replacementInvocations)
360
- })
361
- })
1
+ import { describe, expect, it } from 'vitest'
2
+ import { DEFAULT_ORGANIZATION_MODEL } from '../defaults'
3
+ import { listDomain } from '../helpers'
4
+ import { resolveOrganizationModel } from '../resolve'
5
+ import type { DeepPartial, OrganizationModel } from '../types'
6
+
7
+ // ---------------------------------------------------------------------------
8
+ // Step 5 of om-flatten-single-array-domains: lock in additive-by-id merge
9
+ // across all 9 flattened domains.
10
+ //
11
+ // Before the flatten, every domain was `{ field: Entry[] }` and deepMerge's
12
+ // array branch (`Array.isArray(base) -> return override`) caused tenant
13
+ // overrides to wholesale-replace the array. The systems domain needed a
14
+ // special-case mergeSystemsById helper (sidebar-regression-fix.mdx OD1) to
15
+ // produce additive behavior.
16
+ //
17
+ // After the flatten, every domain is `Record<string, Entry>` keyed by id, so
18
+ // deepMerge's object branch (`isPlainObject(base) && isPlainObject(override)
19
+ // -> deepMerge per key`) gives additive-by-id semantics for free.
20
+ //
21
+ // These tests pin that semantic in place. If anyone re-introduces array shape
22
+ // on these domains, or special-cases merge logic, these tests will surface it.
23
+ // ---------------------------------------------------------------------------
24
+
25
+ describe('flatten: deepMerge produces additive-by-id behavior across all 9 flattened domains', () => {
26
+ // Phase 4 (D1): 'statuses' top-level field removed from OrganizationModel.
27
+ // Status data moved into system.content as (schema:status-flow) / (schema:status) nodes.
28
+ // The remaining 8 domains are all Record<string, Entry> after the flatten.
29
+ const FLATTENED_DOMAINS = [
30
+ 'systems',
31
+ 'roles',
32
+ 'goals',
33
+ 'customers',
34
+ 'actions',
35
+ 'offerings',
36
+ 'resources',
37
+ 'policies'
38
+ ] as const
39
+
40
+ it.each(FLATTENED_DOMAINS)('%s domain shape is Record<string, Entry>, not an array wrapper', (domain) => {
41
+ const value = DEFAULT_ORGANIZATION_MODEL[domain]
42
+ expect(Array.isArray(value)).toBe(false)
43
+ expect(typeof value).toBe('object')
44
+ expect(value).not.toBeNull()
45
+ })
46
+
47
+ it('adds a tenant system to the merged record without dropping defaults', () => {
48
+ const baseSize = Object.keys(DEFAULT_ORGANIZATION_MODEL.systems).length
49
+
50
+ const model = resolveOrganizationModel({
51
+ systems: {
52
+ 'tenant.workspace': {
53
+ id: 'tenant.workspace',
54
+ order: 10000,
55
+ label: 'Tenant Workspace',
56
+ lifecycle: 'active',
57
+ path: '/tenant'
58
+ }
59
+ }
60
+ })
61
+
62
+ expect(Object.keys(model.systems)).toHaveLength(baseSize + 1)
63
+ expect(model.systems['tenant.workspace']?.label).toBe('Tenant Workspace')
64
+ // Default systems survive
65
+ expect(model.systems['dashboard']).toBeDefined()
66
+ expect(model.systems['operations']).toBeDefined()
67
+ })
68
+
69
+ it('merges a tenant role additively without clobbering default roles', () => {
70
+ const baseSize = Object.keys(DEFAULT_ORGANIZATION_MODEL.roles).length
71
+
72
+ const model = resolveOrganizationModel({
73
+ roles: {
74
+ 'role.tenant-lead': {
75
+ id: 'role.tenant-lead',
76
+ order: 10000,
77
+ title: 'Tenant Lead'
78
+ }
79
+ }
80
+ })
81
+
82
+ expect(Object.keys(model.roles)).toHaveLength(baseSize + 1)
83
+ expect(model.roles['role.tenant-lead']?.title).toBe('Tenant Lead')
84
+ })
85
+
86
+ it('merges a tenant action additively while preserving the default action catalog', () => {
87
+ const baseSize = Object.keys(DEFAULT_ORGANIZATION_MODEL.actions).length
88
+ expect(baseSize).toBeGreaterThan(0)
89
+
90
+ const model = resolveOrganizationModel({
91
+ actions: {
92
+ 'tenant.custom-action': {
93
+ id: 'tenant.custom-action',
94
+ order: 10000,
95
+ label: 'Custom Action',
96
+ scope: 'global',
97
+ resourceId: 'tenant-workflow'
98
+ }
99
+ }
100
+ })
101
+
102
+ expect(Object.keys(model.actions)).toHaveLength(baseSize + 1)
103
+ expect(model.actions['tenant.custom-action']?.label).toBe('Custom Action')
104
+ expect(model.actions['lead-gen.company.source']).toBeDefined()
105
+ expect(model.actions['send_reply']).toBeDefined()
106
+ })
107
+
108
+ it('overrides an existing system by id (last write wins on the same map key)', () => {
109
+ const model = resolveOrganizationModel({
110
+ systems: {
111
+ dashboard: {
112
+ id: 'dashboard',
113
+ order: 10,
114
+ label: 'Tenant Dashboard',
115
+ lifecycle: 'active',
116
+ path: '/'
117
+ }
118
+ }
119
+ })
120
+
121
+ expect(model.systems['dashboard']?.label).toBe('Tenant Dashboard')
122
+ // Other defaults remain
123
+ expect(model.systems['operations']).toBeDefined()
124
+ })
125
+
126
+ it('merges nested fields within a single entry deeply (not replacing the whole entry)', () => {
127
+ // Overriding only `label` on a default system preserves its other fields.
128
+ const defaultDashboard = DEFAULT_ORGANIZATION_MODEL.systems['dashboard']
129
+ expect(defaultDashboard).toBeDefined()
130
+
131
+ const model = resolveOrganizationModel({
132
+ systems: {
133
+ dashboard: {
134
+ id: 'dashboard',
135
+ order: defaultDashboard!.order,
136
+ label: 'Renamed'
137
+ }
138
+ }
139
+ })
140
+
141
+ const merged = model.systems['dashboard']
142
+ expect(merged?.label).toBe('Renamed')
143
+ // Path from defaults survives because object merge is deep per-key.
144
+ expect(merged?.path).toBe(defaultDashboard!.path)
145
+ })
146
+
147
+ it('confirms no special-case mergeSystemsById helper exists in resolve.ts', async () => {
148
+ // The OD1 special case from sidebar-regression-fix.mdx is superseded by
149
+ // this flatten. If anyone reintroduces it under any of these names,
150
+ // surface that drift.
151
+ const source = await import('node:fs/promises').then((fs) =>
152
+ fs.readFile(new URL('../resolve.ts', import.meta.url), 'utf-8')
153
+ )
154
+
155
+ expect(source).not.toMatch(/mergeSystemsById/)
156
+ expect(source).not.toMatch(/mergeById/)
157
+ expect(source).not.toMatch(/mergeRecordById/)
158
+ })
159
+
160
+ it('multiple flattened domains receive independent additive merges in one call', () => {
161
+ const baseSystems = Object.keys(DEFAULT_ORGANIZATION_MODEL.systems).length
162
+ const baseRoles = Object.keys(DEFAULT_ORGANIZATION_MODEL.roles).length
163
+ const baseActions = Object.keys(DEFAULT_ORGANIZATION_MODEL.actions).length
164
+
165
+ const model = resolveOrganizationModel({
166
+ systems: {
167
+ 'tenant.a': { id: 'tenant.a', order: 10000, label: 'A', lifecycle: 'active', path: '/a' }
168
+ },
169
+ roles: {
170
+ 'role.a': { id: 'role.a', order: 10000, title: 'A' }
171
+ },
172
+ actions: {
173
+ 'tenant.a': {
174
+ id: 'tenant.a',
175
+ order: 10000,
176
+ label: 'A Action',
177
+ scope: 'global',
178
+ resourceId: 'a-workflow'
179
+ }
180
+ }
181
+ })
182
+
183
+ expect(Object.keys(model.systems)).toHaveLength(baseSystems + 1)
184
+ expect(Object.keys(model.roles)).toHaveLength(baseRoles + 1)
185
+ expect(Object.keys(model.actions)).toHaveLength(baseActions + 1)
186
+ })
187
+
188
+ // ---------------------------------------------------------------------------
189
+ // Override merge edge cases
190
+ // ---------------------------------------------------------------------------
191
+
192
+ it('order override on existing entry takes precedence and re-sorts the listed domain', () => {
193
+ const override: DeepPartial<OrganizationModel> = {
194
+ systems: {
195
+ dashboard: {
196
+ id: 'dashboard',
197
+ order: 1000,
198
+ label: 'Dashboard',
199
+ lifecycle: 'active',
200
+ path: '/'
201
+ }
202
+ }
203
+ }
204
+
205
+ const resolved = resolveOrganizationModel(override)
206
+
207
+ expect(resolved.systems['dashboard']!.order).toBe(1000)
208
+
209
+ const listed = listDomain(resolved.systems)
210
+ const dashboardIndex = listed.findIndex((s) => s.id === 'dashboard')
211
+ const systemsBefore = listed.filter((s) => s.order < 1000)
212
+ expect(dashboardIndex).toBeGreaterThan(systemsBefore.length - 1)
213
+ // All entries with order < 1000 must appear before dashboard in the sorted list
214
+ systemsBefore.forEach((s) => {
215
+ expect(listed.indexOf(s)).toBeLessThan(dashboardIndex)
216
+ })
217
+ })
218
+
219
+ it('override entry omitting order preserves the base order value', () => {
220
+ const baseOrder = DEFAULT_ORGANIZATION_MODEL.systems['dashboard']!.order
221
+
222
+ const override: DeepPartial<OrganizationModel> = {
223
+ systems: {
224
+ dashboard: {
225
+ id: 'dashboard',
226
+ label: 'Renamed Dashboard'
227
+ }
228
+ }
229
+ }
230
+
231
+ const resolved = resolveOrganizationModel(override)
232
+
233
+ expect(resolved.systems['dashboard']!.order).toBe(baseOrder)
234
+ expect(resolved.systems['dashboard']!.label).toBe('Renamed Dashboard')
235
+ })
236
+
237
+ it('throws when system map key does not match the entry id', () => {
238
+ const badOverride: DeepPartial<OrganizationModel> = {
239
+ systems: {
240
+ 'wrong-key': {
241
+ id: 'different-id',
242
+ order: 10,
243
+ label: 'Mismatch',
244
+ lifecycle: 'active',
245
+ path: '/mismatch'
246
+ }
247
+ }
248
+ }
249
+
250
+ expect(() => resolveOrganizationModel(badOverride)).toThrow(/map key/)
251
+ })
252
+
253
+ it('replaces nested arrays rather than merging them', () => {
254
+ // sales.lead-gen has a top-level `actions` array field on the entry.
255
+ // deepMerge hits Array.isArray(existing) => returns override value directly.
256
+ const singleRef = [{ actionId: 'lead-gen.company.source', intent: 'exposes' as const }]
257
+
258
+ const override: DeepPartial<OrganizationModel> = {
259
+ systems: {
260
+ 'sales.lead-gen': {
261
+ id: 'sales.lead-gen',
262
+ order: DEFAULT_ORGANIZATION_MODEL.systems['sales.lead-gen']!.order,
263
+ label: 'Lead Gen',
264
+ actions: singleRef
265
+ }
266
+ }
267
+ }
268
+
269
+ const resolved = resolveOrganizationModel(override)
270
+ const entry = resolved.systems['sales.lead-gen']!
271
+
272
+ expect(entry.actions).toHaveLength(1)
273
+ expect(entry.actions).toEqual(singleRef)
274
+ })
275
+
276
+ it('empty systems override produces same systems domain as no override', () => {
277
+ const withEmpty = resolveOrganizationModel({ systems: {} })
278
+ const withNone = resolveOrganizationModel()
279
+
280
+ expect(withEmpty.systems).toEqual(withNone.systems)
281
+ })
282
+
283
+ it('field set to undefined in override is skipped and retains the base value', () => {
284
+ const basePath = DEFAULT_ORGANIZATION_MODEL.systems['dashboard']!.path
285
+
286
+ const override: DeepPartial<OrganizationModel> = {
287
+ systems: {
288
+ dashboard: {
289
+ id: 'dashboard',
290
+ order: DEFAULT_ORGANIZATION_MODEL.systems['dashboard']!.order,
291
+ label: 'Dashboard',
292
+ path: undefined
293
+ }
294
+ }
295
+ }
296
+
297
+ const resolved = resolveOrganizationModel(override)
298
+
299
+ expect(resolved.systems['dashboard']!.path).toBe(basePath)
300
+ })
301
+
302
+ it('simultaneous override of systems, roles, and actions merges each domain independently without cross-contamination', () => {
303
+ const baseSystems = Object.keys(DEFAULT_ORGANIZATION_MODEL.systems).length
304
+ const baseRoles = Object.keys(DEFAULT_ORGANIZATION_MODEL.roles).length
305
+ const baseActions = Object.keys(DEFAULT_ORGANIZATION_MODEL.actions).length
306
+ const baseCustomers = DEFAULT_ORGANIZATION_MODEL.customers
307
+ const baseGoals = DEFAULT_ORGANIZATION_MODEL.goals
308
+
309
+ const model = resolveOrganizationModel({
310
+ systems: {
311
+ 'multi.sys': { id: 'multi.sys', order: 50000, label: 'Multi Sys', lifecycle: 'active', path: '/multi-sys' }
312
+ },
313
+ roles: {
314
+ 'multi.role': { id: 'multi.role', order: 50000, title: 'Multi Role' }
315
+ },
316
+ actions: {
317
+ 'multi.action': {
318
+ id: 'multi.action',
319
+ order: 50000,
320
+ label: 'Multi Action',
321
+ scope: 'global',
322
+ resourceId: 'multi-workflow'
323
+ }
324
+ }
325
+ })
326
+
327
+ expect(Object.keys(model.systems)).toHaveLength(baseSystems + 1)
328
+ expect(model.systems['multi.sys']).toBeDefined()
329
+
330
+ expect(Object.keys(model.roles)).toHaveLength(baseRoles + 1)
331
+ expect(model.roles['multi.role']).toBeDefined()
332
+
333
+ expect(Object.keys(model.actions)).toHaveLength(baseActions + 1)
334
+ expect(model.actions['multi.action']).toBeDefined()
335
+
336
+ expect(model.customers).toEqual(baseCustomers)
337
+ expect(model.goals).toEqual(baseGoals)
338
+ })
339
+
340
+ it('override sets top-level array field on entry to single-element array replacing the original', () => {
341
+ // The `invocations` field on an action entry is a top-level array.
342
+ // deepMerge sees Array.isArray(existing) and returns override directly.
343
+ const replacementInvocations = [{ kind: 'slash-command' as const, command: '/custom-source' }]
344
+
345
+ const override: DeepPartial<OrganizationModel> = {
346
+ actions: {
347
+ 'lead-gen.company.source': {
348
+ id: 'lead-gen.company.source',
349
+ order: DEFAULT_ORGANIZATION_MODEL.actions['lead-gen.company.source']!.order,
350
+ label: 'Source companies',
351
+ invocations: replacementInvocations
352
+ }
353
+ }
354
+ }
355
+
356
+ const resolved = resolveOrganizationModel(override)
357
+ const entry = resolved.actions['lead-gen.company.source']!
358
+
359
+ expect(entry.invocations).toHaveLength(1)
360
+ expect(entry.invocations).toEqual(replacementInvocations)
361
+ })
362
+ })