@talkpilot/core-db 1.2.0 → 1.2.2
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/README.md +95 -116
- package/README_OLD.md +160 -0
- package/dist/talkpilot/calls/calls.dashboard.d.ts +3 -0
- package/dist/talkpilot/calls/calls.dashboard.d.ts.map +1 -0
- package/dist/talkpilot/calls/calls.dashboard.js +191 -0
- package/dist/talkpilot/calls/calls.dashboard.js.map +1 -0
- package/dist/talkpilot/calls/calls.getters.d.ts +3 -3
- package/dist/talkpilot/calls/calls.getters.d.ts.map +1 -1
- package/dist/talkpilot/calls/calls.getters.js +1 -178
- package/dist/talkpilot/calls/calls.getters.js.map +1 -1
- package/dist/talkpilot/calls/calls.types.d.ts +3 -55
- package/dist/talkpilot/calls/calls.types.d.ts.map +1 -1
- package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts +36 -0
- package/dist/talkpilot/calls/dashboard/calls.dashboard.d.ts.map +1 -0
- package/dist/talkpilot/calls/dashboard/calls.dashboard.js +208 -0
- package/dist/talkpilot/calls/dashboard/calls.dashboard.js.map +1 -0
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts +66 -0
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.d.ts.map +1 -0
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js +3 -0
- package/dist/talkpilot/calls/dashboard/calls.dashboard.types.js.map +1 -0
- package/dist/talkpilot/calls/index.d.ts +1 -0
- package/dist/talkpilot/calls/index.d.ts.map +1 -1
- package/dist/talkpilot/calls/index.js +1 -0
- package/dist/talkpilot/calls/index.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts +2 -2
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js +11 -10
- package/dist/talkpilot/clientsConfig/clientsConfig.getters.js.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts +20 -9
- package/dist/talkpilot/clientsConfig/clientsConfig.types.d.ts.map +1 -1
- package/dist/talkpilot/clientsConfig/clientsConfig.types.js +6 -0
- package/dist/talkpilot/clientsConfig/clientsConfig.types.js.map +1 -1
- package/dist/talkpilot/flows/flows.schema.js +1 -1
- package/dist/talkpilot/phone_numbers/index.d.ts +2 -2
- package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts +1 -1
- package/dist/talkpilot/phone_numbers/phone_numbers.getter.d.ts.map +1 -1
- package/dist/talkpilot/phone_numbers/phone_numbers.getter.js +5 -3
- package/dist/talkpilot/phone_numbers/phone_numbers.getter.js.map +1 -1
- package/dist/talkpilot/phone_numbers/phone_numbers.schema.js +12 -12
- package/dist/talkpilot/phone_numbers/phone_numbers.types.d.ts +4 -4
- package/dist/talkpilot/results/results.getter.d.ts.map +1 -1
- package/dist/talkpilot/results/results.getter.js.map +1 -1
- package/dist/talkpilot/retry_analyze/retryAnalyze.getters.d.ts.map +1 -1
- package/dist/talkpilot/retry_analyze/retryAnalyze.getters.js.map +1 -1
- package/dist/utils/shared.types.d.ts +5 -0
- package/dist/utils/shared.types.d.ts.map +1 -0
- package/dist/utils/shared.types.js +3 -0
- package/dist/utils/shared.types.js.map +1 -0
- package/package.json +2 -1
- 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.dashboard.spec.ts +46 -0
- package/src/talkpilot/calls/__tests__/calls.spec.ts +270 -252
- package/src/talkpilot/calls/calls.getters.ts +248 -446
- package/src/talkpilot/calls/calls.types.ts +113 -171
- package/src/talkpilot/calls/dashboard/calls.dashboard.ts +243 -0
- package/src/talkpilot/calls/dashboard/calls.dashboard.types.ts +70 -0
- package/src/talkpilot/calls/index.ts +3 -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.getters.spec.ts +53 -0
- package/src/talkpilot/clientsConfig/__tests__/clientsConfig.spec.ts +197 -106
- package/src/talkpilot/clientsConfig/clientsConfig.getters.ts +55 -44
- package/src/talkpilot/clientsConfig/clientsConfig.types.ts +127 -94
- package/src/talkpilot/clientsConfig/index.ts +2 -2
- package/src/talkpilot/flows/__tests__/flows.schema.spec.ts +71 -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 +252 -247
- package/src/talkpilot/phone_numbers/index.ts +2 -2
- package/src/talkpilot/phone_numbers/phone_numbers.getter.ts +158 -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 +39 -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 +84 -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/shared.types.ts +4 -0
- package/src/utils/validation.ts +23 -23
|
@@ -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
|
+
});
|
|
@@ -1,117 +1,117 @@
|
|
|
1
|
-
import { getDb } from "../index";
|
|
2
|
-
import {
|
|
3
|
-
ClientAudioBufferDoc,
|
|
4
|
-
ClientAudioBufferType,
|
|
5
|
-
CreateAudioBufferInput,
|
|
6
|
-
} from "./clientsAudioBuffers.types";
|
|
7
|
-
import { Collection, InsertOneResult, DeleteResult } from "mongodb";
|
|
8
|
-
|
|
9
|
-
const getClientAudioBuffersCollection =
|
|
10
|
-
(): Collection<ClientAudioBufferType> => {
|
|
11
|
-
return getDb().collection<ClientAudioBufferType>("clientAudioBuffers");
|
|
12
|
-
};
|
|
13
|
-
|
|
14
|
-
/**
|
|
15
|
-
* Deletes all audio buffers associated with a specific callSid.
|
|
16
|
-
* Useful for clearing context between different tickets in the same call.
|
|
17
|
-
*/
|
|
18
|
-
const deleteAudioBuffersByCallSid = async (
|
|
19
|
-
callSid: string,
|
|
20
|
-
): Promise<DeleteResult> => {
|
|
21
|
-
return getClientAudioBuffersCollection().deleteMany({ callSid });
|
|
22
|
-
};
|
|
23
|
-
|
|
24
|
-
const insertClientAudioBuffer = async (
|
|
25
|
-
audioBufferInput: CreateAudioBufferInput,
|
|
26
|
-
): Promise<InsertOneResult<ClientAudioBufferType>> => {
|
|
27
|
-
const collection = getClientAudioBuffersCollection();
|
|
28
|
-
const realtimeDate = new Date();
|
|
29
|
-
const ttlMs = audioBufferInput.ttlMs ?? 3600_000;
|
|
30
|
-
const expiresAt = new Date(realtimeDate.getTime() + ttlMs);
|
|
31
|
-
|
|
32
|
-
const inputFieldsForDocument = {
|
|
33
|
-
callSid: audioBufferInput.callSid,
|
|
34
|
-
clientId: audioBufferInput.clientId,
|
|
35
|
-
bucketStartMs: audioBufferInput.bucketStartMs,
|
|
36
|
-
bucketDurationMs: audioBufferInput.bucketDurationMs,
|
|
37
|
-
data: audioBufferInput.data,
|
|
38
|
-
};
|
|
39
|
-
const clientAudioBufferMongoDocument: ClientAudioBufferType = {
|
|
40
|
-
...inputFieldsForDocument,
|
|
41
|
-
codec: "mulaw",
|
|
42
|
-
sampleRateHz: 8000,
|
|
43
|
-
channels: 1,
|
|
44
|
-
createdAt: realtimeDate,
|
|
45
|
-
expiresAt: expiresAt,
|
|
46
|
-
};
|
|
47
|
-
return collection.insertOne(clientAudioBufferMongoDocument);
|
|
48
|
-
};
|
|
49
|
-
|
|
50
|
-
const findClientAudioBuffersByCallSidAndRange = async (
|
|
51
|
-
callSid: string,
|
|
52
|
-
startMs?: number,
|
|
53
|
-
endMs?: number,
|
|
54
|
-
): Promise<ClientAudioBufferDoc[]> => {
|
|
55
|
-
const collection = getClientAudioBuffersCollection();
|
|
56
|
-
if (startMs === undefined) {
|
|
57
|
-
return collection.find({ callSid }).toArray();
|
|
58
|
-
}
|
|
59
|
-
const endMsToUse = endMs ?? Date.now();
|
|
60
|
-
return collection
|
|
61
|
-
.find({
|
|
62
|
-
callSid,
|
|
63
|
-
bucketStartMs: {
|
|
64
|
-
$gte: startMs,
|
|
65
|
-
$lte: endMsToUse,
|
|
66
|
-
},
|
|
67
|
-
})
|
|
68
|
-
.toArray();
|
|
69
|
-
};
|
|
70
|
-
|
|
71
|
-
const getAudioChunksByCallSid = async (
|
|
72
|
-
callSid: string,
|
|
73
|
-
limit?: number,
|
|
74
|
-
): Promise<Buffer[]> => {
|
|
75
|
-
const query = getClientAudioBuffersCollection().find({ callSid });
|
|
76
|
-
if (limit) {
|
|
77
|
-
query.sort({ bucketStartMs: -1 }).limit(limit);
|
|
78
|
-
} else {
|
|
79
|
-
query.sort({ bucketStartMs: 1 });
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
const docs = await query.toArray();
|
|
83
|
-
|
|
84
|
-
if (!docs.length) {
|
|
85
|
-
return [];
|
|
86
|
-
}
|
|
87
|
-
const orderedDocs = limit ? docs.reverse() : docs;
|
|
88
|
-
|
|
89
|
-
return orderedDocs.map((chunk) => {
|
|
90
|
-
const data: any = chunk.data;
|
|
91
|
-
if (Buffer.isBuffer(data)) return data;
|
|
92
|
-
if (data?.buffer) return Buffer.from(data.buffer);
|
|
93
|
-
return Buffer.from(data);
|
|
94
|
-
});
|
|
95
|
-
};
|
|
96
|
-
|
|
97
|
-
const getMergedAudioByCallSid = async (
|
|
98
|
-
callSid: string,
|
|
99
|
-
limit: number,
|
|
100
|
-
): Promise<Buffer> => {
|
|
101
|
-
const buffers = await getAudioChunksByCallSid(callSid, limit);
|
|
102
|
-
|
|
103
|
-
if (!buffers.length) {
|
|
104
|
-
return Buffer.alloc(0);
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
return Buffer.concat(buffers);
|
|
108
|
-
};
|
|
109
|
-
|
|
110
|
-
export {
|
|
111
|
-
insertClientAudioBuffer,
|
|
112
|
-
getClientAudioBuffersCollection,
|
|
113
|
-
findClientAudioBuffersByCallSidAndRange,
|
|
114
|
-
getAudioChunksByCallSid,
|
|
115
|
-
getMergedAudioByCallSid,
|
|
116
|
-
deleteAudioBuffersByCallSid,
|
|
117
|
-
};
|
|
1
|
+
import { getDb } from "../index";
|
|
2
|
+
import {
|
|
3
|
+
ClientAudioBufferDoc,
|
|
4
|
+
ClientAudioBufferType,
|
|
5
|
+
CreateAudioBufferInput,
|
|
6
|
+
} from "./clientsAudioBuffers.types";
|
|
7
|
+
import { Collection, InsertOneResult, DeleteResult } from "mongodb";
|
|
8
|
+
|
|
9
|
+
const getClientAudioBuffersCollection =
|
|
10
|
+
(): Collection<ClientAudioBufferType> => {
|
|
11
|
+
return getDb().collection<ClientAudioBufferType>("clientAudioBuffers");
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Deletes all audio buffers associated with a specific callSid.
|
|
16
|
+
* Useful for clearing context between different tickets in the same call.
|
|
17
|
+
*/
|
|
18
|
+
const deleteAudioBuffersByCallSid = async (
|
|
19
|
+
callSid: string,
|
|
20
|
+
): Promise<DeleteResult> => {
|
|
21
|
+
return getClientAudioBuffersCollection().deleteMany({ callSid });
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
const insertClientAudioBuffer = async (
|
|
25
|
+
audioBufferInput: CreateAudioBufferInput,
|
|
26
|
+
): Promise<InsertOneResult<ClientAudioBufferType>> => {
|
|
27
|
+
const collection = getClientAudioBuffersCollection();
|
|
28
|
+
const realtimeDate = new Date();
|
|
29
|
+
const ttlMs = audioBufferInput.ttlMs ?? 3600_000;
|
|
30
|
+
const expiresAt = new Date(realtimeDate.getTime() + ttlMs);
|
|
31
|
+
|
|
32
|
+
const inputFieldsForDocument = {
|
|
33
|
+
callSid: audioBufferInput.callSid,
|
|
34
|
+
clientId: audioBufferInput.clientId,
|
|
35
|
+
bucketStartMs: audioBufferInput.bucketStartMs,
|
|
36
|
+
bucketDurationMs: audioBufferInput.bucketDurationMs,
|
|
37
|
+
data: audioBufferInput.data,
|
|
38
|
+
};
|
|
39
|
+
const clientAudioBufferMongoDocument: ClientAudioBufferType = {
|
|
40
|
+
...inputFieldsForDocument,
|
|
41
|
+
codec: "mulaw",
|
|
42
|
+
sampleRateHz: 8000,
|
|
43
|
+
channels: 1,
|
|
44
|
+
createdAt: realtimeDate,
|
|
45
|
+
expiresAt: expiresAt,
|
|
46
|
+
};
|
|
47
|
+
return collection.insertOne(clientAudioBufferMongoDocument);
|
|
48
|
+
};
|
|
49
|
+
|
|
50
|
+
const findClientAudioBuffersByCallSidAndRange = async (
|
|
51
|
+
callSid: string,
|
|
52
|
+
startMs?: number,
|
|
53
|
+
endMs?: number,
|
|
54
|
+
): Promise<ClientAudioBufferDoc[]> => {
|
|
55
|
+
const collection = getClientAudioBuffersCollection();
|
|
56
|
+
if (startMs === undefined) {
|
|
57
|
+
return collection.find({ callSid }).toArray();
|
|
58
|
+
}
|
|
59
|
+
const endMsToUse = endMs ?? Date.now();
|
|
60
|
+
return collection
|
|
61
|
+
.find({
|
|
62
|
+
callSid,
|
|
63
|
+
bucketStartMs: {
|
|
64
|
+
$gte: startMs,
|
|
65
|
+
$lte: endMsToUse,
|
|
66
|
+
},
|
|
67
|
+
})
|
|
68
|
+
.toArray();
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
const getAudioChunksByCallSid = async (
|
|
72
|
+
callSid: string,
|
|
73
|
+
limit?: number,
|
|
74
|
+
): Promise<Buffer[]> => {
|
|
75
|
+
const query = getClientAudioBuffersCollection().find({ callSid });
|
|
76
|
+
if (limit) {
|
|
77
|
+
query.sort({ bucketStartMs: -1 }).limit(limit);
|
|
78
|
+
} else {
|
|
79
|
+
query.sort({ bucketStartMs: 1 });
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
const docs = await query.toArray();
|
|
83
|
+
|
|
84
|
+
if (!docs.length) {
|
|
85
|
+
return [];
|
|
86
|
+
}
|
|
87
|
+
const orderedDocs = limit ? docs.reverse() : docs;
|
|
88
|
+
|
|
89
|
+
return orderedDocs.map((chunk) => {
|
|
90
|
+
const data: any = chunk.data;
|
|
91
|
+
if (Buffer.isBuffer(data)) return data;
|
|
92
|
+
if (data?.buffer) return Buffer.from(data.buffer);
|
|
93
|
+
return Buffer.from(data);
|
|
94
|
+
});
|
|
95
|
+
};
|
|
96
|
+
|
|
97
|
+
const getMergedAudioByCallSid = async (
|
|
98
|
+
callSid: string,
|
|
99
|
+
limit: number,
|
|
100
|
+
): Promise<Buffer> => {
|
|
101
|
+
const buffers = await getAudioChunksByCallSid(callSid, limit);
|
|
102
|
+
|
|
103
|
+
if (!buffers.length) {
|
|
104
|
+
return Buffer.alloc(0);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
return Buffer.concat(buffers);
|
|
108
|
+
};
|
|
109
|
+
|
|
110
|
+
export {
|
|
111
|
+
insertClientAudioBuffer,
|
|
112
|
+
getClientAudioBuffersCollection,
|
|
113
|
+
findClientAudioBuffersByCallSidAndRange,
|
|
114
|
+
getAudioChunksByCallSid,
|
|
115
|
+
getMergedAudioByCallSid,
|
|
116
|
+
deleteAudioBuffersByCallSid,
|
|
117
|
+
};
|
|
@@ -1,25 +1,25 @@
|
|
|
1
|
-
import { WithId } from "mongodb";
|
|
2
|
-
|
|
3
|
-
export type ClientAudioBufferType = {
|
|
4
|
-
callSid: string;
|
|
5
|
-
clientId: string;
|
|
6
|
-
bucketStartMs: number;
|
|
7
|
-
bucketDurationMs: number;
|
|
8
|
-
codec: "mulaw";
|
|
9
|
-
sampleRateHz: 8000;
|
|
10
|
-
channels: 1;
|
|
11
|
-
data: Buffer;
|
|
12
|
-
expiresAt: Date;
|
|
13
|
-
createdAt: Date;
|
|
14
|
-
};
|
|
15
|
-
|
|
16
|
-
export type CreateAudioBufferInput = {
|
|
17
|
-
callSid: string;
|
|
18
|
-
clientId: string;
|
|
19
|
-
bucketStartMs: number;
|
|
20
|
-
bucketDurationMs: number;
|
|
21
|
-
data: Buffer;
|
|
22
|
-
ttlMs?: number;
|
|
23
|
-
};
|
|
24
|
-
|
|
25
|
-
export type ClientAudioBufferDoc = WithId<ClientAudioBufferType>;
|
|
1
|
+
import { WithId } from "mongodb";
|
|
2
|
+
|
|
3
|
+
export type ClientAudioBufferType = {
|
|
4
|
+
callSid: string;
|
|
5
|
+
clientId: string;
|
|
6
|
+
bucketStartMs: number;
|
|
7
|
+
bucketDurationMs: number;
|
|
8
|
+
codec: "mulaw";
|
|
9
|
+
sampleRateHz: 8000;
|
|
10
|
+
channels: 1;
|
|
11
|
+
data: Buffer;
|
|
12
|
+
expiresAt: Date;
|
|
13
|
+
createdAt: Date;
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
export type CreateAudioBufferInput = {
|
|
17
|
+
callSid: string;
|
|
18
|
+
clientId: string;
|
|
19
|
+
bucketStartMs: number;
|
|
20
|
+
bucketDurationMs: number;
|
|
21
|
+
data: Buffer;
|
|
22
|
+
ttlMs?: number;
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
export type ClientAudioBufferDoc = WithId<ClientAudioBufferType>;
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
export * from "./clientAudioBuffer.getters";
|
|
2
|
-
export * from "./clientsAudioBuffers.types";
|
|
1
|
+
export * from "./clientAudioBuffer.getters";
|
|
2
|
+
export * from "./clientsAudioBuffers.types";
|
|
@@ -1,16 +1,16 @@
|
|
|
1
|
-
import { getDb, Client } from "../index";
|
|
2
|
-
import { Collection } from "mongodb";
|
|
3
|
-
|
|
4
|
-
export const getClientsCollection = (): Collection<Client> => {
|
|
5
|
-
return getDb().collection<Client>("clients");
|
|
6
|
-
};
|
|
7
|
-
|
|
8
|
-
export const getClientByAPIKey = async (
|
|
9
|
-
apiKey: string,
|
|
10
|
-
): Promise<Client | null> => {
|
|
11
|
-
const client = await getClientsCollection().findOne({
|
|
12
|
-
apiKey,
|
|
13
|
-
isActive: true,
|
|
14
|
-
});
|
|
15
|
-
return client ? client : null;
|
|
16
|
-
};
|
|
1
|
+
import { getDb, Client } from "../index";
|
|
2
|
+
import { Collection } from "mongodb";
|
|
3
|
+
|
|
4
|
+
export const getClientsCollection = (): Collection<Client> => {
|
|
5
|
+
return getDb().collection<Client>("clients");
|
|
6
|
+
};
|
|
7
|
+
|
|
8
|
+
export const getClientByAPIKey = async (
|
|
9
|
+
apiKey: string,
|
|
10
|
+
): Promise<Client | null> => {
|
|
11
|
+
const client = await getClientsCollection().findOne({
|
|
12
|
+
apiKey,
|
|
13
|
+
isActive: true,
|
|
14
|
+
});
|
|
15
|
+
return client ? client : null;
|
|
16
|
+
};
|
|
@@ -1,14 +1,14 @@
|
|
|
1
|
-
import { ObjectId } from "mongodb";
|
|
2
|
-
|
|
3
|
-
export type Client = {
|
|
4
|
-
_id?: ObjectId;
|
|
5
|
-
createdAt: Date;
|
|
6
|
-
updatedAt: Date;
|
|
7
|
-
clientId: string;
|
|
8
|
-
isActive?: boolean | null;
|
|
9
|
-
apiKey?: string | null;
|
|
10
|
-
credits: number;
|
|
11
|
-
managedBy?: string;
|
|
12
|
-
};
|
|
13
|
-
|
|
14
|
-
export type ClientDoc = Client & { _id: ObjectId };
|
|
1
|
+
import { ObjectId } from "mongodb";
|
|
2
|
+
|
|
3
|
+
export type Client = {
|
|
4
|
+
_id?: ObjectId;
|
|
5
|
+
createdAt: Date;
|
|
6
|
+
updatedAt: Date;
|
|
7
|
+
clientId: string;
|
|
8
|
+
isActive?: boolean | null;
|
|
9
|
+
apiKey?: string | null;
|
|
10
|
+
credits: number;
|
|
11
|
+
managedBy?: string;
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
export type ClientDoc = Client & { _id: ObjectId };
|