@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.
- package/bundles/hivegpt-hiveai-angular.umd.js +285 -503
- 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 +118 -85
- package/esm2015/lib/components/voice-agent/services/websocket-voice-client.service.js +114 -46
- package/esm2015/lib/components/voice-agent/voice-agent.module.js +3 -5
- package/fesm2015/hivegpt-hiveai-angular.js +223 -422
- 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 +11 -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 +23 -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,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
|
|
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
|
-
*
|
|
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).
|
|
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
|
-
|
|
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
|
-
|
|
21
|
-
this.
|
|
22
|
-
/**
|
|
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
|
-
|
|
39
|
-
this.ws
|
|
40
|
-
|
|
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
|
-
|
|
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 (
|
|
60
|
-
// Ignore non-JSON
|
|
61
|
+
catch (_a) {
|
|
62
|
+
// Ignore non-JSON
|
|
61
63
|
}
|
|
62
64
|
};
|
|
63
|
-
|
|
64
|
-
this.
|
|
65
|
+
socket.onerror = () => {
|
|
66
|
+
this.ngZone.run(() => {
|
|
67
|
+
if (this.ws === socket && socket.readyState !== WebSocket.CLOSED) {
|
|
68
|
+
socket.close();
|
|
69
|
+
}
|
|
70
|
+
});
|
|
65
71
|
};
|
|
66
|
-
|
|
67
|
-
this.ws
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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":"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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\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
|
|
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
|