@talkpilot/core-db 1.1.19 → 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 +48 -0
- package/dist/talkpilot/calls/calls.types.d.ts.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/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 -115
- 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 -94
- 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,115 +1,171 @@
|
|
|
1
|
-
import { ObjectId, Sort, WithId } from "mongodb";
|
|
2
|
-
import { TranscriptionSegment } from "../results";
|
|
3
|
-
|
|
4
|
-
export const CONFERENCE_ROLE_CUSTOMER = "customer" as const;
|
|
5
|
-
export const CONFERENCE_ROLE_SUPERVISOR = "supervisor" as const;
|
|
6
|
-
|
|
7
|
-
export type ConferenceRole =
|
|
8
|
-
| typeof CONFERENCE_ROLE_CUSTOMER
|
|
9
|
-
| typeof CONFERENCE_ROLE_SUPERVISOR
|
|
10
|
-
| null;
|
|
11
|
-
|
|
12
|
-
export type Call = {
|
|
13
|
-
callSid: string;
|
|
14
|
-
flowId: ObjectId;
|
|
15
|
-
clientId: string;
|
|
16
|
-
sessionId: ObjectId | null;
|
|
17
|
-
runId?: string;
|
|
18
|
-
resultId: ObjectId;
|
|
19
|
-
customerPhoneNumber: string;
|
|
20
|
-
agentPhoneNumber: string;
|
|
21
|
-
isIncomingCall: boolean;
|
|
22
|
-
isOutgoingCall: boolean;
|
|
23
|
-
callLength: number;
|
|
24
|
-
transcription?: TranscriptionSegment[];
|
|
25
|
-
status?: string;
|
|
26
|
-
leads?: Record<string, string>[];
|
|
27
|
-
summary?: string;
|
|
28
|
-
recordingUrl?: string;
|
|
29
|
-
env: string;
|
|
30
|
-
updatedAt: Date;
|
|
31
|
-
createdAt: Date;
|
|
32
|
-
isAnsweredByAnsweringMachine?: boolean;
|
|
33
|
-
agentHungUp?: boolean;
|
|
34
|
-
endReason?: string;
|
|
35
|
-
conferenceName?: string | null;
|
|
36
|
-
conferenceSid?: string | null;
|
|
37
|
-
conferenceParticipantSid?: string | null;
|
|
38
|
-
isConferenceCall?: boolean;
|
|
39
|
-
conferenceRole?: ConferenceRole;
|
|
40
|
-
redirectedCall?: boolean;
|
|
41
|
-
toolExecutions?: ToolExecution[];
|
|
42
|
-
};
|
|
43
|
-
|
|
44
|
-
export type ToolExecution = {
|
|
45
|
-
toolName: string;
|
|
46
|
-
executedAt: Date;
|
|
47
|
-
durationMs: number;
|
|
48
|
-
args: Record<string, unknown> | { _redacted: true };
|
|
49
|
-
meta: ToolExecutionMeta;
|
|
50
|
-
status: 'success' | 'error';
|
|
51
|
-
httpStatus?: number;
|
|
52
|
-
response?: Record<string, unknown>;
|
|
53
|
-
};
|
|
54
|
-
|
|
55
|
-
export type ToolExecutionMeta =
|
|
56
|
-
| {
|
|
57
|
-
kind: 'http';
|
|
58
|
-
url: string;
|
|
59
|
-
method: string;
|
|
60
|
-
flowToolId?: string;
|
|
61
|
-
runInBackground?: boolean;
|
|
62
|
-
sensitive?: boolean;
|
|
63
|
-
}
|
|
64
|
-
| { kind: 'internal' };
|
|
65
|
-
|
|
66
|
-
export type CallQueryOptions = {
|
|
67
|
-
sort?: Sort;
|
|
68
|
-
skip?: number;
|
|
69
|
-
limit?: number;
|
|
70
|
-
};
|
|
71
|
-
|
|
72
|
-
export type CallsFilterParams = {
|
|
73
|
-
clientId: string;
|
|
74
|
-
callSid?: string;
|
|
75
|
-
startDate?: Date;
|
|
76
|
-
endDate?: Date;
|
|
77
|
-
status?: string;
|
|
78
|
-
customerPhoneNumber?: string;
|
|
79
|
-
agentPhoneNumber?: string;
|
|
80
|
-
flowId?: string;
|
|
81
|
-
runId?: string;
|
|
82
|
-
sessionId?: string;
|
|
83
|
-
isIncoming?: boolean;
|
|
84
|
-
search?: string;
|
|
85
|
-
};
|
|
86
|
-
|
|
87
|
-
export type ImmutableCallFields =
|
|
88
|
-
| "customerPhoneNumber"
|
|
89
|
-
| "agentPhoneNumber"
|
|
90
|
-
| "isIncomingCall"
|
|
91
|
-
| "isOutgoingCall"
|
|
92
|
-
| "sessionId"
|
|
93
|
-
| "callSid"
|
|
94
|
-
| "flowId"
|
|
95
|
-
| "clientId"
|
|
96
|
-
| "runId"
|
|
97
|
-
| "resultId"
|
|
98
|
-
| "env"
|
|
99
|
-
| "createdAt";
|
|
100
|
-
|
|
101
|
-
export type CallUpdateParams = Partial<Omit<Call, ImmutableCallFields>>;
|
|
102
|
-
|
|
103
|
-
export type CallDoc = WithId<Call>;
|
|
104
|
-
|
|
105
|
-
export type CallsByHour = { hour: string; calls: number };
|
|
106
|
-
|
|
107
|
-
export type CountOpts = {
|
|
108
|
-
isOutgoingCall?: boolean;
|
|
109
|
-
isIncomingCall?: boolean;
|
|
110
|
-
};
|
|
111
|
-
|
|
112
|
-
export type DateRange = {
|
|
113
|
-
since?: Date;
|
|
114
|
-
until?: Date;
|
|
115
|
-
};
|
|
1
|
+
import { ObjectId, Sort, WithId } from "mongodb";
|
|
2
|
+
import { TranscriptionSegment } from "../results";
|
|
3
|
+
|
|
4
|
+
export const CONFERENCE_ROLE_CUSTOMER = "customer" as const;
|
|
5
|
+
export const CONFERENCE_ROLE_SUPERVISOR = "supervisor" as const;
|
|
6
|
+
|
|
7
|
+
export type ConferenceRole =
|
|
8
|
+
| typeof CONFERENCE_ROLE_CUSTOMER
|
|
9
|
+
| typeof CONFERENCE_ROLE_SUPERVISOR
|
|
10
|
+
| null;
|
|
11
|
+
|
|
12
|
+
export type Call = {
|
|
13
|
+
callSid: string;
|
|
14
|
+
flowId: ObjectId;
|
|
15
|
+
clientId: string;
|
|
16
|
+
sessionId: ObjectId | null;
|
|
17
|
+
runId?: string;
|
|
18
|
+
resultId: ObjectId;
|
|
19
|
+
customerPhoneNumber: string;
|
|
20
|
+
agentPhoneNumber: string;
|
|
21
|
+
isIncomingCall: boolean;
|
|
22
|
+
isOutgoingCall: boolean;
|
|
23
|
+
callLength: number;
|
|
24
|
+
transcription?: TranscriptionSegment[];
|
|
25
|
+
status?: string;
|
|
26
|
+
leads?: Record<string, string>[];
|
|
27
|
+
summary?: string;
|
|
28
|
+
recordingUrl?: string;
|
|
29
|
+
env: string;
|
|
30
|
+
updatedAt: Date;
|
|
31
|
+
createdAt: Date;
|
|
32
|
+
isAnsweredByAnsweringMachine?: boolean;
|
|
33
|
+
agentHungUp?: boolean;
|
|
34
|
+
endReason?: string;
|
|
35
|
+
conferenceName?: string | null;
|
|
36
|
+
conferenceSid?: string | null;
|
|
37
|
+
conferenceParticipantSid?: string | null;
|
|
38
|
+
isConferenceCall?: boolean;
|
|
39
|
+
conferenceRole?: ConferenceRole;
|
|
40
|
+
redirectedCall?: boolean;
|
|
41
|
+
toolExecutions?: ToolExecution[];
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
export type ToolExecution = {
|
|
45
|
+
toolName: string;
|
|
46
|
+
executedAt: Date;
|
|
47
|
+
durationMs: number;
|
|
48
|
+
args: Record<string, unknown> | { _redacted: true };
|
|
49
|
+
meta: ToolExecutionMeta;
|
|
50
|
+
status: 'success' | 'error';
|
|
51
|
+
httpStatus?: number;
|
|
52
|
+
response?: Record<string, unknown>;
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
export type ToolExecutionMeta =
|
|
56
|
+
| {
|
|
57
|
+
kind: 'http';
|
|
58
|
+
url: string;
|
|
59
|
+
method: string;
|
|
60
|
+
flowToolId?: string;
|
|
61
|
+
runInBackground?: boolean;
|
|
62
|
+
sensitive?: boolean;
|
|
63
|
+
}
|
|
64
|
+
| { kind: 'internal' };
|
|
65
|
+
|
|
66
|
+
export type CallQueryOptions = {
|
|
67
|
+
sort?: Sort;
|
|
68
|
+
skip?: number;
|
|
69
|
+
limit?: number;
|
|
70
|
+
};
|
|
71
|
+
|
|
72
|
+
export type CallsFilterParams = {
|
|
73
|
+
clientId: string;
|
|
74
|
+
callSid?: string;
|
|
75
|
+
startDate?: Date;
|
|
76
|
+
endDate?: Date;
|
|
77
|
+
status?: string;
|
|
78
|
+
customerPhoneNumber?: string;
|
|
79
|
+
agentPhoneNumber?: string;
|
|
80
|
+
flowId?: string;
|
|
81
|
+
runId?: string;
|
|
82
|
+
sessionId?: string;
|
|
83
|
+
isIncoming?: boolean;
|
|
84
|
+
search?: string;
|
|
85
|
+
};
|
|
86
|
+
|
|
87
|
+
export type ImmutableCallFields =
|
|
88
|
+
| "customerPhoneNumber"
|
|
89
|
+
| "agentPhoneNumber"
|
|
90
|
+
| "isIncomingCall"
|
|
91
|
+
| "isOutgoingCall"
|
|
92
|
+
| "sessionId"
|
|
93
|
+
| "callSid"
|
|
94
|
+
| "flowId"
|
|
95
|
+
| "clientId"
|
|
96
|
+
| "runId"
|
|
97
|
+
| "resultId"
|
|
98
|
+
| "env"
|
|
99
|
+
| "createdAt";
|
|
100
|
+
|
|
101
|
+
export type CallUpdateParams = Partial<Omit<Call, ImmutableCallFields>>;
|
|
102
|
+
|
|
103
|
+
export type CallDoc = WithId<Call>;
|
|
104
|
+
|
|
105
|
+
export type CallsByHour = { hour: string; calls: number };
|
|
106
|
+
|
|
107
|
+
export type CountOpts = {
|
|
108
|
+
isOutgoingCall?: boolean;
|
|
109
|
+
isIncomingCall?: boolean;
|
|
110
|
+
};
|
|
111
|
+
|
|
112
|
+
export type DateRange = {
|
|
113
|
+
since?: Date;
|
|
114
|
+
until?: Date;
|
|
115
|
+
};
|
|
116
|
+
|
|
117
|
+
export type HourlyBucket = {
|
|
118
|
+
hour: string;
|
|
119
|
+
calls: number;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
export type DailyBucket = {
|
|
123
|
+
date: string;
|
|
124
|
+
count: number;
|
|
125
|
+
completed: number;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
export type CallLengthBuckets = {
|
|
129
|
+
short: number;
|
|
130
|
+
medium: number;
|
|
131
|
+
long: number;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
export type DashboardKpis = {
|
|
135
|
+
totalCalls: number;
|
|
136
|
+
avgDurationSeconds: number;
|
|
137
|
+
timeSavedMinutes: number;
|
|
138
|
+
successRate: number;
|
|
139
|
+
completedCount: number;
|
|
140
|
+
failedCount: number;
|
|
141
|
+
noAnswerCount: number;
|
|
142
|
+
busyCount: number;
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
export type DashboardCharts = {
|
|
146
|
+
volumeData: DailyBucket[];
|
|
147
|
+
heatmap: Record<string, HourlyBucket[]>;
|
|
148
|
+
callLengthBuckets: CallLengthBuckets;
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
export type DashboardStatsParams = {
|
|
152
|
+
clientId: string;
|
|
153
|
+
startDate: string;
|
|
154
|
+
endDate: string;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
export type RecentCall = {
|
|
158
|
+
callSid: string;
|
|
159
|
+
customerPhoneNumber: string;
|
|
160
|
+
status: string | undefined;
|
|
161
|
+
callLength: number;
|
|
162
|
+
createdAt: Date;
|
|
163
|
+
summary: string | undefined;
|
|
164
|
+
isIncomingCall: boolean;
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
export type DashboardStatsResult = {
|
|
168
|
+
kpis: DashboardKpis;
|
|
169
|
+
charts: DashboardCharts;
|
|
170
|
+
recentCalls: RecentCall[];
|
|
171
|
+
}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export * from "./calls.types";
|
|
2
|
-
export * from "./calls.getters";
|
|
1
|
+
export * from "./calls.types";
|
|
2
|
+
export * from "./calls.getters";
|
|
@@ -1,160 +1,160 @@
|
|
|
1
|
-
import {
|
|
2
|
-
getClientAudioBuffersCollection,
|
|
3
|
-
insertClientAudioBuffer,
|
|
4
|
-
findClientAudioBuffersByCallSidAndRange,
|
|
5
|
-
getMergedAudioByCallSid,
|
|
6
|
-
getAudioChunksByCallSid,
|
|
7
|
-
} from "../clientAudioBuffer.getters";
|
|
8
|
-
import { ObjectId } from "../../index";
|
|
9
|
-
import { createAudioBufferInput } from "../../../test-utils/factories";
|
|
10
|
-
|
|
11
|
-
describe("clientAudioBuffers getters", () => {
|
|
12
|
-
it('should return the "clientAudioBuffers" collection', () => {
|
|
13
|
-
const collection = getClientAudioBuffersCollection();
|
|
14
|
-
expect(collection.collectionName).toBe("clientAudioBuffers");
|
|
15
|
-
});
|
|
16
|
-
|
|
17
|
-
it("should insert a client audio buffer and return insertedId", async () => {
|
|
18
|
-
const inputToCheck = createAudioBufferInput();
|
|
19
|
-
const result = await insertClientAudioBuffer(inputToCheck);
|
|
20
|
-
expect(result.insertedId).toBeInstanceOf(ObjectId);
|
|
21
|
-
|
|
22
|
-
const savedDoc = await getClientAudioBuffersCollection().findOne({
|
|
23
|
-
_id: result.insertedId,
|
|
24
|
-
});
|
|
25
|
-
expect(savedDoc).toBeDefined();
|
|
26
|
-
expect(savedDoc?.createdAt).toBeInstanceOf(Date);
|
|
27
|
-
expect(savedDoc?.expiresAt).toBeInstanceOf(Date);
|
|
28
|
-
});
|
|
29
|
-
|
|
30
|
-
it("should return only buffers in the specified bucketStartMs range", async () => {
|
|
31
|
-
const callSid = "CA1234567890abcdef1234567890abcdef";
|
|
32
|
-
await insertClientAudioBuffer(
|
|
33
|
-
createAudioBufferInput({ callSid, bucketStartMs: 500 }),
|
|
34
|
-
);
|
|
35
|
-
await insertClientAudioBuffer(
|
|
36
|
-
createAudioBufferInput({ callSid, bucketStartMs: 1000 }),
|
|
37
|
-
);
|
|
38
|
-
await insertClientAudioBuffer(
|
|
39
|
-
createAudioBufferInput({ callSid, bucketStartMs: 3000 }),
|
|
40
|
-
);
|
|
41
|
-
await insertClientAudioBuffer(
|
|
42
|
-
createAudioBufferInput({ callSid, bucketStartMs: 6000 }),
|
|
43
|
-
);
|
|
44
|
-
|
|
45
|
-
const result = await findClientAudioBuffersByCallSidAndRange(
|
|
46
|
-
callSid,
|
|
47
|
-
500,
|
|
48
|
-
3000,
|
|
49
|
-
);
|
|
50
|
-
expect(result).toHaveLength(3);
|
|
51
|
-
const bucketStartValues = result.map((doc) => doc.bucketStartMs);
|
|
52
|
-
const bucketStartsSorted = bucketStartValues.sort((a, b) => a - b);
|
|
53
|
-
expect(bucketStartsSorted).toEqual([500, 1000, 3000]);
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
it("should return all buffers when no range is provided", async () => {
|
|
57
|
-
const callSid = "CA1234567890abcdef1234567890abcdeg";
|
|
58
|
-
await insertClientAudioBuffer(
|
|
59
|
-
createAudioBufferInput({ callSid, bucketStartMs: 500 }),
|
|
60
|
-
);
|
|
61
|
-
await insertClientAudioBuffer(
|
|
62
|
-
createAudioBufferInput({ callSid, bucketStartMs: 1000 }),
|
|
63
|
-
);
|
|
64
|
-
await insertClientAudioBuffer(
|
|
65
|
-
createAudioBufferInput({ callSid, bucketStartMs: 3000 }),
|
|
66
|
-
);
|
|
67
|
-
|
|
68
|
-
const result = await findClientAudioBuffersByCallSidAndRange(callSid);
|
|
69
|
-
expect(result).toHaveLength(3);
|
|
70
|
-
const bucketStartValues = result.map((doc) => doc.bucketStartMs);
|
|
71
|
-
const bucketStartsSorted = bucketStartValues.sort((a, b) => a - b);
|
|
72
|
-
expect(bucketStartsSorted).toEqual([500, 1000, 3000]);
|
|
73
|
-
});
|
|
74
|
-
|
|
75
|
-
it("should merge audio buffers by callSid in chronological order", async () => {
|
|
76
|
-
const callSid = "CA1234567890abcdef1234567890abcdeh";
|
|
77
|
-
|
|
78
|
-
const firstBuffer = Buffer.from([1, 2]);
|
|
79
|
-
const secondBuffer = Buffer.from([3]);
|
|
80
|
-
const thirdBuffer = Buffer.from([4, 5]);
|
|
81
|
-
|
|
82
|
-
await insertClientAudioBuffer(
|
|
83
|
-
createAudioBufferInput({
|
|
84
|
-
callSid,
|
|
85
|
-
bucketStartMs: 1000,
|
|
86
|
-
data: secondBuffer,
|
|
87
|
-
}),
|
|
88
|
-
);
|
|
89
|
-
await insertClientAudioBuffer(
|
|
90
|
-
createAudioBufferInput({
|
|
91
|
-
callSid,
|
|
92
|
-
bucketStartMs: 500,
|
|
93
|
-
data: firstBuffer,
|
|
94
|
-
}),
|
|
95
|
-
);
|
|
96
|
-
await insertClientAudioBuffer(
|
|
97
|
-
createAudioBufferInput({
|
|
98
|
-
callSid,
|
|
99
|
-
bucketStartMs: 3000,
|
|
100
|
-
data: thirdBuffer,
|
|
101
|
-
}),
|
|
102
|
-
);
|
|
103
|
-
|
|
104
|
-
const merged = await getMergedAudioByCallSid(callSid, 10);
|
|
105
|
-
const expected = Buffer.concat([firstBuffer, secondBuffer, thirdBuffer]);
|
|
106
|
-
|
|
107
|
-
expect(merged).toBeInstanceOf(Buffer);
|
|
108
|
-
expect(merged.length).toBe(expected.length);
|
|
109
|
-
expect(merged.equals(expected)).toBe(true);
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
it("should return all audio chunks when less than or equal to limit buffers exist", async () => {
|
|
113
|
-
const callSid = "CA1234567890abcdef1234567890abclt";
|
|
114
|
-
|
|
115
|
-
const limit = 30;
|
|
116
|
-
const buffers: Buffer[] = [];
|
|
117
|
-
for (let i = 0; i < limit; i++) {
|
|
118
|
-
const buf = Buffer.from([i]);
|
|
119
|
-
buffers.push(buf);
|
|
120
|
-
await insertClientAudioBuffer(
|
|
121
|
-
createAudioBufferInput({
|
|
122
|
-
callSid,
|
|
123
|
-
bucketStartMs: 1000 + i * 1000,
|
|
124
|
-
data: buf,
|
|
125
|
-
}),
|
|
126
|
-
);
|
|
127
|
-
}
|
|
128
|
-
|
|
129
|
-
const chunks = await getAudioChunksByCallSid(callSid, limit);
|
|
130
|
-
|
|
131
|
-
expect(chunks).toHaveLength(buffers.length);
|
|
132
|
-
expect(chunks.every((chunk) => chunk instanceof Buffer)).toBe(true);
|
|
133
|
-
expect(Buffer.concat(chunks).equals(Buffer.concat(buffers))).toBe(true);
|
|
134
|
-
});
|
|
135
|
-
|
|
136
|
-
it("should return only the last N audio chunks when more than limit buffers exist", async () => {
|
|
137
|
-
const callSid = "CA1234567890abcdef1234567890abclm";
|
|
138
|
-
|
|
139
|
-
const limit = 30;
|
|
140
|
-
const buffers: Buffer[] = [];
|
|
141
|
-
for (let i = 0; i < 35; i++) {
|
|
142
|
-
const buf = Buffer.from([i]);
|
|
143
|
-
buffers.push(buf);
|
|
144
|
-
await insertClientAudioBuffer(
|
|
145
|
-
createAudioBufferInput({
|
|
146
|
-
callSid,
|
|
147
|
-
bucketStartMs: 1000 + i * 1000,
|
|
148
|
-
data: buf,
|
|
149
|
-
}),
|
|
150
|
-
);
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
const chunks = await getAudioChunksByCallSid(callSid, limit);
|
|
154
|
-
const expectedBuffers = buffers.slice(-limit);
|
|
155
|
-
expect(chunks).toHaveLength(expectedBuffers.length);
|
|
156
|
-
expect(Buffer.concat(chunks).equals(Buffer.concat(expectedBuffers))).toBe(
|
|
157
|
-
true,
|
|
158
|
-
);
|
|
159
|
-
});
|
|
160
|
-
});
|
|
1
|
+
import {
|
|
2
|
+
getClientAudioBuffersCollection,
|
|
3
|
+
insertClientAudioBuffer,
|
|
4
|
+
findClientAudioBuffersByCallSidAndRange,
|
|
5
|
+
getMergedAudioByCallSid,
|
|
6
|
+
getAudioChunksByCallSid,
|
|
7
|
+
} from "../clientAudioBuffer.getters";
|
|
8
|
+
import { ObjectId } from "../../index";
|
|
9
|
+
import { createAudioBufferInput } from "../../../test-utils/factories";
|
|
10
|
+
|
|
11
|
+
describe("clientAudioBuffers getters", () => {
|
|
12
|
+
it('should return the "clientAudioBuffers" collection', () => {
|
|
13
|
+
const collection = getClientAudioBuffersCollection();
|
|
14
|
+
expect(collection.collectionName).toBe("clientAudioBuffers");
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
it("should insert a client audio buffer and return insertedId", async () => {
|
|
18
|
+
const inputToCheck = createAudioBufferInput();
|
|
19
|
+
const result = await insertClientAudioBuffer(inputToCheck);
|
|
20
|
+
expect(result.insertedId).toBeInstanceOf(ObjectId);
|
|
21
|
+
|
|
22
|
+
const savedDoc = await getClientAudioBuffersCollection().findOne({
|
|
23
|
+
_id: result.insertedId,
|
|
24
|
+
});
|
|
25
|
+
expect(savedDoc).toBeDefined();
|
|
26
|
+
expect(savedDoc?.createdAt).toBeInstanceOf(Date);
|
|
27
|
+
expect(savedDoc?.expiresAt).toBeInstanceOf(Date);
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it("should return only buffers in the specified bucketStartMs range", async () => {
|
|
31
|
+
const callSid = "CA1234567890abcdef1234567890abcdef";
|
|
32
|
+
await insertClientAudioBuffer(
|
|
33
|
+
createAudioBufferInput({ callSid, bucketStartMs: 500 }),
|
|
34
|
+
);
|
|
35
|
+
await insertClientAudioBuffer(
|
|
36
|
+
createAudioBufferInput({ callSid, bucketStartMs: 1000 }),
|
|
37
|
+
);
|
|
38
|
+
await insertClientAudioBuffer(
|
|
39
|
+
createAudioBufferInput({ callSid, bucketStartMs: 3000 }),
|
|
40
|
+
);
|
|
41
|
+
await insertClientAudioBuffer(
|
|
42
|
+
createAudioBufferInput({ callSid, bucketStartMs: 6000 }),
|
|
43
|
+
);
|
|
44
|
+
|
|
45
|
+
const result = await findClientAudioBuffersByCallSidAndRange(
|
|
46
|
+
callSid,
|
|
47
|
+
500,
|
|
48
|
+
3000,
|
|
49
|
+
);
|
|
50
|
+
expect(result).toHaveLength(3);
|
|
51
|
+
const bucketStartValues = result.map((doc) => doc.bucketStartMs);
|
|
52
|
+
const bucketStartsSorted = bucketStartValues.sort((a, b) => a - b);
|
|
53
|
+
expect(bucketStartsSorted).toEqual([500, 1000, 3000]);
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
it("should return all buffers when no range is provided", async () => {
|
|
57
|
+
const callSid = "CA1234567890abcdef1234567890abcdeg";
|
|
58
|
+
await insertClientAudioBuffer(
|
|
59
|
+
createAudioBufferInput({ callSid, bucketStartMs: 500 }),
|
|
60
|
+
);
|
|
61
|
+
await insertClientAudioBuffer(
|
|
62
|
+
createAudioBufferInput({ callSid, bucketStartMs: 1000 }),
|
|
63
|
+
);
|
|
64
|
+
await insertClientAudioBuffer(
|
|
65
|
+
createAudioBufferInput({ callSid, bucketStartMs: 3000 }),
|
|
66
|
+
);
|
|
67
|
+
|
|
68
|
+
const result = await findClientAudioBuffersByCallSidAndRange(callSid);
|
|
69
|
+
expect(result).toHaveLength(3);
|
|
70
|
+
const bucketStartValues = result.map((doc) => doc.bucketStartMs);
|
|
71
|
+
const bucketStartsSorted = bucketStartValues.sort((a, b) => a - b);
|
|
72
|
+
expect(bucketStartsSorted).toEqual([500, 1000, 3000]);
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
it("should merge audio buffers by callSid in chronological order", async () => {
|
|
76
|
+
const callSid = "CA1234567890abcdef1234567890abcdeh";
|
|
77
|
+
|
|
78
|
+
const firstBuffer = Buffer.from([1, 2]);
|
|
79
|
+
const secondBuffer = Buffer.from([3]);
|
|
80
|
+
const thirdBuffer = Buffer.from([4, 5]);
|
|
81
|
+
|
|
82
|
+
await insertClientAudioBuffer(
|
|
83
|
+
createAudioBufferInput({
|
|
84
|
+
callSid,
|
|
85
|
+
bucketStartMs: 1000,
|
|
86
|
+
data: secondBuffer,
|
|
87
|
+
}),
|
|
88
|
+
);
|
|
89
|
+
await insertClientAudioBuffer(
|
|
90
|
+
createAudioBufferInput({
|
|
91
|
+
callSid,
|
|
92
|
+
bucketStartMs: 500,
|
|
93
|
+
data: firstBuffer,
|
|
94
|
+
}),
|
|
95
|
+
);
|
|
96
|
+
await insertClientAudioBuffer(
|
|
97
|
+
createAudioBufferInput({
|
|
98
|
+
callSid,
|
|
99
|
+
bucketStartMs: 3000,
|
|
100
|
+
data: thirdBuffer,
|
|
101
|
+
}),
|
|
102
|
+
);
|
|
103
|
+
|
|
104
|
+
const merged = await getMergedAudioByCallSid(callSid, 10);
|
|
105
|
+
const expected = Buffer.concat([firstBuffer, secondBuffer, thirdBuffer]);
|
|
106
|
+
|
|
107
|
+
expect(merged).toBeInstanceOf(Buffer);
|
|
108
|
+
expect(merged.length).toBe(expected.length);
|
|
109
|
+
expect(merged.equals(expected)).toBe(true);
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
it("should return all audio chunks when less than or equal to limit buffers exist", async () => {
|
|
113
|
+
const callSid = "CA1234567890abcdef1234567890abclt";
|
|
114
|
+
|
|
115
|
+
const limit = 30;
|
|
116
|
+
const buffers: Buffer[] = [];
|
|
117
|
+
for (let i = 0; i < limit; i++) {
|
|
118
|
+
const buf = Buffer.from([i]);
|
|
119
|
+
buffers.push(buf);
|
|
120
|
+
await insertClientAudioBuffer(
|
|
121
|
+
createAudioBufferInput({
|
|
122
|
+
callSid,
|
|
123
|
+
bucketStartMs: 1000 + i * 1000,
|
|
124
|
+
data: buf,
|
|
125
|
+
}),
|
|
126
|
+
);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
const chunks = await getAudioChunksByCallSid(callSid, limit);
|
|
130
|
+
|
|
131
|
+
expect(chunks).toHaveLength(buffers.length);
|
|
132
|
+
expect(chunks.every((chunk) => chunk instanceof Buffer)).toBe(true);
|
|
133
|
+
expect(Buffer.concat(chunks).equals(Buffer.concat(buffers))).toBe(true);
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
it("should return only the last N audio chunks when more than limit buffers exist", async () => {
|
|
137
|
+
const callSid = "CA1234567890abcdef1234567890abclm";
|
|
138
|
+
|
|
139
|
+
const limit = 30;
|
|
140
|
+
const buffers: Buffer[] = [];
|
|
141
|
+
for (let i = 0; i < 35; i++) {
|
|
142
|
+
const buf = Buffer.from([i]);
|
|
143
|
+
buffers.push(buf);
|
|
144
|
+
await insertClientAudioBuffer(
|
|
145
|
+
createAudioBufferInput({
|
|
146
|
+
callSid,
|
|
147
|
+
bucketStartMs: 1000 + i * 1000,
|
|
148
|
+
data: buf,
|
|
149
|
+
}),
|
|
150
|
+
);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
const chunks = await getAudioChunksByCallSid(callSid, limit);
|
|
154
|
+
const expectedBuffers = buffers.slice(-limit);
|
|
155
|
+
expect(chunks).toHaveLength(expectedBuffers.length);
|
|
156
|
+
expect(Buffer.concat(chunks).equals(Buffer.concat(expectedBuffers))).toBe(
|
|
157
|
+
true,
|
|
158
|
+
);
|
|
159
|
+
});
|
|
160
|
+
});
|