@hivegpt/hiveai-angular 0.0.581 → 0.0.582

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (25) hide show
  1. package/bundles/hivegpt-hiveai-angular.umd.js +285 -503
  2. package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
  3. package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
  4. package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
  5. package/esm2015/hivegpt-hiveai-angular.js +4 -5
  6. package/esm2015/lib/components/voice-agent/services/audio-analyzer.service.js +3 -3
  7. package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +118 -85
  8. package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +114 -46
  9. package/esm2015/lib/components/voice-agent/voice-agent.module.js +3 -5
  10. package/fesm2015/hivegpt-hiveai-angular.js +223 -422
  11. package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
  12. package/hivegpt-hiveai-angular.d.ts +3 -4
  13. package/hivegpt-hiveai-angular.d.ts.map +1 -1
  14. package/hivegpt-hiveai-angular.metadata.json +1 -1
  15. package/lib/components/voice-agent/services/audio-analyzer.service.d.ts +2 -2
  16. package/lib/components/voice-agent/services/voice-agent.service.d.ts +11 -13
  17. package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
  18. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +23 -20
  19. package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
  20. package/lib/components/voice-agent/voice-agent.module.d.ts +1 -1
  21. package/lib/components/voice-agent/voice-agent.module.d.ts.map +1 -1
  22. package/package.json +1 -1
  23. package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +0 -305
  24. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +0 -62
  25. package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +0 -1
@@ -1,70 +1,83 @@
1
- import { Injectable } from '@angular/core';
1
+ import { Injectable, NgZone } from '@angular/core';
2
2
  import { Subject } from 'rxjs';
3
3
  import * as i0 from "@angular/core";
4
4
  /**
5
- * WebSocket-only client for voice agent signaling.
5
+ * Native WebSocket client for voice session (signaling, transcripts, speaking hints).
6
6
  * CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.
7
7
  *
8
- * Responsibilities:
9
- * - Connect to ws_url (from POST /ai/ask-voice response)
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).
8
+ * Connects to `ws_url` from `POST {baseUrl}/ai/ask-voice-socket`.
9
+ * Parses JSON messages for transcripts and optional assistant/user speaking flags.
13
10
  */
14
11
  export class WebSocketVoiceClientService {
15
- constructor() {
12
+ constructor(ngZone) {
13
+ this.ngZone = ngZone;
16
14
  this.ws = null;
17
- this.roomCreatedSubject = new Subject();
15
+ /** True when {@link disconnect} initiated the close (not counted as remote close). */
16
+ this.closeInitiatedByClient = false;
17
+ this.openedSubject = new Subject();
18
+ this.remoteCloseSubject = new Subject();
18
19
  this.userTranscriptSubject = new Subject();
19
20
  this.botTranscriptSubject = new Subject();
20
- /** Emits room_url when backend sends room_created. */
21
- this.roomCreated$ = this.roomCreatedSubject.asObservable();
22
- /** Emits user transcript updates. */
21
+ this.assistantSpeakingSubject = new Subject();
22
+ this.serverUserSpeakingSubject = new Subject();
23
+ /** Fires once each time the WebSocket reaches OPEN. */
24
+ this.opened$ = this.openedSubject.asObservable();
25
+ /** Fires when the socket closes without a client-initiated {@link disconnect}. */
26
+ this.remoteClose$ = this.remoteCloseSubject.asObservable();
23
27
  this.userTranscript$ = this.userTranscriptSubject.asObservable();
24
- /** Emits bot transcript updates. */
25
28
  this.botTranscript$ = this.botTranscriptSubject.asObservable();
29
+ /** Assistant/bot speaking, when the server sends explicit events (see {@link handleJsonMessage}). */
30
+ this.assistantSpeaking$ = this.assistantSpeakingSubject.asObservable();
31
+ /** User speaking from server-side VAD, if provided. */
32
+ this.serverUserSpeaking$ = this.serverUserSpeakingSubject.asObservable();
26
33
  }
27
- /** Connect to signaling WebSocket. No audio over this connection. */
28
34
  connect(wsUrl) {
29
35
  var _a;
30
36
  if (((_a = this.ws) === null || _a === void 0 ? void 0 : _a.readyState) === WebSocket.OPEN) {
31
37
  return;
32
38
  }
33
39
  if (this.ws) {
40
+ this.closeInitiatedByClient = true;
34
41
  this.ws.close();
35
- this.ws = null;
36
42
  }
37
43
  try {
38
- this.ws = new WebSocket(wsUrl);
39
- this.ws.onmessage = (event) => {
40
- var _a;
44
+ const socket = new WebSocket(wsUrl);
45
+ this.ws = socket;
46
+ socket.onopen = () => {
47
+ if (this.ws !== socket)
48
+ return;
49
+ this.ngZone.run(() => this.openedSubject.next());
50
+ };
51
+ socket.onmessage = (event) => {
52
+ if (this.ws !== socket)
53
+ return;
54
+ if (typeof event.data !== 'string') {
55
+ return;
56
+ }
41
57
  try {
42
58
  const msg = JSON.parse(event.data);
43
- if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'room_created') {
44
- const roomUrl = ((_a = msg.room_url) !== null && _a !== void 0 ? _a : msg.roomUrl);
45
- if (typeof roomUrl === 'string') {
46
- this.roomCreatedSubject.next(roomUrl);
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
- }
59
+ this.ngZone.run(() => this.handleJsonMessage(msg));
58
60
  }
59
- catch (_b) {
60
- // Ignore non-JSON or unknown messages
61
+ catch (_a) {
62
+ // Ignore non-JSON
61
63
  }
62
64
  };
63
- this.ws.onerror = () => {
64
- this.disconnect();
65
+ socket.onerror = () => {
66
+ this.ngZone.run(() => {
67
+ if (this.ws === socket && socket.readyState !== WebSocket.CLOSED) {
68
+ socket.close();
69
+ }
70
+ });
65
71
  };
66
- this.ws.onclose = () => {
67
- this.ws = null;
72
+ socket.onclose = () => {
73
+ if (this.ws === socket) {
74
+ this.ws = null;
75
+ }
76
+ const client = this.closeInitiatedByClient;
77
+ this.closeInitiatedByClient = false;
78
+ if (!client) {
79
+ this.ngZone.run(() => this.remoteCloseSubject.next());
80
+ }
68
81
  };
69
82
  }
70
83
  catch (err) {
@@ -73,23 +86,78 @@ export class WebSocketVoiceClientService {
73
86
  throw err;
74
87
  }
75
88
  }
76
- /** Disconnect and cleanup. */
89
+ handleJsonMessage(msg) {
90
+ const type = msg.type;
91
+ const typeStr = typeof type === 'string' ? type : '';
92
+ if (typeStr === 'session_ready' || typeStr === 'connected' || typeStr === 'voice_session_started') {
93
+ return;
94
+ }
95
+ if (typeStr === 'assistant_speaking' ||
96
+ typeStr === 'bot_speaking') {
97
+ if (msg.active === true || msg.speaking === true) {
98
+ this.assistantSpeakingSubject.next(true);
99
+ }
100
+ else if (msg.active === false || msg.speaking === false) {
101
+ this.assistantSpeakingSubject.next(false);
102
+ }
103
+ return;
104
+ }
105
+ if (typeStr === 'user_speaking') {
106
+ if (msg.active === true || msg.speaking === true) {
107
+ this.serverUserSpeakingSubject.next(true);
108
+ }
109
+ else if (msg.active === false || msg.speaking === false) {
110
+ this.serverUserSpeakingSubject.next(false);
111
+ }
112
+ return;
113
+ }
114
+ if (typeStr === 'input_audio_buffer.speech_started') {
115
+ this.serverUserSpeakingSubject.next(true);
116
+ return;
117
+ }
118
+ if (typeStr === 'input_audio_buffer.speech_stopped') {
119
+ this.serverUserSpeakingSubject.next(false);
120
+ return;
121
+ }
122
+ if (typeStr === 'response.audio.delta') {
123
+ this.assistantSpeakingSubject.next(true);
124
+ return;
125
+ }
126
+ if (typeStr === 'response.audio.done' ||
127
+ typeStr === 'response.output_audio.done') {
128
+ this.assistantSpeakingSubject.next(false);
129
+ return;
130
+ }
131
+ if (typeStr === 'user_transcript' && typeof msg.text === 'string') {
132
+ this.userTranscriptSubject.next({
133
+ text: msg.text,
134
+ final: msg.final === true,
135
+ });
136
+ return;
137
+ }
138
+ if (typeStr === 'bot_transcript' && typeof msg.text === 'string') {
139
+ this.botTranscriptSubject.next(msg.text);
140
+ }
141
+ }
77
142
  disconnect() {
78
- if (this.ws) {
79
- this.ws.close();
80
- this.ws = null;
143
+ if (!this.ws) {
144
+ return;
81
145
  }
146
+ this.closeInitiatedByClient = true;
147
+ this.ws.close();
82
148
  }
83
- /** Whether the WebSocket is open. */
84
149
  get isConnected() {
85
150
  var _a;
86
151
  return ((_a = this.ws) === null || _a === void 0 ? void 0 : _a.readyState) === WebSocket.OPEN;
87
152
  }
88
153
  }
89
- WebSocketVoiceClientService.ɵprov = i0.ɵɵdefineInjectable({ factory: function WebSocketVoiceClientService_Factory() { return new WebSocketVoiceClientService(); }, token: WebSocketVoiceClientService, providedIn: "root" });
154
+ WebSocketVoiceClientService.ɵprov = i0.ɵɵdefineInjectable({ factory: function WebSocketVoiceClientService_Factory() { return new WebSocketVoiceClientService(i0.ɵɵinject(i0.NgZone)); }, token: WebSocketVoiceClientService, providedIn: "root" });
90
155
  WebSocketVoiceClientService.decorators = [
91
156
  { type: Injectable, args: [{
92
157
  providedIn: 'root',
93
158
  },] }
94
159
  ];
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"]}
160
+ WebSocketVoiceClientService.ctorParameters = () => [
161
+ { type: NgZone }
162
+ ];
163
+ //# 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":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAc,OAAO,EAAE,MAAM,MAAM,CAAC;;AAuB3C;;;;;;GAMG;AAIH,MAAM,OAAO,2BAA2B;IAgCtC,YAAoB,MAAc;QAAd,WAAM,GAAN,MAAM,CAAQ;QA/B1B,OAAE,GAAqB,IAAI,CAAC;QACpC,sFAAsF;QAC9E,2BAAsB,GAAG,KAAK,CAAC;QAE/B,kBAAa,GAAG,IAAI,OAAO,EAAQ,CAAC;QACpC,uBAAkB,GAAG,IAAI,OAAO,EAAQ,CAAC;QACzC,0BAAqB,GAAG,IAAI,OAAO,EAAkB,CAAC;QACtD,yBAAoB,GAAG,IAAI,OAAO,EAAU,CAAC;QAC7C,6BAAwB,GAAG,IAAI,OAAO,EAAW,CAAC;QAClD,8BAAyB,GAAG,IAAI,OAAO,EAAW,CAAC;QAE3D,uDAAuD;QACvD,YAAO,GAAqB,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;QAE9D,kFAAkF;QAClF,iBAAY,GAAqB,IAAI,CAAC,kBAAkB,CAAC,YAAY,EAAE,CAAC;QAExE,oBAAe,GACb,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,CAAC;QAE5C,mBAAc,GACZ,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,CAAC;QAE3C,qGAAqG;QACrG,uBAAkB,GAChB,IAAI,CAAC,wBAAwB,CAAC,YAAY,EAAE,CAAC;QAE/C,uDAAuD;QACvD,wBAAmB,GACjB,IAAI,CAAC,yBAAyB,CAAC,YAAY,EAAE,CAAC;IAEX,CAAC;IAEtC,OAAO,CAAC,KAAa;;QACnB,IAAI,CAAA,MAAA,IAAI,CAAC,EAAE,0CAAE,UAAU,MAAK,SAAS,CAAC,IAAI,EAAE;YAC1C,OAAO;SACR;QACD,IAAI,IAAI,CAAC,EAAE,EAAE;YACX,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;YACnC,IAAI,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;SACjB;QAED,IAAI;YACF,MAAM,MAAM,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;YACpC,IAAI,CAAC,EAAE,GAAG,MAAM,CAAC;YACjB,MAAM,CAAC,MAAM,GAAG,GAAG,EAAE;gBACnB,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM;oBAAE,OAAO;gBAC/B,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC,CAAC;YACnD,CAAC,CAAC;YACF,MAAM,CAAC,SAAS,GAAG,CAAC,KAAmB,EAAE,EAAE;gBACzC,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM;oBAAE,OAAO;gBAC/B,IAAI,OAAO,KAAK,CAAC,IAAI,KAAK,QAAQ,EAAE;oBAClC,OAAO;iBACR;gBACD,IAAI;oBACF,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAA4B,CAAC;oBAC9D,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC;iBACpD;gBAAC,WAAM;oBACN,kBAAkB;iBACnB;YACH,CAAC,CAAC;YACF,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE;gBACpB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;oBACnB,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM,IAAI,MAAM,CAAC,UAAU,KAAK,SAAS,CAAC,MAAM,EAAE;wBAChE,MAAM,CAAC,KAAK,EAAE,CAAC;qBAChB;gBACH,CAAC,CAAC,CAAC;YACL,CAAC,CAAC;YACF,MAAM,CAAC,OAAO,GAAG,GAAG,EAAE;gBACpB,IAAI,IAAI,CAAC,EAAE,KAAK,MAAM,EAAE;oBACtB,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;iBAChB;gBACD,MAAM,MAAM,GAAG,IAAI,CAAC,sBAAsB,CAAC;gBAC3C,IAAI,CAAC,sBAAsB,GAAG,KAAK,CAAC;gBACpC,IAAI,CAAC,MAAM,EAAE;oBACX,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,CAAC,CAAC;iBACvD;YACH,CAAC,CAAC;SACH;QAAC,OAAO,GAAG,EAAE;YACZ,OAAO,CAAC,KAAK,CAAC,sCAAsC,EAAE,GAAG,CAAC,CAAC;YAC3D,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;YACf,MAAM,GAAG,CAAC;SACX;IACH,CAAC;IAEO,iBAAiB,CAAC,GAA4B;QACpD,MAAM,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC;QACtB,MAAM,OAAO,GAAG,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC;QAErD,IAAI,OAAO,KAAK,eAAe,IAAI,OAAO,KAAK,WAAW,IAAI,OAAO,KAAK,uBAAuB,EAAE;YACjG,OAAO;SACR;QAED,IACE,OAAO,KAAK,oBAAoB;YAChC,OAAO,KAAK,cAAc,EAC1B;YACA,IAAI,GAAG,CAAC,MAAM,KAAK,IAAI,IAAI,GAAG,CAAC,QAAQ,KAAK,IAAI,EAAE;gBAChD,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aAC1C;iBAAM,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,GAAG,CAAC,QAAQ,KAAK,KAAK,EAAE;gBACzD,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aAC3C;YACD,OAAO;SACR;QAED,IAAI,OAAO,KAAK,eAAe,EAAE;YAC/B,IAAI,GAAG,CAAC,MAAM,KAAK,IAAI,IAAI,GAAG,CAAC,QAAQ,KAAK,IAAI,EAAE;gBAChD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aAC3C;iBAAM,IAAI,GAAG,CAAC,MAAM,KAAK,KAAK,IAAI,GAAG,CAAC,QAAQ,KAAK,KAAK,EAAE;gBACzD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aAC5C;YACD,OAAO;SACR;QAED,IAAI,OAAO,KAAK,mCAAmC,EAAE;YACnD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC1C,OAAO;SACR;QACD,IAAI,OAAO,KAAK,mCAAmC,EAAE;YACnD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC3C,OAAO;SACR;QAED,IAAI,OAAO,KAAK,sBAAsB,EAAE;YACtC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACzC,OAAO;SACR;QACD,IACE,OAAO,KAAK,qBAAqB;YACjC,OAAO,KAAK,4BAA4B,EACxC;YACA,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC1C,OAAO;SACR;QAED,IAAI,OAAO,KAAK,iBAAiB,IAAI,OAAO,GAAG,CAAC,IAAI,KAAK,QAAQ,EAAE;YACjE,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC;gBAC9B,IAAI,EAAE,GAAG,CAAC,IAAI;gBACd,KAAK,EAAE,GAAG,CAAC,KAAK,KAAK,IAAI;aAC1B,CAAC,CAAC;YACH,OAAO;SACR;QACD,IAAI,OAAO,KAAK,gBAAgB,IAAI,OAAO,GAAG,CAAC,IAAI,KAAK,QAAQ,EAAE;YAChE,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;SAC1C;IACH,CAAC;IAED,UAAU;QACR,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE;YACZ,OAAO;SACR;QACD,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;QACnC,IAAI,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;IAClB,CAAC;IAED,IAAI,WAAW;;QACb,OAAO,CAAA,MAAA,IAAI,CAAC,EAAE,0CAAE,UAAU,MAAK,SAAS,CAAC,IAAI,CAAC;IAChD,CAAC;;;;YAjKF,UAAU,SAAC;gBACV,UAAU,EAAE,MAAM;aACnB;;;YAjCoB,MAAM","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\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  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        if (typeof event.data !== 'string') {\n          return;\n        }\n        try {\n          const msg = JSON.parse(event.data) as Record<string, unknown>;\n          this.ngZone.run(() => this.handleJsonMessage(msg));\n        } catch {\n          // Ignore non-JSON\n        }\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 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 + Daily.js only.
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,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtYWdlbnQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9yb2hpdHRoYWt1ci9oaXZlLWdwdC9IaXZlQUktUGFja2FnZXMvQW5ndWxhci9wcm9qZWN0cy9oaXZlZ3B0L2V2ZW50c2dwdC1hbmd1bGFyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3ZvaWNlLWFnZW50Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSw0REFBNEQsQ0FBQztBQUN0RyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUNuRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUN6RSxPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSwyQ0FBMkMsQ0FBQztBQUN4RixPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQztBQUVoRjs7O0dBR0c7QUFrQkgsTUFBTSxPQUFPLGdCQUFnQjs7O1lBakI1QixRQUFRLFNBQUM7Z0JBQ1IsWUFBWSxFQUFFO29CQUNaLHdCQUF3QjtpQkFDekI7Z0JBQ0QsT0FBTyxFQUFFO29CQUNQLFlBQVk7aUJBQ2I7Z0JBQ0QsU0FBUyxFQUFFO29CQUNULGlCQUFpQjtvQkFDakIsb0JBQW9CO29CQUNwQiwyQkFBMkI7b0JBQzNCLHVCQUF1QjtpQkFDeEI7Z0JBQ0QsT0FBTyxFQUFFO29CQUNQLHdCQUF3QjtpQkFDekI7YUFDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgVm9pY2VBZ2VudE1vZGFsQ29tcG9uZW50IH0gZnJvbSAnLi9jb21wb25lbnRzL3ZvaWNlLWFnZW50LW1vZGFsL3ZvaWNlLWFnZW50LW1vZGFsLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBWb2ljZUFnZW50U2VydmljZSB9IGZyb20gJy4vc2VydmljZXMvdm9pY2UtYWdlbnQuc2VydmljZSc7XG5pbXBvcnQgeyBBdWRpb0FuYWx5emVyU2VydmljZSB9IGZyb20gJy4vc2VydmljZXMvYXVkaW8tYW5hbHl6ZXIuc2VydmljZSc7XG5pbXBvcnQgeyBXZWJTb2NrZXRWb2ljZUNsaWVudFNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL3dlYnNvY2tldC12b2ljZS1jbGllbnQuc2VydmljZSc7XG5pbXBvcnQgeyBEYWlseVZvaWNlQ2xpZW50U2VydmljZSB9IGZyb20gJy4vc2VydmljZXMvZGFpbHktdm9pY2UtY2xpZW50LnNlcnZpY2UnO1xuXG4vKipcbiAqIFZvaWNlIGFnZW50IG1vZHVsZS4gVXNlcyBuYXRpdmUgV2ViU29ja2V0ICsgRGFpbHkuanMgb25seS5cbiAqIERvZXMgTk9UIHVzZSBTb2NrZXQuSU8gb3Igbmd4LXNvY2tldC1pby5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbXG4gICAgVm9pY2VBZ2VudE1vZGFsQ29tcG9uZW50XG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGVcbiAgXSxcbiAgcHJvdmlkZXJzOiBbXG4gICAgVm9pY2VBZ2VudFNlcnZpY2UsXG4gICAgQXVkaW9BbmFseXplclNlcnZpY2UsXG4gICAgV2ViU29ja2V0Vm9pY2VDbGllbnRTZXJ2aWNlLFxuICAgIERhaWx5Vm9pY2VDbGllbnRTZXJ2aWNlXG4gIF0sXG4gIGV4cG9ydHM6IFtcbiAgICBWb2ljZUFnZW50TW9kYWxDb21wb25lbnRcbiAgXVxufSlcbmV4cG9ydCBjbGFzcyBWb2ljZUFnZW50TW9kdWxlIHsgfVxuIl19
31
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtYWdlbnQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6Ii9Vc2Vycy9yb2hpdHRoYWt1ci9oaXZlLWdwdC9IaXZlQUktUGFja2FnZXMvQW5ndWxhci9wcm9qZWN0cy9oaXZlZ3B0L2V2ZW50c2dwdC1hbmd1bGFyL3NyYy8iLCJzb3VyY2VzIjpbImxpYi9jb21wb25lbnRzL3ZvaWNlLWFnZW50L3ZvaWNlLWFnZW50Lm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSw0REFBNEQsQ0FBQztBQUN0RyxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUNuRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQUN6RSxPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSwyQ0FBMkMsQ0FBQztBQUV4Rjs7O0dBR0c7QUFpQkgsTUFBTSxPQUFPLGdCQUFnQjs7O1lBaEI1QixRQUFRLFNBQUM7Z0JBQ1IsWUFBWSxFQUFFO29CQUNaLHdCQUF3QjtpQkFDekI7Z0JBQ0QsT0FBTyxFQUFFO29CQUNQLFlBQVk7aUJBQ2I7Z0JBQ0QsU0FBUyxFQUFFO29CQUNULGlCQUFpQjtvQkFDakIsb0JBQW9CO29CQUNwQiwyQkFBMkI7aUJBQzVCO2dCQUNELE9BQU8sRUFBRTtvQkFDUCx3QkFBd0I7aUJBQ3pCO2FBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IFZvaWNlQWdlbnRNb2RhbENvbXBvbmVudCB9IGZyb20gJy4vY29tcG9uZW50cy92b2ljZS1hZ2VudC1tb2RhbC92b2ljZS1hZ2VudC1tb2RhbC5jb21wb25lbnQnO1xuaW1wb3J0IHsgVm9pY2VBZ2VudFNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL3ZvaWNlLWFnZW50LnNlcnZpY2UnO1xuaW1wb3J0IHsgQXVkaW9BbmFseXplclNlcnZpY2UgfSBmcm9tICcuL3NlcnZpY2VzL2F1ZGlvLWFuYWx5emVyLnNlcnZpY2UnO1xuaW1wb3J0IHsgV2ViU29ja2V0Vm9pY2VDbGllbnRTZXJ2aWNlIH0gZnJvbSAnLi9zZXJ2aWNlcy93ZWJzb2NrZXQtdm9pY2UtY2xpZW50LnNlcnZpY2UnO1xuXG4vKipcbiAqIFZvaWNlIGFnZW50IG1vZHVsZS4gVXNlcyBuYXRpdmUgV2ViU29ja2V0IGZvciB0aGUgdm9pY2Ugc2Vzc2lvbi5cbiAqIERvZXMgTk9UIHVzZSBTb2NrZXQuSU8gb3Igbmd4LXNvY2tldC1pby5cbiAqL1xuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbXG4gICAgVm9pY2VBZ2VudE1vZGFsQ29tcG9uZW50XG4gIF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGVcbiAgXSxcbiAgcHJvdmlkZXJzOiBbXG4gICAgVm9pY2VBZ2VudFNlcnZpY2UsXG4gICAgQXVkaW9BbmFseXplclNlcnZpY2UsXG4gICAgV2ViU29ja2V0Vm9pY2VDbGllbnRTZXJ2aWNlXG4gIF0sXG4gIGV4cG9ydHM6IFtcbiAgICBWb2ljZUFnZW50TW9kYWxDb21wb25lbnRcbiAgXVxufSlcbmV4cG9ydCBjbGFzcyBWb2ljZUFnZW50TW9kdWxlIHsgfVxuIl19