@hivegpt/hiveai-angular 0.0.582 → 0.0.584
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 +209 -382
- package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
- package/esm2015/hivegpt-hiveai-angular.js +4 -5
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +174 -165
- package/esm2015/lib/components/voice-agent/voice-agent.module.js +3 -5
- package/fesm2015/hivegpt-hiveai-angular.js +173 -323
- package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
- package/hivegpt-hiveai-angular.d.ts +3 -4
- package/hivegpt-hiveai-angular.d.ts.map +1 -1
- package/hivegpt-hiveai-angular.metadata.json +1 -1
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +24 -15
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/lib/components/voice-agent/voice-agent.module.d.ts +2 -2
- package/lib/components/voice-agent/voice-agent.module.d.ts.map +1 -1
- package/package.json +1 -1
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +0 -163
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +0 -52
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +0 -1
|
@@ -3,12 +3,11 @@
|
|
|
3
3
|
*/
|
|
4
4
|
export * from './public-api';
|
|
5
5
|
export { NotificationSocket as ɵd } from './lib/components/NotificationSocket';
|
|
6
|
-
export { BotHtmlEditorComponent as
|
|
6
|
+
export { BotHtmlEditorComponent as ɵh } from './lib/components/bot-html-editor/bot-html-editor.component';
|
|
7
7
|
export { BotsService as ɵa } from './lib/components/bot.service';
|
|
8
8
|
export { ConversationService as ɵc } from './lib/components/conversation.service';
|
|
9
9
|
export { SocketService as ɵb } from './lib/components/socket-service.service';
|
|
10
10
|
export { TranslationService as ɵe } from './lib/components/translations/translation.service';
|
|
11
|
-
export { VideoPlayerComponent as
|
|
12
|
-
export {
|
|
13
|
-
|
|
14
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGl2ZWdwdC1oaXZlYWktYW5ndWxhci5qcyIsInNvdXJjZVJvb3QiOiIvVXNlcnMvcm9oaXR0aGFrdXIvaGl2ZS1ncHQvSGl2ZUFJLVBhY2thZ2VzL0FuZ3VsYXIvcHJvamVjdHMvaGl2ZWdwdC9ldmVudHNncHQtYW5ndWxhci9zcmMvIiwic291cmNlcyI6WyJoaXZlZ3B0LWhpdmVhaS1hbmd1bGFyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUM7QUFFN0IsT0FBTyxFQUFDLGtCQUFrQixJQUFJLEVBQUUsRUFBQyxNQUFNLHFDQUFxQyxDQUFDO0FBQzdFLE9BQU8sRUFBQyxzQkFBc0IsSUFBSSxFQUFFLEVBQUMsTUFBTSw0REFBNEQsQ0FBQztBQUN4RyxPQUFPLEVBQUMsV0FBVyxJQUFJLEVBQUUsRUFBQyxNQUFNLDhCQUE4QixDQUFDO0FBQy9ELE9BQU8sRUFBQyxtQkFBbUIsSUFBSSxFQUFFLEVBQUMsTUFBTSx1Q0FBdUMsQ0FBQztBQUNoRixPQUFPLEVBQUMsYUFBYSxJQUFJLEVBQUUsRUFBQyxNQUFNLHlDQUF5QyxDQUFDO0FBQzVFLE9BQU8sRUFBQyxrQkFBa0IsSUFBSSxFQUFFLEVBQUMsTUFBTSxtREFBbUQsQ0FBQztBQUMzRixPQUFPLEVBQUMsb0JBQW9CLElBQUksRUFBRSxFQUFDLE1BQU0sc0RBQXNELENBQUM7QUFDaEcsT0FBTyxFQUFDLDJCQUEyQixJQUFJLEVBQUUsRUFBQyxNQUFNLHNFQUFzRSxDQUFDO0FBQ3ZILE9BQU8sRUFBQyxZQUFZLElBQUksRUFBRSxFQUFDLE1BQU0sNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEdlbmVyYXRlZCBidW5kbGUgaW5kZXguIERvIG5vdCBlZGl0LlxuICovXG5cbmV4cG9ydCAqIGZyb20gJy4vcHVibGljLWFwaSc7XG5cbmV4cG9ydCB7Tm90aWZpY2F0aW9uU29ja2V0IGFzIMm1ZH0gZnJvbSAnLi9saWIvY29tcG9uZW50cy9Ob3RpZmljYXRpb25Tb2NrZXQnO1xuZXhwb3J0IHtCb3RIdG1sRWRpdG9yQ29tcG9uZW50IGFzIMm1aX0gZnJvbSAnLi9saWIvY29tcG9uZW50cy9ib3QtaHRtbC1lZGl0b3IvYm90LWh0bWwtZWRpdG9yLmNvbXBvbmVudCc7XG5leHBvcnQge0JvdHNTZXJ2aWNlIGFzIMm1YX0gZnJvbSAnLi9saWIvY29tcG9uZW50cy9ib3Quc2VydmljZSc7XG5leHBvcnQge0NvbnZlcnNhdGlvblNlcnZpY2UgYXMgybVjfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL2NvbnZlcnNhdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCB7U29ja2V0U2VydmljZSBhcyDJtWJ9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvc29ja2V0LXNlcnZpY2Uuc2VydmljZSc7XG5leHBvcnQge1RyYW5zbGF0aW9uU2VydmljZSBhcyDJtWV9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvdHJhbnNsYXRpb25zL3RyYW5zbGF0aW9uLnNlcnZpY2UnO1xuZXhwb3J0IHtWaWRlb1BsYXllckNvbXBvbmVudCBhcyDJtWd9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvdmlkZW8tcGxheWVyL3ZpZGVvLXBsYXllci5jb21wb25lbnQnO1xuZXhwb3J0IHtXZWJTb2NrZXRWb2ljZUNsaWVudFNlcnZpY2UgYXMgybVmfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3NlcnZpY2VzL3dlYnNvY2tldC12b2ljZS1jbGllbnQuc2VydmljZSc7XG5leHBvcnQge1NhZmVIdG1sUGlwZSBhcyDJtWh9IGZyb20gJy4vbGliL3BpcGVzL3NhZmUtaHRtbC5waXBlJzsiXX0=
|
|
11
|
+
export { VideoPlayerComponent as ɵf } from './lib/components/video-player/video-player.component';
|
|
12
|
+
export { SafeHtmlPipe as ɵg } from './lib/pipes/safe-html.pipe';
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGl2ZWdwdC1oaXZlYWktYW5ndWxhci5qcyIsInNvdXJjZVJvb3QiOiIvVXNlcnMvcm9oaXR0aGFrdXIvaGl2ZS1ncHQvSGl2ZUFJLVBhY2thZ2VzL0FuZ3VsYXIvcHJvamVjdHMvaGl2ZWdwdC9ldmVudHNncHQtYW5ndWxhci9zcmMvIiwic291cmNlcyI6WyJoaXZlZ3B0LWhpdmVhaS1hbmd1bGFyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBRUgsY0FBYyxjQUFjLENBQUM7QUFFN0IsT0FBTyxFQUFDLGtCQUFrQixJQUFJLEVBQUUsRUFBQyxNQUFNLHFDQUFxQyxDQUFDO0FBQzdFLE9BQU8sRUFBQyxzQkFBc0IsSUFBSSxFQUFFLEVBQUMsTUFBTSw0REFBNEQsQ0FBQztBQUN4RyxPQUFPLEVBQUMsV0FBVyxJQUFJLEVBQUUsRUFBQyxNQUFNLDhCQUE4QixDQUFDO0FBQy9ELE9BQU8sRUFBQyxtQkFBbUIsSUFBSSxFQUFFLEVBQUMsTUFBTSx1Q0FBdUMsQ0FBQztBQUNoRixPQUFPLEVBQUMsYUFBYSxJQUFJLEVBQUUsRUFBQyxNQUFNLHlDQUF5QyxDQUFDO0FBQzVFLE9BQU8sRUFBQyxrQkFBa0IsSUFBSSxFQUFFLEVBQUMsTUFBTSxtREFBbUQsQ0FBQztBQUMzRixPQUFPLEVBQUMsb0JBQW9CLElBQUksRUFBRSxFQUFDLE1BQU0sc0RBQXNELENBQUM7QUFDaEcsT0FBTyxFQUFDLFlBQVksSUFBSSxFQUFFLEVBQUMsTUFBTSw0QkFBNEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcblxuZXhwb3J0IHtOb3RpZmljYXRpb25Tb2NrZXQgYXMgybVkfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL05vdGlmaWNhdGlvblNvY2tldCc7XG5leHBvcnQge0JvdEh0bWxFZGl0b3JDb21wb25lbnQgYXMgybVofSBmcm9tICcuL2xpYi9jb21wb25lbnRzL2JvdC1odG1sLWVkaXRvci9ib3QtaHRtbC1lZGl0b3IuY29tcG9uZW50JztcbmV4cG9ydCB7Qm90c1NlcnZpY2UgYXMgybVhfSBmcm9tICcuL2xpYi9jb21wb25lbnRzL2JvdC5zZXJ2aWNlJztcbmV4cG9ydCB7Q29udmVyc2F0aW9uU2VydmljZSBhcyDJtWN9IGZyb20gJy4vbGliL2NvbXBvbmVudHMvY29udmVyc2F0aW9uLnNlcnZpY2UnO1xuZXhwb3J0IHtTb2NrZXRTZXJ2aWNlIGFzIMm1Yn0gZnJvbSAnLi9saWIvY29tcG9uZW50cy9zb2NrZXQtc2VydmljZS5zZXJ2aWNlJztcbmV4cG9ydCB7VHJhbnNsYXRpb25TZXJ2aWNlIGFzIMm1ZX0gZnJvbSAnLi9saWIvY29tcG9uZW50cy90cmFuc2xhdGlvbnMvdHJhbnNsYXRpb24uc2VydmljZSc7XG5leHBvcnQge1ZpZGVvUGxheWVyQ29tcG9uZW50IGFzIMm1Zn0gZnJvbSAnLi9saWIvY29tcG9uZW50cy92aWRlby1wbGF5ZXIvdmlkZW8tcGxheWVyLmNvbXBvbmVudCc7XG5leHBvcnQge1NhZmVIdG1sUGlwZSBhcyDJtWd9IGZyb20gJy4vbGliL3BpcGVzL3NhZmUtaHRtbC5waXBlJzsiXX0=
|
|
@@ -1,27 +1,33 @@
|
|
|
1
1
|
import { __awaiter } from "tslib";
|
|
2
2
|
import { isPlatformBrowser } from '@angular/common';
|
|
3
|
-
import { Inject, Injectable, PLATFORM_ID } from '@angular/core';
|
|
4
|
-
import { BehaviorSubject,
|
|
5
|
-
import {
|
|
3
|
+
import { Inject, Injectable, NgZone, PLATFORM_ID } from '@angular/core';
|
|
4
|
+
import { BehaviorSubject, Subject, Subscription } from 'rxjs';
|
|
5
|
+
import { take } from 'rxjs/operators';
|
|
6
|
+
import { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';
|
|
7
|
+
import { WebSocketTransport } from '@pipecat-ai/websocket-transport';
|
|
6
8
|
import { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';
|
|
7
9
|
import { AudioAnalyzerService } from './audio-analyzer.service';
|
|
8
|
-
import { WebSocketVoiceClientService } from './websocket-voice-client.service';
|
|
9
10
|
import * as i0 from "@angular/core";
|
|
10
11
|
import * as i1 from "./audio-analyzer.service";
|
|
11
|
-
import * as i2 from "
|
|
12
|
-
import * as i3 from "../../../services/platform-token-refresh.service";
|
|
12
|
+
import * as i2 from "../../../services/platform-token-refresh.service";
|
|
13
13
|
/**
|
|
14
|
-
* Voice agent orchestrator
|
|
15
|
-
*
|
|
16
|
-
*
|
|
14
|
+
* Voice agent orchestrator using the official PipecatClient SDK.
|
|
15
|
+
*
|
|
16
|
+
* Audio flow (mirrors the React reference implementation):
|
|
17
|
+
* - Local mic: acquired by PipecatClient.initDevices(); local track fed to
|
|
18
|
+
* AudioAnalyzerService for waveform visualisation.
|
|
19
|
+
* - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,
|
|
20
|
+
* played through a hidden <audio> element.
|
|
21
|
+
* - All binary protobuf framing / RTVI protocol handled by
|
|
22
|
+
* @pipecat-ai/client-js + @pipecat-ai/websocket-transport.
|
|
17
23
|
*/
|
|
18
24
|
export class VoiceAgentService {
|
|
19
|
-
constructor(audioAnalyzer,
|
|
25
|
+
constructor(audioAnalyzer, platformTokenRefresh, ngZone,
|
|
20
26
|
/** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */
|
|
21
27
|
platformId) {
|
|
22
28
|
this.audioAnalyzer = audioAnalyzer;
|
|
23
|
-
this.wsClient = wsClient;
|
|
24
29
|
this.platformTokenRefresh = platformTokenRefresh;
|
|
30
|
+
this.ngZone = ngZone;
|
|
25
31
|
this.platformId = platformId;
|
|
26
32
|
this.callStateSubject = new BehaviorSubject('idle');
|
|
27
33
|
this.statusTextSubject = new BehaviorSubject('');
|
|
@@ -33,8 +39,8 @@ export class VoiceAgentService {
|
|
|
33
39
|
this.botTranscriptSubject = new Subject();
|
|
34
40
|
this.callStartTime = 0;
|
|
35
41
|
this.durationInterval = null;
|
|
36
|
-
this.
|
|
37
|
-
this.
|
|
42
|
+
this.pcClient = null;
|
|
43
|
+
this.botAudioElement = null;
|
|
38
44
|
this.subscriptions = new Subscription();
|
|
39
45
|
this.destroy$ = new Subject();
|
|
40
46
|
this.callState$ = this.callStateSubject.asObservable();
|
|
@@ -46,25 +52,17 @@ export class VoiceAgentService {
|
|
|
46
52
|
this.userTranscript$ = this.userTranscriptSubject.asObservable();
|
|
47
53
|
this.botTranscript$ = this.botTranscriptSubject.asObservable();
|
|
48
54
|
this.subscriptions.add(this.audioAnalyzer.audioLevels$.subscribe((levels) => this.audioLevelsSubject.next(levels)));
|
|
49
|
-
this.subscriptions.add(this.wsClient.remoteClose$
|
|
50
|
-
.pipe(takeUntil(this.destroy$))
|
|
51
|
-
.subscribe(() => void this.handleRemoteClose()));
|
|
52
55
|
}
|
|
53
56
|
ngOnDestroy() {
|
|
54
57
|
this.destroy$.next();
|
|
55
58
|
this.subscriptions.unsubscribe();
|
|
56
|
-
this.disconnect();
|
|
59
|
+
void this.disconnect();
|
|
57
60
|
}
|
|
58
|
-
/** Reset to idle
|
|
61
|
+
/** Reset to idle (e.g. when modal re-opens so user can click Start Call). */
|
|
59
62
|
resetToIdle() {
|
|
60
63
|
if (this.callStateSubject.value === 'idle')
|
|
61
64
|
return;
|
|
62
|
-
this.
|
|
63
|
-
this.stopDurationTimer();
|
|
64
|
-
this.callStartTime = 0;
|
|
65
|
-
this.audioAnalyzer.stop();
|
|
66
|
-
this.stopLocalMic();
|
|
67
|
-
this.wsClient.disconnect();
|
|
65
|
+
void this.disconnect();
|
|
68
66
|
this.callStateSubject.next('idle');
|
|
69
67
|
this.statusTextSubject.next('');
|
|
70
68
|
this.durationSubject.next('0:00');
|
|
@@ -72,7 +70,7 @@ export class VoiceAgentService {
|
|
|
72
70
|
connect(apiUrl, token, botId, conversationId, apiKey, eventToken, eventId, eventUrl, domainAuthority, usersApiUrl) {
|
|
73
71
|
return __awaiter(this, void 0, void 0, function* () {
|
|
74
72
|
if (this.callStateSubject.value !== 'idle') {
|
|
75
|
-
console.warn('Call already in progress');
|
|
73
|
+
console.warn('[HiveGpt Voice] Call already in progress');
|
|
76
74
|
return;
|
|
77
75
|
}
|
|
78
76
|
try {
|
|
@@ -85,185 +83,196 @@ export class VoiceAgentService {
|
|
|
85
83
|
.ensureValidAccessToken(token, usersApiUrl)
|
|
86
84
|
.pipe(take(1))
|
|
87
85
|
.toPromise();
|
|
88
|
-
if (ensured === null || ensured === void 0 ? void 0 : ensured.accessToken)
|
|
86
|
+
if (ensured === null || ensured === void 0 ? void 0 : ensured.accessToken)
|
|
89
87
|
accessToken = ensured.accessToken;
|
|
90
|
-
}
|
|
91
88
|
}
|
|
92
89
|
catch (e) {
|
|
93
|
-
console.warn('[HiveGpt Voice] Token refresh
|
|
90
|
+
console.warn('[HiveGpt Voice] Token refresh failed', e);
|
|
94
91
|
}
|
|
95
92
|
}
|
|
96
93
|
const baseUrl = apiUrl.replace(/\/$/, '');
|
|
97
|
-
const
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
94
|
+
const pcClient = new PipecatClient({
|
|
95
|
+
transport: new WebSocketTransport(),
|
|
96
|
+
enableMic: true,
|
|
97
|
+
enableCam: false,
|
|
98
|
+
callbacks: {
|
|
99
|
+
onConnected: () => this.ngZone.run(() => this.onPipecatConnected()),
|
|
100
|
+
onDisconnected: () => this.ngZone.run(() => this.onPipecatDisconnected()),
|
|
101
|
+
onBotReady: () => this.ngZone.run(() => this.onBotReady()),
|
|
102
|
+
onUserTranscript: (data) => this.ngZone.run(() => this.userTranscriptSubject.next({ text: data.text, final: !!data.final })),
|
|
103
|
+
onBotTranscript: (data) => this.ngZone.run(() => this.botTranscriptSubject.next(data.text)),
|
|
104
|
+
onError: (err) => {
|
|
105
|
+
this.ngZone.run(() => {
|
|
106
|
+
console.error('[HiveGpt Voice] PipecatClient error', err);
|
|
107
|
+
this.callStateSubject.next('ended');
|
|
108
|
+
this.statusTextSubject.next('Connection failed');
|
|
109
|
+
});
|
|
110
|
+
},
|
|
111
|
+
},
|
|
112
|
+
});
|
|
113
|
+
this.pcClient = pcClient;
|
|
114
|
+
// Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element
|
|
115
|
+
pcClient.on(RTVIEvent.TrackStarted, (track, participant) => {
|
|
116
|
+
if (!(participant === null || participant === void 0 ? void 0 : participant.local) && track.kind === 'audio') {
|
|
117
|
+
this.ngZone.run(() => this.setupBotAudioTrack(track));
|
|
118
|
+
}
|
|
119
|
+
});
|
|
120
|
+
// Speaking state comes straight from RTVI events
|
|
121
|
+
pcClient.on(RTVIEvent.BotStartedSpeaking, () => this.ngZone.run(() => this.onBotStartedSpeaking()));
|
|
122
|
+
pcClient.on(RTVIEvent.BotStoppedSpeaking, () => this.ngZone.run(() => this.onBotStoppedSpeaking()));
|
|
123
|
+
pcClient.on(RTVIEvent.UserStartedSpeaking, () => this.ngZone.run(() => {
|
|
124
|
+
this.isUserSpeakingSubject.next(true);
|
|
125
|
+
this.callStateSubject.next('listening');
|
|
126
|
+
}));
|
|
127
|
+
pcClient.on(RTVIEvent.UserStoppedSpeaking, () => this.ngZone.run(() => {
|
|
128
|
+
this.isUserSpeakingSubject.next(false);
|
|
129
|
+
if (this.callStateSubject.value === 'listening') {
|
|
130
|
+
this.callStateSubject.next('connected');
|
|
131
|
+
}
|
|
132
|
+
}));
|
|
133
|
+
// Acquire mic (triggers browser permission prompt)
|
|
134
|
+
yield pcClient.initDevices();
|
|
135
|
+
// Build headers using the browser Headers API (required by pipecat's APIRequest type)
|
|
136
|
+
const requestHeaders = new Headers();
|
|
137
|
+
requestHeaders.append('Authorization', `Bearer ${accessToken}`);
|
|
138
|
+
requestHeaders.append('x-api-key', apiKey);
|
|
139
|
+
requestHeaders.append('hive-bot-id', botId);
|
|
140
|
+
requestHeaders.append('domain-authority', domainAuthority);
|
|
141
|
+
requestHeaders.append('eventUrl', eventUrl);
|
|
142
|
+
requestHeaders.append('eventId', eventId);
|
|
143
|
+
requestHeaders.append('eventToken', eventToken);
|
|
144
|
+
requestHeaders.append('ngrok-skip-browser-warning', 'true');
|
|
145
|
+
// POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects
|
|
146
|
+
yield pcClient.startBotAndConnect({
|
|
147
|
+
endpoint: `${baseUrl}/ai/ask-voice-socket`,
|
|
148
|
+
headers: requestHeaders,
|
|
149
|
+
requestData: {
|
|
113
150
|
bot_id: botId,
|
|
114
151
|
conversation_id: conversationId,
|
|
115
152
|
voice: 'alloy',
|
|
116
|
-
}
|
|
153
|
+
},
|
|
117
154
|
});
|
|
118
|
-
if (!res.ok) {
|
|
119
|
-
throw new Error(`HTTP ${res.status}`);
|
|
120
|
-
}
|
|
121
|
-
const json = yield res.json();
|
|
122
|
-
const wsUrl = (typeof (json === null || json === void 0 ? void 0 : json.ws_url) === 'string' && json.ws_url) ||
|
|
123
|
-
(typeof (json === null || json === void 0 ? void 0 : json.rn_ws_url) === 'string' && json.rn_ws_url);
|
|
124
|
-
if (!wsUrl) {
|
|
125
|
-
throw new Error('No ws_url in response');
|
|
126
|
-
}
|
|
127
|
-
const untilCallEnds$ = merge(this.destroy$, this.endCall$);
|
|
128
|
-
this.subscriptions.add(this.wsClient.userTranscript$
|
|
129
|
-
.pipe(takeUntil(untilCallEnds$))
|
|
130
|
-
.subscribe((t) => this.userTranscriptSubject.next(t)));
|
|
131
|
-
this.subscriptions.add(this.wsClient.botTranscript$
|
|
132
|
-
.pipe(takeUntil(untilCallEnds$))
|
|
133
|
-
.subscribe((t) => this.botTranscriptSubject.next(t)));
|
|
134
|
-
this.subscriptions.add(this.wsClient.opened$
|
|
135
|
-
.pipe(takeUntil(untilCallEnds$), take(1))
|
|
136
|
-
.subscribe(() => void this.onWebsocketOpened()));
|
|
137
|
-
this.wsClient.connect(wsUrl);
|
|
138
155
|
}
|
|
139
156
|
catch (error) {
|
|
140
|
-
console.error('
|
|
157
|
+
console.error('[HiveGpt Voice] connect failed', error);
|
|
141
158
|
this.callStateSubject.next('ended');
|
|
142
|
-
yield this.
|
|
159
|
+
yield this.cleanupPipecatClient();
|
|
143
160
|
this.statusTextSubject.next('Connection failed');
|
|
144
161
|
throw error;
|
|
145
162
|
}
|
|
146
163
|
});
|
|
147
164
|
}
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
try {
|
|
154
|
-
yield this.startLocalMic();
|
|
155
|
-
this.statusTextSubject.next('Connected');
|
|
156
|
-
this.callStateSubject.next('connected');
|
|
157
|
-
this.wireSpeakingState();
|
|
158
|
-
}
|
|
159
|
-
catch (err) {
|
|
160
|
-
console.error('[HiveGpt Voice] Mic or session setup failed', err);
|
|
161
|
-
this.callStateSubject.next('ended');
|
|
162
|
-
this.statusTextSubject.next('Microphone unavailable');
|
|
163
|
-
yield this.disconnect();
|
|
164
|
-
}
|
|
165
|
-
});
|
|
165
|
+
onPipecatConnected() {
|
|
166
|
+
this.callStateSubject.next('connected');
|
|
167
|
+
this.statusTextSubject.next('Connected');
|
|
168
|
+
this.isMicMutedSubject.next(false);
|
|
169
|
+
this.startLocalMicAnalyzer();
|
|
166
170
|
}
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
]).pipe(map(([local, server]) => local || server), distinctUntilChanged(), startWith(false));
|
|
175
|
-
this.subscriptions.add(combineLatest([assistantTalking$, userTalking$])
|
|
176
|
-
.pipe(takeUntil(untilCallEnds$))
|
|
177
|
-
.subscribe(([bot, user]) => {
|
|
178
|
-
const current = this.callStateSubject.value;
|
|
179
|
-
if (user) {
|
|
180
|
-
this.isUserSpeakingSubject.next(true);
|
|
181
|
-
this.callStateSubject.next('listening');
|
|
182
|
-
}
|
|
183
|
-
else {
|
|
184
|
-
this.isUserSpeakingSubject.next(false);
|
|
185
|
-
}
|
|
186
|
-
if (user) {
|
|
187
|
-
return;
|
|
188
|
-
}
|
|
189
|
-
if (bot) {
|
|
190
|
-
if (this.callStartTime === 0) {
|
|
191
|
-
this.callStartTime = Date.now();
|
|
192
|
-
this.startDurationTimer();
|
|
193
|
-
}
|
|
194
|
-
this.callStateSubject.next('talking');
|
|
195
|
-
}
|
|
196
|
-
else if (current === 'talking' || current === 'listening') {
|
|
197
|
-
this.callStateSubject.next('connected');
|
|
198
|
-
}
|
|
199
|
-
}));
|
|
171
|
+
onPipecatDisconnected() {
|
|
172
|
+
this.stopDurationTimer();
|
|
173
|
+
this.callStartTime = 0;
|
|
174
|
+
this.audioAnalyzer.stop();
|
|
175
|
+
this.stopBotAudio();
|
|
176
|
+
this.callStateSubject.next('ended');
|
|
177
|
+
this.statusTextSubject.next('Call Ended');
|
|
200
178
|
}
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
throw new Error('No audio track');
|
|
209
|
-
}
|
|
210
|
-
this.localMicStream = stream;
|
|
211
|
-
this.isMicMutedSubject.next(!track.enabled);
|
|
212
|
-
this.audioAnalyzer.start(stream);
|
|
213
|
-
});
|
|
179
|
+
onBotReady() {
|
|
180
|
+
var _a, _b, _c;
|
|
181
|
+
// Retry track wiring in case tracks weren't ready at onConnected
|
|
182
|
+
this.startLocalMicAnalyzer();
|
|
183
|
+
const botTrack = (_c = (_b = (_a = this.pcClient) === null || _a === void 0 ? void 0 : _a.tracks()) === null || _b === void 0 ? void 0 : _b.bot) === null || _c === void 0 ? void 0 : _c.audio;
|
|
184
|
+
if (botTrack)
|
|
185
|
+
this.setupBotAudioTrack(botTrack);
|
|
214
186
|
}
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
187
|
+
startLocalMicAnalyzer() {
|
|
188
|
+
var _a, _b, _c;
|
|
189
|
+
const localTrack = (_c = (_b = (_a = this.pcClient) === null || _a === void 0 ? void 0 : _a.tracks()) === null || _b === void 0 ? void 0 : _b.local) === null || _c === void 0 ? void 0 : _c.audio;
|
|
190
|
+
if (localTrack) {
|
|
191
|
+
this.audioAnalyzer.start(new MediaStream([localTrack]));
|
|
219
192
|
}
|
|
220
193
|
}
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
this.
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
194
|
+
onBotStartedSpeaking() {
|
|
195
|
+
if (this.callStartTime === 0) {
|
|
196
|
+
this.callStartTime = Date.now();
|
|
197
|
+
this.startDurationTimer();
|
|
198
|
+
}
|
|
199
|
+
this.callStateSubject.next('talking');
|
|
200
|
+
}
|
|
201
|
+
onBotStoppedSpeaking() {
|
|
202
|
+
if (this.callStateSubject.value === 'talking') {
|
|
203
|
+
this.callStateSubject.next('connected');
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
setupBotAudioTrack(track) {
|
|
207
|
+
var _a;
|
|
208
|
+
if (!this.botAudioElement) {
|
|
209
|
+
this.botAudioElement = new Audio();
|
|
210
|
+
this.botAudioElement.autoplay = true;
|
|
211
|
+
}
|
|
212
|
+
const existing = (_a = this.botAudioElement.srcObject) === null || _a === void 0 ? void 0 : _a.getAudioTracks()[0];
|
|
213
|
+
if ((existing === null || existing === void 0 ? void 0 : existing.id) === track.id)
|
|
214
|
+
return;
|
|
215
|
+
this.botAudioElement.srcObject = new MediaStream([track]);
|
|
216
|
+
this.botAudioElement.play().catch((err) => console.warn('[HiveGpt Voice] Bot audio play blocked', err));
|
|
217
|
+
}
|
|
218
|
+
stopBotAudio() {
|
|
219
|
+
var _a;
|
|
220
|
+
if (this.botAudioElement) {
|
|
221
|
+
try {
|
|
222
|
+
this.botAudioElement.pause();
|
|
223
|
+
(_a = this.botAudioElement.srcObject) === null || _a === void 0 ? void 0 : _a.getAudioTracks().forEach((t) => t.stop());
|
|
224
|
+
this.botAudioElement.srcObject = null;
|
|
225
|
+
}
|
|
226
|
+
catch (_b) {
|
|
227
|
+
// ignore
|
|
228
|
+
}
|
|
229
|
+
this.botAudioElement = null;
|
|
230
|
+
}
|
|
234
231
|
}
|
|
235
232
|
disconnect() {
|
|
236
233
|
return __awaiter(this, void 0, void 0, function* () {
|
|
237
|
-
this.endCall$.next();
|
|
238
234
|
this.stopDurationTimer();
|
|
239
235
|
this.callStartTime = 0;
|
|
240
236
|
this.audioAnalyzer.stop();
|
|
241
|
-
this.
|
|
242
|
-
this.
|
|
237
|
+
this.stopBotAudio();
|
|
238
|
+
yield this.cleanupPipecatClient();
|
|
243
239
|
this.callStateSubject.next('ended');
|
|
244
240
|
this.statusTextSubject.next('Call Ended');
|
|
245
241
|
});
|
|
246
242
|
}
|
|
243
|
+
cleanupPipecatClient() {
|
|
244
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
245
|
+
if (this.pcClient) {
|
|
246
|
+
try {
|
|
247
|
+
yield this.pcClient.disconnect();
|
|
248
|
+
}
|
|
249
|
+
catch (_a) {
|
|
250
|
+
// ignore
|
|
251
|
+
}
|
|
252
|
+
this.pcClient = null;
|
|
253
|
+
}
|
|
254
|
+
});
|
|
255
|
+
}
|
|
247
256
|
toggleMic() {
|
|
248
|
-
|
|
257
|
+
if (!this.pcClient)
|
|
258
|
+
return;
|
|
249
259
|
const nextMuted = !this.isMicMutedSubject.value;
|
|
250
|
-
|
|
251
|
-
if (track) {
|
|
252
|
-
track.enabled = !nextMuted;
|
|
253
|
-
}
|
|
260
|
+
this.pcClient.enableMic(!nextMuted);
|
|
254
261
|
this.isMicMutedSubject.next(nextMuted);
|
|
262
|
+
if (nextMuted)
|
|
263
|
+
this.isUserSpeakingSubject.next(false);
|
|
255
264
|
}
|
|
256
265
|
startDurationTimer() {
|
|
257
|
-
const
|
|
266
|
+
const tick = () => {
|
|
258
267
|
if (this.callStartTime > 0) {
|
|
259
268
|
const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);
|
|
260
|
-
const
|
|
261
|
-
const
|
|
262
|
-
this.durationSubject.next(`${
|
|
269
|
+
const m = Math.floor(elapsed / 60);
|
|
270
|
+
const s = elapsed % 60;
|
|
271
|
+
this.durationSubject.next(`${m}:${String(s).padStart(2, '0')}`);
|
|
263
272
|
}
|
|
264
273
|
};
|
|
265
|
-
|
|
266
|
-
this.durationInterval = setInterval(
|
|
274
|
+
tick();
|
|
275
|
+
this.durationInterval = setInterval(tick, 1000);
|
|
267
276
|
}
|
|
268
277
|
stopDurationTimer() {
|
|
269
278
|
if (this.durationInterval) {
|
|
@@ -272,7 +281,7 @@ export class VoiceAgentService {
|
|
|
272
281
|
}
|
|
273
282
|
}
|
|
274
283
|
}
|
|
275
|
-
VoiceAgentService.ɵprov = i0.ɵɵdefineInjectable({ factory: function VoiceAgentService_Factory() { return new VoiceAgentService(i0.ɵɵinject(i1.AudioAnalyzerService), i0.ɵɵinject(i2.
|
|
284
|
+
VoiceAgentService.ɵprov = i0.ɵɵdefineInjectable({ factory: function VoiceAgentService_Factory() { return new VoiceAgentService(i0.ɵɵinject(i1.AudioAnalyzerService), i0.ɵɵinject(i2.PlatformTokenRefreshService), i0.ɵɵinject(i0.NgZone), i0.ɵɵinject(i0.PLATFORM_ID)); }, token: VoiceAgentService, providedIn: "root" });
|
|
276
285
|
VoiceAgentService.decorators = [
|
|
277
286
|
{ type: Injectable, args: [{
|
|
278
287
|
providedIn: 'root',
|
|
@@ -280,8 +289,8 @@ VoiceAgentService.decorators = [
|
|
|
280
289
|
];
|
|
281
290
|
VoiceAgentService.ctorParameters = () => [
|
|
282
291
|
{ type: AudioAnalyzerService },
|
|
283
|
-
{ type: WebSocketVoiceClientService },
|
|
284
292
|
{ type: PlatformTokenRefreshService },
|
|
293
|
+
{ type: NgZone },
|
|
285
294
|
{ type: Object, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
|
|
286
295
|
];
|
|
287
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-agent.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/voice-agent.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { isPlatformBrowser } from '@angular/common';\nimport { Inject, Injectable, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  concat,\n  merge,\n  Observable,\n  of,\n  Subject,\n  Subscription,\n  timer,\n} from 'rxjs';\nimport {\n  distinctUntilChanged,\n  map,\n  startWith,\n  switchMap,\n  take,\n  takeUntil,\n} from 'rxjs/operators';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.service';\nimport { WebSocketVoiceClientService } from './websocket-voice-client.service';\n\nexport type CallState =\n  | 'idle'\n  | 'connecting'\n  | 'connected'\n  | 'listening'\n  | 'talking'\n  | 'ended';\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Voice agent orchestrator: single WebSocket (`ws_url` from POST /ai/ask-voice-socket)\n * for session events, transcripts, and optional speaking hints; local mic for capture\n * and waveform only (no Daily/WebRTC room).\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class VoiceAgentService implements OnDestroy {\n  private callStateSubject = new BehaviorSubject<CallState>('idle');\n  private statusTextSubject = new BehaviorSubject<string>('');\n  private durationSubject = new BehaviorSubject<string>('00:00');\n  private isMicMutedSubject = new BehaviorSubject<boolean>(false);\n  private isUserSpeakingSubject = new BehaviorSubject<boolean>(false);\n  private audioLevelsSubject = new BehaviorSubject<number[]>([]);\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  private callStartTime = 0;\n  private durationInterval: ReturnType<typeof setInterval> | null = null;\n\n  private localMicStream: MediaStream | null = null;\n  private readonly endCall$ = new Subject<void>();\n\n  private subscriptions = new Subscription();\n  private destroy$ = new Subject<void>();\n\n  callState$: Observable<CallState> = this.callStateSubject.asObservable();\n  statusText$: Observable<string> = this.statusTextSubject.asObservable();\n  duration$: Observable<string> = this.durationSubject.asObservable();\n  isMicMuted$: Observable<boolean> = this.isMicMutedSubject.asObservable();\n  isUserSpeaking$: Observable<boolean> =\n    this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private wsClient: WebSocketVoiceClientService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n    this.subscriptions.add(\n      this.wsClient.remoteClose$\n        .pipe(takeUntil(this.destroy$))\n        .subscribe(() => void this.handleRemoteClose()),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    this.disconnect();\n  }\n\n  /** Reset to idle state (e.g. when modal opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.wsClient.disconnect();\n    this.callStateSubject.next('idle');\n    this.statusTextSubject.next('');\n    this.durationSubject.next('0:00');\n  }\n\n  async connect(\n    apiUrl: string,\n    token: string,\n    botId: string,\n    conversationId: string,\n    apiKey: string,\n    eventToken: string,\n    eventId: string,\n    eventUrl: string,\n    domainAuthority: string,\n    usersApiUrl?: string,\n  ): Promise<void> {\n    if (this.callStateSubject.value !== 'idle') {\n      console.warn('Call already in progress');\n      return;\n    }\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = token;\n      if (usersApiUrl && isPlatformBrowser(this.platformId)) {\n        try {\n          const ensured = await this.platformTokenRefresh\n            .ensureValidAccessToken(token, usersApiUrl)\n            .pipe(take(1))\n            .toPromise();\n          if (ensured?.accessToken) {\n            accessToken = ensured.accessToken;\n          }\n        } catch (e) {\n          console.warn(\n            '[HiveGpt Voice] Token refresh before connect failed',\n            e,\n          );\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n      const postUrl = `${baseUrl}/ai/ask-voice-socket`;\n\n      const headers: Record<string, string> = {\n        'Content-Type': 'application/json',\n        Authorization: `Bearer ${accessToken}`,\n        'x-api-key': apiKey,\n        'hive-bot-id': botId,\n        'domain-authority': domainAuthority,\n        eventUrl,\n        eventId,\n        eventToken,\n        'ngrok-skip-browser-warning': 'true',\n      };\n\n      const res = await fetch(postUrl, {\n        method: 'POST',\n        headers,\n        body: JSON.stringify({\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        }),\n      });\n\n      if (!res.ok) {\n        throw new Error(`HTTP ${res.status}`);\n      }\n\n      const json = await res.json();\n      const wsUrl =\n        (typeof json?.ws_url === 'string' && json.ws_url) ||\n        (typeof json?.rn_ws_url === 'string' && json.rn_ws_url);\n      if (!wsUrl) {\n        throw new Error('No ws_url in response');\n      }\n\n      const untilCallEnds$ = merge(this.destroy$, this.endCall$);\n\n      this.subscriptions.add(\n        this.wsClient.userTranscript$\n          .pipe(takeUntil(untilCallEnds$))\n          .subscribe((t) => this.userTranscriptSubject.next(t)),\n      );\n      this.subscriptions.add(\n        this.wsClient.botTranscript$\n          .pipe(takeUntil(untilCallEnds$))\n          .subscribe((t) => this.botTranscriptSubject.next(t)),\n      );\n\n      this.subscriptions.add(\n        this.wsClient.opened$\n          .pipe(takeUntil(untilCallEnds$), take(1))\n          .subscribe(() => void this.onWebsocketOpened()),\n      );\n\n      this.wsClient.connect(wsUrl);\n    } catch (error) {\n      console.error('Error connecting voice agent:', error);\n      this.callStateSubject.next('ended');\n      await this.disconnect();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private async onWebsocketOpened(): Promise<void> {\n    if (this.callStateSubject.value !== 'connecting') {\n      return;\n    }\n    try {\n      await this.startLocalMic();\n      this.statusTextSubject.next('Connected');\n      this.callStateSubject.next('connected');\n      this.wireSpeakingState();\n    } catch (err) {\n      console.error('[HiveGpt Voice] Mic or session setup failed', err);\n      this.callStateSubject.next('ended');\n      this.statusTextSubject.next('Microphone unavailable');\n      await this.disconnect();\n    }\n  }\n\n  private wireSpeakingState(): void {\n    const untilCallEnds$ = merge(this.destroy$, this.endCall$);\n\n    const transcriptDrivenAssistant$ = this.wsClient.botTranscript$.pipe(\n      switchMap(() => concat(of(true), timer(800).pipe(map(() => false)))),\n      distinctUntilChanged(),\n    );\n\n    const assistantTalking$ = merge(\n      this.wsClient.assistantSpeaking$,\n      transcriptDrivenAssistant$,\n    ).pipe(distinctUntilChanged(), startWith(false));\n\n    const userTalking$ = combineLatest([\n      this.audioAnalyzer.isUserSpeaking$,\n      this.wsClient.serverUserSpeaking$.pipe(startWith(false)),\n    ]).pipe(\n      map(([local, server]) => local || server),\n      distinctUntilChanged(),\n      startWith(false),\n    );\n\n    this.subscriptions.add(\n      combineLatest([assistantTalking$, userTalking$])\n        .pipe(takeUntil(untilCallEnds$))\n        .subscribe(([bot, user]) => {\n          const current = this.callStateSubject.value;\n          if (user) {\n            this.isUserSpeakingSubject.next(true);\n            this.callStateSubject.next('listening');\n          } else {\n            this.isUserSpeakingSubject.next(false);\n          }\n          if (user) {\n            return;\n          }\n          if (bot) {\n            if (this.callStartTime === 0) {\n              this.callStartTime = Date.now();\n              this.startDurationTimer();\n            }\n            this.callStateSubject.next('talking');\n          } else if (current === 'talking' || current === 'listening') {\n            this.callStateSubject.next('connected');\n          }\n        }),\n    );\n  }\n\n  private async startLocalMic(): Promise<void> {\n    this.stopLocalMic();\n    const stream = await navigator.mediaDevices.getUserMedia({ audio: true });\n    const track = stream.getAudioTracks()[0];\n    if (!track) {\n      stream.getTracks().forEach((t) => t.stop());\n      throw new Error('No audio track');\n    }\n    this.localMicStream = stream;\n    this.isMicMutedSubject.next(!track.enabled);\n    this.audioAnalyzer.start(stream);\n  }\n\n  private stopLocalMic(): void {\n    if (this.localMicStream) {\n      this.localMicStream.getTracks().forEach((t) => t.stop());\n      this.localMicStream = null;\n    }\n  }\n\n  private async handleRemoteClose(): Promise<void> {\n    const state = this.callStateSubject.value;\n    if (state === 'idle' || state === 'ended') return;\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Connection lost');\n  }\n\n  async disconnect(): Promise<void> {\n    this.endCall$.next();\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopLocalMic();\n    this.wsClient.disconnect();\n\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  toggleMic(): void {\n    const nextMuted = !this.isMicMutedSubject.value;\n    const track = this.localMicStream?.getAudioTracks()[0];\n    if (track) {\n      track.enabled = !nextMuted;\n    }\n    this.isMicMutedSubject.next(nextMuted);\n  }\n\n  private startDurationTimer(): void {\n    const updateDuration = () => {\n      if (this.callStartTime > 0) {\n        const elapsed = Math.floor((Date.now() - this.callStartTime) / 1000);\n        const minutes = Math.floor(elapsed / 60);\n        const seconds = elapsed % 60;\n        this.durationSubject.next(\n          `${minutes}:${String(seconds).padStart(2, '0')}`,\n        );\n      }\n    };\n    updateDuration();\n    this.durationInterval = setInterval(updateDuration, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
|
|
296
|
+
//# 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, NgZone, OnDestroy, PLATFORM_ID } from '@angular/core';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { take } from 'rxjs/operators';\nimport { PipecatClient, RTVIEvent } from '@pipecat-ai/client-js';\nimport { WebSocketTransport } from '@pipecat-ai/websocket-transport';\nimport { PlatformTokenRefreshService } from '../../../services/platform-token-refresh.service';\nimport { AudioAnalyzerService } from './audio-analyzer.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 using the official PipecatClient SDK.\n *\n * Audio flow (mirrors the React reference implementation):\n *  - Local mic: acquired by PipecatClient.initDevices(); local track fed to\n *    AudioAnalyzerService for waveform visualisation.\n *  - Bot audio: received as a MediaStreamTrack via RTVIEvent.TrackStarted,\n *    played through a hidden <audio> element.\n *  - All binary protobuf framing / RTVI protocol handled by\n *    @pipecat-ai/client-js + @pipecat-ai/websocket-transport.\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 pcClient: PipecatClient | null = null;\n  private botAudioElement: HTMLAudioElement | 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> = this.isUserSpeakingSubject.asObservable();\n  audioLevels$: Observable<number[]> = this.audioLevelsSubject.asObservable();\n  userTranscript$: Observable<TranscriptData> = this.userTranscriptSubject.asObservable();\n  botTranscript$: Observable<string> = this.botTranscriptSubject.asObservable();\n\n  constructor(\n    private audioAnalyzer: AudioAnalyzerService,\n    private platformTokenRefresh: PlatformTokenRefreshService,\n    private ngZone: NgZone,\n    /** `Object` not `object` — ngc metadata collection rejects the `object` type in DI params. */\n    @Inject(PLATFORM_ID) private platformId: Object,\n  ) {\n    this.subscriptions.add(\n      this.audioAnalyzer.audioLevels$.subscribe((levels) =>\n        this.audioLevelsSubject.next(levels),\n      ),\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.subscriptions.unsubscribe();\n    void this.disconnect();\n  }\n\n  /** Reset to idle (e.g. when modal re-opens so user can click Start Call). */\n  resetToIdle(): void {\n    if (this.callStateSubject.value === 'idle') return;\n    void this.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('[HiveGpt Voice] Call already in progress');\n      return;\n    }\n\n    try {\n      this.callStateSubject.next('connecting');\n      this.statusTextSubject.next('Connecting...');\n\n      let accessToken = token;\n      if (usersApiUrl && isPlatformBrowser(this.platformId)) {\n        try {\n          const ensured = await this.platformTokenRefresh\n            .ensureValidAccessToken(token, usersApiUrl)\n            .pipe(take(1))\n            .toPromise();\n          if (ensured?.accessToken) accessToken = ensured.accessToken;\n        } catch (e) {\n          console.warn('[HiveGpt Voice] Token refresh failed', e);\n        }\n      }\n\n      const baseUrl = apiUrl.replace(/\\/$/, '');\n\n      const pcClient = new PipecatClient({\n        transport: new WebSocketTransport(),\n        enableMic: true,\n        enableCam: false,\n        callbacks: {\n          onConnected: () => this.ngZone.run(() => this.onPipecatConnected()),\n          onDisconnected: () => this.ngZone.run(() => this.onPipecatDisconnected()),\n          onBotReady: () => this.ngZone.run(() => this.onBotReady()),\n          onUserTranscript: (data) =>\n            this.ngZone.run(() =>\n              this.userTranscriptSubject.next({ text: data.text, final: !!data.final }),\n            ),\n          onBotTranscript: (data) =>\n            this.ngZone.run(() => this.botTranscriptSubject.next(data.text)),\n          onError: (err) => {\n            this.ngZone.run(() => {\n              console.error('[HiveGpt Voice] PipecatClient error', err);\n              this.callStateSubject.next('ended');\n              this.statusTextSubject.next('Connection failed');\n            });\n          },\n        },\n      });\n\n      this.pcClient = pcClient;\n\n      // Bot audio arrives as a MediaStreamTrack — wire to a hidden <audio> element\n      pcClient.on(\n        RTVIEvent.TrackStarted,\n        (track: MediaStreamTrack, participant?: { local?: boolean }) => {\n          if (!participant?.local && track.kind === 'audio') {\n            this.ngZone.run(() => this.setupBotAudioTrack(track));\n          }\n        },\n      );\n\n      // Speaking state comes straight from RTVI events\n      pcClient.on(RTVIEvent.BotStartedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStartedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.BotStoppedSpeaking, () =>\n        this.ngZone.run(() => this.onBotStoppedSpeaking()),\n      );\n      pcClient.on(RTVIEvent.UserStartedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(true);\n          this.callStateSubject.next('listening');\n        }),\n      );\n      pcClient.on(RTVIEvent.UserStoppedSpeaking, () =>\n        this.ngZone.run(() => {\n          this.isUserSpeakingSubject.next(false);\n          if (this.callStateSubject.value === 'listening') {\n            this.callStateSubject.next('connected');\n          }\n        }),\n      );\n\n      // Acquire mic (triggers browser permission prompt)\n      await pcClient.initDevices();\n\n      // Build headers using the browser Headers API (required by pipecat's APIRequest type)\n      const requestHeaders = new Headers();\n      requestHeaders.append('Authorization', `Bearer ${accessToken}`);\n      requestHeaders.append('x-api-key', apiKey);\n      requestHeaders.append('hive-bot-id', botId);\n      requestHeaders.append('domain-authority', domainAuthority);\n      requestHeaders.append('eventUrl', eventUrl);\n      requestHeaders.append('eventId', eventId);\n      requestHeaders.append('eventToken', eventToken);\n      requestHeaders.append('ngrok-skip-browser-warning', 'true');\n\n      // POST to /ai/ask-voice-socket → receives { ws_url } → WebSocketTransport connects\n      await pcClient.startBotAndConnect({\n        endpoint: `${baseUrl}/ai/ask-voice-socket`,\n        headers: requestHeaders,\n        requestData: {\n          bot_id: botId,\n          conversation_id: conversationId,\n          voice: 'alloy',\n        },\n      });\n    } catch (error) {\n      console.error('[HiveGpt Voice] connect failed', error);\n      this.callStateSubject.next('ended');\n      await this.cleanupPipecatClient();\n      this.statusTextSubject.next('Connection failed');\n      throw error;\n    }\n  }\n\n  private onPipecatConnected(): void {\n    this.callStateSubject.next('connected');\n    this.statusTextSubject.next('Connected');\n    this.isMicMutedSubject.next(false);\n    this.startLocalMicAnalyzer();\n  }\n\n  private onPipecatDisconnected(): void {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private onBotReady(): void {\n    // Retry track wiring in case tracks weren't ready at onConnected\n    this.startLocalMicAnalyzer();\n    const botTrack = (this.pcClient?.tracks() as { bot?: { audio?: MediaStreamTrack } })\n      ?.bot?.audio;\n    if (botTrack) this.setupBotAudioTrack(botTrack);\n  }\n\n  private startLocalMicAnalyzer(): void {\n    const localTrack = (this.pcClient?.tracks() as { local?: { audio?: MediaStreamTrack } })\n      ?.local?.audio;\n    if (localTrack) {\n      this.audioAnalyzer.start(new MediaStream([localTrack]));\n    }\n  }\n\n  private onBotStartedSpeaking(): void {\n    if (this.callStartTime === 0) {\n      this.callStartTime = Date.now();\n      this.startDurationTimer();\n    }\n    this.callStateSubject.next('talking');\n  }\n\n  private onBotStoppedSpeaking(): void {\n    if (this.callStateSubject.value === 'talking') {\n      this.callStateSubject.next('connected');\n    }\n  }\n\n  private setupBotAudioTrack(track: MediaStreamTrack): void {\n    if (!this.botAudioElement) {\n      this.botAudioElement = new Audio();\n      this.botAudioElement.autoplay = true;\n    }\n    const existing = (this.botAudioElement.srcObject as MediaStream | null)\n      ?.getAudioTracks()[0];\n    if (existing?.id === track.id) return;\n\n    this.botAudioElement.srcObject = new MediaStream([track]);\n    this.botAudioElement.play().catch((err) =>\n      console.warn('[HiveGpt Voice] Bot audio play blocked', err),\n    );\n  }\n\n  private stopBotAudio(): void {\n    if (this.botAudioElement) {\n      try {\n        this.botAudioElement.pause();\n        (this.botAudioElement.srcObject as MediaStream | null)\n          ?.getAudioTracks()\n          .forEach((t) => t.stop());\n        this.botAudioElement.srcObject = null;\n      } catch {\n        // ignore\n      }\n      this.botAudioElement = null;\n    }\n  }\n\n  async disconnect(): Promise<void> {\n    this.stopDurationTimer();\n    this.callStartTime = 0;\n    this.audioAnalyzer.stop();\n    this.stopBotAudio();\n    await this.cleanupPipecatClient();\n    this.callStateSubject.next('ended');\n    this.statusTextSubject.next('Call Ended');\n  }\n\n  private async cleanupPipecatClient(): Promise<void> {\n    if (this.pcClient) {\n      try {\n        await this.pcClient.disconnect();\n      } catch {\n        // ignore\n      }\n      this.pcClient = null;\n    }\n  }\n\n  toggleMic(): void {\n    if (!this.pcClient) return;\n    const nextMuted = !this.isMicMutedSubject.value;\n    this.pcClient.enableMic(!nextMuted);\n    this.isMicMutedSubject.next(nextMuted);\n    if (nextMuted) this.isUserSpeakingSubject.next(false);\n  }\n\n  private startDurationTimer(): void {\n    const tick = () => {\n      if (this.callStartTime > 0) {\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      }\n    };\n    tick();\n    this.durationInterval = setInterval(tick, 1000);\n  }\n\n  private stopDurationTimer(): void {\n    if (this.durationInterval) {\n      clearInterval(this.durationInterval);\n      this.durationInterval = null;\n    }\n  }\n}\n"]}
|