@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.
Files changed (25) hide show
  1. package/bundles/hivegpt-hiveai-angular.umd.js +420 -490
  2. package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
  3. package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
  4. package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
  5. package/esm2015/hivegpt-hiveai-angular.js +4 -5
  6. package/esm2015/lib/components/voice-agent/services/audio-analyzer.service.js +3 -3
  7. package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +195 -83
  8. package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +160 -49
  9. package/esm2015/lib/components/voice-agent/voice-agent.module.js +3 -5
  10. package/fesm2015/hivegpt-hiveai-angular.js +338 -416
  11. package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
  12. package/hivegpt-hiveai-angular.d.ts +3 -4
  13. package/hivegpt-hiveai-angular.d.ts.map +1 -1
  14. package/hivegpt-hiveai-angular.metadata.json +1 -1
  15. package/lib/components/voice-agent/services/audio-analyzer.service.d.ts +2 -2
  16. package/lib/components/voice-agent/services/voice-agent.service.d.ts +22 -13
  17. package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
  18. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +30 -20
  19. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
  20. package/lib/components/voice-agent/voice-agent.module.d.ts +1 -1
  21. package/lib/components/voice-agent/voice-agent.module.d.ts.map +1 -1
  22. package/package.json +1 -1
  23. package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +0 -305
  24. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +0 -62
  25. 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 ɵj } from './lib/components/bot-html-editor/bot-html-editor.component';
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 ɵh } from './lib/components/video-player/video-player.component';
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 ɵi } from './lib/pipes/safe-html.pipe';
15
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGl2ZWdwdC1oaXZlYWktYW5ndWxhci5qcyIsInNvdXJjZVJvb3QiOiIvVXNlcnMvcm9oaXR0aGFrdXIvaGl2ZS1ncHQvSGl2ZUFJLVBhY2thZ2VzL0FuZ3VsYXIvcHJvamVjdHMvaGl2ZWdwdC9ldmVudHNncHQtYW5ndWxhci9zcmMvIiwic291cmNlcyI6WyJoaXZlZ3B0LWhpdmVhaS1hbmd1bGFyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUM7QUFFN0IsT0FBTyxFQUFDLGtCQUFrQixJQUFJLEVBQUUsRUFBQyxNQUFNLHFDQUFxQyxDQUFDO0FBQzdFLE9BQU8sRUFBQyxzQkFBc0IsSUFBSSxFQUFFLEVBQUMsTUFBTSw0REFBNEQsQ0FBQztBQUN4RyxPQUFPLEVBQUMsV0FBVyxJQUFJLEVBQUUsRUFBQyxNQUFNLDhCQUE4QixDQUFDO0FBQy9ELE9BQU8sRUFBQyxtQkFBbUIsSUFBSSxFQUFFLEVBQUMsTUFBTSx1Q0FBdUMsQ0FBQztBQUNoRixPQUFPLEVBQUMsYUFBYSxJQUFJLEVBQUUsRUFBQyxNQUFNLHlDQUF5QyxDQUFDO0FBQzVFLE9BQU8sRUFBQyxrQkFBa0IsSUFBSSxFQUFFLEVBQUMsTUFBTSxtREFBbUQsQ0FBQztBQUMzRixPQUFPLEVBQUMsb0JBQW9CLElBQUksRUFBRSxFQUFDLE1BQU0sc0RBQXNELENBQUM7QUFDaEcsT0FBTyxFQUFDLHVCQUF1QixJQUFJLEVBQUUsRUFBQyxNQUFNLGtFQUFrRSxDQUFDO0FBQy9HLE9BQU8sRUFBQywyQkFBMkIsSUFBSSxFQUFFLEVBQUMsTUFBTSxzRUFBc0UsQ0FBQztBQUN2SCxPQUFPLEVBQUMsWUFBWSxJQUFJLEVBQUUsRUFBQyxNQUFNLDRCQUE0QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuXG5leHBvcnQge05vdGlmaWNhdGlvblNvY2tldCBhcyDJtWR9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvTm90aWZpY2F0aW9uU29ja2V0JztcbmV4cG9ydCB7Qm90SHRtbEVkaXRvckNvbXBvbmVudCBhcyDJtWp9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvYm90LWh0bWwtZWRpdG9yL2JvdC1odG1sLWVkaXRvci5jb21wb25lbnQnO1xuZXhwb3J0IHtCb3RzU2VydmljZSBhcyDJtWF9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvYm90LnNlcnZpY2UnO1xuZXhwb3J0IHtDb252ZXJzYXRpb25TZXJ2aWNlIGFzIMm1Y30gZnJvbSAnLi9saWIvY29tcG9uZW50cy9jb252ZXJzYXRpb24uc2VydmljZSc7XG5leHBvcnQge1NvY2tldFNlcnZpY2UgYXMgybVifSBmcm9tICcuL2xpYi9jb21wb25lbnRzL3NvY2tldC1zZXJ2aWNlLnNlcnZpY2UnO1xuZXhwb3J0IHtUcmFuc2xhdGlvblNlcnZpY2UgYXMgybVlfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL3RyYW5zbGF0aW9ucy90cmFuc2xhdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCB7VmlkZW9QbGF5ZXJDb21wb25lbnQgYXMgybVofSBmcm9tICcuL2xpYi9jb21wb25lbnRzL3ZpZGVvLXBsYXllci92aWRlby1wbGF5ZXIuY29tcG9uZW50JztcbmV4cG9ydCB7RGFpbHlWb2ljZUNsaWVudFNlcnZpY2UgYXMgybVnfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3NlcnZpY2VzL2RhaWx5LXZvaWNlLWNsaWVudC5zZXJ2aWNlJztcbmV4cG9ydCB7V2ViU29ja2V0Vm9pY2VDbGllbnRTZXJ2aWNlIGFzIMm1Zn0gZnJvbSAnLi9saWIvY29tcG9uZW50cy92b2ljZS1hZ2VudC9zZXJ2aWNlcy93ZWJzb2NrZXQtdm9pY2UtY2xpZW50LnNlcnZpY2UnO1xuZXhwb3J0IHtTYWZlSHRtbFBpcGUgYXMgybVpfSBmcm9tICcuL2xpYi9waXBlcy9zYWZlLWh0bWwucGlwZSc7Il19
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 only.
6
- * Do NOT use isUserSpeaking$ for call state; speaking state must come from Daily.js.
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 { filter, take, takeUntil } from 'rxjs/operators';
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 "./daily-voice-client.service";
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. Coordinates WebSocket (signaling) and Daily.js (WebRTC audio).
17
- *
18
- * CRITICAL: This service must NEVER use Socket.IO or ngx-socket-io. Voice flow uses only:
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, dailyClient, platformTokenRefresh,
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.rn_ws_url;
131
- if (!wsUrl || typeof wsUrl !== 'string') {
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
- // Subscribe to room_created BEFORE connecting to avoid race
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(this.destroy$))
136
+ .pipe(takeUntil(untilCallEnds$))
152
137
  .subscribe((t) => this.userTranscriptSubject.next(t)));
153
138
  this.subscriptions.add(this.wsClient.botTranscript$
154
- .pipe(takeUntil(this.destroy$))
139
+ .pipe(takeUntil(untilCallEnds$))
155
140
  .subscribe((t) => this.botTranscriptSubject.next(t)));
156
- // Connect signaling WebSocket (no audio over WS)
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
- onRoomCreated(roomUrl) {
155
+ onWebsocketOpened() {
169
156
  return __awaiter(this, void 0, void 0, function* () {
170
- // Connect Daily.js for WebRTC audio
171
- yield this.dailyClient.connect(roomUrl);
172
- // Waveform: use local mic stream from Daily client
173
- this.dailyClient.localStream$
174
- .pipe(filter((s) => s != null), take(1))
175
- .subscribe((stream) => {
176
- this.audioAnalyzer.start(stream);
177
- });
178
- this.subscriptions.add(this.dailyClient.userSpeaking$.subscribe((s) => this.isUserSpeakingSubject.next(s)));
179
- this.subscriptions.add(combineLatest([
180
- this.dailyClient.speaking$,
181
- this.dailyClient.userSpeaking$,
182
- ]).subscribe(([bot, user]) => {
183
- const current = this.callStateSubject.value;
184
- if (current === 'connecting' && !bot) {
185
- return;
186
- }
187
- if (current === 'connecting' && bot) {
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
- if (user) {
194
- this.callStateSubject.next('listening');
195
- }
196
- else if (bot) {
197
- this.callStateSubject.next('talking');
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
- else if (current === 'talking' || current === 'listening') {
200
- this.callStateSubject.next('connected');
247
+ catch (_a) {
248
+ // Ignore undecodable chunks; server may mix non-audio binary events.
201
249
  }
202
- }));
203
- this.subscriptions.add(this.dailyClient.micMuted$.subscribe((muted) => this.isMicMutedSubject.next(muted)));
204
- this.statusTextSubject.next('Connecting...');
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
- // Daily first, then WebSocket
212
- yield this.dailyClient.disconnect();
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
- const current = this.isMicMutedSubject.value;
220
- this.dailyClient.setMuted(!current);
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.DailyVoiceClientService), i0.ɵɵinject(i4.PlatformTokenRefreshService), i0.ɵɵinject(i0.PLATFORM_ID)); }, token: VoiceAgentService, providedIn: "root" });
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"]}