@hivegpt/hiveai-angular 0.0.573 → 0.0.575
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/hivegpt-hiveai-angular.umd.js +526 -271
- package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
- package/esm2015/lib/components/chat-drawer/chat-drawer.component.js +6 -6
- package/esm2015/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.js +85 -54
- package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +153 -63
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +149 -85
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +79 -34
- package/fesm2015/hivegpt-hiveai-angular.js +465 -236
- package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
- package/hivegpt-hiveai-angular.metadata.json +1 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +7 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +37 -3
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +19 -6
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +5 -12
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
- 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,
|
|
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(
|
|
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,155 @@ 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('
|
|
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
|
-
this.statusTextSubject.next('Connecting
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
98
|
+
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) => {
|
|
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
|
-
|
|
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;
|
|
156
|
+
if (!wsUrl || typeof wsUrl !== 'string') {
|
|
157
|
+
throw new Error('No ws_url in response');
|
|
158
|
+
}
|
|
159
|
+
// Subscribe to room_created BEFORE connecting to avoid race
|
|
109
160
|
this.wsClient.roomCreated$
|
|
110
161
|
.pipe(take(1), takeUntil(this.destroy$))
|
|
111
162
|
.subscribe((roomUrl) => __awaiter(this, void 0, void 0, function* () {
|
|
112
|
-
|
|
163
|
+
try {
|
|
164
|
+
yield this.onRoomCreated(roomUrl, micStream !== null && micStream !== void 0 ? micStream : undefined);
|
|
165
|
+
}
|
|
166
|
+
catch (err) {
|
|
167
|
+
console.error('Daily join failed:', err);
|
|
168
|
+
this.callStateSubject.next('ended');
|
|
169
|
+
this.statusTextSubject.next('Connection failed');
|
|
170
|
+
yield this.disconnect();
|
|
171
|
+
throw err;
|
|
172
|
+
}
|
|
113
173
|
}));
|
|
114
|
-
|
|
115
|
-
this.
|
|
116
|
-
this.wsClient.connect(wsUrl);
|
|
174
|
+
// Connect signaling WebSocket (no audio over WS)
|
|
175
|
+
yield this.wsClient.connect(wsUrl);
|
|
117
176
|
}
|
|
118
|
-
catch (
|
|
177
|
+
catch (error) {
|
|
178
|
+
console.error('Error connecting voice agent:', error);
|
|
119
179
|
this.callStateSubject.next('ended');
|
|
180
|
+
yield this.disconnect();
|
|
181
|
+
this.statusTextSubject.next('Connection failed');
|
|
182
|
+
throw error;
|
|
120
183
|
}
|
|
121
184
|
});
|
|
122
185
|
}
|
|
123
|
-
onRoomCreated(roomUrl) {
|
|
186
|
+
onRoomCreated(roomUrl, micStream) {
|
|
124
187
|
return __awaiter(this, void 0, void 0, function* () {
|
|
125
|
-
yield this.dailyClient.connect(roomUrl);
|
|
126
|
-
|
|
127
|
-
this.
|
|
128
|
-
|
|
129
|
-
this.
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
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([
|
|
188
|
+
yield this.dailyClient.connect(roomUrl, undefined, micStream);
|
|
189
|
+
this.callSubscriptions.unsubscribe();
|
|
190
|
+
this.callSubscriptions = new Subscription();
|
|
191
|
+
// Waveform: use local mic stream from Daily client
|
|
192
|
+
this.callSubscriptions.add(this.dailyClient.localStream$
|
|
193
|
+
.pipe(filter((s) => s != null), take(1))
|
|
194
|
+
.subscribe((stream) => {
|
|
195
|
+
this.audioAnalyzer.start(stream);
|
|
196
|
+
}));
|
|
197
|
+
this.callSubscriptions.add(this.dailyClient.userSpeaking$.subscribe((s) => this.isUserSpeakingSubject.next(s)));
|
|
198
|
+
this.callSubscriptions.add(combineLatest([
|
|
150
199
|
this.dailyClient.speaking$,
|
|
151
200
|
this.dailyClient.userSpeaking$,
|
|
152
201
|
]).subscribe(([bot, user]) => {
|
|
153
202
|
const current = this.callStateSubject.value;
|
|
203
|
+
if (current === 'connecting' && !bot) {
|
|
204
|
+
return;
|
|
205
|
+
}
|
|
154
206
|
if (current === 'connecting' && bot) {
|
|
155
207
|
this.callStartTime = Date.now();
|
|
156
208
|
this.startDurationTimer();
|
|
@@ -164,15 +216,21 @@ export class VoiceAgentService {
|
|
|
164
216
|
this.callStateSubject.next('talking');
|
|
165
217
|
}
|
|
166
218
|
else {
|
|
167
|
-
|
|
219
|
+
// Between bot turns: stay on listening to avoid flicker via 'connected'
|
|
220
|
+
this.callStateSubject.next('listening');
|
|
168
221
|
}
|
|
169
222
|
}));
|
|
223
|
+
this.callSubscriptions.add(this.dailyClient.micMuted$.subscribe((muted) => this.isMicMutedSubject.next(muted)));
|
|
224
|
+
this.statusTextSubject.next('Connecting...');
|
|
170
225
|
});
|
|
171
226
|
}
|
|
172
227
|
disconnect() {
|
|
173
228
|
return __awaiter(this, void 0, void 0, function* () {
|
|
229
|
+
this.callSubscriptions.unsubscribe();
|
|
230
|
+
this.callSubscriptions = new Subscription();
|
|
174
231
|
this.stopDurationTimer();
|
|
175
232
|
this.audioAnalyzer.stop();
|
|
233
|
+
// Daily first, then WebSocket
|
|
176
234
|
yield this.dailyClient.disconnect();
|
|
177
235
|
this.wsClient.disconnect();
|
|
178
236
|
this.callStateSubject.next('ended');
|
|
@@ -184,16 +242,22 @@ export class VoiceAgentService {
|
|
|
184
242
|
this.dailyClient.setMuted(!current);
|
|
185
243
|
}
|
|
186
244
|
startDurationTimer() {
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
245
|
+
const updateDuration = () => {
|
|
246
|
+
if (this.callStartTime > 0) {
|
|
247
|
+
const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);
|
|
248
|
+
const minutes = Math.floor(elapsed / 60);
|
|
249
|
+
const seconds = elapsed % 60;
|
|
250
|
+
this.durationSubject.next(`${minutes}:${String(seconds).padStart(2, '0')}`);
|
|
251
|
+
}
|
|
252
|
+
};
|
|
253
|
+
updateDuration();
|
|
254
|
+
this.durationInterval = setInterval(updateDuration, 1000);
|
|
193
255
|
}
|
|
194
256
|
stopDurationTimer() {
|
|
195
|
-
if (this.durationInterval)
|
|
257
|
+
if (this.durationInterval) {
|
|
196
258
|
clearInterval(this.durationInterval);
|
|
259
|
+
this.durationInterval = null;
|
|
260
|
+
}
|
|
197
261
|
}
|
|
198
262
|
}
|
|
199
263
|
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 +273,4 @@ VoiceAgentService.ctorParameters = () => [
|
|
|
209
273
|
{ type: PlatformTokenRefreshService },
|
|
210
274
|
{ type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
|
|
211
275
|
];
|
|
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 to agent...');\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 = `https://1356-103-210-33-236.ngrok-free.app/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"]}
|
|
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"]}
|
|
@@ -11,6 +11,7 @@ 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;
|
|
14
15
|
export class WebSocketVoiceClientService {
|
|
15
16
|
constructor() {
|
|
16
17
|
this.ws = null;
|
|
@@ -24,54 +25,98 @@ export class WebSocketVoiceClientService {
|
|
|
24
25
|
/** Emits bot transcript updates. */
|
|
25
26
|
this.botTranscript$ = this.botTranscriptSubject.asObservable();
|
|
26
27
|
}
|
|
27
|
-
/**
|
|
28
|
+
/**
|
|
29
|
+
* Connect to signaling WebSocket. No audio over this connection.
|
|
30
|
+
* Resolves when the socket is open; rejects if the connection fails.
|
|
31
|
+
*/
|
|
28
32
|
connect(wsUrl) {
|
|
29
33
|
var _a;
|
|
30
34
|
if (((_a = this.ws) === null || _a === void 0 ? void 0 : _a.readyState) === WebSocket.OPEN) {
|
|
31
|
-
return;
|
|
35
|
+
return Promise.resolve();
|
|
32
36
|
}
|
|
33
37
|
if (this.ws) {
|
|
34
38
|
this.ws.close();
|
|
35
39
|
this.ws = null;
|
|
36
40
|
}
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
41
|
+
return new Promise((resolve, reject) => {
|
|
42
|
+
let settled = false;
|
|
43
|
+
const timeout = setTimeout(() => {
|
|
40
44
|
var _a;
|
|
45
|
+
if (settled)
|
|
46
|
+
return;
|
|
47
|
+
settled = true;
|
|
41
48
|
try {
|
|
42
|
-
|
|
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);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
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
|
-
});
|
|
54
|
-
}
|
|
55
|
-
else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'bot_transcript' && typeof msg.text === 'string') {
|
|
56
|
-
this.botTranscriptSubject.next(msg.text);
|
|
57
|
-
}
|
|
49
|
+
(_a = this.ws) === null || _a === void 0 ? void 0 : _a.close();
|
|
58
50
|
}
|
|
59
51
|
catch (_b) {
|
|
60
|
-
|
|
52
|
+
/* ignore */
|
|
61
53
|
}
|
|
62
|
-
};
|
|
63
|
-
this.ws.onerror = () => {
|
|
64
|
-
this.disconnect();
|
|
65
|
-
};
|
|
66
|
-
this.ws.onclose = () => {
|
|
67
54
|
this.ws = null;
|
|
55
|
+
reject(new Error('WebSocket connection timed out'));
|
|
56
|
+
}, WS_CONNECT_TIMEOUT_MS);
|
|
57
|
+
const clear = () => {
|
|
58
|
+
clearTimeout(timeout);
|
|
68
59
|
};
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
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);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
catch (_b) {
|
|
91
|
+
// Ignore non-JSON or unknown messages
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
ws.onerror = () => {
|
|
95
|
+
if (!settled) {
|
|
96
|
+
settled = true;
|
|
97
|
+
clear();
|
|
98
|
+
this.disconnect();
|
|
99
|
+
reject(new Error('WebSocket connection failed'));
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
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);
|
|
116
|
+
this.ws = null;
|
|
117
|
+
reject(err instanceof Error ? err : new Error(String(err)));
|
|
118
|
+
}
|
|
119
|
+
});
|
|
75
120
|
}
|
|
76
121
|
/** Disconnect and cleanup. */
|
|
77
122
|
disconnect() {
|
|
@@ -92,4 +137,4 @@ WebSocketVoiceClientService.decorators = [
|
|
|
92
137
|
providedIn: 'root',
|
|
93
138
|
},] }
|
|
94
139
|
];
|
|
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"]}
|
|
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"]}
|