@hivegpt/hiveai-angular 0.0.575 → 0.0.577

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 (20) hide show
  1. package/bundles/hivegpt-hiveai-angular.umd.js +195 -328
  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/voice-agent/components/voice-agent-modal/voice-agent-modal.component.js +54 -85
  6. package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +14 -22
  7. package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +45 -67
  8. package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +34 -79
  9. package/fesm2015/hivegpt-hiveai-angular.js +142 -247
  10. package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
  11. package/hivegpt-hiveai-angular.metadata.json +1 -1
  12. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +1 -7
  13. package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
  14. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +3 -5
  15. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +1 -1
  16. package/lib/components/voice-agent/services/voice-agent.service.d.ts +1 -3
  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 +12 -5
  19. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
  20. package/package.json +1 -1
@@ -43,8 +43,6 @@ export class VoiceAgentService {
43
43
  this.callStartTime = 0;
44
44
  this.durationInterval = null;
45
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();
48
46
  this.destroy$ = new Subject();
49
47
  this.callState$ = this.callStateSubject.asObservable();
50
48
  this.statusText$ = this.statusTextSubject.asObservable();
@@ -56,15 +54,6 @@ export class VoiceAgentService {
56
54
  this.botTranscript$ = this.botTranscriptSubject.asObservable();
57
55
  // Waveform visualization only - do NOT use for speaking state
58
56
  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)));
68
57
  }
69
58
  ngOnDestroy() {
70
59
  this.destroy$.next();
@@ -75,8 +64,6 @@ export class VoiceAgentService {
75
64
  resetToIdle() {
76
65
  if (this.callStateSubject.value === 'idle')
77
66
  return;
78
- this.callSubscriptions.unsubscribe();
79
- this.callSubscriptions = new Subscription();
80
67
  this.stopDurationTimer();
81
68
  this.audioAnalyzer.stop();
82
69
  this.wsClient.disconnect();
@@ -86,8 +73,7 @@ export class VoiceAgentService {
86
73
  this.statusTextSubject.next('');
87
74
  this.durationSubject.next('0:00');
88
75
  }
89
- connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl, existingMicStream) {
90
- var _a;
76
+ connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl) {
91
77
  return __awaiter(this, void 0, void 0, function* () {
92
78
  if (this.callStateSubject.value !== 'idle') {
93
79
  console.warn('Call already in progress');
@@ -96,41 +82,26 @@ export class VoiceAgentService {
96
82
  try {
97
83
  this.callStateSubject.next('connecting');
98
84
  this.statusTextSubject.next('Connecting...');
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) => {
85
+ 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
+ if (usersApiUrl && isPlatformBrowser(this.platformId)) {
90
+ try {
91
+ const ensured = yield this.platformTokenRefresh
92
+ .ensureValidAccessToken(token, usersApiUrl)
93
+ .pipe(take(1))
94
+ .toPromise();
95
+ if (ensured === null || ensured === void 0 ? void 0 : ensured.accessToken) {
96
+ accessToken = ensured.accessToken;
97
+ }
98
+ }
99
+ catch (e) {
106
100
  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
- ]);
101
+ }
102
+ }
103
+ const baseUrl = apiUrl.replace(/\/$/, '');
104
+ const postUrl = `${baseUrl}/ai/ask-voice`;
134
105
  const headers = {
135
106
  'Content-Type': 'application/json',
136
107
  Authorization: `Bearer ${accessToken}`,
@@ -146,7 +117,11 @@ export class VoiceAgentService {
146
117
  const res = yield fetch(postUrl, {
147
118
  method: 'POST',
148
119
  headers,
149
- body,
120
+ body: JSON.stringify({
121
+ bot_id: botId,
122
+ conversation_id: conversationId,
123
+ voice: 'alloy',
124
+ }),
150
125
  });
151
126
  if (!res.ok) {
152
127
  throw new Error(`HTTP ${res.status}`);
@@ -161,7 +136,7 @@ export class VoiceAgentService {
161
136
  .pipe(take(1), takeUntil(this.destroy$))
162
137
  .subscribe((roomUrl) => __awaiter(this, void 0, void 0, function* () {
163
138
  try {
164
- yield this.onRoomCreated(roomUrl, micStream !== null && micStream !== void 0 ? micStream : undefined);
139
+ yield this.onRoomCreated(roomUrl);
165
140
  }
166
141
  catch (err) {
167
142
  console.error('Daily join failed:', err);
@@ -171,8 +146,15 @@ export class VoiceAgentService {
171
146
  throw err;
172
147
  }
173
148
  }));
149
+ // Forward transcripts from WebSocket
150
+ this.subscriptions.add(this.wsClient.userTranscript$
151
+ .pipe(takeUntil(this.destroy$))
152
+ .subscribe((t) => this.userTranscriptSubject.next(t)));
153
+ this.subscriptions.add(this.wsClient.botTranscript$
154
+ .pipe(takeUntil(this.destroy$))
155
+ .subscribe((t) => this.botTranscriptSubject.next(t)));
174
156
  // Connect signaling WebSocket (no audio over WS)
175
- yield this.wsClient.connect(wsUrl);
157
+ this.wsClient.connect(wsUrl);
176
158
  }
177
159
  catch (error) {
178
160
  console.error('Error connecting voice agent:', error);
@@ -183,19 +165,18 @@ export class VoiceAgentService {
183
165
  }
184
166
  });
185
167
  }
186
- onRoomCreated(roomUrl, micStream) {
168
+ onRoomCreated(roomUrl) {
187
169
  return __awaiter(this, void 0, void 0, function* () {
188
- yield this.dailyClient.connect(roomUrl, undefined, micStream);
189
- this.callSubscriptions.unsubscribe();
190
- this.callSubscriptions = new Subscription();
170
+ // Connect Daily.js for WebRTC audio
171
+ yield this.dailyClient.connect(roomUrl);
191
172
  // Waveform: use local mic stream from Daily client
192
- this.callSubscriptions.add(this.dailyClient.localStream$
173
+ this.dailyClient.localStream$
193
174
  .pipe(filter((s) => s != null), take(1))
194
175
  .subscribe((stream) => {
195
176
  this.audioAnalyzer.start(stream);
196
- }));
197
- this.callSubscriptions.add(this.dailyClient.userSpeaking$.subscribe((s) => this.isUserSpeakingSubject.next(s)));
198
- this.callSubscriptions.add(combineLatest([
177
+ });
178
+ this.subscriptions.add(this.dailyClient.userSpeaking$.subscribe((s) => this.isUserSpeakingSubject.next(s)));
179
+ this.subscriptions.add(combineLatest([
199
180
  this.dailyClient.speaking$,
200
181
  this.dailyClient.userSpeaking$,
201
182
  ]).subscribe(([bot, user]) => {
@@ -215,19 +196,16 @@ export class VoiceAgentService {
215
196
  else if (bot) {
216
197
  this.callStateSubject.next('talking');
217
198
  }
218
- else {
219
- // Between bot turns: stay on listening to avoid flicker via 'connected'
220
- this.callStateSubject.next('listening');
199
+ else if (current === 'talking' || current === 'listening') {
200
+ this.callStateSubject.next('connected');
221
201
  }
222
202
  }));
223
- this.callSubscriptions.add(this.dailyClient.micMuted$.subscribe((muted) => this.isMicMutedSubject.next(muted)));
203
+ this.subscriptions.add(this.dailyClient.micMuted$.subscribe((muted) => this.isMicMutedSubject.next(muted)));
224
204
  this.statusTextSubject.next('Connecting...');
225
205
  });
226
206
  }
227
207
  disconnect() {
228
208
  return __awaiter(this, void 0, void 0, function* () {
229
- this.callSubscriptions.unsubscribe();
230
- this.callSubscriptions = new Subscription();
231
209
  this.stopDurationTimer();
232
210
  this.audioAnalyzer.stop();
233
211
  // Daily first, then WebSocket
@@ -273,4 +251,4 @@ VoiceAgentService.ctorParameters = () => [
273
251
  { type: PlatformTokenRefreshService },
274
252
  { type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
275
253
  ];
276
- //# 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 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, micStream ?? undefined);\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      // Connect signaling WebSocket (no audio over WS)\n      await 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(\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"]}
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"]}
@@ -11,7 +11,6 @@ import * as i0 from "@angular/core";
11
11
  * - Emit roomCreated$, userTranscript$, botTranscript$
12
12
  * - NO audio logic, NO mic logic. Audio is handled by Daily.js (WebRTC).
13
13
  */
14
- const WS_CONNECT_TIMEOUT_MS = 10000;
15
14
  export class WebSocketVoiceClientService {
16
15
  constructor() {
17
16
  this.ws = null;
@@ -25,98 +24,54 @@ export class WebSocketVoiceClientService {
25
24
  /** Emits bot transcript updates. */
26
25
  this.botTranscript$ = this.botTranscriptSubject.asObservable();
27
26
  }
28
- /**
29
- * Connect to signaling WebSocket. No audio over this connection.
30
- * Resolves when the socket is open; rejects if the connection fails.
31
- */
27
+ /** Connect to signaling WebSocket. No audio over this connection. */
32
28
  connect(wsUrl) {
33
29
  var _a;
34
30
  if (((_a = this.ws) === null || _a === void 0 ? void 0 : _a.readyState) === WebSocket.OPEN) {
35
- return Promise.resolve();
31
+ return;
36
32
  }
37
33
  if (this.ws) {
38
34
  this.ws.close();
39
35
  this.ws = null;
40
36
  }
41
- return new Promise((resolve, reject) => {
42
- let settled = false;
43
- const timeout = setTimeout(() => {
37
+ try {
38
+ this.ws = new WebSocket(wsUrl);
39
+ this.ws.onmessage = (event) => {
44
40
  var _a;
45
- if (settled)
46
- return;
47
- settled = true;
48
41
  try {
49
- (_a = this.ws) === null || _a === void 0 ? void 0 : _a.close();
50
- }
51
- catch (_b) {
52
- /* ignore */
53
- }
54
- this.ws = null;
55
- reject(new Error('WebSocket connection timed out'));
56
- }, WS_CONNECT_TIMEOUT_MS);
57
- const clear = () => {
58
- clearTimeout(timeout);
59
- };
60
- try {
61
- const ws = new WebSocket(wsUrl);
62
- this.ws = ws;
63
- ws.onopen = () => {
64
- if (settled)
65
- return;
66
- settled = true;
67
- clear();
68
- resolve();
69
- };
70
- ws.onmessage = (event) => {
71
- var _a;
72
- try {
73
- const msg = JSON.parse(event.data);
74
- if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'room_created') {
75
- const roomUrl = ((_a = msg.room_url) !== null && _a !== void 0 ? _a : msg.roomUrl);
76
- if (typeof roomUrl === 'string') {
77
- this.roomCreatedSubject.next(roomUrl);
78
- }
79
- }
80
- else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'user_transcript' && typeof msg.text === 'string') {
81
- this.userTranscriptSubject.next({
82
- text: msg.text,
83
- final: msg.final === true,
84
- });
85
- }
86
- else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'bot_transcript' && typeof msg.text === 'string') {
87
- this.botTranscriptSubject.next(msg.text);
42
+ const msg = JSON.parse(event.data);
43
+ if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'room_created') {
44
+ const roomUrl = ((_a = msg.room_url) !== null && _a !== void 0 ? _a : msg.roomUrl);
45
+ if (typeof roomUrl === 'string') {
46
+ this.roomCreatedSubject.next(roomUrl);
88
47
  }
89
48
  }
90
- catch (_b) {
91
- // Ignore non-JSON or unknown messages
49
+ else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'user_transcript' && typeof msg.text === 'string') {
50
+ this.userTranscriptSubject.next({
51
+ text: msg.text,
52
+ final: msg.final === true,
53
+ });
92
54
  }
93
- };
94
- ws.onerror = () => {
95
- if (!settled) {
96
- settled = true;
97
- clear();
98
- this.disconnect();
99
- reject(new Error('WebSocket connection failed'));
100
- return;
55
+ else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'bot_transcript' && typeof msg.text === 'string') {
56
+ this.botTranscriptSubject.next(msg.text);
101
57
  }
102
- this.disconnect();
103
- };
104
- ws.onclose = () => {
105
- this.ws = null;
106
- if (!settled) {
107
- settled = true;
108
- clear();
109
- reject(new Error('WebSocket connection failed'));
110
- }
111
- };
112
- }
113
- catch (err) {
114
- clear();
115
- console.error('WebSocketVoiceClient: connect failed', err);
58
+ }
59
+ catch (_b) {
60
+ // Ignore non-JSON or unknown messages
61
+ }
62
+ };
63
+ this.ws.onerror = () => {
64
+ this.disconnect();
65
+ };
66
+ this.ws.onclose = () => {
116
67
  this.ws = null;
117
- reject(err instanceof Error ? err : new Error(String(err)));
118
- }
119
- });
68
+ };
69
+ }
70
+ catch (err) {
71
+ console.error('WebSocketVoiceClient: connect failed', err);
72
+ this.ws = null;
73
+ throw err;
74
+ }
120
75
  }
121
76
  /** Disconnect and cleanup. */
122
77
  disconnect() {
@@ -137,4 +92,4 @@ WebSocketVoiceClientService.decorators = [
137
92
  providedIn: 'root',
138
93
  },] }
139
94
  ];
140
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/websocket-voice-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\n\n/** WebSocket message types from backend signaling. */\nexport interface WsMessageRoomCreated {\n  type: 'room_created';\n  room_url: string;\n}\n\nexport interface WsMessageUserTranscript {\n  type: 'user_transcript';\n  text: string;\n  final?: boolean;\n}\n\nexport interface WsMessageBotTranscript {\n  type: 'bot_transcript';\n  text: string;\n}\n\nexport type WsMessage =\n  | WsMessageRoomCreated\n  | WsMessageUserTranscript\n  | WsMessageBotTranscript;\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * WebSocket-only client for voice agent signaling.\n * CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.\n *\n * Responsibilities:\n * - Connect to ws_url (from POST /ai/ask-voice response)\n * - Parse JSON messages (room_created, user_transcript, bot_transcript)\n * - Emit roomCreated$, userTranscript$, botTranscript$\n * - NO audio logic, NO mic logic. Audio is handled by Daily.js (WebRTC).\n */\nconst WS_CONNECT_TIMEOUT_MS = 10_000;\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class WebSocketVoiceClientService {\n  private ws: WebSocket | null = null;\n  private roomCreatedSubject = new Subject<string>();\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  /** Emits room_url when backend sends room_created. */\n  roomCreated$: Observable<string> = this.roomCreatedSubject.asObservable();\n\n  /** Emits user transcript updates. */\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n\n  /** Emits bot transcript updates. */\n  botTranscript$: Observable<string> =\n    this.botTranscriptSubject.asObservable();\n\n  /**\n   * Connect to signaling WebSocket. No audio over this connection.\n   * Resolves when the socket is open; rejects if the connection fails.\n   */\n  connect(wsUrl: string): Promise<void> {\n    if (this.ws?.readyState === WebSocket.OPEN) {\n      return Promise.resolve();\n    }\n    if (this.ws) {\n      this.ws.close();\n      this.ws = null;\n    }\n\n    return new Promise((resolve, reject) => {\n      let settled = false;\n      const timeout = setTimeout(() => {\n        if (settled) return;\n        settled = true;\n        try {\n          this.ws?.close();\n        } catch {\n          /* ignore */\n        }\n        this.ws = null;\n        reject(new Error('WebSocket connection timed out'));\n      }, WS_CONNECT_TIMEOUT_MS);\n\n      const clear = (): void => {\n        clearTimeout(timeout);\n      };\n\n      try {\n        const ws = new WebSocket(wsUrl);\n        this.ws = ws;\n\n        ws.onopen = () => {\n          if (settled) return;\n          settled = true;\n          clear();\n          resolve();\n        };\n\n        ws.onmessage = (event: MessageEvent) => {\n          try {\n            const msg = JSON.parse(event.data) as Record<string, unknown>;\n            if (msg?.type === 'room_created') {\n              const roomUrl = (msg.room_url ?? msg.roomUrl) as string | undefined;\n              if (typeof roomUrl === 'string') {\n                this.roomCreatedSubject.next(roomUrl);\n              }\n            } else if (msg?.type === 'user_transcript' && typeof msg.text === 'string') {\n              this.userTranscriptSubject.next({\n                text: msg.text,\n                final: msg.final === true,\n              });\n            } else if (msg?.type === 'bot_transcript' && typeof msg.text === 'string') {\n              this.botTranscriptSubject.next(msg.text);\n            }\n          } catch {\n            // Ignore non-JSON or unknown messages\n          }\n        };\n\n        ws.onerror = () => {\n          if (!settled) {\n            settled = true;\n            clear();\n            this.disconnect();\n            reject(new Error('WebSocket connection failed'));\n            return;\n          }\n          this.disconnect();\n        };\n\n        ws.onclose = () => {\n          this.ws = null;\n          if (!settled) {\n            settled = true;\n            clear();\n            reject(new Error('WebSocket connection failed'));\n          }\n        };\n      } catch (err) {\n        clear();\n        console.error('WebSocketVoiceClient: connect failed', err);\n        this.ws = null;\n        reject(err instanceof Error ? err : new Error(String(err)));\n      }\n    });\n  }\n\n  /** Disconnect and cleanup. */\n  disconnect(): void {\n    if (this.ws) {\n      this.ws.close();\n      this.ws = null;\n    }\n  }\n\n  /** Whether the WebSocket is open. */\n  get isConnected(): boolean {\n    return this.ws?.readyState === WebSocket.OPEN;\n  }\n}\n"]}
95
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/websocket-voice-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\n\n/** WebSocket message types from backend signaling. */\nexport interface WsMessageRoomCreated {\n  type: 'room_created';\n  room_url: string;\n}\n\nexport interface WsMessageUserTranscript {\n  type: 'user_transcript';\n  text: string;\n  final?: boolean;\n}\n\nexport interface WsMessageBotTranscript {\n  type: 'bot_transcript';\n  text: string;\n}\n\nexport type WsMessage =\n  | WsMessageRoomCreated\n  | WsMessageUserTranscript\n  | WsMessageBotTranscript;\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * WebSocket-only client for voice agent signaling.\n * CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.\n *\n * Responsibilities:\n * - Connect to ws_url (from POST /ai/ask-voice response)\n * - Parse JSON messages (room_created, user_transcript, bot_transcript)\n * - Emit roomCreated$, userTranscript$, botTranscript$\n * - NO audio logic, NO mic logic. Audio is handled by Daily.js (WebRTC).\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class WebSocketVoiceClientService {\n  private ws: WebSocket | null = null;\n  private roomCreatedSubject = new Subject<string>();\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  /** Emits room_url when backend sends room_created. */\n  roomCreated$: Observable<string> = this.roomCreatedSubject.asObservable();\n\n  /** Emits user transcript updates. */\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n\n  /** Emits bot transcript updates. */\n  botTranscript$: Observable<string> =\n    this.botTranscriptSubject.asObservable();\n\n  /** Connect to signaling WebSocket. No audio over this connection. */\n  connect(wsUrl: string): void {\n    if (this.ws?.readyState === WebSocket.OPEN) {\n      return;\n    }\n    if (this.ws) {\n      this.ws.close();\n      this.ws = null;\n    }\n\n    try {\n      this.ws = new WebSocket(wsUrl);\n      this.ws.onmessage = (event: MessageEvent) => {\n        try {\n          const msg = JSON.parse(event.data) as Record<string, unknown>;\n          if (msg?.type === 'room_created') {\n            const roomUrl = (msg.room_url ?? msg.roomUrl) as string | undefined;\n            if (typeof roomUrl === 'string') {\n              this.roomCreatedSubject.next(roomUrl);\n            }\n          } else if (msg?.type === 'user_transcript' && typeof msg.text === 'string') {\n            this.userTranscriptSubject.next({\n              text: msg.text,\n              final: msg.final === true,\n            });\n          } else if (msg?.type === 'bot_transcript' && typeof msg.text === 'string') {\n            this.botTranscriptSubject.next(msg.text);\n          }\n        } catch {\n          // Ignore non-JSON or unknown messages\n        }\n      };\n      this.ws.onerror = () => {\n        this.disconnect();\n      };\n      this.ws.onclose = () => {\n        this.ws = null;\n      };\n    } catch (err) {\n      console.error('WebSocketVoiceClient: connect failed', err);\n      this.ws = null;\n      throw err;\n    }\n  }\n\n  /** Disconnect and cleanup. */\n  disconnect(): void {\n    if (this.ws) {\n      this.ws.close();\n      this.ws = null;\n    }\n  }\n\n  /** Whether the WebSocket is open. */\n  get isConnected(): boolean {\n    return this.ws?.readyState === WebSocket.OPEN;\n  }\n}\n"]}