@sinequa/assistant 3.8.0 → 3.9.0

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 (151) hide show
  1. package/chat/charts/chart/chart.component.d.ts +13 -13
  2. package/chat/chat-message/chat-message.component.d.ts +85 -81
  3. package/chat/chat-message/i18n/en.json +11 -0
  4. package/chat/chat-message/i18n/fr.json +11 -0
  5. package/chat/chat-reference/chat-reference.component.d.ts +14 -14
  6. package/chat/chat-reference/i18n/en.json +4 -0
  7. package/chat/chat-reference/i18n/fr.json +4 -0
  8. package/chat/chat-settings-v3/chat-settings-v3.component.d.ts +48 -50
  9. package/chat/chat-settings-v3/i18n/en.json +14 -0
  10. package/chat/chat-settings-v3/i18n/fr.json +14 -0
  11. package/chat/chat.component.d.ts +388 -1405
  12. package/chat/chat.service.d.ts +247 -228
  13. package/chat/debug-message/debug-message.component.d.ts +17 -17
  14. package/chat/debug-message/i18n/en.json +3 -0
  15. package/chat/debug-message/i18n/fr.json +3 -0
  16. package/chat/dialogs/delete-saved-chat.component.d.ts +22 -0
  17. package/chat/dialogs/i18n/en.json +19 -0
  18. package/chat/dialogs/i18n/fr.json +19 -0
  19. package/chat/dialogs/rename-saved-chat.component.d.ts +21 -0
  20. package/chat/dialogs/updates.component.d.ts +15 -0
  21. package/chat/documents-upload/document-list/document-list.component.d.ts +68 -77
  22. package/chat/documents-upload/document-overview/document-overview.component.d.ts +31 -41
  23. package/chat/documents-upload/document-upload/document-upload.component.d.ts +96 -98
  24. package/chat/documents-upload/documents-upload.model.d.ts +66 -66
  25. package/chat/documents-upload/documents-upload.service.d.ts +170 -174
  26. package/chat/documents-upload/i18n/en.json +24 -0
  27. package/chat/documents-upload/i18n/fr.json +24 -0
  28. package/chat/format-icon/format-icon.component.d.ts +10 -10
  29. package/chat/format-icon/icons.d.ts +5 -5
  30. package/chat/i18n/en.json +42 -0
  31. package/chat/i18n/fr.json +42 -0
  32. package/chat/index.d.ts +5 -5
  33. package/chat/initials-avatar/initials-avatar.component.d.ts +35 -35
  34. package/chat/instance-manager.service.d.ts +28 -28
  35. package/chat/pipes/message-content.pipe.d.ts +16 -0
  36. package/chat/prompt.component.d.ts +25 -21
  37. package/chat/public-api.d.ts +17 -17
  38. package/chat/references/i18n/en.json +6 -0
  39. package/chat/references/i18n/fr.json +6 -0
  40. package/chat/references/inline-image-reference.d.ts +21 -0
  41. package/chat/references/inline-page-reference.d.ts +21 -0
  42. package/chat/rest-chat.service.d.ts +31 -33
  43. package/chat/saved-chats/i18n/en.json +4 -0
  44. package/chat/saved-chats/i18n/fr.json +4 -0
  45. package/chat/saved-chats/saved-chats.component.d.ts +30 -36
  46. package/chat/services/app.service.d.ts +8 -0
  47. package/chat/services/dialog.service.d.ts +12 -0
  48. package/chat/services/notification.service.d.ts +10 -0
  49. package/chat/services/principal.service.d.ts +7 -0
  50. package/chat/services/search.service.d.ts +7 -0
  51. package/chat/services/signalR.web.service.d.ts +45 -0
  52. package/chat/services/ui.service.d.ts +13 -0
  53. package/chat/services/user-settings.service.d.ts +7 -0
  54. package/chat/token-progress-bar/i18n/en.json +4 -0
  55. package/chat/token-progress-bar/i18n/fr.json +4 -0
  56. package/chat/token-progress-bar/token-progress-bar.component.d.ts +24 -27
  57. package/chat/tooltip/tooltip.component.d.ts +12 -0
  58. package/chat/tooltip/tooltip.directive.d.ts +81 -0
  59. package/chat/types/message-content.types.d.ts +54 -0
  60. package/chat/types/message-reference.types.d.ts +11 -0
  61. package/chat/types.d.ts +913 -873
  62. package/chat/unified-plugins/embedded-image-reference.plugin.d.ts +3 -0
  63. package/chat/unified-plugins/embedded-page-reference.plugin.d.ts +3 -0
  64. package/chat/utils/assistant-json.d.ts +2 -0
  65. package/chat/websocket-chat.service.d.ts +102 -103
  66. package/esm2022/chat/charts/chart/chart.component.mjs +40 -0
  67. package/esm2022/chat/chat-message/chat-message.component.mjs +351 -0
  68. package/esm2022/chat/chat-reference/chat-reference.component.mjs +40 -0
  69. package/esm2022/chat/chat-settings-v3/chat-settings-v3.component.mjs +118 -0
  70. package/esm2022/chat/chat.component.mjs +1090 -0
  71. package/esm2022/chat/chat.service.mjs +417 -0
  72. package/esm2022/chat/debug-message/debug-message.component.mjs +43 -0
  73. package/esm2022/chat/dialogs/delete-saved-chat.component.mjs +81 -0
  74. package/esm2022/chat/dialogs/rename-saved-chat.component.mjs +84 -0
  75. package/esm2022/chat/dialogs/updates.component.mjs +61 -0
  76. package/esm2022/chat/documents-upload/document-list/document-list.component.mjs +140 -0
  77. package/esm2022/chat/documents-upload/document-overview/document-overview.component.mjs +65 -0
  78. package/esm2022/chat/documents-upload/document-upload/document-upload.component.mjs +256 -0
  79. package/{esm2020 → esm2022}/chat/documents-upload/documents-upload.model.mjs +1 -1
  80. package/esm2022/chat/documents-upload/documents-upload.service.mjs +291 -0
  81. package/{esm2020 → esm2022}/chat/format-icon/format-icon.component.mjs +23 -23
  82. package/{esm2020 → esm2022}/chat/format-icon/icons.mjs +137 -137
  83. package/{esm2020 → esm2022}/chat/initials-avatar/initials-avatar.component.mjs +60 -60
  84. package/esm2022/chat/instance-manager.service.mjs +46 -0
  85. package/esm2022/chat/pipes/message-content.pipe.mjs +34 -0
  86. package/esm2022/chat/prompt.component.mjs +88 -0
  87. package/{esm2020 → esm2022}/chat/public-api.mjs +18 -18
  88. package/esm2022/chat/references/inline-image-reference.mjs +110 -0
  89. package/esm2022/chat/references/inline-page-reference.mjs +110 -0
  90. package/esm2022/chat/rest-chat.service.mjs +296 -0
  91. package/esm2022/chat/saved-chats/saved-chats.component.mjs +82 -0
  92. package/esm2022/chat/services/app.service.mjs +19 -0
  93. package/esm2022/chat/services/dialog.service.mjs +40 -0
  94. package/esm2022/chat/services/notification.service.mjs +25 -0
  95. package/esm2022/chat/services/principal.service.mjs +16 -0
  96. package/esm2022/chat/services/search.service.mjs +13 -0
  97. package/esm2022/chat/services/signalR.web.service.mjs +79 -0
  98. package/esm2022/chat/services/ui.service.mjs +61 -0
  99. package/esm2022/chat/services/user-settings.service.mjs +22 -0
  100. package/{esm2020 → esm2022}/chat/sinequa-assistant-chat.mjs +4 -4
  101. package/esm2022/chat/token-progress-bar/token-progress-bar.component.mjs +52 -0
  102. package/esm2022/chat/tooltip/tooltip.component.mjs +44 -0
  103. package/esm2022/chat/tooltip/tooltip.directive.mjs +203 -0
  104. package/esm2022/chat/types/message-content.types.mjs +2 -0
  105. package/esm2022/chat/types/message-reference.types.mjs +2 -0
  106. package/esm2022/chat/types.mjs +130 -0
  107. package/esm2022/chat/unified-plugins/embedded-image-reference.plugin.mjs +57 -0
  108. package/esm2022/chat/unified-plugins/embedded-page-reference.plugin.mjs +57 -0
  109. package/esm2022/chat/utils/assistant-json.mjs +12 -0
  110. package/esm2022/chat/websocket-chat.service.mjs +654 -0
  111. package/{esm2020 → esm2022}/public-api.mjs +2 -2
  112. package/{esm2020 → esm2022}/sinequa-assistant.mjs +4 -4
  113. package/fesm2022/sinequa-assistant-chat.mjs +5340 -0
  114. package/fesm2022/sinequa-assistant-chat.mjs.map +1 -0
  115. package/{fesm2015 → fesm2022}/sinequa-assistant.mjs +3 -3
  116. package/index.d.ts +5 -5
  117. package/package.json +52 -25
  118. package/public-api.d.ts +1 -1
  119. package/chat/messages/de.d.ts +0 -4
  120. package/chat/messages/en.d.ts +0 -4
  121. package/chat/messages/fr.d.ts +0 -4
  122. package/chat/messages/index.d.ts +0 -4
  123. package/esm2020/chat/charts/chart/chart.component.mjs +0 -40
  124. package/esm2020/chat/chat-message/chat-message.component.mjs +0 -263
  125. package/esm2020/chat/chat-reference/chat-reference.component.mjs +0 -40
  126. package/esm2020/chat/chat-settings-v3/chat-settings-v3.component.mjs +0 -117
  127. package/esm2020/chat/chat.component.mjs +0 -1069
  128. package/esm2020/chat/chat.service.mjs +0 -333
  129. package/esm2020/chat/debug-message/debug-message.component.mjs +0 -43
  130. package/esm2020/chat/documents-upload/document-list/document-list.component.mjs +0 -191
  131. package/esm2020/chat/documents-upload/document-overview/document-overview.component.mjs +0 -80
  132. package/esm2020/chat/documents-upload/document-upload/document-upload.component.mjs +0 -258
  133. package/esm2020/chat/documents-upload/documents-upload.service.mjs +0 -289
  134. package/esm2020/chat/instance-manager.service.mjs +0 -46
  135. package/esm2020/chat/messages/de.mjs +0 -4
  136. package/esm2020/chat/messages/en.mjs +0 -4
  137. package/esm2020/chat/messages/fr.mjs +0 -4
  138. package/esm2020/chat/messages/index.mjs +0 -9
  139. package/esm2020/chat/prompt.component.mjs +0 -88
  140. package/esm2020/chat/rest-chat.service.mjs +0 -241
  141. package/esm2020/chat/saved-chats/saved-chats.component.mjs +0 -175
  142. package/esm2020/chat/token-progress-bar/token-progress-bar.component.mjs +0 -54
  143. package/esm2020/chat/types.mjs +0 -112
  144. package/esm2020/chat/websocket-chat.service.mjs +0 -641
  145. package/fesm2015/sinequa-assistant-chat.mjs +0 -4200
  146. package/fesm2015/sinequa-assistant-chat.mjs.map +0 -1
  147. package/fesm2020/sinequa-assistant-chat.mjs +0 -4171
  148. package/fesm2020/sinequa-assistant-chat.mjs.map +0 -1
  149. package/fesm2020/sinequa-assistant.mjs +0 -9
  150. package/fesm2020/sinequa-assistant.mjs.map +0 -1
  151. /package/{fesm2015 → fesm2022}/sinequa-assistant.mjs.map +0 -0
@@ -0,0 +1,654 @@
1
+ import { Injectable, inject } from "@angular/core";
2
+ import { HttpTransportType, LogLevel } from "@microsoft/signalr";
3
+ import { Observable, Subject, catchError, defer, forkJoin, from, fromEvent, map, merge, mergeMap, of, switchMap, take, takeUntil, tap, throwError } from "rxjs";
4
+ import { getToken, globalConfig } from "@sinequa/atomic";
5
+ import { ChatService } from "./chat.service";
6
+ import { SignalRWebService } from "./services/signalR.web.service";
7
+ import * as i0 from "@angular/core";
8
+ export class WebSocketChatService extends ChatService {
9
+ constructor() {
10
+ super();
11
+ this._messageHandlers = new Map();
12
+ this._actionMap = new Map();
13
+ this._progress = undefined;
14
+ this._attachments = [];
15
+ this._debugMessages = [];
16
+ this.signalRService = inject(SignalRWebService);
17
+ }
18
+ /**
19
+ * Initialize the assistant process.
20
+ * It includes building and starting a connection, executing parallel requests for models and functions, and handling errors during the process.
21
+ * ⚠️ This method MUST be called ONLY if the user is loggedIn and once when the assistant is initialized.
22
+ *
23
+ * @returns An Observable<boolean> indicating the success of the initialization process.
24
+ */
25
+ init() {
26
+ // Ensure all logic is executed when subscribed to the observable
27
+ return defer(() => {
28
+ this.getRequestsUrl();
29
+ return from(
30
+ // Build the connection
31
+ this.buildConnection()).pipe(tap(() => this.initMessageHandlers()),
32
+ // Start the connection
33
+ switchMap(() => this.startConnection()),
34
+ // Execute parallel requests for models and functions
35
+ switchMap(() => forkJoin([
36
+ this.listModels(),
37
+ this.listFunctions()
38
+ ])),
39
+ // Map the results of parallel requests to a boolean indicating success
40
+ map(([models, functions]) => {
41
+ const result = !!models && !!functions;
42
+ this.initProcess$.next(result);
43
+ return result;
44
+ }),
45
+ // Any errors during the process are caught, logged, and re-thrown to propagate the error further
46
+ catchError((error) => {
47
+ console.error('Error occurred:', error);
48
+ return throwError(() => error);
49
+ }), take(1));
50
+ });
51
+ }
52
+ /**
53
+ * Define the assistant endpoint to use for the websocket requests
54
+ * It can be overridden by the app config
55
+ */
56
+ getRequestsUrl() {
57
+ if (this.assistantConfig$.value.connectionSettings.websocketEndpoint) {
58
+ this.REQUEST_URL = this.assistantConfig$.value.connectionSettings.websocketEndpoint;
59
+ }
60
+ else {
61
+ throw new Error(`The property 'websocketEndpoint' must be provided when attempting to use 'WebSocket' in assistant instance`);
62
+ }
63
+ }
64
+ overrideUser() {
65
+ const { userOverrideActive, userOverride } = globalConfig;
66
+ if (!(userOverrideActive && userOverride)) {
67
+ this.userOverride$.next(false);
68
+ return;
69
+ }
70
+ // Prepare the payload to send to the OverrideUser method
71
+ const data = {
72
+ instanceId: this.chatInstanceId,
73
+ user: userOverride.username,
74
+ domain: userOverride.domain
75
+ };
76
+ // Invoke the OverrideUser method and handle errors
77
+ this.connection.invoke('OverrideUser', data)
78
+ .then((res) => this.userOverride$.next(!!res))
79
+ .catch(error => {
80
+ console.error('Error invoking OverrideUser:', error);
81
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
82
+ });
83
+ }
84
+ listModels() {
85
+ const modelsSubject$ = new Subject();
86
+ this.connection.on('ListModels', (res) => {
87
+ this.models = res.models?.filter(model => !!model.enable);
88
+ modelsSubject$.next(this.models);
89
+ modelsSubject$.complete();
90
+ });
91
+ // Send the request to get the list of models
92
+ this.connection.invoke('ListModels', { debug: this.assistantConfig$.value.defaultValues.debug })
93
+ .catch(error => {
94
+ console.error('Error invoking ListModels:', error);
95
+ modelsSubject$.error(new Error(error));
96
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
97
+ });
98
+ return modelsSubject$.asObservable();
99
+ }
100
+ listFunctions() {
101
+ const functionsSubject$ = new Subject();
102
+ this.connection.on('ListFunctions', (res) => {
103
+ this.functions = res.functions?.filter(func => func.enabled);
104
+ functionsSubject$.next(this.functions);
105
+ functionsSubject$.complete();
106
+ });
107
+ // Send the request to get the list of functions
108
+ this.connection.invoke('ListFunctions', { debug: this.assistantConfig$.value.defaultValues.debug })
109
+ .catch(error => {
110
+ console.error('Error invoking ListFunctions:', error);
111
+ functionsSubject$.error(new Error(error));
112
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
113
+ });
114
+ return functionsSubject$.asObservable();
115
+ }
116
+ fetch(messages, query) {
117
+ // Start streaming by invoking the Chat method
118
+ this.streaming$.next(true);
119
+ // Prepare the payload to send to the Chat method
120
+ const data = {
121
+ history: messages,
122
+ functions: this.assistantConfig$.value.defaultValues.functions?.filter(func => func.enabled).map(func => func.name),
123
+ debug: this.assistantConfig$.value.defaultValues.debug,
124
+ serviceSettings: {
125
+ service_id: this.assistantConfig$.value.defaultValues.service_id,
126
+ model_id: this.assistantConfig$.value.defaultValues.model_id,
127
+ top_p: this.assistantConfig$.value.defaultValues.top_p,
128
+ temperature: this.assistantConfig$.value.defaultValues.temperature,
129
+ max_tokens: this.assistantConfig$.value.defaultValues.max_tokens,
130
+ ...this.assistantConfig$.value.additionalServiceSettings
131
+ },
132
+ appQuery: {
133
+ app: this.appService.appName,
134
+ query
135
+ },
136
+ genericChatErrorMessage: this.assistantConfig$.value.globalSettings.genericChatErrorMessage ? this.transloco.translate(this.assistantConfig$.value.globalSettings.genericChatErrorMessage) : ""
137
+ };
138
+ if (this.assistantConfig$.value.savedChatSettings.enabled) {
139
+ data.instanceId = this.chatInstanceId;
140
+ data.savedChatId = this.savedChatId;
141
+ }
142
+ // Initialize the response with an empty assistant message
143
+ this._response = [{ role: "assistant", content: "", additionalProperties: { display: true } }]; // here display: true is needed in order to be able to show the progress
144
+ // Create a Subject to signal completion
145
+ const completion$ = new Subject();
146
+ // Create observables for each non-global handler in the _messageHandlers map (default and eventual custom ones) once it is triggered by the hub connection
147
+ const observables = Array
148
+ .from(this._messageHandlers.entries())
149
+ .filter(([eventName, eventHandler]) => !eventHandler.isGlobalHandler)
150
+ .map(([eventName, eventHandler]) => fromEvent(this.connection, eventName).pipe(mergeMap((event) => {
151
+ // Wrap the handler in a try-catch block to prevent the entire stream from failing if an error occurs in a single handler
152
+ try {
153
+ // Execute the handler and emit the result
154
+ // NB: here we could use [eventHandler.handler(event)] which behind the scenes mergeMap interprets this array as an observable sequence with one item, which it then emits
155
+ return of(eventHandler.handler(event));
156
+ }
157
+ catch (error) {
158
+ console.error(`Error in event handler for ${eventName}:`, error);
159
+ // Use throwError to propagate the error downstream
160
+ return throwError(() => new Error(`Error in event handler for ${eventName}: ${error}`));
161
+ }
162
+ })));
163
+ // Then merge them into a single observable in order to simulate the streaming behavior
164
+ const combined$ = merge(...observables).pipe(map(() => {
165
+ // Define $progress from the _actionMap
166
+ const actions = Array.from(this._actionMap.values());
167
+ this._progress = actions.length > 0
168
+ ? actions.map((a) => ({
169
+ title: a.displayName ?? "",
170
+ content: a.displayValue ?? "",
171
+ done: a.executionTime !== undefined,
172
+ time: a.executionTime,
173
+ }))
174
+ : undefined;
175
+ // Always update ONLY the first assistant message of the _response with the new $progress, $attachment and $debug
176
+ // Assuming that the first assistant message is always visible since the hub does not send hidden messages by design
177
+ // So even if the first assistant message is hidden (display: false), the _response[0] will and should contain :
178
+ // - $progress, $attachment and $debug
179
+ // - the content of the first visible assistant message in the workflow
180
+ // This is mandatory in order to match the behavior of consecutive messages and maintain consistency with the chatHistory
181
+ if (!!this._progress || this._attachments.length > 0 || this._debugMessages.length > 0) {
182
+ this._response[0].additionalProperties.$progress = this._progress;
183
+ this._response[0].additionalProperties.$attachment = this._attachments;
184
+ this._response[0].additionalProperties.$debug = this._debugMessages;
185
+ }
186
+ // Return the result
187
+ return { history: [...messages, ...this._response], executionTime: this._executionTime, executionTimeMilliseconds: this._executionTimeMilliseconds };
188
+ }), takeUntil(completion$));
189
+ // return a new Observable that emits the result of the combined stream and handles the eventual errors of the invocation of the Chat method
190
+ return new Observable(observer => {
191
+ // Subscribe to combined stream
192
+ combined$.subscribe({
193
+ next: (value) => observer.next(value),
194
+ error: (err) => observer.error(err)
195
+ });
196
+ // Invoke the Chat method and handle errors
197
+ this.connection.invoke('Chat', data)
198
+ .then(() => {
199
+ // If a valid assistant message with (display: true) was found, update it
200
+ // and it should always the case
201
+ const index = this.firstVisibleAssistantMessageIndex(this.chatHistory);
202
+ if (index !== -1) {
203
+ this.chatHistory[index].additionalProperties.$progress = this._progress;
204
+ this.chatHistory[index].additionalProperties.$attachment = this._attachments;
205
+ this.chatHistory[index].additionalProperties.$debug = this._debugMessages;
206
+ }
207
+ // Save/update the chat if savedChat enabled
208
+ if (this.assistantConfig$.value.savedChatSettings.enabled && this.chatHistory.some((msg) => msg.additionalProperties?.isUserInput === true)) {
209
+ const action = !this.savedChatId ? this.addSavedChat(this.chatHistory).pipe(tap(() => this.listSavedChat())) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);
210
+ action.pipe(take(1)).subscribe({
211
+ next: () => { },
212
+ error: (error) => {
213
+ this.streaming$.next(false);
214
+ observer.error(error);
215
+ },
216
+ complete: () => {
217
+ this.streaming$.next(false);
218
+ observer.complete();
219
+ }
220
+ });
221
+ }
222
+ else {
223
+ this.streaming$.next(false);
224
+ observer.complete();
225
+ }
226
+ })
227
+ .catch(error => {
228
+ console.error('Error invoking Chat:', error);
229
+ this.streaming$.next(false);
230
+ // Emit the error to the newly created observable
231
+ observer.error(error);
232
+ // Return a resolved promise to handle the error and prevent unhandled promise rejection
233
+ return Promise.resolve();
234
+ })
235
+ .finally(() => {
236
+ // This block concerns ONLY the completion of the "Chat" method invocation.
237
+ // This means the completion of the combined$ stream.
238
+ // It does not take into account the completion of the entire fetch method (the observable returned by fetch) and which depends on the completion of the save chat action if enabled
239
+ this._response = []; // Clear the _response
240
+ this._actionMap.clear(); // Clear the _actionMap
241
+ this._progress = undefined; // Clear the _progress
242
+ this._attachments = []; // Clear the _attachments
243
+ this._debugMessages = []; // Clear the _debugMessages
244
+ this._executionTime = ""; // Clear the _executionTime
245
+ this._executionTimeMilliseconds = undefined; // Clear the _executionTimeMilliseconds
246
+ completion$.next(); // Emit a signal to complete the observables
247
+ completion$.complete(); // Complete the subject
248
+ });
249
+ });
250
+ }
251
+ stopGeneration() {
252
+ // Start stopping generation by invoking the CancelTasks method
253
+ this.stoppingGeneration$.next(true);
254
+ // Create a Subject to hold the result of the CancelTasks method
255
+ const stopGenerationSubject$ = new Subject();
256
+ this.connection.on('CancelTasks', (res) => {
257
+ // When the generation is stopped before streaming any VISIBLE assistant message, this means that $progress, $attachment and $debug properties will be lost.
258
+ // However, the "ContextMessage" frames will be persisted in the chatHistory and the assistant may reference them in the next generation.
259
+ // This leads to the problem of referencing undisplayed attachments in the next generation.
260
+ // To solve this problem, we need to persist $progress, $attachment and $debug properties by adding a new assistant message with empty content and these properties.
261
+ if (this._response.length === 1 && this._response[0].content === "") {
262
+ this.chatHistory?.push({ role: "assistant", content: "", additionalProperties: { display: true, $progress: this._progress, $attachment: this._attachments, $debug: this._debugMessages } });
263
+ }
264
+ stopGenerationSubject$.next(!!res); // Emit the result of the CancelTasks method
265
+ stopGenerationSubject$.complete(); // Complete the subject
266
+ this.stoppingGeneration$.next(false); // Complete stopping generation
267
+ });
268
+ // Invoke the CancelTasks method and handle errors
269
+ this.connection.invoke('CancelTasks')
270
+ .catch(error => {
271
+ console.error('Error invoking CancelTasks:', error);
272
+ stopGenerationSubject$.error(new Error(error));
273
+ this.stoppingGeneration$.next(false); // Complete stopping generation
274
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
275
+ });
276
+ return stopGenerationSubject$.asObservable();
277
+ }
278
+ listSavedChat() {
279
+ if (!this.assistantConfig$.value.savedChatSettings.enabled) {
280
+ return;
281
+ }
282
+ const data = {
283
+ appName: this.appService.appName,
284
+ instanceId: this.chatInstanceId,
285
+ debug: this.assistantConfig$.value.defaultValues.debug
286
+ };
287
+ this.connection.on('SavedChatList', (res) => {
288
+ this.savedChats$.next(res.savedChats); // emits the result to the savedChats$ subject
289
+ });
290
+ // Invoke the method SavedChatList
291
+ this.connection.invoke('SavedChatList', data)
292
+ .catch(error => {
293
+ console.error('Error invoking SavedChatList:', error);
294
+ return Promise.resolve();
295
+ });
296
+ }
297
+ getSavedChat(id) {
298
+ const savedChatSubject$ = new Subject();
299
+ const data = {
300
+ appName: this.appService.appName,
301
+ instanceId: this.chatInstanceId,
302
+ savedChatId: id,
303
+ debug: this.assistantConfig$.value.defaultValues.debug
304
+ };
305
+ this.connection.on('SavedChatGet', (res) => {
306
+ this.generateAuditEvent('ast-saved-chat.load', { duration: res.executionTimeMilliseconds }, res.savedChat.id);
307
+ savedChatSubject$.next(res.savedChat);
308
+ savedChatSubject$.complete();
309
+ });
310
+ // Invoke the method SavedChatGet
311
+ this.connection.invoke('SavedChatGet', data)
312
+ .catch(error => {
313
+ console.error('Error invoking SavedChatGet:', error);
314
+ savedChatSubject$.error(new Error(error));
315
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
316
+ });
317
+ return savedChatSubject$.asObservable();
318
+ }
319
+ addSavedChat(messages) {
320
+ const addSavedChatSubject$ = new Subject();
321
+ const data = {
322
+ appName: this.appService.appName,
323
+ instanceId: this.chatInstanceId,
324
+ savedChatId: this.chatId,
325
+ history: messages,
326
+ debug: this.assistantConfig$.value.defaultValues.debug
327
+ };
328
+ this.connection.on('SavedChatAdd', (res) => {
329
+ this.setSavedChatId(res.savedChat.id); // Persist the savedChatId
330
+ this.generateAuditEvent('ast-saved-chat.add', { duration: res.executionTimeMilliseconds }, res.savedChat.id); // Generate audit event
331
+ addSavedChatSubject$.next(res);
332
+ addSavedChatSubject$.complete();
333
+ });
334
+ // Invoke the method SavedChatAdd
335
+ this.connection.invoke('SavedChatAdd', data)
336
+ .catch(error => {
337
+ console.error('Error invoking SavedChatAdd:', error);
338
+ addSavedChatSubject$.error(new Error(error));
339
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
340
+ });
341
+ return addSavedChatSubject$.asObservable();
342
+ }
343
+ updateSavedChat(id, name, messages) {
344
+ const updateSavedChatSubject$ = new Subject();
345
+ const data = {
346
+ appName: this.appService.appName,
347
+ instanceId: this.chatInstanceId,
348
+ savedChatId: id,
349
+ debug: this.assistantConfig$.value.defaultValues.debug
350
+ };
351
+ if (name)
352
+ data["title"] = name;
353
+ if (messages)
354
+ data["history"] = messages;
355
+ this.connection.on('SavedChatUpdate', (res) => {
356
+ updateSavedChatSubject$.next(res);
357
+ updateSavedChatSubject$.complete();
358
+ });
359
+ // Invoke the method SavedChatUpdate
360
+ this.connection.invoke('SavedChatUpdate', data)
361
+ .catch(error => {
362
+ console.error('Error invoking SavedChatUpdate:', error);
363
+ updateSavedChatSubject$.error(new Error(error));
364
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
365
+ });
366
+ return updateSavedChatSubject$.asObservable();
367
+ }
368
+ deleteSavedChat(ids) {
369
+ const deleteSavedChatSubject$ = new Subject();
370
+ const data = {
371
+ appName: this.appService.appName,
372
+ instanceId: this.chatInstanceId,
373
+ SavedChatIds: ids,
374
+ debug: this.assistantConfig$.value.defaultValues.debug
375
+ };
376
+ this.connection.on('SavedChatDelete', (res) => {
377
+ deleteSavedChatSubject$.next(res);
378
+ deleteSavedChatSubject$.complete();
379
+ });
380
+ // Invoke the method SavedChatDelete
381
+ this.connection.invoke('SavedChatDelete', data)
382
+ .catch(error => {
383
+ console.error('Error invoking SavedChatDelete:', error);
384
+ deleteSavedChatSubject$.error(new Error(error));
385
+ return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream
386
+ });
387
+ return deleteSavedChatSubject$.asObservable();
388
+ }
389
+ /**
390
+ * Initialize out-of-the-box handlers
391
+ * It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with frame message(s)
392
+ */
393
+ initMessageHandlers() {
394
+ this.addMessageHandler("Error", {
395
+ handler: (error) => {
396
+ console.error(error);
397
+ this.notificationsService.error(error);
398
+ },
399
+ isGlobalHandler: true
400
+ });
401
+ this.addMessageHandler("Quota", {
402
+ handler: (message) => {
403
+ try {
404
+ this.updateQuota(message.quota);
405
+ }
406
+ catch (error) {
407
+ console.error(error);
408
+ }
409
+ },
410
+ isGlobalHandler: true
411
+ });
412
+ this.addMessageHandler("Debug", { handler: () => { },
413
+ isGlobalHandler: true
414
+ });
415
+ this.addMessageHandler("ActionStart", { handler: (action) => this._actionMap.set(action.guid, action),
416
+ isGlobalHandler: false });
417
+ this.addMessageHandler("ActionResult", {
418
+ handler: (action) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),
419
+ isGlobalHandler: false
420
+ });
421
+ this.addMessageHandler("ActionStop", {
422
+ handler: (action) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),
423
+ isGlobalHandler: false
424
+ });
425
+ this.addMessageHandler("ContextMessage", {
426
+ handler: (message) => {
427
+ this._attachments.push(message.additionalProperties);
428
+ },
429
+ isGlobalHandler: false
430
+ });
431
+ this.addMessageHandler("Message", {
432
+ handler: (message) => this._response.at(-1).content += message.delta ?? "",
433
+ isGlobalHandler: false
434
+ });
435
+ this.addMessageHandler("History", {
436
+ handler: (history) => {
437
+ // The ChatHistory is updated: it is the current copy concatenated with the new items ONLY (it can have multiple messages: the context messages + the response message)
438
+ // This is mandatory to not lose the previous updates of the chatHistory when the assistant is streaming multiple message steps
439
+ this.chatHistory = [...this.chatHistory, ...(history.history.slice(this.chatHistory.length))];
440
+ // Emit the updated chat usage metrics
441
+ if (!!this.chatHistory.at(-1)?.additionalProperties.usageMetrics) {
442
+ this.updateChatUsageMetrics(this.chatHistory.at(-1).additionalProperties.usageMetrics);
443
+ }
444
+ this._executionTime = history.executionTime;
445
+ this._executionTimeMilliseconds = history.executionTimeMilliseconds;
446
+ },
447
+ isGlobalHandler: false
448
+ });
449
+ this.addMessageHandler("SuggestedActions", {
450
+ handler: (message) => {
451
+ // Since after the "History" and "MessageBreak" that this event is caught,
452
+ // $suggestedAction needs to be updated directly to the last visible "assistant" message in the _response and the chatHistory
453
+ this._response.at(-1).additionalProperties.$suggestedAction = (this._response.at(-1).additionalProperties.$suggestedAction || []).concat(message.suggestedActions);
454
+ const index = this.lastVisibleAssistantMessageIndex(this.chatHistory);
455
+ if (index !== -1) {
456
+ this.chatHistory[index].additionalProperties.$suggestedAction = (this.chatHistory[index].additionalProperties.$suggestedAction || []).concat(message.suggestedActions);
457
+ }
458
+ },
459
+ isGlobalHandler: false
460
+ });
461
+ this.addMessageHandler("DebugDisplay", {
462
+ handler: (message) => this._debugMessages = this._debugMessages.concat(message),
463
+ isGlobalHandler: false
464
+ });
465
+ this.addMessageHandler("MessageBreak", {
466
+ handler: () => {
467
+ // Generate audit event
468
+ const details = {
469
+ 'duration': this._executionTimeMilliseconds !== undefined ? this._executionTimeMilliseconds : this._executionTime,
470
+ 'role': this.chatHistory.at(-1).role, // 'assistant'
471
+ 'rank': this.chatHistory.length - 1,
472
+ 'generation-tokencount': this.chatHistory.at(-1).additionalProperties.usageMetrics?.completionTokenCount,
473
+ 'prompt-tokencount': this.chatHistory.at(-1).additionalProperties.usageMetrics?.promptTokenCount,
474
+ 'attachments': JSON.stringify(this._attachments.map(({ recordId, contextId, parts, type }) => ({
475
+ recordId,
476
+ contextId,
477
+ parts: parts.map(({ partId, text }) => {
478
+ if (!!this.assistantConfig$.value?.auditSettings?.logContent)
479
+ return { partId, text };
480
+ return { partId };
481
+ }),
482
+ type
483
+ })))
484
+ };
485
+ if (!!this.assistantConfig$.value?.auditSettings?.logContent)
486
+ details['text'] = this.chatHistory.at(-1).content;
487
+ this.generateAuditEvent('ast-message', details);
488
+ // Push a new assistant message to the _response array ONLY if the content of the last message is not empty
489
+ if (this._response.at(-1).content !== "") {
490
+ this._response.push({ role: "assistant", content: "", additionalProperties: { display: true } });
491
+ }
492
+ },
493
+ isGlobalHandler: false
494
+ });
495
+ }
496
+ /**
497
+ * Override and register the entire _messageHandlers map by merging the provided map with the default one
498
+ * @param _messageHandlers
499
+ */
500
+ overrideMessageHandlers(_messageHandlers) {
501
+ // Clear the already registered global chat handlers before merging the new ones
502
+ this._messageHandlers.forEach((eventHandler, eventName) => {
503
+ if (eventHandler.isGlobalHandler) {
504
+ this.unsubscribeMessageHandler(eventName);
505
+ }
506
+ });
507
+ // Merge the new event handlers with the existing ones
508
+ this._messageHandlers = new Map([...this._messageHandlers, ..._messageHandlers]);
509
+ // Register the global handlers among the merged map
510
+ this._messageHandlers.forEach((eventHandler, eventName) => {
511
+ if (eventHandler.isGlobalHandler) {
512
+ this.registerMessageHandler(eventName, eventHandler);
513
+ }
514
+ });
515
+ }
516
+ /**
517
+ * Add a listener for a specific event.
518
+ * If a listener for this same event already exists, it will be overridden.
519
+ * If the listener has "isGlobalHandler" set to true, it will be registered to the hub connection.
520
+ * @param eventName Name of the event to register a listener for
521
+ * @param eventHandler The handler to be called when the event is received
522
+ */
523
+ addMessageHandler(eventName, eventHandler) {
524
+ this._messageHandlers.set(eventName, eventHandler);
525
+ if (eventHandler.isGlobalHandler) {
526
+ this.registerMessageHandler(eventName, eventHandler);
527
+ }
528
+ }
529
+ /**
530
+ * Dynamically register a listener for a specific event.
531
+ * If a listener for this event already exists, it will be overridden.
532
+ * @param eventName Name of the event to register a listener for
533
+ * @param eventHandler The handler to be called when the event is received
534
+ */
535
+ registerMessageHandler(eventName, eventHandler) {
536
+ if (!this.connection) {
537
+ console.log("No connection found to register the listener" + eventName);
538
+ return;
539
+ }
540
+ this.connection.on(eventName, (data) => {
541
+ eventHandler.handler(data);
542
+ });
543
+ }
544
+ /**
545
+ * Remove a listener for a specific event from the _messageHandlers map and unsubscribe from receiving messages for this event from the SignalR hub.
546
+ * @param eventName Name of the event to remove the listener for
547
+ */
548
+ removeMessageHandler(eventName) {
549
+ this._messageHandlers.delete(eventName);
550
+ this.unsubscribeMessageHandler(eventName);
551
+ }
552
+ /**
553
+ * Unsubscribe from receiving messages for a specific event from the SignalR hub.
554
+ * ALL its related listeners will be removed from hub connection
555
+ * This is needed to prevent accumulating old listeners when overriding the entire _messageHandlers map
556
+ * @param eventName Name of the event
557
+ */
558
+ unsubscribeMessageHandler(eventName) {
559
+ this.connection.off(eventName);
560
+ }
561
+ /**
562
+ * Build a connection to the signalR websocket and register default listeners to the methods defined in the server hub class
563
+ * @param options The options for the connection. It overrides the default options
564
+ * @param logLevel Define the log level displayed in the console
565
+ * @returns Promise that resolves when the connection is built
566
+ */
567
+ buildConnection(options) {
568
+ return new Promise((resolve, reject) => {
569
+ if (!this.REQUEST_URL) {
570
+ reject(new Error("No endpoint provided to connect the websocket to"));
571
+ return;
572
+ }
573
+ const logLevel = this._getLogLevel();
574
+ this.connection = this.signalRService.buildConnection(this.REQUEST_URL, { ...this.defaultOptions, ...options }, logLevel, true);
575
+ const signalRServerTimeoutInMilliseconds = this.assistantConfig$.value?.connectionSettings.signalRServerTimeoutInMilliseconds;
576
+ if (signalRServerTimeoutInMilliseconds) {
577
+ this.connection.serverTimeoutInMilliseconds = signalRServerTimeoutInMilliseconds;
578
+ }
579
+ resolve();
580
+ });
581
+ }
582
+ /**
583
+ * Start the connection
584
+ * @returns Promise that resolves when the connection is started
585
+ */
586
+ startConnection() {
587
+ return this.signalRService.startConnection(this.connection);
588
+ }
589
+ /**
590
+ * Stop the connection
591
+ * @returns Promise that resolves when the connection is stopped
592
+ */
593
+ stopConnection() {
594
+ return this.signalRService.stopConnection(this.connection);
595
+ }
596
+ _getTransports() {
597
+ switch (this.assistantConfig$.value?.connectionSettings.signalRTransport) {
598
+ case "WebSockets":
599
+ return HttpTransportType.WebSockets;
600
+ case "ServerSentEvents":
601
+ return HttpTransportType.ServerSentEvents;
602
+ case "LongPolling":
603
+ return HttpTransportType.LongPolling;
604
+ default:
605
+ return HttpTransportType.None;
606
+ }
607
+ }
608
+ _getLogLevel() {
609
+ switch (this.assistantConfig$.value?.connectionSettings.signalRLogLevel) {
610
+ case "Critical":
611
+ return LogLevel.Critical; // Log level for diagnostic messages that indicate a failure that will terminate the entire application.
612
+ case "Debug":
613
+ return LogLevel.Debug; // Log level for low severity diagnostic messages.
614
+ case "Error":
615
+ return LogLevel.Error; // Log level for diagnostic messages that indicate a failure in the current operation.
616
+ case "Information":
617
+ return LogLevel.Information; // Log level for informational diagnostic messages.
618
+ case "None":
619
+ return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.
620
+ case "Trace":
621
+ return LogLevel.Trace; // Log level for very low severity diagnostic messages.
622
+ case "Warning":
623
+ return LogLevel.Warning; // Log level for diagnostic messages that indicate a non-fatal problem.
624
+ default:
625
+ return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.
626
+ }
627
+ }
628
+ get defaultOptions() {
629
+ let headers = {
630
+ "sinequa-force-camel-case": "true",
631
+ "x-language": this.getCurrentLocaleName(),
632
+ "ui-language": this.getCurrentLocaleName(),
633
+ };
634
+ const token = getToken();
635
+ if (token) {
636
+ headers = { ...headers, "sinequa-csrf-token": token };
637
+ }
638
+ // For the first GET request sent by signalR to start a WebSocket protocol,
639
+ // as far as we know, signalR only lets us tweak the request with this access token factory
640
+ // so we pass along the Sinequa CSRF token to pass the CSRF check..
641
+ return {
642
+ transport: this._getTransports(),
643
+ withCredentials: true,
644
+ headers,
645
+ accessTokenFactory: () => token || ""
646
+ };
647
+ }
648
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WebSocketChatService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
649
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WebSocketChatService }); }
650
+ }
651
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: WebSocketChatService, decorators: [{
652
+ type: Injectable
653
+ }], ctorParameters: () => [] });
654
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/websocket-chat.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACnD,OAAO,EAAE,iBAAiB,EAAiB,QAAQ,EAAkB,MAAM,oBAAoB,CAAC;AAChG,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,SAAS,EAAE,GAAG,EAAE,KAAK,EAAE,QAAQ,EAAE,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,SAAS,EAAE,GAAG,EAAE,UAAU,EAAE,MAAM,MAAM,CAAC;AAEhK,OAAO,EAAS,QAAQ,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAEhE,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAqB,iBAAiB,EAAE,MAAM,gCAAgC,CAAC;;AAItF,MAAM,OAAO,oBAAqB,SAAQ,WAAW;IAenD;QACE,KAAK,EAAE,CAAC;QAZF,qBAAgB,GAAqC,IAAI,GAAG,EAAE,CAAC;QAE/D,eAAU,GAAG,IAAI,GAAG,EAAyB,CAAC;QAC9C,cAAS,GAA+B,SAAS,CAAC;QAGlD,iBAAY,GAA4B,EAAE,CAAC;QAC3C,mBAAc,GAAmB,EAAE,CAAC;QAErC,mBAAc,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;IAIlD,CAAC;IAED;;;;;;OAMG;IACH,IAAI;QACF,iEAAiE;QACjE,OAAO,KAAK,CAAC,GAAG,EAAE;YAChB,IAAI,CAAC,cAAc,EAAE,CAAC;YAEtB,OAAO,IAAI;YACT,uBAAuB;YACvB,IAAI,CAAC,eAAe,EAAE,CACrB,CAAC,IAAI,CACN,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC;YACrC,uBAAuB;YACvB,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC;YACvC,qDAAqD;YACrD,SAAS,CAAC,GAAG,EAAE,CACb,QAAQ,CAAC;gBACP,IAAI,CAAC,UAAU,EAAE;gBACjB,IAAI,CAAC,aAAa,EAAE;aACrB,CAAC,CACH;YACD,uEAAuE;YACvE,GAAG,CAAC,CAAC,CAAC,MAAM,EAAE,SAAS,CAAC,EAAE,EAAE;gBAC1B,MAAM,MAAM,GAAG,CAAC,CAAC,MAAM,IAAI,CAAC,CAAC,SAAS,CAAC;gBACvC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;gBAC/B,OAAO,MAAM,CAAC;YAChB,CAAC,CAAC;YACF,iGAAiG;YACjG,UAAU,CAAC,CAAC,KAAK,EAAE,EAAE;gBACnB,OAAO,CAAC,KAAK,CAAC,iBAAiB,EAAE,KAAK,CAAC,CAAC;gBACxC,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;YACjC,CAAC,CAAC,EACF,IAAI,CAAC,CAAC,CAAC,CACR,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,kBAAkB,CAAC,iBAAiB,EAAE,CAAC;YACtE,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,kBAAkB,CAAC,iBAAiB,CAAC;QACvF,CAAC;aAAM,CAAC;YACN,MAAM,IAAI,KAAK,CAAC,4GAA4G,CAAC,CAAC;QAChI,CAAC;IACH,CAAC;IAED,YAAY;QACV,MAAM,EAAE,kBAAkB,EAAE,YAAY,EAAE,GAAG,YAAY,CAAC;QAC1D,IAAI,CAAC,CAAC,kBAAkB,IAAI,YAAY,CAAC,EAAE,CAAC;YAC1C,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAC/B,OAAO;QACT,CAAC;QAED,yDAAyD;QACzD,MAAM,IAAI,GAAG;YACX,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,IAAI,EAAE,YAAY,CAAC,QAAQ;YAC3B,MAAM,EAAE,YAAY,CAAC,MAAM;SAC5B,CAAA;QAED,mDAAmD;QACnD,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,cAAc,EAAE,IAAI,CAAC;aAC1C,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;aAC7C,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;YACrD,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAC;IACP,CAAC;IAED,UAAU;QACR,MAAM,cAAc,GAAG,IAAI,OAAO,EAAsC,CAAC;QAEzE,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,YAAY,EAAE,CAAC,GAAG,EAAE,EAAE;YACxC,IAAI,CAAC,MAAM,GAAI,GAAG,CAAC,MAA6C,EAAE,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;YAClG,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACjC,cAAc,CAAC,QAAQ,EAAE,CAAA;QAC3B,CAAC,CAAC,CAAC;QAEH,6CAA6C;QAC7C,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,YAAY,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;aAC/F,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,4BAA4B,EAAE,KAAK,CAAC,CAAC;YACnD,cAAc,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;YACvC,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAA;QAEJ,OAAO,cAAc,CAAC,YAAY,EAAE,CAAC;IACvC,CAAC;IAED,aAAa;QACX,MAAM,iBAAiB,GAAG,IAAI,OAAO,EAA8B,CAAC;QAEpE,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,eAAe,EAAE,CAAC,GAAG,EAAE,EAAE;YAC3C,IAAI,CAAC,SAAS,GAAI,GAAG,CAAC,SAAwC,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC7F,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YACvC,iBAAiB,CAAC,QAAQ,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAC;QAEH,gDAAgD;QAChD,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,eAAe,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;aAClG,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,+BAA+B,EAAE,KAAK,CAAC,CAAC;YACtD,iBAAiB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;YAC1C,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAA;QAEJ,OAAO,iBAAiB,CAAC,YAAY,EAAE,CAAC;IAC1C,CAAC;IAED,KAAK,CAAC,QAAuB,EAAE,KAAY;QACzC,8CAA8C;QAC9C,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAE3B,iDAAiD;QACjD,MAAM,IAAI,GAAgB;YACxB,OAAO,EAAE,QAAQ;YACjB,SAAS,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,SAAS,EAAE,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC;YACpH,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;YACvD,eAAe,EAAE;gBACf,UAAU,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,UAAU;gBACjE,QAAQ,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,QAAQ;gBAC7D,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;gBACvD,WAAW,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,WAAW;gBACnE,UAAU,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,UAAU;gBACjE,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,yBAAyB;aAC1D;YACD,QAAQ,EAAE;gBACR,GAAG,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO;gBAC5B,KAAK;aACN;YACD,uBAAuB,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,cAAc,CAAC,uBAAuB,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,cAAc,CAAC,uBAAuB,CAAC,CAAC,CAAC,CAAC,EAAE;SAClM,CAAA;QACD,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,iBAAiB,CAAC,OAAO,EAAE,CAAC;YAC3D,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC;YACtC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QACtC,CAAC;QAED,0DAA0D;QAC1D,IAAI,CAAC,SAAS,GAAG,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,EAAE,EAAE,oBAAoB,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,EAAE,CAAC,CAAA,CAAC,wEAAwE;QAEvK,wCAAwC;QACxC,MAAM,WAAW,GAAG,IAAI,OAAO,EAAQ,CAAC;QAExC,2JAA2J;QAC3J,MAAM,WAAW,GAAG,KAAK;aACtB,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,CAAC;aACrC,MAAM,CAAC,CAAC,CAAC,SAAS,EAAE,YAAY,CAAC,EAAE,EAAE,CAAC,CAAC,YAAY,CAAC,eAAe,CAAC;aACpE,GAAG,CAAC,CAAC,CAAC,SAAS,EAAE,YAAY,CAAC,EAAE,EAAE,CAAC,SAAS,CAAM,IAAI,CAAC,UAAW,EAAE,SAAS,CAAC,CAAC,IAAI,CAChF,QAAQ,CAAC,CAAC,KAAK,EAAE,EAAE;YACjB,yHAAyH;YACzH,IAAI,CAAC;gBACH,0CAA0C;gBAC1C,0KAA0K;gBAC1K,OAAO,EAAE,CAAC,YAAY,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;YACzC,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,OAAO,CAAC,KAAK,CAAC,8BAA8B,SAAS,GAAG,EAAE,KAAK,CAAC,CAAC;gBACjE,mDAAmD;gBACnD,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,IAAI,KAAK,CAAC,8BAA8B,SAAS,KAAK,KAAK,EAAE,CAAC,CAAC,CAAC;YAC1F,CAAC;QACH,CAAC,CAAC,CACH,CAAC,CAAC;QAEP,uFAAuF;QACvF,MAAM,SAAS,GAAG,KAAK,CAAC,GAAG,WAAW,CAAC,CAAC,IAAI,CAC1C,GAAG,CAAC,GAAG,EAAE;YACP,uCAAuC;YACvC,MAAM,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,CAAC,CAAC;YACrD,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,MAAM,GAAG,CAAC;gBACjB,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBAChB,KAAK,EAAE,CAAC,CAAC,WAAW,IAAI,EAAE;oBAC1B,OAAO,EAAE,CAAC,CAAC,YAAY,IAAI,EAAE;oBAC7B,IAAI,EAAE,CAAC,CAAC,aAAa,KAAK,SAAS;oBACnC,IAAI,EAAE,CAAC,CAAC,aAAa;iBACtB,CAAC,CAAC;gBACP,CAAC,CAAC,SAAS,CAAC;YAE9B,iHAAiH;YACjH,oHAAoH;YACpH,gHAAgH;YAChH,uCAAuC;YACvC,wEAAwE;YACxE,yHAAyH;YACzH,IAAG,CAAC,CAAC,IAAI,CAAC,SAAS,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACtF,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,oBAAoB,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;gBAClE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,oBAAoB,CAAC,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC;gBACvE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,oBAAoB,CAAC,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC;YACtE,CAAC;YAED,oBAAoB;YACpB,OAAO,EAAE,OAAO,EAAE,CAAC,GAAG,QAAQ,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,EAAE,aAAa,EAAE,IAAI,CAAC,cAAc,EAAE,yBAAyB,EAAE,IAAI,CAAC,0BAA0B,EAAE,CAAC;QACvJ,CAAC,CAAC,EACF,SAAS,CAAC,WAAW,CAAC,CACvB,CAAC;QAEF,4IAA4I;QAC5I,OAAO,IAAI,UAAU,CAAC,QAAQ,CAAC,EAAE;YAC/B,+BAA+B;YAC/B,SAAS,CAAC,SAAS,CAAC;gBAClB,IAAI,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC;gBACrC,KAAK,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC;aACpC,CAAC,CAAC;YAEH,2CAA2C;YAC3C,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,MAAM,EAAE,IAAI,CAAC;iBAClC,IAAI,CAAC,GAAG,EAAE;gBACT,yEAAyE;gBACzE,gCAAgC;gBAChC,MAAM,KAAK,GAAG,IAAI,CAAC,iCAAiC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACvE,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE,CAAC;oBACjB,IAAI,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,oBAAoB,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;oBACzE,IAAI,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,oBAAoB,CAAC,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC;oBAC9E,IAAI,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,oBAAoB,CAAC,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC;gBAC7E,CAAC;gBACD,4CAA4C;gBAC5C,IAAI,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,iBAAiB,CAAC,OAAO,IAAI,IAAI,CAAC,WAAY,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,oBAAoB,EAAE,WAAW,KAAK,IAAI,CAAC,EAAE,CAAC;oBAC9I,MAAM,MAAM,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,WAAY,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,WAAW,EAAE,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;oBACpL,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;wBAC7B,IAAI,EAAE,GAAG,EAAE,GAAE,CAAC;wBACd,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE;4BACf,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;4BAC5B,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;wBACxB,CAAC;wBACD,QAAQ,EAAE,GAAG,EAAE;4BACb,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;4BAC5B,QAAQ,CAAC,QAAQ,EAAE,CAAC;wBACtB,CAAC;qBACF,CAAC,CAAC;gBACL,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;oBAC5B,QAAQ,CAAC,QAAQ,EAAE,CAAC;gBACtB,CAAC;YACH,CAAC,CAAC;iBACD,KAAK,CAAC,KAAK,CAAC,EAAE;gBACb,OAAO,CAAC,KAAK,CAAC,sBAAsB,EAAE,KAAK,CAAC,CAAC;gBAC7C,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC5B,iDAAiD;gBACjD,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;gBACtB,wFAAwF;gBACxF,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;YAC3B,CAAC,CAAC;iBACD,OAAO,CAAC,GAAG,EAAE;gBACZ,2EAA2E;gBAC3E,qDAAqD;gBACrD,oLAAoL;gBACpL,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC,sBAAsB;gBAC3C,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC,CAAC,uBAAuB;gBAChD,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC,CAAC,sBAAsB;gBAClD,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC,CAAC,yBAAyB;gBACjD,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC,CAAC,2BAA2B;gBACrD,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC,CAAC,2BAA2B;gBACrD,IAAI,CAAC,0BAA0B,GAAG,SAAS,CAAC,CAAC,uCAAuC;gBACpF,WAAW,CAAC,IAAI,EAAE,CAAC,CAAC,4CAA4C;gBAChE,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC,uBAAuB;YACjD,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;IACL,CAAC;IAED,cAAc;QACZ,+DAA+D;QAC/D,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACpC,gEAAgE;QAChE,MAAM,sBAAsB,GAAG,IAAI,OAAO,EAAW,CAAC;QAEtD,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,aAAa,EAAE,CAAC,GAAG,EAAE,EAAE;YACzC,4JAA4J;YAC5J,yIAAyI;YACzI,2FAA2F;YAC3F,oKAAoK;YACpK,IAAI,IAAI,CAAC,SAAS,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,OAAO,KAAK,EAAE,EAAE,CAAC;gBACpE,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,EAAC,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,EAAE,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,CAAC,SAAS,EAAE,WAAW,EAAE,IAAI,CAAC,YAAY,EAAE,MAAM,EAAE,IAAI,CAAC,cAAc,EAAC,EAAC,CAAC,CAAC;YAC1L,CAAC;YACD,sBAAsB,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,4CAA4C;YAChF,sBAAsB,CAAC,QAAQ,EAAE,CAAC,CAAC,uBAAuB;YAC1D,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,+BAA+B;QACvE,CAAC,CAAC,CAAC;QAEH,kDAAkD;QAClD,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,aAAa,CAAC;aACnC,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAC;YACpD,sBAAsB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;YAC/C,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,+BAA+B;YACrE,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAC;QAEL,OAAO,sBAAsB,CAAC,YAAY,EAAE,CAAC;IAC/C,CAAC;IAED,aAAa;QACX,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,iBAAiB,CAAC,OAAO,EAAE,CAAC;YAC5D,OAAO;QACT,CAAC;QAED,MAAM,IAAI,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO;YAChC,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACxD,CAAC;QAEF,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,eAAe,EAAE,CAAC,GAAG,EAAE,EAAE;YAC3C,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,8CAA8C;QACvF,CAAC,CAAC,CAAC;QAEH,kCAAkC;QAClC,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,eAAe,EAAE,IAAI,CAAC;aAC3C,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,+BAA+B,EAAE,KAAK,CAAC,CAAC;YACtD,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;QAC3B,CAAC,CAAC,CAAC;IACP,CAAC;IAED,YAAY,CAAC,EAAU;QACrB,MAAM,iBAAiB,GAAG,IAAI,OAAO,EAAgC,CAAC;QAEtE,MAAM,IAAI,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO;YAChC,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,WAAW,EAAE,EAAE;YACf,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACxD,CAAC;QAEF,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,cAAc,EAAE,CAAC,GAAG,EAAE,EAAE;YAC1C,IAAI,CAAC,kBAAkB,CAAC,qBAAqB,EAAE,EAAE,QAAQ,EAAE,GAAG,CAAC,yBAAyB,EAAE,EAAE,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,CAAA;YAC7G,iBAAiB,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YACtC,iBAAiB,CAAC,QAAQ,EAAE,CAAA;QAC9B,CAAC,CAAC,CAAC;QAEH,iCAAiC;QACjC,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,cAAc,EAAE,IAAI,CAAC;aAC1C,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;YACrD,iBAAiB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;YAC1C,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAA;QAEJ,OAAO,iBAAiB,CAAC,YAAY,EAAE,CAAC;IAC1C,CAAC;IAED,YAAY,CAAC,QAAuB;QAClC,MAAM,oBAAoB,GAAG,IAAI,OAAO,EAAqB,CAAC;QAE9D,MAAM,IAAI,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO;YAChC,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,WAAW,EAAE,IAAI,CAAC,MAAM;YACxB,OAAO,EAAE,QAAQ;YACjB,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACxD,CAAC;QAEF,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,cAAc,EAAE,CAAC,GAAG,EAAE,EAAE;YAC1C,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,0BAA0B;YACjE,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,EAAE,EAAE,QAAQ,EAAE,GAAG,CAAC,yBAAyB,EAAE,EAAE,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,uBAAuB;YACrI,oBAAoB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAC/B,oBAAoB,CAAC,QAAQ,EAAE,CAAA;QACjC,CAAC,CAAC,CAAC;QAEH,iCAAiC;QACjC,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,cAAc,EAAE,IAAI,CAAC;aAC1C,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;YACrD,oBAAoB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;YAC7C,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAA;QAEJ,OAAO,oBAAoB,CAAC,YAAY,EAAE,CAAC;IAC7C,CAAC;IAED,eAAe,CAAC,EAAU,EAAE,IAAa,EAAE,QAAwB;QACjE,MAAM,uBAAuB,GAAG,IAAI,OAAO,EAAqB,CAAC;QAEjE,MAAM,IAAI,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO;YAChC,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,WAAW,EAAE,EAAE;YACf,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACxD,CAAC;QAEF,IAAG,IAAI;YAAE,IAAI,CAAC,OAAO,CAAC,GAAG,IAAI,CAAC;QAC9B,IAAG,QAAQ;YAAE,IAAI,CAAC,SAAS,CAAC,GAAG,QAAQ,CAAC;QAExC,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,iBAAiB,EAAE,CAAC,GAAG,EAAE,EAAE;YAC7C,uBAAuB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAClC,uBAAuB,CAAC,QAAQ,EAAE,CAAA;QACpC,CAAC,CAAC,CAAC;QAEH,oCAAoC;QACpC,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC;aAC7C,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,iCAAiC,EAAE,KAAK,CAAC,CAAC;YACxD,uBAAuB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;YAChD,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAA;QAEJ,OAAO,uBAAuB,CAAC,YAAY,EAAE,CAAC;IAChD,CAAC;IAED,eAAe,CAAC,GAAa;QAC3B,MAAM,uBAAuB,GAAG,IAAI,OAAO,EAA2B,CAAC;QAEvE,MAAM,IAAI,GAAG;YACX,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO;YAChC,UAAU,EAAE,IAAI,CAAC,cAAc;YAC/B,YAAY,EAAE,GAAG;YACjB,KAAK,EAAE,IAAI,CAAC,gBAAgB,CAAC,KAAM,CAAC,aAAa,CAAC,KAAK;SACxD,CAAC;QAEF,IAAI,CAAC,UAAW,CAAC,EAAE,CAAC,iBAAiB,EAAE,CAAC,GAAG,EAAE,EAAE;YAC7C,uBAAuB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAClC,uBAAuB,CAAC,QAAQ,EAAE,CAAC;QACrC,CAAC,CAAC,CAAC;QAEH,oCAAoC;QACpC,IAAI,CAAC,UAAW,CAAC,MAAM,CAAC,iBAAiB,EAAE,IAAI,CAAC;aAC7C,KAAK,CAAC,KAAK,CAAC,EAAE;YACb,OAAO,CAAC,KAAK,CAAC,iCAAiC,EAAE,KAAK,CAAC,CAAC;YACxD,uBAAuB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;YAChD,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,yIAAyI;QACrK,CAAC,CAAC,CAAA;QAEJ,OAAO,uBAAuB,CAAC,YAAY,EAAE,CAAC;IAChD,CAAC;IAED;;;OAGG;IACH,mBAAmB;QACjB,IAAI,CAAC,iBAAiB,CACpB,OAAO,EACP;YACE,OAAO,EAAE,CAAC,KAAiB,EAAE,EAAE;gBAC7B,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;gBACrB,IAAI,CAAC,oBAAoB,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;YACzC,CAAC;YACD,eAAe,EAAE,IAAI;SACtB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,OAAO,EACP;YACE,OAAO,EAAE,CAAC,OAAmB,EAAE,EAAE;gBAC/B,IAAI,CAAC;oBACH,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,KAAK,CAAC,CAAA;gBACjC,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;gBACvB,CAAC;YACH,CAAC;YACD,eAAe,EAAE,IAAI;SACtB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,OAAO,EACP,EAAE,OAAO,EAAE,GAAG,EAAE,GAAE,CAAC;YACjB,eAAe,EAAE,IAAI;SACtB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,aAAa,EACb,EAAE,OAAO,EAAE,CAAC,MAAwB,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC;YAC/E,eAAe,EAAE,KAAK,EACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,cAAc,EACd;YACE,OAAO,EAAE,CAAC,MAAyB,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,GAAG,MAAM,EAAE,CAAC;YAC5H,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,YAAY,EACZ;YACE,OAAO,EAAE,CAAC,MAAuB,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,GAAG,MAAM,EAAE,CAAC;YAC1H,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,gBAAgB,EAChB;YACE,OAAO,EAAE,CAAC,OAA4B,EAAE,EAAE;gBACxC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,oBAAoB,CAAC,CAAC;YACvD,CAAC;YACD,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,SAAS,EACT;YACE,OAAO,EAAE,CAAC,OAAqB,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,OAAO,IAAI,OAAO,CAAC,KAAK,IAAI,EAAE;YACzF,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,SAAS,EACT;YACE,OAAO,EAAE,CAAC,OAAqB,EAAE,EAAE;gBACjC,uKAAuK;gBACvK,+HAA+H;gBAC/H,IAAI,CAAC,WAAW,GAAG,CAAC,GAAG,IAAI,CAAC,WAAY,EAAE,GAAG,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,WAAY,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;gBAChG,sCAAsC;gBACtC,IAAI,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,oBAAoB,CAAC,YAAY,EAAE,CAAC;oBACjE,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,YAAa,CAAC,CAAC;gBAC3F,CAAC;gBACD,IAAI,CAAC,cAAc,GAAG,OAAO,CAAC,aAAa,CAAC;gBAC5C,IAAI,CAAC,0BAA0B,GAAG,OAAO,CAAC,yBAAyB,CAAC;YACtE,CAAC;YACD,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,kBAAkB,EAClB;YACE,OAAO,EAAE,CAAC,OAA8B,EAAE,EAAE;gBAC1C,0EAA0E;gBAC1E,6HAA6H;gBAC7H,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,gBAAgB,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC;gBACrK,MAAM,KAAK,GAAG,IAAI,CAAC,gCAAgC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBACtE,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE,CAAC;oBACjB,IAAI,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,oBAAoB,CAAC,gBAAgB,GAAG,CAAC,IAAI,CAAC,WAAY,CAAC,KAAK,CAAC,CAAC,oBAAoB,CAAC,gBAAgB,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,gBAAgB,CAAC,CAAC;gBAC3K,CAAC;YACH,CAAC;YACD,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,cAAc,EACd;YACE,OAAO,EAAE,CAAC,OAA0B,EAAE,EAAE,CAAC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,OAAO,CAAC;YAClG,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;QACF,IAAI,CAAC,iBAAiB,CACpB,cAAc,EACd;YACE,OAAO,EAAE,GAAG,EAAE;gBACZ,uBAAuB;gBACvB,MAAM,OAAO,GAAG;oBACd,UAAU,EAAE,IAAI,CAAC,0BAA0B,KAAK,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc;oBACjH,MAAM,EAAE,IAAI,CAAC,WAAY,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,IAAI,EAAE,cAAc;oBACtD,MAAM,EAAE,IAAI,CAAC,WAAY,CAAC,MAAM,GAAG,CAAC;oBACpC,uBAAuB,EAAE,IAAI,CAAC,WAAY,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,YAAY,EAAE,oBAAoB;oBAC1G,mBAAmB,EAAE,IAAI,CAAC,WAAY,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,oBAAoB,CAAC,YAAY,EAAE,gBAAgB;oBAClG,aAAa,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,EAAE,QAAQ,EAAE,SAAS,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,CAAC;wBAC/E,QAAQ;wBACR,SAAS;wBACT,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE;4BACpC,IAAI,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,aAAa,EAAE,UAAU;gCAAE,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC;4BACtF,OAAO,EAAE,MAAM,EAAE,CAAC;wBACpB,CAAC,CAAC;wBACF,IAAI;qBACL,CAAC,CAAC,CAAC;iBACnB,CAAC;gBAEF,IAAI,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,aAAa,EAAE,UAAU;oBAC1D,OAAO,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,WAAY,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,OAAO,CAAC;gBAEtD,IAAI,CAAC,kBAAkB,CAAC,aAAa,EAAE,OAAO,CAAC,CAAC;gBAChD,2GAA2G;gBAC3G,IAAI,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAE,CAAC,OAAO,KAAK,EAAE,EAAE,CAAC;oBAC1C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAC,IAAI,EAAE,WAAW,EAAE,OAAO,EAAE,EAAE,EAAE,oBAAoB,EAAE,EAAC,OAAO,EAAE,IAAI,EAAC,EAAC,CAAC,CAAA;gBAC9F,CAAC;YACH,CAAC;YACD,eAAe,EAAE,KAAK;SACvB,CACF,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,uBAAuB,CAAI,gBAAgD;QACzE,gFAAgF;QAChF,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC,YAAY,EAAE,SAAS,EAAE,EAAE;YACxD,IAAG,YAAY,CAAC,eAAe,EAAE,CAAC;gBAChC,IAAI,CAAC,yBAAyB,CAAC,SAAS,CAAC,CAAC;YAC5C,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,sDAAsD;QACtD,IAAI,CAAC,gBAAgB,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,gBAAgB,EAAE,GAAG,gBAAgB,CAAC,CAAC,CAAC;QAEjF,oDAAoD;QACpD,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC,YAAY,EAAE,SAAS,EAAE,EAAE;YACxD,IAAG,YAAY,CAAC,eAAe,EAAE,CAAC;gBAChC,IAAI,CAAC,sBAAsB,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;YACvD,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;OAMG;IACH,iBAAiB,CAAI,SAAiB,EAAE,YAA+B;QACrE,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QACnD,IAAG,YAAY,CAAC,eAAe,EAAE,CAAC;YAChC,IAAI,CAAC,sBAAsB,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QACvD,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACO,sBAAsB,CAAI,SAAiB,EAAE,YAA+B;QACpF,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YACrB,OAAO,CAAC,GAAG,CAAC,8CAA8C,GAAG,SAAS,CAAC,CAAC;YACxE,OAAO;QACT,CAAC;QAED,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,IAAO,EAAE,EAAE;YACxC,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,SAAiB;QACpC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QACxC,IAAI,CAAC,yBAAyB,CAAC,SAAS,CAAC,CAAC;IAC5C,CAAC;IAED;;;;;OAKG;IACO,yBAAyB,CAAC,SAAiB;QACnD,IAAI,CAAC,UAAW,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IAClC,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,OAA2B;QACzC,OAAO,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YAC3C,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;gBACpB,MAAM,CAAC,IAAI,KAAK,CAAC,kDAAkD,CAAC,CAAC,CAAC;gBACtE,OAAO;YACX,CAAC;YACD,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;YACrC,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC,WAAW,EAAE,EAAC,GAAG,IAAI,CAAC,cAAc,EAAE,GAAG,OAAO,EAAC,EAAE,QAAQ,EAAE,IAAI,CAAC,CAAC;YAE9H,MAAM,kCAAkC,GAAG,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,kBAAkB,CAAC,kCAAkC,CAAC;YAC9H,IAAI,kCAAkC,EAAE,CAAC;gBACvC,IAAI,CAAC,UAAU,CAAC,2BAA2B,GAAG,kCAAkC,CAAC;YACnF,CAAC;YAED,OAAO,EAAE,CAAC;QACZ,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH,eAAe;QACb,OAAO,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC9D,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,OAAO,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IAC7D,CAAC;IAEO,cAAc;QACpB,QAAQ,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,kBAAkB,CAAC,gBAAgB,EAAE,CAAC;YACzE,KAAK,YAAY;gBACf,OAAO,iBAAiB,CAAC,UAAU,CAAC;YACtC,KAAK,kBAAkB;gBACrB,OAAO,iBAAiB,CAAC,gBAAgB,CAAC;YAC5C,KAAK,aAAa;gBAChB,OAAO,iBAAiB,CAAC,WAAW,CAAC;YACvC;gBACE,OAAO,iBAAiB,CAAC,IAAI,CAAC;QAClC,CAAC;IACH,CAAC;IAEO,YAAY;QAClB,QAAQ,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,kBAAkB,CAAC,eAAe,EAAE,CAAC;YACxE,KAAK,UAAU;gBACb,OAAO,QAAQ,CAAC,QAAQ,CAAC,CAAC,wGAAwG;YACpI,KAAK,OAAO;gBACV,OAAO,QAAQ,CAAC,KAAK,CAAC,CAAC,kDAAkD;YAC3E,KAAK,OAAO;gBACV,OAAO,QAAQ,CAAC,KAAK,CAAC,CAAC,sFAAsF;YAC/G,KAAK,aAAa;gBAChB,OAAO,QAAQ,CAAC,WAAW,CAAC,CAAC,mDAAmD;YAClF,KAAK,MAAM;gBACT,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,oHAAoH;YAC5I,KAAK,OAAO;gBACV,OAAO,QAAQ,CAAC,KAAK,CAAC,CAAC,uDAAuD;YAChF,KAAK,SAAS;gBACZ,OAAO,QAAQ,CAAC,OAAO,CAAC,CAAC,uEAAuE;YAClG;gBACE,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,oHAAoH;QAC9I,CAAC;IACH,CAAC;IAED,IAAI,cAAc;QAChB,IAAI,OAAO,GAAmB;YAC5B,0BAA0B,EAAE,MAAM;YAClC,YAAY,EAAE,IAAI,CAAC,oBAAoB,EAAE;YACzC,aAAa,EAAE,IAAI,CAAC,oBAAoB,EAAE;SAC3C,CAAC;QACF,MAAM,KAAK,GAAG,QAAQ,EAAE,CAAC;QACzB,IAAG,KAAK,EAAC,CAAC;YACR,OAAO,GAAG,EAAC,GAAG,OAAO,EAAE,oBAAoB,EAAE,KAAK,EAAC,CAAC;QACtD,CAAC;QAED,2EAA2E;QAC3E,2FAA2F;QAC3F,mEAAmE;QACnE,OAAO;YACL,SAAS,EAAE,IAAI,CAAC,cAAc,EAAE;YAChC,eAAe,EAAE,IAAI;YACrB,OAAO;YACP,kBAAkB,EAAE,GAAG,EAAE,CAAC,KAAK,IAAI,EAAE;SACtC,CAAA;IACH,CAAC;+GAzvBU,oBAAoB;mHAApB,oBAAoB;;4FAApB,oBAAoB;kBADhC,UAAU","sourcesContent":["import { Injectable, inject } from \"@angular/core\";\nimport { HttpTransportType, HubConnection, LogLevel, MessageHeaders } from \"@microsoft/signalr\";\nimport { Observable, Subject, catchError, defer, forkJoin, from, fromEvent, map, merge, mergeMap, of, switchMap, take, takeUntil, tap, throwError } from \"rxjs\";\n\nimport { Query, getToken, globalConfig } from \"@sinequa/atomic\";\n\nimport { ChatService } from \"./chat.service\";\nimport { ConnectionOptions, SignalRWebService } from \"./services/signalR.web.service\";\nimport { ActionMessage, ActionResultEvent, ActionStartEvent, ActionStopEvent, ChatContextAttachment, ChatMessage, ChatPayload, ChatProgress, ChatResponse, ContextMessageEvent, DebugMessage, DebugMessageEvent, DeleteSavedChatResponse, ErrorEvent, GllmFunction, GllmModelDescription, HistoryEvent, MessageEvent, MessageHandler, QuotaEvent, SavedChatHistory, SavedChatResponse, SuggestedActionsEvent } from \"./types\";\n\n@Injectable()\nexport class WebSocketChatService extends ChatService {\n\n  public connection: HubConnection | undefined;\n\n  private _messageHandlers: Map<string, MessageHandler<any>> = new Map();\n  private _response: ChatMessage[];\n  private _actionMap = new Map<string, ActionMessage>();\n  private _progress: ChatProgress[] | undefined = undefined;\n  private _executionTime: string;\n  private _executionTimeMilliseconds?: number;\n  private _attachments: ChatContextAttachment[] = [];\n  private _debugMessages: DebugMessage[] = [];\n\n  public signalRService = inject(SignalRWebService);\n\n  constructor() {\n    super();\n  }\n\n  /**\n   * Initialize the assistant process.\n   * It includes building and starting a connection, executing parallel requests for models and functions, and handling errors during the process.\n   * ⚠️ This method MUST be called ONLY if the user is loggedIn and once when the assistant is initialized.\n   *\n   * @returns An Observable<boolean> indicating the success of the initialization process.\n   */\n  init(): Observable<boolean> {\n    // Ensure all logic is executed when subscribed to the observable\n    return defer(() => {\n      this.getRequestsUrl();\n\n      return from(\n        // Build the connection\n        this.buildConnection()\n        ).pipe(\n        tap(() => this.initMessageHandlers()),\n        // Start the connection\n        switchMap(() => this.startConnection()),\n        // Execute parallel requests for models and functions\n        switchMap(() =>\n          forkJoin([\n            this.listModels(),\n            this.listFunctions()\n          ])\n        ),\n        // Map the results of parallel requests to a boolean indicating success\n        map(([models, functions]) => {\n          const result = !!models && !!functions;\n          this.initProcess$.next(result);\n          return result;\n        }),\n        // Any errors during the process are caught, logged, and re-thrown to propagate the error further\n        catchError((error) => {\n          console.error('Error occurred:', error);\n          return throwError(() => error);\n        }),\n        take(1)\n      );\n    });\n  }\n\n  /**\n   * Define the assistant endpoint to use for the websocket requests\n   * It can be overridden by the app config\n   */\n  getRequestsUrl() {\n    if (this.assistantConfig$.value!.connectionSettings.websocketEndpoint) {\n      this.REQUEST_URL = this.assistantConfig$.value!.connectionSettings.websocketEndpoint;\n    } else {\n      throw new Error(`The property 'websocketEndpoint' must be provided when attempting to use 'WebSocket' in assistant instance`);\n    }\n  }\n\n  overrideUser(): void {\n    const { userOverrideActive, userOverride } = globalConfig;\n    if (!(userOverrideActive && userOverride)) {\n      this.userOverride$.next(false);\n      return;\n    }\n\n    // Prepare the payload to send to the OverrideUser method\n    const data = {\n      instanceId: this.chatInstanceId,\n      user: userOverride.username,\n      domain: userOverride.domain\n    }\n\n    // Invoke the OverrideUser method and handle errors\n    this.connection!.invoke('OverrideUser', data)\n      .then((res) => this.userOverride$.next(!!res))\n      .catch(error => {\n        console.error('Error invoking OverrideUser:', error);\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      });\n  }\n\n  listModels(): Observable<GllmModelDescription[] | undefined> {\n    const modelsSubject$ = new Subject<GllmModelDescription[] | undefined>();\n\n    this.connection!.on('ListModels', (res) => {\n      this.models = (res.models as GllmModelDescription[] | undefined)?.filter(model => !!model.enable);\n      modelsSubject$.next(this.models);\n      modelsSubject$.complete()\n    });\n\n    // Send the request to get the list of models\n    this.connection!.invoke('ListModels', { debug: this.assistantConfig$.value!.defaultValues.debug })\n      .catch(error => {\n        console.error('Error invoking ListModels:', error);\n        modelsSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return modelsSubject$.asObservable();\n  }\n\n  listFunctions(): Observable<GllmFunction[] | undefined> {\n    const functionsSubject$ = new Subject<GllmFunction[] | undefined>();\n\n    this.connection!.on('ListFunctions', (res) => {\n      this.functions = (res.functions as GllmFunction[] | undefined)?.filter(func => func.enabled);\n      functionsSubject$.next(this.functions);\n      functionsSubject$.complete()\n    });\n\n    // Send the request to get the list of functions\n    this.connection!.invoke('ListFunctions', { debug: this.assistantConfig$.value!.defaultValues.debug })\n      .catch(error => {\n        console.error('Error invoking ListFunctions:', error);\n        functionsSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return functionsSubject$.asObservable();\n  }\n\n  fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse> {\n    // Start streaming by invoking the Chat method\n    this.streaming$.next(true);\n\n    // Prepare the payload to send to the Chat method\n    const data: ChatPayload = {\n      history: messages,\n      functions: this.assistantConfig$.value!.defaultValues.functions?.filter(func => func.enabled).map(func => func.name),\n      debug: this.assistantConfig$.value!.defaultValues.debug,\n      serviceSettings: {\n        service_id: this.assistantConfig$.value!.defaultValues.service_id,\n        model_id: this.assistantConfig$.value!.defaultValues.model_id,\n        top_p: this.assistantConfig$.value!.defaultValues.top_p,\n        temperature: this.assistantConfig$.value!.defaultValues.temperature,\n        max_tokens: this.assistantConfig$.value!.defaultValues.max_tokens,\n        ...this.assistantConfig$.value!.additionalServiceSettings\n      },\n      appQuery: {\n        app: this.appService.appName,\n        query\n      },\n      genericChatErrorMessage: this.assistantConfig$.value!.globalSettings.genericChatErrorMessage ? this.transloco.translate(this.assistantConfig$.value!.globalSettings.genericChatErrorMessage) : \"\"\n    }\n    if (this.assistantConfig$.value!.savedChatSettings.enabled) {\n      data.instanceId = this.chatInstanceId;\n      data.savedChatId = this.savedChatId;\n    }\n\n    // Initialize the response with an empty assistant message\n    this._response = [{ role: \"assistant\", content: \"\", additionalProperties: { display: true } }] // here display: true is needed in order to be able to show the progress\n\n    // Create a Subject to signal completion\n    const completion$ = new Subject<void>();\n\n    // Create observables for each non-global handler in the _messageHandlers map (default and eventual custom ones) once it is triggered by the hub connection\n    const observables = Array\n      .from(this._messageHandlers.entries())\n      .filter(([eventName, eventHandler]) => !eventHandler.isGlobalHandler)\n      .map(([eventName, eventHandler]) => fromEvent<any>(this.connection!, eventName).pipe(\n          mergeMap((event) => {\n            // Wrap the handler in a try-catch block to prevent the entire stream from failing if an error occurs in a single handler\n            try {\n              // Execute the handler and emit the result\n              // NB: here we could use [eventHandler.handler(event)] which behind the scenes mergeMap interprets this array as an observable sequence with one item, which it then emits\n              return of(eventHandler.handler(event));\n            } catch (error) {\n              console.error(`Error in event handler for ${eventName}:`, error);\n              // Use throwError to propagate the error downstream\n              return throwError(() => new Error(`Error in event handler for ${eventName}: ${error}`));\n            }\n          })\n        ));\n\n    // Then merge them into a single observable in order to simulate the streaming behavior\n    const combined$ = merge(...observables).pipe(\n      map(() => {\n        // Define $progress from the _actionMap\n        const actions = Array.from(this._actionMap.values());\n        this._progress = actions.length > 0\n                          ? actions.map((a) => ({\n                                title: a.displayName ?? \"\",\n                                content: a.displayValue ?? \"\",\n                                done: a.executionTime !== undefined,\n                                time: a.executionTime,\n                              }))\n                          : undefined;\n\n        // Always update ONLY the first assistant message of the _response with the new $progress, $attachment and $debug\n        // Assuming that the first assistant message is always visible since the hub does not send hidden messages by design\n        // So even if the first assistant message is hidden (display: false), the _response[0] will and should contain :\n        //  - $progress, $attachment and $debug\n        //  - the content of the first visible assistant message in the workflow\n        // This is mandatory in order to match the behavior of consecutive messages and maintain consistency with the chatHistory\n        if(!!this._progress || this._attachments.length > 0 || this._debugMessages.length > 0) {\n          this._response[0].additionalProperties.$progress = this._progress;\n          this._response[0].additionalProperties.$attachment = this._attachments;\n          this._response[0].additionalProperties.$debug = this._debugMessages;\n        }\n\n        // Return the result\n        return { history: [...messages, ...this._response], executionTime: this._executionTime, executionTimeMilliseconds: this._executionTimeMilliseconds };\n      }),\n      takeUntil(completion$), // Complete the observable when completion$ emits\n    );\n\n    // return a new Observable that emits the result of the combined stream and handles the eventual errors of the invocation of the Chat method\n    return new Observable(observer => {\n      // Subscribe to combined stream\n      combined$.subscribe({\n        next: (value) => observer.next(value),\n        error: (err) => observer.error(err)\n      });\n\n      // Invoke the Chat method and handle errors\n      this.connection!.invoke('Chat', data)\n        .then(() => {\n          // If a valid assistant message with (display: true) was found, update it\n          // and it should always the case\n          const index = this.firstVisibleAssistantMessageIndex(this.chatHistory);\n          if (index !== -1) {\n            this.chatHistory![index].additionalProperties.$progress = this._progress;\n            this.chatHistory![index].additionalProperties.$attachment = this._attachments;\n            this.chatHistory![index].additionalProperties.$debug = this._debugMessages;\n          }\n          // Save/update the chat if savedChat enabled\n          if (this.assistantConfig$.value!.savedChatSettings.enabled && this.chatHistory!.some((msg) => msg.additionalProperties?.isUserInput === true)) {\n            const action = !this.savedChatId ? this.addSavedChat(this.chatHistory!).pipe(tap(() => this.listSavedChat())) : this.updateSavedChat(this.savedChatId, undefined, this.chatHistory);\n            action.pipe(take(1)).subscribe({\n              next: () => {},\n              error: (error) => {\n                this.streaming$.next(false);\n                observer.error(error);\n              },\n              complete: () => {\n                this.streaming$.next(false);\n                observer.complete();\n              }\n            });\n          } else {\n            this.streaming$.next(false);\n            observer.complete();\n          }\n        })\n        .catch(error => {\n          console.error('Error invoking Chat:', error);\n          this.streaming$.next(false);\n          // Emit the error to the newly created observable\n          observer.error(error);\n          // Return a resolved promise to handle the error and prevent unhandled promise rejection\n          return Promise.resolve();\n        })\n        .finally(() => {\n          // This block concerns ONLY the completion of the \"Chat\" method invocation.\n          // This means the completion of the combined$ stream.\n          // It does not take into account the completion of the entire fetch method (the observable returned by fetch) and which depends on the completion of the save chat action if enabled\n          this._response = []; // Clear the _response\n          this._actionMap.clear(); // Clear the _actionMap\n          this._progress = undefined; // Clear the _progress\n          this._attachments = []; // Clear the _attachments\n          this._debugMessages = []; // Clear the _debugMessages\n          this._executionTime = \"\"; // Clear the _executionTime\n          this._executionTimeMilliseconds = undefined; // Clear the _executionTimeMilliseconds\n          completion$.next(); // Emit a signal to complete the observables\n          completion$.complete(); // Complete the subject\n        });\n    });\n  }\n\n  stopGeneration(): Observable<boolean> {\n    // Start stopping generation by invoking the CancelTasks method\n    this.stoppingGeneration$.next(true);\n    // Create a Subject to hold the result of the CancelTasks method\n    const stopGenerationSubject$ = new Subject<boolean>();\n\n    this.connection!.on('CancelTasks', (res) => {\n      // When the generation is stopped before streaming any VISIBLE assistant message, this means that $progress, $attachment and $debug properties will be lost.\n      // However, the \"ContextMessage\" frames will be persisted in the chatHistory and the assistant may reference them in the next generation.\n      // This leads to the problem of referencing undisplayed attachments in the next generation.\n      // To solve this problem, we need to persist $progress, $attachment and $debug properties by adding a new assistant message with empty content and these properties.\n      if (this._response.length === 1 && this._response[0].content === \"\") {\n        this.chatHistory?.push({role: \"assistant\", content: \"\", additionalProperties: {display: true, $progress: this._progress, $attachment: this._attachments, $debug: this._debugMessages}});\n      }\n      stopGenerationSubject$.next(!!res); // Emit the result of the CancelTasks method\n      stopGenerationSubject$.complete(); // Complete the subject\n      this.stoppingGeneration$.next(false); // Complete stopping generation\n    });\n\n    // Invoke the CancelTasks method and handle errors\n    this.connection!.invoke('CancelTasks')\n      .catch(error => {\n        console.error('Error invoking CancelTasks:', error);\n        stopGenerationSubject$.error(new Error(error));\n        this.stoppingGeneration$.next(false); // Complete stopping generation\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      });\n\n    return stopGenerationSubject$.asObservable();\n  }\n\n  listSavedChat(): void {\n    if (!this.assistantConfig$.value!.savedChatSettings.enabled) {\n      return;\n    }\n\n    const data = {\n      appName: this.appService.appName,\n      instanceId: this.chatInstanceId,\n      debug: this.assistantConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatList', (res) => {\n      this.savedChats$.next(res.savedChats); // emits the result to the savedChats$ subject\n    });\n\n    // Invoke the method SavedChatList\n    this.connection!.invoke('SavedChatList', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatList:', error);\n        return Promise.resolve();\n      });\n  }\n\n  getSavedChat(id: string): Observable<SavedChatHistory | undefined> {\n    const savedChatSubject$ = new Subject<SavedChatHistory | undefined>();\n\n    const data = {\n      appName: this.appService.appName,\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.assistantConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatGet', (res) => {\n      this.generateAuditEvent('ast-saved-chat.load', { duration: res.executionTimeMilliseconds }, res.savedChat.id)\n      savedChatSubject$.next(res.savedChat);\n      savedChatSubject$.complete()\n    });\n\n    // Invoke the method SavedChatGet\n    this.connection!.invoke('SavedChatGet', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatGet:', error);\n        savedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return savedChatSubject$.asObservable();\n  }\n\n  addSavedChat(messages: ChatMessage[]): Observable<SavedChatResponse> {\n    const addSavedChatSubject$ = new Subject<SavedChatResponse>();\n\n    const data = {\n      appName: this.appService.appName,\n      instanceId: this.chatInstanceId,\n      savedChatId: this.chatId,\n      history: messages,\n      debug: this.assistantConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatAdd', (res) => {\n      this.setSavedChatId(res.savedChat.id); // Persist the savedChatId\n      this.generateAuditEvent('ast-saved-chat.add', { duration: res.executionTimeMilliseconds }, res.savedChat.id); // Generate audit event\n      addSavedChatSubject$.next(res);\n      addSavedChatSubject$.complete()\n    });\n\n    // Invoke the method SavedChatAdd\n    this.connection!.invoke('SavedChatAdd', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatAdd:', error);\n        addSavedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return addSavedChatSubject$.asObservable();\n  }\n\n  updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChatResponse> {\n    const updateSavedChatSubject$ = new Subject<SavedChatResponse>();\n\n    const data = {\n      appName: this.appService.appName,\n      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.assistantConfig$.value!.defaultValues.debug\n    };\n\n    if(name) data[\"title\"] = name;\n    if(messages) data[\"history\"] = messages;\n\n    this.connection!.on('SavedChatUpdate', (res) => {\n      updateSavedChatSubject$.next(res);\n      updateSavedChatSubject$.complete()\n    });\n\n    // Invoke the method SavedChatUpdate\n    this.connection!.invoke('SavedChatUpdate', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatUpdate:', error);\n        updateSavedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return updateSavedChatSubject$.asObservable();\n  }\n\n  deleteSavedChat(ids: string[]): Observable<DeleteSavedChatResponse> {\n    const deleteSavedChatSubject$ = new Subject<DeleteSavedChatResponse>();\n\n    const data = {\n      appName: this.appService.appName,\n      instanceId: this.chatInstanceId,\n      SavedChatIds: ids,\n      debug: this.assistantConfig$.value!.defaultValues.debug\n    };\n\n    this.connection!.on('SavedChatDelete', (res) => {\n      deleteSavedChatSubject$.next(res);\n      deleteSavedChatSubject$.complete();\n    });\n\n    // Invoke the method SavedChatDelete\n    this.connection!.invoke('SavedChatDelete', data)\n      .catch(error => {\n        console.error('Error invoking SavedChatDelete:', error);\n        deleteSavedChatSubject$.error(new Error(error));\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection when no further error handling exists downstream\n      })\n\n    return deleteSavedChatSubject$.asObservable();\n  }\n\n  /**\n   * Initialize out-of-the-box handlers\n   * It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with frame message(s)\n   */\n  initMessageHandlers() {\n    this.addMessageHandler(\n      \"Error\",\n      {\n        handler: (error: ErrorEvent) => {\n          console.error(error);\n          this.notificationsService.error(error);\n        },\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"Quota\",\n      {\n        handler: (message: QuotaEvent) => {\n          try {\n            this.updateQuota(message.quota)\n          } catch (error) {\n            console.error(error);\n          }\n        },\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"Debug\",\n      { handler: () => {},\n        isGlobalHandler: true\n      }\n    );\n    this.addMessageHandler(\n      \"ActionStart\",\n      { handler: (action: ActionStartEvent) => this._actionMap.set(action.guid, action),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ActionResult\",\n      {\n        handler: (action: ActionResultEvent) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ActionStop\",\n      {\n        handler: (action: ActionStopEvent) => this._actionMap.set(action.guid, { ...this._actionMap.get(action.guid), ...action }),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"ContextMessage\",\n      {\n        handler: (message: ContextMessageEvent) => {\n          this._attachments.push(message.additionalProperties);\n        },\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"Message\",\n      {\n        handler: (message: MessageEvent) => this._response.at(-1)!.content += message.delta ?? \"\",\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"History\",\n      {\n        handler: (history: HistoryEvent) => {\n          // The ChatHistory is updated: it is the current copy concatenated with the new items ONLY (it can have multiple messages: the context messages + the response message)\n          // This is mandatory to not lose the previous updates of the chatHistory when the assistant is streaming multiple message steps\n          this.chatHistory = [...this.chatHistory!, ...(history.history.slice(this.chatHistory!.length))];\n          // Emit the updated chat usage metrics\n          if (!!this.chatHistory.at(-1)?.additionalProperties.usageMetrics) {\n            this.updateChatUsageMetrics(this.chatHistory.at(-1)!.additionalProperties.usageMetrics!);\n          }\n          this._executionTime = history.executionTime;\n          this._executionTimeMilliseconds = history.executionTimeMilliseconds;\n        },\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"SuggestedActions\",\n      {\n        handler: (message: SuggestedActionsEvent) => {\n          // Since after the \"History\" and \"MessageBreak\" that this event is caught,\n          // $suggestedAction needs to be updated directly to the last visible \"assistant\" message in the _response and the chatHistory\n          this._response.at(-1)!.additionalProperties.$suggestedAction = (this._response.at(-1)!.additionalProperties.$suggestedAction || []).concat(message.suggestedActions);\n          const index = this.lastVisibleAssistantMessageIndex(this.chatHistory);\n          if (index !== -1) {\n            this.chatHistory![index].additionalProperties.$suggestedAction = (this.chatHistory![index].additionalProperties.$suggestedAction || []).concat(message.suggestedActions);\n          }\n        },\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"DebugDisplay\",\n      {\n        handler: (message: DebugMessageEvent) => this._debugMessages = this._debugMessages.concat(message),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"MessageBreak\",\n      {\n        handler: () => {\n          // Generate audit event\n          const details = {\n            'duration': this._executionTimeMilliseconds !== undefined ? this._executionTimeMilliseconds : this._executionTime,\n            'role': this.chatHistory!.at(-1)!.role, // 'assistant'\n            'rank': this.chatHistory!.length - 1,\n            'generation-tokencount': this.chatHistory!.at(-1)!.additionalProperties.usageMetrics?.completionTokenCount,\n            'prompt-tokencount': this.chatHistory!.at(-1)!.additionalProperties.usageMetrics?.promptTokenCount,\n            'attachments': JSON.stringify(this._attachments.map(({ recordId, contextId, parts, type }) => ({\n                            recordId,\n                            contextId,\n                            parts: parts.map(({ partId, text }) => {\n                              if (!!this.assistantConfig$.value?.auditSettings?.logContent) return { partId, text };\n                              return { partId };\n                            }),\n                            type\n                          })))\n          };\n\n          if (!!this.assistantConfig$.value?.auditSettings?.logContent)\n            details['text'] = this.chatHistory!.at(-1)!.content;\n\n          this.generateAuditEvent('ast-message', details);\n          // Push a new assistant message to the _response array ONLY if the content of the last message is not empty\n          if (this._response.at(-1)!.content !== \"\") {\n            this._response.push({role: \"assistant\", content: \"\", additionalProperties: {display: true}})\n          }\n        },\n        isGlobalHandler: false\n      }\n    );\n  }\n\n  /**\n   * Override and register the entire _messageHandlers map by merging the provided map with the default one\n   * @param _messageHandlers\n   */\n  overrideMessageHandlers<T>(_messageHandlers: Map<string, MessageHandler<T>>) {\n    // Clear the already registered global chat handlers before merging the new ones\n    this._messageHandlers.forEach((eventHandler, eventName) => {\n      if(eventHandler.isGlobalHandler) {\n        this.unsubscribeMessageHandler(eventName);\n      }\n    });\n\n    // Merge the new event handlers with the existing ones\n    this._messageHandlers = new Map([...this._messageHandlers, ..._messageHandlers]);\n\n    // Register the global handlers among the merged map\n    this._messageHandlers.forEach((eventHandler, eventName) => {\n      if(eventHandler.isGlobalHandler) {\n        this.registerMessageHandler(eventName, eventHandler);\n      }\n    });\n  }\n\n  /**\n   * Add a listener for a specific event.\n   * If a listener for this same event already exists, it will be overridden.\n   * If the listener has \"isGlobalHandler\" set to true, it will be registered to the hub connection.\n   * @param eventName Name of the event to register a listener for\n   * @param eventHandler The handler to be called when the event is received\n   */\n  addMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>) {\n    this._messageHandlers.set(eventName, eventHandler);\n    if(eventHandler.isGlobalHandler) {\n      this.registerMessageHandler(eventName, eventHandler);\n    }\n  }\n\n  /**\n   * Dynamically register a listener for a specific event.\n   * If a listener for this event already exists, it will be overridden.\n   * @param eventName Name of the event to register a listener for\n   * @param eventHandler The handler to be called when the event is received\n   */\n  protected registerMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>) {\n    if (!this.connection) {\n      console.log(\"No connection found to register the listener\" + eventName);\n      return;\n    }\n\n    this.connection.on(eventName, (data: T) => {\n      eventHandler.handler(data);\n    });\n  }\n\n  /**\n   * Remove a listener for a specific event from the _messageHandlers map and unsubscribe from receiving messages for this event from the SignalR hub.\n   * @param eventName Name of the event to remove the listener for\n   */\n  removeMessageHandler(eventName: string) {\n    this._messageHandlers.delete(eventName);\n    this.unsubscribeMessageHandler(eventName);\n  }\n\n  /**\n   * Unsubscribe from receiving messages for a specific event from the SignalR hub.\n   * ALL its related listeners will be removed from hub connection\n   * This is needed to prevent accumulating old listeners when overriding the entire _messageHandlers map\n   * @param eventName Name of the event\n   */\n  protected unsubscribeMessageHandler(eventName: string) {\n    this.connection!.off(eventName);\n  }\n\n  /**\n   * Build a connection to the signalR websocket and register default listeners to the methods defined in the server hub class\n   * @param options The options for the connection. It overrides the default options\n   * @param logLevel Define the log level displayed in the console\n   * @returns Promise that resolves when the connection is built\n   */\n  buildConnection(options?: ConnectionOptions): Promise<void> {\n    return new Promise<void>((resolve, reject) => {\n      if (!this.REQUEST_URL) {\n          reject(new Error(\"No endpoint provided to connect the websocket to\"));\n          return;\n      }\n      const logLevel = this._getLogLevel();\n      this.connection = this.signalRService.buildConnection(this.REQUEST_URL, {...this.defaultOptions, ...options}, logLevel, true);\n\n      const signalRServerTimeoutInMilliseconds = this.assistantConfig$.value?.connectionSettings.signalRServerTimeoutInMilliseconds;\n      if (signalRServerTimeoutInMilliseconds) {\n        this.connection.serverTimeoutInMilliseconds = signalRServerTimeoutInMilliseconds;\n      }\n\n      resolve();\n    });\n  }\n\n  /**\n   * Start the connection\n   * @returns Promise that resolves when the connection is started\n   */\n  startConnection(): Promise<void> {\n    return this.signalRService.startConnection(this.connection);\n  }\n\n  /**\n   * Stop the connection\n   * @returns Promise that resolves when the connection is stopped\n   */\n  stopConnection(): Promise<void> {\n    return this.signalRService.stopConnection(this.connection);\n  }\n\n  private _getTransports(): HttpTransportType {\n    switch (this.assistantConfig$.value?.connectionSettings.signalRTransport) {\n      case \"WebSockets\":\n        return HttpTransportType.WebSockets;\n      case \"ServerSentEvents\":\n        return HttpTransportType.ServerSentEvents;\n      case \"LongPolling\":\n        return HttpTransportType.LongPolling;\n      default:\n        return HttpTransportType.None;\n    }\n  }\n\n  private _getLogLevel(): LogLevel {\n    switch (this.assistantConfig$.value?.connectionSettings.signalRLogLevel) {\n      case \"Critical\":\n        return LogLevel.Critical; // Log level for diagnostic messages that indicate a failure that will terminate the entire application.\n      case \"Debug\":\n        return LogLevel.Debug; // Log level for low severity diagnostic messages.\n      case \"Error\":\n        return LogLevel.Error; // Log level for diagnostic messages that indicate a failure in the current operation.\n      case \"Information\":\n        return LogLevel.Information; // Log level for informational diagnostic messages.\n      case \"None\":\n        return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.\n      case \"Trace\":\n        return LogLevel.Trace; // Log level for very low severity diagnostic messages.\n      case \"Warning\":\n        return LogLevel.Warning; // Log level for diagnostic messages that indicate a non-fatal problem.\n      default:\n        return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.\n    }\n  }\n\n  get defaultOptions(): ConnectionOptions {\n    let headers: MessageHeaders = {\n      \"sinequa-force-camel-case\": \"true\",\n      \"x-language\": this.getCurrentLocaleName(),\n      \"ui-language\": this.getCurrentLocaleName(),\n    };\n    const token = getToken();\n    if(token){\n      headers = {...headers, \"sinequa-csrf-token\": token};\n    }\n\n    // For the first GET request sent by signalR to start a WebSocket protocol,\n    // as far as we know, signalR only lets us tweak the request with this access token factory\n    // so we pass along the Sinequa CSRF token to pass the CSRF check..\n    return {\n      transport: this._getTransports(),\n      withCredentials: true,\n      headers,\n      accessTokenFactory: () => token || \"\"\n    }\n  }\n}\n"]}