@hivegpt/hiveai-angular 0.0.582 → 0.0.583

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.
@@ -34,6 +34,9 @@ export class VoiceAgentService {
34
34
  this.callStartTime = 0;
35
35
  this.durationInterval = null;
36
36
  this.localMicStream = null;
37
+ this.remoteAudioContext = null;
38
+ this.pendingRemoteAudio = [];
39
+ this.remoteAudioPlaying = false;
37
40
  this.endCall$ = new Subject();
38
41
  this.subscriptions = new Subscription();
39
42
  this.destroy$ = new Subject();
@@ -49,6 +52,9 @@ export class VoiceAgentService {
49
52
  this.subscriptions.add(this.wsClient.remoteClose$
50
53
  .pipe(takeUntil(this.destroy$))
51
54
  .subscribe(() => void this.handleRemoteClose()));
55
+ this.subscriptions.add(this.wsClient.audioChunk$
56
+ .pipe(takeUntil(this.destroy$))
57
+ .subscribe((chunk) => this.enqueueRemoteAudio(chunk)));
52
58
  }
53
59
  ngOnDestroy() {
54
60
  this.destroy$.next();
@@ -64,6 +70,7 @@ export class VoiceAgentService {
64
70
  this.callStartTime = 0;
65
71
  this.audioAnalyzer.stop();
66
72
  this.stopLocalMic();
73
+ this.resetRemoteAudioPlayback();
67
74
  this.wsClient.disconnect();
68
75
  this.callStateSubject.next('idle');
69
76
  this.statusTextSubject.next('');
@@ -218,6 +225,76 @@ export class VoiceAgentService {
218
225
  this.localMicStream = null;
219
226
  }
220
227
  }
228
+ enqueueRemoteAudio(chunk) {
229
+ this.pendingRemoteAudio.push(chunk.slice(0));
230
+ if (!this.remoteAudioPlaying) {
231
+ void this.playRemoteAudioQueue();
232
+ }
233
+ }
234
+ playRemoteAudioQueue() {
235
+ return __awaiter(this, void 0, void 0, function* () {
236
+ this.remoteAudioPlaying = true;
237
+ const context = this.getOrCreateRemoteAudioContext();
238
+ while (this.pendingRemoteAudio.length > 0) {
239
+ const chunk = this.pendingRemoteAudio.shift();
240
+ if (!chunk)
241
+ continue;
242
+ try {
243
+ const decoded = yield this.decodeAudioChunk(context, chunk);
244
+ this.assistantAudioStarted();
245
+ yield this.playDecodedBuffer(context, decoded);
246
+ }
247
+ catch (_a) {
248
+ // Ignore undecodable chunks; server may mix non-audio binary events.
249
+ }
250
+ }
251
+ this.remoteAudioPlaying = false;
252
+ this.assistantAudioStopped();
253
+ });
254
+ }
255
+ getOrCreateRemoteAudioContext() {
256
+ if (!this.remoteAudioContext || this.remoteAudioContext.state === 'closed') {
257
+ this.remoteAudioContext = new AudioContext();
258
+ }
259
+ if (this.remoteAudioContext.state === 'suspended') {
260
+ void this.remoteAudioContext.resume();
261
+ }
262
+ return this.remoteAudioContext;
263
+ }
264
+ decodeAudioChunk(context, chunk) {
265
+ return new Promise((resolve, reject) => {
266
+ context.decodeAudioData(chunk.slice(0), resolve, reject);
267
+ });
268
+ }
269
+ playDecodedBuffer(context, buffer) {
270
+ return new Promise((resolve) => {
271
+ const source = context.createBufferSource();
272
+ source.buffer = buffer;
273
+ source.connect(context.destination);
274
+ source.onended = () => resolve();
275
+ source.start();
276
+ });
277
+ }
278
+ assistantAudioStarted() {
279
+ if (this.callStartTime === 0) {
280
+ this.callStartTime = Date.now();
281
+ this.startDurationTimer();
282
+ }
283
+ this.callStateSubject.next('talking');
284
+ }
285
+ assistantAudioStopped() {
286
+ if (this.callStateSubject.value === 'talking') {
287
+ this.callStateSubject.next('connected');
288
+ }
289
+ }
290
+ resetRemoteAudioPlayback() {
291
+ this.pendingRemoteAudio = [];
292
+ this.remoteAudioPlaying = false;
293
+ if (this.remoteAudioContext && this.remoteAudioContext.state !== 'closed') {
294
+ this.remoteAudioContext.close().catch(() => { });
295
+ }
296
+ this.remoteAudioContext = null;
297
+ }
221
298
  handleRemoteClose() {
222
299
  return __awaiter(this, void 0, void 0, function* () {
223
300
  const state = this.callStateSubject.value;
@@ -228,6 +305,7 @@ export class VoiceAgentService {
228
305
  this.callStartTime = 0;
229
306
  this.audioAnalyzer.stop();
230
307
  this.stopLocalMic();
308
+ this.resetRemoteAudioPlayback();
231
309
  this.callStateSubject.next('ended');
232
310
  this.statusTextSubject.next('Connection lost');
233
311
  });
@@ -239,6 +317,7 @@ export class VoiceAgentService {
239
317
  this.callStartTime = 0;
240
318
  this.audioAnalyzer.stop();
241
319
  this.stopLocalMic();
320
+ this.resetRemoteAudioPlayback();
242
321
  this.wsClient.disconnect();
243
322
  this.callStateSubject.next('ended');
244
323
  this.statusTextSubject.next('Call Ended');
@@ -284,4 +363,4 @@ VoiceAgentService.ctorParameters = () => [
284
363
  { type: PlatformTokenRefreshService },
285
364
  { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
286
365
  ];
287
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["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,EAAa,WAAW,EAAE,MAAM,eAAe,CAAC;AAC3E,OAAO,EACL,eAAe,EACf,aAAa,EACb,MAAM,EACN,KAAK,EAEL,EAAE,EACF,OAAO,EACP,YAAY,EACZ,KAAK,GACN,MAAM,MAAM,CAAC;AACd,OAAO,EACL,oBAAoB,EACpB,GAAG,EACH,SAAS,EACT,SAAS,EACT,IAAI,EACJ,SAAS,GACV,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAE,2BAA2B,EAAE,MAAM,kDAAkD,CAAC;AAC/F,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,2BAA2B,EAAE,MAAM,kCAAkC,CAAC;;;;;AAe/E;;;;GAIG;AAIH,MAAM,OAAO,iBAAiB;IA8B5B,YACU,aAAmC,EACnC,QAAqC,EACrC,oBAAiD;IACzD,8FAA8F;IACjE,UAAkB;QAJvC,kBAAa,GAAb,aAAa,CAAsB;QACnC,aAAQ,GAAR,QAAQ,CAA6B;QACrC,yBAAoB,GAApB,oBAAoB,CAA6B;QAE5B,eAAU,GAAV,UAAU,CAAQ;QAlCzC,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,mBAAc,GAAuB,IAAI,CAAC;QACjC,aAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;QAExC,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,GACb,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QAC5C,iBAAY,GAAyB,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAC5E,oBAAe,GACb,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QAC5C,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;QACF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,QAAQ,CAAC,YAAY;aACvB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC,GAAG,EAAE,CAAC,KAAK,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAClD,CAAC;IACJ,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,aAAa,CAAC,WAAW,EAAE,CAAC;QACjC,IAAI,CAAC,UAAU,EAAE,CAAC;IACpB,CAAC;IAED,gFAAgF;IAChF,WAAW;QACT,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,MAAM;YAAE,OAAO;QACnD,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,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,QAAQ,CAAC,UAAU,EAAE,CAAC;QAC3B,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;IAEK,OAAO,CACX,MAAc,EACd,KAAa,EACb,KAAa,EACb,cAAsB,EACtB,MAAc,EACd,UAAkB,EAClB,OAAe,EACf,QAAgB,EAChB,eAAuB,EACvB,WAAoB;;YAEpB,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,MAAM,EAAE;gBAC1C,OAAO,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC;gBACzC,OAAO;aACR;YAED,IAAI;gBACF,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;gBACzC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;gBAE7C,IAAI,WAAW,GAAG,KAAK,CAAC;gBACxB,IAAI,WAAW,IAAI,iBAAiB,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE;oBACrD,IAAI;wBACF,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,oBAAoB;6BAC5C,sBAAsB,CAAC,KAAK,EAAE,WAAW,CAAC;6BAC1C,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;6BACb,SAAS,EAAE,CAAC;wBACf,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,WAAW,EAAE;4BACxB,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC;yBACnC;qBACF;oBAAC,OAAO,CAAC,EAAE;wBACV,OAAO,CAAC,IAAI,CACV,qDAAqD,EACrD,CAAC,CACF,CAAC;qBACH;iBACF;gBAED,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;gBAC1C,MAAM,OAAO,GAAG,GAAG,OAAO,sBAAsB,CAAC;gBAEjD,MAAM,OAAO,GAA2B;oBACtC,cAAc,EAAE,kBAAkB;oBAClC,aAAa,EAAE,UAAU,WAAW,EAAE;oBACtC,WAAW,EAAE,MAAM;oBACnB,aAAa,EAAE,KAAK;oBACpB,kBAAkB,EAAE,eAAe;oBACnC,QAAQ;oBACR,OAAO;oBACP,UAAU;oBACV,4BAA4B,EAAE,MAAM;iBACrC,CAAC;gBAEF,MAAM,GAAG,GAAG,MAAM,KAAK,CAAC,OAAO,EAAE;oBAC/B,MAAM,EAAE,MAAM;oBACd,OAAO;oBACP,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;wBACnB,MAAM,EAAE,KAAK;wBACb,eAAe,EAAE,cAAc;wBAC/B,KAAK,EAAE,OAAO;qBACf,CAAC;iBACH,CAAC,CAAC;gBAEH,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE;oBACX,MAAM,IAAI,KAAK,CAAC,QAAQ,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC;iBACvC;gBAED,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,IAAI,EAAE,CAAC;gBAC9B,MAAM,KAAK,GACT,CAAC,OAAO,CAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,MAAM,CAAA,KAAK,QAAQ,IAAI,IAAI,CAAC,MAAM,CAAC;oBACjD,CAAC,OAAO,CAAA,IAAI,aAAJ,IAAI,uBAAJ,IAAI,CAAE,SAAS,CAAA,KAAK,QAAQ,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC;gBAC1D,IAAI,CAAC,KAAK,EAAE;oBACV,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC;iBAC1C;gBAED,MAAM,cAAc,GAAG,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;gBAE3D,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,QAAQ,CAAC,eAAe;qBAC1B,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;qBAC/B,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CACxD,CAAC;gBACF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,QAAQ,CAAC,cAAc;qBACzB,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;qBAC/B,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CACvD,CAAC;gBAEF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,QAAQ,CAAC,OAAO;qBAClB,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC;qBACxC,SAAS,CAAC,GAAG,EAAE,CAAC,KAAK,IAAI,CAAC,iBAAiB,EAAE,CAAC,CAClD,CAAC;gBAEF,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;aAC9B;YAAC,OAAO,KAAK,EAAE;gBACd,OAAO,CAAC,KAAK,CAAC,+BAA+B,EAAE,KAAK,CAAC,CAAC;gBACtD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBACpC,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;gBACxB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;gBACjD,MAAM,KAAK,CAAC;aACb;QACH,CAAC;KAAA;IAEa,iBAAiB;;YAC7B,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,YAAY,EAAE;gBAChD,OAAO;aACR;YACD,IAAI;gBACF,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;gBAC3B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACzC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACxC,IAAI,CAAC,iBAAiB,EAAE,CAAC;aAC1B;YAAC,OAAO,GAAG,EAAE;gBACZ,OAAO,CAAC,KAAK,CAAC,6CAA6C,EAAE,GAAG,CAAC,CAAC;gBAClE,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;gBACtD,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;aACzB;QACH,CAAC;KAAA;IAEO,iBAAiB;QACvB,MAAM,cAAc,GAAG,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAE3D,MAAM,0BAA0B,GAAG,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,CAClE,SAAS,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EACpE,oBAAoB,EAAE,CACvB,CAAC;QAEF,MAAM,iBAAiB,GAAG,KAAK,CAC7B,IAAI,CAAC,QAAQ,CAAC,kBAAkB,EAChC,0BAA0B,CAC3B,CAAC,IAAI,CAAC,oBAAoB,EAAE,EAAE,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC;QAEjD,MAAM,YAAY,GAAG,aAAa,CAAC;YACjC,IAAI,CAAC,aAAa,CAAC,eAAe;YAClC,IAAI,CAAC,QAAQ,CAAC,mBAAmB,CAAC,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;SACzD,CAAC,CAAC,IAAI,CACL,GAAG,CAAC,CAAC,CAAC,KAAK,EAAE,MAAM,CAAC,EAAE,EAAE,CAAC,KAAK,IAAI,MAAM,CAAC,EACzC,oBAAoB,EAAE,EACtB,SAAS,CAAC,KAAK,CAAC,CACjB,CAAC;QAEF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,aAAa,CAAC,CAAC,iBAAiB,EAAE,YAAY,CAAC,CAAC;aAC7C,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC;aAC/B,SAAS,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,EAAE;YACzB,MAAM,OAAO,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC;YAC5C,IAAI,IAAI,EAAE;gBACR,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBACtC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;aACzC;iBAAM;gBACL,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aACxC;YACD,IAAI,IAAI,EAAE;gBACR,OAAO;aACR;YACD,IAAI,GAAG,EAAE;gBACP,IAAI,IAAI,CAAC,aAAa,KAAK,CAAC,EAAE;oBAC5B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;oBAChC,IAAI,CAAC,kBAAkB,EAAE,CAAC;iBAC3B;gBACD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;aACvC;iBAAM,IAAI,OAAO,KAAK,SAAS,IAAI,OAAO,KAAK,WAAW,EAAE;gBAC3D,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;aACzC;QACH,CAAC,CAAC,CACL,CAAC;IACJ,CAAC;IAEa,aAAa;;YACzB,IAAI,CAAC,YAAY,EAAE,CAAC;YACpB,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,YAAY,CAAC,YAAY,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;YAC1E,MAAM,KAAK,GAAG,MAAM,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,CAAC;YACzC,IAAI,CAAC,KAAK,EAAE;gBACV,MAAM,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;gBAC5C,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;aACnC;YACD,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC;YAC7B,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAC5C,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACnC,CAAC;KAAA;IAEO,YAAY;QAClB,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,IAAI,CAAC,cAAc,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;YACzD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;SAC5B;IACH,CAAC;IAEa,iBAAiB;;YAC7B,MAAM,KAAK,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC;YAC1C,IAAI,KAAK,KAAK,MAAM,IAAI,KAAK,KAAK,OAAO;gBAAE,OAAO;YAClD,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;YACrB,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;YACvB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;YAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;YACpB,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACjD,CAAC;KAAA;IAEK,UAAU;;YACd,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;YACrB,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC;YACvB,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;YAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;YACpB,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;YAE3B,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACpC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAC5C,CAAC;KAAA;IAED,SAAS;;QACP,MAAM,SAAS,GAAG,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC;QAChD,MAAM,KAAK,GAAG,MAAA,IAAI,CAAC,cAAc,0CAAE,cAAc,GAAG,CAAC,CAAC,CAAC;QACvD,IAAI,KAAK,EAAE;YACT,KAAK,CAAC,OAAO,GAAG,CAAC,SAAS,CAAC;SAC5B;QACD,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACzC,CAAC;IAEO,kBAAkB;QACxB,MAAM,cAAc,GAAG,GAAG,EAAE;YAC1B,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,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,EAAE,CAAC,CAAC;gBACzC,MAAM,OAAO,GAAG,OAAO,GAAG,EAAE,CAAC;gBAC7B,IAAI,CAAC,eAAe,CAAC,IAAI,CACvB,GAAG,OAAO,IAAI,MAAM,CAAC,OAAO,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CACjD,CAAC;aACH;QACH,CAAC,CAAC;QACF,cAAc,EAAE,CAAC;QACjB,IAAI,CAAC,gBAAgB,GAAG,WAAW,CAAC,cAAc,EAAE,IAAI,CAAC,CAAC;IAC5D,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;;;;YA5TF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YAvBQ,oBAAoB;YACpB,2BAA2B;YAF3B,2BAA2B;YA4DS,MAAM,uBAA9C,MAAM,SAAC,WAAW","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  concat,\n  merge,\n  Observable,\n  of,\n  Subject,\n  Subscription,\n  timer,\n} from 'rxjs';\nimport {\n  distinctUntilChanged,\n  map,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n} from 'rxjs/operators';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\nimport { WebSocketVoiceClientService } from './websocket-voice-client.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: single WebSocket (`ws_url` from POST /ai/ask-voice-socket)\n * for session events, transcripts, and optional speaking hints; local mic for capture\n * and waveform only (no Daily/WebRTC room).\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 localMicStream: MediaStream | null = null;\n  private readonly endCall$ = new Subject<void>();\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> =\n    this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private wsClient: WebSocketVoiceClientService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\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    this.subscriptions.add(\n      this.wsClient.remoteClose$\n        .pipe(takeUntil(this.destroy$))\n        .subscribe(() => void this.handleRemoteClose()),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    this.disconnect();\n  }\n\n  /** Reset to idle state (e.g. when modal opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.wsClient.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('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) {\n            accessToken = ensured.accessToken;\n          }\n        } catch (e) {\n          console.warn(\n            '[HiveGpt Voice] Token refresh before connect failed',\n            e,\n          );\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n      const postUrl = `${baseUrl}/ai/ask-voice-socket`;\n\n      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n        Authorization: `Bearer ${accessToken}`,\n        'x-api-key': apiKey,\n        'hive-bot-id': botId,\n        'domain-authority': domainAuthority,\n        eventUrl,\n        eventId,\n        eventToken,\n        'ngrok-skip-browser-warning': 'true',\n      };\n\n      const res = await fetch(postUrl, {\n        method: 'POST',\n        headers,\n        body: JSON.stringify({\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        }),\n      });\n\n      if (!res.ok) {\n        throw new Error(`HTTP ${res.status}`);\n      }\n\n      const json = await res.json();\n      const wsUrl =\n        (typeof json?.ws_url === 'string' && json.ws_url) ||\n        (typeof json?.rn_ws_url === 'string' && json.rn_ws_url);\n      if (!wsUrl) {\n        throw new Error('No ws_url in response');\n      }\n\n      const untilCallEnds$ = merge(this.destroy$, this.endCall$);\n\n      this.subscriptions.add(\n        this.wsClient.userTranscript$\n          .pipe(takeUntil(untilCallEnds$))\n          .subscribe((t) => this.userTranscriptSubject.next(t)),\n      );\n      this.subscriptions.add(\n        this.wsClient.botTranscript$\n          .pipe(takeUntil(untilCallEnds$))\n          .subscribe((t) => this.botTranscriptSubject.next(t)),\n      );\n\n      this.subscriptions.add(\n        this.wsClient.opened$\n          .pipe(takeUntil(untilCallEnds$), take(1))\n          .subscribe(() => void this.onWebsocketOpened()),\n      );\n\n      this.wsClient.connect(wsUrl);\n    } catch (error) {\n      console.error('Error connecting voice agent:', error);\n      this.callStateSubject.next('ended');\n      await this.disconnect();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private async onWebsocketOpened(): Promise<void> {\n    if (this.callStateSubject.value !== 'connecting') {\n      return;\n    }\n    try {\n      await this.startLocalMic();\n      this.statusTextSubject.next('Connected');\n      this.callStateSubject.next('connected');\n      this.wireSpeakingState();\n    } catch (err) {\n      console.error('[HiveGpt Voice] Mic or session setup failed', err);\n      this.callStateSubject.next('ended');\n      this.statusTextSubject.next('Microphone unavailable');\n      await this.disconnect();\n    }\n  }\n\n  private wireSpeakingState(): void {\n    const untilCallEnds$ = merge(this.destroy$, this.endCall$);\n\n    const transcriptDrivenAssistant$ = this.wsClient.botTranscript$.pipe(\n      switchMap(() => concat(of(true), timer(800).pipe(map(() => false)))),\n      distinctUntilChanged(),\n    );\n\n    const assistantTalking$ = merge(\n      this.wsClient.assistantSpeaking$,\n      transcriptDrivenAssistant$,\n    ).pipe(distinctUntilChanged(), startWith(false));\n\n    const userTalking$ = combineLatest([\n      this.audioAnalyzer.isUserSpeaking$,\n      this.wsClient.serverUserSpeaking$.pipe(startWith(false)),\n    ]).pipe(\n      map(([local, server]) => local || server),\n      distinctUntilChanged(),\n      startWith(false),\n    );\n\n    this.subscriptions.add(\n      combineLatest([assistantTalking$, userTalking$])\n        .pipe(takeUntil(untilCallEnds$))\n        .subscribe(([bot, user]) => {\n          const current = this.callStateSubject.value;\n          if (user) {\n            this.isUserSpeakingSubject.next(true);\n            this.callStateSubject.next('listening');\n          } else {\n            this.isUserSpeakingSubject.next(false);\n          }\n          if (user) {\n            return;\n          }\n          if (bot) {\n            if (this.callStartTime === 0) {\n              this.callStartTime = Date.now();\n              this.startDurationTimer();\n            }\n            this.callStateSubject.next('talking');\n          } else if (current === 'talking' || current === 'listening') {\n            this.callStateSubject.next('connected');\n          }\n        }),\n    );\n  }\n\n  private async startLocalMic(): Promise<void> {\n    this.stopLocalMic();\n    const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\n    const track = stream.getAudioTracks()[0];\n    if (!track) {\n      stream.getTracks().forEach((t) => t.stop());\n      throw new Error('No audio track');\n    }\n    this.localMicStream = stream;\n    this.isMicMutedSubject.next(!track.enabled);\n    this.audioAnalyzer.start(stream);\n  }\n\n  private stopLocalMic(): void {\n    if (this.localMicStream) {\n      this.localMicStream.getTracks().forEach((t) => t.stop());\n      this.localMicStream = null;\n    }\n  }\n\n  private async handleRemoteClose(): Promise<void> {\n    const state = this.callStateSubject.value;\n    if (state === 'idle' || state === 'ended') return;\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Connection lost');\n  }\n\n  async disconnect(): Promise<void> {\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.wsClient.disconnect();\n\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  toggleMic(): void {\n    const nextMuted = !this.isMicMutedSubject.value;\n    const track = this.localMicStream?.getAudioTracks()[0];\n    if (track) {\n      track.enabled = !nextMuted;\n    }\n    this.isMicMutedSubject.next(nextMuted);\n  }\n\n  private startDurationTimer(): void {\n    const updateDuration = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const minutes = Math.floor(elapsed / 60);\n        const seconds = elapsed % 60;\n        this.durationSubject.next(\n          `${minutes}:${String(seconds).padStart(2, '0')}`,\n        );\n      }\n    };\n    updateDuration();\n    this.durationInterval = setInterval(updateDuration, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
366
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  concat,\n  merge,\n  Observable,\n  of,\n  Subject,\n  Subscription,\n  timer,\n} from 'rxjs';\nimport {\n  distinctUntilChanged,\n  map,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n} from 'rxjs/operators';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\nimport { WebSocketVoiceClientService } from './websocket-voice-client.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: single WebSocket (`ws_url` from POST /ai/ask-voice-socket)\n * for session events, transcripts, and optional speaking hints; local mic for capture\n * and waveform only (no Daily/WebRTC room).\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 localMicStream: MediaStream | null = null;\n  private remoteAudioContext: AudioContext | null = null;\n  private pendingRemoteAudio: ArrayBuffer[] = [];\n  private remoteAudioPlaying = false;\n  private readonly endCall$ = new Subject<void>();\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> =\n    this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private wsClient: WebSocketVoiceClientService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\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    this.subscriptions.add(\n      this.wsClient.remoteClose$\n        .pipe(takeUntil(this.destroy$))\n        .subscribe(() => void this.handleRemoteClose()),\n    );\n    this.subscriptions.add(\n      this.wsClient.audioChunk$\n        .pipe(takeUntil(this.destroy$))\n        .subscribe((chunk) => this.enqueueRemoteAudio(chunk)),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    this.disconnect();\n  }\n\n  /** Reset to idle state (e.g. when modal opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.resetRemoteAudioPlayback();\n    this.wsClient.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('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) {\n            accessToken = ensured.accessToken;\n          }\n        } catch (e) {\n          console.warn(\n            '[HiveGpt Voice] Token refresh before connect failed',\n            e,\n          );\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n      const postUrl = `${baseUrl}/ai/ask-voice-socket`;\n\n      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n        Authorization: `Bearer ${accessToken}`,\n        'x-api-key': apiKey,\n        'hive-bot-id': botId,\n        'domain-authority': domainAuthority,\n        eventUrl,\n        eventId,\n        eventToken,\n        'ngrok-skip-browser-warning': 'true',\n      };\n\n      const res = await fetch(postUrl, {\n        method: 'POST',\n        headers,\n        body: JSON.stringify({\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        }),\n      });\n\n      if (!res.ok) {\n        throw new Error(`HTTP ${res.status}`);\n      }\n\n      const json = await res.json();\n      const wsUrl =\n        (typeof json?.ws_url === 'string' && json.ws_url) ||\n        (typeof json?.rn_ws_url === 'string' && json.rn_ws_url);\n      if (!wsUrl) {\n        throw new Error('No ws_url in response');\n      }\n\n      const untilCallEnds$ = merge(this.destroy$, this.endCall$);\n\n      this.subscriptions.add(\n        this.wsClient.userTranscript$\n          .pipe(takeUntil(untilCallEnds$))\n          .subscribe((t) => this.userTranscriptSubject.next(t)),\n      );\n      this.subscriptions.add(\n        this.wsClient.botTranscript$\n          .pipe(takeUntil(untilCallEnds$))\n          .subscribe((t) => this.botTranscriptSubject.next(t)),\n      );\n\n      this.subscriptions.add(\n        this.wsClient.opened$\n          .pipe(takeUntil(untilCallEnds$), take(1))\n          .subscribe(() => void this.onWebsocketOpened()),\n      );\n\n      this.wsClient.connect(wsUrl);\n    } catch (error) {\n      console.error('Error connecting voice agent:', error);\n      this.callStateSubject.next('ended');\n      await this.disconnect();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private async onWebsocketOpened(): Promise<void> {\n    if (this.callStateSubject.value !== 'connecting') {\n      return;\n    }\n    try {\n      await this.startLocalMic();\n      this.statusTextSubject.next('Connected');\n      this.callStateSubject.next('connected');\n      this.wireSpeakingState();\n    } catch (err) {\n      console.error('[HiveGpt Voice] Mic or session setup failed', err);\n      this.callStateSubject.next('ended');\n      this.statusTextSubject.next('Microphone unavailable');\n      await this.disconnect();\n    }\n  }\n\n  private wireSpeakingState(): void {\n    const untilCallEnds$ = merge(this.destroy$, this.endCall$);\n\n    const transcriptDrivenAssistant$ = this.wsClient.botTranscript$.pipe(\n      switchMap(() => concat(of(true), timer(800).pipe(map(() => false)))),\n      distinctUntilChanged(),\n    );\n\n    const assistantTalking$ = merge(\n      this.wsClient.assistantSpeaking$,\n      transcriptDrivenAssistant$,\n    ).pipe(distinctUntilChanged(), startWith(false));\n\n    const userTalking$ = combineLatest([\n      this.audioAnalyzer.isUserSpeaking$,\n      this.wsClient.serverUserSpeaking$.pipe(startWith(false)),\n    ]).pipe(\n      map(([local, server]) => local || server),\n      distinctUntilChanged(),\n      startWith(false),\n    );\n\n    this.subscriptions.add(\n      combineLatest([assistantTalking$, userTalking$])\n        .pipe(takeUntil(untilCallEnds$))\n        .subscribe(([bot, user]) => {\n          const current = this.callStateSubject.value;\n          if (user) {\n            this.isUserSpeakingSubject.next(true);\n            this.callStateSubject.next('listening');\n          } else {\n            this.isUserSpeakingSubject.next(false);\n          }\n          if (user) {\n            return;\n          }\n          if (bot) {\n            if (this.callStartTime === 0) {\n              this.callStartTime = Date.now();\n              this.startDurationTimer();\n            }\n            this.callStateSubject.next('talking');\n          } else if (current === 'talking' || current === 'listening') {\n            this.callStateSubject.next('connected');\n          }\n        }),\n    );\n  }\n\n  private async startLocalMic(): Promise<void> {\n    this.stopLocalMic();\n    const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\n    const track = stream.getAudioTracks()[0];\n    if (!track) {\n      stream.getTracks().forEach((t) => t.stop());\n      throw new Error('No audio track');\n    }\n    this.localMicStream = stream;\n    this.isMicMutedSubject.next(!track.enabled);\n    this.audioAnalyzer.start(stream);\n  }\n\n  private stopLocalMic(): void {\n    if (this.localMicStream) {\n      this.localMicStream.getTracks().forEach((t) => t.stop());\n      this.localMicStream = null;\n    }\n  }\n\n  private enqueueRemoteAudio(chunk: ArrayBuffer): void {\n    this.pendingRemoteAudio.push(chunk.slice(0));\n    if (!this.remoteAudioPlaying) {\n      void this.playRemoteAudioQueue();\n    }\n  }\n\n  private async playRemoteAudioQueue(): Promise<void> {\n    this.remoteAudioPlaying = true;\n    const context = this.getOrCreateRemoteAudioContext();\n    while (this.pendingRemoteAudio.length > 0) {\n      const chunk = this.pendingRemoteAudio.shift();\n      if (!chunk) continue;\n      try {\n        const decoded = await this.decodeAudioChunk(context, chunk);\n        this.assistantAudioStarted();\n        await this.playDecodedBuffer(context, decoded);\n      } catch {\n        // Ignore undecodable chunks; server may mix non-audio binary events.\n      }\n    }\n    this.remoteAudioPlaying = false;\n    this.assistantAudioStopped();\n  }\n\n  private getOrCreateRemoteAudioContext(): AudioContext {\n    if (!this.remoteAudioContext || this.remoteAudioContext.state === 'closed') {\n      this.remoteAudioContext = new AudioContext();\n    }\n    if (this.remoteAudioContext.state === 'suspended') {\n      void this.remoteAudioContext.resume();\n    }\n    return this.remoteAudioContext;\n  }\n\n  private decodeAudioChunk(\n    context: AudioContext,\n    chunk: ArrayBuffer,\n  ): Promise<AudioBuffer> {\n    return new Promise((resolve, reject) => {\n      context.decodeAudioData(chunk.slice(0), resolve, reject);\n    });\n  }\n\n  private playDecodedBuffer(\n    context: AudioContext,\n    buffer: AudioBuffer,\n  ): Promise<void> {\n    return new Promise((resolve) => {\n      const source = context.createBufferSource();\n      source.buffer = buffer;\n      source.connect(context.destination);\n      source.onended = () => resolve();\n      source.start();\n    });\n  }\n\n  private assistantAudioStarted(): void {\n    if (this.callStartTime === 0) {\n      this.callStartTime = Date.now();\n      this.startDurationTimer();\n    }\n    this.callStateSubject.next('talking');\n  }\n\n  private assistantAudioStopped(): void {\n    if (this.callStateSubject.value === 'talking') {\n      this.callStateSubject.next('connected');\n    }\n  }\n\n  private resetRemoteAudioPlayback(): void {\n    this.pendingRemoteAudio = [];\n    this.remoteAudioPlaying = false;\n    if (this.remoteAudioContext && this.remoteAudioContext.state !== 'closed') {\n      this.remoteAudioContext.close().catch(() => {});\n    }\n    this.remoteAudioContext = null;\n  }\n\n  private async handleRemoteClose(): Promise<void> {\n    const state = this.callStateSubject.value;\n    if (state === 'idle' || state === 'ended') return;\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.resetRemoteAudioPlayback();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Connection lost');\n  }\n\n  async disconnect(): Promise<void> {\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.resetRemoteAudioPlayback();\n    this.wsClient.disconnect();\n\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  toggleMic(): void {\n    const nextMuted = !this.isMicMutedSubject.value;\n    const track = this.localMicStream?.getAudioTracks()[0];\n    if (track) {\n      track.enabled = !nextMuted;\n    }\n    this.isMicMutedSubject.next(nextMuted);\n  }\n\n  private startDurationTimer(): void {\n    const updateDuration = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const minutes = Math.floor(elapsed / 60);\n        const seconds = elapsed % 60;\n        this.durationSubject.next(\n          `${minutes}:${String(seconds).padStart(2, '0')}`,\n        );\n      }\n    };\n    updateDuration();\n    this.durationInterval = setInterval(updateDuration, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
@@ -1,3 +1,4 @@
1
+ import { __awaiter } from "tslib";
1
2
  import { Injectable, NgZone } from '@angular/core';
2
3
  import { Subject } from 'rxjs';
3
4
  import * as i0 from "@angular/core";
@@ -20,6 +21,7 @@ export class WebSocketVoiceClientService {
20
21
  this.botTranscriptSubject = new Subject();
21
22
  this.assistantSpeakingSubject = new Subject();
22
23
  this.serverUserSpeakingSubject = new Subject();
24
+ this.audioChunkSubject = new Subject();
23
25
  /** Fires once each time the WebSocket reaches OPEN. */
24
26
  this.opened$ = this.openedSubject.asObservable();
25
27
  /** Fires when the socket closes without a client-initiated {@link disconnect}. */
@@ -30,6 +32,8 @@ export class WebSocketVoiceClientService {
30
32
  this.assistantSpeaking$ = this.assistantSpeakingSubject.asObservable();
31
33
  /** User speaking from server-side VAD, if provided. */
32
34
  this.serverUserSpeaking$ = this.serverUserSpeakingSubject.asObservable();
35
+ /** Binary audio frames from server (when backend streams bot audio over WS). */
36
+ this.audioChunk$ = this.audioChunkSubject.asObservable();
33
37
  }
34
38
  connect(wsUrl) {
35
39
  var _a;
@@ -51,16 +55,7 @@ export class WebSocketVoiceClientService {
51
55
  socket.onmessage = (event) => {
52
56
  if (this.ws !== socket)
53
57
  return;
54
- if (typeof event.data !== 'string') {
55
- return;
56
- }
57
- try {
58
- const msg = JSON.parse(event.data);
59
- this.ngZone.run(() => this.handleJsonMessage(msg));
60
- }
61
- catch (_a) {
62
- // Ignore non-JSON
63
- }
58
+ void this.handleIncomingMessage(event.data);
64
59
  };
65
60
  socket.onerror = () => {
66
61
  this.ngZone.run(() => {
@@ -86,6 +81,54 @@ export class WebSocketVoiceClientService {
86
81
  throw err;
87
82
  }
88
83
  }
84
+ handleIncomingMessage(payload) {
85
+ return __awaiter(this, void 0, void 0, function* () {
86
+ if (typeof payload === 'string') {
87
+ this.handleJsonString(payload);
88
+ return;
89
+ }
90
+ if (payload instanceof ArrayBuffer) {
91
+ this.handleBinaryMessage(payload);
92
+ return;
93
+ }
94
+ if (payload instanceof Blob) {
95
+ const ab = yield payload.arrayBuffer();
96
+ this.handleBinaryMessage(ab);
97
+ }
98
+ });
99
+ }
100
+ handleJsonString(jsonText) {
101
+ try {
102
+ const msg = JSON.parse(jsonText);
103
+ this.ngZone.run(() => this.handleJsonMessage(msg));
104
+ }
105
+ catch (_a) {
106
+ // Ignore non-JSON
107
+ }
108
+ }
109
+ handleBinaryMessage(buffer) {
110
+ // Some backends wrap JSON events inside binary WS frames.
111
+ const maybeText = this.tryDecodeUtf8(buffer);
112
+ if (maybeText !== null) {
113
+ this.handleJsonString(maybeText);
114
+ return;
115
+ }
116
+ // Otherwise treat binary as streamed assistant audio.
117
+ this.ngZone.run(() => this.audioChunkSubject.next(buffer));
118
+ }
119
+ tryDecodeUtf8(buffer) {
120
+ try {
121
+ const text = new TextDecoder('utf-8', { fatal: true }).decode(buffer);
122
+ const trimmed = text.trim();
123
+ if (!trimmed || (trimmed[0] !== '{' && trimmed[0] !== '[')) {
124
+ return null;
125
+ }
126
+ return trimmed;
127
+ }
128
+ catch (_a) {
129
+ return null;
130
+ }
131
+ }
89
132
  handleJsonMessage(msg) {
90
133
  const type = msg.type;
91
134
  const typeStr = typeof type === 'string' ? type : '';
@@ -160,4 +203,4 @@ WebSocketVoiceClientService.decorators = [
160
203
  WebSocketVoiceClientService.ctorParameters = () => [
161
204
  { type: NgZone }
162
205
  ];
163
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/websocket-voice-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\n\n/** WebSocket message types from backend (voice session over a single WS). */\nexport interface WsMessageUserTranscript {\n  type: 'user_transcript';\n  text: string;\n  final?: boolean;\n}\n\nexport interface WsMessageBotTranscript {\n  type: 'bot_transcript';\n  text: string;\n}\n\nexport type WsMessage =\n  | WsMessageUserTranscript\n  | WsMessageBotTranscript;\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Native WebSocket client for voice session (signaling, transcripts, speaking hints).\n * CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.\n *\n * Connects to `ws_url` from `POST {baseUrl}/ai/ask-voice-socket`.\n * Parses JSON messages for transcripts and optional assistant/user speaking flags.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class WebSocketVoiceClientService {\n  private ws: WebSocket | null = null;\n  /** True when {@link disconnect} initiated the close (not counted as remote close). */\n  private closeInitiatedByClient = false;\n\n  private openedSubject = new Subject<void>();\n  private remoteCloseSubject = new Subject<void>();\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n  private assistantSpeakingSubject = new Subject<boolean>();\n  private serverUserSpeakingSubject = new Subject<boolean>();\n\n  /** Fires once each time the WebSocket reaches OPEN. */\n  opened$: Observable<void> = this.openedSubject.asObservable();\n\n  /** Fires when the socket closes without a client-initiated {@link disconnect}. */\n  remoteClose$: Observable<void> = this.remoteCloseSubject.asObservable();\n\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n\n  botTranscript$: Observable<string> =\n    this.botTranscriptSubject.asObservable();\n\n  /** Assistant/bot speaking, when the server sends explicit events (see {@link handleJsonMessage}). */\n  assistantSpeaking$: Observable<boolean> =\n    this.assistantSpeakingSubject.asObservable();\n\n  /** User speaking from server-side VAD, if provided. */\n  serverUserSpeaking$: Observable<boolean> =\n    this.serverUserSpeakingSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  connect(wsUrl: string): void {\n    if (this.ws?.readyState === WebSocket.OPEN) {\n      return;\n    }\n    if (this.ws) {\n      this.closeInitiatedByClient = true;\n      this.ws.close();\n    }\n\n    try {\n      const socket = new WebSocket(wsUrl);\n      this.ws = socket;\n      socket.onopen = () => {\n        if (this.ws !== socket) return;\n        this.ngZone.run(() => this.openedSubject.next());\n      };\n      socket.onmessage = (event: MessageEvent) => {\n        if (this.ws !== socket) return;\n        if (typeof event.data !== 'string') {\n          return;\n        }\n        try {\n          const msg = JSON.parse(event.data) as Record<string, unknown>;\n          this.ngZone.run(() => this.handleJsonMessage(msg));\n        } catch {\n          // Ignore non-JSON\n        }\n      };\n      socket.onerror = () => {\n        this.ngZone.run(() => {\n          if (this.ws === socket && socket.readyState !== WebSocket.CLOSED) {\n            socket.close();\n          }\n        });\n      };\n      socket.onclose = () => {\n        if (this.ws === socket) {\n          this.ws = null;\n        }\n        const client = this.closeInitiatedByClient;\n        this.closeInitiatedByClient = false;\n        if (!client) {\n          this.ngZone.run(() => this.remoteCloseSubject.next());\n        }\n      };\n    } catch (err) {\n      console.error('WebSocketVoiceClient: connect failed', err);\n      this.ws = null;\n      throw err;\n    }\n  }\n\n  private handleJsonMessage(msg: Record<string, unknown>): void {\n    const type = msg.type;\n    const typeStr = typeof type === 'string' ? type : '';\n\n    if (typeStr === 'session_ready' || typeStr === 'connected' || typeStr === 'voice_session_started') {\n      return;\n    }\n\n    if (\n      typeStr === 'assistant_speaking' ||\n      typeStr === 'bot_speaking'\n    ) {\n      if (msg.active === true || msg.speaking === true) {\n        this.assistantSpeakingSubject.next(true);\n      } else if (msg.active === false || msg.speaking === false) {\n        this.assistantSpeakingSubject.next(false);\n      }\n      return;\n    }\n\n    if (typeStr === 'user_speaking') {\n      if (msg.active === true || msg.speaking === true) {\n        this.serverUserSpeakingSubject.next(true);\n      } else if (msg.active === false || msg.speaking === false) {\n        this.serverUserSpeakingSubject.next(false);\n      }\n      return;\n    }\n\n    if (typeStr === 'input_audio_buffer.speech_started') {\n      this.serverUserSpeakingSubject.next(true);\n      return;\n    }\n    if (typeStr === 'input_audio_buffer.speech_stopped') {\n      this.serverUserSpeakingSubject.next(false);\n      return;\n    }\n\n    if (typeStr === 'response.audio.delta') {\n      this.assistantSpeakingSubject.next(true);\n      return;\n    }\n    if (\n      typeStr === 'response.audio.done' ||\n      typeStr === 'response.output_audio.done'\n    ) {\n      this.assistantSpeakingSubject.next(false);\n      return;\n    }\n\n    if (typeStr === 'user_transcript' && typeof msg.text === 'string') {\n      this.userTranscriptSubject.next({\n        text: msg.text,\n        final: msg.final === true,\n      });\n      return;\n    }\n    if (typeStr === 'bot_transcript' && typeof msg.text === 'string') {\n      this.botTranscriptSubject.next(msg.text);\n    }\n  }\n\n  disconnect(): void {\n    if (!this.ws) {\n      return;\n    }\n    this.closeInitiatedByClient = true;\n    this.ws.close();\n  }\n\n  get isConnected(): boolean {\n    return this.ws?.readyState === WebSocket.OPEN;\n  }\n}\n"]}
206
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/websocket-voice-client.service.ts"],"names":[],"mappings":";AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAc,OAAO,EAAE,MAAM,MAAM,CAAC;;AAuB3C;;;;;;GAMG;AAIH,MAAM,OAAO,2BAA2B;IAoCtC,YAAoB,MAAc;QAAd,WAAM,GAAN,MAAM,CAAQ;QAnC1B,OAAE,GAAqB,IAAI,CAAC;QACpC,sFAAsF;QAC9E,2BAAsB,GAAG,KAAK,CAAC;QAE/B,kBAAa,GAAG,IAAI,OAAO,EAAQ,CAAC;QACpC,uBAAkB,GAAG,IAAI,OAAO,EAAQ,CAAC;QACzC,0BAAqB,GAAG,IAAI,OAAO,EAAkB,CAAC;QACtD,yBAAoB,GAAG,IAAI,OAAO,EAAU,CAAC;QAC7C,6BAAwB,GAAG,IAAI,OAAO,EAAW,CAAC;QAClD,8BAAyB,GAAG,IAAI,OAAO,EAAW,CAAC;QACnD,sBAAiB,GAAG,IAAI,OAAO,EAAe,CAAC;QAEvD,uDAAuD;QACvD,YAAO,GAAqB,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;QAE9D,kFAAkF;QAClF,iBAAY,GAAqB,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAExE,oBAAe,GACb,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QAE5C,mBAAc,GACZ,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,CAAC;QAE3C,qGAAqG;QACrG,uBAAkB,GAChB,IAAI,CAAC,wBAAwB,CAAC,YAAY,EAAE,CAAC;QAE/C,uDAAuD;QACvD,wBAAmB,GACjB,IAAI,CAAC,yBAAyB,CAAC,YAAY,EAAE,CAAC;QAEhD,gFAAgF;QAChF,gBAAW,GAA4B,IAAI,CAAC,iBAAiB,CAAC,YAAY,EAAE,CAAC;IAExC,CAAC;IAEtC,OAAO,CAAC,KAAa;;QACnB,IAAI,CAAA,MAAA,IAAI,CAAC,EAAE,0CAAE,UAAU,MAAK,SAAS,CAAC,IAAI,EAAE;YAC1C,OAAO;SACR;QACD,IAAI,IAAI,CAAC,EAAE,EAAE;YACX,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;YACnC,IAAI,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;SACjB;QAED,IAAI;YACF,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;YACpC,IAAI,CAAC,EAAE,GAAG,MAAM,CAAC;YACjB,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE;gBACnB,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM;oBAAE,OAAO;gBAC/B,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC,CAAC;YACnD,CAAC,CAAC;YACF,MAAM,CAAC,SAAS,GAAG,CAAC,KAAmB,EAAE,EAAE;gBACzC,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM;oBAAE,OAAO;gBAC/B,KAAK,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YAC9C,CAAC,CAAC;YACF,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE;gBACpB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;oBACnB,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM,IAAI,MAAM,CAAC,UAAU,KAAK,SAAS,CAAC,MAAM,EAAE;wBAChE,MAAM,CAAC,KAAK,EAAE,CAAC;qBAChB;gBACH,CAAC,CAAC,CAAC;YACL,CAAC,CAAC;YACF,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE;gBACpB,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM,EAAE;oBACtB,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;iBAChB;gBACD,MAAM,MAAM,GAAG,IAAI,CAAC,sBAAsB,CAAC;gBAC3C,IAAI,CAAC,sBAAsB,GAAG,KAAK,CAAC;gBACpC,IAAI,CAAC,MAAM,EAAE;oBACX,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,CAAC,CAAC;iBACvD;YACH,CAAC,CAAC;SACH;QAAC,OAAO,GAAG,EAAE;YACZ,OAAO,CAAC,KAAK,CAAC,sCAAsC,EAAE,GAAG,CAAC,CAAC;YAC3D,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;YACf,MAAM,GAAG,CAAC;SACX;IACH,CAAC;IAEa,qBAAqB,CAAC,OAAgB;;YAClD,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE;gBAC/B,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC;gBAC/B,OAAO;aACR;YACD,IAAI,OAAO,YAAY,WAAW,EAAE;gBAClC,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;gBAClC,OAAO;aACR;YACD,IAAI,OAAO,YAAY,IAAI,EAAE;gBAC3B,MAAM,EAAE,GAAG,MAAM,OAAO,CAAC,WAAW,EAAE,CAAC;gBACvC,IAAI,CAAC,mBAAmB,CAAC,EAAE,CAAC,CAAC;aAC9B;QACH,CAAC;KAAA;IAEO,gBAAgB,CAAC,QAAgB;QACvC,IAAI;YACF,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,QAAQ,CAA4B,CAAC;YAC5D,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC;SACpD;QAAC,WAAM;YACN,kBAAkB;SACnB;IACH,CAAC;IAEO,mBAAmB,CAAC,MAAmB;QAC7C,0DAA0D;QAC1D,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QAC7C,IAAI,SAAS,KAAK,IAAI,EAAE;YACtB,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAAC;YACjC,OAAO;SACR;QACD,sDAAsD;QACtD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;IAC7D,CAAC;IAEO,aAAa,CAAC,MAAmB;QACvC,IAAI;YACF,MAAM,IAAI,GAAG,IAAI,WAAW,CAAC,OAAO,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACtE,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;YAC5B,IAAI,CAAC,OAAO,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,GAAG,IAAI,OAAO,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,EAAE;gBAC1D,OAAO,IAAI,CAAC;aACb;YACD,OAAO,OAAO,CAAC;SAChB;QAAC,WAAM;YACN,OAAO,IAAI,CAAC;SACb;IACH,CAAC;IAEO,iBAAiB,CAAC,GAA4B;QACpD,MAAM,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC;QACtB,MAAM,OAAO,GAAG,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC;QAErD,IAAI,OAAO,KAAK,eAAe,IAAI,OAAO,KAAK,WAAW,IAAI,OAAO,KAAK,uBAAuB,EAAE;YACjG,OAAO;SACR;QAED,IACE,OAAO,KAAK,oBAAoB;YAChC,OAAO,KAAK,cAAc,EAC1B;YACA,IAAI,GAAG,CAAC,MAAM,KAAK,IAAI,IAAI,GAAG,CAAC,QAAQ,KAAK,IAAI,EAAE;gBAChD,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aAC1C;iBAAM,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,GAAG,CAAC,QAAQ,KAAK,KAAK,EAAE;gBACzD,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aAC3C;YACD,OAAO;SACR;QAED,IAAI,OAAO,KAAK,eAAe,EAAE;YAC/B,IAAI,GAAG,CAAC,MAAM,KAAK,IAAI,IAAI,GAAG,CAAC,QAAQ,KAAK,IAAI,EAAE;gBAChD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aAC3C;iBAAM,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,GAAG,CAAC,QAAQ,KAAK,KAAK,EAAE;gBACzD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aAC5C;YACD,OAAO;SACR;QAED,IAAI,OAAO,KAAK,mCAAmC,EAAE;YACnD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC1C,OAAO;SACR;QACD,IAAI,OAAO,KAAK,mCAAmC,EAAE;YACnD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3C,OAAO;SACR;QAED,IAAI,OAAO,KAAK,sBAAsB,EAAE;YACtC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACzC,OAAO;SACR;QACD,IACE,OAAO,KAAK,qBAAqB;YACjC,OAAO,KAAK,4BAA4B,EACxC;YACA,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC1C,OAAO;SACR;QAED,IAAI,OAAO,KAAK,iBAAiB,IAAI,OAAO,GAAG,CAAC,IAAI,KAAK,QAAQ,EAAE;YACjE,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC;gBAC9B,IAAI,EAAE,GAAG,CAAC,IAAI;gBACd,KAAK,EAAE,GAAG,CAAC,KAAK,KAAK,IAAI;aAC1B,CAAC,CAAC;YACH,OAAO;SACR;QACD,IAAI,OAAO,KAAK,gBAAgB,IAAI,OAAO,GAAG,CAAC,IAAI,KAAK,QAAQ,EAAE;YAChE,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;SAC1C;IACH,CAAC;IAED,UAAU;QACR,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE;YACZ,OAAO;SACR;QACD,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;QACnC,IAAI,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;IAClB,CAAC;IAED,IAAI,WAAW;;QACb,OAAO,CAAA,MAAA,IAAI,CAAC,EAAE,0CAAE,UAAU,MAAK,SAAS,CAAC,IAAI,CAAC;IAChD,CAAC;;;;YA7MF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YAjCoB,MAAM","sourcesContent":["import { Injectable, NgZone } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\n\n/** WebSocket message types from backend (voice session over a single WS). */\nexport interface WsMessageUserTranscript {\n  type: 'user_transcript';\n  text: string;\n  final?: boolean;\n}\n\nexport interface WsMessageBotTranscript {\n  type: 'bot_transcript';\n  text: string;\n}\n\nexport type WsMessage =\n  | WsMessageUserTranscript\n  | WsMessageBotTranscript;\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Native WebSocket client for voice session (signaling, transcripts, speaking hints).\n * CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.\n *\n * Connects to `ws_url` from `POST {baseUrl}/ai/ask-voice-socket`.\n * Parses JSON messages for transcripts and optional assistant/user speaking flags.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class WebSocketVoiceClientService {\n  private ws: WebSocket | null = null;\n  /** True when {@link disconnect} initiated the close (not counted as remote close). */\n  private closeInitiatedByClient = false;\n\n  private openedSubject = new Subject<void>();\n  private remoteCloseSubject = new Subject<void>();\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n  private assistantSpeakingSubject = new Subject<boolean>();\n  private serverUserSpeakingSubject = new Subject<boolean>();\n  private audioChunkSubject = new Subject<ArrayBuffer>();\n\n  /** Fires once each time the WebSocket reaches OPEN. */\n  opened$: Observable<void> = this.openedSubject.asObservable();\n\n  /** Fires when the socket closes without a client-initiated {@link disconnect}. */\n  remoteClose$: Observable<void> = this.remoteCloseSubject.asObservable();\n\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n\n  botTranscript$: Observable<string> =\n    this.botTranscriptSubject.asObservable();\n\n  /** Assistant/bot speaking, when the server sends explicit events (see {@link handleJsonMessage}). */\n  assistantSpeaking$: Observable<boolean> =\n    this.assistantSpeakingSubject.asObservable();\n\n  /** User speaking from server-side VAD, if provided. */\n  serverUserSpeaking$: Observable<boolean> =\n    this.serverUserSpeakingSubject.asObservable();\n\n  /** Binary audio frames from server (when backend streams bot audio over WS). */\n  audioChunk$: Observable<ArrayBuffer> = this.audioChunkSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  connect(wsUrl: string): void {\n    if (this.ws?.readyState === WebSocket.OPEN) {\n      return;\n    }\n    if (this.ws) {\n      this.closeInitiatedByClient = true;\n      this.ws.close();\n    }\n\n    try {\n      const socket = new WebSocket(wsUrl);\n      this.ws = socket;\n      socket.onopen = () => {\n        if (this.ws !== socket) return;\n        this.ngZone.run(() => this.openedSubject.next());\n      };\n      socket.onmessage = (event: MessageEvent) => {\n        if (this.ws !== socket) return;\n        void this.handleIncomingMessage(event.data);\n      };\n      socket.onerror = () => {\n        this.ngZone.run(() => {\n          if (this.ws === socket && socket.readyState !== WebSocket.CLOSED) {\n            socket.close();\n          }\n        });\n      };\n      socket.onclose = () => {\n        if (this.ws === socket) {\n          this.ws = null;\n        }\n        const client = this.closeInitiatedByClient;\n        this.closeInitiatedByClient = false;\n        if (!client) {\n          this.ngZone.run(() => this.remoteCloseSubject.next());\n        }\n      };\n    } catch (err) {\n      console.error('WebSocketVoiceClient: connect failed', err);\n      this.ws = null;\n      throw err;\n    }\n  }\n\n  private async handleIncomingMessage(payload: unknown): Promise<void> {\n    if (typeof payload === 'string') {\n      this.handleJsonString(payload);\n      return;\n    }\n    if (payload instanceof ArrayBuffer) {\n      this.handleBinaryMessage(payload);\n      return;\n    }\n    if (payload instanceof Blob) {\n      const ab = await payload.arrayBuffer();\n      this.handleBinaryMessage(ab);\n    }\n  }\n\n  private handleJsonString(jsonText: string): void {\n    try {\n      const msg = JSON.parse(jsonText) as Record<string, unknown>;\n      this.ngZone.run(() => this.handleJsonMessage(msg));\n    } catch {\n      // Ignore non-JSON\n    }\n  }\n\n  private handleBinaryMessage(buffer: ArrayBuffer): void {\n    // Some backends wrap JSON events inside binary WS frames.\n    const maybeText = this.tryDecodeUtf8(buffer);\n    if (maybeText !== null) {\n      this.handleJsonString(maybeText);\n      return;\n    }\n    // Otherwise treat binary as streamed assistant audio.\n    this.ngZone.run(() => this.audioChunkSubject.next(buffer));\n  }\n\n  private tryDecodeUtf8(buffer: ArrayBuffer): string | null {\n    try {\n      const text = new TextDecoder('utf-8', { fatal: true }).decode(buffer);\n      const trimmed = text.trim();\n      if (!trimmed || (trimmed[0] !== '{' && trimmed[0] !== '[')) {\n        return null;\n      }\n      return trimmed;\n    } catch {\n      return null;\n    }\n  }\n\n  private handleJsonMessage(msg: Record<string, unknown>): void {\n    const type = msg.type;\n    const typeStr = typeof type === 'string' ? type : '';\n\n    if (typeStr === 'session_ready' || typeStr === 'connected' || typeStr === 'voice_session_started') {\n      return;\n    }\n\n    if (\n      typeStr === 'assistant_speaking' ||\n      typeStr === 'bot_speaking'\n    ) {\n      if (msg.active === true || msg.speaking === true) {\n        this.assistantSpeakingSubject.next(true);\n      } else if (msg.active === false || msg.speaking === false) {\n        this.assistantSpeakingSubject.next(false);\n      }\n      return;\n    }\n\n    if (typeStr === 'user_speaking') {\n      if (msg.active === true || msg.speaking === true) {\n        this.serverUserSpeakingSubject.next(true);\n      } else if (msg.active === false || msg.speaking === false) {\n        this.serverUserSpeakingSubject.next(false);\n      }\n      return;\n    }\n\n    if (typeStr === 'input_audio_buffer.speech_started') {\n      this.serverUserSpeakingSubject.next(true);\n      return;\n    }\n    if (typeStr === 'input_audio_buffer.speech_stopped') {\n      this.serverUserSpeakingSubject.next(false);\n      return;\n    }\n\n    if (typeStr === 'response.audio.delta') {\n      this.assistantSpeakingSubject.next(true);\n      return;\n    }\n    if (\n      typeStr === 'response.audio.done' ||\n      typeStr === 'response.output_audio.done'\n    ) {\n      this.assistantSpeakingSubject.next(false);\n      return;\n    }\n\n    if (typeStr === 'user_transcript' && typeof msg.text === 'string') {\n      this.userTranscriptSubject.next({\n        text: msg.text,\n        final: msg.final === true,\n      });\n      return;\n    }\n    if (typeStr === 'bot_transcript' && typeof msg.text === 'string') {\n      this.botTranscriptSubject.next(msg.text);\n    }\n  }\n\n  disconnect(): void {\n    if (!this.ws) {\n      return;\n    }\n    this.closeInitiatedByClient = true;\n    this.ws.close();\n  }\n\n  get isConnected(): boolean {\n    return this.ws?.readyState === WebSocket.OPEN;\n  }\n}\n"]}
@@ -823,6 +823,7 @@ class WebSocketVoiceClientService {
823
823
  this.botTranscriptSubject = new Subject();
824
824
  this.assistantSpeakingSubject = new Subject();
825
825
  this.serverUserSpeakingSubject = new Subject();
826
+ this.audioChunkSubject = new Subject();
826
827
  /** Fires once each time the WebSocket reaches OPEN. */
827
828
  this.opened$ = this.openedSubject.asObservable();
828
829
  /** Fires when the socket closes without a client-initiated {@link disconnect}. */
@@ -833,6 +834,8 @@ class WebSocketVoiceClientService {
833
834
  this.assistantSpeaking$ = this.assistantSpeakingSubject.asObservable();
834
835
  /** User speaking from server-side VAD, if provided. */
835
836
  this.serverUserSpeaking$ = this.serverUserSpeakingSubject.asObservable();
837
+ /** Binary audio frames from server (when backend streams bot audio over WS). */
838
+ this.audioChunk$ = this.audioChunkSubject.asObservable();
836
839
  }
837
840
  connect(wsUrl) {
838
841
  var _a;
@@ -854,16 +857,7 @@ class WebSocketVoiceClientService {
854
857
  socket.onmessage = (event) => {
855
858
  if (this.ws !== socket)
856
859
  return;
857
- if (typeof event.data !== 'string') {
858
- return;
859
- }
860
- try {
861
- const msg = JSON.parse(event.data);
862
- this.ngZone.run(() => this.handleJsonMessage(msg));
863
- }
864
- catch (_a) {
865
- // Ignore non-JSON
866
- }
860
+ void this.handleIncomingMessage(event.data);
867
861
  };
868
862
  socket.onerror = () => {
869
863
  this.ngZone.run(() => {
@@ -889,6 +883,54 @@ class WebSocketVoiceClientService {
889
883
  throw err;
890
884
  }
891
885
  }
886
+ handleIncomingMessage(payload) {
887
+ return __awaiter(this, void 0, void 0, function* () {
888
+ if (typeof payload === 'string') {
889
+ this.handleJsonString(payload);
890
+ return;
891
+ }
892
+ if (payload instanceof ArrayBuffer) {
893
+ this.handleBinaryMessage(payload);
894
+ return;
895
+ }
896
+ if (payload instanceof Blob) {
897
+ const ab = yield payload.arrayBuffer();
898
+ this.handleBinaryMessage(ab);
899
+ }
900
+ });
901
+ }
902
+ handleJsonString(jsonText) {
903
+ try {
904
+ const msg = JSON.parse(jsonText);
905
+ this.ngZone.run(() => this.handleJsonMessage(msg));
906
+ }
907
+ catch (_a) {
908
+ // Ignore non-JSON
909
+ }
910
+ }
911
+ handleBinaryMessage(buffer) {
912
+ // Some backends wrap JSON events inside binary WS frames.
913
+ const maybeText = this.tryDecodeUtf8(buffer);
914
+ if (maybeText !== null) {
915
+ this.handleJsonString(maybeText);
916
+ return;
917
+ }
918
+ // Otherwise treat binary as streamed assistant audio.
919
+ this.ngZone.run(() => this.audioChunkSubject.next(buffer));
920
+ }
921
+ tryDecodeUtf8(buffer) {
922
+ try {
923
+ const text = new TextDecoder('utf-8', { fatal: true }).decode(buffer);
924
+ const trimmed = text.trim();
925
+ if (!trimmed || (trimmed[0] !== '{' && trimmed[0] !== '[')) {
926
+ return null;
927
+ }
928
+ return trimmed;
929
+ }
930
+ catch (_a) {
931
+ return null;
932
+ }
933
+ }
892
934
  handleJsonMessage(msg) {
893
935
  const type = msg.type;
894
936
  const typeStr = typeof type === 'string' ? type : '';
@@ -988,6 +1030,9 @@ class VoiceAgentService {
988
1030
  this.callStartTime = 0;
989
1031
  this.durationInterval = null;
990
1032
  this.localMicStream = null;
1033
+ this.remoteAudioContext = null;
1034
+ this.pendingRemoteAudio = [];
1035
+ this.remoteAudioPlaying = false;
991
1036
  this.endCall$ = new Subject();
992
1037
  this.subscriptions = new Subscription();
993
1038
  this.destroy$ = new Subject();
@@ -1003,6 +1048,9 @@ class VoiceAgentService {
1003
1048
  this.subscriptions.add(this.wsClient.remoteClose$
1004
1049
  .pipe(takeUntil(this.destroy$))
1005
1050
  .subscribe(() => void this.handleRemoteClose()));
1051
+ this.subscriptions.add(this.wsClient.audioChunk$
1052
+ .pipe(takeUntil(this.destroy$))
1053
+ .subscribe((chunk) => this.enqueueRemoteAudio(chunk)));
1006
1054
  }
1007
1055
  ngOnDestroy() {
1008
1056
  this.destroy$.next();
@@ -1018,6 +1066,7 @@ class VoiceAgentService {
1018
1066
  this.callStartTime = 0;
1019
1067
  this.audioAnalyzer.stop();
1020
1068
  this.stopLocalMic();
1069
+ this.resetRemoteAudioPlayback();
1021
1070
  this.wsClient.disconnect();
1022
1071
  this.callStateSubject.next('idle');
1023
1072
  this.statusTextSubject.next('');
@@ -1172,6 +1221,76 @@ class VoiceAgentService {
1172
1221
  this.localMicStream = null;
1173
1222
  }
1174
1223
  }
1224
+ enqueueRemoteAudio(chunk) {
1225
+ this.pendingRemoteAudio.push(chunk.slice(0));
1226
+ if (!this.remoteAudioPlaying) {
1227
+ void this.playRemoteAudioQueue();
1228
+ }
1229
+ }
1230
+ playRemoteAudioQueue() {
1231
+ return __awaiter(this, void 0, void 0, function* () {
1232
+ this.remoteAudioPlaying = true;
1233
+ const context = this.getOrCreateRemoteAudioContext();
1234
+ while (this.pendingRemoteAudio.length > 0) {
1235
+ const chunk = this.pendingRemoteAudio.shift();
1236
+ if (!chunk)
1237
+ continue;
1238
+ try {
1239
+ const decoded = yield this.decodeAudioChunk(context, chunk);
1240
+ this.assistantAudioStarted();
1241
+ yield this.playDecodedBuffer(context, decoded);
1242
+ }
1243
+ catch (_a) {
1244
+ // Ignore undecodable chunks; server may mix non-audio binary events.
1245
+ }
1246
+ }
1247
+ this.remoteAudioPlaying = false;
1248
+ this.assistantAudioStopped();
1249
+ });
1250
+ }
1251
+ getOrCreateRemoteAudioContext() {
1252
+ if (!this.remoteAudioContext || this.remoteAudioContext.state === 'closed') {
1253
+ this.remoteAudioContext = new AudioContext();
1254
+ }
1255
+ if (this.remoteAudioContext.state === 'suspended') {
1256
+ void this.remoteAudioContext.resume();
1257
+ }
1258
+ return this.remoteAudioContext;
1259
+ }
1260
+ decodeAudioChunk(context, chunk) {
1261
+ return new Promise((resolve, reject) => {
1262
+ context.decodeAudioData(chunk.slice(0), resolve, reject);
1263
+ });
1264
+ }
1265
+ playDecodedBuffer(context, buffer) {
1266
+ return new Promise((resolve) => {
1267
+ const source = context.createBufferSource();
1268
+ source.buffer = buffer;
1269
+ source.connect(context.destination);
1270
+ source.onended = () => resolve();
1271
+ source.start();
1272
+ });
1273
+ }
1274
+ assistantAudioStarted() {
1275
+ if (this.callStartTime === 0) {
1276
+ this.callStartTime = Date.now();
1277
+ this.startDurationTimer();
1278
+ }
1279
+ this.callStateSubject.next('talking');
1280
+ }
1281
+ assistantAudioStopped() {
1282
+ if (this.callStateSubject.value === 'talking') {
1283
+ this.callStateSubject.next('connected');
1284
+ }
1285
+ }
1286
+ resetRemoteAudioPlayback() {
1287
+ this.pendingRemoteAudio = [];
1288
+ this.remoteAudioPlaying = false;
1289
+ if (this.remoteAudioContext && this.remoteAudioContext.state !== 'closed') {
1290
+ this.remoteAudioContext.close().catch(() => { });
1291
+ }
1292
+ this.remoteAudioContext = null;
1293
+ }
1175
1294
  handleRemoteClose() {
1176
1295
  return __awaiter(this, void 0, void 0, function* () {
1177
1296
  const state = this.callStateSubject.value;
@@ -1182,6 +1301,7 @@ class VoiceAgentService {
1182
1301
  this.callStartTime = 0;
1183
1302
  this.audioAnalyzer.stop();
1184
1303
  this.stopLocalMic();
1304
+ this.resetRemoteAudioPlayback();
1185
1305
  this.callStateSubject.next('ended');
1186
1306
  this.statusTextSubject.next('Connection lost');
1187
1307
  });
@@ -1193,6 +1313,7 @@ class VoiceAgentService {
1193
1313
  this.callStartTime = 0;
1194
1314
  this.audioAnalyzer.stop();
1195
1315
  this.stopLocalMic();
1316
+ this.resetRemoteAudioPlayback();
1196
1317
  this.wsClient.disconnect();
1197
1318
  this.callStateSubject.next('ended');
1198
1319
  this.statusTextSubject.next('Call Ended');