@elevasis/core 0.26.0 → 0.27.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 +5 -5
- package/dist/index.js +209 -173
- package/dist/knowledge/index.d.ts +21 -21
- package/dist/organization-model/index.d.ts +5 -5
- package/dist/organization-model/index.js +209 -173
- package/dist/test-utils/index.d.ts +2 -2
- package/dist/test-utils/index.js +182 -126
- package/package.json +1 -1
- package/src/_gen/__tests__/__snapshots__/contracts.md.snap +976 -1063
- package/src/business/acquisition/api-schemas.test.ts +1962 -1841
- package/src/business/acquisition/api-schemas.ts +1461 -1464
- package/src/business/acquisition/crm-next-action.test.ts +45 -25
- package/src/business/acquisition/crm-next-action.ts +227 -220
- package/src/business/acquisition/crm-priority.test.ts +41 -8
- package/src/business/acquisition/crm-priority.ts +365 -349
- package/src/business/acquisition/crm-state-actions.test.ts +208 -153
- package/src/business/acquisition/derive-actions.test.ts +90 -13
- package/src/business/acquisition/derive-actions.ts +8 -139
- package/src/business/acquisition/ontology-validation.ts +72 -158
- package/src/business/pdf/sections/investment.ts +1 -1
- package/src/business/pdf/sections/summary-investment.ts +1 -1
- package/src/execution/engine/tools/tool-maps.ts +872 -831
- package/src/organization-model/__tests__/cross-ref.test.ts +167 -0
- package/src/organization-model/__tests__/published-zero-leak.test.ts +60 -1
- package/src/organization-model/__tests__/resolve.test.ts +1 -1
- package/src/organization-model/__tests__/schema-refinements.test.ts +72 -0
- package/src/organization-model/cross-ref.ts +175 -0
- package/src/organization-model/domains/branding.ts +6 -6
- package/src/organization-model/domains/sales.test.ts +104 -218
- package/src/organization-model/domains/sales.ts +212 -375
- package/src/organization-model/index.ts +1 -0
- package/src/organization-model/schema-refinements.ts +667 -0
- package/src/organization-model/schema.ts +8 -715
- package/src/reference/_generated/contracts.md +976 -1063
|
@@ -1,218 +1,104 @@
|
|
|
1
|
-
import { describe,
|
|
2
|
-
import {
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
it('
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
})
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
expect(
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
expect(
|
|
61
|
-
})
|
|
62
|
-
|
|
63
|
-
it
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
it('
|
|
93
|
-
expect(
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
})
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
'follow_up_due',
|
|
106
|
-
'waiting',
|
|
107
|
-
'stale',
|
|
108
|
-
'closed_low'
|
|
109
|
-
])
|
|
110
|
-
expect(CRM_PRIORITY_BUCKETS.map((bucket) => bucket.rank)).toEqual([10, 20, 30, 40, 50])
|
|
111
|
-
})
|
|
112
|
-
|
|
113
|
-
it('includes display metadata for each bucket', () => {
|
|
114
|
-
for (const bucket of CRM_PRIORITY_BUCKETS) {
|
|
115
|
-
expect(bucket.label).toEqual(expect.any(String))
|
|
116
|
-
expect(bucket.rank).toEqual(expect.any(Number))
|
|
117
|
-
expect(bucket.color).toEqual(expect.any(String))
|
|
118
|
-
}
|
|
119
|
-
})
|
|
120
|
-
|
|
121
|
-
it('maps CRM states and closed stages into priority rules', () => {
|
|
122
|
-
expect(DEFAULT_CRM_PRIORITY_RULE_CONFIG.closedStageKeys).toEqual(['closed_won', 'closed_lost'])
|
|
123
|
-
expect(DEFAULT_CRM_PRIORITY_RULE_CONFIG.followUpAfterDaysByStateKey.discovery_link_sent).toBe(3)
|
|
124
|
-
expect(DEFAULT_CRM_PRIORITY_RULE_CONFIG.staleAfterDays).toBeGreaterThan(0)
|
|
125
|
-
})
|
|
126
|
-
})
|
|
127
|
-
|
|
128
|
-
describe('getValidStatesForStage', () => {
|
|
129
|
-
it('returns all 8 states for the interested stage', () => {
|
|
130
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'interested')
|
|
131
|
-
expect(states).toHaveLength(8)
|
|
132
|
-
expect(states.map((s) => s.stateKey)).toContain('discovery_replied')
|
|
133
|
-
expect(states.map((s) => s.stateKey)).toContain('followup_3_sent')
|
|
134
|
-
})
|
|
135
|
-
|
|
136
|
-
it('returns empty array for the proposal stage (no states defined)', () => {
|
|
137
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'proposal')
|
|
138
|
-
expect(states).toEqual([])
|
|
139
|
-
})
|
|
140
|
-
|
|
141
|
-
it('returns empty array for the closing stage (no states defined)', () => {
|
|
142
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'closing')
|
|
143
|
-
expect(states).toEqual([])
|
|
144
|
-
})
|
|
145
|
-
|
|
146
|
-
it('returns empty array for closed_won stage', () => {
|
|
147
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'closed_won')
|
|
148
|
-
expect(states).toEqual([])
|
|
149
|
-
})
|
|
150
|
-
|
|
151
|
-
it('returns empty array for closed_lost stage', () => {
|
|
152
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'closed_lost')
|
|
153
|
-
expect(states).toEqual([])
|
|
154
|
-
})
|
|
155
|
-
|
|
156
|
-
it('returns empty array for nurturing stage (no states defined)', () => {
|
|
157
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'nurturing')
|
|
158
|
-
expect(states).toEqual([])
|
|
159
|
-
})
|
|
160
|
-
|
|
161
|
-
it('returns empty array for an unknown stage key', () => {
|
|
162
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'unknown_stage')
|
|
163
|
-
expect(states).toEqual([])
|
|
164
|
-
})
|
|
165
|
-
|
|
166
|
-
it('returns empty array for an empty string stage key', () => {
|
|
167
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, '')
|
|
168
|
-
expect(states).toEqual([])
|
|
169
|
-
})
|
|
170
|
-
|
|
171
|
-
it('is case-sensitive — "Interested" does not match "interested"', () => {
|
|
172
|
-
const states = getValidStatesForStage(CRM_PIPELINE_DEFINITION, 'Interested')
|
|
173
|
-
expect(states).toEqual([])
|
|
174
|
-
})
|
|
175
|
-
|
|
176
|
-
it('works with the lead-gen member pipeline for known stage', () => {
|
|
177
|
-
const states = getValidStatesForStage(ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE, 'outreach')
|
|
178
|
-
expect(states.length).toBeGreaterThan(0)
|
|
179
|
-
expect(states.map((s) => s.stateKey)).toContain('personalized')
|
|
180
|
-
})
|
|
181
|
-
|
|
182
|
-
it('works with the lead-gen member pipeline for unknown stage', () => {
|
|
183
|
-
const states = getValidStatesForStage(ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE, 'nonexistent')
|
|
184
|
-
expect(states).toEqual([])
|
|
185
|
-
})
|
|
186
|
-
})
|
|
187
|
-
|
|
188
|
-
describe('findPipeline', () => {
|
|
189
|
-
it('finds a pipeline by pipelineKey when present', () => {
|
|
190
|
-
const found = findPipeline(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-member'], 'lead-gen')
|
|
191
|
-
expect(found).toBe(ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE)
|
|
192
|
-
})
|
|
193
|
-
|
|
194
|
-
it('returns undefined for an unknown pipelineKey', () => {
|
|
195
|
-
const found = findPipeline(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-member'], 'crm')
|
|
196
|
-
expect(found).toBeUndefined()
|
|
197
|
-
})
|
|
198
|
-
|
|
199
|
-
it('returns undefined for an empty array', () => {
|
|
200
|
-
const found = findPipeline([], 'lead-gen')
|
|
201
|
-
expect(found).toBeUndefined()
|
|
202
|
-
})
|
|
203
|
-
})
|
|
204
|
-
|
|
205
|
-
describe('LEAD_GEN_PIPELINE_DEFINITIONS', () => {
|
|
206
|
-
it('has entries for acq.list-member and acq.list-company', () => {
|
|
207
|
-
expect(LEAD_GEN_PIPELINE_DEFINITIONS).toHaveProperty('acq.list-member')
|
|
208
|
-
expect(LEAD_GEN_PIPELINE_DEFINITIONS).toHaveProperty('acq.list-company')
|
|
209
|
-
})
|
|
210
|
-
|
|
211
|
-
it('acq.list-member entry contains the ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE', () => {
|
|
212
|
-
expect(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-member']).toContain(ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE)
|
|
213
|
-
})
|
|
214
|
-
|
|
215
|
-
it('acq.list-company entry contains the ACQ_LIST_COMPANIES_LEAD_GEN_PIPELINE', () => {
|
|
216
|
-
expect(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-company']).toContain(ACQ_LIST_COMPANIES_LEAD_GEN_PIPELINE)
|
|
217
|
-
})
|
|
218
|
-
})
|
|
1
|
+
import { describe, expect, it } from 'vitest'
|
|
2
|
+
import {
|
|
3
|
+
ACQ_LIST_COMPANIES_LEAD_GEN_PIPELINE,
|
|
4
|
+
ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE,
|
|
5
|
+
LEAD_GEN_PIPELINE_DEFINITIONS,
|
|
6
|
+
findPipeline,
|
|
7
|
+
getValidStatesForStage,
|
|
8
|
+
type StatefulPipelineDefinition
|
|
9
|
+
} from './sales'
|
|
10
|
+
|
|
11
|
+
const TEST_CRM_PIPELINE: StatefulPipelineDefinition = {
|
|
12
|
+
pipelineKey: 'crm',
|
|
13
|
+
label: 'CRM',
|
|
14
|
+
entityKey: 'crm.deal',
|
|
15
|
+
stages: [
|
|
16
|
+
{
|
|
17
|
+
stageKey: 'interested',
|
|
18
|
+
label: 'Interested',
|
|
19
|
+
color: 'blue',
|
|
20
|
+
states: [
|
|
21
|
+
{ stateKey: 'discovery_replied', label: 'Discovery Replied' },
|
|
22
|
+
{ stateKey: 'discovery_link_sent', label: 'Discovery Link Sent' }
|
|
23
|
+
]
|
|
24
|
+
},
|
|
25
|
+
{ stageKey: 'proposal', label: 'Proposal', color: 'yellow', states: [] }
|
|
26
|
+
]
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
describe('sales domain published surface', () => {
|
|
30
|
+
it('does not export Elevasis CRM runtime constants from the generic core sales domain', async () => {
|
|
31
|
+
const sales = await import('./sales')
|
|
32
|
+
|
|
33
|
+
expect(Object.keys(sales)).not.toEqual(
|
|
34
|
+
expect.arrayContaining([
|
|
35
|
+
'CRM_PIPELINE_DEFINITION',
|
|
36
|
+
'CRM_DISCOVERY_REPLIED_STATE',
|
|
37
|
+
'CRM_PRIORITY_BUCKETS',
|
|
38
|
+
'DEFAULT_CRM_PRIORITY_RULE_CONFIG',
|
|
39
|
+
'DEFAULT_CRM_NEXT_ACTION_RULE_CONFIG'
|
|
40
|
+
])
|
|
41
|
+
)
|
|
42
|
+
})
|
|
43
|
+
})
|
|
44
|
+
|
|
45
|
+
describe('getValidStatesForStage', () => {
|
|
46
|
+
it('returns states for a known stage from a caller-supplied pipeline definition', () => {
|
|
47
|
+
const states = getValidStatesForStage(TEST_CRM_PIPELINE, 'interested')
|
|
48
|
+
|
|
49
|
+
expect(states).toEqual([
|
|
50
|
+
{ stateKey: 'discovery_replied', label: 'Discovery Replied' },
|
|
51
|
+
{ stateKey: 'discovery_link_sent', label: 'Discovery Link Sent' }
|
|
52
|
+
])
|
|
53
|
+
})
|
|
54
|
+
|
|
55
|
+
it('returns empty array for a stage with no states', () => {
|
|
56
|
+
expect(getValidStatesForStage(TEST_CRM_PIPELINE, 'proposal')).toEqual([])
|
|
57
|
+
})
|
|
58
|
+
|
|
59
|
+
it('returns empty array for an unknown stage key', () => {
|
|
60
|
+
expect(getValidStatesForStage(TEST_CRM_PIPELINE, 'unknown_stage')).toEqual([])
|
|
61
|
+
})
|
|
62
|
+
|
|
63
|
+
it('is case-sensitive', () => {
|
|
64
|
+
expect(getValidStatesForStage(TEST_CRM_PIPELINE, 'Interested')).toEqual([])
|
|
65
|
+
})
|
|
66
|
+
|
|
67
|
+
it('works with the lead-gen member pipeline for known stage', () => {
|
|
68
|
+
const states = getValidStatesForStage(ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE, 'outreach')
|
|
69
|
+
expect(states.length).toBeGreaterThan(0)
|
|
70
|
+
expect(states.map((s) => s.stateKey)).toContain('personalized')
|
|
71
|
+
})
|
|
72
|
+
})
|
|
73
|
+
|
|
74
|
+
describe('findPipeline', () => {
|
|
75
|
+
it('finds a pipeline by pipelineKey when present', () => {
|
|
76
|
+
const found = findPipeline(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-member'], 'lead-gen')
|
|
77
|
+
expect(found).toBe(ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE)
|
|
78
|
+
})
|
|
79
|
+
|
|
80
|
+
it('returns undefined for an unknown pipelineKey', () => {
|
|
81
|
+
const found = findPipeline(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-member'], 'crm')
|
|
82
|
+
expect(found).toBeUndefined()
|
|
83
|
+
})
|
|
84
|
+
|
|
85
|
+
it('returns undefined for an empty array', () => {
|
|
86
|
+
const found = findPipeline([], 'lead-gen')
|
|
87
|
+
expect(found).toBeUndefined()
|
|
88
|
+
})
|
|
89
|
+
})
|
|
90
|
+
|
|
91
|
+
describe('LEAD_GEN_PIPELINE_DEFINITIONS', () => {
|
|
92
|
+
it('has entries for acq.list-member and acq.list-company', () => {
|
|
93
|
+
expect(LEAD_GEN_PIPELINE_DEFINITIONS).toHaveProperty('acq.list-member')
|
|
94
|
+
expect(LEAD_GEN_PIPELINE_DEFINITIONS).toHaveProperty('acq.list-company')
|
|
95
|
+
})
|
|
96
|
+
|
|
97
|
+
it('acq.list-member entry contains the ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE', () => {
|
|
98
|
+
expect(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-member']).toContain(ACQ_LIST_MEMBERS_LEAD_GEN_PIPELINE)
|
|
99
|
+
})
|
|
100
|
+
|
|
101
|
+
it('acq.list-company entry contains the ACQ_LIST_COMPANIES_LEAD_GEN_PIPELINE', () => {
|
|
102
|
+
expect(LEAD_GEN_PIPELINE_DEFINITIONS['acq.list-company']).toContain(ACQ_LIST_COMPANIES_LEAD_GEN_PIPELINE)
|
|
103
|
+
})
|
|
104
|
+
})
|