@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.
- package/dist/index.d.ts +3214 -2501
- package/dist/index.js +3112 -1222
- package/dist/knowledge/index.d.ts +1108 -1264
- package/dist/knowledge/index.js +112 -9
- package/dist/organization-model/index.d.ts +3214 -2501
- package/dist/organization-model/index.js +3112 -1222
- package/dist/test-utils/index.d.ts +985 -1103
- package/dist/test-utils/index.js +2464 -1165
- package/package.json +5 -5
- package/src/README.md +14 -14
- package/src/__tests__/publish.test.ts +24 -24
- package/src/__tests__/template-core-compatibility.test.ts +9 -80
- package/src/_gen/__tests__/__snapshots__/contracts.md.snap +2389 -2121
- package/src/_gen/__tests__/scaffold-contracts.test.ts +30 -30
- package/src/auth/multi-tenancy/credentials/__tests__/encryption.test.ts +217 -217
- package/src/auth/multi-tenancy/credentials/server/encryption.ts +69 -69
- package/src/auth/multi-tenancy/credentials/server/kek-loader.ts +37 -37
- package/src/auth/multi-tenancy/index.ts +26 -26
- package/src/auth/multi-tenancy/invitations/api-schemas.ts +104 -104
- package/src/auth/multi-tenancy/memberships/api-schemas.ts +143 -143
- package/src/auth/multi-tenancy/memberships/index.ts +26 -26
- package/src/auth/multi-tenancy/memberships/membership.ts +130 -130
- package/src/auth/multi-tenancy/organizations/__tests__/api-schemas.test.ts +194 -194
- package/src/auth/multi-tenancy/organizations/api-schemas.ts +136 -136
- package/src/auth/multi-tenancy/permissions.test.ts +42 -42
- package/src/auth/multi-tenancy/permissions.ts +123 -123
- package/src/auth/multi-tenancy/role-management/api-schemas.ts +78 -78
- package/src/auth/multi-tenancy/role-management/index.ts +16 -16
- package/src/auth/multi-tenancy/theme-presets.ts +45 -45
- package/src/auth/multi-tenancy/types.ts +57 -57
- package/src/auth/multi-tenancy/users/api-schemas.ts +165 -165
- package/src/business/README.md +2 -2
- package/src/business/acquisition/activity-events.test.ts +250 -250
- package/src/business/acquisition/activity-events.ts +93 -93
- package/src/business/acquisition/api-schemas.test.ts +1883 -1843
- package/src/business/acquisition/api-schemas.ts +1493 -1500
- package/src/business/acquisition/build-templates.test.ts +240 -240
- package/src/business/acquisition/build-templates.ts +83 -41
- package/src/business/acquisition/crm-next-action.test.ts +262 -262
- package/src/business/acquisition/crm-next-action.ts +220 -220
- package/src/business/acquisition/crm-priority.test.ts +216 -216
- package/src/business/acquisition/crm-priority.ts +349 -349
- package/src/business/acquisition/crm-state-actions.test.ts +153 -151
- package/src/business/acquisition/deal-ownership.test.ts +351 -351
- package/src/business/acquisition/deal-ownership.ts +120 -120
- package/src/business/acquisition/derive-actions.test.ts +129 -104
- package/src/business/acquisition/derive-actions.ts +74 -84
- package/src/business/acquisition/index.ts +171 -170
- package/src/business/acquisition/ontology-validation.ts +309 -0
- package/src/business/acquisition/stateful.ts +30 -30
- package/src/business/acquisition/types.ts +396 -392
- package/src/business/clients/api-schemas.test.ts +115 -115
- package/src/business/clients/api-schemas.ts +158 -158
- package/src/business/clients/index.ts +1 -1
- package/src/business/crm/api-schemas.ts +40 -40
- package/src/business/crm/index.ts +1 -1
- package/src/business/deals/api-schemas.ts +87 -87
- package/src/business/deals/index.ts +1 -1
- package/src/business/index.ts +5 -5
- package/src/business/projects/types.ts +144 -144
- package/src/commands/queue/types/task.ts +15 -15
- package/src/execution/core/runner-types.ts +61 -61
- package/src/execution/core/sse-executions.ts +7 -7
- package/src/execution/engine/__tests__/fixtures/test-agents.ts +10 -10
- package/src/execution/engine/agent/core/__tests__/agent.test.ts +16 -16
- package/src/execution/engine/agent/core/__tests__/error-passthrough.test.ts +4 -4
- package/src/execution/engine/agent/core/types.ts +25 -25
- package/src/execution/engine/agent/index.ts +6 -6
- package/src/execution/engine/agent/reasoning/__tests__/request-builder.test.ts +24 -24
- package/src/execution/engine/index.ts +443 -443
- package/src/execution/engine/tools/integration/server/adapters/apify/__tests__/apify-run-actor.integration.test.ts +298 -298
- package/src/execution/engine/tools/integration/server/adapters/apify/apify-adapter.test.ts +55 -55
- package/src/execution/engine/tools/integration/server/adapters/apify/apify-adapter.ts +107 -107
- package/src/execution/engine/tools/integration/server/adapters/apollo/apollo-adapter.test.ts +48 -48
- package/src/execution/engine/tools/integration/server/adapters/apollo/apollo-adapter.ts +99 -99
- package/src/execution/engine/tools/integration/server/adapters/apollo/index.ts +1 -1
- package/src/execution/engine/tools/integration/server/adapters/attio/__tests__/attio-crud.integration.test.ts +363 -363
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/get-record/index.test.ts +162 -162
- package/src/execution/engine/tools/integration/server/adapters/attio/fetch/list-records/index.test.ts +316 -316
- package/src/execution/engine/tools/integration/server/adapters/clickup/clickup-adapter.test.ts +18 -18
- package/src/execution/engine/tools/integration/server/adapters/clickup/clickup-adapter.ts +194 -194
- package/src/execution/engine/tools/integration/server/adapters/clickup/index.ts +7 -7
- package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-adapter.ts +204 -204
- package/src/execution/engine/tools/integration/server/adapters/gmail/gmail-tools.ts +105 -105
- package/src/execution/engine/tools/integration/server/adapters/google-calendar/google-calendar-adapter.ts +428 -428
- package/src/execution/engine/tools/integration/server/adapters/google-calendar/index.ts +2 -2
- package/src/execution/engine/tools/integration/server/adapters/google-sheets/__tests__/google-sheets.integration.test.ts +261 -261
- package/src/execution/engine/tools/integration/server/adapters/instantly/instantly-tools.ts +1474 -1474
- package/src/execution/engine/tools/integration/server/adapters/millionverifier/millionverifier-tools.ts +103 -103
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.test.ts +88 -88
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/send-email/index.ts +141 -141
- package/src/execution/engine/tools/integration/server/adapters/resend/fetch/utils/types.ts +76 -76
- package/src/execution/engine/tools/integration/server/adapters/signature-api/signature-api-tools.ts +182 -182
- package/src/execution/engine/tools/integration/server/adapters/stripe/stripe-tools.ts +310 -310
- package/src/execution/engine/tools/integration/service.test.ts +239 -239
- package/src/execution/engine/tools/integration/service.ts +172 -172
- package/src/execution/engine/tools/integration/tool.ts +255 -255
- package/src/execution/engine/tools/lead-service-types.ts +1005 -1005
- package/src/execution/engine/tools/messages.ts +43 -43
- package/src/execution/engine/tools/platform/acquisition/company-tools.ts +7 -7
- package/src/execution/engine/tools/platform/acquisition/contact-tools.ts +6 -6
- package/src/execution/engine/tools/platform/acquisition/list-tools.ts +6 -6
- package/src/execution/engine/tools/platform/acquisition/types.ts +280 -280
- package/src/execution/engine/tools/platform/email/types.ts +97 -97
- package/src/execution/engine/tools/registry.ts +704 -704
- package/src/execution/engine/tools/tool-maps.ts +831 -831
- package/src/execution/engine/tools/types.ts +234 -234
- package/src/execution/engine/workflow/types.ts +202 -202
- package/src/execution/external/__tests__/api-schemas.test.ts +127 -127
- package/src/execution/external/api-schemas.ts +40 -40
- package/src/execution/external/index.ts +1 -1
- package/src/index.ts +18 -18
- package/src/integrations/credentials/__tests__/api-schemas.test.ts +420 -420
- package/src/integrations/credentials/api-schemas.ts +146 -146
- package/src/integrations/credentials/schemas.ts +200 -200
- package/src/integrations/oauth/__tests__/provider-registry.test.ts +7 -7
- package/src/integrations/oauth/provider-registry.ts +74 -74
- package/src/integrations/oauth/server/credentials.ts +43 -43
- package/src/integrations/webhook-endpoints/__tests__/api-schemas.test.ts +327 -327
- package/src/integrations/webhook-endpoints/api-schemas.ts +103 -103
- package/src/integrations/webhook-endpoints/types.ts +58 -58
- package/src/knowledge/README.md +33 -32
- package/src/knowledge/__tests__/queries.test.ts +633 -541
- package/src/knowledge/format.ts +100 -99
- package/src/knowledge/index.ts +5 -5
- package/src/knowledge/published.ts +5 -5
- package/src/knowledge/queries.ts +274 -222
- package/src/operations/activities/api-schemas.ts +80 -80
- package/src/operations/activities/types.ts +64 -64
- package/src/organization-model/README.md +149 -109
- package/src/organization-model/__tests__/content-kinds-registry.test.ts +210 -0
- package/src/organization-model/__tests__/defaults.test.ts +168 -194
- package/src/organization-model/__tests__/domains/actions.test.ts +78 -0
- package/src/organization-model/__tests__/domains/customers.test.ts +48 -44
- package/src/organization-model/__tests__/domains/entities.test.ts +56 -0
- package/src/organization-model/__tests__/domains/goals.test.ts +110 -96
- package/src/organization-model/__tests__/domains/identity.test.ts +4 -3
- package/src/organization-model/__tests__/domains/navigation.test.ts +222 -166
- package/src/organization-model/__tests__/domains/offerings.test.ts +83 -88
- package/src/organization-model/__tests__/domains/policies.test.ts +323 -0
- package/src/organization-model/__tests__/domains/resource-mappings.test.ts +30 -30
- package/src/organization-model/__tests__/domains/resources.test.ts +396 -175
- package/src/organization-model/__tests__/domains/roles.test.ts +463 -402
- package/src/organization-model/__tests__/domains/statuses.test.ts +13 -10
- package/src/organization-model/__tests__/domains/systems.test.ts +209 -193
- package/src/organization-model/__tests__/flatten-additive-merge.test.ts +362 -0
- package/src/organization-model/__tests__/foundation.test.ts +47 -75
- package/src/organization-model/__tests__/get-resources-for-system.test.ts +144 -0
- package/src/organization-model/__tests__/graph.test.ts +1336 -149
- package/src/organization-model/__tests__/icons.test.ts +10 -1
- package/src/organization-model/__tests__/knowledge.test.ts +418 -61
- package/src/organization-model/__tests__/lookup-helpers.test.ts +438 -0
- package/src/organization-model/__tests__/migration-helpers.test.ts +591 -0
- package/src/organization-model/__tests__/prospecting-ssot.test.ts +103 -94
- package/src/organization-model/__tests__/recursive-system-schema.test.ts +549 -0
- package/src/organization-model/__tests__/resolve.test.ts +303 -42
- package/src/organization-model/__tests__/schema.test.ts +863 -153
- package/src/organization-model/__tests__/surface-projection.test.ts +284 -174
- package/src/organization-model/catalogs/lead-gen.ts +144 -0
- package/src/organization-model/content-kinds/config.ts +36 -0
- package/src/organization-model/content-kinds/index.ts +78 -0
- package/src/organization-model/content-kinds/pipeline.ts +68 -0
- package/src/organization-model/content-kinds/registry.ts +44 -0
- package/src/organization-model/content-kinds/status.ts +71 -0
- package/src/organization-model/content-kinds/template.ts +83 -0
- package/src/organization-model/content-kinds/types.ts +117 -0
- package/src/organization-model/contracts.ts +27 -17
- package/src/organization-model/defaults.ts +489 -107
- package/src/organization-model/domains/actions.ts +333 -0
- package/src/organization-model/domains/customers.ts +10 -7
- package/src/organization-model/domains/entities.ts +144 -0
- package/src/organization-model/domains/goals.ts +9 -6
- package/src/organization-model/domains/knowledge.ts +128 -54
- package/src/organization-model/domains/navigation.ts +139 -416
- package/src/organization-model/domains/offerings.ts +15 -10
- package/src/organization-model/domains/policies.ts +102 -0
- package/src/organization-model/domains/projects.ts +6 -40
- package/src/organization-model/domains/prospecting.ts +395 -514
- package/src/organization-model/domains/resources.ts +173 -81
- package/src/organization-model/domains/roles.ts +96 -93
- package/src/organization-model/domains/sales.test.ts +218 -218
- package/src/organization-model/domains/sales.ts +380 -589
- package/src/organization-model/domains/shared.ts +8 -8
- package/src/organization-model/domains/statuses.ts +298 -89
- package/src/organization-model/domains/systems.ts +240 -38
- package/src/organization-model/foundation.ts +35 -48
- package/src/organization-model/graph/build.ts +1035 -279
- package/src/organization-model/graph/index.ts +4 -4
- package/src/organization-model/graph/link.ts +10 -10
- package/src/organization-model/graph/schema.ts +77 -56
- package/src/organization-model/graph/types.ts +75 -56
- package/src/organization-model/helpers.ts +312 -59
- package/src/organization-model/icons.ts +78 -66
- package/src/organization-model/index.ts +129 -16
- package/src/organization-model/migration-helpers.ts +252 -0
- package/src/organization-model/ontology.ts +661 -0
- package/src/organization-model/organization-graph.mdx +110 -89
- package/src/organization-model/organization-model.mdx +226 -171
- package/src/organization-model/published.ts +295 -139
- package/src/organization-model/resolve.ts +139 -21
- package/src/organization-model/schema.ts +841 -301
- package/src/organization-model/surface-projection.ts +212 -218
- package/src/organization-model/types.ts +181 -90
- package/src/platform/api/types.ts +38 -38
- package/src/platform/constants/versions.ts +3 -3
- package/src/platform/index.ts +23 -23
- package/src/platform/registry/__tests__/command-view.test.ts +5 -7
- package/src/platform/registry/__tests__/resource-link.test.ts +35 -30
- package/src/platform/registry/__tests__/resource-registry.integration.test.ts +17 -32
- package/src/platform/registry/__tests__/resource-registry.nested-systems.test.ts +245 -0
- package/src/platform/registry/__tests__/resource-registry.test.ts +2053 -2051
- package/src/platform/registry/__tests__/validation.test.ts +1347 -1343
- package/src/platform/registry/command-view.ts +10 -10
- package/src/platform/registry/index.ts +103 -103
- package/src/platform/registry/resource-link.ts +32 -32
- package/src/platform/registry/resource-registry.ts +890 -878
- package/src/platform/registry/serialization.ts +295 -295
- package/src/platform/registry/serialized-types.ts +166 -166
- package/src/platform/registry/stats-types.ts +68 -68
- package/src/platform/registry/types.ts +425 -425
- package/src/platform/registry/validation.ts +745 -743
- package/src/platform/utils/__tests__/validation.test.ts +1084 -1084
- package/src/platform/utils/validation.ts +425 -425
- package/src/projects/api-schemas.test.ts +39 -39
- package/src/projects/api-schemas.ts +291 -291
- package/src/reference/_generated/contracts.md +2389 -2121
- package/src/reference/glossary.md +76 -76
- package/src/scaffold-registry/__tests__/index.test.ts +206 -206
- package/src/scaffold-registry/__tests__/schema.test.ts +166 -166
- package/src/scaffold-registry/index.ts +392 -392
- package/src/scaffold-registry/schema.ts +243 -243
- package/src/server.ts +289 -289
- package/src/supabase/database.types.ts +3153 -3093
- package/src/test-utils/README.md +37 -37
- package/src/test-utils/entities.ts +108 -108
- package/src/test-utils/fixtures/memberships.ts +82 -82
- package/src/test-utils/index.ts +12 -12
- package/src/test-utils/organization-model.ts +65 -65
- package/src/test-utils/published.ts +6 -6
- package/src/test-utils/rls/RLSTestContext.ts +588 -588
- package/src/test-utils/test-utils.test.ts +44 -49
- package/src/organization-model/__tests__/domains/operations.test.ts +0 -203
- package/src/organization-model/domains/features.ts +0 -31
- package/src/organization-model/domains/operations.ts +0 -85
|
@@ -0,0 +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 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
|
+
})
|
|
@@ -2,56 +2,32 @@ import { describe, expect, it } from 'vitest'
|
|
|
2
2
|
import { createFoundationOrganizationModel } from '../foundation'
|
|
3
3
|
|
|
4
4
|
describe('createFoundationOrganizationModel', () => {
|
|
5
|
-
it('builds the foundation model from
|
|
5
|
+
it('builds the foundation model from derived sidebar navigation', () => {
|
|
6
6
|
const result = createFoundationOrganizationModel({
|
|
7
7
|
branding: { organizationName: 'Acme', productName: 'Acme OS', shortName: 'Acme' }
|
|
8
8
|
})
|
|
9
9
|
|
|
10
10
|
expect(result.canonical.branding.organizationName).toBe('Acme')
|
|
11
11
|
expect(result.homeLabel).toBe('Dashboard')
|
|
12
|
-
expect(result.model.navigation.defaultSurfaceId).toBe('
|
|
12
|
+
expect(result.model.navigation.defaultSurfaceId).toBe('dashboard')
|
|
13
13
|
|
|
14
14
|
const surfaces = result.model.navigation.surfaces
|
|
15
|
-
expect(surfaces.find((s) => s.id === '
|
|
16
|
-
expect(surfaces.find((s) => s.id === '
|
|
15
|
+
expect(surfaces.find((s) => s.id === 'dashboard')).toMatchObject({ path: '/', surfaceType: 'dashboard' })
|
|
16
|
+
expect(surfaces.find((s) => s.id === 'sales')).toBeDefined()
|
|
17
|
+
expect(surfaces.find((s) => s.id === 'clients')).toMatchObject({ path: '/clients' })
|
|
17
18
|
expect(surfaces.find((s) => s.id === 'projects')).toBeDefined()
|
|
18
|
-
expect(surfaces.find((s) => s.id === 'operations')).toBeDefined()
|
|
19
|
-
expect(surfaces.find((s) => s.id === 'settings')).toBeDefined()
|
|
19
|
+
expect(surfaces.find((s) => s.id === 'operations-overview')).toBeDefined()
|
|
20
|
+
expect(surfaces.find((s) => s.id === 'settings-account')).toBeDefined()
|
|
20
21
|
|
|
21
|
-
expect(result.quickAccessSurfaceIds).toContain('
|
|
22
|
+
expect(result.quickAccessSurfaceIds).toContain('sales')
|
|
23
|
+
expect(result.quickAccessSurfaceIds).toContain('clients')
|
|
22
24
|
expect(result.quickAccessSurfaceIds).toContain('projects')
|
|
23
|
-
expect(result.quickAccessSurfaceIds).toContain('
|
|
24
|
-
expect(result.quickAccessSurfaceIds).toContain('crm')
|
|
25
|
+
expect(result.quickAccessSurfaceIds).toContain('operations-overview')
|
|
25
26
|
})
|
|
26
27
|
|
|
27
|
-
it('passes deeper overrides through to the canonical model', () => {
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
sales: {
|
|
31
|
-
pipelines: [
|
|
32
|
-
{
|
|
33
|
-
id: 'custom-pipeline',
|
|
34
|
-
label: 'Custom Pipeline',
|
|
35
|
-
description: 'A custom pipeline',
|
|
36
|
-
entityId: 'crm.deal',
|
|
37
|
-
stages: [
|
|
38
|
-
{
|
|
39
|
-
id: 'stage-1',
|
|
40
|
-
label: 'Stage 1',
|
|
41
|
-
color: 'blue',
|
|
42
|
-
order: 0,
|
|
43
|
-
semanticClass: 'open' as const,
|
|
44
|
-
surfaceIds: ['crm.pipeline'],
|
|
45
|
-
resourceIds: []
|
|
46
|
-
}
|
|
47
|
-
]
|
|
48
|
-
}
|
|
49
|
-
]
|
|
50
|
-
}
|
|
51
|
-
})
|
|
52
|
-
|
|
53
|
-
expect(result.canonical.sales.pipelines).toHaveLength(1)
|
|
54
|
-
expect(result.canonical.sales.pipelines[0]?.id).toBe('custom-pipeline')
|
|
28
|
+
it.skip('passes deeper overrides through to the canonical model (deferred - Phase 4: sales domain removed)', () => {
|
|
29
|
+
// Previously tested: result.canonical.sales.pipelines.
|
|
30
|
+
// Re-enable with a content-node based pipeline override test if needed.
|
|
55
31
|
})
|
|
56
32
|
|
|
57
33
|
it('exposes a working getOrganizationSurface lookup', () => {
|
|
@@ -59,47 +35,43 @@ describe('createFoundationOrganizationModel', () => {
|
|
|
59
35
|
branding: { organizationName: 'Acme', productName: 'Acme OS', shortName: 'Acme' }
|
|
60
36
|
})
|
|
61
37
|
|
|
62
|
-
const
|
|
63
|
-
expect(
|
|
64
|
-
expect(
|
|
65
|
-
expect(
|
|
66
|
-
|
|
67
|
-
expect(result.getOrganizationSurface('nonexistent')).toBeUndefined()
|
|
68
|
-
})
|
|
38
|
+
const clientsSurface = result.getOrganizationSurface('clients')
|
|
39
|
+
expect(clientsSurface).toBeDefined()
|
|
40
|
+
expect(clientsSurface?.id).toBe('clients')
|
|
41
|
+
expect(clientsSurface?.path).toBe('/clients')
|
|
69
42
|
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
label: 'CRM',
|
|
76
|
-
description: 'CRM workspace',
|
|
77
|
-
enabled: true,
|
|
78
|
-
color: 'blue',
|
|
79
|
-
entityIds: [],
|
|
80
|
-
surfaceIds: ['custom.home'],
|
|
81
|
-
resourceIds: [],
|
|
82
|
-
capabilityIds: []
|
|
83
|
-
}
|
|
84
|
-
],
|
|
43
|
+
expect(result.getOrganizationSurface('nonexistent')).toBeUndefined()
|
|
44
|
+
})
|
|
45
|
+
|
|
46
|
+
it('does not require hardcoded system-backed foundation surfaces', () => {
|
|
47
|
+
const result = createFoundationOrganizationModel({
|
|
85
48
|
navigation: {
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
49
|
+
sidebar: {
|
|
50
|
+
primary: {
|
|
51
|
+
dashboard: {
|
|
52
|
+
type: 'surface',
|
|
53
|
+
label: 'Home',
|
|
54
|
+
path: '/',
|
|
55
|
+
surfaceType: 'dashboard',
|
|
56
|
+
order: 10,
|
|
57
|
+
targets: { systems: ['dashboard'] }
|
|
58
|
+
},
|
|
59
|
+
custom: {
|
|
60
|
+
type: 'surface',
|
|
61
|
+
label: 'Custom',
|
|
62
|
+
path: '/custom',
|
|
63
|
+
surfaceType: 'page',
|
|
64
|
+
order: 20,
|
|
65
|
+
targets: { systems: [] }
|
|
66
|
+
}
|
|
67
|
+
},
|
|
68
|
+
bottom: {}
|
|
69
|
+
}
|
|
100
70
|
}
|
|
101
|
-
}
|
|
71
|
+
})
|
|
102
72
|
|
|
103
|
-
expect(
|
|
73
|
+
expect(result.model.navigation.defaultSurfaceId).toBe('dashboard')
|
|
74
|
+
expect(result.homeLabel).toBe('Home')
|
|
75
|
+
expect(result.getOrganizationSurface('custom')).toMatchObject({ label: 'Custom', path: '/custom' })
|
|
104
76
|
})
|
|
105
77
|
})
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Unit tests for getResourcesForSystem helper.
|
|
3
|
+
*
|
|
4
|
+
* Coverage:
|
|
5
|
+
* - Empty model (no resources)
|
|
6
|
+
* - Single leaf resource, exact match
|
|
7
|
+
* - includeDescendants: true returns descendants
|
|
8
|
+
* - Deep nesting (three levels)
|
|
9
|
+
* - Segment-boundary safety: 'sales' must NOT match 'salesforce.foo'
|
|
10
|
+
* - Root path with includeDescendants: false
|
|
11
|
+
* - Root path with includeDescendants: true
|
|
12
|
+
* - Mixed resources, only matching ones returned
|
|
13
|
+
*/
|
|
14
|
+
import { describe, it, expect } from 'vitest'
|
|
15
|
+
import { getResourcesForSystem } from '../helpers'
|
|
16
|
+
import type { OrganizationModel } from '../types'
|
|
17
|
+
import type { ResourceEntry } from '../domains/resources'
|
|
18
|
+
|
|
19
|
+
// ---------------------------------------------------------------------------
|
|
20
|
+
// Minimal fixture builder
|
|
21
|
+
// ---------------------------------------------------------------------------
|
|
22
|
+
|
|
23
|
+
function makeModel(resources: Record<string, ResourceEntry>): OrganizationModel {
|
|
24
|
+
return {
|
|
25
|
+
systems: {},
|
|
26
|
+
resources,
|
|
27
|
+
roles: {},
|
|
28
|
+
goals: {},
|
|
29
|
+
knowledge: {},
|
|
30
|
+
actions: {},
|
|
31
|
+
entities: {},
|
|
32
|
+
policies: {},
|
|
33
|
+
navigation: { sidebar: { primary: {}, bottom: {} } }
|
|
34
|
+
} as unknown as OrganizationModel
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function makeWorkflow(id: string, systemPath: string): ResourceEntry {
|
|
38
|
+
return {
|
|
39
|
+
id,
|
|
40
|
+
order: 10,
|
|
41
|
+
kind: 'workflow',
|
|
42
|
+
systemPath,
|
|
43
|
+
status: 'active'
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// ---------------------------------------------------------------------------
|
|
48
|
+
// Tests
|
|
49
|
+
// ---------------------------------------------------------------------------
|
|
50
|
+
|
|
51
|
+
describe('getResourcesForSystem', () => {
|
|
52
|
+
it('returns empty array when model has no resources', () => {
|
|
53
|
+
const model = makeModel({})
|
|
54
|
+
expect(getResourcesForSystem(model, 'sales')).toEqual([])
|
|
55
|
+
})
|
|
56
|
+
|
|
57
|
+
it('returns empty array when no resources match the path', () => {
|
|
58
|
+
const model = makeModel({
|
|
59
|
+
'wf-a': makeWorkflow('wf-a', 'marketing')
|
|
60
|
+
})
|
|
61
|
+
expect(getResourcesForSystem(model, 'sales')).toEqual([])
|
|
62
|
+
})
|
|
63
|
+
|
|
64
|
+
it('returns a single resource with exact match', () => {
|
|
65
|
+
const r = makeWorkflow('wf-a', 'sales')
|
|
66
|
+
const model = makeModel({ 'wf-a': r })
|
|
67
|
+
const result = getResourcesForSystem(model, 'sales')
|
|
68
|
+
expect(result).toHaveLength(1)
|
|
69
|
+
expect(result[0].id).toBe('wf-a')
|
|
70
|
+
})
|
|
71
|
+
|
|
72
|
+
it('does NOT return descendants when includeDescendants is false (default)', () => {
|
|
73
|
+
const model = makeModel({
|
|
74
|
+
'wf-a': makeWorkflow('wf-a', 'sales'),
|
|
75
|
+
'wf-b': makeWorkflow('wf-b', 'sales.crm'),
|
|
76
|
+
'wf-c': makeWorkflow('wf-c', 'sales.lead-gen')
|
|
77
|
+
})
|
|
78
|
+
const result = getResourcesForSystem(model, 'sales')
|
|
79
|
+
expect(result).toHaveLength(1)
|
|
80
|
+
expect(result[0].id).toBe('wf-a')
|
|
81
|
+
})
|
|
82
|
+
|
|
83
|
+
it('returns exact match AND descendants when includeDescendants is true', () => {
|
|
84
|
+
const model = makeModel({
|
|
85
|
+
'wf-a': makeWorkflow('wf-a', 'sales'),
|
|
86
|
+
'wf-b': makeWorkflow('wf-b', 'sales.crm'),
|
|
87
|
+
'wf-c': makeWorkflow('wf-c', 'sales.lead-gen'),
|
|
88
|
+
'wf-d': makeWorkflow('wf-d', 'marketing')
|
|
89
|
+
})
|
|
90
|
+
const result = getResourcesForSystem(model, 'sales', { includeDescendants: true })
|
|
91
|
+
const ids = result.map((r) => r.id).sort()
|
|
92
|
+
expect(ids).toEqual(['wf-a', 'wf-b', 'wf-c'])
|
|
93
|
+
})
|
|
94
|
+
|
|
95
|
+
it('matches three-level deep descendants', () => {
|
|
96
|
+
const model = makeModel({
|
|
97
|
+
'wf-a': makeWorkflow('wf-a', 'sales.crm.dispositions'),
|
|
98
|
+
'wf-b': makeWorkflow('wf-b', 'sales.crm'),
|
|
99
|
+
'wf-c': makeWorkflow('wf-c', 'sales')
|
|
100
|
+
})
|
|
101
|
+
const result = getResourcesForSystem(model, 'sales', { includeDescendants: true })
|
|
102
|
+
expect(result).toHaveLength(3)
|
|
103
|
+
})
|
|
104
|
+
|
|
105
|
+
it('segment-boundary: "sales" does NOT match "salesforce.foo"', () => {
|
|
106
|
+
const model = makeModel({
|
|
107
|
+
'wf-a': makeWorkflow('wf-a', 'salesforce.foo'),
|
|
108
|
+
'wf-b': makeWorkflow('wf-b', 'salesforce'),
|
|
109
|
+
'wf-c': makeWorkflow('wf-c', 'sales')
|
|
110
|
+
})
|
|
111
|
+
const result = getResourcesForSystem(model, 'sales', { includeDescendants: true })
|
|
112
|
+
// Only exact 'sales' match — 'salesforce' and 'salesforce.foo' must NOT appear
|
|
113
|
+
expect(result).toHaveLength(1)
|
|
114
|
+
expect(result[0].id).toBe('wf-c')
|
|
115
|
+
})
|
|
116
|
+
|
|
117
|
+
it('segment-boundary: "sys" does NOT match "sys2.foo" or "system.bar"', () => {
|
|
118
|
+
const model = makeModel({
|
|
119
|
+
'wf-a': makeWorkflow('wf-a', 'sys'),
|
|
120
|
+
'wf-b': makeWorkflow('wf-b', 'sys.platform'),
|
|
121
|
+
'wf-c': makeWorkflow('wf-c', 'sys2.foo'),
|
|
122
|
+
'wf-d': makeWorkflow('wf-d', 'system.bar')
|
|
123
|
+
})
|
|
124
|
+
const result = getResourcesForSystem(model, 'sys', { includeDescendants: true })
|
|
125
|
+
const ids = result.map((r) => r.id).sort()
|
|
126
|
+
expect(ids).toEqual(['wf-a', 'wf-b'])
|
|
127
|
+
})
|
|
128
|
+
|
|
129
|
+
it('root path with includeDescendants false returns only exact root resources', () => {
|
|
130
|
+
const model = makeModel({
|
|
131
|
+
'wf-a': makeWorkflow('wf-a', 'sys'),
|
|
132
|
+
'wf-b': makeWorkflow('wf-b', 'sys.platform'),
|
|
133
|
+
'wf-c': makeWorkflow('wf-c', 'sys.diagnostic')
|
|
134
|
+
})
|
|
135
|
+
const result = getResourcesForSystem(model, 'sys')
|
|
136
|
+
expect(result).toHaveLength(1)
|
|
137
|
+
expect(result[0].id).toBe('wf-a')
|
|
138
|
+
})
|
|
139
|
+
|
|
140
|
+
it('handles undefined resources gracefully via nullish coalescing', () => {
|
|
141
|
+
const model = { systems: {}, resources: undefined } as unknown as OrganizationModel
|
|
142
|
+
expect(getResourcesForSystem(model, 'sales')).toEqual([])
|
|
143
|
+
})
|
|
144
|
+
})
|