@hivegpt/hiveai-angular 0.0.574 → 0.0.576
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 +542 -278
- package/bundles/hivegpt-hiveai-angular.umd.js.map +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js +1 -1
- package/bundles/hivegpt-hiveai-angular.umd.min.js.map +1 -1
- package/esm2015/lib/components/chat-drawer/chat-drawer.component.js +6 -6
- package/esm2015/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.js +85 -54
- package/esm2015/lib/components/voice-agent/services/daily-voice-client.service.js +153 -63
- package/esm2015/lib/components/voice-agent/services/voice-agent.service.js +160 -88
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +81 -34
- package/fesm2015/hivegpt-hiveai-angular.js +478 -239
- package/fesm2015/hivegpt-hiveai-angular.js.map +1 -1
- package/hivegpt-hiveai-angular.metadata.json +1 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts +7 -1
- package/lib/components/voice-agent/components/voice-agent-modal/voice-agent-modal.component.d.ts.map +1 -1
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts +37 -3
- package/lib/components/voice-agent/services/daily-voice-client.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/voice-agent.service.d.ts +19 -6
- package/lib/components/voice-agent/services/voice-agent.service.d.ts.map +1 -1
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts +5 -12
- package/lib/components/voice-agent/services/websocket-voice-client.service.d.ts.map +1 -1
- package/package.json +1 -1
|
@@ -11,6 +11,7 @@ import * as i0 from "@angular/core";
|
|
|
11
11
|
* - Emit roomCreated$, userTranscript$, botTranscript$
|
|
12
12
|
* - NO audio logic, NO mic logic. Audio is handled by Daily.js (WebRTC).
|
|
13
13
|
*/
|
|
14
|
+
const WS_CONNECT_TIMEOUT_MS = 10000;
|
|
14
15
|
export class WebSocketVoiceClientService {
|
|
15
16
|
constructor() {
|
|
16
17
|
this.ws = null;
|
|
@@ -24,54 +25,100 @@ export class WebSocketVoiceClientService {
|
|
|
24
25
|
/** Emits bot transcript updates. */
|
|
25
26
|
this.botTranscript$ = this.botTranscriptSubject.asObservable();
|
|
26
27
|
}
|
|
27
|
-
/**
|
|
28
|
+
/**
|
|
29
|
+
* Connect to signaling WebSocket. No audio over this connection.
|
|
30
|
+
* Resolves when the socket is open; rejects if the connection fails.
|
|
31
|
+
*/
|
|
28
32
|
connect(wsUrl) {
|
|
29
33
|
var _a;
|
|
30
34
|
if (((_a = this.ws) === null || _a === void 0 ? void 0 : _a.readyState) === WebSocket.OPEN) {
|
|
31
|
-
return;
|
|
35
|
+
return Promise.resolve();
|
|
32
36
|
}
|
|
33
37
|
if (this.ws) {
|
|
34
38
|
this.ws.close();
|
|
35
39
|
this.ws = null;
|
|
36
40
|
}
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
41
|
+
return new Promise((resolve, reject) => {
|
|
42
|
+
let settled = false;
|
|
43
|
+
const timeout = setTimeout(() => {
|
|
40
44
|
var _a;
|
|
45
|
+
if (settled)
|
|
46
|
+
return;
|
|
47
|
+
settled = true;
|
|
41
48
|
try {
|
|
42
|
-
|
|
43
|
-
if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'room_created') {
|
|
44
|
-
const roomUrl = ((_a = msg.room_url) !== null && _a !== void 0 ? _a : msg.roomUrl);
|
|
45
|
-
if (typeof roomUrl === 'string') {
|
|
46
|
-
this.roomCreatedSubject.next(roomUrl);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'user_transcript' && typeof msg.text === 'string') {
|
|
50
|
-
this.userTranscriptSubject.next({
|
|
51
|
-
text: msg.text,
|
|
52
|
-
final: msg.final === true,
|
|
53
|
-
});
|
|
54
|
-
}
|
|
55
|
-
else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'bot_transcript' && typeof msg.text === 'string') {
|
|
56
|
-
this.botTranscriptSubject.next(msg.text);
|
|
57
|
-
}
|
|
49
|
+
(_a = this.ws) === null || _a === void 0 ? void 0 : _a.close();
|
|
58
50
|
}
|
|
59
51
|
catch (_b) {
|
|
60
|
-
|
|
52
|
+
/* ignore */
|
|
61
53
|
}
|
|
62
|
-
};
|
|
63
|
-
this.ws.onerror = () => {
|
|
64
|
-
this.disconnect();
|
|
65
|
-
};
|
|
66
|
-
this.ws.onclose = () => {
|
|
67
54
|
this.ws = null;
|
|
55
|
+
reject(new Error('WebSocket connection timed out'));
|
|
56
|
+
}, WS_CONNECT_TIMEOUT_MS);
|
|
57
|
+
const clear = () => {
|
|
58
|
+
clearTimeout(timeout);
|
|
68
59
|
};
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
60
|
+
try {
|
|
61
|
+
const ws = new WebSocket(wsUrl);
|
|
62
|
+
this.ws = ws;
|
|
63
|
+
ws.onopen = () => {
|
|
64
|
+
if (settled)
|
|
65
|
+
return;
|
|
66
|
+
settled = true;
|
|
67
|
+
clear();
|
|
68
|
+
resolve();
|
|
69
|
+
};
|
|
70
|
+
ws.onmessage = (event) => {
|
|
71
|
+
var _a;
|
|
72
|
+
try {
|
|
73
|
+
const msg = JSON.parse(event.data);
|
|
74
|
+
if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'room_created') {
|
|
75
|
+
const roomUrl = ((_a = msg.room_url) !== null && _a !== void 0 ? _a : msg.roomUrl);
|
|
76
|
+
if (typeof roomUrl === 'string') {
|
|
77
|
+
this.roomCreatedSubject.next(roomUrl);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'user_transcript' && typeof msg.text === 'string') {
|
|
81
|
+
this.userTranscriptSubject.next({
|
|
82
|
+
text: msg.text,
|
|
83
|
+
final: msg.final === true,
|
|
84
|
+
});
|
|
85
|
+
}
|
|
86
|
+
else if ((msg === null || msg === void 0 ? void 0 : msg.type) === 'bot_transcript' && typeof msg.text === 'string') {
|
|
87
|
+
this.botTranscriptSubject.next(msg.text);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
catch (_b) {
|
|
91
|
+
// Ignore non-JSON or unknown messages
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
ws.onerror = () => {
|
|
95
|
+
if (!settled) {
|
|
96
|
+
settled = true;
|
|
97
|
+
clear();
|
|
98
|
+
this.disconnect();
|
|
99
|
+
reject(new Error('WebSocket connection failed'));
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
// After onopen, some environments fire onerror spuriously; closing here can
|
|
103
|
+
// kill the socket before room_created is delivered. Let onclose clean up.
|
|
104
|
+
console.warn('WebSocketVoiceClient: onerror after open (not forcing disconnect)');
|
|
105
|
+
};
|
|
106
|
+
ws.onclose = () => {
|
|
107
|
+
this.ws = null;
|
|
108
|
+
if (!settled) {
|
|
109
|
+
settled = true;
|
|
110
|
+
clear();
|
|
111
|
+
reject(new Error('WebSocket connection failed'));
|
|
112
|
+
}
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
catch (err) {
|
|
116
|
+
clear();
|
|
117
|
+
console.error('WebSocketVoiceClient: connect failed', err);
|
|
118
|
+
this.ws = null;
|
|
119
|
+
reject(err instanceof Error ? err : new Error(String(err)));
|
|
120
|
+
}
|
|
121
|
+
});
|
|
75
122
|
}
|
|
76
123
|
/** Disconnect and cleanup. */
|
|
77
124
|
disconnect() {
|
|
@@ -92,4 +139,4 @@ WebSocketVoiceClientService.decorators = [
|
|
|
92
139
|
providedIn: 'root',
|
|
93
140
|
},] }
|
|
94
141
|
];
|
|
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"]}
|
|
142
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-voice-client.service.js","sourceRoot":"/Users/rohitthakur/hive-gpt/HiveAI-Packages/Angular/projects/hivegpt/eventsgpt-angular/src/","sources":["lib/components/voice-agent/services/websocket-voice-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\n\n/** WebSocket message types from backend signaling. */\nexport interface WsMessageRoomCreated {\n  type: 'room_created';\n  room_url: string;\n}\n\nexport interface WsMessageUserTranscript {\n  type: 'user_transcript';\n  text: string;\n  final?: boolean;\n}\n\nexport interface WsMessageBotTranscript {\n  type: 'bot_transcript';\n  text: string;\n}\n\nexport type WsMessage =\n  | WsMessageRoomCreated\n  | WsMessageUserTranscript\n  | WsMessageBotTranscript;\n\nexport interface TranscriptData {\n  text: string;\n  final: boolean;\n}\n\n/**\n * WebSocket-only client for voice agent signaling.\n * CRITICAL: Uses native WebSocket only. NO Socket.IO, NO ngx-socket-io.\n *\n * Responsibilities:\n * - Connect to ws_url (from POST /ai/ask-voice response)\n * - Parse JSON messages (room_created, user_transcript, bot_transcript)\n * - Emit roomCreated$, userTranscript$, botTranscript$\n * - NO audio logic, NO mic logic. Audio is handled by Daily.js (WebRTC).\n */\nconst WS_CONNECT_TIMEOUT_MS = 10_000;\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class WebSocketVoiceClientService {\n  private ws: WebSocket | null = null;\n  private roomCreatedSubject = new Subject<string>();\n  private userTranscriptSubject = new Subject<TranscriptData>();\n  private botTranscriptSubject = new Subject<string>();\n\n  /** Emits room_url when backend sends room_created. */\n  roomCreated$: Observable<string> = this.roomCreatedSubject.asObservable();\n\n  /** Emits user transcript updates. */\n  userTranscript$: Observable<TranscriptData> =\n    this.userTranscriptSubject.asObservable();\n\n  /** Emits bot transcript updates. */\n  botTranscript$: Observable<string> =\n    this.botTranscriptSubject.asObservable();\n\n  /**\n   * Connect to signaling WebSocket. No audio over this connection.\n   * Resolves when the socket is open; rejects if the connection fails.\n   */\n  connect(wsUrl: string): Promise<void> {\n    if (this.ws?.readyState === WebSocket.OPEN) {\n      return Promise.resolve();\n    }\n    if (this.ws) {\n      this.ws.close();\n      this.ws = null;\n    }\n\n    return new Promise((resolve, reject) => {\n      let settled = false;\n      const timeout = setTimeout(() => {\n        if (settled) return;\n        settled = true;\n        try {\n          this.ws?.close();\n        } catch {\n          /* ignore */\n        }\n        this.ws = null;\n        reject(new Error('WebSocket connection timed out'));\n      }, WS_CONNECT_TIMEOUT_MS);\n\n      const clear = (): void => {\n        clearTimeout(timeout);\n      };\n\n      try {\n        const ws = new WebSocket(wsUrl);\n        this.ws = ws;\n\n        ws.onopen = () => {\n          if (settled) return;\n          settled = true;\n          clear();\n          resolve();\n        };\n\n        ws.onmessage = (event: MessageEvent) => {\n          try {\n            const msg = JSON.parse(event.data) as Record<string, unknown>;\n            if (msg?.type === 'room_created') {\n              const roomUrl = (msg.room_url ?? msg.roomUrl) as string | undefined;\n              if (typeof roomUrl === 'string') {\n                this.roomCreatedSubject.next(roomUrl);\n              }\n            } else if (msg?.type === 'user_transcript' && typeof msg.text === 'string') {\n              this.userTranscriptSubject.next({\n                text: msg.text,\n                final: msg.final === true,\n              });\n            } else if (msg?.type === 'bot_transcript' && typeof msg.text === 'string') {\n              this.botTranscriptSubject.next(msg.text);\n            }\n          } catch {\n            // Ignore non-JSON or unknown messages\n          }\n        };\n\n        ws.onerror = () => {\n          if (!settled) {\n            settled = true;\n            clear();\n            this.disconnect();\n            reject(new Error('WebSocket connection failed'));\n            return;\n          }\n          // After onopen, some environments fire onerror spuriously; closing here can\n          // kill the socket before room_created is delivered. Let onclose clean up.\n          console.warn(\n            'WebSocketVoiceClient: onerror after open (not forcing disconnect)',\n          );\n        };\n\n        ws.onclose = () => {\n          this.ws = null;\n          if (!settled) {\n            settled = true;\n            clear();\n            reject(new Error('WebSocket connection failed'));\n          }\n        };\n      } catch (err) {\n        clear();\n        console.error('WebSocketVoiceClient: connect failed', err);\n        this.ws = null;\n        reject(err instanceof Error ? err : new Error(String(err)));\n      }\n    });\n  }\n\n  /** Disconnect and cleanup. */\n  disconnect(): void {\n    if (this.ws) {\n      this.ws.close();\n      this.ws = null;\n    }\n  }\n\n  /** Whether the WebSocket is open. */\n  get isConnected(): boolean {\n    return this.ws?.readyState === WebSocket.OPEN;\n  }\n}\n"]}
|