@hivegpt/hiveai-angular 0.0.595 → 0.0.597
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/esm2020/lib/components/chat-drawer/chat-drawer.component.mjs +2 -2
- package/esm2020/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.mjs +24 -15
- package/esm2020/lib/components/voice-agent/services/audio-analyzer.service.mjs +14 -7
- package/esm2020/lib/components/voice-agent/services/voice-agent.service.mjs +84 -23
- package/fesm2015/hivegpt-hiveai-angular.mjs +131 -45
- package/fesm2015/hivegpt-hiveai-angular.mjs.map +1 -1
- package/fesm2020/hivegpt-hiveai-angular.mjs +121 -44
- package/fesm2020/hivegpt-hiveai-angular.mjs.map +1 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +6 -4
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
- package/lib/components/voice-agent/services/audio-analyzer.service.d.ts +3 -0
- package/lib/components/voice-agent/services/audio-analyzer.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +7 -0
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { isPlatformBrowser } from '@angular/common';
|
|
2
2
|
import { Inject, Injectable, PLATFORM_ID } from '@angular/core';
|
|
3
3
|
import { BehaviorSubject, Subject, Subscription } from 'rxjs';
|
|
4
|
-
import { take } from 'rxjs/operators';
|
|
4
|
+
import { distinctUntilChanged, take, throttleTime } from 'rxjs/operators';
|
|
5
5
|
import { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';
|
|
6
6
|
import { WebSocketTransport } from '@pipecat-ai/websocket-transport';
|
|
7
7
|
import * as i0 from "@angular/core";
|
|
@@ -38,17 +38,18 @@ export class VoiceAgentService {
|
|
|
38
38
|
this.durationInterval = null;
|
|
39
39
|
this.pcClient = null;
|
|
40
40
|
this.botAudioElement = null;
|
|
41
|
+
this.lifecycleToken = 0;
|
|
41
42
|
this.subscriptions = new Subscription();
|
|
42
43
|
this.destroy$ = new Subject();
|
|
43
44
|
this.callState$ = this.callStateSubject.asObservable();
|
|
44
45
|
this.statusText$ = this.statusTextSubject.asObservable();
|
|
45
46
|
this.duration$ = this.durationSubject.asObservable();
|
|
46
47
|
this.isMicMuted$ = this.isMicMutedSubject.asObservable();
|
|
47
|
-
this.isUserSpeaking$ = this.isUserSpeakingSubject.asObservable();
|
|
48
|
+
this.isUserSpeaking$ = this.isUserSpeakingSubject.asObservable().pipe(distinctUntilChanged());
|
|
48
49
|
this.audioLevels$ = this.audioLevelsSubject.asObservable();
|
|
49
50
|
this.userTranscript$ = this.userTranscriptSubject.asObservable();
|
|
50
51
|
this.botTranscript$ = this.botTranscriptSubject.asObservable();
|
|
51
|
-
this.subscriptions.add(this.audioAnalyzer.audioLevels$.subscribe((levels) => this.audioLevelsSubject.next(levels)));
|
|
52
|
+
this.subscriptions.add(this.audioAnalyzer.audioLevels$.pipe(throttleTime(50)).subscribe((levels) => this.audioLevelsSubject.next(levels)));
|
|
52
53
|
}
|
|
53
54
|
ngOnDestroy() {
|
|
54
55
|
this.destroy$.next();
|
|
@@ -57,18 +58,31 @@ export class VoiceAgentService {
|
|
|
57
58
|
}
|
|
58
59
|
/** Reset to idle (e.g. when modal re-opens so user can click Start Call). */
|
|
59
60
|
resetToIdle() {
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
61
|
+
void this.prepareFreshSession();
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* Hard reset voice session so each modal open starts from a clean state.
|
|
65
|
+
* Closes any previous client and prevents stale callbacks from mutating UI state.
|
|
66
|
+
*/
|
|
67
|
+
async prepareFreshSession() {
|
|
68
|
+
this.lifecycleToken += 1;
|
|
69
|
+
this.stopDurationTimer();
|
|
70
|
+
this.callStartTime = 0;
|
|
71
|
+
this.audioAnalyzer.stop();
|
|
72
|
+
this.stopBotAudio();
|
|
73
|
+
this.isUserSpeakingSubject.next(false);
|
|
74
|
+
this.isMicMutedSubject.next(false);
|
|
75
|
+
this.durationSubject.next('00:00');
|
|
64
76
|
this.statusTextSubject.next('');
|
|
65
|
-
this.
|
|
77
|
+
this.callStateSubject.next('idle');
|
|
78
|
+
await this.cleanupPipecatClient();
|
|
66
79
|
}
|
|
67
80
|
async connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl) {
|
|
68
81
|
if (this.callStateSubject.value !== 'idle') {
|
|
69
82
|
console.warn('[HiveGpt Voice] Call already in progress');
|
|
70
83
|
return;
|
|
71
84
|
}
|
|
85
|
+
const connectToken = ++this.lifecycleToken;
|
|
72
86
|
try {
|
|
73
87
|
this.callStateSubject.next('connecting');
|
|
74
88
|
this.statusTextSubject.next('Connecting...');
|
|
@@ -92,13 +106,32 @@ export class VoiceAgentService {
|
|
|
92
106
|
enableMic: true,
|
|
93
107
|
enableCam: false,
|
|
94
108
|
callbacks: {
|
|
95
|
-
onConnected: () => this.ngZone.run(() =>
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
109
|
+
onConnected: () => this.ngZone.run(() => {
|
|
110
|
+
if (!this.isLifecycleActive(connectToken))
|
|
111
|
+
return;
|
|
112
|
+
this.onPipecatConnected();
|
|
113
|
+
}),
|
|
114
|
+
onDisconnected: () => this.ngZone.run(() => {
|
|
115
|
+
if (!this.isLifecycleActive(connectToken))
|
|
116
|
+
return;
|
|
117
|
+
this.onPipecatDisconnected();
|
|
118
|
+
}),
|
|
119
|
+
onBotReady: () => this.ngZone.run(() => {
|
|
120
|
+
if (!this.isLifecycleActive(connectToken))
|
|
121
|
+
return;
|
|
122
|
+
this.onBotReady();
|
|
123
|
+
}),
|
|
124
|
+
onUserTranscript: (data) => this.ngZone.run(() => this.isLifecycleActive(connectToken) &&
|
|
125
|
+
this.userTranscriptSubject.next({ text: data.text, final: !!data.final })),
|
|
126
|
+
onBotTranscript: (data) => this.ngZone.run(() => {
|
|
127
|
+
if (!this.isLifecycleActive(connectToken))
|
|
128
|
+
return;
|
|
129
|
+
this.botTranscriptSubject.next(data.text);
|
|
130
|
+
}),
|
|
100
131
|
onError: (err) => {
|
|
101
132
|
this.ngZone.run(() => {
|
|
133
|
+
if (!this.isLifecycleActive(connectToken))
|
|
134
|
+
return;
|
|
102
135
|
console.error('[HiveGpt Voice] PipecatClient error', err);
|
|
103
136
|
this.callStateSubject.next('ended');
|
|
104
137
|
this.statusTextSubject.next('Connection failed');
|
|
@@ -109,19 +142,33 @@ export class VoiceAgentService {
|
|
|
109
142
|
this.pcClient = pcClient;
|
|
110
143
|
// Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element
|
|
111
144
|
pcClient.on(RTVIEvent.TrackStarted, (track, participant) => {
|
|
145
|
+
if (!this.isLifecycleActive(connectToken))
|
|
146
|
+
return;
|
|
112
147
|
if (!participant?.local && track.kind === 'audio') {
|
|
113
148
|
this.ngZone.run(() => this.setupBotAudioTrack(track));
|
|
114
149
|
}
|
|
115
150
|
});
|
|
116
151
|
// Speaking state comes straight from RTVI events
|
|
117
|
-
pcClient.on(RTVIEvent.BotStartedSpeaking, () => this.ngZone.run(() =>
|
|
118
|
-
|
|
152
|
+
pcClient.on(RTVIEvent.BotStartedSpeaking, () => this.ngZone.run(() => {
|
|
153
|
+
if (!this.isLifecycleActive(connectToken))
|
|
154
|
+
return;
|
|
155
|
+
this.onBotStartedSpeaking();
|
|
156
|
+
}));
|
|
157
|
+
pcClient.on(RTVIEvent.BotStoppedSpeaking, () => this.ngZone.run(() => {
|
|
158
|
+
if (!this.isLifecycleActive(connectToken))
|
|
159
|
+
return;
|
|
160
|
+
this.onBotStoppedSpeaking();
|
|
161
|
+
}));
|
|
119
162
|
pcClient.on(RTVIEvent.UserStartedSpeaking, () => this.ngZone.run(() => {
|
|
163
|
+
if (!this.isLifecycleActive(connectToken))
|
|
164
|
+
return;
|
|
120
165
|
this.isUserSpeakingSubject.next(true);
|
|
121
166
|
this.callStateSubject.next('listening');
|
|
122
167
|
this.statusTextSubject.next('Listening...');
|
|
123
168
|
}));
|
|
124
169
|
pcClient.on(RTVIEvent.UserStoppedSpeaking, () => this.ngZone.run(() => {
|
|
170
|
+
if (!this.isLifecycleActive(connectToken))
|
|
171
|
+
return;
|
|
125
172
|
this.isUserSpeakingSubject.next(false);
|
|
126
173
|
if (this.callStateSubject.value === 'listening') {
|
|
127
174
|
// Brief 'Processing...' while we wait for the bot to respond.
|
|
@@ -153,6 +200,8 @@ export class VoiceAgentService {
|
|
|
153
200
|
});
|
|
154
201
|
}
|
|
155
202
|
catch (error) {
|
|
203
|
+
if (!this.isLifecycleActive(connectToken))
|
|
204
|
+
return;
|
|
156
205
|
console.error('[HiveGpt Voice] connect failed', error);
|
|
157
206
|
this.callStateSubject.next('ended');
|
|
158
207
|
await this.cleanupPipecatClient();
|
|
@@ -166,8 +215,10 @@ export class VoiceAgentService {
|
|
|
166
215
|
this.startDurationTimer();
|
|
167
216
|
this.callStateSubject.next('connected');
|
|
168
217
|
this.statusTextSubject.next('Connected');
|
|
169
|
-
|
|
170
|
-
this.
|
|
218
|
+
// Preserve any mute action the user made during "connecting".
|
|
219
|
+
const shouldBeMuted = this.isMicMutedSubject.value;
|
|
220
|
+
this.pcClient?.enableMic(!shouldBeMuted);
|
|
221
|
+
this.isMicMutedSubject.next(shouldBeMuted);
|
|
171
222
|
}
|
|
172
223
|
onPipecatDisconnected() {
|
|
173
224
|
this.stopDurationTimer();
|
|
@@ -210,11 +261,14 @@ export class VoiceAgentService {
|
|
|
210
261
|
if (!this.botAudioElement) {
|
|
211
262
|
this.botAudioElement = new Audio();
|
|
212
263
|
this.botAudioElement.autoplay = true;
|
|
264
|
+
this.botAudioElement.playsInline = true;
|
|
265
|
+
this.botAudioElement.setAttribute('playsinline', '');
|
|
213
266
|
}
|
|
214
267
|
const existing = this.botAudioElement.srcObject
|
|
215
268
|
?.getAudioTracks()[0];
|
|
216
269
|
if (existing?.id === track.id)
|
|
217
270
|
return;
|
|
271
|
+
this.botAudioElement.pause();
|
|
218
272
|
this.botAudioElement.srcObject = new MediaStream([track]);
|
|
219
273
|
this.botAudioElement.play().catch((err) => console.warn('[HiveGpt Voice] Bot audio play blocked', err));
|
|
220
274
|
}
|
|
@@ -234,6 +288,7 @@ export class VoiceAgentService {
|
|
|
234
288
|
}
|
|
235
289
|
}
|
|
236
290
|
async disconnect() {
|
|
291
|
+
this.lifecycleToken += 1;
|
|
237
292
|
this.stopDurationTimer();
|
|
238
293
|
this.callStartTime = 0;
|
|
239
294
|
this.audioAnalyzer.stop();
|
|
@@ -242,6 +297,9 @@ export class VoiceAgentService {
|
|
|
242
297
|
this.callStateSubject.next('ended');
|
|
243
298
|
this.statusTextSubject.next('Call Ended');
|
|
244
299
|
}
|
|
300
|
+
isLifecycleActive(token) {
|
|
301
|
+
return token === this.lifecycleToken;
|
|
302
|
+
}
|
|
245
303
|
async cleanupPipecatClient() {
|
|
246
304
|
if (this.pcClient) {
|
|
247
305
|
try {
|
|
@@ -254,11 +312,12 @@ export class VoiceAgentService {
|
|
|
254
312
|
}
|
|
255
313
|
}
|
|
256
314
|
toggleMic() {
|
|
257
|
-
if (!this.pcClient)
|
|
258
|
-
return;
|
|
259
315
|
const nextMuted = !this.isMicMutedSubject.value;
|
|
260
|
-
|
|
316
|
+
// Reflect the user's intent immediately, even if connect is still in progress.
|
|
261
317
|
this.isMicMutedSubject.next(nextMuted);
|
|
318
|
+
if (this.pcClient) {
|
|
319
|
+
this.pcClient.enableMic(!nextMuted);
|
|
320
|
+
}
|
|
262
321
|
if (nextMuted)
|
|
263
322
|
this.isUserSpeakingSubject.next(false);
|
|
264
323
|
}
|
|
@@ -268,11 +327,13 @@ export class VoiceAgentService {
|
|
|
268
327
|
const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);
|
|
269
328
|
const m = Math.floor(elapsed / 60);
|
|
270
329
|
const s = elapsed % 60;
|
|
271
|
-
this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);
|
|
330
|
+
this.ngZone.run(() => this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`));
|
|
272
331
|
}
|
|
273
332
|
};
|
|
274
333
|
tick();
|
|
275
|
-
this.
|
|
334
|
+
this.ngZone.runOutsideAngular(() => {
|
|
335
|
+
this.durationInterval = setInterval(tick, 1000);
|
|
336
|
+
});
|
|
276
337
|
}
|
|
277
338
|
stopDurationTimer() {
|
|
278
339
|
if (this.durationInterval) {
|
|
@@ -292,4 +353,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
|
|
|
292
353
|
type: Inject,
|
|
293
354
|
args: [PLATFORM_ID]
|
|
294
355
|
}] }]; } });
|
|
295
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"","sources":["../../../../../../../../projects/hivegpt/eventsgpt-angular/src/lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,iBAAiB,CAAC;AACpD,OAAO,EAAE,MAAM,EAAE,UAAU,EAAqB,WAAW,EAAE,MAAM,eAAe,CAAC;AACnF,OAAO,EAAE,eAAe,EAAc,OAAO,EAAE,YAAY,EAAE,MAAM,MAAM,CAAC;AAC1E,OAAO,EAAE,IAAI,EAAE,MAAM,gBAAgB,CAAC;AACtC,OAAO,EAAE,aAAa,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC;AACjE,OAAO,EAAE,kBAAkB,EAAE,MAAM,iCAAiC,CAAC;;;;AAiBrE;;;;;;;;;;GAUG;AAIH,MAAM,OAAO,iBAAiB;IA4B5B,YACU,aAAmC,EACnC,oBAAiD,EACjD,MAAc;IACtB,8FAA8F;IACjE,UAAkB;QAJvC,kBAAa,GAAb,aAAa,CAAsB;QACnC,yBAAoB,GAApB,oBAAoB,CAA6B;QACjD,WAAM,GAAN,MAAM,CAAQ;QAEO,eAAU,GAAV,UAAU,CAAQ;QAhCzC,qBAAgB,GAAG,IAAI,eAAe,CAAY,MAAM,CAAC,CAAC;QAC1D,sBAAiB,GAAG,IAAI,eAAe,CAAS,EAAE,CAAC,CAAC;QACpD,oBAAe,GAAG,IAAI,eAAe,CAAS,OAAO,CAAC,CAAC;QACvD,sBAAiB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QACxD,0BAAqB,GAAG,IAAI,eAAe,CAAU,KAAK,CAAC,CAAC;QAC5D,uBAAkB,GAAG,IAAI,eAAe,CAAW,EAAE,CAAC,CAAC;QACvD,0BAAqB,GAAG,IAAI,OAAO,EAAkB,CAAC;QACtD,yBAAoB,GAAG,IAAI,OAAO,EAAU,CAAC;QAE7C,kBAAa,GAAG,CAAC,CAAC;QAClB,qBAAgB,GAA0C,IAAI,CAAC;QAE/D,aAAQ,GAAyB,IAAI,CAAC;QACtC,oBAAe,GAA4B,IAAI,CAAC;QAEhD,kBAAa,GAAG,IAAI,YAAY,EAAE,CAAC;QACnC,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAEvC,eAAU,GAA0B,IAAI,CAAC,gBAAgB,CAAC,YAAY,EAAE,CAAC;QACzE,gBAAW,GAAuB,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;QACxE,cAAS,GAAuB,IAAI,CAAC,eAAe,CAAC,YAAY,EAAE,CAAC;QACpE,gBAAW,GAAwB,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;QACzE,oBAAe,GAAwB,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QACjF,iBAAY,GAAyB,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAC5E,oBAAe,GAA+B,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QACxF,mBAAc,GAAuB,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,CAAC;QAS5E,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC,MAAM,EAAE,EAAE,CACnD,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,MAAM,CAAC,CACrC,CACF,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC;QACjC,KAAK,IAAI,CAAC,UAAU,EAAE,CAAC;IACzB,CAAC;IAED,6EAA6E;IAC7E,WAAW;QACT,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,MAAM;YAAE,OAAO;QACnD,KAAK,IAAI,CAAC,UAAU,EAAE,CAAC;QACvB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACnC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAChC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IACpC,CAAC;IAED,KAAK,CAAC,OAAO,CACX,MAAc,EACd,KAAa,EACb,KAAa,EACb,cAAsB,EACtB,MAAc,EACd,UAAkB,EAClB,OAAe,EACf,QAAgB,EAChB,eAAuB,EACvB,WAAoB;QAEpB,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,MAAM,EAAE;YAC1C,OAAO,CAAC,IAAI,CAAC,0CAA0C,CAAC,CAAC;YACzD,OAAO;SACR;QAED,IAAI;YACF,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACzC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YAE7C,IAAI,WAAW,GAAG,KAAK,CAAC;YACxB,IAAI,WAAW,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE;gBACrD,IAAI;oBACF,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,oBAAoB;yBAC5C,sBAAsB,CAAC,KAAK,EAAE,WAAW,CAAC;yBAC1C,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;yBACb,SAAS,EAAE,CAAC;oBACf,IAAI,OAAO,EAAE,WAAW;wBAAE,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;iBAC7D;gBAAC,OAAO,CAAC,EAAE;oBACV,OAAO,CAAC,IAAI,CAAC,sCAAsC,EAAE,CAAC,CAAC,CAAC;iBACzD;aACF;YAED,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;YAE1C,MAAM,QAAQ,GAAG,IAAI,aAAa,CAAC;gBACjC,SAAS,EAAE,IAAI,kBAAkB,EAAE;gBACnC,SAAS,EAAE,IAAI;gBACf,SAAS,EAAE,KAAK;gBAChB,SAAS,EAAE;oBACT,WAAW,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,EAAE,CAAC;oBACnE,cAAc,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC;oBACzE,UAAU,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;oBAC1D,gBAAgB,EAAE,CAAC,IAAI,EAAE,EAAE,CACzB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CACnB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAC1E;oBACH,eAAe,EAAE,CAAC,IAAI,EAAE,EAAE,CACxB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBAClE,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;wBACf,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;4BACnB,OAAO,CAAC,KAAK,CAAC,qCAAqC,EAAE,GAAG,CAAC,CAAC;4BAC1D,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;4BACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;wBACnD,CAAC,CAAC,CAAC;oBACL,CAAC;iBACF;aACF,CAAC,CAAC;YAEH,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;YAEzB,6EAA6E;YAC7E,QAAQ,CAAC,EAAE,CACT,SAAS,CAAC,YAAY,EACtB,CAAC,KAAuB,EAAE,WAAiC,EAAE,EAAE;gBAC7D,IAAI,CAAC,WAAW,EAAE,KAAK,IAAI,KAAK,CAAC,IAAI,KAAK,OAAO,EAAE;oBACjD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC,CAAC;iBACvD;YACH,CAAC,CACF,CAAC;YAEF,iDAAiD;YACjD,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,kBAAkB,EAAE,GAAG,EAAE,CAC7C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC,CACnD,CAAC;YACF,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,kBAAkB,EAAE,GAAG,EAAE,CAC7C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC,CACnD,CAAC;YACF,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,mBAAmB,EAAE,GAAG,EAAE,CAC9C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gBACnB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACtC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;YAC9C,CAAC,CAAC,CACH,CAAC;YACF,QAAQ,CAAC,EAAE,CAAC,SAAS,CAAC,mBAAmB,EAAE,GAAG,EAAE,CAC9C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gBACnB,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBACvC,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,WAAW,EAAE;oBAC/C,8DAA8D;oBAC9D,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;oBACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;iBAC9C;YACH,CAAC,CAAC,CACH,CAAC;YAEF,mDAAmD;YACnD,MAAM,QAAQ,CAAC,WAAW,EAAE,CAAC;YAE7B,sFAAsF;YACtF,MAAM,cAAc,GAAG,IAAI,OAAO,EAAE,CAAC;YACrC,cAAc,CAAC,MAAM,CAAC,eAAe,EAAE,UAAU,WAAW,EAAE,CAAC,CAAC;YAChE,cAAc,CAAC,MAAM,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;YAC3C,cAAc,CAAC,MAAM,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC;YAC5C,cAAc,CAAC,MAAM,CAAC,kBAAkB,EAAE,eAAe,CAAC,CAAC;YAC3D,cAAc,CAAC,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;YAC5C,cAAc,CAAC,MAAM,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;YAC1C,cAAc,CAAC,MAAM,CAAC,YAAY,EAAE,UAAU,CAAC,CAAC;YAChD,cAAc,CAAC,MAAM,CAAC,4BAA4B,EAAE,MAAM,CAAC,CAAC;YAE5D,mFAAmF;YACnF,MAAM,QAAQ,CAAC,kBAAkB,CAAC;gBAChC,QAAQ,EAAE,GAAG,OAAO,sBAAsB;gBAC1C,OAAO,EAAE,cAAc;gBACvB,WAAW,EAAE;oBACX,MAAM,EAAE,KAAK;oBACb,eAAe,EAAE,cAAc;oBAC/B,KAAK,EAAE,OAAO;iBACf;aACF,CAAC,CAAC;SACJ;QAAC,OAAO,KAAK,EAAE;YACd,OAAO,CAAC,KAAK,CAAC,gCAAgC,EAAE,KAAK,CAAC,CAAC;YACvD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACpC,MAAM,IAAI,CAAC,oBAAoB,EAAE,CAAC;YAClC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;YACjD,MAAM,KAAK,CAAC;SACb;IACH,CAAC;IAEO,kBAAkB;QACxB,gEAAgE;QAChE,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAChC,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACzC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACnC,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC/B,CAAC;IAEO,qBAAqB;QAC3B,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;QACvB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;QAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAC5C,CAAC;IAEO,UAAU;QAChB,kEAAkE;QAClE,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC7B,MAAM,QAAQ,GAAI,IAAI,CAAC,QAAQ,EAAE,MAAM,EAA6C;YAClF,EAAE,GAAG,EAAE,KAAK,CAAC;QACf,IAAI,QAAQ;YAAE,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAChD,qEAAqE;QACrE,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IAC9C,CAAC;IAEO,qBAAqB;QAC3B,MAAM,UAAU,GAAI,IAAI,CAAC,QAAQ,EAAE,MAAM,EAA+C;YACtF,EAAE,KAAK,EAAE,KAAK,CAAC;QACjB,IAAI,UAAU,EAAE;YACd,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,IAAI,WAAW,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;SACzD;IACH,CAAC;IAEO,oBAAoB;QAC1B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACtC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC1C,2DAA2D;QAC3D,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC;IAEO,oBAAoB;QAC1B,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,SAAS,EAAE;YAC7C,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACxC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;SAC7C;IACH,CAAC;IAEO,kBAAkB,CAAC,KAAuB;QAChD,IAAI,CAAC,IAAI,CAAC,eAAe,EAAE;YACzB,IAAI,CAAC,eAAe,GAAG,IAAI,KAAK,EAAE,CAAC;YACnC,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,IAAI,CAAC;SACtC;QACD,MAAM,QAAQ,GAAI,IAAI,CAAC,eAAe,CAAC,SAAgC;YACrE,EAAE,cAAc,EAAE,CAAC,CAAC,CAAC,CAAC;QACxB,IAAI,QAAQ,EAAE,EAAE,KAAK,KAAK,CAAC,EAAE;YAAE,OAAO;QAEtC,IAAI,CAAC,eAAe,CAAC,SAAS,GAAG,IAAI,WAAW,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;QAC1D,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC,GAAG,EAAE,EAAE,CACxC,OAAO,CAAC,IAAI,CAAC,wCAAwC,EAAE,GAAG,CAAC,CAC5D,CAAC;IACJ,CAAC;IAEO,YAAY;QAClB,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,IAAI;gBACF,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;gBAC5B,IAAI,CAAC,eAAe,CAAC,SAAgC;oBACpD,EAAE,cAAc,EAAE;qBACjB,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;gBAC5B,IAAI,CAAC,eAAe,CAAC,SAAS,GAAG,IAAI,CAAC;aACvC;YAAC,MAAM;gBACN,SAAS;aACV;YACD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC;SAC7B;IACH,CAAC;IAED,KAAK,CAAC,UAAU;QACd,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;QACvB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;QAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,MAAM,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAClC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IAC5C,CAAC;IAEO,KAAK,CAAC,oBAAoB;QAChC,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI;gBACF,MAAM,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;aAClC;YAAC,MAAM;gBACN,SAAS;aACV;YACD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;SACtB;IACH,CAAC;IAED,SAAS;QACP,IAAI,CAAC,IAAI,CAAC,QAAQ;YAAE,OAAO;QAC3B,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC;QAChD,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,CAAC;QACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACvC,IAAI,SAAS;YAAE,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACxD,CAAC;IAEO,kBAAkB;QACxB,MAAM,IAAI,GAAG,GAAG,EAAE;YAChB,IAAI,IAAI,CAAC,aAAa,GAAG,CAAC,EAAE;gBAC1B,MAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,aAAa,CAAC,GAAG,IAAI,CAAC,CAAC;gBACrE,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,CAAC;gBACnC,MAAM,CAAC,GAAG,OAAO,GAAG,EAAE,CAAC;gBACvB,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC;aACjE;QACH,CAAC,CAAC;QACF,IAAI,EAAE,CAAC;QACP,IAAI,CAAC,gBAAgB,GAAG,WAAW,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;IAClD,CAAC;IAEO,iBAAiB;QACvB,IAAI,IAAI,CAAC,gBAAgB,EAAE;YACzB,aAAa,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YACrC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;SAC9B;IACH,CAAC;;+GA3TU,iBAAiB,uHAiClB,WAAW;mHAjCV,iBAAiB,cAFhB,MAAM;4FAEP,iBAAiB;kBAH7B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB;;0BAkCI,MAAM;2BAAC,WAAW","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, NgZone, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';\nimport { WebSocketTransport } from '@pipecat-ai/websocket-transport';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\n\nexport type CallState =\n  | 'idle'\n  | 'connecting'\n  | 'connected'\n  | 'listening'\n  | 'talking'\n  | 'ended';\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Voice agent orchestrator using the official PipecatClient SDK.\n *\n * Audio flow (mirrors the React reference implementation):\n *  - Local mic: acquired by PipecatClient.initDevices(); local track fed to\n *    AudioAnalyzerService for waveform visualisation.\n *  - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,\n *    played through a hidden <audio> element.\n *  - All binary protobuf framing / RTVI protocol handled by\n *    @pipecat-ai/client-js + @pipecat-ai/websocket-transport.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class VoiceAgentService implements OnDestroy {\n  private callStateSubject = new BehaviorSubject<CallState>('idle');\n  private statusTextSubject = new BehaviorSubject<string>('');\n  private durationSubject = new BehaviorSubject<string>('00:00');\n  private isMicMutedSubject = new BehaviorSubject<boolean>(false);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  private callStartTime = 0;\n  private durationInterval: ReturnType<typeof setInterval> | null = null;\n\n  private pcClient: PipecatClient | null = null;\n  private botAudioElement: HTMLAudioElement | null = null;\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\n\n  callState$: Observable<CallState> = this.callStateSubject.asObservable();\n  statusText$: Observable<string> = this.statusTextSubject.asObservable();\n  duration$: Observable<string> = this.durationSubject.asObservable();\n  isMicMuted$: Observable<boolean> = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> = this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    private ngZone: NgZone,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    void this.disconnect();\n  }\n\n  /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    void this.disconnect();\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('0:00');\n  }\n\n  async connect(\n    apiUrl: string,\n    token: string,\n    botId: string,\n    conversationId: string,\n    apiKey: string,\n    eventToken: string,\n    eventId: string,\n    eventUrl: string,\n    domainAuthority: string,\n    usersApiUrl?: string,\n  ): Promise<void> {\n    if (this.callStateSubject.value !== 'idle') {\n      console.warn('[HiveGpt Voice] Call already in progress');\n      return;\n    }\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = token;\n      if (usersApiUrl && isPlatformBrowser(this.platformId)) {\n        try {\n          const ensured = await this.platformTokenRefresh\n            .ensureValidAccessToken(token, usersApiUrl)\n            .pipe(take(1))\n            .toPromise();\n          if (ensured?.accessToken) accessToken = ensured.accessToken;\n        } catch (e) {\n          console.warn('[HiveGpt Voice] Token refresh failed', e);\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n\n      const pcClient = new PipecatClient({\n        transport: new WebSocketTransport(),\n        enableMic: true,\n        enableCam: false,\n        callbacks: {\n          onConnected: () => this.ngZone.run(() => this.onPipecatConnected()),\n          onDisconnected: () => this.ngZone.run(() => this.onPipecatDisconnected()),\n          onBotReady: () => this.ngZone.run(() => this.onBotReady()),\n          onUserTranscript: (data) =>\n            this.ngZone.run(() =>\n              this.userTranscriptSubject.next({ text: data.text, final: !!data.final }),\n            ),\n          onBotTranscript: (data) =>\n            this.ngZone.run(() => this.botTranscriptSubject.next(data.text)),\n          onError: (err) => {\n            this.ngZone.run(() => {\n              console.error('[HiveGpt Voice] PipecatClient error', err);\n              this.callStateSubject.next('ended');\n              this.statusTextSubject.next('Connection failed');\n            });\n          },\n        },\n      });\n\n      this.pcClient = pcClient;\n\n      // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element\n      pcClient.on(\n        RTVIEvent.TrackStarted,\n        (track: MediaStreamTrack, participant?: { local?: boolean }) => {\n          if (!participant?.local && track.kind === 'audio') {\n            this.ngZone.run(() => this.setupBotAudioTrack(track));\n          }\n        },\n      );\n\n      // Speaking state comes straight from RTVI events\n      pcClient.on(RTVIEvent.BotStartedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStartedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.BotStoppedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStoppedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.UserStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(true);\n          this.callStateSubject.next('listening');\n          this.statusTextSubject.next('Listening...');\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(false);\n          if (this.callStateSubject.value === 'listening') {\n            // Brief 'Processing...' while we wait for the bot to respond.\n            this.callStateSubject.next('connected');\n            this.statusTextSubject.next('Processing...');\n          }\n        }),\n      );\n\n      // Acquire mic (triggers browser permission prompt)\n      await pcClient.initDevices();\n\n      // Build headers using the browser Headers API (required by pipecat's APIRequest type)\n      const requestHeaders = new Headers();\n      requestHeaders.append('Authorization', `Bearer ${accessToken}`);\n      requestHeaders.append('x-api-key', apiKey);\n      requestHeaders.append('hive-bot-id', botId);\n      requestHeaders.append('domain-authority', domainAuthority);\n      requestHeaders.append('eventUrl', eventUrl);\n      requestHeaders.append('eventId', eventId);\n      requestHeaders.append('eventToken', eventToken);\n      requestHeaders.append('ngrok-skip-browser-warning', 'true');\n\n      // POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects\n      await pcClient.startBotAndConnect({\n        endpoint: `${baseUrl}/ai/ask-voice-socket`,\n        headers: requestHeaders,\n        requestData: {\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        },\n      });\n    } catch (error) {\n      console.error('[HiveGpt Voice] connect failed', error);\n      this.callStateSubject.next('ended');\n      await this.cleanupPipecatClient();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private onPipecatConnected(): void {\n    // Start the duration timer from the moment the session is live.\n    this.callStartTime = Date.now();\n    this.startDurationTimer();\n    this.callStateSubject.next('connected');\n    this.statusTextSubject.next('Connected');\n    this.isMicMutedSubject.next(false);\n    this.startLocalMicAnalyzer();\n  }\n\n  private onPipecatDisconnected(): void {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private onBotReady(): void {\n    // Retry track wiring in case tracks weren't ready at onConnected.\n    this.startLocalMicAnalyzer();\n    const botTrack = (this.pcClient?.tracks() as { bot?: { audio?: MediaStreamTrack } })\n      ?.bot?.audio;\n    if (botTrack) this.setupBotAudioTrack(botTrack);\n    // Bot is initialised — signal that we're now waiting for user input.\n    this.statusTextSubject.next('Listening...');\n  }\n\n  private startLocalMicAnalyzer(): void {\n    const localTrack = (this.pcClient?.tracks() as { local?: { audio?: MediaStreamTrack } })\n      ?.local?.audio;\n    if (localTrack) {\n      this.audioAnalyzer.start(new MediaStream([localTrack]));\n    }\n  }\n\n  private onBotStartedSpeaking(): void {\n    this.callStateSubject.next('talking');\n    this.statusTextSubject.next('Talking...');\n    // Mark user as no longer speaking when bot takes the turn.\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private onBotStoppedSpeaking(): void {\n    if (this.callStateSubject.value === 'talking') {\n      this.callStateSubject.next('connected');\n      this.statusTextSubject.next('Listening...');\n    }\n  }\n\n  private setupBotAudioTrack(track: MediaStreamTrack): void {\n    if (!this.botAudioElement) {\n      this.botAudioElement = new Audio();\n      this.botAudioElement.autoplay = true;\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.srcObject = new MediaStream([track]);\n    this.botAudioElement.play().catch((err) =>\n      console.warn('[HiveGpt Voice] Bot audio play blocked', err),\n    );\n  }\n\n  private stopBotAudio(): void {\n    if (this.botAudioElement) {\n      try {\n        this.botAudioElement.pause();\n        (this.botAudioElement.srcObject as MediaStream | null)\n          ?.getAudioTracks()\n          .forEach((t) => t.stop());\n        this.botAudioElement.srcObject = null;\n      } catch {\n        // ignore\n      }\n      this.botAudioElement = null;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    await this.cleanupPipecatClient();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private async cleanupPipecatClient(): Promise<void> {\n    if (this.pcClient) {\n      try {\n        await this.pcClient.disconnect();\n      } catch {\n        // ignore\n      }\n      this.pcClient = null;\n    }\n  }\n\n  toggleMic(): void {\n    if (!this.pcClient) return;\n    const nextMuted = !this.isMicMutedSubject.value;\n    this.pcClient.enableMic(!nextMuted);\n    this.isMicMutedSubject.next(nextMuted);\n    if (nextMuted) this.isUserSpeakingSubject.next(false);\n  }\n\n  private startDurationTimer(): void {\n    const tick = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const m = Math.floor(elapsed / 60);\n        const s = elapsed % 60;\n        this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);\n      }\n    };\n    tick();\n    this.durationInterval = setInterval(tick, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
|
|
356
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"","sources":["../../../../../../../../projects/hivegpt/eventsgpt-angular/src/lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, NgZone, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { distinctUntilChanged, take, throttleTime } from 'rxjs/operators';\nimport { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';\nimport { WebSocketTransport } from '@pipecat-ai/websocket-transport';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\n\nexport type CallState =\n  | 'idle'\n  | 'connecting'\n  | 'connected'\n  | 'listening'\n  | 'talking'\n  | 'ended';\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Voice agent orchestrator using the official PipecatClient SDK.\n *\n * Audio flow (mirrors the React reference implementation):\n *  - Local mic: acquired by PipecatClient.initDevices(); local track fed to\n *    AudioAnalyzerService for waveform visualisation.\n *  - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,\n *    played through a hidden <audio> element.\n *  - All binary protobuf framing / RTVI protocol handled by\n *    @pipecat-ai/client-js + @pipecat-ai/websocket-transport.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class VoiceAgentService implements OnDestroy {\n  private callStateSubject = new BehaviorSubject<CallState>('idle');\n  private statusTextSubject = new BehaviorSubject<string>('');\n  private durationSubject = new BehaviorSubject<string>('00:00');\n  private isMicMutedSubject = new BehaviorSubject<boolean>(false);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  private callStartTime = 0;\n  private durationInterval: ReturnType<typeof setInterval> | null = null;\n\n  private pcClient: PipecatClient | null = null;\n  private botAudioElement: HTMLAudioElement | null = null;\n  private lifecycleToken = 0;\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\n\n  callState$: Observable<CallState> = this.callStateSubject.asObservable();\n  statusText$: Observable<string> = this.statusTextSubject.asObservable();\n  duration$: Observable<string> = this.durationSubject.asObservable();\n  isMicMuted$: Observable<boolean> = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable().pipe(distinctUntilChanged());\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> = this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    private ngZone: NgZone,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.pipe(throttleTime(50)).subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    void this.disconnect();\n  }\n\n  /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */\n  resetToIdle(): void {\n    void this.prepareFreshSession();\n  }\n\n  /**\n   * Hard reset voice session so each modal open starts from a clean state.\n   * Closes any previous client and prevents stale callbacks from mutating UI state.\n   */\n  async prepareFreshSession(): Promise<void> {\n    this.lifecycleToken += 1;\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.isUserSpeakingSubject.next(false);\n    this.isMicMutedSubject.next(false);\n    this.durationSubject.next('00:00');\n    this.statusTextSubject.next('');\n    this.callStateSubject.next('idle');\n    await this.cleanupPipecatClient();\n  }\n\n  async connect(\n    apiUrl: string,\n    token: string,\n    botId: string,\n    conversationId: string,\n    apiKey: string,\n    eventToken: string,\n    eventId: string,\n    eventUrl: string,\n    domainAuthority: string,\n    usersApiUrl?: string,\n  ): Promise<void> {\n    if (this.callStateSubject.value !== 'idle') {\n      console.warn('[HiveGpt Voice] Call already in progress');\n      return;\n    }\n\n    const connectToken = ++this.lifecycleToken;\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = token;\n      if (usersApiUrl && isPlatformBrowser(this.platformId)) {\n        try {\n          const ensured = await this.platformTokenRefresh\n            .ensureValidAccessToken(token, usersApiUrl)\n            .pipe(take(1))\n            .toPromise();\n          if (ensured?.accessToken) accessToken = ensured.accessToken;\n        } catch (e) {\n          console.warn('[HiveGpt Voice] Token refresh failed', e);\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n\n      const pcClient = new PipecatClient({\n        transport: new WebSocketTransport(),\n        enableMic: true,\n        enableCam: false,\n        callbacks: {\n          onConnected: () =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.onPipecatConnected();\n            }),\n          onDisconnected: () =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.onPipecatDisconnected();\n            }),\n          onBotReady: () =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.onBotReady();\n            }),\n          onUserTranscript: (data) =>\n            this.ngZone.run(() =>\n              this.isLifecycleActive(connectToken) &&\n              this.userTranscriptSubject.next({ text: data.text, final: !!data.final }),\n            ),\n          onBotTranscript: (data) =>\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              this.botTranscriptSubject.next(data.text);\n            }),\n          onError: (err) => {\n            this.ngZone.run(() => {\n              if (!this.isLifecycleActive(connectToken)) return;\n              console.error('[HiveGpt Voice] PipecatClient error', err);\n              this.callStateSubject.next('ended');\n              this.statusTextSubject.next('Connection failed');\n            });\n          },\n        },\n      });\n\n      this.pcClient = pcClient;\n\n      // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element\n      pcClient.on(\n        RTVIEvent.TrackStarted,\n        (track: MediaStreamTrack, participant?: { local?: boolean }) => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          if (!participant?.local && track.kind === 'audio') {\n            this.ngZone.run(() => this.setupBotAudioTrack(track));\n          }\n        },\n      );\n\n      // Speaking state comes straight from RTVI events\n      pcClient.on(RTVIEvent.BotStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.onBotStartedSpeaking();\n        }),\n      );\n      pcClient.on(RTVIEvent.BotStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.onBotStoppedSpeaking();\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.isUserSpeakingSubject.next(true);\n          this.callStateSubject.next('listening');\n          this.statusTextSubject.next('Listening...');\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          if (!this.isLifecycleActive(connectToken)) return;\n          this.isUserSpeakingSubject.next(false);\n          if (this.callStateSubject.value === 'listening') {\n            // Brief 'Processing...' while we wait for the bot to respond.\n            this.callStateSubject.next('connected');\n            this.statusTextSubject.next('Processing...');\n          }\n        }),\n      );\n\n      // Acquire mic (triggers browser permission prompt)\n      await pcClient.initDevices();\n\n      // Build headers using the browser Headers API (required by pipecat's APIRequest type)\n      const requestHeaders = new Headers();\n      requestHeaders.append('Authorization', `Bearer ${accessToken}`);\n      requestHeaders.append('x-api-key', apiKey);\n      requestHeaders.append('hive-bot-id', botId);\n      requestHeaders.append('domain-authority', domainAuthority);\n      requestHeaders.append('eventUrl', eventUrl);\n      requestHeaders.append('eventId', eventId);\n      requestHeaders.append('eventToken', eventToken);\n      requestHeaders.append('ngrok-skip-browser-warning', 'true');\n\n      // POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects\n      await pcClient.startBotAndConnect({\n        endpoint: `${baseUrl}/ai/ask-voice-socket`,\n        headers: requestHeaders,\n        requestData: {\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        },\n      });\n    } catch (error) {\n      if (!this.isLifecycleActive(connectToken)) return;\n      console.error('[HiveGpt Voice] connect failed', error);\n      this.callStateSubject.next('ended');\n      await this.cleanupPipecatClient();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private onPipecatConnected(): void {\n    // Start the duration timer from the moment the session is live.\n    this.callStartTime = Date.now();\n    this.startDurationTimer();\n    this.callStateSubject.next('connected');\n    this.statusTextSubject.next('Connected');\n    // Preserve any mute action the user made during \"connecting\".\n    const shouldBeMuted = this.isMicMutedSubject.value;\n    this.pcClient?.enableMic(!shouldBeMuted);\n    this.isMicMutedSubject.next(shouldBeMuted);\n  }\n\n  private onPipecatDisconnected(): void {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private onBotReady(): void {\n    // Retry track wiring in case tracks weren't ready at onConnected.\n    this.startLocalMicAnalyzer();\n    const botTrack = (this.pcClient?.tracks() as { bot?: { audio?: MediaStreamTrack } })\n      ?.bot?.audio;\n    if (botTrack) this.setupBotAudioTrack(botTrack);\n    // Bot is initialised — signal that we're now waiting for user input.\n    this.statusTextSubject.next('Listening...');\n  }\n\n  private startLocalMicAnalyzer(): void {\n    const localTrack = (this.pcClient?.tracks() as { local?: { audio?: MediaStreamTrack } })\n      ?.local?.audio;\n    if (localTrack) {\n      this.audioAnalyzer.start(new MediaStream([localTrack]));\n    }\n  }\n\n  private onBotStartedSpeaking(): void {\n    this.callStateSubject.next('talking');\n    this.statusTextSubject.next('Talking...');\n    // Mark user as no longer speaking when bot takes the turn.\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private onBotStoppedSpeaking(): void {\n    if (this.callStateSubject.value === 'talking') {\n      this.callStateSubject.next('connected');\n      this.statusTextSubject.next('Listening...');\n    }\n  }\n\n  private setupBotAudioTrack(track: MediaStreamTrack): void {\n    if (!this.botAudioElement) {\n      this.botAudioElement = new Audio();\n      this.botAudioElement.autoplay = true;\n      (this.botAudioElement as any).playsInline = true;\n      this.botAudioElement.setAttribute('playsinline', '');\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.pause();\n    this.botAudioElement.srcObject = new MediaStream([track]);\n    this.botAudioElement.play().catch((err) =>\n      console.warn('[HiveGpt Voice] Bot audio play blocked', err),\n    );\n  }\n\n  private stopBotAudio(): void {\n    if (this.botAudioElement) {\n      try {\n        this.botAudioElement.pause();\n        (this.botAudioElement.srcObject as MediaStream | null)\n          ?.getAudioTracks()\n          .forEach((t) => t.stop());\n        this.botAudioElement.srcObject = null;\n      } catch {\n        // ignore\n      }\n      this.botAudioElement = null;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    this.lifecycleToken += 1;\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    await this.cleanupPipecatClient();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private isLifecycleActive(token: number): boolean {\n    return token === this.lifecycleToken;\n  }\n\n  private async cleanupPipecatClient(): Promise<void> {\n    if (this.pcClient) {\n      try {\n        await this.pcClient.disconnect();\n      } catch {\n        // ignore\n      }\n      this.pcClient = null;\n    }\n  }\n\n  toggleMic(): void {\n    const nextMuted = !this.isMicMutedSubject.value;\n    // Reflect the user's intent immediately, even if connect is still in progress.\n    this.isMicMutedSubject.next(nextMuted);\n    if (this.pcClient) {\n      this.pcClient.enableMic(!nextMuted);\n    }\n    if (nextMuted) this.isUserSpeakingSubject.next(false);\n  }\n\n  private startDurationTimer(): void {\n    const tick = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const m = Math.floor(elapsed / 60);\n        const s = elapsed % 60;\n        this.ngZone.run(() => this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`));\n      }\n    };\n    tick();\n    this.ngZone.runOutsideAngular(() => {\n      this.durationInterval = setInterval(tick, 1000);\n    });\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
|