@elevasis/core 0.15.0 → 0.15.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.
- package/dist/index.d.ts +60 -0
- package/dist/index.js +198 -1
- package/dist/organization-model/index.d.ts +60 -0
- package/dist/organization-model/index.js +198 -1
- package/dist/test-utils/index.d.ts +399 -363
- package/dist/test-utils/index.js +198 -1
- package/package.json +3 -3
- package/src/_gen/__tests__/__snapshots__/contracts.md.snap +826 -703
- package/src/business/acquisition/activity-events.ts +12 -3
- package/src/business/acquisition/api-schemas.test.ts +315 -4
- package/src/business/acquisition/api-schemas.ts +1192 -1097
- package/src/business/acquisition/build-templates.ts +44 -0
- package/src/business/acquisition/crm-next-action.test.ts +262 -0
- package/src/business/acquisition/crm-next-action.ts +220 -0
- package/src/business/acquisition/crm-priority.test.ts +216 -0
- package/src/business/acquisition/crm-priority.ts +349 -0
- package/src/business/acquisition/crm-state-actions.test.ts +12 -21
- package/src/business/acquisition/deal-ownership.test.ts +351 -0
- package/src/business/acquisition/deal-ownership.ts +120 -0
- package/src/business/acquisition/derive-actions.test.ts +101 -37
- package/src/business/acquisition/derive-actions.ts +102 -87
- package/src/business/acquisition/index.ts +10 -0
- package/src/business/acquisition/types.ts +51 -8
- package/src/execution/engine/index.ts +4 -3
- package/src/execution/engine/tools/lead-service-types.ts +44 -30
- package/src/execution/engine/tools/platform/acquisition/list-tools.ts +6 -5
- package/src/execution/engine/tools/platform/acquisition/types.ts +3 -1
- package/src/execution/engine/tools/registry.ts +4 -3
- package/src/execution/engine/tools/tool-maps.ts +3 -1
- package/src/organization-model/domains/prospecting.ts +204 -1
- package/src/organization-model/domains/sales.test.ts +29 -0
- package/src/organization-model/domains/sales.ts +102 -0
- package/src/organization-model/types.ts +2 -2
- package/src/platform/constants/versions.ts +1 -1
- package/src/reference/_generated/contracts.md +826 -703
- package/src/supabase/database.types.ts +2978 -2958
|
@@ -0,0 +1,216 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest'
|
|
2
|
+
import { evaluateCrmDealPriority, resolveCrmPriorityRuleConfig } from './crm-priority'
|
|
3
|
+
|
|
4
|
+
// ---------------------------------------------------------------------------
|
|
5
|
+
// Helpers
|
|
6
|
+
// ---------------------------------------------------------------------------
|
|
7
|
+
|
|
8
|
+
function makeEvent(type: string, timestamp: string): Record<string, unknown> {
|
|
9
|
+
return { type, timestamp }
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
const INBOUND = 'reply_received'
|
|
13
|
+
const OUTBOUND = 'reply_sent_to_lead'
|
|
14
|
+
const NOW = '2026-04-30T12:00:00.000Z'
|
|
15
|
+
|
|
16
|
+
function makeDeal(
|
|
17
|
+
overrides: Partial<{
|
|
18
|
+
stage_key: string | null
|
|
19
|
+
state_key: string | null
|
|
20
|
+
activity_log: unknown[]
|
|
21
|
+
updated_at: string
|
|
22
|
+
created_at: string
|
|
23
|
+
}> = {}
|
|
24
|
+
) {
|
|
25
|
+
return {
|
|
26
|
+
stage_key: overrides.stage_key ?? 'interested',
|
|
27
|
+
state_key: overrides.state_key ?? 'discovery_replied',
|
|
28
|
+
activity_log: overrides.activity_log ?? [],
|
|
29
|
+
updated_at: overrides.updated_at ?? NOW,
|
|
30
|
+
created_at: overrides.created_at ?? NOW
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
// ---------------------------------------------------------------------------
|
|
35
|
+
// Closed deals → closed_low
|
|
36
|
+
// ---------------------------------------------------------------------------
|
|
37
|
+
|
|
38
|
+
describe('evaluateCrmDealPriority — closed deals', () => {
|
|
39
|
+
it('returns closed_low for closed_won stage', () => {
|
|
40
|
+
const result = evaluateCrmDealPriority(makeDeal({ stage_key: 'closed_won' }), { now: NOW })
|
|
41
|
+
expect(result.bucketKey).toBe('closed_low')
|
|
42
|
+
})
|
|
43
|
+
|
|
44
|
+
it('returns closed_low for closed_lost stage', () => {
|
|
45
|
+
const result = evaluateCrmDealPriority(makeDeal({ stage_key: 'closed_lost' }), { now: NOW })
|
|
46
|
+
expect(result.bucketKey).toBe('closed_low')
|
|
47
|
+
})
|
|
48
|
+
})
|
|
49
|
+
|
|
50
|
+
// ---------------------------------------------------------------------------
|
|
51
|
+
// needs_response — ownership === 'us'
|
|
52
|
+
// ---------------------------------------------------------------------------
|
|
53
|
+
|
|
54
|
+
describe('evaluateCrmDealPriority — needs_response (ownership === us)', () => {
|
|
55
|
+
it('returns needs_response when lead replied last (inbound after outbound)', () => {
|
|
56
|
+
const result = evaluateCrmDealPriority(
|
|
57
|
+
makeDeal({
|
|
58
|
+
state_key: 'discovery_replied',
|
|
59
|
+
activity_log: [makeEvent(OUTBOUND, '2026-04-28T10:00:00Z'), makeEvent(INBOUND, '2026-04-29T10:00:00Z')]
|
|
60
|
+
}),
|
|
61
|
+
{ now: NOW }
|
|
62
|
+
)
|
|
63
|
+
expect(result.bucketKey).toBe('needs_response')
|
|
64
|
+
})
|
|
65
|
+
|
|
66
|
+
it('returns needs_response when only inbound event present', () => {
|
|
67
|
+
const result = evaluateCrmDealPriority(
|
|
68
|
+
makeDeal({
|
|
69
|
+
state_key: 'discovery_replied',
|
|
70
|
+
activity_log: [makeEvent(INBOUND, '2026-04-29T10:00:00Z')]
|
|
71
|
+
}),
|
|
72
|
+
{ now: NOW }
|
|
73
|
+
)
|
|
74
|
+
expect(result.bucketKey).toBe('needs_response')
|
|
75
|
+
})
|
|
76
|
+
|
|
77
|
+
it('returns needs_response regardless of state_key when ownership is us', () => {
|
|
78
|
+
// reply_sent is normally a "we acted" state, but if a new inbound arrives ownership flips
|
|
79
|
+
const result = evaluateCrmDealPriority(
|
|
80
|
+
makeDeal({
|
|
81
|
+
state_key: 'reply_sent',
|
|
82
|
+
activity_log: [makeEvent(OUTBOUND, '2026-04-27T10:00:00Z'), makeEvent(INBOUND, '2026-04-29T10:00:00Z')]
|
|
83
|
+
}),
|
|
84
|
+
{ now: NOW }
|
|
85
|
+
)
|
|
86
|
+
expect(result.bucketKey).toBe('needs_response')
|
|
87
|
+
})
|
|
88
|
+
})
|
|
89
|
+
|
|
90
|
+
// ---------------------------------------------------------------------------
|
|
91
|
+
// waiting — ownership === 'them' AND age < staleAfterDays
|
|
92
|
+
// ---------------------------------------------------------------------------
|
|
93
|
+
|
|
94
|
+
describe('evaluateCrmDealPriority — waiting (ownership === them, not stale)', () => {
|
|
95
|
+
it('returns waiting when we replied last and age is below staleAfterDays', () => {
|
|
96
|
+
const result = evaluateCrmDealPriority(
|
|
97
|
+
makeDeal({
|
|
98
|
+
state_key: 'reply_sent',
|
|
99
|
+
// No followUpAfterDaysByStateKey entry for reply_sent within window
|
|
100
|
+
activity_log: [makeEvent(OUTBOUND, '2026-04-29T10:00:00Z')]
|
|
101
|
+
}),
|
|
102
|
+
{ now: NOW }
|
|
103
|
+
)
|
|
104
|
+
// Age = 1 day, staleAfterDays = 14 → waiting
|
|
105
|
+
// reply_sent has followUpAfterDaysByStateKey=3, nextActionAt is 2026-05-02, not yet due
|
|
106
|
+
expect(result.bucketKey).toBe('waiting')
|
|
107
|
+
})
|
|
108
|
+
|
|
109
|
+
it('returns follow_up_due when follow-up window elapses (ownership them)', () => {
|
|
110
|
+
// discovery_link_sent: followUpAfterDaysByStateKey = 3
|
|
111
|
+
// Activity at 2026-04-26, now = 2026-04-30 → age 4 days → nextActionAt = 2026-04-29 (past)
|
|
112
|
+
// updated_at set before activity so latestActivityAt comes from activity_log
|
|
113
|
+
const result = evaluateCrmDealPriority(
|
|
114
|
+
makeDeal({
|
|
115
|
+
state_key: 'discovery_link_sent',
|
|
116
|
+
activity_log: [makeEvent(OUTBOUND, '2026-04-26T10:00:00Z')],
|
|
117
|
+
updated_at: '2026-04-25T00:00:00Z',
|
|
118
|
+
created_at: '2026-04-25T00:00:00Z'
|
|
119
|
+
}),
|
|
120
|
+
{ now: NOW }
|
|
121
|
+
)
|
|
122
|
+
expect(result.bucketKey).toBe('follow_up_due')
|
|
123
|
+
})
|
|
124
|
+
})
|
|
125
|
+
|
|
126
|
+
// ---------------------------------------------------------------------------
|
|
127
|
+
// stale — ownership === 'them' AND age >= staleAfterDays
|
|
128
|
+
// ---------------------------------------------------------------------------
|
|
129
|
+
|
|
130
|
+
describe('evaluateCrmDealPriority — stale (ownership them, age >= staleAfterDays)', () => {
|
|
131
|
+
it('returns stale when we sent last and age >= staleAfterDays', () => {
|
|
132
|
+
// Activity at 2026-04-10, now = 2026-04-30 → age 20 days >= staleAfterDays 14
|
|
133
|
+
// updated_at set before activity so latestActivityAt comes from activity_log
|
|
134
|
+
const result = evaluateCrmDealPriority(
|
|
135
|
+
makeDeal({
|
|
136
|
+
state_key: 'reply_sent',
|
|
137
|
+
activity_log: [makeEvent(OUTBOUND, '2026-04-10T10:00:00Z')],
|
|
138
|
+
updated_at: '2026-04-09T00:00:00Z',
|
|
139
|
+
created_at: '2026-04-09T00:00:00Z'
|
|
140
|
+
}),
|
|
141
|
+
{ now: NOW }
|
|
142
|
+
)
|
|
143
|
+
expect(result.bucketKey).toBe('stale')
|
|
144
|
+
})
|
|
145
|
+
})
|
|
146
|
+
|
|
147
|
+
// ---------------------------------------------------------------------------
|
|
148
|
+
// No activity (ownership null)
|
|
149
|
+
// ---------------------------------------------------------------------------
|
|
150
|
+
|
|
151
|
+
describe('evaluateCrmDealPriority — no ownership signal', () => {
|
|
152
|
+
it('returns waiting when no activity log', () => {
|
|
153
|
+
const result = evaluateCrmDealPriority(makeDeal({ activity_log: [] }), { now: NOW })
|
|
154
|
+
expect(result.bucketKey).toBe('waiting')
|
|
155
|
+
})
|
|
156
|
+
|
|
157
|
+
it('returns stale when no activity and updated_at is very old', () => {
|
|
158
|
+
const result = evaluateCrmDealPriority(
|
|
159
|
+
{
|
|
160
|
+
stage_key: 'interested',
|
|
161
|
+
state_key: 'discovery_replied',
|
|
162
|
+
activity_log: [],
|
|
163
|
+
updated_at: '2026-01-01T00:00:00Z',
|
|
164
|
+
created_at: '2026-01-01T00:00:00Z'
|
|
165
|
+
},
|
|
166
|
+
{ now: NOW }
|
|
167
|
+
)
|
|
168
|
+
expect(result.bucketKey).toBe('stale')
|
|
169
|
+
})
|
|
170
|
+
})
|
|
171
|
+
|
|
172
|
+
// ---------------------------------------------------------------------------
|
|
173
|
+
// disabled config
|
|
174
|
+
// ---------------------------------------------------------------------------
|
|
175
|
+
|
|
176
|
+
describe('evaluateCrmDealPriority — disabled config', () => {
|
|
177
|
+
it('returns waiting when config.enabled is false', () => {
|
|
178
|
+
const config = resolveCrmPriorityRuleConfig({ enabled: false })
|
|
179
|
+
const result = evaluateCrmDealPriority(makeDeal({ activity_log: [makeEvent(INBOUND, '2026-04-29T10:00:00Z')] }), {
|
|
180
|
+
now: NOW,
|
|
181
|
+
config
|
|
182
|
+
})
|
|
183
|
+
expect(result.bucketKey).toBe('waiting')
|
|
184
|
+
})
|
|
185
|
+
})
|
|
186
|
+
|
|
187
|
+
// ---------------------------------------------------------------------------
|
|
188
|
+
// resolveCrmPriorityRuleConfig — override merging
|
|
189
|
+
// ---------------------------------------------------------------------------
|
|
190
|
+
|
|
191
|
+
describe('resolveCrmPriorityRuleConfig — override merging', () => {
|
|
192
|
+
it('merges staleAfterDays and closedStageKeys from org override', () => {
|
|
193
|
+
const config = resolveCrmPriorityRuleConfig({
|
|
194
|
+
crm: {
|
|
195
|
+
priority: {
|
|
196
|
+
staleAfterDays: 7,
|
|
197
|
+
closedStageKeys: ['custom_closed']
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
})
|
|
201
|
+
expect(config.staleAfterDays).toBe(7)
|
|
202
|
+
expect(config.closedStageKeys).toEqual(['custom_closed'])
|
|
203
|
+
})
|
|
204
|
+
|
|
205
|
+
it('falls back to defaults for invalid input', () => {
|
|
206
|
+
const config = resolveCrmPriorityRuleConfig('not-valid')
|
|
207
|
+
expect(config.staleAfterDays).toBe(14)
|
|
208
|
+
expect(config.enabled).toBe(true)
|
|
209
|
+
})
|
|
210
|
+
|
|
211
|
+
it('does not have needsResponseStateKeys or needsResponseActivityTypes', () => {
|
|
212
|
+
const config = resolveCrmPriorityRuleConfig()
|
|
213
|
+
expect(config).not.toHaveProperty('needsResponseStateKeys')
|
|
214
|
+
expect(config).not.toHaveProperty('needsResponseActivityTypes')
|
|
215
|
+
})
|
|
216
|
+
})
|
|
@@ -0,0 +1,349 @@
|
|
|
1
|
+
import { z } from 'zod'
|
|
2
|
+
import {
|
|
3
|
+
DEFAULT_CRM_PRIORITY_RULE_CONFIG,
|
|
4
|
+
type CrmPriorityBucketDefinition,
|
|
5
|
+
type CrmPriorityBucketKey,
|
|
6
|
+
type CrmPriorityRuleConfig
|
|
7
|
+
} from '../../organization-model/domains/sales'
|
|
8
|
+
import { getDealOwnership } from './deal-ownership'
|
|
9
|
+
import type { DealPriority } from './types'
|
|
10
|
+
|
|
11
|
+
export interface CrmPriorityInput {
|
|
12
|
+
stage_key: string | null
|
|
13
|
+
state_key: string | null
|
|
14
|
+
activity_log: unknown
|
|
15
|
+
updated_at: string
|
|
16
|
+
created_at: string
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export interface EvaluateCrmDealPriorityOptions {
|
|
20
|
+
config?: CrmPriorityRuleConfig | ResolvedCrmPriorityRuleConfig
|
|
21
|
+
now?: Date | string
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export const CrmPriorityBucketKeySchema = z.enum(['needs_response', 'follow_up_due', 'waiting', 'stale', 'closed_low'])
|
|
25
|
+
|
|
26
|
+
export const CrmPriorityBucketOverrideSchema = z
|
|
27
|
+
.object({
|
|
28
|
+
label: z.string().trim().min(1).max(80).optional(),
|
|
29
|
+
color: z.string().trim().min(1).max(40).optional(),
|
|
30
|
+
rank: z.number().int().min(0).max(10000).optional()
|
|
31
|
+
})
|
|
32
|
+
.strict()
|
|
33
|
+
|
|
34
|
+
const CrmPriorityBucketDefinitionSchema = z
|
|
35
|
+
.object({
|
|
36
|
+
bucketKey: CrmPriorityBucketKeySchema,
|
|
37
|
+
label: z.string(),
|
|
38
|
+
rank: z.number().int(),
|
|
39
|
+
color: z.string()
|
|
40
|
+
})
|
|
41
|
+
.strict()
|
|
42
|
+
|
|
43
|
+
const CrmPriorityRuleConfigSchema = z
|
|
44
|
+
.object({
|
|
45
|
+
enabled: z.boolean().optional(),
|
|
46
|
+
buckets: z.array(CrmPriorityBucketDefinitionSchema),
|
|
47
|
+
closedStageKeys: z.array(z.string()),
|
|
48
|
+
followUpAfterDaysByStateKey: z.record(z.string(), z.number().int()),
|
|
49
|
+
staleAfterDays: z.number().int()
|
|
50
|
+
})
|
|
51
|
+
.strict()
|
|
52
|
+
|
|
53
|
+
export const CrmPriorityOverrideSchema = z
|
|
54
|
+
.object({
|
|
55
|
+
enabled: z.boolean().optional(),
|
|
56
|
+
staleAfterDays: z.number().int().min(1).max(365).optional(),
|
|
57
|
+
bucketOrder: z.array(CrmPriorityBucketKeySchema).optional(),
|
|
58
|
+
buckets: z.partialRecord(CrmPriorityBucketKeySchema, CrmPriorityBucketOverrideSchema).optional(),
|
|
59
|
+
followUpAfterDaysByStateKey: z
|
|
60
|
+
.record(z.string().trim().min(1).max(120), z.number().int().min(0).max(365))
|
|
61
|
+
.optional(),
|
|
62
|
+
closedStageKeys: z.array(z.string().trim().min(1).max(120)).optional()
|
|
63
|
+
})
|
|
64
|
+
.strict()
|
|
65
|
+
|
|
66
|
+
export type CrmPriorityBucketOverride = z.infer<typeof CrmPriorityBucketOverrideSchema>
|
|
67
|
+
export type CrmPriorityOverride = z.infer<typeof CrmPriorityOverrideSchema>
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* Resolved CRM priority settings used by evaluators.
|
|
71
|
+
*
|
|
72
|
+
* When enabled is false, priority evaluation is intentionally neutral: all deals
|
|
73
|
+
* receive the configured waiting bucket and no follow-up/stale urgency is computed.
|
|
74
|
+
*/
|
|
75
|
+
export type ResolvedCrmPriorityRuleConfig = CrmPriorityRuleConfig & {
|
|
76
|
+
enabled: boolean
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
interface ParsedActivity {
|
|
80
|
+
type: string | null
|
|
81
|
+
occurredAt: string
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
const MS_PER_DAY = 24 * 60 * 60 * 1000
|
|
85
|
+
|
|
86
|
+
export function evaluateCrmDealPriority(
|
|
87
|
+
input: CrmPriorityInput,
|
|
88
|
+
options: EvaluateCrmDealPriorityOptions = {}
|
|
89
|
+
): DealPriority {
|
|
90
|
+
const config = resolveCrmPriorityRuleConfig(options.config)
|
|
91
|
+
const now = parseDate(options.now ?? new Date()) ?? new Date()
|
|
92
|
+
const latestActivityAt = getLatestActivityAt(input)
|
|
93
|
+
const nextActionAt = getNextActionAt(input.state_key, latestActivityAt, config)
|
|
94
|
+
|
|
95
|
+
if (!config.enabled) {
|
|
96
|
+
return buildPriority('waiting', config, {
|
|
97
|
+
reason: 'CRM priority evaluation is disabled.',
|
|
98
|
+
latestActivityAt,
|
|
99
|
+
nextActionAt: null
|
|
100
|
+
})
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
if (input.stage_key && config.closedStageKeys.includes(input.stage_key)) {
|
|
104
|
+
return buildPriority('closed_low', config, {
|
|
105
|
+
reason: `Deal is in ${input.stage_key}.`,
|
|
106
|
+
latestActivityAt,
|
|
107
|
+
nextActionAt: null
|
|
108
|
+
})
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
const ownership = getDealOwnership(input)
|
|
112
|
+
|
|
113
|
+
if (ownership === 'us') {
|
|
114
|
+
return buildPriority('needs_response', config, {
|
|
115
|
+
reason: 'Lead replied last — we owe the next move.',
|
|
116
|
+
latestActivityAt,
|
|
117
|
+
nextActionAt
|
|
118
|
+
})
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ownership === 'them' or null: check age against staleAfterDays first
|
|
122
|
+
if (latestActivityAt) {
|
|
123
|
+
const latestActivityDate = parseDate(latestActivityAt)
|
|
124
|
+
if (latestActivityDate && now.getTime() - latestActivityDate.getTime() >= config.staleAfterDays * MS_PER_DAY) {
|
|
125
|
+
return buildPriority('stale', config, {
|
|
126
|
+
reason: `No meaningful activity in ${config.staleAfterDays} days.`,
|
|
127
|
+
latestActivityAt,
|
|
128
|
+
nextActionAt: null
|
|
129
|
+
})
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
if (ownership === 'them') {
|
|
134
|
+
if (nextActionAt) {
|
|
135
|
+
const nextActionDate = parseDate(nextActionAt)
|
|
136
|
+
if (nextActionDate && nextActionDate <= now) {
|
|
137
|
+
return buildPriority('follow_up_due', config, {
|
|
138
|
+
reason: 'Configured follow-up window has elapsed.',
|
|
139
|
+
latestActivityAt,
|
|
140
|
+
nextActionAt
|
|
141
|
+
})
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
return buildPriority('waiting', config, {
|
|
146
|
+
reason: 'Waiting on their reply.',
|
|
147
|
+
latestActivityAt,
|
|
148
|
+
nextActionAt
|
|
149
|
+
})
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
// ownership === null (no inbound/outbound events): fall back to follow_up/waiting
|
|
153
|
+
if (nextActionAt) {
|
|
154
|
+
const nextActionDate = parseDate(nextActionAt)
|
|
155
|
+
if (nextActionDate && nextActionDate <= now) {
|
|
156
|
+
return buildPriority('follow_up_due', config, {
|
|
157
|
+
reason: 'Configured follow-up window has elapsed.',
|
|
158
|
+
latestActivityAt,
|
|
159
|
+
nextActionAt
|
|
160
|
+
})
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
return buildPriority('waiting', config, {
|
|
164
|
+
reason: 'Waiting until the configured next action time.',
|
|
165
|
+
latestActivityAt,
|
|
166
|
+
nextActionAt
|
|
167
|
+
})
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
return buildPriority('waiting', config, {
|
|
171
|
+
reason: 'No immediate response or follow-up is due.',
|
|
172
|
+
latestActivityAt,
|
|
173
|
+
nextActionAt: null
|
|
174
|
+
})
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
export function resolveCrmPriorityRuleConfig(input?: unknown): ResolvedCrmPriorityRuleConfig {
|
|
178
|
+
const candidate = extractPriorityOverride(input)
|
|
179
|
+
const fullConfigResult = CrmPriorityRuleConfigSchema.safeParse(candidate)
|
|
180
|
+
|
|
181
|
+
if (fullConfigResult.success) return buildResolvedConfig(fullConfigResult.data)
|
|
182
|
+
|
|
183
|
+
const result = CrmPriorityOverrideSchema.safeParse(candidate)
|
|
184
|
+
|
|
185
|
+
if (!result.success) return buildDefaultResolvedConfig()
|
|
186
|
+
|
|
187
|
+
return mergeCrmPriorityOverride(result.data)
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
function mergeCrmPriorityOverride(override: CrmPriorityOverride): ResolvedCrmPriorityRuleConfig {
|
|
191
|
+
const defaultConfig = buildDefaultResolvedConfig()
|
|
192
|
+
const bucketOverrides = override.buckets ?? {}
|
|
193
|
+
const orderedBucketKeys = override.bucketOrder ? [...new Set(override.bucketOrder)] : []
|
|
194
|
+
const orderRanks = new Map<CrmPriorityBucketKey, number>()
|
|
195
|
+
|
|
196
|
+
orderedBucketKeys.forEach((bucketKey, index) => {
|
|
197
|
+
orderRanks.set(bucketKey, (index + 1) * 10)
|
|
198
|
+
})
|
|
199
|
+
|
|
200
|
+
const buckets = defaultConfig.buckets
|
|
201
|
+
.map((bucket) => {
|
|
202
|
+
const bucketOverride = bucketOverrides[bucket.bucketKey]
|
|
203
|
+
|
|
204
|
+
return {
|
|
205
|
+
...bucket,
|
|
206
|
+
rank: bucketOverride?.rank ?? orderRanks.get(bucket.bucketKey) ?? bucket.rank,
|
|
207
|
+
label: bucketOverride?.label ?? bucket.label,
|
|
208
|
+
color: bucketOverride?.color ?? bucket.color
|
|
209
|
+
}
|
|
210
|
+
})
|
|
211
|
+
.sort((a, b) => a.rank - b.rank)
|
|
212
|
+
|
|
213
|
+
return {
|
|
214
|
+
enabled: override.enabled ?? defaultConfig.enabled,
|
|
215
|
+
buckets,
|
|
216
|
+
closedStageKeys: override.closedStageKeys ?? defaultConfig.closedStageKeys,
|
|
217
|
+
followUpAfterDaysByStateKey: {
|
|
218
|
+
...defaultConfig.followUpAfterDaysByStateKey,
|
|
219
|
+
...(override.followUpAfterDaysByStateKey ?? {})
|
|
220
|
+
},
|
|
221
|
+
staleAfterDays: override.staleAfterDays ?? defaultConfig.staleAfterDays
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
function buildDefaultResolvedConfig(): ResolvedCrmPriorityRuleConfig {
|
|
226
|
+
return buildResolvedConfig(DEFAULT_CRM_PRIORITY_RULE_CONFIG)
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
function buildResolvedConfig(config: CrmPriorityRuleConfig & { enabled?: boolean }): ResolvedCrmPriorityRuleConfig {
|
|
230
|
+
return {
|
|
231
|
+
enabled: config.enabled ?? true,
|
|
232
|
+
buckets: config.buckets.map((bucket) => ({ ...bucket })),
|
|
233
|
+
closedStageKeys: [...config.closedStageKeys],
|
|
234
|
+
followUpAfterDaysByStateKey: { ...config.followUpAfterDaysByStateKey },
|
|
235
|
+
staleAfterDays: config.staleAfterDays
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
function extractPriorityOverride(input: unknown): unknown {
|
|
240
|
+
if (!isPlainRecord(input)) return input
|
|
241
|
+
|
|
242
|
+
const crm = input.crm
|
|
243
|
+
if (isPlainRecord(crm) && Object.prototype.hasOwnProperty.call(crm, 'priority')) {
|
|
244
|
+
return crm.priority
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
return input
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
function isPlainRecord(value: unknown): value is Record<string, unknown> {
|
|
251
|
+
return Boolean(value) && typeof value === 'object' && !Array.isArray(value)
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
function buildPriority(
|
|
255
|
+
bucketKey: CrmPriorityBucketKey,
|
|
256
|
+
config: CrmPriorityRuleConfig,
|
|
257
|
+
values: Pick<DealPriority, 'reason' | 'latestActivityAt' | 'nextActionAt'>
|
|
258
|
+
): DealPriority {
|
|
259
|
+
const bucket = findBucket(config.buckets, bucketKey)
|
|
260
|
+
|
|
261
|
+
return {
|
|
262
|
+
bucketKey: bucket.bucketKey,
|
|
263
|
+
rank: bucket.rank,
|
|
264
|
+
label: bucket.label,
|
|
265
|
+
color: bucket.color,
|
|
266
|
+
reason: values.reason,
|
|
267
|
+
latestActivityAt: values.latestActivityAt,
|
|
268
|
+
nextActionAt: values.nextActionAt
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
|
|
272
|
+
function findBucket(
|
|
273
|
+
buckets: CrmPriorityBucketDefinition[],
|
|
274
|
+
bucketKey: CrmPriorityBucketKey
|
|
275
|
+
): CrmPriorityBucketDefinition {
|
|
276
|
+
const bucket = buckets.find((candidate) => candidate.bucketKey === bucketKey)
|
|
277
|
+
if (bucket) return bucket
|
|
278
|
+
|
|
279
|
+
return {
|
|
280
|
+
bucketKey,
|
|
281
|
+
label: bucketKey,
|
|
282
|
+
rank: Number.MAX_SAFE_INTEGER,
|
|
283
|
+
color: 'gray'
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
function getLatestActivityAt(input: CrmPriorityInput): string | null {
|
|
288
|
+
const timestamps = getParsedActivities(input.activity_log).map((activity) => activity.occurredAt)
|
|
289
|
+
const fallback = normalizeTimestamp(input.updated_at) ?? normalizeTimestamp(input.created_at)
|
|
290
|
+
if (fallback) timestamps.push(fallback)
|
|
291
|
+
|
|
292
|
+
return timestamps.sort((a, b) => new Date(b).getTime() - new Date(a).getTime())[0] ?? null
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
function getParsedActivities(activityLog: unknown): ParsedActivity[] {
|
|
296
|
+
if (!Array.isArray(activityLog)) return []
|
|
297
|
+
|
|
298
|
+
const parsed: ParsedActivity[] = []
|
|
299
|
+
|
|
300
|
+
for (const entry of activityLog) {
|
|
301
|
+
if (!entry || typeof entry !== 'object' || Array.isArray(entry)) continue
|
|
302
|
+
|
|
303
|
+
const record = entry as Record<string, unknown>
|
|
304
|
+
const occurredAt =
|
|
305
|
+
normalizeTimestamp(record.timestamp) ??
|
|
306
|
+
normalizeTimestamp(record.occurredAt) ??
|
|
307
|
+
normalizeTimestamp(record.createdAt) ??
|
|
308
|
+
normalizeTimestamp(record.updatedAt) ??
|
|
309
|
+
normalizeTimestamp(record.sentAt)
|
|
310
|
+
|
|
311
|
+
if (!occurredAt) continue
|
|
312
|
+
|
|
313
|
+
parsed.push({
|
|
314
|
+
type: typeof record.type === 'string' ? record.type : null,
|
|
315
|
+
occurredAt
|
|
316
|
+
})
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
return parsed
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
function getNextActionAt(
|
|
323
|
+
stateKey: string | null,
|
|
324
|
+
latestActivityAt: string | null,
|
|
325
|
+
config: CrmPriorityRuleConfig
|
|
326
|
+
): string | null {
|
|
327
|
+
if (!stateKey || !latestActivityAt) return null
|
|
328
|
+
|
|
329
|
+
const days = config.followUpAfterDaysByStateKey[stateKey]
|
|
330
|
+
if (days === undefined) return null
|
|
331
|
+
|
|
332
|
+
const latestActivityDate = parseDate(latestActivityAt)
|
|
333
|
+
if (!latestActivityDate) return null
|
|
334
|
+
|
|
335
|
+
return new Date(latestActivityDate.getTime() + days * MS_PER_DAY).toISOString()
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
function normalizeTimestamp(value: unknown): string | null {
|
|
339
|
+
const date = parseDate(value)
|
|
340
|
+
return date ? date.toISOString() : null
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
function parseDate(value: unknown): Date | null {
|
|
344
|
+
if (value instanceof Date) return Number.isNaN(value.getTime()) ? null : value
|
|
345
|
+
if (typeof value !== 'string') return null
|
|
346
|
+
|
|
347
|
+
const date = new Date(value)
|
|
348
|
+
return Number.isNaN(date.getTime()) ? null : date
|
|
349
|
+
}
|
|
@@ -39,27 +39,18 @@ describe('CRM deal action derivation', () => {
|
|
|
39
39
|
expectedAction
|
|
40
40
|
]
|
|
41
41
|
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
expect(deriveActions(deal('interested', 'discovery_nudging'))).toEqual([
|
|
55
|
-
{ key: 'move_to_proposal', label: 'Move to Proposal' },
|
|
56
|
-
{ key: 'move_to_closed_lost', label: 'Close Lost' },
|
|
57
|
-
{ key: 'move_to_nurturing', label: 'Move to Nurturing' },
|
|
58
|
-
{ key: 'send_reply', label: 'Send Reply', payloadSchema: expect.any(Object) },
|
|
59
|
-
{ key: 'send_nudge', label: 'Send Nudge' },
|
|
60
|
-
{ key: 'mark_no_show', label: 'Mark No-Show' }
|
|
61
|
-
])
|
|
62
|
-
})
|
|
42
|
+
expect(deriveActions(deal('interested', stateKey))).toEqual(expected)
|
|
43
|
+
})
|
|
44
|
+
|
|
45
|
+
it('adds nudge and no-show actions for interested/discovery_nudging', () => {
|
|
46
|
+
expect(deriveActions(deal('interested', 'discovery_nudging'))).toEqual([
|
|
47
|
+
{ key: 'move_to_proposal', label: 'Move to Proposal' },
|
|
48
|
+
{ key: 'move_to_closed_lost', label: 'Close Lost' },
|
|
49
|
+
{ key: 'move_to_nurturing', label: 'Move to Nurturing' },
|
|
50
|
+
{ key: 'send_nudge', label: 'Send Nudge' },
|
|
51
|
+
{ key: 'mark_no_show', label: 'Mark No-Show' }
|
|
52
|
+
])
|
|
53
|
+
})
|
|
63
54
|
|
|
64
55
|
it('derives exact proposal and closing transitions', () => {
|
|
65
56
|
expect(deriveActions(deal('proposal'))).toEqual([
|