@sinequa/assistant 3.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/chat/chat-message/chat-message.component.d.ts +157 -0
- package/chat/chat-reference/chat-reference.component.d.ts +118 -0
- package/chat/chat-settings-v3/chat-settings-v3.component.d.ts +41 -0
- package/chat/chat.component.d.ts +1112 -0
- package/chat/chat.service.d.ts +1046 -0
- package/chat/format-icon/format-icon.component.d.ts +10 -0
- package/chat/format-icon/icons.d.ts +5 -0
- package/chat/index.d.ts +5 -0
- package/chat/initials-avatar/initials-avatar.component.d.ts +35 -0
- package/chat/instance-manager.service.d.ts +28 -0
- package/chat/messages/de.d.ts +4 -0
- package/chat/messages/en.d.ts +4 -0
- package/chat/messages/fr.d.ts +4 -0
- package/chat/messages/index.d.ts +4 -0
- package/chat/public-api.d.ts +11 -0
- package/chat/rest-chat.service.d.ts +28 -0
- package/chat/saved-chats/saved-chats.component.d.ts +37 -0
- package/chat/styles/assistant.scss +61 -0
- package/chat/styles/references.scss +23 -0
- package/chat/types.d.ts +1241 -0
- package/chat/websocket-chat.service.d.ts +97 -0
- package/esm2020/chat/chat-message/chat-message.component.mjs +181 -0
- package/esm2020/chat/chat-reference/chat-reference.component.mjs +40 -0
- package/esm2020/chat/chat-settings-v3/chat-settings-v3.component.mjs +103 -0
- package/esm2020/chat/chat.component.mjs +369 -0
- package/esm2020/chat/chat.service.mjs +185 -0
- package/esm2020/chat/format-icon/format-icon.component.mjs +23 -0
- package/esm2020/chat/format-icon/icons.mjs +138 -0
- package/esm2020/chat/initials-avatar/initials-avatar.component.mjs +60 -0
- package/esm2020/chat/instance-manager.service.mjs +46 -0
- package/esm2020/chat/messages/de.mjs +4 -0
- package/esm2020/chat/messages/en.mjs +4 -0
- package/esm2020/chat/messages/fr.mjs +4 -0
- package/esm2020/chat/messages/index.mjs +9 -0
- package/esm2020/chat/public-api.mjs +12 -0
- package/esm2020/chat/rest-chat.service.mjs +164 -0
- package/esm2020/chat/saved-chats/saved-chats.component.mjs +132 -0
- package/esm2020/chat/sinequa-assistant-chat.mjs +5 -0
- package/esm2020/chat/types.mjs +85 -0
- package/esm2020/chat/websocket-chat.service.mjs +430 -0
- package/esm2020/public-api.mjs +3 -0
- package/esm2020/sinequa-assistant.mjs +5 -0
- package/fesm2015/sinequa-assistant-chat.mjs +1925 -0
- package/fesm2015/sinequa-assistant-chat.mjs.map +1 -0
- package/fesm2015/sinequa-assistant.mjs +9 -0
- package/fesm2015/sinequa-assistant.mjs.map +1 -0
- package/fesm2020/sinequa-assistant-chat.mjs +1911 -0
- package/fesm2020/sinequa-assistant-chat.mjs.map +1 -0
- package/fesm2020/sinequa-assistant.mjs +9 -0
- package/fesm2020/sinequa-assistant.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/package.json +46 -0
- package/public-api.d.ts +1 -0
|
@@ -0,0 +1,430 @@
|
|
|
1
|
+
import { Injectable, inject } from "@angular/core";
|
|
2
|
+
import { AuthenticationService } from "@sinequa/core/login";
|
|
3
|
+
import { SignalRWebService } from "@sinequa/core/web-services";
|
|
4
|
+
import { HttpTransportType, LogLevel } from "@microsoft/signalr";
|
|
5
|
+
import { ChatService } from "./chat.service";
|
|
6
|
+
import { merge, fromEvent, Subject, catchError, filter, forkJoin, map, shareReplay, switchMap, tap, throwError, takeUntil } from "rxjs";
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
export class WebSocketChatService extends ChatService {
|
|
9
|
+
constructor() {
|
|
10
|
+
super();
|
|
11
|
+
this.connectionBuilt$ = new Subject(); // Emit when the connection is built
|
|
12
|
+
this.connectionStarted$ = new Subject(); // Emit when the connection is started
|
|
13
|
+
this.messageHandlers = new Map();
|
|
14
|
+
this.actionMap = new Map();
|
|
15
|
+
this.content = "";
|
|
16
|
+
this.attachments = [];
|
|
17
|
+
this.signalRService = inject(SignalRWebService);
|
|
18
|
+
this.authenticationService = inject(AuthenticationService);
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Initialize the chat process after the login is complete.
|
|
22
|
+
* It includes building and starting a connection, executing parallel requests for models and functions, and handling errors during the process.
|
|
23
|
+
*
|
|
24
|
+
* @returns An Observable<boolean> indicating the success of the initialization process.
|
|
25
|
+
*/
|
|
26
|
+
init() {
|
|
27
|
+
return this.loginService.events.pipe(filter((e) => e.type === 'login-complete'), tap(() => this.getRequestsUrl()),
|
|
28
|
+
// Build the connection and handle the completion with the connectionBuilt$ subject
|
|
29
|
+
switchMap(() => this.buildConnection()), tap(() => {
|
|
30
|
+
this.initMessageHandlers();
|
|
31
|
+
this.connectionBuilt$.next();
|
|
32
|
+
}),
|
|
33
|
+
// Start the connection and handle the completion with the connectionStarted$ subject
|
|
34
|
+
switchMap(() => this.startConnection()),
|
|
35
|
+
// Execute parallel requests for models and functions
|
|
36
|
+
switchMap(() => {
|
|
37
|
+
this.connectionStarted$.next();
|
|
38
|
+
return forkJoin([
|
|
39
|
+
this.listModels(),
|
|
40
|
+
this.listFunctions()
|
|
41
|
+
]);
|
|
42
|
+
}),
|
|
43
|
+
// Map the results of parallel requests to a boolean indicating success
|
|
44
|
+
map(([models, functions]) => {
|
|
45
|
+
this.initProcess$.next(true);
|
|
46
|
+
return !!models && !!functions;
|
|
47
|
+
}),
|
|
48
|
+
// Any errors during the process are caught, logged, and re-thrown to propagate the error further
|
|
49
|
+
catchError((error) => {
|
|
50
|
+
console.error('Error occurred:', error);
|
|
51
|
+
return throwError(() => error);
|
|
52
|
+
}),
|
|
53
|
+
// Cache and replay the emitted value for subsequent subscribers
|
|
54
|
+
shareReplay(1));
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Define the assistant endpoint to use for the websocket requests
|
|
58
|
+
* It can be overridden by the app config
|
|
59
|
+
*/
|
|
60
|
+
getRequestsUrl() {
|
|
61
|
+
if (this.chatConfig$.value.globalSettings.websocketEndpoint) {
|
|
62
|
+
this.REQUEST_URL = this.chatConfig$.value.globalSettings.websocketEndpoint;
|
|
63
|
+
}
|
|
64
|
+
else {
|
|
65
|
+
throw new Error(`The property 'websocketEndpoint' must be provided when attempting to use 'WebSocket' in assistant instance`);
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
listModels() {
|
|
69
|
+
const modelsSubject = new Subject();
|
|
70
|
+
this.connection.on('ListModels', (res) => {
|
|
71
|
+
this.models = res.models?.filter(model => !!model.enable);
|
|
72
|
+
modelsSubject.next(this.models);
|
|
73
|
+
modelsSubject.complete();
|
|
74
|
+
});
|
|
75
|
+
// Send the request to get the list of models
|
|
76
|
+
this.connection.invoke('ListModels', { debug: this.chatConfig$.value.debug })
|
|
77
|
+
.catch(error => {
|
|
78
|
+
console.error('Error invoking ListModels:', error);
|
|
79
|
+
modelsSubject.complete();
|
|
80
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection
|
|
81
|
+
});
|
|
82
|
+
return modelsSubject.asObservable();
|
|
83
|
+
}
|
|
84
|
+
listFunctions() {
|
|
85
|
+
const functionsSubject = new Subject();
|
|
86
|
+
this.connection.on('ListFunctions', (res) => {
|
|
87
|
+
this.functions = res.functions;
|
|
88
|
+
functionsSubject.next(this.functions);
|
|
89
|
+
functionsSubject.complete();
|
|
90
|
+
});
|
|
91
|
+
// Send the request to get the list of functions
|
|
92
|
+
this.connection.invoke('ListFunctions', { debug: this.chatConfig$.value.debug })
|
|
93
|
+
.catch(error => {
|
|
94
|
+
console.error('Error invoking ListFunctions:', error);
|
|
95
|
+
functionsSubject.complete();
|
|
96
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection
|
|
97
|
+
});
|
|
98
|
+
return functionsSubject.asObservable();
|
|
99
|
+
}
|
|
100
|
+
fetch(messages, query = this.searchService.query) {
|
|
101
|
+
// Start streaming by invoking the Chat method
|
|
102
|
+
this.streaming$.next(true);
|
|
103
|
+
// Prepare the payload to send to the Chat method
|
|
104
|
+
const data = {
|
|
105
|
+
history: messages,
|
|
106
|
+
functions: this.chatConfig$.value.functions,
|
|
107
|
+
debug: this.chatConfig$.value.debug,
|
|
108
|
+
serviceSettings: this.chatConfig$.value.serviceSettings,
|
|
109
|
+
contextSettings: {
|
|
110
|
+
...this.chatConfig$.value.contextSettings,
|
|
111
|
+
app: this.appService.appName,
|
|
112
|
+
query
|
|
113
|
+
}
|
|
114
|
+
};
|
|
115
|
+
if (this.chatConfig$.value.saveChats) {
|
|
116
|
+
data.instanceId = this.chatInstanceId;
|
|
117
|
+
data.savedChatId = this.savedChatId;
|
|
118
|
+
}
|
|
119
|
+
let response = { role: "assistant", content: "", additionalProperties: { display: true } }; // here display: true is needed in order to be able to show the progress
|
|
120
|
+
// Create a Subject to signal completion
|
|
121
|
+
const completion$ = new Subject();
|
|
122
|
+
// Create observables for each non-global handler in the messageHandlers map (default and eventual custom ones) once it is triggered by the hub connection
|
|
123
|
+
const observables = Array
|
|
124
|
+
.from(this.messageHandlers.entries())
|
|
125
|
+
.filter(([eventName, eventHandler]) => !eventHandler.isGlobalHandler)
|
|
126
|
+
.map(([eventName, eventHandler]) => {
|
|
127
|
+
return fromEvent(this.connection, eventName).pipe(map((event) => eventHandler.handler(event)) // Execute the corresponding handler
|
|
128
|
+
);
|
|
129
|
+
});
|
|
130
|
+
// Then merge them into a single observable in order to simulate the streaming behavior
|
|
131
|
+
const combined$ = merge(...observables).pipe(takeUntil(completion$) // Complete the observable when completion$ emits
|
|
132
|
+
);
|
|
133
|
+
// Invoke the Chat method
|
|
134
|
+
this.connection.invoke('Chat', data)
|
|
135
|
+
.then(() => this.notifyAudit(this.chatHistory, this.chatConfig$.value.serviceSettings.service_id)) // When the server indicates it has successfully finished invoking the method, notify the audit service with the recent chat history
|
|
136
|
+
.catch(error => {
|
|
137
|
+
console.error('Error invoking Chat:', error);
|
|
138
|
+
return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection
|
|
139
|
+
})
|
|
140
|
+
.finally(() => {
|
|
141
|
+
this.streaming$.next(false); // Complete streaming regardless of success or error
|
|
142
|
+
this.actionMap.clear(); // Clear the actionMap
|
|
143
|
+
this.content = ""; // Clear the content
|
|
144
|
+
this.attachments = []; // Clear the attachments
|
|
145
|
+
this.executionTime = ""; // Clear the executionTime
|
|
146
|
+
completion$.next(); // Emit a signal to complete the observables
|
|
147
|
+
completion$.complete(); // Complete the subject
|
|
148
|
+
});
|
|
149
|
+
// Return the merged observables
|
|
150
|
+
return combined$.pipe(map(() => {
|
|
151
|
+
// Define $progress from the actionMap
|
|
152
|
+
const actions = Array.from(this.actionMap.values());
|
|
153
|
+
const $progress = actions.length > 0
|
|
154
|
+
? actions.map((a) => ({
|
|
155
|
+
title: a.displayName ?? "",
|
|
156
|
+
content: a.displayValue ?? "",
|
|
157
|
+
done: a.executionTime !== undefined,
|
|
158
|
+
time: a.executionTime,
|
|
159
|
+
}))
|
|
160
|
+
: undefined;
|
|
161
|
+
// Define the attachment used in the context of the response message
|
|
162
|
+
const $attachment = this.attachments;
|
|
163
|
+
// As soon as the first content or $progress is defined, the assistant is considered as streaming
|
|
164
|
+
if (!!this.content || $progress || $attachment.length > 0) {
|
|
165
|
+
response = { ...response, content: this.content, additionalProperties: { ...response.additionalProperties, $progress, $attachment } };
|
|
166
|
+
}
|
|
167
|
+
// Return the result
|
|
168
|
+
return { history: [...messages, response], executionTime: this.executionTime };
|
|
169
|
+
}));
|
|
170
|
+
}
|
|
171
|
+
listSavedChat() {
|
|
172
|
+
const data = {
|
|
173
|
+
instanceId: this.chatInstanceId,
|
|
174
|
+
debug: this.chatConfig$.value.debug
|
|
175
|
+
};
|
|
176
|
+
this.connection.on('SavedChatList', (res) => {
|
|
177
|
+
this.savedChats$.next(res.savedChats); // emits the result to the savedChats$ subject
|
|
178
|
+
});
|
|
179
|
+
// Invoke the method SavedChatList
|
|
180
|
+
this.connection.invoke('SavedChatList', data)
|
|
181
|
+
.catch(error => {
|
|
182
|
+
console.error('Error invoking SavedChatList:', error);
|
|
183
|
+
return Promise.resolve();
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
getSavedChat(id) {
|
|
187
|
+
const savedChatSubject = new Subject();
|
|
188
|
+
const data = {
|
|
189
|
+
instanceId: this.chatInstanceId,
|
|
190
|
+
savedChatId: id,
|
|
191
|
+
debug: this.chatConfig$.value.debug
|
|
192
|
+
};
|
|
193
|
+
this.connection.on('SavedChatGet', (res) => {
|
|
194
|
+
savedChatSubject.next(res.savedChat);
|
|
195
|
+
savedChatSubject.complete();
|
|
196
|
+
});
|
|
197
|
+
// Invoke the method SavedChatGet
|
|
198
|
+
this.connection.invoke('SavedChatGet', data)
|
|
199
|
+
.catch(error => {
|
|
200
|
+
console.error('Error invoking SavedChatGet:', error);
|
|
201
|
+
savedChatSubject.complete();
|
|
202
|
+
return Promise.resolve();
|
|
203
|
+
});
|
|
204
|
+
return savedChatSubject.asObservable();
|
|
205
|
+
}
|
|
206
|
+
deleteSavedChat(ids) {
|
|
207
|
+
const deleteSavedChatSubject = new Subject();
|
|
208
|
+
const data = {
|
|
209
|
+
instanceId: this.chatInstanceId,
|
|
210
|
+
SavedChatIds: ids,
|
|
211
|
+
debug: this.chatConfig$.value.debug
|
|
212
|
+
};
|
|
213
|
+
this.connection.on('SavedChatDelete', (res) => {
|
|
214
|
+
deleteSavedChatSubject.next(res.deleteCount);
|
|
215
|
+
deleteSavedChatSubject.complete();
|
|
216
|
+
});
|
|
217
|
+
// Invoke the method SavedChatDelete
|
|
218
|
+
this.connection.invoke('SavedChatDelete', data)
|
|
219
|
+
.catch(error => {
|
|
220
|
+
console.error('Error invoking SavedChatDelete:', error);
|
|
221
|
+
deleteSavedChatSubject.complete();
|
|
222
|
+
return Promise.resolve();
|
|
223
|
+
});
|
|
224
|
+
return deleteSavedChatSubject.asObservable();
|
|
225
|
+
}
|
|
226
|
+
/**
|
|
227
|
+
* Initialize out-of-the-box handlers
|
|
228
|
+
* It is a placeholder for non-streaming scenarios, where you invoke a specific hub method, and the server responds with a single message or a result
|
|
229
|
+
*/
|
|
230
|
+
initMessageHandlers() {
|
|
231
|
+
this.addMessageHandler("Debug", { handler: (debug) => console.log(debug),
|
|
232
|
+
isGlobalHandler: true });
|
|
233
|
+
this.addMessageHandler("ActionStart", { handler: (action) => this.actionMap.set(action.guid, action),
|
|
234
|
+
isGlobalHandler: false });
|
|
235
|
+
this.addMessageHandler("ActionResult", {
|
|
236
|
+
handler: (action) => this.actionMap.set(action.guid, { ...this.actionMap.get(action.guid), ...action }),
|
|
237
|
+
isGlobalHandler: false
|
|
238
|
+
});
|
|
239
|
+
this.addMessageHandler("ActionStop", {
|
|
240
|
+
handler: (action) => this.actionMap.set(action.guid, { ...this.actionMap.get(action.guid), ...action }),
|
|
241
|
+
isGlobalHandler: false
|
|
242
|
+
});
|
|
243
|
+
this.addMessageHandler("ContextMessage", {
|
|
244
|
+
handler: (message) => this.attachments.push(message.metadata),
|
|
245
|
+
isGlobalHandler: false
|
|
246
|
+
});
|
|
247
|
+
this.addMessageHandler("Message", {
|
|
248
|
+
handler: (message) => this.content += message ?? "",
|
|
249
|
+
isGlobalHandler: false
|
|
250
|
+
});
|
|
251
|
+
this.addMessageHandler("History", {
|
|
252
|
+
handler: (history) => {
|
|
253
|
+
this.chatHistory = history.history;
|
|
254
|
+
this.executionTime = history.executionTime;
|
|
255
|
+
},
|
|
256
|
+
isGlobalHandler: false
|
|
257
|
+
});
|
|
258
|
+
this.addMessageHandler("Error", {
|
|
259
|
+
handler: (error) => {
|
|
260
|
+
console.error(error);
|
|
261
|
+
this.notificationsService.error(error);
|
|
262
|
+
},
|
|
263
|
+
isGlobalHandler: true
|
|
264
|
+
});
|
|
265
|
+
this.addMessageHandler("Quota", {
|
|
266
|
+
handler: (message) => {
|
|
267
|
+
if (message.quota.maxQuotaReached) {
|
|
268
|
+
const msg = `Sorry, you have exceeded the allowed quota. Please retry starting from ${this.formatDateTime(message.quota.nextResetUTC)}.`;
|
|
269
|
+
console.error(msg);
|
|
270
|
+
this.notificationsService.error(msg);
|
|
271
|
+
}
|
|
272
|
+
},
|
|
273
|
+
isGlobalHandler: true
|
|
274
|
+
});
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* Override and register the entire messageHandlers map by merging the provided map with the default one
|
|
278
|
+
* @param messageHandlers
|
|
279
|
+
*/
|
|
280
|
+
overrideMessageHandlers(messageHandlers) {
|
|
281
|
+
// Clear the already registered global chat handlers before merging the new ones
|
|
282
|
+
this.messageHandlers.forEach((eventHandler, eventName) => {
|
|
283
|
+
if (eventHandler.isGlobalHandler) {
|
|
284
|
+
this.unsubscribeMessageHandler(eventName);
|
|
285
|
+
}
|
|
286
|
+
});
|
|
287
|
+
// Merge the new event handlers with the existing ones
|
|
288
|
+
this.messageHandlers = new Map([...this.messageHandlers, ...messageHandlers]);
|
|
289
|
+
// Register the global chat handlers among the merged map
|
|
290
|
+
this.messageHandlers.forEach((eventHandler, eventName) => {
|
|
291
|
+
if (eventHandler.isGlobalHandler) {
|
|
292
|
+
this.registerMessageHandler(eventName, eventHandler);
|
|
293
|
+
}
|
|
294
|
+
});
|
|
295
|
+
}
|
|
296
|
+
/**
|
|
297
|
+
* Add a listener for a specific event.
|
|
298
|
+
* If a listener for this same event already exists, it will be overridden.
|
|
299
|
+
* If the listener has "isChatGlobalHandler" set to true, it will be registered to the hub connection.
|
|
300
|
+
* @param eventName Name of the event to register a listener for
|
|
301
|
+
* @param eventHandler The handler to be called when the event is received
|
|
302
|
+
*/
|
|
303
|
+
addMessageHandler(eventName, eventHandler) {
|
|
304
|
+
this.messageHandlers.set(eventName, eventHandler);
|
|
305
|
+
if (eventHandler.isGlobalHandler) {
|
|
306
|
+
this.registerMessageHandler(eventName, eventHandler);
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
/**
|
|
310
|
+
* Dynamically register a listener for a specific event.
|
|
311
|
+
* If a listener for this event already exists, it will be overridden.
|
|
312
|
+
* @param eventName Name of the event to register a listener for
|
|
313
|
+
* @param eventHandler The handler to be called when the event is received
|
|
314
|
+
*/
|
|
315
|
+
registerMessageHandler(eventName, eventHandler) {
|
|
316
|
+
if (!this.connection) {
|
|
317
|
+
console.log("No connection found to register the listener" + eventName);
|
|
318
|
+
return;
|
|
319
|
+
}
|
|
320
|
+
this.connection.on(eventName, (data) => {
|
|
321
|
+
eventHandler.handler(data);
|
|
322
|
+
});
|
|
323
|
+
}
|
|
324
|
+
/**
|
|
325
|
+
* Remove a listener for a specific event from the messageHandlers map and unsubscribe from receiving messages for this event from the SignalR hub.
|
|
326
|
+
* @param eventName Name of the event to remove the listener for
|
|
327
|
+
*/
|
|
328
|
+
removeMessageHandler(eventName) {
|
|
329
|
+
this.messageHandlers.delete(eventName);
|
|
330
|
+
this.unsubscribeMessageHandler(eventName);
|
|
331
|
+
}
|
|
332
|
+
/**
|
|
333
|
+
* Unsubscribe from receiving messages for a specific event from the SignalR hub.
|
|
334
|
+
* ALL its related listeners will be removed from hub connection
|
|
335
|
+
* This is needed to prevent accumulating old listeners when overriding the entire messageHandlers map
|
|
336
|
+
* @param eventName Name of the event
|
|
337
|
+
*/
|
|
338
|
+
unsubscribeMessageHandler(eventName) {
|
|
339
|
+
this.connection.off(eventName);
|
|
340
|
+
}
|
|
341
|
+
/**
|
|
342
|
+
* Build a connection to the signalR websocket and register default listeners to the methods defined in the server hub class
|
|
343
|
+
* @param options The options for the connection. It overrides the default options
|
|
344
|
+
* @param logLevel Define the log level displayed in the console
|
|
345
|
+
* @returns Promise that resolves when the connection is built
|
|
346
|
+
*/
|
|
347
|
+
buildConnection(options) {
|
|
348
|
+
return new Promise((resolve, reject) => {
|
|
349
|
+
if (!this.REQUEST_URL) {
|
|
350
|
+
reject(new Error("No endpoint provided to connect the websocket to"));
|
|
351
|
+
return;
|
|
352
|
+
}
|
|
353
|
+
const logLevel = this.getLogLevel();
|
|
354
|
+
this.connection = this.signalRService.buildConnection(this.REQUEST_URL, { ...this.defaultOptions, ...options }, logLevel);
|
|
355
|
+
resolve();
|
|
356
|
+
});
|
|
357
|
+
}
|
|
358
|
+
/**
|
|
359
|
+
* Start the connection
|
|
360
|
+
* @returns Promise that resolves when the connection is started
|
|
361
|
+
*/
|
|
362
|
+
startConnection() {
|
|
363
|
+
return this.signalRService.startConnection(this.connection);
|
|
364
|
+
}
|
|
365
|
+
/**
|
|
366
|
+
* Stop the connection
|
|
367
|
+
* @returns Promise that resolves when the connection is stopped
|
|
368
|
+
*/
|
|
369
|
+
stopConnection() {
|
|
370
|
+
return this.signalRService.stopConnection(this.connection);
|
|
371
|
+
}
|
|
372
|
+
getTransports() {
|
|
373
|
+
switch (this.chatConfig$.value?.globalSettings.signalRTransport) {
|
|
374
|
+
case "WebSockets":
|
|
375
|
+
return HttpTransportType.WebSockets;
|
|
376
|
+
case "ServerSentEvents":
|
|
377
|
+
return HttpTransportType.ServerSentEvents;
|
|
378
|
+
case "LongPolling":
|
|
379
|
+
return HttpTransportType.LongPolling;
|
|
380
|
+
default:
|
|
381
|
+
return HttpTransportType.None;
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
getLogLevel() {
|
|
385
|
+
switch (this.chatConfig$.value?.globalSettings.signalRLogLevel) {
|
|
386
|
+
case "Critical":
|
|
387
|
+
return LogLevel.Critical; // Log level for diagnostic messages that indicate a failure that will terminate the entire application.
|
|
388
|
+
case "Debug":
|
|
389
|
+
return LogLevel.Debug; // Log level for low severity diagnostic messages.
|
|
390
|
+
case "Error":
|
|
391
|
+
return LogLevel.Error; // Log level for diagnostic messages that indicate a failure in the current operation.
|
|
392
|
+
case "Information":
|
|
393
|
+
return LogLevel.Information; // Log level for informational diagnostic messages.
|
|
394
|
+
case "None":
|
|
395
|
+
return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.
|
|
396
|
+
case "Trace":
|
|
397
|
+
return LogLevel.Trace; // Log level for very low severity diagnostic messages.
|
|
398
|
+
case "Warning":
|
|
399
|
+
return LogLevel.Warning; // Log level for diagnostic messages that indicate a non-fatal problem.
|
|
400
|
+
default:
|
|
401
|
+
return LogLevel.None; // The highest possible log level. Used when configuring logging to indicate that no log messages should be emitted.
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
get defaultOptions() {
|
|
405
|
+
let headers = {
|
|
406
|
+
"sinequa-force-camel-case": "true",
|
|
407
|
+
"x-language": this.intlService.currentLocale.name,
|
|
408
|
+
"ui-language": this.intlService.currentLocale.name,
|
|
409
|
+
};
|
|
410
|
+
if (this.authenticationService.processedCredentials) {
|
|
411
|
+
headers = { ...headers, "sinequa-csrf-token": this.authenticationService.processedCredentials.data.csrfToken };
|
|
412
|
+
}
|
|
413
|
+
;
|
|
414
|
+
// For the first GET request sent by signalR to start a WebSocket protocol,
|
|
415
|
+
// as far as we know, signalR only lets us tweak the request with this access token factory
|
|
416
|
+
// so we pass along the Sinequa CSRF token to pass the CSRF check..
|
|
417
|
+
return {
|
|
418
|
+
transport: this.getTransports(),
|
|
419
|
+
withCredentials: true,
|
|
420
|
+
headers,
|
|
421
|
+
accessTokenFactory: () => this.authenticationService.processedCredentials?.data?.csrfToken || ""
|
|
422
|
+
};
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
WebSocketChatService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: WebSocketChatService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
426
|
+
WebSocketChatService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: WebSocketChatService });
|
|
427
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: WebSocketChatService, decorators: [{
|
|
428
|
+
type: Injectable
|
|
429
|
+
}], ctorParameters: function () { return []; } });
|
|
430
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-chat.service.js","sourceRoot":"","sources":["../../../../projects/assistant/chat/websocket-chat.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from \"@angular/core\";\nimport { AuthenticationService } from \"@sinequa/core/login\";\nimport { ConnectionOptions, SignalRWebService } from \"@sinequa/core/web-services\";\nimport { HttpTransportType, HubConnection, LogLevel, MessageHeaders } from \"@microsoft/signalr\";\nimport { ActionMessage, ActionResultEvent, ActionStartEvent, ActionStopEvent, MessageEvent, ChatMessage, ChatPayload, ChatResponse, GllmFunction, GllmModelDescription, MessageHandler, HistoryEvent, ErrorEvent, QuotaEvent, ChatContextAttachment, ContextMessageEvent, SavedChatHistory } from \"./types\";\nimport { ChatService } from \"./chat.service\";\nimport { merge, fromEvent, Observable, Subject, catchError, filter, forkJoin, map, shareReplay, switchMap, tap, throwError, takeUntil } from \"rxjs\";\n\n@Injectable()\nexport class WebSocketChatService extends ChatService {\n\n  public connection: HubConnection | undefined;\n  public connectionBuilt$ = new Subject<void>(); // Emit when the connection is built\n  public connectionStarted$ = new Subject<void>(); // Emit when the connection is started\n\n  private messageHandlers: Map<string, MessageHandler<any>> = new Map();\n\n  private actionMap = new Map<string, ActionMessage>();\n  private content = \"\";\n  private executionTime: string;\n  private attachments: ChatContextAttachment[] = [];\n\n  public signalRService = inject(SignalRWebService);\n  public authenticationService = inject(AuthenticationService);\n\n  constructor() {\n    super();\n  }\n\n  /**\n   * Initialize the chat process after the login is complete.\n   * It includes building and starting a connection, executing parallel requests for models and functions, and handling errors during the process.\n   *\n   * @returns An Observable<boolean> indicating the success of the initialization process.\n   */\n  init(): Observable<boolean> {\n    return this.loginService.events.pipe(\n      filter((e) => e.type === 'login-complete'),\n      tap(() => this.getRequestsUrl()),\n      // Build the connection and handle the completion with the connectionBuilt$ subject\n      switchMap(() => this.buildConnection()),\n      tap(() => {\n        this.initMessageHandlers();\n        this.connectionBuilt$.next();\n      }),\n      // Start the connection and handle the completion with the connectionStarted$ subject\n      switchMap(() => this.startConnection()),\n      // Execute parallel requests for models and functions\n      switchMap(() => {\n        this.connectionStarted$.next();\n        return 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        this.initProcess$.next(true);\n        return !!models && !!functions\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      // Cache and replay the emitted value for subsequent subscribers\n      shareReplay(1)\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.chatConfig$.value!.globalSettings.websocketEndpoint) {\n      this.REQUEST_URL = this.chatConfig$.value!.globalSettings.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  listModels(): Observable<GllmModelDescription[] | undefined> {\n    const modelsSubject = new Subject<GllmModelDescription[] | undefined>();\n\n    this.connection!.on('ListModels', (res) => {\n      this.models = res.models?.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.chatConfig$.value!.debug })\n      .catch(error => {\n        console.error('Error invoking ListModels:', error);\n        modelsSubject.complete()\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection\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;\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.chatConfig$.value!.debug })\n      .catch(error => {\n        console.error('Error invoking ListFunctions:', error);\n        functionsSubject.complete()\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection\n      })\n\n    return functionsSubject.asObservable();\n  }\n\n  fetch(messages: ChatMessage[], query = this.searchService.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.chatConfig$.value!.functions,\n      debug: this.chatConfig$.value!.debug,\n      serviceSettings: this.chatConfig$.value!.serviceSettings,\n      contextSettings: {\n        ...this.chatConfig$.value!.contextSettings,\n        app: this.appService.appName,\n        query\n      }\n    }\n    if (this.chatConfig$.value!.saveChats) {\n      data.instanceId = this.chatInstanceId;\n      data.savedChatId = this.savedChatId;\n    }\n\n    let response: ChatMessage = {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]) => {\n        return fromEvent<any>(this.connection!, eventName).pipe(\n          map((event) => eventHandler.handler(event)) // Execute the corresponding handler\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      takeUntil(completion$) // Complete the observable when completion$ emits\n    );\n\n    // Invoke the Chat method\n    this.connection!.invoke('Chat', data)\n      .then(() => this.notifyAudit(this.chatHistory!, this.chatConfig$.value!.serviceSettings.service_id)) // When the server indicates it has successfully finished invoking the method, notify the audit service with the recent chat history\n      .catch(error => {\n        console.error('Error invoking Chat:', error);\n        return Promise.resolve(); // Return a resolved promise to handle the error and prevent unhandled promise rejection\n      })\n      .finally(() => {\n        this.streaming$.next(false); // Complete streaming regardless of success or error\n        this.actionMap.clear(); // Clear the actionMap\n        this.content = \"\"; // Clear the content\n        this.attachments = []; // Clear the attachments\n        this.executionTime = \"\"; // Clear the executionTime\n        completion$.next(); // Emit a signal to complete the observables\n        completion$.complete(); // Complete the subject\n      });\n\n    // Return the merged observables\n    return combined$.pipe(\n      map(() => {\n        // Define $progress from the actionMap\n        const actions = Array.from(this.actionMap.values());\n        const $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        // Define the attachment used in the context of the response message\n        const $attachment = this.attachments;\n\n        // As soon as the first content or $progress is defined, the assistant is considered as streaming\n        if(!!this.content || $progress || $attachment.length > 0) {\n          response = {...response, content: this.content, additionalProperties: {...response.additionalProperties, $progress, $attachment} };\n        }\n\n        // Return the result\n        return { history: [...messages, response], executionTime: this.executionTime };\n      })\n    );\n  }\n\n  listSavedChat(): void {\n    const data = {\n      instanceId: this.chatInstanceId,\n      debug: this.chatConfig$.value!.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      instanceId: this.chatInstanceId,\n      savedChatId: id,\n      debug: this.chatConfig$.value!.debug\n    };\n\n    this.connection!.on('SavedChatGet', (res) => {\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.complete()\n        return Promise.resolve();\n      })\n\n    return savedChatSubject.asObservable();\n  }\n\n  deleteSavedChat(ids: string[]): Observable<number> {\n    const deleteSavedChatSubject = new Subject<number>();\n\n    const data = {\n      instanceId: this.chatInstanceId,\n      SavedChatIds: ids,\n      debug: this.chatConfig$.value!.debug\n    };\n\n    this.connection!.on('SavedChatDelete', (res) => {\n      deleteSavedChatSubject.next(res.deleteCount);\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.complete();\n        return Promise.resolve();\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 a single message or a result\n   */\n  initMessageHandlers() {\n    this.addMessageHandler(\n      \"Debug\",\n      { handler: (debug: any) => console.log(debug),\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) => this.attachments.push(message.metadata),\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"Message\",\n      {\n        handler: (message: MessageEvent) => this.content += message ?? \"\",\n        isGlobalHandler: false\n      }\n    );\n    this.addMessageHandler(\n      \"History\",\n      {\n        handler: (history: HistoryEvent) => {\n          this.chatHistory = history.history\n          this.executionTime = history.executionTime;\n        },\n        isGlobalHandler: false\n      }\n    );\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          if(message.quota.maxQuotaReached) {\n            const msg = `Sorry, you have exceeded the allowed quota. Please retry starting from ${this.formatDateTime(message.quota.nextResetUTC)}.`;\n            console.error(msg);\n            this.notificationsService.error(msg);\n          }\n        },\n        isGlobalHandler: true\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 chat 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 \"isChatGlobalHandler\" 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);\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.chatConfig$.value?.globalSettings.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.chatConfig$.value?.globalSettings.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.intlService.currentLocale.name,\n      \"ui-language\": this.intlService.currentLocale.name,\n    };\n    if (this.authenticationService.processedCredentials) {\n      headers = {...headers, \"sinequa-csrf-token\": this.authenticationService.processedCredentials.data.csrfToken};\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: () => this.authenticationService.processedCredentials?.data?.csrfToken || \"\"\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
// We'd like to not have a primary entry point but ng-packagr requires one currently
|
|
2
|
+
export const sinequaAssistantModule = undefined;
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3Byb2plY3RzL2Fzc2lzdGFudC9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLG9GQUFvRjtBQUNwRixNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBRyxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBXZSdkIGxpa2UgdG8gbm90IGhhdmUgYSBwcmltYXJ5IGVudHJ5IHBvaW50IGJ1dCBuZy1wYWNrYWdyIHJlcXVpcmVzIG9uZSBjdXJyZW50bHlcbmV4cG9ydCBjb25zdCBzaW5lcXVhQXNzaXN0YW50TW9kdWxlID0gdW5kZWZpbmVkO1xuIl19
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './public-api';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZXF1YS1hc3Npc3RhbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9wcm9qZWN0cy9hc3Npc3RhbnQvc2luZXF1YS1hc3Npc3RhbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcbiJdfQ==
|