@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.
- package/bundles/hivegpt-hiveai-angular.umd.js +158 -10
- package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +80 -1
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +54 -11
- package/fesm2015/hivegpt-hiveai-angular.js +131 -10
- package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
- package/hivegpt-hiveai-angular.metadata.json +1 -1
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +11 -0
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +7 -0
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -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":"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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 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":";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;IAiC5B,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;QArCzC,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;QAC1C,uBAAkB,GAAwB,IAAI,CAAC;QAC/C,uBAAkB,GAAkB,EAAE,CAAC;QACvC,uBAAkB,GAAG,KAAK,CAAC;QAClB,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;QACF,IAAI,CAAC,aAAa,CAAC,GAAG,CACpB,IAAI,CAAC,QAAQ,CAAC,WAAW;aACtB,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC,CACxD,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,wBAAwB,EAAE,CAAC;QAChC,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;IAEO,kBAAkB,CAAC,KAAkB;QAC3C,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QAC7C,IAAI,CAAC,IAAI,CAAC,kBAAkB,EAAE;YAC5B,KAAK,IAAI,CAAC,oBAAoB,EAAE,CAAC;SAClC;IACH,CAAC;IAEa,oBAAoB;;YAChC,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;YAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,6BAA6B,EAAE,CAAC;YACrD,OAAO,IAAI,CAAC,kBAAkB,CAAC,MAAM,GAAG,CAAC,EAAE;gBACzC,MAAM,KAAK,GAAG,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,CAAC;gBAC9C,IAAI,CAAC,KAAK;oBAAE,SAAS;gBACrB,IAAI;oBACF,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;oBAC5D,IAAI,CAAC,qBAAqB,EAAE,CAAC;oBAC7B,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;iBAChD;gBAAC,WAAM;oBACN,qEAAqE;iBACtE;aACF;YACD,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;YAChC,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC/B,CAAC;KAAA;IAEO,6BAA6B;QACnC,IAAI,CAAC,IAAI,CAAC,kBAAkB,IAAI,IAAI,CAAC,kBAAkB,CAAC,KAAK,KAAK,QAAQ,EAAE;YAC1E,IAAI,CAAC,kBAAkB,GAAG,IAAI,YAAY,EAAE,CAAC;SAC9C;QACD,IAAI,IAAI,CAAC,kBAAkB,CAAC,KAAK,KAAK,WAAW,EAAE;YACjD,KAAK,IAAI,CAAC,kBAAkB,CAAC,MAAM,EAAE,CAAC;SACvC;QACD,OAAO,IAAI,CAAC,kBAAkB,CAAC;IACjC,CAAC;IAEO,gBAAgB,CACtB,OAAqB,EACrB,KAAkB;QAElB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,OAAO,CAAC,eAAe,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC;QAC3D,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,iBAAiB,CACvB,OAAqB,EACrB,MAAmB;QAEnB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE;YAC7B,MAAM,MAAM,GAAG,OAAO,CAAC,kBAAkB,EAAE,CAAC;YAC5C,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC;YACvB,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;YACpC,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC;YACjC,MAAM,CAAC,KAAK,EAAE,CAAC;QACjB,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,qBAAqB;QAC3B,IAAI,IAAI,CAAC,aAAa,KAAK,CAAC,EAAE;YAC5B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YAChC,IAAI,CAAC,kBAAkB,EAAE,CAAC;SAC3B;QACD,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACxC,CAAC;IAEO,qBAAqB;QAC3B,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAK,KAAK,SAAS,EAAE;YAC7C,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;SACzC;IACH,CAAC;IAEO,wBAAwB;QAC9B,IAAI,CAAC,kBAAkB,GAAG,EAAE,CAAC;QAC7B,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;QAChC,IAAI,IAAI,CAAC,kBAAkB,IAAI,IAAI,CAAC,kBAAkB,CAAC,KAAK,KAAK,QAAQ,EAAE;YACzE,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;SACjD;QACD,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;IACjC,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,wBAAwB,EAAE,CAAC;YAChC,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,wBAAwB,EAAE,CAAC;YAChC,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;;;;YAvZF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YAvBQ,oBAAoB;YACpB,2BAA2B;YAF3B,2BAA2B;YA+DS,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 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
|
-
|
|
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":"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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  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
|
-
|
|
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');
|