@hivegpt/hiveai-angular 0.0.581 → 0.0.583
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/hivegpt-hiveai-angular.umd.js +420 -490
- 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/audio-analyzer.service.js +3 -3
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +195 -83
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +160 -49
- package/esm2015/lib/components/voice-agent/voice-agent.module.js +3 -5
- package/fesm2015/hivegpt-hiveai-angular.js +338 -416
- 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/audio-analyzer.service.d.ts +2 -2
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +22 -13
- 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 +30 -20
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
- package/lib/components/voice-agent/voice-agent.module.d.ts +1 -1
- 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/daily-voice-client.service.js +0 -305
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +0 -62
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +0 -1
|
@@ -1,71 +1,79 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { __awaiter } from "tslib";
|
|
2
|
+
import { Injectable, NgZone } from '@angular/core';
|
|
2
3
|
import { Subject } from 'rxjs';
|
|
3
4
|
import * as i0 from "@angular/core";
|
|
4
5
|
/**
|
|
5
|
-
* WebSocket
|
|
6
|
+
* Native WebSocket client for voice session (signaling, transcripts, speaking hints).
|
|
6
7
|
* CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.
|
|
7
8
|
*
|
|
8
|
-
*
|
|
9
|
-
*
|
|
10
|
-
* - Parse JSON messages (room_created, user_transcript, bot_transcript)
|
|
11
|
-
* - Emit roomCreated$, userTranscript$, botTranscript$
|
|
12
|
-
* - NO audio logic, NO mic logic. Audio is handled by Daily.js (WebRTC).
|
|
9
|
+
* Connects to `ws_url` from `POST {baseUrl}/ai/ask-voice-socket`.
|
|
10
|
+
* Parses JSON messages for transcripts and optional assistant/user speaking flags.
|
|
13
11
|
*/
|
|
14
12
|
export class WebSocketVoiceClientService {
|
|
15
|
-
constructor() {
|
|
13
|
+
constructor(ngZone) {
|
|
14
|
+
this.ngZone = ngZone;
|
|
16
15
|
this.ws = null;
|
|
17
|
-
|
|
16
|
+
/** True when {@link disconnect} initiated the close (not counted as remote close). */
|
|
17
|
+
this.closeInitiatedByClient = false;
|
|
18
|
+
this.openedSubject = new Subject();
|
|
19
|
+
this.remoteCloseSubject = new Subject();
|
|
18
20
|
this.userTranscriptSubject = new Subject();
|
|
19
21
|
this.botTranscriptSubject = new Subject();
|
|
20
|
-
|
|
21
|
-
this.
|
|
22
|
-
|
|
22
|
+
this.assistantSpeakingSubject = new Subject();
|
|
23
|
+
this.serverUserSpeakingSubject = new Subject();
|
|
24
|
+
this.audioChunkSubject = new Subject();
|
|
25
|
+
/** Fires once each time the WebSocket reaches OPEN. */
|
|
26
|
+
this.opened$ = this.openedSubject.asObservable();
|
|
27
|
+
/** Fires when the socket closes without a client-initiated {@link disconnect}. */
|
|
28
|
+
this.remoteClose$ = this.remoteCloseSubject.asObservable();
|
|
23
29
|
this.userTranscript$ = this.userTranscriptSubject.asObservable();
|
|
24
|
-
/** Emits bot transcript updates. */
|
|
25
30
|
this.botTranscript$ = this.botTranscriptSubject.asObservable();
|
|
31
|
+
/** Assistant/bot speaking, when the server sends explicit events (see {@link handleJsonMessage}). */
|
|
32
|
+
this.assistantSpeaking$ = this.assistantSpeakingSubject.asObservable();
|
|
33
|
+
/** User speaking from server-side VAD, if provided. */
|
|
34
|
+
this.serverUserSpeaking$ = this.serverUserSpeakingSubject.asObservable();
|
|
35
|
+
/** Binary audio frames from server (when backend streams bot audio over WS). */
|
|
36
|
+
this.audioChunk$ = this.audioChunkSubject.asObservable();
|
|
26
37
|
}
|
|
27
|
-
/** Connect to signaling WebSocket. No audio over this connection. */
|
|
28
38
|
connect(wsUrl) {
|
|
29
39
|
var _a;
|
|
30
40
|
if (((_a = this.ws) === null || _a === void 0 ? void 0 : _a.readyState) === WebSocket.OPEN) {
|
|
31
41
|
return;
|
|
32
42
|
}
|
|
33
43
|
if (this.ws) {
|
|
44
|
+
this.closeInitiatedByClient = true;
|
|
34
45
|
this.ws.close();
|
|
35
|
-
this.ws = null;
|
|
36
46
|
}
|
|
37
47
|
try {
|
|
38
|
-
|
|
39
|
-
this.ws
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
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);
|
|
48
|
+
const socket = new WebSocket(wsUrl);
|
|
49
|
+
this.ws = socket;
|
|
50
|
+
socket.onopen = () => {
|
|
51
|
+
if (this.ws !== socket)
|
|
52
|
+
return;
|
|
53
|
+
this.ngZone.run(() => this.openedSubject.next());
|
|
54
|
+
};
|
|
55
|
+
socket.onmessage = (event) => {
|
|
56
|
+
if (this.ws !== socket)
|
|
57
|
+
return;
|
|
58
|
+
void this.handleIncomingMessage(event.data);
|
|
59
|
+
};
|
|
60
|
+
socket.onerror = () => {
|
|
61
|
+
this.ngZone.run(() => {
|
|
62
|
+
if (this.ws === socket && socket.readyState !== WebSocket.CLOSED) {
|
|
63
|
+
socket.close();
|
|
57
64
|
}
|
|
65
|
+
});
|
|
66
|
+
};
|
|
67
|
+
socket.onclose = () => {
|
|
68
|
+
if (this.ws === socket) {
|
|
69
|
+
this.ws = null;
|
|
58
70
|
}
|
|
59
|
-
|
|
60
|
-
|
|
71
|
+
const client = this.closeInitiatedByClient;
|
|
72
|
+
this.closeInitiatedByClient = false;
|
|
73
|
+
if (!client) {
|
|
74
|
+
this.ngZone.run(() => this.remoteCloseSubject.next());
|
|
61
75
|
}
|
|
62
76
|
};
|
|
63
|
-
this.ws.onerror = () => {
|
|
64
|
-
this.disconnect();
|
|
65
|
-
};
|
|
66
|
-
this.ws.onclose = () => {
|
|
67
|
-
this.ws = null;
|
|
68
|
-
};
|
|
69
77
|
}
|
|
70
78
|
catch (err) {
|
|
71
79
|
console.error('WebSocketVoiceClient: connect failed', err);
|
|
@@ -73,23 +81,126 @@ export class WebSocketVoiceClientService {
|
|
|
73
81
|
throw err;
|
|
74
82
|
}
|
|
75
83
|
}
|
|
76
|
-
|
|
84
|
+
handleIncomingMessage(payload) {
|
|
85
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
86
|
+
if (typeof payload === 'string') {
|
|
87
|
+
this.handleJsonString(payload);
|
|
88
|
+
return;
|
|
89
|
+
}
|
|
90
|
+
if (payload instanceof ArrayBuffer) {
|
|
91
|
+
this.handleBinaryMessage(payload);
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
if (payload instanceof Blob) {
|
|
95
|
+
const ab = yield payload.arrayBuffer();
|
|
96
|
+
this.handleBinaryMessage(ab);
|
|
97
|
+
}
|
|
98
|
+
});
|
|
99
|
+
}
|
|
100
|
+
handleJsonString(jsonText) {
|
|
101
|
+
try {
|
|
102
|
+
const msg = JSON.parse(jsonText);
|
|
103
|
+
this.ngZone.run(() => this.handleJsonMessage(msg));
|
|
104
|
+
}
|
|
105
|
+
catch (_a) {
|
|
106
|
+
// Ignore non-JSON
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
handleBinaryMessage(buffer) {
|
|
110
|
+
// Some backends wrap JSON events inside binary WS frames.
|
|
111
|
+
const maybeText = this.tryDecodeUtf8(buffer);
|
|
112
|
+
if (maybeText !== null) {
|
|
113
|
+
this.handleJsonString(maybeText);
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
// Otherwise treat binary as streamed assistant audio.
|
|
117
|
+
this.ngZone.run(() => this.audioChunkSubject.next(buffer));
|
|
118
|
+
}
|
|
119
|
+
tryDecodeUtf8(buffer) {
|
|
120
|
+
try {
|
|
121
|
+
const text = new TextDecoder('utf-8', { fatal: true }).decode(buffer);
|
|
122
|
+
const trimmed = text.trim();
|
|
123
|
+
if (!trimmed || (trimmed[0] !== '{' && trimmed[0] !== '[')) {
|
|
124
|
+
return null;
|
|
125
|
+
}
|
|
126
|
+
return trimmed;
|
|
127
|
+
}
|
|
128
|
+
catch (_a) {
|
|
129
|
+
return null;
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
handleJsonMessage(msg) {
|
|
133
|
+
const type = msg.type;
|
|
134
|
+
const typeStr = typeof type === 'string' ? type : '';
|
|
135
|
+
if (typeStr === 'session_ready' || typeStr === 'connected' || typeStr === 'voice_session_started') {
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
if (typeStr === 'assistant_speaking' ||
|
|
139
|
+
typeStr === 'bot_speaking') {
|
|
140
|
+
if (msg.active === true || msg.speaking === true) {
|
|
141
|
+
this.assistantSpeakingSubject.next(true);
|
|
142
|
+
}
|
|
143
|
+
else if (msg.active === false || msg.speaking === false) {
|
|
144
|
+
this.assistantSpeakingSubject.next(false);
|
|
145
|
+
}
|
|
146
|
+
return;
|
|
147
|
+
}
|
|
148
|
+
if (typeStr === 'user_speaking') {
|
|
149
|
+
if (msg.active === true || msg.speaking === true) {
|
|
150
|
+
this.serverUserSpeakingSubject.next(true);
|
|
151
|
+
}
|
|
152
|
+
else if (msg.active === false || msg.speaking === false) {
|
|
153
|
+
this.serverUserSpeakingSubject.next(false);
|
|
154
|
+
}
|
|
155
|
+
return;
|
|
156
|
+
}
|
|
157
|
+
if (typeStr === 'input_audio_buffer.speech_started') {
|
|
158
|
+
this.serverUserSpeakingSubject.next(true);
|
|
159
|
+
return;
|
|
160
|
+
}
|
|
161
|
+
if (typeStr === 'input_audio_buffer.speech_stopped') {
|
|
162
|
+
this.serverUserSpeakingSubject.next(false);
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
if (typeStr === 'response.audio.delta') {
|
|
166
|
+
this.assistantSpeakingSubject.next(true);
|
|
167
|
+
return;
|
|
168
|
+
}
|
|
169
|
+
if (typeStr === 'response.audio.done' ||
|
|
170
|
+
typeStr === 'response.output_audio.done') {
|
|
171
|
+
this.assistantSpeakingSubject.next(false);
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
if (typeStr === 'user_transcript' && typeof msg.text === 'string') {
|
|
175
|
+
this.userTranscriptSubject.next({
|
|
176
|
+
text: msg.text,
|
|
177
|
+
final: msg.final === true,
|
|
178
|
+
});
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
if (typeStr === 'bot_transcript' && typeof msg.text === 'string') {
|
|
182
|
+
this.botTranscriptSubject.next(msg.text);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
77
185
|
disconnect() {
|
|
78
|
-
if (this.ws) {
|
|
79
|
-
|
|
80
|
-
this.ws = null;
|
|
186
|
+
if (!this.ws) {
|
|
187
|
+
return;
|
|
81
188
|
}
|
|
189
|
+
this.closeInitiatedByClient = true;
|
|
190
|
+
this.ws.close();
|
|
82
191
|
}
|
|
83
|
-
/** Whether the WebSocket is open. */
|
|
84
192
|
get isConnected() {
|
|
85
193
|
var _a;
|
|
86
194
|
return ((_a = this.ws) === null || _a === void 0 ? void 0 : _a.readyState) === WebSocket.OPEN;
|
|
87
195
|
}
|
|
88
196
|
}
|
|
89
|
-
WebSocketVoiceClientService.ɵprov = i0.ɵɵdefineInjectable({ factory: function WebSocketVoiceClientService_Factory() { return new WebSocketVoiceClientService(); }, token: WebSocketVoiceClientService, providedIn: "root" });
|
|
197
|
+
WebSocketVoiceClientService.ɵprov = i0.ɵɵdefineInjectable({ factory: function WebSocketVoiceClientService_Factory() { return new WebSocketVoiceClientService(i0.ɵɵinject(i0.NgZone)); }, token: WebSocketVoiceClientService, providedIn: "root" });
|
|
90
198
|
WebSocketVoiceClientService.decorators = [
|
|
91
199
|
{ type: Injectable, args: [{
|
|
92
200
|
providedIn: 'root',
|
|
93
201
|
},] }
|
|
94
202
|
];
|
|
95
|
-
|
|
203
|
+
WebSocketVoiceClientService.ctorParameters = () => [
|
|
204
|
+
{ type: NgZone }
|
|
205
|
+
];
|
|
206
|
+
//# 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, NgZone } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\n\n/** WebSocket message types from backend (voice session over a single WS). */\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  | WsMessageUserTranscript\n  | WsMessageBotTranscript;\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * Native WebSocket client for voice session (signaling, transcripts, speaking hints).\n * CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.\n *\n * Connects to `ws_url` from `POST {baseUrl}/ai/ask-voice-socket`.\n * Parses JSON messages for transcripts and optional assistant/user speaking flags.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class WebSocketVoiceClientService {\n  private ws: WebSocket | null = null;\n  /** True when {@link disconnect} initiated the close (not counted as remote close). */\n  private closeInitiatedByClient = false;\n\n  private openedSubject = new Subject<void>();\n  private remoteCloseSubject = new Subject<void>();\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n  private assistantSpeakingSubject = new Subject<boolean>();\n  private serverUserSpeakingSubject = new Subject<boolean>();\n  private audioChunkSubject = new Subject<ArrayBuffer>();\n\n  /** Fires once each time the WebSocket reaches OPEN. */\n  opened$: Observable<void> = this.openedSubject.asObservable();\n\n  /** Fires when the socket closes without a client-initiated {@link disconnect}. */\n  remoteClose$: Observable<void> = this.remoteCloseSubject.asObservable();\n\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n\n  botTranscript$: Observable<string> =\n    this.botTranscriptSubject.asObservable();\n\n  /** Assistant/bot speaking, when the server sends explicit events (see {@link handleJsonMessage}). */\n  assistantSpeaking$: Observable<boolean> =\n    this.assistantSpeakingSubject.asObservable();\n\n  /** User speaking from server-side VAD, if provided. */\n  serverUserSpeaking$: Observable<boolean> =\n    this.serverUserSpeakingSubject.asObservable();\n\n  /** Binary audio frames from server (when backend streams bot audio over WS). */\n  audioChunk$: Observable<ArrayBuffer> = this.audioChunkSubject.asObservable();\n\n  constructor(private ngZone: NgZone) {}\n\n  connect(wsUrl: string): void {\n    if (this.ws?.readyState === WebSocket.OPEN) {\n      return;\n    }\n    if (this.ws) {\n      this.closeInitiatedByClient = true;\n      this.ws.close();\n    }\n\n    try {\n      const socket = new WebSocket(wsUrl);\n      this.ws = socket;\n      socket.onopen = () => {\n        if (this.ws !== socket) return;\n        this.ngZone.run(() => this.openedSubject.next());\n      };\n      socket.onmessage = (event: MessageEvent) => {\n        if (this.ws !== socket) return;\n        void this.handleIncomingMessage(event.data);\n      };\n      socket.onerror = () => {\n        this.ngZone.run(() => {\n          if (this.ws === socket && socket.readyState !== WebSocket.CLOSED) {\n            socket.close();\n          }\n        });\n      };\n      socket.onclose = () => {\n        if (this.ws === socket) {\n          this.ws = null;\n        }\n        const client = this.closeInitiatedByClient;\n        this.closeInitiatedByClient = false;\n        if (!client) {\n          this.ngZone.run(() => this.remoteCloseSubject.next());\n        }\n      };\n    } catch (err) {\n      console.error('WebSocketVoiceClient: connect failed', err);\n      this.ws = null;\n      throw err;\n    }\n  }\n\n  private async handleIncomingMessage(payload: unknown): Promise<void> {\n    if (typeof payload === 'string') {\n      this.handleJsonString(payload);\n      return;\n    }\n    if (payload instanceof ArrayBuffer) {\n      this.handleBinaryMessage(payload);\n      return;\n    }\n    if (payload instanceof Blob) {\n      const ab = await payload.arrayBuffer();\n      this.handleBinaryMessage(ab);\n    }\n  }\n\n  private handleJsonString(jsonText: string): void {\n    try {\n      const msg = JSON.parse(jsonText) as Record<string, unknown>;\n      this.ngZone.run(() => this.handleJsonMessage(msg));\n    } catch {\n      // Ignore non-JSON\n    }\n  }\n\n  private handleBinaryMessage(buffer: ArrayBuffer): void {\n    // Some backends wrap JSON events inside binary WS frames.\n    const maybeText = this.tryDecodeUtf8(buffer);\n    if (maybeText !== null) {\n      this.handleJsonString(maybeText);\n      return;\n    }\n    // Otherwise treat binary as streamed assistant audio.\n    this.ngZone.run(() => this.audioChunkSubject.next(buffer));\n  }\n\n  private tryDecodeUtf8(buffer: ArrayBuffer): string | null {\n    try {\n      const text = new TextDecoder('utf-8', { fatal: true }).decode(buffer);\n      const trimmed = text.trim();\n      if (!trimmed || (trimmed[0] !== '{' && trimmed[0] !== '[')) {\n        return null;\n      }\n      return trimmed;\n    } catch {\n      return null;\n    }\n  }\n\n  private handleJsonMessage(msg: Record<string, unknown>): void {\n    const type = msg.type;\n    const typeStr = typeof type === 'string' ? type : '';\n\n    if (typeStr === 'session_ready' || typeStr === 'connected' || typeStr === 'voice_session_started') {\n      return;\n    }\n\n    if (\n      typeStr === 'assistant_speaking' ||\n      typeStr === 'bot_speaking'\n    ) {\n      if (msg.active === true || msg.speaking === true) {\n        this.assistantSpeakingSubject.next(true);\n      } else if (msg.active === false || msg.speaking === false) {\n        this.assistantSpeakingSubject.next(false);\n      }\n      return;\n    }\n\n    if (typeStr === 'user_speaking') {\n      if (msg.active === true || msg.speaking === true) {\n        this.serverUserSpeakingSubject.next(true);\n      } else if (msg.active === false || msg.speaking === false) {\n        this.serverUserSpeakingSubject.next(false);\n      }\n      return;\n    }\n\n    if (typeStr === 'input_audio_buffer.speech_started') {\n      this.serverUserSpeakingSubject.next(true);\n      return;\n    }\n    if (typeStr === 'input_audio_buffer.speech_stopped') {\n      this.serverUserSpeakingSubject.next(false);\n      return;\n    }\n\n    if (typeStr === 'response.audio.delta') {\n      this.assistantSpeakingSubject.next(true);\n      return;\n    }\n    if (\n      typeStr === 'response.audio.done' ||\n      typeStr === 'response.output_audio.done'\n    ) {\n      this.assistantSpeakingSubject.next(false);\n      return;\n    }\n\n    if (typeStr === 'user_transcript' && typeof msg.text === 'string') {\n      this.userTranscriptSubject.next({\n        text: msg.text,\n        final: msg.final === true,\n      });\n      return;\n    }\n    if (typeStr === 'bot_transcript' && typeof msg.text === 'string') {\n      this.botTranscriptSubject.next(msg.text);\n    }\n  }\n\n  disconnect(): void {\n    if (!this.ws) {\n      return;\n    }\n    this.closeInitiatedByClient = true;\n    this.ws.close();\n  }\n\n  get isConnected(): boolean {\n    return this.ws?.readyState === WebSocket.OPEN;\n  }\n}\n"]}
|
|
@@ -4,9 +4,8 @@ import { VoiceAgentModalComponent } from './components/voice-agent-modal/voice-a
|
|
|
4
4
|
import { VoiceAgentService } from './services/voice-agent.service';
|
|
5
5
|
import { AudioAnalyzerService } from './services/audio-analyzer.service';
|
|
6
6
|
import { WebSocketVoiceClientService } from './services/websocket-voice-client.service';
|
|
7
|
-
import { DailyVoiceClientService } from './services/daily-voice-client.service';
|
|
8
7
|
/**
|
|
9
|
-
* Voice agent module. Uses native WebSocket
|
|
8
|
+
* Voice agent module. Uses native WebSocket for the voice session.
|
|
10
9
|
* Does NOT use Socket.IO or ngx-socket-io.
|
|
11
10
|
*/
|
|
12
11
|
export class VoiceAgentModule {
|
|
@@ -22,12 +21,11 @@ VoiceAgentModule.decorators = [
|
|
|
22
21
|
providers: [
|
|
23
22
|
VoiceAgentService,
|
|
24
23
|
AudioAnalyzerService,
|
|
25
|
-
WebSocketVoiceClientService
|
|
26
|
-
DailyVoiceClientService
|
|
24
|
+
WebSocketVoiceClientService
|
|
27
25
|
],
|
|
28
26
|
exports: [
|
|
29
27
|
VoiceAgentModalComponent
|
|
30
28
|
]
|
|
31
29
|
},] }
|
|
32
30
|
];
|
|
33
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtYWdlbnQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9yb2hpdHRoYWt1ci9oaXZlLWdwdC9IaXZlQUktUGFja2FnZXMvQW5ndWxhci9wcm9qZWN0cy9oaXZlZ3B0L2V2ZW50c2dwdC1hbmd1bGFyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3ZvaWNlLWFnZW50Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSw0REFBNEQsQ0FBQztBQUN0RyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUNuRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUN6RSxPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSwyQ0FBMkMsQ0FBQztBQUV4Rjs7O0dBR0c7QUFpQkgsTUFBTSxPQUFPLGdCQUFnQjs7O1lBaEI1QixRQUFRLFNBQUM7Z0JBQ1IsWUFBWSxFQUFFO29CQUNaLHdCQUF3QjtpQkFDekI7Z0JBQ0QsT0FBTyxFQUFFO29CQUNQLFlBQVk7aUJBQ2I7Z0JBQ0QsU0FBUyxFQUFFO29CQUNULGlCQUFpQjtvQkFDakIsb0JBQW9CO29CQUNwQiwyQkFBMkI7aUJBQzVCO2dCQUNELE9BQU8sRUFBRTtvQkFDUCx3QkFBd0I7aUJBQ3pCO2FBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFZvaWNlQWdlbnRNb2RhbENvbXBvbmVudCB9IGZyb20gJy4vY29tcG9uZW50cy92b2ljZS1hZ2VudC1tb2RhbC92b2ljZS1hZ2VudC1tb2RhbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVm9pY2VBZ2VudFNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL3ZvaWNlLWFnZW50LnNlcnZpY2UnO1xuaW1wb3J0IHsgQXVkaW9BbmFseXplclNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL2F1ZGlvLWFuYWx5emVyLnNlcnZpY2UnO1xuaW1wb3J0IHsgV2ViU29ja2V0Vm9pY2VDbGllbnRTZXJ2aWNlIH0gZnJvbSAnLi9zZXJ2aWNlcy93ZWJzb2NrZXQtdm9pY2UtY2xpZW50LnNlcnZpY2UnO1xuXG4vKipcbiAqIFZvaWNlIGFnZW50IG1vZHVsZS4gVXNlcyBuYXRpdmUgV2ViU29ja2V0IGZvciB0aGUgdm9pY2Ugc2Vzc2lvbi5cbiAqIERvZXMgTk9UIHVzZSBTb2NrZXQuSU8gb3Igbmd4LXNvY2tldC1pby5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbXG4gICAgVm9pY2VBZ2VudE1vZGFsQ29tcG9uZW50XG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGVcbiAgXSxcbiAgcHJvdmlkZXJzOiBbXG4gICAgVm9pY2VBZ2VudFNlcnZpY2UsXG4gICAgQXVkaW9BbmFseXplclNlcnZpY2UsXG4gICAgV2ViU29ja2V0Vm9pY2VDbGllbnRTZXJ2aWNlXG4gIF0sXG4gIGV4cG9ydHM6IFtcbiAgICBWb2ljZUFnZW50TW9kYWxDb21wb25lbnRcbiAgXVxufSlcbmV4cG9ydCBjbGFzcyBWb2ljZUFnZW50TW9kdWxlIHsgfVxuIl19
|