@talkpilot/core-db 1.1.18 → 1.2.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/.cursor/rules/development.mdc +65 -65
- package/DEVELOPMENT.md +98 -98
- package/README.md +160 -160
- package/dist/talkpilot/calls/calls.getters.d.ts +2 -1
- package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.getters.js +176 -0
- package/dist/talkpilot/calls/calls.getters.js.map +1 -1
- package/dist/talkpilot/calls/calls.types.d.ts +49 -2
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.types.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +1 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +13 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +2 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
- package/jest.config.js +19 -19
- package/package.json +45 -45
- package/src/__tests__/setup.ts +20 -20
- package/src/connection.ts +42 -42
- package/src/index.ts +16 -16
- package/src/municipal/__tests__/validation.spec.ts +62 -62
- package/src/municipal/cities/cities.getters.ts +50 -50
- package/src/municipal/cities/cities.types.ts +11 -11
- package/src/municipal/cities/index.ts +2 -2
- package/src/municipal/departmentsSubjects/departmentsSubjects.getters.ts +282 -282
- package/src/municipal/departmentsSubjects/departmentsSubjects.types.ts +72 -72
- package/src/municipal/departmentsSubjects/index.ts +9 -9
- package/src/municipal/index.ts +21 -21
- package/src/municipal/mongodb-client.ts +61 -61
- package/src/municipal/streets/index.ts +2 -2
- package/src/municipal/streets/streets.getters.ts +125 -125
- package/src/municipal/streets/streets.types.ts +18 -18
- package/src/municipal/systemInstructions/__tests__/getters.spec.ts +113 -113
- package/src/municipal/systemInstructions/__tests__/setters.spec.ts +274 -274
- package/src/municipal/systemInstructions/index.ts +7 -7
- package/src/municipal/systemInstructions/instructions.getters.ts +57 -57
- package/src/municipal/systemInstructions/instructions.setters.ts +119 -119
- package/src/municipal/systemInstructions/instructions.types.ts +30 -30
- package/src/municipal/tickets/__tests__/tickets.getters.spec.ts +66 -66
- package/src/municipal/tickets/index.ts +2 -2
- package/src/municipal/tickets/tickets.getters.ts +261 -261
- package/src/municipal/tickets/tickets.types.ts +43 -43
- package/src/municipal/utils/types.ts +11 -11
- package/src/talkpilot/__tests__/db.spec.ts +38 -38
- package/src/talkpilot/__tests__/mongodb-client.spec.ts +18 -18
- package/src/talkpilot/__tests__/validation.spec.ts +68 -68
- package/src/talkpilot/agents/__tests__/agents.getters.spec.ts +29 -29
- package/src/talkpilot/agents/agents.getters.ts +34 -34
- package/src/talkpilot/agents/agents.types.ts +14 -14
- package/src/talkpilot/agents/index.ts +2 -2
- package/src/talkpilot/backgroundToolResults/__tests__/backgroundToolResults.getters.spec.ts +147 -147
- package/src/talkpilot/backgroundToolResults/backgroundToolResults.getters.ts +65 -65
- package/src/talkpilot/backgroundToolResults/backgroundToolResults.types.ts +23 -23
- package/src/talkpilot/backgroundToolResults/index.ts +2 -2
- package/src/talkpilot/calls/__tests__/callStats.utils.spec.ts +128 -128
- package/src/talkpilot/calls/__tests__/calls.spec.ts +252 -252
- package/src/talkpilot/calls/calls.getters.ts +446 -248
- package/src/talkpilot/calls/calls.types.ts +171 -116
- package/src/talkpilot/calls/index.ts +2 -2
- package/src/talkpilot/clientAudioBuffers/__tests__/clientAudioBuffer.getters.spec.ts +160 -160
- package/src/talkpilot/clientAudioBuffers/clientAudioBuffer.getters.ts +117 -117
- package/src/talkpilot/clientAudioBuffers/clientsAudioBuffers.types.ts +25 -25
- package/src/talkpilot/clientAudioBuffers/index.ts +2 -2
- package/src/talkpilot/clients/clients.getters.ts +16 -16
- package/src/talkpilot/clients/clients.types.ts +14 -14
- package/src/talkpilot/clients/index.ts +2 -2
- package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +106 -106
- package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +44 -22
- package/src/talkpilot/clientsConfig/clientsConfig.types.ts +94 -92
- package/src/talkpilot/clientsConfig/index.ts +2 -2
- package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +67 -67
- package/src/talkpilot/flows/flows.getter.ts +14 -14
- package/src/talkpilot/flows/flows.schema.ts +153 -153
- package/src/talkpilot/flows/flows.types.ts +184 -184
- package/src/talkpilot/flows/index.ts +2 -2
- package/src/talkpilot/groups/__tests__/groups.spec.ts +90 -90
- package/src/talkpilot/groups/__tests__/phone.utils.spec.ts +32 -32
- package/src/talkpilot/groups/groups.getters.ts +30 -30
- package/src/talkpilot/groups/groups.types.ts +29 -29
- package/src/talkpilot/groups/index.ts +3 -3
- package/src/talkpilot/groups/phone.utils.ts +46 -46
- package/src/talkpilot/index.ts +29 -29
- package/src/talkpilot/leads/index.ts +2 -2
- package/src/talkpilot/leads/leads.getter.ts +6 -6
- package/src/talkpilot/leads/leads.schema.ts +33 -33
- package/src/talkpilot/leads/leads.types.ts +20 -20
- package/src/talkpilot/mongodb-client.ts +78 -78
- package/src/talkpilot/phone_numbers/__tests__/phone_numbers.spec.ts +247 -247
- package/src/talkpilot/phone_numbers/index.ts +2 -2
- package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +154 -154
- package/src/talkpilot/phone_numbers/phone_numbers.schema.ts +17 -17
- package/src/talkpilot/phone_numbers/phone_numbers.types.ts +30 -30
- package/src/talkpilot/plans/__tests__/plans.spec.ts +70 -70
- package/src/talkpilot/plans/index.ts +2 -2
- package/src/talkpilot/plans/plans.getters.ts +132 -132
- package/src/talkpilot/plans/plans.types.ts +89 -89
- package/src/talkpilot/results/index.ts +7 -7
- package/src/talkpilot/results/results.getter.ts +35 -35
- package/src/talkpilot/results/results.schema.ts +25 -25
- package/src/talkpilot/results/results.types.ts +34 -34
- package/src/talkpilot/retry_analyze/__tests__/retryAnalyze.getters.spec.ts +156 -156
- package/src/talkpilot/retry_analyze/index.ts +2 -2
- package/src/talkpilot/retry_analyze/retryAnalyze.getters.ts +75 -75
- package/src/talkpilot/retry_analyze/retryAnalyze.types.ts +13 -13
- package/src/talkpilot/sessions/__tests__/sessions.spec.ts +147 -147
- package/src/talkpilot/sessions/index.ts +2 -2
- package/src/talkpilot/sessions/sessions.getter.ts +92 -92
- package/src/talkpilot/sessions/sessions.schema.ts +34 -34
- package/src/talkpilot/sessions/sessions.types.ts +30 -30
- package/src/talkpilot/subscriptions/__tests__/subscriptions.getters.utils.spec.ts +45 -45
- package/src/talkpilot/subscriptions/index.ts +3 -3
- package/src/talkpilot/subscriptions/subscriptions.getters.ts +146 -146
- package/src/talkpilot/subscriptions/subscriptions.getters.utils.ts +33 -33
- package/src/talkpilot/subscriptions/subscriptions.types.ts +66 -66
- package/src/talkpilot/utils/__tests__/query.utils.spec.ts +49 -49
- package/src/talkpilot/utils/query.utils.ts +21 -21
- package/src/test-utils/db-utils.ts +24 -24
- package/src/test-utils/factories/index.ts +12 -12
- package/src/test-utils/factories/municipal/cities.ts +16 -16
- package/src/test-utils/factories/municipal/departmentsSubjects.ts +37 -37
- package/src/test-utils/factories/municipal/streets.ts +22 -22
- package/src/test-utils/factories/municipal/tickets.ts +39 -39
- package/src/test-utils/factories/talkpilot/agents.ts +19 -19
- package/src/test-utils/factories/talkpilot/calls.ts +37 -37
- package/src/test-utils/factories/talkpilot/clientAudioBuffers.ts +20 -20
- package/src/test-utils/factories/talkpilot/clientsConfig.ts +18 -18
- package/src/test-utils/factories/talkpilot/flows.ts +33 -33
- package/src/test-utils/factories/talkpilot/groups.ts +33 -33
- package/src/test-utils/factories/talkpilot/phone_numbers.ts +22 -22
- package/src/test-utils/factories/talkpilot/sessions.ts +35 -35
- package/src/utils/validation.ts +23 -23
- package/tsconfig.json +23 -23
|
@@ -1,184 +1,184 @@
|
|
|
1
|
-
import { ObjectId } from "mongodb";
|
|
2
|
-
|
|
3
|
-
/* ---------- Basic helpers ---------- */
|
|
4
|
-
|
|
5
|
-
export type Position = { x: number; y: number };
|
|
6
|
-
export type HttpMethod = "GET" | "POST" | "PUT" | "PATCH" | "DELETE";
|
|
7
|
-
|
|
8
|
-
export type QuestionType = "open" | "multiple" | "table";
|
|
9
|
-
export type SelectionType = "single" | "multiple";
|
|
10
|
-
|
|
11
|
-
export type Question = {
|
|
12
|
-
id: string;
|
|
13
|
-
type: QuestionType;
|
|
14
|
-
text: string;
|
|
15
|
-
repeatAnswer?: boolean;
|
|
16
|
-
// For 'multiple' type questions:
|
|
17
|
-
answers?: string[] | null;
|
|
18
|
-
selectionType?: SelectionType | null; // 'single' = choose one, 'multiple' = choose one or more
|
|
19
|
-
// For 'table' type questions:
|
|
20
|
-
tableHeader?: string | null;
|
|
21
|
-
tableRows?: string[] | null;
|
|
22
|
-
tableColumns?: string[] | null; // These are the answer options shared across all rows
|
|
23
|
-
// Note: 'table' questions can also have selectionType to indicate if each row allows single or multiple selections
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
export type NodeKind =
|
|
27
|
-
| "start"
|
|
28
|
-
| "say"
|
|
29
|
-
| "condition"
|
|
30
|
-
| "improvise"
|
|
31
|
-
| "tool"
|
|
32
|
-
| "sendMessage"
|
|
33
|
-
| "questionsList"
|
|
34
|
-
| "end";
|
|
35
|
-
type BaseNode = {
|
|
36
|
-
id: string;
|
|
37
|
-
type: NodeKind;
|
|
38
|
-
text: string;
|
|
39
|
-
|
|
40
|
-
name?: string | null;
|
|
41
|
-
position: Position;
|
|
42
|
-
children?: string[] | null;
|
|
43
|
-
successStatus?: string | null;
|
|
44
|
-
questions?: Question[] | null;
|
|
45
|
-
};
|
|
46
|
-
|
|
47
|
-
export type StartNode = BaseNode & { type: "start" };
|
|
48
|
-
|
|
49
|
-
export type SayNode = BaseNode & { type: "say" };
|
|
50
|
-
|
|
51
|
-
export type ConditionNode = BaseNode & {
|
|
52
|
-
type: "condition";
|
|
53
|
-
conditions: Array<{
|
|
54
|
-
conditionString: string;
|
|
55
|
-
nodeType?: string | null;
|
|
56
|
-
id?: string | null;
|
|
57
|
-
}>;
|
|
58
|
-
defaultNodeType?: string | null;
|
|
59
|
-
defaultChild?: { id?: string } | null;
|
|
60
|
-
};
|
|
61
|
-
|
|
62
|
-
export type ImproviseNode = BaseNode & { type: "improvise" };
|
|
63
|
-
|
|
64
|
-
export type FlowTool = {
|
|
65
|
-
id: string;
|
|
66
|
-
name: string;
|
|
67
|
-
description?: string;
|
|
68
|
-
url: string;
|
|
69
|
-
method: HttpMethod;
|
|
70
|
-
headers?: Record<string, string>;
|
|
71
|
-
parameters: Record<string, unknown>;
|
|
72
|
-
bodyFormat?: "json" | "form-data";
|
|
73
|
-
preventAudioDuringTool?: boolean;
|
|
74
|
-
sendUserAuthToken?: boolean;
|
|
75
|
-
runInBackground?: boolean;
|
|
76
|
-
backgroundToolType?: "backgroundToolAlways" | "backgroundToolOnce";
|
|
77
|
-
backgroundContinuationInstructions?: string;
|
|
78
|
-
};
|
|
79
|
-
|
|
80
|
-
export type SendMethod = "sms" | "whatsapp";
|
|
81
|
-
|
|
82
|
-
export type SendMessageNode = BaseNode & {
|
|
83
|
-
type: "sendMessage";
|
|
84
|
-
/** Which channel to use. If omitted, decide at runtime (e.g., default to whatsapp). */
|
|
85
|
-
send_method: SendMethod;
|
|
86
|
-
/** Destination phone (E.164 or templated string like "{{userPhone}}"). */
|
|
87
|
-
phone_number: string;
|
|
88
|
-
};
|
|
89
|
-
|
|
90
|
-
export type ToolNode = BaseNode & {
|
|
91
|
-
type: "tool";
|
|
92
|
-
name: string;
|
|
93
|
-
text: string;
|
|
94
|
-
};
|
|
95
|
-
|
|
96
|
-
export type QuestionsListNode = BaseNode & {
|
|
97
|
-
type: "questionsList";
|
|
98
|
-
questions: Question[];
|
|
99
|
-
};
|
|
100
|
-
|
|
101
|
-
export type EndNode = BaseNode & { type: "end" };
|
|
102
|
-
|
|
103
|
-
export type Interaction =
|
|
104
|
-
| StartNode
|
|
105
|
-
| SayNode
|
|
106
|
-
| ConditionNode
|
|
107
|
-
| ImproviseNode
|
|
108
|
-
| ToolNode
|
|
109
|
-
| SendMessageNode
|
|
110
|
-
| QuestionsListNode
|
|
111
|
-
| EndNode;
|
|
112
|
-
|
|
113
|
-
export type Edge = {
|
|
114
|
-
id: string;
|
|
115
|
-
source: string;
|
|
116
|
-
target: string;
|
|
117
|
-
label?: string | null;
|
|
118
|
-
};
|
|
119
|
-
|
|
120
|
-
export type MetaKey = {
|
|
121
|
-
key: string;
|
|
122
|
-
required?: boolean;
|
|
123
|
-
};
|
|
124
|
-
|
|
125
|
-
export type UserProperty = {
|
|
126
|
-
name: string;
|
|
127
|
-
description: string;
|
|
128
|
-
};
|
|
129
|
-
|
|
130
|
-
export type InterruptionSettings = {
|
|
131
|
-
enableInterruptionDetection?: boolean;
|
|
132
|
-
interruptionWindowSeconds?: number;
|
|
133
|
-
interruptionThresholdSeconds?: number;
|
|
134
|
-
interruptionInstruction?: string;
|
|
135
|
-
};
|
|
136
|
-
|
|
137
|
-
export type SilenceEscalationSettings = {
|
|
138
|
-
enableSilenceDetection?: boolean;
|
|
139
|
-
firstWarningSilenceSeconds?: number;
|
|
140
|
-
firstWarningInstruction?: string;
|
|
141
|
-
secondWarningSilenceSeconds?: number;
|
|
142
|
-
secondWarningInstruction?: string;
|
|
143
|
-
disconnectSilenceSeconds?: number;
|
|
144
|
-
silenceTimeoutSeconds?: number;
|
|
145
|
-
};
|
|
146
|
-
|
|
147
|
-
export type ConversationSettings = {
|
|
148
|
-
interruptions?: InterruptionSettings;
|
|
149
|
-
silence?: SilenceEscalationSettings;
|
|
150
|
-
liveTimeUpdates?: boolean;
|
|
151
|
-
};
|
|
152
|
-
|
|
153
|
-
/* ---------- The Flow object ---------- */
|
|
154
|
-
|
|
155
|
-
export type Flow = {
|
|
156
|
-
freeTextContent?: string;
|
|
157
|
-
isFreeText?: boolean;
|
|
158
|
-
_id?: ObjectId;
|
|
159
|
-
|
|
160
|
-
flowName: string;
|
|
161
|
-
clientId: string;
|
|
162
|
-
|
|
163
|
-
systemInstructions: string;
|
|
164
|
-
initialSentence: string;
|
|
165
|
-
voice?: string;
|
|
166
|
-
|
|
167
|
-
insights: string[];
|
|
168
|
-
conversationSettings?: ConversationSettings;
|
|
169
|
-
|
|
170
|
-
interactions: Interaction[];
|
|
171
|
-
|
|
172
|
-
edges?: Edge[] | null;
|
|
173
|
-
metaKeys?: MetaKey[] | null;
|
|
174
|
-
userProperties?: UserProperty[] | null;
|
|
175
|
-
tools?: FlowTool[];
|
|
176
|
-
afterCallEvent?: string;
|
|
177
|
-
|
|
178
|
-
// AI provider configuration
|
|
179
|
-
aiProvider?: any;
|
|
180
|
-
|
|
181
|
-
// Timestamps
|
|
182
|
-
createdAt?: Date;
|
|
183
|
-
updatedAt?: Date;
|
|
184
|
-
};
|
|
1
|
+
import { ObjectId } from "mongodb";
|
|
2
|
+
|
|
3
|
+
/* ---------- Basic helpers ---------- */
|
|
4
|
+
|
|
5
|
+
export type Position = { x: number; y: number };
|
|
6
|
+
export type HttpMethod = "GET" | "POST" | "PUT" | "PATCH" | "DELETE";
|
|
7
|
+
|
|
8
|
+
export type QuestionType = "open" | "multiple" | "table";
|
|
9
|
+
export type SelectionType = "single" | "multiple";
|
|
10
|
+
|
|
11
|
+
export type Question = {
|
|
12
|
+
id: string;
|
|
13
|
+
type: QuestionType;
|
|
14
|
+
text: string;
|
|
15
|
+
repeatAnswer?: boolean;
|
|
16
|
+
// For 'multiple' type questions:
|
|
17
|
+
answers?: string[] | null;
|
|
18
|
+
selectionType?: SelectionType | null; // 'single' = choose one, 'multiple' = choose one or more
|
|
19
|
+
// For 'table' type questions:
|
|
20
|
+
tableHeader?: string | null;
|
|
21
|
+
tableRows?: string[] | null;
|
|
22
|
+
tableColumns?: string[] | null; // These are the answer options shared across all rows
|
|
23
|
+
// Note: 'table' questions can also have selectionType to indicate if each row allows single or multiple selections
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
export type NodeKind =
|
|
27
|
+
| "start"
|
|
28
|
+
| "say"
|
|
29
|
+
| "condition"
|
|
30
|
+
| "improvise"
|
|
31
|
+
| "tool"
|
|
32
|
+
| "sendMessage"
|
|
33
|
+
| "questionsList"
|
|
34
|
+
| "end";
|
|
35
|
+
type BaseNode = {
|
|
36
|
+
id: string;
|
|
37
|
+
type: NodeKind;
|
|
38
|
+
text: string;
|
|
39
|
+
|
|
40
|
+
name?: string | null;
|
|
41
|
+
position: Position;
|
|
42
|
+
children?: string[] | null;
|
|
43
|
+
successStatus?: string | null;
|
|
44
|
+
questions?: Question[] | null;
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
export type StartNode = BaseNode & { type: "start" };
|
|
48
|
+
|
|
49
|
+
export type SayNode = BaseNode & { type: "say" };
|
|
50
|
+
|
|
51
|
+
export type ConditionNode = BaseNode & {
|
|
52
|
+
type: "condition";
|
|
53
|
+
conditions: Array<{
|
|
54
|
+
conditionString: string;
|
|
55
|
+
nodeType?: string | null;
|
|
56
|
+
id?: string | null;
|
|
57
|
+
}>;
|
|
58
|
+
defaultNodeType?: string | null;
|
|
59
|
+
defaultChild?: { id?: string } | null;
|
|
60
|
+
};
|
|
61
|
+
|
|
62
|
+
export type ImproviseNode = BaseNode & { type: "improvise" };
|
|
63
|
+
|
|
64
|
+
export type FlowTool = {
|
|
65
|
+
id: string;
|
|
66
|
+
name: string;
|
|
67
|
+
description?: string;
|
|
68
|
+
url: string;
|
|
69
|
+
method: HttpMethod;
|
|
70
|
+
headers?: Record<string, string>;
|
|
71
|
+
parameters: Record<string, unknown>;
|
|
72
|
+
bodyFormat?: "json" | "form-data";
|
|
73
|
+
preventAudioDuringTool?: boolean;
|
|
74
|
+
sendUserAuthToken?: boolean;
|
|
75
|
+
runInBackground?: boolean;
|
|
76
|
+
backgroundToolType?: "backgroundToolAlways" | "backgroundToolOnce";
|
|
77
|
+
backgroundContinuationInstructions?: string;
|
|
78
|
+
};
|
|
79
|
+
|
|
80
|
+
export type SendMethod = "sms" | "whatsapp";
|
|
81
|
+
|
|
82
|
+
export type SendMessageNode = BaseNode & {
|
|
83
|
+
type: "sendMessage";
|
|
84
|
+
/** Which channel to use. If omitted, decide at runtime (e.g., default to whatsapp). */
|
|
85
|
+
send_method: SendMethod;
|
|
86
|
+
/** Destination phone (E.164 or templated string like "{{userPhone}}"). */
|
|
87
|
+
phone_number: string;
|
|
88
|
+
};
|
|
89
|
+
|
|
90
|
+
export type ToolNode = BaseNode & {
|
|
91
|
+
type: "tool";
|
|
92
|
+
name: string;
|
|
93
|
+
text: string;
|
|
94
|
+
};
|
|
95
|
+
|
|
96
|
+
export type QuestionsListNode = BaseNode & {
|
|
97
|
+
type: "questionsList";
|
|
98
|
+
questions: Question[];
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
export type EndNode = BaseNode & { type: "end" };
|
|
102
|
+
|
|
103
|
+
export type Interaction =
|
|
104
|
+
| StartNode
|
|
105
|
+
| SayNode
|
|
106
|
+
| ConditionNode
|
|
107
|
+
| ImproviseNode
|
|
108
|
+
| ToolNode
|
|
109
|
+
| SendMessageNode
|
|
110
|
+
| QuestionsListNode
|
|
111
|
+
| EndNode;
|
|
112
|
+
|
|
113
|
+
export type Edge = {
|
|
114
|
+
id: string;
|
|
115
|
+
source: string;
|
|
116
|
+
target: string;
|
|
117
|
+
label?: string | null;
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
export type MetaKey = {
|
|
121
|
+
key: string;
|
|
122
|
+
required?: boolean;
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
export type UserProperty = {
|
|
126
|
+
name: string;
|
|
127
|
+
description: string;
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
export type InterruptionSettings = {
|
|
131
|
+
enableInterruptionDetection?: boolean;
|
|
132
|
+
interruptionWindowSeconds?: number;
|
|
133
|
+
interruptionThresholdSeconds?: number;
|
|
134
|
+
interruptionInstruction?: string;
|
|
135
|
+
};
|
|
136
|
+
|
|
137
|
+
export type SilenceEscalationSettings = {
|
|
138
|
+
enableSilenceDetection?: boolean;
|
|
139
|
+
firstWarningSilenceSeconds?: number;
|
|
140
|
+
firstWarningInstruction?: string;
|
|
141
|
+
secondWarningSilenceSeconds?: number;
|
|
142
|
+
secondWarningInstruction?: string;
|
|
143
|
+
disconnectSilenceSeconds?: number;
|
|
144
|
+
silenceTimeoutSeconds?: number;
|
|
145
|
+
};
|
|
146
|
+
|
|
147
|
+
export type ConversationSettings = {
|
|
148
|
+
interruptions?: InterruptionSettings;
|
|
149
|
+
silence?: SilenceEscalationSettings;
|
|
150
|
+
liveTimeUpdates?: boolean;
|
|
151
|
+
};
|
|
152
|
+
|
|
153
|
+
/* ---------- The Flow object ---------- */
|
|
154
|
+
|
|
155
|
+
export type Flow = {
|
|
156
|
+
freeTextContent?: string;
|
|
157
|
+
isFreeText?: boolean;
|
|
158
|
+
_id?: ObjectId;
|
|
159
|
+
|
|
160
|
+
flowName: string;
|
|
161
|
+
clientId: string;
|
|
162
|
+
|
|
163
|
+
systemInstructions: string;
|
|
164
|
+
initialSentence: string;
|
|
165
|
+
voice?: string;
|
|
166
|
+
|
|
167
|
+
insights: string[];
|
|
168
|
+
conversationSettings?: ConversationSettings;
|
|
169
|
+
|
|
170
|
+
interactions: Interaction[];
|
|
171
|
+
|
|
172
|
+
edges?: Edge[] | null;
|
|
173
|
+
metaKeys?: MetaKey[] | null;
|
|
174
|
+
userProperties?: UserProperty[] | null;
|
|
175
|
+
tools?: FlowTool[];
|
|
176
|
+
afterCallEvent?: string;
|
|
177
|
+
|
|
178
|
+
// AI provider configuration
|
|
179
|
+
aiProvider?: any;
|
|
180
|
+
|
|
181
|
+
// Timestamps
|
|
182
|
+
createdAt?: Date;
|
|
183
|
+
updatedAt?: Date;
|
|
184
|
+
};
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export { getFlowsCollection, findFlowById } from "./flows.getter";
|
|
2
|
-
export type * from "./flows.types";
|
|
1
|
+
export { getFlowsCollection, findFlowById } from "./flows.getter";
|
|
2
|
+
export type * from "./flows.types";
|
|
@@ -1,90 +1,90 @@
|
|
|
1
|
-
import { ObjectId } from "mongodb";
|
|
2
|
-
import { faker } from "@faker-js/faker";
|
|
3
|
-
import {
|
|
4
|
-
getGroupsCollection,
|
|
5
|
-
findGroups,
|
|
6
|
-
createGroup,
|
|
7
|
-
updateGroup,
|
|
8
|
-
removeGroup,
|
|
9
|
-
} from "../groups.getters";
|
|
10
|
-
import { createGroup as createGroupDoc } from "../../../test-utils/factories";
|
|
11
|
-
|
|
12
|
-
describe("db.groups", () => {
|
|
13
|
-
describe("getGroupsCollection", () => {
|
|
14
|
-
it('returns the "groups" collection', () => {
|
|
15
|
-
expect(getGroupsCollection().collectionName).toBe("groups");
|
|
16
|
-
});
|
|
17
|
-
});
|
|
18
|
-
|
|
19
|
-
describe("findGroups", () => {
|
|
20
|
-
it("returns groups matching the provided filter", async () => {
|
|
21
|
-
const clientId = faker.string.uuid();
|
|
22
|
-
const g1 = createGroupDoc({ clientId, name: "Group A" });
|
|
23
|
-
const g2 = createGroupDoc({ clientId, name: "Group B" });
|
|
24
|
-
const gOther = createGroupDoc({
|
|
25
|
-
clientId: faker.string.uuid(),
|
|
26
|
-
name: "Group C",
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
await getGroupsCollection().insertMany([g1, g2, gOther]);
|
|
30
|
-
|
|
31
|
-
const result = await findGroups({ clientId });
|
|
32
|
-
const names = result.map((g) => g.name).sort();
|
|
33
|
-
|
|
34
|
-
expect(result).toHaveLength(2);
|
|
35
|
-
expect(names).toEqual(["Group A", "Group B"]);
|
|
36
|
-
});
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
describe("createGroup", () => {
|
|
40
|
-
it("inserts a group and returns insertedId", async () => {
|
|
41
|
-
const toInsert = createGroupDoc({ name: "New Group" });
|
|
42
|
-
|
|
43
|
-
const insertedId = await createGroup(toInsert);
|
|
44
|
-
expect(insertedId).toBeInstanceOf(ObjectId);
|
|
45
|
-
|
|
46
|
-
const fromDb = await getGroupsCollection().findOne({ _id: insertedId });
|
|
47
|
-
expect(fromDb?.name).toBe("New Group");
|
|
48
|
-
expect(fromDb?.clientId).toBe(toInsert.clientId);
|
|
49
|
-
});
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
describe("updateGroup", () => {
|
|
53
|
-
it("updates the group and sets updatedAt, returning the updated document", async () => {
|
|
54
|
-
const original = createGroupDoc({
|
|
55
|
-
name: "Before",
|
|
56
|
-
description: "Old desc",
|
|
57
|
-
});
|
|
58
|
-
const { insertedId } = await getGroupsCollection().insertOne(original);
|
|
59
|
-
const before = await getGroupsCollection().findOne({ _id: insertedId });
|
|
60
|
-
|
|
61
|
-
// Add a small delay to ensure updatedAt will be different
|
|
62
|
-
await new Promise((resolve) => setTimeout(resolve, 10));
|
|
63
|
-
|
|
64
|
-
const res = await updateGroup(
|
|
65
|
-
{ _id: insertedId },
|
|
66
|
-
{ name: "After", description: "New desc" },
|
|
67
|
-
);
|
|
68
|
-
|
|
69
|
-
expect(res?._id).toEqual(insertedId);
|
|
70
|
-
expect(res?.name).toBe("After");
|
|
71
|
-
expect(res?.description).toBe("New desc");
|
|
72
|
-
expect(new Date(res!.updatedAt).getTime()).toBeGreaterThan(
|
|
73
|
-
new Date(before!.updatedAt).getTime(),
|
|
74
|
-
);
|
|
75
|
-
});
|
|
76
|
-
});
|
|
77
|
-
|
|
78
|
-
describe("removeGroup", () => {
|
|
79
|
-
it("removes a group by string id", async () => {
|
|
80
|
-
const { insertedId } =
|
|
81
|
-
await getGroupsCollection().insertOne(createGroupDoc());
|
|
82
|
-
|
|
83
|
-
const result = await removeGroup(insertedId.toHexString());
|
|
84
|
-
expect(result.deletedCount).toBe(1);
|
|
85
|
-
|
|
86
|
-
const check = await getGroupsCollection().findOne({ _id: insertedId });
|
|
87
|
-
expect(check).toBeNull();
|
|
88
|
-
});
|
|
89
|
-
});
|
|
90
|
-
});
|
|
1
|
+
import { ObjectId } from "mongodb";
|
|
2
|
+
import { faker } from "@faker-js/faker";
|
|
3
|
+
import {
|
|
4
|
+
getGroupsCollection,
|
|
5
|
+
findGroups,
|
|
6
|
+
createGroup,
|
|
7
|
+
updateGroup,
|
|
8
|
+
removeGroup,
|
|
9
|
+
} from "../groups.getters";
|
|
10
|
+
import { createGroup as createGroupDoc } from "../../../test-utils/factories";
|
|
11
|
+
|
|
12
|
+
describe("db.groups", () => {
|
|
13
|
+
describe("getGroupsCollection", () => {
|
|
14
|
+
it('returns the "groups" collection', () => {
|
|
15
|
+
expect(getGroupsCollection().collectionName).toBe("groups");
|
|
16
|
+
});
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
describe("findGroups", () => {
|
|
20
|
+
it("returns groups matching the provided filter", async () => {
|
|
21
|
+
const clientId = faker.string.uuid();
|
|
22
|
+
const g1 = createGroupDoc({ clientId, name: "Group A" });
|
|
23
|
+
const g2 = createGroupDoc({ clientId, name: "Group B" });
|
|
24
|
+
const gOther = createGroupDoc({
|
|
25
|
+
clientId: faker.string.uuid(),
|
|
26
|
+
name: "Group C",
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
await getGroupsCollection().insertMany([g1, g2, gOther]);
|
|
30
|
+
|
|
31
|
+
const result = await findGroups({ clientId });
|
|
32
|
+
const names = result.map((g) => g.name).sort();
|
|
33
|
+
|
|
34
|
+
expect(result).toHaveLength(2);
|
|
35
|
+
expect(names).toEqual(["Group A", "Group B"]);
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
describe("createGroup", () => {
|
|
40
|
+
it("inserts a group and returns insertedId", async () => {
|
|
41
|
+
const toInsert = createGroupDoc({ name: "New Group" });
|
|
42
|
+
|
|
43
|
+
const insertedId = await createGroup(toInsert);
|
|
44
|
+
expect(insertedId).toBeInstanceOf(ObjectId);
|
|
45
|
+
|
|
46
|
+
const fromDb = await getGroupsCollection().findOne({ _id: insertedId });
|
|
47
|
+
expect(fromDb?.name).toBe("New Group");
|
|
48
|
+
expect(fromDb?.clientId).toBe(toInsert.clientId);
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
describe("updateGroup", () => {
|
|
53
|
+
it("updates the group and sets updatedAt, returning the updated document", async () => {
|
|
54
|
+
const original = createGroupDoc({
|
|
55
|
+
name: "Before",
|
|
56
|
+
description: "Old desc",
|
|
57
|
+
});
|
|
58
|
+
const { insertedId } = await getGroupsCollection().insertOne(original);
|
|
59
|
+
const before = await getGroupsCollection().findOne({ _id: insertedId });
|
|
60
|
+
|
|
61
|
+
// Add a small delay to ensure updatedAt will be different
|
|
62
|
+
await new Promise((resolve) => setTimeout(resolve, 10));
|
|
63
|
+
|
|
64
|
+
const res = await updateGroup(
|
|
65
|
+
{ _id: insertedId },
|
|
66
|
+
{ name: "After", description: "New desc" },
|
|
67
|
+
);
|
|
68
|
+
|
|
69
|
+
expect(res?._id).toEqual(insertedId);
|
|
70
|
+
expect(res?.name).toBe("After");
|
|
71
|
+
expect(res?.description).toBe("New desc");
|
|
72
|
+
expect(new Date(res!.updatedAt).getTime()).toBeGreaterThan(
|
|
73
|
+
new Date(before!.updatedAt).getTime(),
|
|
74
|
+
);
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
describe("removeGroup", () => {
|
|
79
|
+
it("removes a group by string id", async () => {
|
|
80
|
+
const { insertedId } =
|
|
81
|
+
await getGroupsCollection().insertOne(createGroupDoc());
|
|
82
|
+
|
|
83
|
+
const result = await removeGroup(insertedId.toHexString());
|
|
84
|
+
expect(result.deletedCount).toBe(1);
|
|
85
|
+
|
|
86
|
+
const check = await getGroupsCollection().findOne({ _id: insertedId });
|
|
87
|
+
expect(check).toBeNull();
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
});
|
|
@@ -1,32 +1,32 @@
|
|
|
1
|
-
import { enrichPhoneNumber, isValidE164 } from "../phone.utils";
|
|
2
|
-
|
|
3
|
-
describe("Phone Utilities", () => {
|
|
4
|
-
describe("enrichPhoneNumber", () => {
|
|
5
|
-
it("should enrich Israeli number", () => {
|
|
6
|
-
const result = enrichPhoneNumber("+972507725874");
|
|
7
|
-
expect(result?.e164).toBe("+972507725874");
|
|
8
|
-
expect(result?.region).toBeDefined();
|
|
9
|
-
expect(result?.countryCallingCode).toBe("972");
|
|
10
|
-
});
|
|
11
|
-
|
|
12
|
-
it("should enrich US number", () => {
|
|
13
|
-
const result = enrichPhoneNumber("+12025551234");
|
|
14
|
-
expect(result?.region).toBe("US");
|
|
15
|
-
expect(result?.countryCallingCode).toBe("1");
|
|
16
|
-
});
|
|
17
|
-
|
|
18
|
-
it("should return null for invalid", () => {
|
|
19
|
-
expect(enrichPhoneNumber("invalid")).toBeNull();
|
|
20
|
-
expect(enrichPhoneNumber("0507725874")).toBeNull();
|
|
21
|
-
});
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
describe("isValidE164", () => {
|
|
25
|
-
it("should validate E.164 format", () => {
|
|
26
|
-
expect(isValidE164("+972508832983")).toBe(true);
|
|
27
|
-
expect(isValidE164("+12025551234")).toBe(true);
|
|
28
|
-
expect(isValidE164("0501234567")).toBe(false);
|
|
29
|
-
expect(isValidE164("invalid")).toBe(false);
|
|
30
|
-
});
|
|
31
|
-
});
|
|
32
|
-
});
|
|
1
|
+
import { enrichPhoneNumber, isValidE164 } from "../phone.utils";
|
|
2
|
+
|
|
3
|
+
describe("Phone Utilities", () => {
|
|
4
|
+
describe("enrichPhoneNumber", () => {
|
|
5
|
+
it("should enrich Israeli number", () => {
|
|
6
|
+
const result = enrichPhoneNumber("+972507725874");
|
|
7
|
+
expect(result?.e164).toBe("+972507725874");
|
|
8
|
+
expect(result?.region).toBeDefined();
|
|
9
|
+
expect(result?.countryCallingCode).toBe("972");
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
it("should enrich US number", () => {
|
|
13
|
+
const result = enrichPhoneNumber("+12025551234");
|
|
14
|
+
expect(result?.region).toBe("US");
|
|
15
|
+
expect(result?.countryCallingCode).toBe("1");
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
it("should return null for invalid", () => {
|
|
19
|
+
expect(enrichPhoneNumber("invalid")).toBeNull();
|
|
20
|
+
expect(enrichPhoneNumber("0507725874")).toBeNull();
|
|
21
|
+
});
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
describe("isValidE164", () => {
|
|
25
|
+
it("should validate E.164 format", () => {
|
|
26
|
+
expect(isValidE164("+972508832983")).toBe(true);
|
|
27
|
+
expect(isValidE164("+12025551234")).toBe(true);
|
|
28
|
+
expect(isValidE164("0501234567")).toBe(false);
|
|
29
|
+
expect(isValidE164("invalid")).toBe(false);
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
});
|