@hivegpt/hiveai-angular 0.0.581 → 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 +420 -490
- 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/hivegpt-hiveai-angular.js +4 -5
- package/esm2015/lib/components/voice-agent/services/audio-analyzer.service.js +3 -3
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +195 -83
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +160 -49
- package/esm2015/lib/components/voice-agent/voice-agent.module.js +3 -5
- package/fesm2015/hivegpt-hiveai-angular.js +338 -416
- package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
- package/hivegpt-hiveai-angular.d.ts +3 -4
- package/hivegpt-hiveai-angular.d.ts.map +1 -1
- package/hivegpt-hiveai-angular.metadata.json +1 -1
- package/lib/components/voice-agent/services/audio-analyzer.service.d.ts +2 -2
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +22 -13
- 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 +30 -20
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
- package/lib/components/voice-agent/voice-agent.module.d.ts +1 -1
- package/lib/components/voice-agent/voice-agent.module.d.ts.map +1 -1
- package/package.json +1 -1
- package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +0 -305
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +0 -62
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +0 -1
|
@@ -3,13 +3,12 @@
|
|
|
3
3
|
*/
|
|
4
4
|
export * from './public-api';
|
|
5
5
|
export { NotificationSocket as ɵd } from './lib/components/NotificationSocket';
|
|
6
|
-
export { BotHtmlEditorComponent as
|
|
6
|
+
export { BotHtmlEditorComponent as ɵi } from './lib/components/bot-html-editor/bot-html-editor.component';
|
|
7
7
|
export { BotsService as ɵa } from './lib/components/bot.service';
|
|
8
8
|
export { ConversationService as ɵc } from './lib/components/conversation.service';
|
|
9
9
|
export { SocketService as ɵb } from './lib/components/socket-service.service';
|
|
10
10
|
export { TranslationService as ɵe } from './lib/components/translations/translation.service';
|
|
11
|
-
export { VideoPlayerComponent as
|
|
12
|
-
export { DailyVoiceClientService as ɵg } from './lib/components/voice-agent/services/daily-voice-client.service';
|
|
11
|
+
export { VideoPlayerComponent as ɵg } from './lib/components/video-player/video-player.component';
|
|
13
12
|
export { WebSocketVoiceClientService as ɵf } from './lib/components/voice-agent/services/websocket-voice-client.service';
|
|
14
|
-
export { SafeHtmlPipe as
|
|
15
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
13
|
+
export { SafeHtmlPipe as ɵh } from './lib/pipes/safe-html.pipe';
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGl2ZWdwdC1oaXZlYWktYW5ndWxhci5qcyIsInNvdXJjZVJvb3QiOiIvVXNlcnMvcm9oaXR0aGFrdXIvaGl2ZS1ncHQvSGl2ZUFJLVBhY2thZ2VzL0FuZ3VsYXIvcHJvamVjdHMvaGl2ZWdwdC9ldmVudHNncHQtYW5ndWxhci9zcmMvIiwic291cmNlcyI6WyJoaXZlZ3B0LWhpdmVhaS1hbmd1bGFyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUM7QUFFN0IsT0FBTyxFQUFDLGtCQUFrQixJQUFJLEVBQUUsRUFBQyxNQUFNLHFDQUFxQyxDQUFDO0FBQzdFLE9BQU8sRUFBQyxzQkFBc0IsSUFBSSxFQUFFLEVBQUMsTUFBTSw0REFBNEQsQ0FBQztBQUN4RyxPQUFPLEVBQUMsV0FBVyxJQUFJLEVBQUUsRUFBQyxNQUFNLDhCQUE4QixDQUFDO0FBQy9ELE9BQU8sRUFBQyxtQkFBbUIsSUFBSSxFQUFFLEVBQUMsTUFBTSx1Q0FBdUMsQ0FBQztBQUNoRixPQUFPLEVBQUMsYUFBYSxJQUFJLEVBQUUsRUFBQyxNQUFNLHlDQUF5QyxDQUFDO0FBQzVFLE9BQU8sRUFBQyxrQkFBa0IsSUFBSSxFQUFFLEVBQUMsTUFBTSxtREFBbUQsQ0FBQztBQUMzRixPQUFPLEVBQUMsb0JBQW9CLElBQUksRUFBRSxFQUFDLE1BQU0sc0RBQXNELENBQUM7QUFDaEcsT0FBTyxFQUFDLDJCQUEyQixJQUFJLEVBQUUsRUFBQyxNQUFNLHNFQUFzRSxDQUFDO0FBQ3ZILE9BQU8sRUFBQyxZQUFZLElBQUksRUFBRSxFQUFDLE1BQU0sNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vcHVibGljLWFwaSc7XG5cbmV4cG9ydCB7Tm90aWZpY2F0aW9uU29ja2V0IGFzIMm1ZH0gZnJvbSAnLi9saWIvY29tcG9uZW50cy9Ob3RpZmljYXRpb25Tb2NrZXQnO1xuZXhwb3J0IHtCb3RIdG1sRWRpdG9yQ29tcG9uZW50IGFzIMm1aX0gZnJvbSAnLi9saWIvY29tcG9uZW50cy9ib3QtaHRtbC1lZGl0b3IvYm90LWh0bWwtZWRpdG9yLmNvbXBvbmVudCc7XG5leHBvcnQge0JvdHNTZXJ2aWNlIGFzIMm1YX0gZnJvbSAnLi9saWIvY29tcG9uZW50cy9ib3Quc2VydmljZSc7XG5leHBvcnQge0NvbnZlcnNhdGlvblNlcnZpY2UgYXMgybVjfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL2NvbnZlcnNhdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCB7U29ja2V0U2VydmljZSBhcyDJtWJ9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvc29ja2V0LXNlcnZpY2Uuc2VydmljZSc7XG5leHBvcnQge1RyYW5zbGF0aW9uU2VydmljZSBhcyDJtWV9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvdHJhbnNsYXRpb25zL3RyYW5zbGF0aW9uLnNlcnZpY2UnO1xuZXhwb3J0IHtWaWRlb1BsYXllckNvbXBvbmVudCBhcyDJtWd9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvdmlkZW8tcGxheWVyL3ZpZGVvLXBsYXllci5jb21wb25lbnQnO1xuZXhwb3J0IHtXZWJTb2NrZXRWb2ljZUNsaWVudFNlcnZpY2UgYXMgybVmfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3NlcnZpY2VzL3dlYnNvY2tldC12b2ljZS1jbGllbnQuc2VydmljZSc7XG5leHBvcnQge1NhZmVIdG1sUGlwZSBhcyDJtWh9IGZyb20gJy4vbGliL3BpcGVzL3NhZmUtaHRtbC5waXBlJzsiXX0=
|
|
@@ -2,8 +2,8 @@ import { Injectable } from '@angular/core';
|
|
|
2
2
|
import { BehaviorSubject } from 'rxjs';
|
|
3
3
|
import * as i0 from "@angular/core";
|
|
4
4
|
/**
|
|
5
|
-
* Audio analyzer for waveform visualization
|
|
6
|
-
*
|
|
5
|
+
* Audio analyzer for waveform visualization and local (mic) speaking detection.
|
|
6
|
+
* VoiceAgentService may combine this with WebSocket server events for call state.
|
|
7
7
|
*/
|
|
8
8
|
export class AudioAnalyzerService {
|
|
9
9
|
constructor() {
|
|
@@ -122,4 +122,4 @@ AudioAnalyzerService.decorators = [
|
|
|
122
122
|
providedIn: 'root'
|
|
123
123
|
},] }
|
|
124
124
|
];
|
|
125
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio-analyzer.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/audio-analyzer.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\n\n/**\n * Audio analyzer for waveform visualization only.\n * Do NOT use isUserSpeaking$ for call state; speaking state must come from Daily.js.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class AudioAnalyzerService {\n  private audioContext: AudioContext | null = null;\n  private analyserNode: AnalyserNode | null = null;\n  private dataArray: Uint8Array | null = null;\n  private animationFrameId: number | null = null;\n  private isRunning = false;\n\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n\n  // Adaptive noise floor detection\n  private noiseFloor = 0;\n  private noiseFloorSamples: number[] = [];\n  private readonly NOISE_FLOOR_SAMPLE_COUNT = 30;\n  private readonly SPEAKING_THRESHOLD_MULTIPLIER = 2.5;\n  private readonly WAVEFORM_BAR_COUNT = 60;\n  // Amplify raw amplitude so normal speech (±10–20 units) maps to visible levels\n  private readonly SENSITIVITY_MULTIPLIER = 5;\n\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n\n  start(stream: MediaStream): void {\n    if (this.isRunning) {\n      this.stop();\n    }\n\n    try {\n      this.audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();\n      const source = this.audioContext.createMediaStreamSource(stream);\n      \n      this.analyserNode = this.audioContext.createAnalyser();\n      this.analyserNode.fftSize = 2048;\n      this.analyserNode.smoothingTimeConstant = 0.4;\n      \n      const bufferLength = this.analyserNode.frequencyBinCount;\n      this.dataArray = new Uint8Array(bufferLength);\n      \n      source.connect(this.analyserNode);\n      \n      this.isRunning = true;\n      this.noiseFloorSamples = [];\n      this.noiseFloor = 0;\n      \n      this.analyze();\n    } catch (error) {\n      console.error('Error starting audio analyzer:', error);\n      this.stop();\n    }\n  }\n\n  stop(): void {\n    this.isRunning = false;\n    \n    if (this.animationFrameId !== null) {\n      cancelAnimationFrame(this.animationFrameId);\n      this.animationFrameId = null;\n    }\n\n    if (this.analyserNode) {\n      this.analyserNode.disconnect();\n      this.analyserNode = null;\n    }\n\n    if (this.audioContext && this.audioContext.state !== 'closed') {\n      this.audioContext.close().catch(console.error);\n      this.audioContext = null;\n    }\n\n    this.dataArray = null;\n    this.audioLevelsSubject.next([]);\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private analyze(): void {\n    if (!this.isRunning || !this.analyserNode || !this.dataArray) {\n      return;\n    }\n\n    this.analyserNode.getByteTimeDomainData(this.dataArray);\n    \n    // Calculate RMS (Root Mean Square) volume\n    const rms = this.calculateRMS(this.dataArray);\n    \n    // Learn noise floor during initial samples\n    if (this.noiseFloorSamples.length < this.NOISE_FLOOR_SAMPLE_COUNT) {\n      this.noiseFloorSamples.push(rms);\n      if (this.noiseFloorSamples.length === this.NOISE_FLOOR_SAMPLE_COUNT) {\n        // Calculate average noise floor\n        const sum = this.noiseFloorSamples.reduce((a, b) => a + b, 0);\n        this.noiseFloor = sum / this.NOISE_FLOOR_SAMPLE_COUNT;\n      }\n    } else {\n      // Update noise floor adaptively (moving average)\n      this.noiseFloor = this.noiseFloor * 0.99 + rms * 0.01;\n    }\n\n    // Detect if user is speaking\n    const threshold = this.noiseFloor * this.SPEAKING_THRESHOLD_MULTIPLIER;\n    const isSpeaking = rms > threshold;\n    this.isUserSpeakingSubject.next(isSpeaking);\n\n    // Generate waveform bars\n    const bars = this.generateWaveformBars(this.dataArray);\n    this.audioLevelsSubject.next(bars);\n\n    this.animationFrameId = requestAnimationFrame(() => this.analyze());\n  }\n\n  private calculateRMS(data: Uint8Array): number {\n    let sum = 0;\n    for (let i = 0; i < data.length; i++) {\n      const normalized = (data[i] - 128) / 128;\n      sum += normalized * normalized;\n    }\n    return Math.sqrt(sum / data.length);\n  }\n\n  private generateWaveformBars(data: Uint8Array): number[] {\n    const bars: number[] = [];\n    const step = Math.floor(data.length / this.WAVEFORM_BAR_COUNT);\n    \n    for (let i = 0; i < this.WAVEFORM_BAR_COUNT; i++) {\n      const index = i * step;\n      const value = data[index];\n      // Normalize and amplify so quiet speech produces visible movement\n      const normalized = Math.abs((value - 128) / 128) * 100 * this.SENSITIVITY_MULTIPLIER;\n      bars.push(Math.min(100, Math.max(0, normalized)));\n    }\n    \n    return bars;\n  }\n}\n"]}
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audio-analyzer.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/audio-analyzer.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\n\n/**\n * Audio analyzer for waveform visualization and local (mic) speaking detection.\n * VoiceAgentService may combine this with WebSocket server events for call state.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class AudioAnalyzerService {\n  private audioContext: AudioContext | null = null;\n  private analyserNode: AnalyserNode | null = null;\n  private dataArray: Uint8Array | null = null;\n  private animationFrameId: number | null = null;\n  private isRunning = false;\n\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n\n  // Adaptive noise floor detection\n  private noiseFloor = 0;\n  private noiseFloorSamples: number[] = [];\n  private readonly NOISE_FLOOR_SAMPLE_COUNT = 30;\n  private readonly SPEAKING_THRESHOLD_MULTIPLIER = 2.5;\n  private readonly WAVEFORM_BAR_COUNT = 60;\n  // Amplify raw amplitude so normal speech (±10–20 units) maps to visible levels\n  private readonly SENSITIVITY_MULTIPLIER = 5;\n\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> = this.isUserSpeakingSubject.asObservable();\n\n  start(stream: MediaStream): void {\n    if (this.isRunning) {\n      this.stop();\n    }\n\n    try {\n      this.audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();\n      const source = this.audioContext.createMediaStreamSource(stream);\n      \n      this.analyserNode = this.audioContext.createAnalyser();\n      this.analyserNode.fftSize = 2048;\n      this.analyserNode.smoothingTimeConstant = 0.4;\n      \n      const bufferLength = this.analyserNode.frequencyBinCount;\n      this.dataArray = new Uint8Array(bufferLength);\n      \n      source.connect(this.analyserNode);\n      \n      this.isRunning = true;\n      this.noiseFloorSamples = [];\n      this.noiseFloor = 0;\n      \n      this.analyze();\n    } catch (error) {\n      console.error('Error starting audio analyzer:', error);\n      this.stop();\n    }\n  }\n\n  stop(): void {\n    this.isRunning = false;\n    \n    if (this.animationFrameId !== null) {\n      cancelAnimationFrame(this.animationFrameId);\n      this.animationFrameId = null;\n    }\n\n    if (this.analyserNode) {\n      this.analyserNode.disconnect();\n      this.analyserNode = null;\n    }\n\n    if (this.audioContext && this.audioContext.state !== 'closed') {\n      this.audioContext.close().catch(console.error);\n      this.audioContext = null;\n    }\n\n    this.dataArray = null;\n    this.audioLevelsSubject.next([]);\n    this.isUserSpeakingSubject.next(false);\n  }\n\n  private analyze(): void {\n    if (!this.isRunning || !this.analyserNode || !this.dataArray) {\n      return;\n    }\n\n    this.analyserNode.getByteTimeDomainData(this.dataArray);\n    \n    // Calculate RMS (Root Mean Square) volume\n    const rms = this.calculateRMS(this.dataArray);\n    \n    // Learn noise floor during initial samples\n    if (this.noiseFloorSamples.length < this.NOISE_FLOOR_SAMPLE_COUNT) {\n      this.noiseFloorSamples.push(rms);\n      if (this.noiseFloorSamples.length === this.NOISE_FLOOR_SAMPLE_COUNT) {\n        // Calculate average noise floor\n        const sum = this.noiseFloorSamples.reduce((a, b) => a + b, 0);\n        this.noiseFloor = sum / this.NOISE_FLOOR_SAMPLE_COUNT;\n      }\n    } else {\n      // Update noise floor adaptively (moving average)\n      this.noiseFloor = this.noiseFloor * 0.99 + rms * 0.01;\n    }\n\n    // Detect if user is speaking\n    const threshold = this.noiseFloor * this.SPEAKING_THRESHOLD_MULTIPLIER;\n    const isSpeaking = rms > threshold;\n    this.isUserSpeakingSubject.next(isSpeaking);\n\n    // Generate waveform bars\n    const bars = this.generateWaveformBars(this.dataArray);\n    this.audioLevelsSubject.next(bars);\n\n    this.animationFrameId = requestAnimationFrame(() => this.analyze());\n  }\n\n  private calculateRMS(data: Uint8Array): number {\n    let sum = 0;\n    for (let i = 0; i < data.length; i++) {\n      const normalized = (data[i] - 128) / 128;\n      sum += normalized * normalized;\n    }\n    return Math.sqrt(sum / data.length);\n  }\n\n  private generateWaveformBars(data: Uint8Array): number[] {\n    const bars: number[] = [];\n    const step = Math.floor(data.length / this.WAVEFORM_BAR_COUNT);\n    \n    for (let i = 0; i < this.WAVEFORM_BAR_COUNT; i++) {\n      const index = i * step;\n      const value = data[index];\n      // Normalize and amplify so quiet speech produces visible movement\n      const normalized = Math.abs((value - 128) / 128) * 100 * this.SENSITIVITY_MULTIPLIER;\n      bars.push(Math.min(100, Math.max(0, normalized)));\n    }\n    \n    return bars;\n  }\n}\n"]}
|
|
@@ -1,35 +1,26 @@
|
|
|
1
1
|
import { __awaiter } from "tslib";
|
|
2
2
|
import { isPlatformBrowser } from '@angular/common';
|
|
3
3
|
import { Inject, Injectable, PLATFORM_ID } from '@angular/core';
|
|
4
|
-
import { BehaviorSubject, combineLatest, Subject, Subscription, } from 'rxjs';
|
|
5
|
-
import {
|
|
4
|
+
import { BehaviorSubject, combineLatest, concat, merge, of, Subject, Subscription, timer, } from 'rxjs';
|
|
5
|
+
import { distinctUntilChanged, map, startWith, switchMap, take, takeUntil, } from 'rxjs/operators';
|
|
6
6
|
import { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';
|
|
7
7
|
import { AudioAnalyzerService } from './audio-analyzer.service';
|
|
8
8
|
import { WebSocketVoiceClientService } from './websocket-voice-client.service';
|
|
9
|
-
import { DailyVoiceClientService } from './daily-voice-client.service';
|
|
10
9
|
import * as i0 from "@angular/core";
|
|
11
10
|
import * as i1 from "./audio-analyzer.service";
|
|
12
11
|
import * as i2 from "./websocket-voice-client.service";
|
|
13
|
-
import * as i3 from "
|
|
14
|
-
import * as i4 from "../../../services/platform-token-refresh.service";
|
|
12
|
+
import * as i3 from "../../../services/platform-token-refresh.service";
|
|
15
13
|
/**
|
|
16
|
-
* Voice agent orchestrator
|
|
17
|
-
*
|
|
18
|
-
*
|
|
19
|
-
* - Native WebSocket (WebSocketVoiceClientService) for signaling (room_created, transcripts)
|
|
20
|
-
* - Daily.js (DailyVoiceClientService) for WebRTC audio. Audio does NOT flow over WebSocket.
|
|
21
|
-
*
|
|
22
|
-
* - Maintains callState, statusText, duration, isMicMuted, isUserSpeaking, audioLevels
|
|
23
|
-
* - Uses WebSocket for room_created and transcripts only (no audio)
|
|
24
|
-
* - Uses Daily.js for all audio, mic, and real-time speaking detection
|
|
14
|
+
* Voice agent orchestrator: single WebSocket (`ws_url` from POST /ai/ask-voice-socket)
|
|
15
|
+
* for session events, transcripts, and optional speaking hints; local mic for capture
|
|
16
|
+
* and waveform only (no Daily/WebRTC room).
|
|
25
17
|
*/
|
|
26
18
|
export class VoiceAgentService {
|
|
27
|
-
constructor(audioAnalyzer, wsClient,
|
|
19
|
+
constructor(audioAnalyzer, wsClient, platformTokenRefresh,
|
|
28
20
|
/** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */
|
|
29
21
|
platformId) {
|
|
30
22
|
this.audioAnalyzer = audioAnalyzer;
|
|
31
23
|
this.wsClient = wsClient;
|
|
32
|
-
this.dailyClient = dailyClient;
|
|
33
24
|
this.platformTokenRefresh = platformTokenRefresh;
|
|
34
25
|
this.platformId = platformId;
|
|
35
26
|
this.callStateSubject = new BehaviorSubject('idle');
|
|
@@ -42,6 +33,11 @@ export class VoiceAgentService {
|
|
|
42
33
|
this.botTranscriptSubject = new Subject();
|
|
43
34
|
this.callStartTime = 0;
|
|
44
35
|
this.durationInterval = null;
|
|
36
|
+
this.localMicStream = null;
|
|
37
|
+
this.remoteAudioContext = null;
|
|
38
|
+
this.pendingRemoteAudio = [];
|
|
39
|
+
this.remoteAudioPlaying = false;
|
|
40
|
+
this.endCall$ = new Subject();
|
|
45
41
|
this.subscriptions = new Subscription();
|
|
46
42
|
this.destroy$ = new Subject();
|
|
47
43
|
this.callState$ = this.callStateSubject.asObservable();
|
|
@@ -52,8 +48,13 @@ export class VoiceAgentService {
|
|
|
52
48
|
this.audioLevels$ = this.audioLevelsSubject.asObservable();
|
|
53
49
|
this.userTranscript$ = this.userTranscriptSubject.asObservable();
|
|
54
50
|
this.botTranscript$ = this.botTranscriptSubject.asObservable();
|
|
55
|
-
// Waveform visualization only - do NOT use for speaking state
|
|
56
51
|
this.subscriptions.add(this.audioAnalyzer.audioLevels$.subscribe((levels) => this.audioLevelsSubject.next(levels)));
|
|
52
|
+
this.subscriptions.add(this.wsClient.remoteClose$
|
|
53
|
+
.pipe(takeUntil(this.destroy$))
|
|
54
|
+
.subscribe(() => void this.handleRemoteClose()));
|
|
55
|
+
this.subscriptions.add(this.wsClient.audioChunk$
|
|
56
|
+
.pipe(takeUntil(this.destroy$))
|
|
57
|
+
.subscribe((chunk) => this.enqueueRemoteAudio(chunk)));
|
|
57
58
|
}
|
|
58
59
|
ngOnDestroy() {
|
|
59
60
|
this.destroy$.next();
|
|
@@ -64,11 +65,13 @@ export class VoiceAgentService {
|
|
|
64
65
|
resetToIdle() {
|
|
65
66
|
if (this.callStateSubject.value === 'idle')
|
|
66
67
|
return;
|
|
68
|
+
this.endCall$.next();
|
|
67
69
|
this.stopDurationTimer();
|
|
70
|
+
this.callStartTime = 0;
|
|
68
71
|
this.audioAnalyzer.stop();
|
|
72
|
+
this.stopLocalMic();
|
|
73
|
+
this.resetRemoteAudioPlayback();
|
|
69
74
|
this.wsClient.disconnect();
|
|
70
|
-
// Fire-and-forget: Daily disconnect is async; connect() will await if needed
|
|
71
|
-
void this.dailyClient.disconnect();
|
|
72
75
|
this.callStateSubject.next('idle');
|
|
73
76
|
this.statusTextSubject.next('');
|
|
74
77
|
this.durationSubject.next('0:00');
|
|
@@ -83,9 +86,6 @@ export class VoiceAgentService {
|
|
|
83
86
|
this.callStateSubject.next('connecting');
|
|
84
87
|
this.statusTextSubject.next('Connecting...');
|
|
85
88
|
let accessToken = token;
|
|
86
|
-
// Align with chat drawer token handling: always delegate to
|
|
87
|
-
// PlatformTokenRefreshService when we have a usersApiUrl, so it can
|
|
88
|
-
// fall back to stored tokens even if the caller passed an empty token.
|
|
89
89
|
if (usersApiUrl && isPlatformBrowser(this.platformId)) {
|
|
90
90
|
try {
|
|
91
91
|
const ensured = yield this.platformTokenRefresh
|
|
@@ -101,7 +101,7 @@ export class VoiceAgentService {
|
|
|
101
101
|
}
|
|
102
102
|
}
|
|
103
103
|
const baseUrl = apiUrl.replace(/\/$/, '');
|
|
104
|
-
const postUrl = `${baseUrl}/ai/ask-voice`;
|
|
104
|
+
const postUrl = `${baseUrl}/ai/ask-voice-socket`;
|
|
105
105
|
const headers = {
|
|
106
106
|
'Content-Type': 'application/json',
|
|
107
107
|
Authorization: `Bearer ${accessToken}`,
|
|
@@ -113,7 +113,6 @@ export class VoiceAgentService {
|
|
|
113
113
|
eventToken,
|
|
114
114
|
'ngrok-skip-browser-warning': 'true',
|
|
115
115
|
};
|
|
116
|
-
// POST to get ws_url for signaling
|
|
117
116
|
const res = yield fetch(postUrl, {
|
|
118
117
|
method: 'POST',
|
|
119
118
|
headers,
|
|
@@ -127,33 +126,21 @@ export class VoiceAgentService {
|
|
|
127
126
|
throw new Error(`HTTP ${res.status}`);
|
|
128
127
|
}
|
|
129
128
|
const json = yield res.json();
|
|
130
|
-
const wsUrl = json === null || json === void 0 ? void 0 : json.
|
|
131
|
-
|
|
129
|
+
const wsUrl = (typeof (json === null || json === void 0 ? void 0 : json.ws_url) === 'string' && json.ws_url) ||
|
|
130
|
+
(typeof (json === null || json === void 0 ? void 0 : json.rn_ws_url) === 'string' && json.rn_ws_url);
|
|
131
|
+
if (!wsUrl) {
|
|
132
132
|
throw new Error('No ws_url in response');
|
|
133
133
|
}
|
|
134
|
-
|
|
135
|
-
this.wsClient.roomCreated$
|
|
136
|
-
.pipe(take(1), takeUntil(this.destroy$))
|
|
137
|
-
.subscribe((roomUrl) => __awaiter(this, void 0, void 0, function* () {
|
|
138
|
-
try {
|
|
139
|
-
yield this.onRoomCreated(roomUrl);
|
|
140
|
-
}
|
|
141
|
-
catch (err) {
|
|
142
|
-
console.error('Daily join failed:', err);
|
|
143
|
-
this.callStateSubject.next('ended');
|
|
144
|
-
this.statusTextSubject.next('Connection failed');
|
|
145
|
-
yield this.disconnect();
|
|
146
|
-
throw err;
|
|
147
|
-
}
|
|
148
|
-
}));
|
|
149
|
-
// Forward transcripts from WebSocket
|
|
134
|
+
const untilCallEnds$ = merge(this.destroy$, this.endCall$);
|
|
150
135
|
this.subscriptions.add(this.wsClient.userTranscript$
|
|
151
|
-
.pipe(takeUntil(
|
|
136
|
+
.pipe(takeUntil(untilCallEnds$))
|
|
152
137
|
.subscribe((t) => this.userTranscriptSubject.next(t)));
|
|
153
138
|
this.subscriptions.add(this.wsClient.botTranscript$
|
|
154
|
-
.pipe(takeUntil(
|
|
139
|
+
.pipe(takeUntil(untilCallEnds$))
|
|
155
140
|
.subscribe((t) => this.botTranscriptSubject.next(t)));
|
|
156
|
-
|
|
141
|
+
this.subscriptions.add(this.wsClient.opened$
|
|
142
|
+
.pipe(takeUntil(untilCallEnds$), take(1))
|
|
143
|
+
.subscribe(() => void this.onWebsocketOpened()));
|
|
157
144
|
this.wsClient.connect(wsUrl);
|
|
158
145
|
}
|
|
159
146
|
catch (error) {
|
|
@@ -165,59 +152,185 @@ export class VoiceAgentService {
|
|
|
165
152
|
}
|
|
166
153
|
});
|
|
167
154
|
}
|
|
168
|
-
|
|
155
|
+
onWebsocketOpened() {
|
|
169
156
|
return __awaiter(this, void 0, void 0, function* () {
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
.
|
|
175
|
-
.
|
|
176
|
-
this.
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
this.
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
157
|
+
if (this.callStateSubject.value !== 'connecting') {
|
|
158
|
+
return;
|
|
159
|
+
}
|
|
160
|
+
try {
|
|
161
|
+
yield this.startLocalMic();
|
|
162
|
+
this.statusTextSubject.next('Connected');
|
|
163
|
+
this.callStateSubject.next('connected');
|
|
164
|
+
this.wireSpeakingState();
|
|
165
|
+
}
|
|
166
|
+
catch (err) {
|
|
167
|
+
console.error('[HiveGpt Voice] Mic or session setup failed', err);
|
|
168
|
+
this.callStateSubject.next('ended');
|
|
169
|
+
this.statusTextSubject.next('Microphone unavailable');
|
|
170
|
+
yield this.disconnect();
|
|
171
|
+
}
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
wireSpeakingState() {
|
|
175
|
+
const untilCallEnds$ = merge(this.destroy$, this.endCall$);
|
|
176
|
+
const transcriptDrivenAssistant$ = this.wsClient.botTranscript$.pipe(switchMap(() => concat(of(true), timer(800).pipe(map(() => false)))), distinctUntilChanged());
|
|
177
|
+
const assistantTalking$ = merge(this.wsClient.assistantSpeaking$, transcriptDrivenAssistant$).pipe(distinctUntilChanged(), startWith(false));
|
|
178
|
+
const userTalking$ = combineLatest([
|
|
179
|
+
this.audioAnalyzer.isUserSpeaking$,
|
|
180
|
+
this.wsClient.serverUserSpeaking$.pipe(startWith(false)),
|
|
181
|
+
]).pipe(map(([local, server]) => local || server), distinctUntilChanged(), startWith(false));
|
|
182
|
+
this.subscriptions.add(combineLatest([assistantTalking$, userTalking$])
|
|
183
|
+
.pipe(takeUntil(untilCallEnds$))
|
|
184
|
+
.subscribe(([bot, user]) => {
|
|
185
|
+
const current = this.callStateSubject.value;
|
|
186
|
+
if (user) {
|
|
187
|
+
this.isUserSpeakingSubject.next(true);
|
|
188
|
+
this.callStateSubject.next('listening');
|
|
189
|
+
}
|
|
190
|
+
else {
|
|
191
|
+
this.isUserSpeakingSubject.next(false);
|
|
192
|
+
}
|
|
193
|
+
if (user) {
|
|
194
|
+
return;
|
|
195
|
+
}
|
|
196
|
+
if (bot) {
|
|
197
|
+
if (this.callStartTime === 0) {
|
|
188
198
|
this.callStartTime = Date.now();
|
|
189
199
|
this.startDurationTimer();
|
|
190
|
-
this.callStateSubject.next('talking');
|
|
191
|
-
return;
|
|
192
200
|
}
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
201
|
+
this.callStateSubject.next('talking');
|
|
202
|
+
}
|
|
203
|
+
else if (current === 'talking' || current === 'listening') {
|
|
204
|
+
this.callStateSubject.next('connected');
|
|
205
|
+
}
|
|
206
|
+
}));
|
|
207
|
+
}
|
|
208
|
+
startLocalMic() {
|
|
209
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
210
|
+
this.stopLocalMic();
|
|
211
|
+
const stream = yield navigator.mediaDevices.getUserMedia({ audio: true });
|
|
212
|
+
const track = stream.getAudioTracks()[0];
|
|
213
|
+
if (!track) {
|
|
214
|
+
stream.getTracks().forEach((t) => t.stop());
|
|
215
|
+
throw new Error('No audio track');
|
|
216
|
+
}
|
|
217
|
+
this.localMicStream = stream;
|
|
218
|
+
this.isMicMutedSubject.next(!track.enabled);
|
|
219
|
+
this.audioAnalyzer.start(stream);
|
|
220
|
+
});
|
|
221
|
+
}
|
|
222
|
+
stopLocalMic() {
|
|
223
|
+
if (this.localMicStream) {
|
|
224
|
+
this.localMicStream.getTracks().forEach((t) => t.stop());
|
|
225
|
+
this.localMicStream = null;
|
|
226
|
+
}
|
|
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);
|
|
198
246
|
}
|
|
199
|
-
|
|
200
|
-
|
|
247
|
+
catch (_a) {
|
|
248
|
+
// Ignore undecodable chunks; server may mix non-audio binary events.
|
|
201
249
|
}
|
|
202
|
-
}
|
|
203
|
-
this.
|
|
204
|
-
this.
|
|
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
|
+
}
|
|
298
|
+
handleRemoteClose() {
|
|
299
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
300
|
+
const state = this.callStateSubject.value;
|
|
301
|
+
if (state === 'idle' || state === 'ended')
|
|
302
|
+
return;
|
|
303
|
+
this.endCall$.next();
|
|
304
|
+
this.stopDurationTimer();
|
|
305
|
+
this.callStartTime = 0;
|
|
306
|
+
this.audioAnalyzer.stop();
|
|
307
|
+
this.stopLocalMic();
|
|
308
|
+
this.resetRemoteAudioPlayback();
|
|
309
|
+
this.callStateSubject.next('ended');
|
|
310
|
+
this.statusTextSubject.next('Connection lost');
|
|
205
311
|
});
|
|
206
312
|
}
|
|
207
313
|
disconnect() {
|
|
208
314
|
return __awaiter(this, void 0, void 0, function* () {
|
|
315
|
+
this.endCall$.next();
|
|
209
316
|
this.stopDurationTimer();
|
|
317
|
+
this.callStartTime = 0;
|
|
210
318
|
this.audioAnalyzer.stop();
|
|
211
|
-
|
|
212
|
-
|
|
319
|
+
this.stopLocalMic();
|
|
320
|
+
this.resetRemoteAudioPlayback();
|
|
213
321
|
this.wsClient.disconnect();
|
|
214
322
|
this.callStateSubject.next('ended');
|
|
215
323
|
this.statusTextSubject.next('Call Ended');
|
|
216
324
|
});
|
|
217
325
|
}
|
|
218
326
|
toggleMic() {
|
|
219
|
-
|
|
220
|
-
this.
|
|
327
|
+
var _a;
|
|
328
|
+
const nextMuted = !this.isMicMutedSubject.value;
|
|
329
|
+
const track = (_a = this.localMicStream) === null || _a === void 0 ? void 0 : _a.getAudioTracks()[0];
|
|
330
|
+
if (track) {
|
|
331
|
+
track.enabled = !nextMuted;
|
|
332
|
+
}
|
|
333
|
+
this.isMicMutedSubject.next(nextMuted);
|
|
221
334
|
}
|
|
222
335
|
startDurationTimer() {
|
|
223
336
|
const updateDuration = () => {
|
|
@@ -238,7 +351,7 @@ export class VoiceAgentService {
|
|
|
238
351
|
}
|
|
239
352
|
}
|
|
240
353
|
}
|
|
241
|
-
VoiceAgentService.ɵprov = i0.ɵɵdefineInjectable({ factory: function VoiceAgentService_Factory() { return new VoiceAgentService(i0.ɵɵinject(i1.AudioAnalyzerService), i0.ɵɵinject(i2.WebSocketVoiceClientService), i0.ɵɵinject(i3.
|
|
354
|
+
VoiceAgentService.ɵprov = i0.ɵɵdefineInjectable({ factory: function VoiceAgentService_Factory() { return new VoiceAgentService(i0.ɵɵinject(i1.AudioAnalyzerService), i0.ɵɵinject(i2.WebSocketVoiceClientService), i0.ɵɵinject(i3.PlatformTokenRefreshService), i0.ɵɵinject(i0.PLATFORM_ID)); }, token: VoiceAgentService, providedIn: "root" });
|
|
242
355
|
VoiceAgentService.decorators = [
|
|
243
356
|
{ type: Injectable, args: [{
|
|
244
357
|
providedIn: 'root',
|
|
@@ -247,8 +360,7 @@ VoiceAgentService.decorators = [
|
|
|
247
360
|
VoiceAgentService.ctorParameters = () => [
|
|
248
361
|
{ type: AudioAnalyzerService },
|
|
249
362
|
{ type: WebSocketVoiceClientService },
|
|
250
|
-
{ type: DailyVoiceClientService },
|
|
251
363
|
{ type: PlatformTokenRefreshService },
|
|
252
364
|
{ type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
|
|
253
365
|
];
|
|
254
|
-
//# 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  Observable,\n  Subject,\n  Subscription,\n} from 'rxjs';\nimport { filter, take, takeUntil } from 'rxjs/operators';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\nimport { WebSocketVoiceClientService } from './websocket-voice-client.service';\nimport { DailyVoiceClientService } from './daily-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. Coordinates WebSocket (signaling) and Daily.js (WebRTC audio).\n *\n * CRITICAL: This service must NEVER use Socket.IO or ngx-socket-io. Voice flow uses only:\n * - Native WebSocket (WebSocketVoiceClientService) for signaling (room_created, transcripts)\n * - Daily.js (DailyVoiceClientService) for WebRTC audio. Audio does NOT flow over WebSocket.\n *\n * - Maintains callState, statusText, duration, isMicMuted, isUserSpeaking, audioLevels\n * - Uses WebSocket for room_created and transcripts only (no audio)\n * - Uses Daily.js for all audio, mic, and real-time speaking detection\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 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 dailyClient: DailyVoiceClientService,\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    // Waveform visualization only - do NOT use for speaking state\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    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.stopDurationTimer();\n    this.audioAnalyzer.stop();\n    this.wsClient.disconnect();\n    // Fire-and-forget: Daily disconnect is async; connect() will await if needed\n    void this.dailyClient.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      // Align with chat drawer token handling: always delegate to\n      // PlatformTokenRefreshService when we have a usersApiUrl, so it can\n      // fall back to stored tokens even if the caller passed an empty 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`;\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      // POST to get ws_url for signaling\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 = json?.rn_ws_url;\n      if (!wsUrl || typeof wsUrl !== 'string') {\n        throw new Error('No ws_url in response');\n      }\n\n      // Subscribe to room_created BEFORE connecting to avoid race\n      this.wsClient.roomCreated$\n        .pipe(take(1), takeUntil(this.destroy$))\n        .subscribe(async (roomUrl) => {\n          try {\n            await this.onRoomCreated(roomUrl);\n          } catch (err) {\n            console.error('Daily join failed:', err);\n            this.callStateSubject.next('ended');\n            this.statusTextSubject.next('Connection failed');\n            await this.disconnect();\n            throw err;\n          }\n        });\n\n      // Forward transcripts from WebSocket\n      this.subscriptions.add(\n        this.wsClient.userTranscript$\n          .pipe(takeUntil(this.destroy$))\n          .subscribe((t) => this.userTranscriptSubject.next(t)),\n      );\n      this.subscriptions.add(\n        this.wsClient.botTranscript$\n          .pipe(takeUntil(this.destroy$))\n          .subscribe((t) => this.botTranscriptSubject.next(t)),\n      );\n\n      // Connect signaling WebSocket (no audio over WS)\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 onRoomCreated(roomUrl: string): Promise<void> {\n    // Connect Daily.js for WebRTC audio\n    await this.dailyClient.connect(roomUrl);\n\n    // Waveform: use local mic stream from Daily client\n    this.dailyClient.localStream$\n      .pipe(\n        filter((s): s is MediaStream => s != null),\n        take(1),\n      )\n      .subscribe((stream) => {\n        this.audioAnalyzer.start(stream);\n      });\n\n    this.subscriptions.add(\n      this.dailyClient.userSpeaking$.subscribe((s) =>\n        this.isUserSpeakingSubject.next(s),\n      ),\n    );\n    this.subscriptions.add(\n      combineLatest([\n        this.dailyClient.speaking$,\n        this.dailyClient.userSpeaking$,\n      ]).subscribe(([bot, user]) => {\n        const current = this.callStateSubject.value;\n        if (current === 'connecting' && !bot) {\n          return;\n        }\n        if (current === 'connecting' && bot) {\n          this.callStartTime = Date.now();\n          this.startDurationTimer();\n          this.callStateSubject.next('talking');\n          return;\n        }\n        if (user) {\n          this.callStateSubject.next('listening');\n        } else if (bot) {\n          this.callStateSubject.next('talking');\n        } else if (current === 'talking' || current === 'listening') {\n          this.callStateSubject.next('connected');\n        }\n      }),\n    );\n\n    this.subscriptions.add(\n      this.dailyClient.micMuted$.subscribe((muted) =>\n        this.isMicMutedSubject.next(muted),\n      ),\n    );\n\n    this.statusTextSubject.next('Connecting...');\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n\n    // Daily first, then WebSocket\n    await this.dailyClient.disconnect();\n    this.wsClient.disconnect();\n\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  toggleMic(): void {\n    const current = this.isMicMutedSubject.value;\n    this.dailyClient.setMuted(!current);\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"]}
|