@hivegpt/hiveai-angular 0.0.574 → 0.0.576

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 (21) hide show
  1. package/bundles/hivegpt-hiveai-angular.umd.js +542 -278
  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/lib/components/chat-drawer/chat-drawer.component.js +6 -6
  6. package/esm2015/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.js +85 -54
  7. package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +153 -63
  8. package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +160 -88
  9. package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +81 -34
  10. package/fesm2015/hivegpt-hiveai-angular.js +478 -239
  11. package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
  12. package/hivegpt-hiveai-angular.metadata.json +1 -1
  13. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +7 -1
  14. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
  15. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +37 -3
  16. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +1 -1
  17. package/lib/components/voice-agent/services/voice-agent.service.d.ts +19 -6
  18. package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
  19. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +5 -12
  20. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
  21. package/package.json +1 -1
@@ -12,8 +12,21 @@ import * as i1 from "./audio-analyzer.service";
12
12
  import * as i2 from "./websocket-voice-client.service";
13
13
  import * as i3 from "./daily-voice-client.service";
14
14
  import * as i4 from "../../../services/platform-token-refresh.service";
15
+ /**
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
25
+ */
15
26
  export class VoiceAgentService {
16
- constructor(audioAnalyzer, wsClient, dailyClient, platformTokenRefresh, platformId) {
27
+ constructor(audioAnalyzer, wsClient, dailyClient, platformTokenRefresh,
28
+ /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */
29
+ platformId) {
17
30
  this.audioAnalyzer = audioAnalyzer;
18
31
  this.wsClient = wsClient;
19
32
  this.dailyClient = dailyClient;
@@ -22,7 +35,7 @@ export class VoiceAgentService {
22
35
  this.callStateSubject = new BehaviorSubject('idle');
23
36
  this.statusTextSubject = new BehaviorSubject('');
24
37
  this.durationSubject = new BehaviorSubject('00:00');
25
- this.isMicMutedSubject = new BehaviorSubject(true);
38
+ this.isMicMutedSubject = new BehaviorSubject(false);
26
39
  this.isUserSpeakingSubject = new BehaviorSubject(false);
27
40
  this.audioLevelsSubject = new BehaviorSubject([]);
28
41
  this.userTranscriptSubject = new Subject();
@@ -30,6 +43,8 @@ export class VoiceAgentService {
30
43
  this.callStartTime = 0;
31
44
  this.durationInterval = null;
32
45
  this.subscriptions = new Subscription();
46
+ /** Per-call only; cleared on disconnect / reset / new room so handlers do not stack. */
47
+ this.callSubscriptions = new Subscription();
33
48
  this.destroy$ = new Subject();
34
49
  this.callState$ = this.callStateSubject.asObservable();
35
50
  this.statusText$ = this.statusTextSubject.asObservable();
@@ -39,118 +54,163 @@ export class VoiceAgentService {
39
54
  this.audioLevels$ = this.audioLevelsSubject.asObservable();
40
55
  this.userTranscript$ = this.userTranscriptSubject.asObservable();
41
56
  this.botTranscript$ = this.botTranscriptSubject.asObservable();
57
+ // Waveform visualization only - do NOT use for speaking state
42
58
  this.subscriptions.add(this.audioAnalyzer.audioLevels$.subscribe((levels) => this.audioLevelsSubject.next(levels)));
59
+ // Transcripts: single subscription for service lifetime (avoid stacking on each connect()).
60
+ // WebSocket is disconnected between calls; no replay — new subscribers (setupVoiceTranscripts)
61
+ // only receive messages from the new WS after connect.
62
+ this.subscriptions.add(this.wsClient.userTranscript$
63
+ .pipe(takeUntil(this.destroy$))
64
+ .subscribe((t) => this.userTranscriptSubject.next(t)));
65
+ this.subscriptions.add(this.wsClient.botTranscript$
66
+ .pipe(takeUntil(this.destroy$))
67
+ .subscribe((t) => this.botTranscriptSubject.next(t)));
43
68
  }
44
69
  ngOnDestroy() {
45
70
  this.destroy$.next();
46
71
  this.subscriptions.unsubscribe();
47
72
  this.disconnect();
48
73
  }
49
- /**
50
- * Tear down transports and reset UI state so a new `connect()` can run.
51
- * `connect()` only proceeds from `idle`; use this after `ended` or when reopening the modal.
52
- */
74
+ /** Reset to idle state (e.g. when modal opens so user can click Start Call). */
53
75
  resetToIdle() {
76
+ if (this.callStateSubject.value === 'idle')
77
+ return;
78
+ this.callSubscriptions.unsubscribe();
79
+ this.callSubscriptions = new Subscription();
54
80
  this.stopDurationTimer();
55
81
  this.audioAnalyzer.stop();
56
82
  this.wsClient.disconnect();
83
+ // Fire-and-forget: Daily disconnect is async; connect() will await if needed
57
84
  void this.dailyClient.disconnect();
58
85
  this.callStateSubject.next('idle');
59
86
  this.statusTextSubject.next('');
60
- this.durationSubject.next('00:00');
61
- this.isMicMutedSubject.next(true);
62
- this.isUserSpeakingSubject.next(false);
63
- this.audioLevelsSubject.next([]);
87
+ this.durationSubject.next('0:00');
64
88
  }
65
- connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl) {
89
+ connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl, existingMicStream) {
90
+ var _a;
66
91
  return __awaiter(this, void 0, void 0, function* () {
67
- if (this.callStateSubject.value !== 'idle')
92
+ if (this.callStateSubject.value !== 'idle') {
93
+ console.warn('Call already in progress');
68
94
  return;
95
+ }
69
96
  try {
70
97
  this.callStateSubject.next('connecting');
71
98
  this.statusTextSubject.next('Connecting...');
72
- let accessToken = token;
73
- if (usersApiUrl && isPlatformBrowser(this.platformId)) {
74
- try {
75
- const ensured = yield this.platformTokenRefresh
76
- .ensureValidAccessToken(token, usersApiUrl)
77
- .pipe(take(1))
78
- .toPromise();
79
- if (ensured === null || ensured === void 0 ? void 0 : ensured.accessToken) {
80
- accessToken = ensured.accessToken;
81
- }
82
- }
83
- catch (_a) { }
84
- }
85
- const base = (apiUrl || '').replace(/\/+$/, '');
86
- const postUrl = `${base}/ai/ask-voice`;
87
- // Same as chat-drawer `/ai/ask` headers: use `eventId` (camelCase), value from host `this.eventId`.
88
- const eventIdHeader = (eventId && String(eventId).trim()) || '';
99
+ const tokenPromise = usersApiUrl && isPlatformBrowser(this.platformId)
100
+ ? this.platformTokenRefresh
101
+ .ensureValidAccessToken(token, usersApiUrl)
102
+ .pipe(take(1))
103
+ .toPromise()
104
+ .then((ensured) => { var _a; return (_a = ensured === null || ensured === void 0 ? void 0 : ensured.accessToken) !== null && _a !== void 0 ? _a : token; })
105
+ .catch((e) => {
106
+ console.warn('[HiveGpt Voice] Token refresh before connect failed', e);
107
+ return token;
108
+ })
109
+ : Promise.resolve(token);
110
+ const prepPromise = Promise.resolve().then(() => {
111
+ const baseUrl = apiUrl.replace(/\/$/, '');
112
+ return {
113
+ postUrl: `${baseUrl}/ai/ask-voice`,
114
+ body: JSON.stringify({
115
+ bot_id: botId,
116
+ conversation_id: conversationId,
117
+ voice: 'alloy',
118
+ }),
119
+ };
120
+ });
121
+ const micPromise = (existingMicStream === null || existingMicStream === void 0 ? void 0 : existingMicStream.getAudioTracks().some((t) => t.readyState === 'live'))
122
+ ? Promise.resolve(existingMicStream)
123
+ : isPlatformBrowser(this.platformId) &&
124
+ ((_a = navigator.mediaDevices) === null || _a === void 0 ? void 0 : _a.getUserMedia)
125
+ ? navigator.mediaDevices
126
+ .getUserMedia({ audio: true })
127
+ .catch(() => undefined)
128
+ : Promise.resolve(undefined);
129
+ const [accessToken, { postUrl, body }, micStream] = yield Promise.all([
130
+ tokenPromise,
131
+ prepPromise,
132
+ micPromise,
133
+ ]);
134
+ const headers = {
135
+ 'Content-Type': 'application/json',
136
+ Authorization: `Bearer ${accessToken}`,
137
+ 'x-api-key': apiKey,
138
+ 'hive-bot-id': botId,
139
+ 'domain-authority': domainAuthority,
140
+ eventUrl,
141
+ eventId,
142
+ eventToken,
143
+ 'ngrok-skip-browser-warning': 'true',
144
+ };
145
+ // POST to get ws_url for signaling
89
146
  const res = yield fetch(postUrl, {
90
147
  method: 'POST',
91
- headers: {
92
- 'Content-Type': 'application/json',
93
- Authorization: `Bearer ${accessToken}`,
94
- 'domain-authority': domainAuthority,
95
- eventtoken: eventToken,
96
- eventurl: eventUrl,
97
- 'hive-bot-id': botId,
98
- 'x-api-key': apiKey,
99
- eventId: eventIdHeader,
100
- },
101
- body: JSON.stringify({
102
- bot_id: botId,
103
- conversation_id: conversationId,
104
- voice: 'alloy',
105
- }),
148
+ headers,
149
+ body,
106
150
  });
151
+ if (!res.ok) {
152
+ throw new Error(`HTTP ${res.status}`);
153
+ }
107
154
  const json = yield res.json();
108
155
  const wsUrl = json === null || json === void 0 ? void 0 : json.rn_ws_url;
109
- this.wsClient.roomCreated$
110
- .pipe(take(1), takeUntil(this.destroy$))
111
- .subscribe((roomUrl) => __awaiter(this, void 0, void 0, function* () {
112
- yield this.onRoomCreated(roomUrl);
113
- }));
114
- this.subscriptions.add(this.wsClient.userTranscript$.subscribe((t) => this.userTranscriptSubject.next(t)));
115
- this.subscriptions.add(this.wsClient.botTranscript$.subscribe((t) => this.botTranscriptSubject.next(t)));
116
- this.wsClient.connect(wsUrl);
156
+ if (!wsUrl || typeof wsUrl !== 'string') {
157
+ throw new Error('No ws_url in response');
158
+ }
159
+ // Subscribe before connect so the first room_created is never missed.
160
+ // Await until Daily join completes — callers must not treat WS "open" as call ready.
161
+ let roomCreatedSub;
162
+ const roomJoined = new Promise((resolve, reject) => {
163
+ roomCreatedSub = this.wsClient.roomCreated$
164
+ .pipe(take(1), takeUntil(this.destroy$))
165
+ .subscribe((roomUrl) => __awaiter(this, void 0, void 0, function* () {
166
+ try {
167
+ yield this.onRoomCreated(roomUrl, micStream !== null && micStream !== void 0 ? micStream : undefined);
168
+ resolve();
169
+ }
170
+ catch (err) {
171
+ console.error('Daily join failed:', err);
172
+ reject(err);
173
+ }
174
+ }), (err) => reject(err));
175
+ });
176
+ try {
177
+ yield this.wsClient.connect(wsUrl);
178
+ yield roomJoined;
179
+ }
180
+ catch (e) {
181
+ roomCreatedSub === null || roomCreatedSub === void 0 ? void 0 : roomCreatedSub.unsubscribe();
182
+ throw e;
183
+ }
117
184
  }
118
- catch (e) {
185
+ catch (error) {
186
+ console.error('Error connecting voice agent:', error);
119
187
  this.callStateSubject.next('ended');
188
+ yield this.disconnect();
189
+ this.statusTextSubject.next('Connection failed');
190
+ throw error;
120
191
  }
121
192
  });
122
193
  }
123
- onRoomCreated(roomUrl) {
194
+ onRoomCreated(roomUrl, micStream) {
124
195
  return __awaiter(this, void 0, void 0, function* () {
125
- yield this.dailyClient.connect(roomUrl);
126
- // 🔴 Start MUTED
127
- this.dailyClient.setMuted(true);
128
- this.isMicMutedSubject.next(true);
129
- this.statusTextSubject.next('Listening to agent...');
130
- // Enable mic on FIRST bot speech
131
- let handled = false;
132
- this.dailyClient.speaking$.pipe(filter(Boolean), take(1)).subscribe(() => {
133
- if (handled)
134
- return;
135
- handled = true;
136
- console.log('[VoiceFlow] First bot response → enabling mic');
137
- this.dailyClient.setMuted(false);
138
- this.statusTextSubject.next('You can speak now');
139
- });
140
- // ⛑️ Fallback (if bot fails)
141
- setTimeout(() => {
142
- if (!handled) {
143
- console.warn('[VoiceFlow] Fallback → enabling mic');
144
- this.dailyClient.setMuted(false);
145
- this.statusTextSubject.next('You can speak now');
146
- }
147
- }, 8000);
148
- // rest same
149
- this.subscriptions.add(combineLatest([
196
+ yield this.dailyClient.connect(roomUrl, undefined, micStream);
197
+ this.callSubscriptions.unsubscribe();
198
+ this.callSubscriptions = new Subscription();
199
+ // Waveform: use local mic stream from Daily client
200
+ this.callSubscriptions.add(this.dailyClient.localStream$
201
+ .pipe(filter((s) => s != null), take(1))
202
+ .subscribe((stream) => {
203
+ this.audioAnalyzer.start(stream);
204
+ }));
205
+ this.callSubscriptions.add(this.dailyClient.userSpeaking$.subscribe((s) => this.isUserSpeakingSubject.next(s)));
206
+ this.callSubscriptions.add(combineLatest([
150
207
  this.dailyClient.speaking$,
151
208
  this.dailyClient.userSpeaking$,
152
209
  ]).subscribe(([bot, user]) => {
153
210
  const current = this.callStateSubject.value;
211
+ if (current === 'connecting' && !bot) {
212
+ return;
213
+ }
154
214
  if (current === 'connecting' && bot) {
155
215
  this.callStartTime = Date.now();
156
216
  this.startDurationTimer();
@@ -164,15 +224,21 @@ export class VoiceAgentService {
164
224
  this.callStateSubject.next('talking');
165
225
  }
166
226
  else {
167
- this.callStateSubject.next('connected');
227
+ // Between bot turns: stay on listening to avoid flicker via 'connected'
228
+ this.callStateSubject.next('listening');
168
229
  }
169
230
  }));
231
+ this.callSubscriptions.add(this.dailyClient.micMuted$.subscribe((muted) => this.isMicMutedSubject.next(muted)));
232
+ this.statusTextSubject.next('Connecting...');
170
233
  });
171
234
  }
172
235
  disconnect() {
173
236
  return __awaiter(this, void 0, void 0, function* () {
237
+ this.callSubscriptions.unsubscribe();
238
+ this.callSubscriptions = new Subscription();
174
239
  this.stopDurationTimer();
175
240
  this.audioAnalyzer.stop();
241
+ // Daily first, then WebSocket
176
242
  yield this.dailyClient.disconnect();
177
243
  this.wsClient.disconnect();
178
244
  this.callStateSubject.next('ended');
@@ -184,16 +250,22 @@ export class VoiceAgentService {
184
250
  this.dailyClient.setMuted(!current);
185
251
  }
186
252
  startDurationTimer() {
187
- this.durationInterval = setInterval(() => {
188
- const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);
189
- const m = Math.floor(elapsed / 60);
190
- const s = elapsed % 60;
191
- this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);
192
- }, 1000);
253
+ const updateDuration = () => {
254
+ if (this.callStartTime > 0) {
255
+ const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);
256
+ const minutes = Math.floor(elapsed / 60);
257
+ const seconds = elapsed % 60;
258
+ this.durationSubject.next(`${minutes}:${String(seconds).padStart(2, '0')}`);
259
+ }
260
+ };
261
+ updateDuration();
262
+ this.durationInterval = setInterval(updateDuration, 1000);
193
263
  }
194
264
  stopDurationTimer() {
195
- if (this.durationInterval)
265
+ if (this.durationInterval) {
196
266
  clearInterval(this.durationInterval);
267
+ this.durationInterval = null;
268
+ }
197
269
  }
198
270
  }
199
271
  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" });
@@ -209,4 +281,4 @@ VoiceAgentService.ctorParameters = () => [
209
281
  { type: PlatformTokenRefreshService },
210
282
  { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
211
283
  ];
212
- //# 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@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>(true);\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$ = this.callStateSubject.asObservable();\n  statusText$ = this.statusTextSubject.asObservable();\n  duration$ = this.durationSubject.asObservable();\n  isMicMuted$ = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$ = this.isUserSpeakingSubject.asObservable();\n  audioLevels$ = this.audioLevelsSubject.asObservable();\n  userTranscript$ = this.userTranscriptSubject.asObservable();\n  botTranscript$ = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private wsClient: WebSocketVoiceClientService,\n    private dailyClient: DailyVoiceClientService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    this.disconnect();\n  }\n\n  /**\n   * Tear down transports and reset UI state so a new `connect()` can run.\n   * `connect()` only proceeds from `idle`; use this after `ended` or when reopening the modal.\n   */\n  resetToIdle(): void {\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n    this.wsClient.disconnect();\n    void this.dailyClient.disconnect();\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('00:00');\n    this.isMicMutedSubject.next(true);\n    this.isUserSpeakingSubject.next(false);\n    this.audioLevelsSubject.next([]);\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') return;\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = token;\n\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\n          if (ensured?.accessToken) {\n            accessToken = ensured.accessToken;\n          }\n        } catch {}\n      }\n\n      const base = (apiUrl || '').replace(/\\/+$/, '');\n      const postUrl = `${base}/ai/ask-voice`;\n      // Same as chat-drawer `/ai/ask` headers: use `eventId` (camelCase), value from host `this.eventId`.\n      const eventIdHeader = (eventId && String(eventId).trim()) || '';\n\n      const res = await fetch(postUrl, {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json',\n          Authorization: `Bearer ${accessToken}`,\n          'domain-authority': domainAuthority,\n          eventtoken: eventToken,\n          eventurl: eventUrl,\n          'hive-bot-id': botId,\n          'x-api-key': apiKey,\n          eventId: eventIdHeader,\n        },\n        body: JSON.stringify({\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        }),\n      });\n\n      const json = await res.json();\n      const wsUrl = json?.rn_ws_url;\n\n      this.wsClient.roomCreated$\n        .pipe(take(1), takeUntil(this.destroy$))\n        .subscribe(async (roomUrl) => {\n          await this.onRoomCreated(roomUrl);\n        });\n\n      this.subscriptions.add(\n        this.wsClient.userTranscript$.subscribe((t) =>\n          this.userTranscriptSubject.next(t),\n        ),\n      );\n\n      this.subscriptions.add(\n        this.wsClient.botTranscript$.subscribe((t) =>\n          this.botTranscriptSubject.next(t),\n        ),\n      );\n\n      this.wsClient.connect(wsUrl);\n    } catch (e) {\n      this.callStateSubject.next('ended');\n    }\n  }\n\n  private async onRoomCreated(roomUrl: string): Promise<void> {\n    await this.dailyClient.connect(roomUrl);\n\n    // 🔴 Start MUTED\n    this.dailyClient.setMuted(true);\n    this.isMicMutedSubject.next(true);\n\n    this.statusTextSubject.next('Listening to agent...');\n\n    // ✅ Enable mic on FIRST bot speech\n    let handled = false;\n\n    this.dailyClient.speaking$.pipe(filter(Boolean), take(1)).subscribe(() => {\n      if (handled) return;\n      handled = true;\n\n      console.log('[VoiceFlow] First bot response → enabling mic');\n\n      this.dailyClient.setMuted(false);\n      this.statusTextSubject.next('You can speak now');\n    });\n\n    // ⛑️ Fallback (if bot fails)\n    setTimeout(() => {\n      if (!handled) {\n        console.warn('[VoiceFlow] Fallback → enabling mic');\n        this.dailyClient.setMuted(false);\n        this.statusTextSubject.next('You can speak now');\n      }\n    }, 8000);\n\n    // rest same\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\n        if (current === 'connecting' && bot) {\n          this.callStartTime = Date.now();\n          this.startDurationTimer();\n          this.callStateSubject.next('talking');\n          return;\n        }\n\n        if (user) {\n          this.callStateSubject.next('listening');\n        } else if (bot) {\n          this.callStateSubject.next('talking');\n        } else {\n          this.callStateSubject.next('connected');\n        }\n      }),\n    );\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.audioAnalyzer.stop();\n\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    this.durationInterval = setInterval(() => {\n      const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n      const m = Math.floor(elapsed / 60);\n      const s = elapsed % 60;\n      this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);\n    }, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) clearInterval(this.durationInterval);\n  }\n}\n"]}
284
+ //# 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  /** Per-call only; cleared on disconnect / reset / new room so handlers do not stack. */\n  private callSubscriptions = 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    // Transcripts: single subscription for service lifetime (avoid stacking on each connect()).\n    // WebSocket is disconnected between calls; no replay — new subscribers (setupVoiceTranscripts)\n    // only receive messages from the new WS after connect.\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\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.callSubscriptions.unsubscribe();\n    this.callSubscriptions = new Subscription();\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    existingMicStream?: MediaStream | null,\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      const tokenPromise =\n        usersApiUrl && isPlatformBrowser(this.platformId)\n          ? this.platformTokenRefresh\n              .ensureValidAccessToken(token, usersApiUrl)\n              .pipe(take(1))\n              .toPromise()\n              .then((ensured) => ensured?.accessToken ?? token)\n              .catch((e) => {\n                console.warn(\n                  '[HiveGpt Voice] Token refresh before connect failed',\n                  e,\n                );\n                return token;\n              })\n          : Promise.resolve(token);\n\n      const prepPromise = Promise.resolve().then(() => {\n        const baseUrl = apiUrl.replace(/\\/$/, '');\n        return {\n          postUrl: `${baseUrl}/ai/ask-voice`,\n          body: JSON.stringify({\n            bot_id: botId,\n            conversation_id: conversationId,\n            voice: 'alloy',\n          }),\n        };\n      });\n\n      const micPromise =\n        existingMicStream?.getAudioTracks().some((t) => t.readyState === 'live')\n          ? Promise.resolve(existingMicStream)\n          : isPlatformBrowser(this.platformId) &&\n              navigator.mediaDevices?.getUserMedia\n            ? navigator.mediaDevices\n                .getUserMedia({ audio: true })\n                .catch(() => undefined)\n            : Promise.resolve(undefined);\n\n      const [accessToken, { postUrl, body }, micStream] = await Promise.all([\n        tokenPromise,\n        prepPromise,\n        micPromise,\n      ]);\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,\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 before connect so the first room_created is never missed.\n      // Await until Daily join completes — callers must not treat WS \"open\" as call ready.\n      let roomCreatedSub: Subscription | undefined;\n      const roomJoined = new Promise<void>((resolve, reject) => {\n        roomCreatedSub = this.wsClient.roomCreated$\n          .pipe(take(1), takeUntil(this.destroy$))\n          .subscribe(\n            async (roomUrl) => {\n              try {\n                await this.onRoomCreated(roomUrl, micStream ?? undefined);\n                resolve();\n              } catch (err) {\n                console.error('Daily join failed:', err);\n                reject(err);\n              }\n            },\n            (err) => reject(err),\n          );\n      });\n\n      try {\n        await this.wsClient.connect(wsUrl);\n        await roomJoined;\n      } catch (e) {\n        roomCreatedSub?.unsubscribe();\n        throw e;\n      }\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(\n    roomUrl: string,\n    micStream?: MediaStream,\n  ): Promise<void> {\n    await this.dailyClient.connect(roomUrl, undefined, micStream);\n\n    this.callSubscriptions.unsubscribe();\n    this.callSubscriptions = new Subscription();\n\n    // Waveform: use local mic stream from Daily client\n    this.callSubscriptions.add(\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\n    this.callSubscriptions.add(\n      this.dailyClient.userSpeaking$.subscribe((s) =>\n        this.isUserSpeakingSubject.next(s),\n      ),\n    );\n    this.callSubscriptions.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 {\n          // Between bot turns: stay on listening to avoid flicker via 'connected'\n          this.callStateSubject.next('listening');\n        }\n      }),\n    );\n\n    this.callSubscriptions.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.callSubscriptions.unsubscribe();\n    this.callSubscriptions = new Subscription();\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"]}