@sinequa/assistant 3.2.0 → 3.2.2
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 +1 -0
- package/chat/chat.component.d.ts +13 -4
- package/chat/chat.service.d.ts +12 -5
- package/chat/rest-chat.service.d.ts +2 -1
- package/chat/saved-chats/saved-chats.component.d.ts +0 -1
- package/chat/websocket-chat.service.d.ts +13 -11
- package/esm2020/chat/chat-message/chat-message.component.mjs +23 -3
- package/esm2020/chat/chat.component.mjs +54 -53
- package/esm2020/chat/chat.service.mjs +1 -1
- package/esm2020/chat/rest-chat.service.mjs +38 -14
- package/esm2020/chat/saved-chats/saved-chats.component.mjs +4 -5
- package/esm2020/chat/websocket-chat.service.mjs +73 -40
- package/fesm2015/sinequa-assistant-chat.mjs +186 -110
- package/fesm2015/sinequa-assistant-chat.mjs.map +1 -1
- package/fesm2020/sinequa-assistant-chat.mjs +186 -109
- package/fesm2020/sinequa-assistant-chat.mjs.map +1 -1
- package/package.json +1 -1
|
@@ -145,6 +145,7 @@ export declare class ChatMessageComponent implements OnChanges, AfterViewInit {
|
|
|
145
145
|
*/
|
|
146
146
|
referencePlugin: (tree: Node) => Node<import("unist").Data>;
|
|
147
147
|
placeholderPlugin: (tree: Node) => Node<import("unist").Data>;
|
|
148
|
+
getLinkText(node: any): string;
|
|
148
149
|
/**
|
|
149
150
|
* Reformat [ids: 12.2, 42.5] to [12.2][42.5]
|
|
150
151
|
*/
|
package/chat/chat.component.d.ts
CHANGED
|
@@ -933,6 +933,13 @@ export declare class ChatComponent extends AbstractFacet implements OnInit, OnCh
|
|
|
933
933
|
instanceId: string;
|
|
934
934
|
/** Define the query to use to fetch answers */
|
|
935
935
|
query: Query;
|
|
936
|
+
/** Function that determines whether the chat should be reloaded after the query changes
|
|
937
|
+
* If not provided, the chat will be reloaded by default
|
|
938
|
+
* @param prevQuery The previous query
|
|
939
|
+
* @param newQuery The new query
|
|
940
|
+
* @returns true if the chat should be reloaded, false otherwise
|
|
941
|
+
*/
|
|
942
|
+
queryChangeShouldTriggerReload: (prevQuery: any, newQuery: any) => boolean;
|
|
936
943
|
/** Define the protocol to be used for this chat instance*/
|
|
937
944
|
protocol: 'REST' | 'WEBSOCKET';
|
|
938
945
|
/** Map of listeners overriding default registered ones*/
|
|
@@ -1052,7 +1059,7 @@ export declare class ChatComponent extends AbstractFacet implements OnInit, OnCh
|
|
|
1052
1059
|
error: EventEmitter<any>;
|
|
1053
1060
|
_config: EventEmitter<ChatConfig>;
|
|
1054
1061
|
messageList?: ElementRef<HTMLUListElement>;
|
|
1055
|
-
questionInput?: ElementRef<
|
|
1062
|
+
questionInput?: ElementRef<HTMLTextAreaElement>;
|
|
1056
1063
|
loadingTpl?: TemplateRef<any>;
|
|
1057
1064
|
chatService: ChatService;
|
|
1058
1065
|
config: ChatConfig;
|
|
@@ -1079,6 +1086,7 @@ export declare class ChatComponent extends AbstractFacet implements OnInit, OnCh
|
|
|
1079
1086
|
instantiateChatService(): void;
|
|
1080
1087
|
get actions(): Action[];
|
|
1081
1088
|
private handleChanges;
|
|
1089
|
+
private triggerReloadAfterQueryChange;
|
|
1082
1090
|
private addScrollListener;
|
|
1083
1091
|
updateModelDescription(): void;
|
|
1084
1092
|
submitQuestion(): void;
|
|
@@ -1098,6 +1106,7 @@ export declare class ChatComponent extends AbstractFacet implements OnInit, OnCh
|
|
|
1098
1106
|
scrollDown(): void;
|
|
1099
1107
|
/**
|
|
1100
1108
|
* Start a new chat with the defaultValues settings
|
|
1109
|
+
* The savedChatId in the chat service will be reset, so that the upcoming saved chat operations will be performed on the fresh new chat
|
|
1101
1110
|
* If the savedChat feature is enabled, the list of saved chats will be refreshed
|
|
1102
1111
|
*/
|
|
1103
1112
|
newChat(): void;
|
|
@@ -1111,7 +1120,7 @@ export declare class ChatComponent extends AbstractFacet implements OnInit, OnCh
|
|
|
1111
1120
|
* If the last message is from the user, a request to the assistant is made to get an answer
|
|
1112
1121
|
* If the last message is from the assistant, the conversation is loaded right away
|
|
1113
1122
|
* @param messages The list of messages of the chat
|
|
1114
|
-
* @param chatId The id of the chat. If
|
|
1123
|
+
* @param chatId The id of the chat. If provided (ie. an existing discussion in the saved chat index), update the savedChatId in the chat service for the upcoming saved chat operations
|
|
1115
1124
|
*/
|
|
1116
1125
|
openChat(messages: RawMessage[], chatId?: string): void;
|
|
1117
1126
|
/**
|
|
@@ -1125,7 +1134,7 @@ export declare class ChatComponent extends AbstractFacet implements OnInit, OnCh
|
|
|
1125
1134
|
editMessage(index: number): void;
|
|
1126
1135
|
regenerateMessage(index: number): void;
|
|
1127
1136
|
onKeyUp(event: KeyboardEvent): void;
|
|
1128
|
-
|
|
1137
|
+
calculateHeight(): void;
|
|
1129
1138
|
static ɵfac: i0.ɵɵFactoryDeclaration<ChatComponent, never>;
|
|
1130
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<ChatComponent, "sq-chat-v3", never, { "instanceId": "instanceId"; "query": "query"; "protocol": "protocol"; "messageHandlers": "messageHandlers"; "automaticScrollToLastResponse": "automaticScrollToLastResponse"; "chat": "chat"; "customAssistantIcon": "customAssistantIcon"; }, { "data": "data"; "referenceClicked": "referenceClicked"; "openPreview": "openPreview"; "loading$": "loading"; "error": "error"; "_config": "config"; }, ["loadingTpl"], never, true>;
|
|
1139
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ChatComponent, "sq-chat-v3", never, { "instanceId": "instanceId"; "query": "query"; "queryChangeShouldTriggerReload": "queryChangeShouldTriggerReload"; "protocol": "protocol"; "messageHandlers": "messageHandlers"; "automaticScrollToLastResponse": "automaticScrollToLastResponse"; "chat": "chat"; "customAssistantIcon": "customAssistantIcon"; }, { "data": "data"; "referenceClicked": "referenceClicked"; "openPreview": "openPreview"; "loading$": "loading"; "error": "error"; "_config": "config"; }, ["loadingTpl"], never, true>;
|
|
1131
1140
|
}
|
package/chat/chat.service.d.ts
CHANGED
|
@@ -31,7 +31,7 @@ export declare abstract class ChatService {
|
|
|
31
31
|
loadSavedChat$: BehaviorSubject<SavedChat | undefined>;
|
|
32
32
|
/** Instance ID of the chat service defining the assistant instance */
|
|
33
33
|
private _chatInstanceId;
|
|
34
|
-
/** ID of the current chat discussion which is used to
|
|
34
|
+
/** ID of the current chat discussion which is used to update/get/delete it */
|
|
35
35
|
private _savedChatId;
|
|
36
36
|
userSettingsService: UserSettingsWebService;
|
|
37
37
|
notificationsService: NotificationsService;
|
|
@@ -64,12 +64,12 @@ export declare abstract class ChatService {
|
|
|
64
64
|
* Get the ID of the current chat discussion which is used to save/get/delete it
|
|
65
65
|
* @returns The ID of the current chat discussion
|
|
66
66
|
*/
|
|
67
|
-
get savedChatId(): string;
|
|
67
|
+
get savedChatId(): string | undefined;
|
|
68
68
|
/**
|
|
69
69
|
* Persist the ID of the current chat discussion which is used to save/get/delete it
|
|
70
70
|
* @param savedChatId The ID of the current chat discussion which is used to save/get/delete it
|
|
71
71
|
*/
|
|
72
|
-
setSavedChatId(savedChatId: string): void;
|
|
72
|
+
setSavedChatId(savedChatId: string | undefined): void;
|
|
73
73
|
/**
|
|
74
74
|
* Initialize the chat config by managing ONLY sub-object **defaultValues** configs of the standard app config (defined in the customization json tab ) and the user preferences.
|
|
75
75
|
* To do so, a tracking mechanism is implemented to notify the user about the available updates in the defaultValues object of the standard app config.
|
|
@@ -118,13 +118,20 @@ export declare abstract class ChatService {
|
|
|
118
118
|
* @param id The id of the saved chat
|
|
119
119
|
*/
|
|
120
120
|
abstract getSavedChat(id: string): Observable<SavedChatHistory | undefined>;
|
|
121
|
+
/**
|
|
122
|
+
* Save a chat with the given messages
|
|
123
|
+
* @param messages The messages to add to the saved chat index
|
|
124
|
+
* @returns The saved chat
|
|
125
|
+
*/
|
|
126
|
+
abstract addSavedChat(messages: ChatMessage[]): Observable<SavedChat>;
|
|
121
127
|
/**
|
|
122
128
|
* Update a saved chat with the given id.
|
|
123
129
|
* @param id The id of the saved chat
|
|
124
|
-
* @param name The new name of the saved chat
|
|
130
|
+
* @param name The new name of the saved chat, if provided
|
|
131
|
+
* @param messages The messages to update the saved chat history, if provided
|
|
125
132
|
* @returns True if the saved chat has been successfully updated
|
|
126
133
|
*/
|
|
127
|
-
abstract updateSavedChat(id: string, name
|
|
134
|
+
abstract updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChat>;
|
|
128
135
|
/**
|
|
129
136
|
* Bulk delete of saved chats matching the given ids
|
|
130
137
|
* @param ids List of ids of the saved chats to delete
|
|
@@ -22,8 +22,9 @@ export declare class RestChatService extends ChatService {
|
|
|
22
22
|
listFunctions(): Observable<GllmFunction[] | undefined>;
|
|
23
23
|
fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse>;
|
|
24
24
|
listSavedChat(): void;
|
|
25
|
+
addSavedChat(messages: ChatMessage[]): Observable<SavedChat>;
|
|
25
26
|
getSavedChat(id: string): Observable<SavedChatHistory | undefined>;
|
|
26
|
-
updateSavedChat(id: string, name
|
|
27
|
+
updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChat>;
|
|
27
28
|
deleteSavedChat(ids: string[]): Observable<number>;
|
|
28
29
|
static ɵfac: i0.ɵɵFactoryDeclaration<RestChatService, never>;
|
|
29
30
|
static ɵprov: i0.ɵɵInjectableDeclaration<RestChatService>;
|
|
@@ -24,7 +24,6 @@ export declare class SavedChatsComponent implements OnInit, OnDestroy {
|
|
|
24
24
|
auditService: AuditWebService;
|
|
25
25
|
modalService: ModalService;
|
|
26
26
|
notificationsService: NotificationsService;
|
|
27
|
-
currentChatId?: string;
|
|
28
27
|
ngOnInit(): void;
|
|
29
28
|
ngOnDestroy(): void;
|
|
30
29
|
instantiateChatService(): void;
|
|
@@ -10,11 +10,12 @@ export declare class WebSocketChatService extends ChatService {
|
|
|
10
10
|
connection: HubConnection | undefined;
|
|
11
11
|
connectionBuilt$: Subject<void>;
|
|
12
12
|
connectionStarted$: Subject<void>;
|
|
13
|
-
private
|
|
14
|
-
private
|
|
15
|
-
private
|
|
16
|
-
private
|
|
17
|
-
private
|
|
13
|
+
private _messageHandlers;
|
|
14
|
+
private _actionMap;
|
|
15
|
+
private _progress;
|
|
16
|
+
private _content;
|
|
17
|
+
private _executionTime;
|
|
18
|
+
private _attachments;
|
|
18
19
|
signalRService: SignalRWebService;
|
|
19
20
|
authenticationService: AuthenticationService;
|
|
20
21
|
constructor();
|
|
@@ -35,7 +36,8 @@ export declare class WebSocketChatService extends ChatService {
|
|
|
35
36
|
fetch(messages: ChatMessage[], query: Query): Observable<ChatResponse>;
|
|
36
37
|
listSavedChat(): void;
|
|
37
38
|
getSavedChat(id: string): Observable<SavedChatHistory | undefined>;
|
|
38
|
-
|
|
39
|
+
addSavedChat(messages: ChatMessage[]): Observable<SavedChat>;
|
|
40
|
+
updateSavedChat(id: string, name?: string, messages?: ChatMessage[]): Observable<SavedChat>;
|
|
39
41
|
deleteSavedChat(ids: string[]): Observable<number>;
|
|
40
42
|
/**
|
|
41
43
|
* Initialize out-of-the-box handlers
|
|
@@ -43,10 +45,10 @@ export declare class WebSocketChatService extends ChatService {
|
|
|
43
45
|
*/
|
|
44
46
|
initMessageHandlers(): void;
|
|
45
47
|
/**
|
|
46
|
-
* Override and register the entire
|
|
47
|
-
* @param
|
|
48
|
+
* Override and register the entire _messageHandlers map by merging the provided map with the default one
|
|
49
|
+
* @param _messageHandlers
|
|
48
50
|
*/
|
|
49
|
-
overrideMessageHandlers<T>(
|
|
51
|
+
overrideMessageHandlers<T>(_messageHandlers: Map<string, MessageHandler<T>>): void;
|
|
50
52
|
/**
|
|
51
53
|
* Add a listener for a specific event.
|
|
52
54
|
* If a listener for this same event already exists, it will be overridden.
|
|
@@ -63,14 +65,14 @@ export declare class WebSocketChatService extends ChatService {
|
|
|
63
65
|
*/
|
|
64
66
|
protected registerMessageHandler<T>(eventName: string, eventHandler: MessageHandler<T>): void;
|
|
65
67
|
/**
|
|
66
|
-
* Remove a listener for a specific event from the
|
|
68
|
+
* Remove a listener for a specific event from the _messageHandlers map and unsubscribe from receiving messages for this event from the SignalR hub.
|
|
67
69
|
* @param eventName Name of the event to remove the listener for
|
|
68
70
|
*/
|
|
69
71
|
removeMessageHandler(eventName: string): void;
|
|
70
72
|
/**
|
|
71
73
|
* Unsubscribe from receiving messages for a specific event from the SignalR hub.
|
|
72
74
|
* ALL its related listeners will be removed from hub connection
|
|
73
|
-
* This is needed to prevent accumulating old listeners when overriding the entire
|
|
75
|
+
* This is needed to prevent accumulating old listeners when overriding the entire _messageHandlers map
|
|
74
76
|
* @param eventName Name of the event
|
|
75
77
|
*/
|
|
76
78
|
protected unsubscribeMessageHandler(eventName: string): void;
|
|
@@ -135,6 +135,26 @@ export class ChatMessageComponent {
|
|
|
135
135
|
}
|
|
136
136
|
this.referenceClicked.emit(record);
|
|
137
137
|
}
|
|
138
|
+
getLinkText(node) {
|
|
139
|
+
if (node.text) {
|
|
140
|
+
return node.text; // Return directly if text is provided in node.text ([Example link](https://example.com))
|
|
141
|
+
}
|
|
142
|
+
else if (node.children && node.children.length > 0) {
|
|
143
|
+
// Recursively search for text content in child nodes
|
|
144
|
+
for (const child of node.children) {
|
|
145
|
+
if (child.type === 'text' && child.value) {
|
|
146
|
+
return child.value; // Return the value of the first text node found ([**Emphasized Link Text**](https://example.com))
|
|
147
|
+
}
|
|
148
|
+
else if (child.children && child.children.length > 0) {
|
|
149
|
+
const textContent = this.getLinkText(child); // Recursively search child nodes ([](https://example.com))
|
|
150
|
+
if (textContent) {
|
|
151
|
+
return textContent; // Return text content if found
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
return 'link'; // Return empty string if no text content is found
|
|
157
|
+
}
|
|
138
158
|
/**
|
|
139
159
|
* Reformat [ids: 12.2, 42.5] to [12.2][42.5]
|
|
140
160
|
*/
|
|
@@ -152,11 +172,11 @@ export class ChatMessageComponent {
|
|
|
152
172
|
}
|
|
153
173
|
}
|
|
154
174
|
ChatMessageComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatMessageComponent, deps: [{ token: i1.SearchService }, { token: i2.UIService }, { token: i3.PrincipalWebService }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
155
|
-
ChatMessageComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatMessageComponent, isStandalone: true, selector: "sq-chat-message", inputs: { message: "message", conversation: "conversation", assistantIcon: "assistantIcon", streaming: "streaming", canEdit: "canEdit", canRegenerate: "canRegenerate", canCopy: "canCopy" }, outputs: { referenceClicked: "referenceClicked", edit: "edit", regenerate: "regenerate", openPreview: "openPreview" }, usesOnChanges: true, ngImport: i0, template: "<!-- Message icon -->\n<span class=\"message-icon\" [title]=\"message.role\">\n <i [ngClass]=\"assistantIcon\" [style.--sq-size.px]=\"24\" *ngIf=\"message.role === 'assistant'\"></i>\n <sq-initials-avatar [fullName]=\"name\" *ngIf=\"message.role !== 'assistant'\"></sq-initials-avatar>\n</span>\n\n<!-- Message body -->\n<div class=\"flex-grow-1\" style=\"min-width: 0;\" [ngClass]=\"'message-'+message.role\">\n\n <div *ngIf=\"message.additionalProperties.$progress as progress\" class=\"small ms-3 mb-2\">\n <a href=\"#\" (click)=\"collapseProgress = !collapseProgress; false\" class=\"text-muted\">\n View progress\n <i class=\"fas fa-chevron-{{collapseProgress? 'right' : 'down'}}\"></i>\n </a>\n <sq-collapse [collapsed]=\"collapseProgress\">\n <ng-template>\n <ul class=\"list-unstyled\">\n <li *ngFor=\"let step of progress\">\n <i class=\"fas fa-fw fa-check text-success\" *ngIf=\"step.done\"></i>\n <i class=\"fas fa-fw fa-spin fa-circle-notch\" *ngIf=\"!step.done\"></i>\n <span class=\"ms-2 fw-bold\">{{step.title}}</span>\n <span *ngIf=\"step.content\">: {{step.content}}</span>\n </li>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n <div class=\"message-content\" *ngIf=\"message.content\">\n\n <remark *ngIf=\"message.role === 'assistant'\" [markdown]=\"message.content\" [processor]=\"processor\">\n\n <ng-template remarkTemplate=\"chat-reference\" let-ref>\n\n <a *ngIf=\"referenceMap.get(ref.refId) as attachment; else staticRefTpl\"\n (click)=\"openDocument(attachment.record)\"\n class=\"reference\"\n [sqTooltip]=\"attachment\"\n [sqTooltipTemplate]=\"tooltipTpl\"\n [hoverableTooltip]=\"true\"\n >{{ref.refId}}</a>\n\n <ng-template #staticRefTpl>\n <span class=\"reference\">{{ref.refId}}</span>\n </ng-template>\n\n </ng-template>\n\n <ng-template remarkTemplate=\"streaming-placeholder\">\n <span class=\"placeholder-glow\" *ngIf=\"streaming\">\n <span class=\"placeholder ms-1\"></span>\n </span>\n </ng-template>\n\n <ng-template remarkTemplate=\"code\" let-node>\n <div class=\"card mb-2\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <span>{{node.lang}}</span>\n <button class=\"btn btn-light btn-sm\" (click)=\"copyToClipboard(node.value)\"><i class=\"far fa-fw fa-clipboard\"></i> Copy code</button>\n </div>\n <pre class=\"language-{{node.lang}} my-0 rounded-0 rounded-bottom\"><code class=\"language-{{node.lang}}\">{{node.value}}</code></pre>\n </div>\n </ng-template>\n\n </remark>\n\n <p *ngIf=\"message.role === 'user'\">{{message.content}}</p>\n\n <!-- List of reference, if any -->\n <div *ngIf=\"references?.length\" class=\"references\">\n <span class=\"references-title\">References</span> <i class=\"fas references-collapse\" [class.fa-chevron-down]=\"showReferences\" [class.fa-chevron-right]=\"!showReferences\" (click)=\"showReferences=!showReferences\"></i>\n <sq-collapse [collapsed]=\"!showReferences\">\n <ng-template>\n <ul>\n <ng-container *ngFor=\"let reference of references\">\n <li *ngIf=\"referenceMap.get(reference) as attachment\" class=\"text-truncate\">\n <sq-chat-reference [class.expanded]=\"attachment.$expanded\" [attachment]=\"attachment\" [reference]=\"reference\"\n (openPreview)=\"openPreview.emit($event)\" (openDocument)=\"openDocument($event)\"></sq-chat-reference>\n </li>\n </ng-container>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n \n </div>\n\n <!-- Edit / Regenerate floating actions -->\n <div class=\"sq-chat-message-action\">\n <button class=\"btn btn-sm\" *ngIf=\"canEdit\" sqTooltip=\"Edit message\"\n (click)=\"edit.emit(message)\">\n <i class=\"fas fa-edit\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canCopy\" sqTooltip=\"Copy to clipboard\"\n (click)=\"copyToClipboard(message.content)\">\n <i class=\"far fa-clipboard\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canRegenerate\" sqTooltip=\"Regenerate message\"\n (click)=\"regenerate.emit(message)\">\n <i class=\"fas fa-sync-alt\"></i>\n </button>\n </div>\n\n <ng-template #tooltipTpl let-ref>\n <sq-chat-reference class=\"expanded\"\n [attachment]=\"ref\"\n [reference]=\"ref.contextId\"\n [partId]=\"ref.$partId\"\n (openPreview)=\"openPreview.emit($event)\"\n (openDocument)=\"openDocument($event)\">\n </sq-chat-reference>\n </ng-template>\n\n</div>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}:host{display:flex}:host:not(:hover) .sq-chat-message-action{display:none}.message-content{padding:var(--ast-message-padding, var(--ast-size-3, .75rem));border-radius:var(--ast-message-border-radius, var(--ast-size-4, 1rem));display:inline-block;max-width:100%}.message-content .references{margin-top:var(--ast-size-5, 1.25rem)}.message-content .references ul{border-left:.2rem solid var(--ast-secondary-color, #FF732E);padding-left:var(--ast-size-5, 1.25rem);margin-top:var(--ast-size-2, .5rem)}.message-content .references .references-title{font-weight:var(--font-weight-bold, 500)}.message-content .references .references-collapse{cursor:pointer;margin-left:var(--ast-size-2, .5rem)}.message-content ::ng-deep p:last-child{margin-bottom:0}.message-content ::ng-deep .placeholder-glow .placeholder{animation-duration:.4s;width:12px;height:var(--ast-size-4, 1rem);vertical-align:text-bottom}.message-content ::ng-deep img{max-width:100%}.message-content ::ng-deep table{display:block;border:1px solid;border-color:var(--ast-message-table-border-color, #ccc);border-collapse:collapse;margin:0;padding:0;min-width:100%;overflow-x:auto;table-layout:fixed}.message-content ::ng-deep table tr{background-color:var(--ast-message-table-tr-bg, #f8f8f8);border:1px solid;border-color:var(--ast-message-table-tr-border-color, #ddd);padding:.35em}.message-content ::ng-deep table th,.message-content ::ng-deep table td{padding:.625em;text-align:center}.message-content ::ng-deep table th{font-size:.85em;letter-spacing:.1em;text-transform:uppercase}.message-content ::ng-deep .reference{color:var(--ast-message-reference-color, black)!important}.message-assistant .message-content{background:var(--ast-secondary-bg, #FFF8F1)}.message-user .message-content{background:var(--ast-primary-bg, #f2f8fe);font-weight:var(--ast-user-font-weight, var(--font-weight-bold, 500))}.sq-chat-message-action{position:absolute;right:calc(0rem - var(--ast-size-2, .5rem));top:var(--ast-size-2, .5rem);display:flex;flex-direction:column}.message-icon{margin-top:var(--ast-size-3, .75rem);margin-right:var(--ast-size-4, 1rem)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }, { kind: "ngmodule", type: CollapseModule }, { kind: "component", type: i5.Collapse, selector: "sq-collapse", inputs: ["collapsed"] }, { kind: "ngmodule", type: RemarkModule }, { kind: "component", type: i6.RemarkComponent, selector: "remark", inputs: ["markdown", "processor", "debug"] }, { kind: "directive", type: i6.RemarkTemplateDirective, selector: "[remarkTemplate]", inputs: ["remarkTemplate"] }, { kind: "component", type: InitialsAvatarComponent, selector: "sq-initials-avatar", inputs: ["fullName", "size"] }, { kind: "component", type: ChatReferenceComponent, selector: "sq-chat-reference", inputs: ["reference", "attachment", "partId"], outputs: ["openDocument", "openPreview"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
175
|
+
ChatMessageComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.0.6", type: ChatMessageComponent, isStandalone: true, selector: "sq-chat-message", inputs: { message: "message", conversation: "conversation", assistantIcon: "assistantIcon", streaming: "streaming", canEdit: "canEdit", canRegenerate: "canRegenerate", canCopy: "canCopy" }, outputs: { referenceClicked: "referenceClicked", edit: "edit", regenerate: "regenerate", openPreview: "openPreview" }, usesOnChanges: true, ngImport: i0, template: "<!-- Message icon -->\n<span class=\"message-icon\" [title]=\"message.role\">\n <i [ngClass]=\"assistantIcon\" [style.--sq-size.px]=\"24\" *ngIf=\"message.role === 'assistant'\"></i>\n <sq-initials-avatar [fullName]=\"name\" *ngIf=\"message.role !== 'assistant'\"></sq-initials-avatar>\n</span>\n\n<!-- Message body -->\n<div class=\"flex-grow-1\" style=\"min-width: 0;\" [ngClass]=\"'message-'+message.role\">\n\n <div *ngIf=\"message.additionalProperties.$progress as progress\" class=\"small ms-3 mb-2\">\n <a href=\"#\" (click)=\"collapseProgress = !collapseProgress; false\" class=\"text-muted\">\n View progress\n <i class=\"fas fa-chevron-{{collapseProgress? 'right' : 'down'}}\"></i>\n </a>\n <sq-collapse [collapsed]=\"collapseProgress\">\n <ng-template>\n <ul class=\"list-unstyled\">\n <li *ngFor=\"let step of progress\">\n <i class=\"fas fa-fw fa-check text-success\" *ngIf=\"step.done\"></i>\n <i class=\"fas fa-fw fa-spin fa-circle-notch\" *ngIf=\"!step.done\"></i>\n <span class=\"ms-2 fw-bold\">{{step.title}}</span>\n <span *ngIf=\"step.content\">: {{step.content}}</span>\n </li>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n <div class=\"message-content\" *ngIf=\"message.content\">\n\n <remark *ngIf=\"message.role === 'assistant'\" [markdown]=\"message.content\" [processor]=\"processor\">\n\n <ng-template remarkTemplate=\"chat-reference\" let-ref>\n\n <a *ngIf=\"referenceMap.get(ref.refId) as attachment; else staticRefTpl\"\n (click)=\"openDocument(attachment.record)\"\n class=\"reference\"\n [sqTooltip]=\"attachment\"\n [sqTooltipTemplate]=\"tooltipTpl\"\n [hoverableTooltip]=\"true\"\n >{{ref.refId}}</a>\n\n <ng-template #staticRefTpl>\n <span class=\"reference\">{{ref.refId}}</span>\n </ng-template>\n\n </ng-template>\n\n <ng-template remarkTemplate=\"streaming-placeholder\">\n <span class=\"placeholder-glow\" *ngIf=\"streaming\">\n <span class=\"placeholder ms-1\"></span>\n </span>\n </ng-template>\n\n <ng-template remarkTemplate=\"code\" let-node>\n <div class=\"card mb-2\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <span>{{node.lang}}</span>\n <button class=\"btn btn-light btn-sm\" (click)=\"copyToClipboard(node.value)\"><i class=\"far fa-fw fa-clipboard\"></i> Copy code</button>\n </div>\n <pre class=\"language-{{node.lang}} my-0 rounded-0 rounded-bottom\"><code class=\"language-{{node.lang}}\">{{node.value}}</code></pre>\n </div>\n </ng-template>\n\n <ng-template remarkTemplate=\"link\" let-node>\n <a [href]=\"node.url\" target=\"_blank\" rel=\"noopener noreferrer\">{{getLinkText(node)}}</a>\n </ng-template>\n\n </remark>\n\n <p *ngIf=\"message.role === 'user'\">{{message.content}}</p>\n\n <!-- List of reference, if any -->\n <div *ngIf=\"references?.length\" class=\"references\">\n <span class=\"references-title\">References</span> <i class=\"fas references-collapse\" [class.fa-chevron-down]=\"showReferences\" [class.fa-chevron-right]=\"!showReferences\" (click)=\"showReferences=!showReferences\"></i>\n <sq-collapse [collapsed]=\"!showReferences\">\n <ng-template>\n <ul>\n <ng-container *ngFor=\"let reference of references\">\n <li *ngIf=\"referenceMap.get(reference) as attachment\" class=\"text-truncate\">\n <sq-chat-reference [class.expanded]=\"attachment.$expanded\" [attachment]=\"attachment\" [reference]=\"reference\"\n (openPreview)=\"openPreview.emit($event)\" (openDocument)=\"openDocument($event)\"></sq-chat-reference>\n </li>\n </ng-container>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n </div>\n\n <!-- Edit / Regenerate floating actions -->\n <div class=\"sq-chat-message-action\">\n <button class=\"btn btn-sm\" *ngIf=\"canEdit\" sqTooltip=\"Edit message\"\n (click)=\"edit.emit(message)\">\n <i class=\"fas fa-edit\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canCopy\" sqTooltip=\"Copy to clipboard\"\n (click)=\"copyToClipboard(message.content)\">\n <i class=\"far fa-clipboard\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canRegenerate\" sqTooltip=\"Regenerate message\"\n (click)=\"regenerate.emit(message)\">\n <i class=\"fas fa-sync-alt\"></i>\n </button>\n </div>\n\n <ng-template #tooltipTpl let-ref>\n <sq-chat-reference class=\"expanded\"\n [attachment]=\"ref\"\n [reference]=\"ref.contextId\"\n [partId]=\"ref.$partId\"\n (openPreview)=\"openPreview.emit($event)\"\n (openDocument)=\"openDocument($event)\">\n </sq-chat-reference>\n </ng-template>\n\n</div>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}:host{display:flex}:host:not(:hover) .sq-chat-message-action{display:none}.message-content{padding:var(--ast-message-padding, var(--ast-size-3, .75rem));border-radius:var(--ast-message-border-radius, var(--ast-size-4, 1rem));display:inline-block;max-width:100%}.message-content .references{margin-top:var(--ast-size-5, 1.25rem)}.message-content .references ul{border-left:.2rem solid var(--ast-secondary-color, #FF732E);padding-left:var(--ast-size-5, 1.25rem);margin-top:var(--ast-size-2, .5rem)}.message-content .references .references-title{font-weight:var(--font-weight-bold, 500)}.message-content .references .references-collapse{cursor:pointer;margin-left:var(--ast-size-2, .5rem)}.message-content ::ng-deep p:last-child{margin-bottom:0}.message-content ::ng-deep .placeholder-glow .placeholder{animation-duration:.4s;width:12px;height:var(--ast-size-4, 1rem);vertical-align:text-bottom}.message-content ::ng-deep img{max-width:100%}.message-content ::ng-deep table{display:block;border:1px solid;border-color:var(--ast-message-table-border-color, #ccc);border-collapse:collapse;margin:0;padding:0;min-width:100%;overflow-x:auto;table-layout:fixed}.message-content ::ng-deep table tr{background-color:var(--ast-message-table-tr-bg, #f8f8f8);border:1px solid;border-color:var(--ast-message-table-tr-border-color, #ddd);padding:.35em}.message-content ::ng-deep table th,.message-content ::ng-deep table td{padding:.625em;text-align:center}.message-content ::ng-deep table th{font-size:.85em;letter-spacing:.1em;text-transform:uppercase}.message-content ::ng-deep .reference{color:var(--ast-message-reference-color, black)!important}.message-content ::ng-deep ul,.message-content ::ng-deep ol{display:flex;flex-direction:column;gap:.5rem;padding-right:2rem;margin-left:0;margin-right:0;padding-left:40px;unicode-bidi:-webkit-isolate;unicode-bidi:isolate;list-style:disc}.message-content ::ng-deep p{margin-top:.5rem}.message-assistant .message-content{background:var(--ast-secondary-bg, #FFF8F1)}.message-user .message-content{background:var(--ast-primary-bg, #f2f8fe);font-weight:var(--ast-user-font-weight, var(--font-weight-bold, 500))}.message-user .message-content p{white-space:pre-line}.sq-chat-message-action{position:absolute;right:calc(0rem - var(--ast-size-2, .5rem));top:0;display:flex;flex-direction:column}.message-icon{margin-top:var(--ast-size-3, .75rem);margin-right:var(--ast-size-4, 1rem)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: UtilsModule }, { kind: "directive", type: i2.TooltipDirective, selector: "[sqTooltip]", inputs: ["sqTooltip", "sqTooltipData", "sqTooltipTemplate", "placement", "fallbackPlacements", "delay", "hoverableTooltip", "tooltipClass"] }, { kind: "ngmodule", type: CollapseModule }, { kind: "component", type: i5.Collapse, selector: "sq-collapse", inputs: ["collapsed"] }, { kind: "ngmodule", type: RemarkModule }, { kind: "component", type: i6.RemarkComponent, selector: "remark", inputs: ["markdown", "processor", "debug"] }, { kind: "directive", type: i6.RemarkTemplateDirective, selector: "[remarkTemplate]", inputs: ["remarkTemplate"] }, { kind: "component", type: InitialsAvatarComponent, selector: "sq-initials-avatar", inputs: ["fullName", "size"] }, { kind: "component", type: ChatReferenceComponent, selector: "sq-chat-reference", inputs: ["reference", "attachment", "partId"], outputs: ["openDocument", "openPreview"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
156
176
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImport: i0, type: ChatMessageComponent, decorators: [{
|
|
157
177
|
type: Component,
|
|
158
178
|
args: [{ selector: "sq-chat-message", changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [CommonModule, UtilsModule, CollapseModule, RemarkModule,
|
|
159
|
-
InitialsAvatarComponent, ChatReferenceComponent], template: "<!-- Message icon -->\n<span class=\"message-icon\" [title]=\"message.role\">\n <i [ngClass]=\"assistantIcon\" [style.--sq-size.px]=\"24\" *ngIf=\"message.role === 'assistant'\"></i>\n <sq-initials-avatar [fullName]=\"name\" *ngIf=\"message.role !== 'assistant'\"></sq-initials-avatar>\n</span>\n\n<!-- Message body -->\n<div class=\"flex-grow-1\" style=\"min-width: 0;\" [ngClass]=\"'message-'+message.role\">\n\n <div *ngIf=\"message.additionalProperties.$progress as progress\" class=\"small ms-3 mb-2\">\n <a href=\"#\" (click)=\"collapseProgress = !collapseProgress; false\" class=\"text-muted\">\n View progress\n <i class=\"fas fa-chevron-{{collapseProgress? 'right' : 'down'}}\"></i>\n </a>\n <sq-collapse [collapsed]=\"collapseProgress\">\n <ng-template>\n <ul class=\"list-unstyled\">\n <li *ngFor=\"let step of progress\">\n <i class=\"fas fa-fw fa-check text-success\" *ngIf=\"step.done\"></i>\n <i class=\"fas fa-fw fa-spin fa-circle-notch\" *ngIf=\"!step.done\"></i>\n <span class=\"ms-2 fw-bold\">{{step.title}}</span>\n <span *ngIf=\"step.content\">: {{step.content}}</span>\n </li>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n <div class=\"message-content\" *ngIf=\"message.content\">\n\n <remark *ngIf=\"message.role === 'assistant'\" [markdown]=\"message.content\" [processor]=\"processor\">\n\n <ng-template remarkTemplate=\"chat-reference\" let-ref>\n\n <a *ngIf=\"referenceMap.get(ref.refId) as attachment; else staticRefTpl\"\n (click)=\"openDocument(attachment.record)\"\n class=\"reference\"\n [sqTooltip]=\"attachment\"\n [sqTooltipTemplate]=\"tooltipTpl\"\n [hoverableTooltip]=\"true\"\n >{{ref.refId}}</a>\n\n <ng-template #staticRefTpl>\n <span class=\"reference\">{{ref.refId}}</span>\n </ng-template>\n\n </ng-template>\n\n <ng-template remarkTemplate=\"streaming-placeholder\">\n <span class=\"placeholder-glow\" *ngIf=\"streaming\">\n <span class=\"placeholder ms-1\"></span>\n </span>\n </ng-template>\n\n <ng-template remarkTemplate=\"code\" let-node>\n <div class=\"card mb-2\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <span>{{node.lang}}</span>\n <button class=\"btn btn-light btn-sm\" (click)=\"copyToClipboard(node.value)\"><i class=\"far fa-fw fa-clipboard\"></i> Copy code</button>\n </div>\n <pre class=\"language-{{node.lang}} my-0 rounded-0 rounded-bottom\"><code class=\"language-{{node.lang}}\">{{node.value}}</code></pre>\n </div>\n </ng-template>\n\n </remark>\n\n <p *ngIf=\"message.role === 'user'\">{{message.content}}</p>\n\n <!-- List of reference, if any -->\n <div *ngIf=\"references?.length\" class=\"references\">\n <span class=\"references-title\">References</span> <i class=\"fas references-collapse\" [class.fa-chevron-down]=\"showReferences\" [class.fa-chevron-right]=\"!showReferences\" (click)=\"showReferences=!showReferences\"></i>\n <sq-collapse [collapsed]=\"!showReferences\">\n <ng-template>\n <ul>\n <ng-container *ngFor=\"let reference of references\">\n <li *ngIf=\"referenceMap.get(reference) as attachment\" class=\"text-truncate\">\n <sq-chat-reference [class.expanded]=\"attachment.$expanded\" [attachment]=\"attachment\" [reference]=\"reference\"\n (openPreview)=\"openPreview.emit($event)\" (openDocument)=\"openDocument($event)\"></sq-chat-reference>\n </li>\n </ng-container>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n
|
|
179
|
+
InitialsAvatarComponent, ChatReferenceComponent], template: "<!-- Message icon -->\n<span class=\"message-icon\" [title]=\"message.role\">\n <i [ngClass]=\"assistantIcon\" [style.--sq-size.px]=\"24\" *ngIf=\"message.role === 'assistant'\"></i>\n <sq-initials-avatar [fullName]=\"name\" *ngIf=\"message.role !== 'assistant'\"></sq-initials-avatar>\n</span>\n\n<!-- Message body -->\n<div class=\"flex-grow-1\" style=\"min-width: 0;\" [ngClass]=\"'message-'+message.role\">\n\n <div *ngIf=\"message.additionalProperties.$progress as progress\" class=\"small ms-3 mb-2\">\n <a href=\"#\" (click)=\"collapseProgress = !collapseProgress; false\" class=\"text-muted\">\n View progress\n <i class=\"fas fa-chevron-{{collapseProgress? 'right' : 'down'}}\"></i>\n </a>\n <sq-collapse [collapsed]=\"collapseProgress\">\n <ng-template>\n <ul class=\"list-unstyled\">\n <li *ngFor=\"let step of progress\">\n <i class=\"fas fa-fw fa-check text-success\" *ngIf=\"step.done\"></i>\n <i class=\"fas fa-fw fa-spin fa-circle-notch\" *ngIf=\"!step.done\"></i>\n <span class=\"ms-2 fw-bold\">{{step.title}}</span>\n <span *ngIf=\"step.content\">: {{step.content}}</span>\n </li>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n <div class=\"message-content\" *ngIf=\"message.content\">\n\n <remark *ngIf=\"message.role === 'assistant'\" [markdown]=\"message.content\" [processor]=\"processor\">\n\n <ng-template remarkTemplate=\"chat-reference\" let-ref>\n\n <a *ngIf=\"referenceMap.get(ref.refId) as attachment; else staticRefTpl\"\n (click)=\"openDocument(attachment.record)\"\n class=\"reference\"\n [sqTooltip]=\"attachment\"\n [sqTooltipTemplate]=\"tooltipTpl\"\n [hoverableTooltip]=\"true\"\n >{{ref.refId}}</a>\n\n <ng-template #staticRefTpl>\n <span class=\"reference\">{{ref.refId}}</span>\n </ng-template>\n\n </ng-template>\n\n <ng-template remarkTemplate=\"streaming-placeholder\">\n <span class=\"placeholder-glow\" *ngIf=\"streaming\">\n <span class=\"placeholder ms-1\"></span>\n </span>\n </ng-template>\n\n <ng-template remarkTemplate=\"code\" let-node>\n <div class=\"card mb-2\">\n <div class=\"card-header d-flex justify-content-between align-items-center\">\n <span>{{node.lang}}</span>\n <button class=\"btn btn-light btn-sm\" (click)=\"copyToClipboard(node.value)\"><i class=\"far fa-fw fa-clipboard\"></i> Copy code</button>\n </div>\n <pre class=\"language-{{node.lang}} my-0 rounded-0 rounded-bottom\"><code class=\"language-{{node.lang}}\">{{node.value}}</code></pre>\n </div>\n </ng-template>\n\n <ng-template remarkTemplate=\"link\" let-node>\n <a [href]=\"node.url\" target=\"_blank\" rel=\"noopener noreferrer\">{{getLinkText(node)}}</a>\n </ng-template>\n\n </remark>\n\n <p *ngIf=\"message.role === 'user'\">{{message.content}}</p>\n\n <!-- List of reference, if any -->\n <div *ngIf=\"references?.length\" class=\"references\">\n <span class=\"references-title\">References</span> <i class=\"fas references-collapse\" [class.fa-chevron-down]=\"showReferences\" [class.fa-chevron-right]=\"!showReferences\" (click)=\"showReferences=!showReferences\"></i>\n <sq-collapse [collapsed]=\"!showReferences\">\n <ng-template>\n <ul>\n <ng-container *ngFor=\"let reference of references\">\n <li *ngIf=\"referenceMap.get(reference) as attachment\" class=\"text-truncate\">\n <sq-chat-reference [class.expanded]=\"attachment.$expanded\" [attachment]=\"attachment\" [reference]=\"reference\"\n (openPreview)=\"openPreview.emit($event)\" (openDocument)=\"openDocument($event)\"></sq-chat-reference>\n </li>\n </ng-container>\n </ul>\n </ng-template>\n </sq-collapse>\n </div>\n\n </div>\n\n <!-- Edit / Regenerate floating actions -->\n <div class=\"sq-chat-message-action\">\n <button class=\"btn btn-sm\" *ngIf=\"canEdit\" sqTooltip=\"Edit message\"\n (click)=\"edit.emit(message)\">\n <i class=\"fas fa-edit\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canCopy\" sqTooltip=\"Copy to clipboard\"\n (click)=\"copyToClipboard(message.content)\">\n <i class=\"far fa-clipboard\"></i>\n </button>\n <button class=\"btn btn-sm\" *ngIf=\"canRegenerate\" sqTooltip=\"Regenerate message\"\n (click)=\"regenerate.emit(message)\">\n <i class=\"fas fa-sync-alt\"></i>\n </button>\n </div>\n\n <ng-template #tooltipTpl let-ref>\n <sq-chat-reference class=\"expanded\"\n [attachment]=\"ref\"\n [reference]=\"ref.contextId\"\n [partId]=\"ref.$partId\"\n (openPreview)=\"openPreview.emit($event)\"\n (openDocument)=\"openDocument($event)\">\n </sq-chat-reference>\n </ng-template>\n\n</div>\n", styles: [".ast-primary{color:var(--ast-primary-color, #005DA7);background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover{background-color:var(--ast-primary-bg, #f2f8fe)}.ast-primary-hover:hover{color:var(--ast-primary-color, #005DA7)}.ast-secondary{color:var(--ast-secondary-color, #FF732E);background-color:var(--ast-secondary-bg, #FFF8F1)}.ast-btn{border:0;text-align:left;padding-top:.5rem;padding-bottom:.5rem;display:flex;align-items:center}.dark{--ast-primary-bg: #0d0701;--ast-primary-color: #008cd1;--ast-secondary-bg: #00070e;--ast-secondary-color: #ffa258;--ast-input-bg: #070707;--ast-input-color: rgba(222, 218, 218, .75);--ast-muted-color: rgba(222, 218, 218, .75);--ast-saved-chat-hover-background: #262421;--ast-message-table-border-color: #333333;--ast-message-table-tr-bg: #070707;--ast-message-table-tr-border-color: #222222;--ast-reference-icon-color: white;--ast-reference-icon-active-color: black;--ast-reference-passages-color: white;--ast-reference-expanded-hover-bg: #262421;--ast-message-reference-color: black}:host{display:flex}:host:not(:hover) .sq-chat-message-action{display:none}.message-content{padding:var(--ast-message-padding, var(--ast-size-3, .75rem));border-radius:var(--ast-message-border-radius, var(--ast-size-4, 1rem));display:inline-block;max-width:100%}.message-content .references{margin-top:var(--ast-size-5, 1.25rem)}.message-content .references ul{border-left:.2rem solid var(--ast-secondary-color, #FF732E);padding-left:var(--ast-size-5, 1.25rem);margin-top:var(--ast-size-2, .5rem)}.message-content .references .references-title{font-weight:var(--font-weight-bold, 500)}.message-content .references .references-collapse{cursor:pointer;margin-left:var(--ast-size-2, .5rem)}.message-content ::ng-deep p:last-child{margin-bottom:0}.message-content ::ng-deep .placeholder-glow .placeholder{animation-duration:.4s;width:12px;height:var(--ast-size-4, 1rem);vertical-align:text-bottom}.message-content ::ng-deep img{max-width:100%}.message-content ::ng-deep table{display:block;border:1px solid;border-color:var(--ast-message-table-border-color, #ccc);border-collapse:collapse;margin:0;padding:0;min-width:100%;overflow-x:auto;table-layout:fixed}.message-content ::ng-deep table tr{background-color:var(--ast-message-table-tr-bg, #f8f8f8);border:1px solid;border-color:var(--ast-message-table-tr-border-color, #ddd);padding:.35em}.message-content ::ng-deep table th,.message-content ::ng-deep table td{padding:.625em;text-align:center}.message-content ::ng-deep table th{font-size:.85em;letter-spacing:.1em;text-transform:uppercase}.message-content ::ng-deep .reference{color:var(--ast-message-reference-color, black)!important}.message-content ::ng-deep ul,.message-content ::ng-deep ol{display:flex;flex-direction:column;gap:.5rem;padding-right:2rem;margin-left:0;margin-right:0;padding-left:40px;unicode-bidi:-webkit-isolate;unicode-bidi:isolate;list-style:disc}.message-content ::ng-deep p{margin-top:.5rem}.message-assistant .message-content{background:var(--ast-secondary-bg, #FFF8F1)}.message-user .message-content{background:var(--ast-primary-bg, #f2f8fe);font-weight:var(--ast-user-font-weight, var(--font-weight-bold, 500))}.message-user .message-content p{white-space:pre-line}.sq-chat-message-action{position:absolute;right:calc(0rem - var(--ast-size-2, .5rem));top:0;display:flex;flex-direction:column}.message-icon{margin-top:var(--ast-size-3, .75rem);margin-right:var(--ast-size-4, 1rem)}\n"] }]
|
|
160
180
|
}], ctorParameters: function () { return [{ type: i1.SearchService }, { type: i2.UIService }, { type: i3.PrincipalWebService }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }]; }, propDecorators: { message: [{
|
|
161
181
|
type: Input
|
|
162
182
|
}], conversation: [{
|
|
@@ -180,4 +200,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.0.6", ngImpor
|
|
|
180
200
|
}], openPreview: [{
|
|
181
201
|
type: Output
|
|
182
202
|
}] } });
|
|
183
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1tZXNzYWdlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Fzc2lzdGFudC9jaGF0L2NoYXQtbWVzc2FnZS9jaGF0LW1lc3NhZ2UuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYXNzaXN0YW50L2NoYXQvY2hhdC1tZXNzYWdlL2NoYXQtbWVzc2FnZS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxLQUFLLEVBQWEsTUFBTSxFQUErRCxNQUFNLGVBQWUsQ0FBQztBQUt4SyxPQUFPLEVBQUUsT0FBTyxFQUFhLE1BQU0sU0FBUyxDQUFDO0FBQzdDLE9BQU8sV0FBVyxNQUFNLGNBQWMsQ0FBQztBQUN2QyxPQUFPLEVBQUUsS0FBSyxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUd6RCxPQUFPLEVBQWEsV0FBVyxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDbkUsT0FBTyxTQUFTLE1BQU0sWUFBWSxDQUFDO0FBQ25DLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sOEJBQThCLENBQUM7QUFDOUQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLFlBQVksQ0FBQztBQUMxQyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSw4Q0FBOEMsQ0FBQztBQUN2RixPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSw0Q0FBNEMsQ0FBQztBQUVwRixPQUFPLGlDQUFpQyxDQUFDO0FBQ3pDLE9BQU8sNkNBQTZDLENBQUM7Ozs7Ozs7O0FBZXJELE1BQU0sT0FBTyxvQkFBb0I7SUF5Qi9CLFlBQ1MsYUFBNEIsRUFDNUIsRUFBYSxFQUNiLGdCQUFxQyxFQUNyQyxHQUFzQixFQUN0QixFQUFjO1FBSmQsa0JBQWEsR0FBYixhQUFhLENBQWU7UUFDNUIsT0FBRSxHQUFGLEVBQUUsQ0FBVztRQUNiLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBcUI7UUFDckMsUUFBRyxHQUFILEdBQUcsQ0FBbUI7UUFDdEIsT0FBRSxHQUFGLEVBQUUsQ0FBWTtRQXpCZCxZQUFPLEdBQVksS0FBSyxDQUFDO1FBQ3pCLGtCQUFhLEdBQVksS0FBSyxDQUFDO1FBQy9CLFlBQU8sR0FBWSxLQUFLLENBQUM7UUFDeEIscUJBQWdCLEdBQUcsSUFBSSxZQUFZLEVBQVUsQ0FBQztRQUM5QyxTQUFJLEdBQUcsSUFBSSxZQUFZLEVBQWUsQ0FBQztRQUN2QyxlQUFVLEdBQUcsSUFBSSxZQUFZLEVBQWUsQ0FBQztRQUM3QyxnQkFBVyxHQUFHLElBQUksWUFBWSxFQUF5QixDQUFDO1FBSWxFLGVBQVUsR0FBYSxFQUFFLENBQUM7UUFDMUIsaUJBQVksR0FBRyxJQUFJLEdBQUcsRUFBaUMsQ0FBQztRQUN4RCxtQkFBYyxHQUFZLElBQUksQ0FBQztRQTREL0I7OztXQUdHO1FBQ0gsb0JBQWUsR0FBRyxDQUFDLElBQVUsRUFBRSxFQUFFO1lBRS9CLE1BQU0sVUFBVSxHQUFHLElBQUksR0FBRyxFQUFVLENBQUM7WUFFckMsdUJBQXVCO1lBQ3ZCLEtBQUssQ0FBQyxJQUFJLEVBQUUsTUFBTSxFQUFFLENBQUMsSUFBVSxFQUFFLEtBQWEsRUFBRSxNQUFjLEVBQUUsRUFBRTtnQkFDaEUsSUFBSSxJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQztnQkFFdEIsSUFBSSxHQUFHLElBQUksQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsQ0FBQztnQkFDckMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLG1CQUFtQixDQUFDLElBQUksQ0FBQyxDQUFDO2dCQUUvQyxtQ0FBbUM7Z0JBQ25DLElBQUksT0FBTyxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUU7b0JBQ3hCLE9BQU8sUUFBUSxDQUFDO2lCQUNqQjtnQkFFRCxNQUFNLEtBQUssR0FBa0MsRUFBRSxDQUFDO2dCQUVoRCxLQUFLLElBQUksS0FBSyxJQUFJLE9BQU8sRUFBRTtvQkFDekIsTUFBTSxLQUFLLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDO29CQUM5QixNQUFNLENBQUMsR0FBRyxDQUFDLEdBQUcsS0FBSyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztvQkFDL0Isd0NBQXdDO29CQUN4QyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsR0FBRyxDQUFDLEVBQUU7d0JBQ2hCLFVBQVUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLHVDQUF1Qzt3QkFFN0QscURBQXFEO3dCQUNyRCxNQUFNLE9BQU8sR0FBRyxLQUFLLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxHQUFHLEVBQUUsQ0FBQyxFQUFFLENBQUM7d0JBQzNDLElBQUksS0FBSyxDQUFDLEtBQU0sR0FBRyxPQUFPLENBQUMsR0FBRyxFQUFFOzRCQUM5QixLQUFLLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxLQUFNLENBQUMsRUFBRSxHQUFHLEVBQUUsS0FBSyxDQUFDLEtBQU0sRUFBRSxDQUFDLENBQUM7eUJBQ25HO3dCQUVELDhCQUE4Qjt3QkFDOUIsS0FBSyxDQUFDLElBQUksQ0FBQyxFQUFFLElBQUksRUFBRSxnQkFBZ0IsRUFBRSxLQUFLLEVBQUUsR0FBRyxFQUFFLEtBQUssQ0FBQyxLQUFNLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sRUFBUyxDQUFDLENBQUM7cUJBQzNGO2lCQUNGO2dCQUVELG1DQUFtQztnQkFDbkMsSUFBSSxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRTtvQkFDdEIsT0FBTyxRQUFRLENBQUM7aUJBQ2pCO2dCQUVELElBQUksS0FBSyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBRSxDQUFDLEdBQUcsR0FBRyxJQUFJLENBQUMsTUFBTSxFQUFFO29CQUNuQyxLQUFLLENBQUMsSUFBSSxDQUFDLEVBQUUsSUFBSSxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFFLENBQUMsR0FBRyxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsRUFBRSxHQUFHLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUM7aUJBQ3ZHO2dCQUVELGlGQUFpRjtnQkFDakYsTUFBTSxDQUFDLFFBQVEsQ0FBQyxNQUFNLENBQUMsS0FBSyxFQUFFLENBQUMsRUFBRSxHQUFHLEtBQUssQ0FBQyxDQUFDO2dCQUUzQyxPQUFPLEtBQUssR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUMsOENBQThDO1lBQzdFLENBQUMsQ0FBQyxDQUFDO1lBRUgsSUFBSSxVQUFVLENBQUMsSUFBSSxHQUFHLENBQUMsRUFBRTtnQkFDdkIsSUFBSSxDQUFDLFVBQVUsR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxNQUFNLEVBQUUsQ0FBQztxQkFDOUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQztxQkFDckIsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDO2dCQUNwQixJQUFJLENBQUMsR0FBRyxDQUFDLGFBQWEsRUFBRSxDQUFDO2FBQzFCO1lBRUQsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDLENBQUE7UUFFRCxzQkFBaUIsR0FBRyxDQUFDLElBQVUsRUFBRSxFQUFFO1lBRWpDLEtBQUssQ0FBQyxJQUFJLEVBQUUsTUFBTSxFQUFFLENBQUMsSUFBVSxFQUFFLEtBQWEsRUFBRSxNQUFjLEVBQUUsRUFBRTtnQkFDaEUsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsRUFBRSxJQUFJLEVBQUUsdUJBQXVCLEVBQVMsQ0FBQyxDQUFDO2dCQUMvRCxPQUFPLElBQUksQ0FBQztZQUNkLENBQUMsRUFBRSxJQUFJLENBQUMsQ0FBQztZQUVULE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQyxDQUFBO0lBdkhHLENBQUM7SUFYTCxJQUFJLElBQUk7UUFDTixPQUFPLENBQUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsRUFBRTtZQUMxQyxDQUFDLENBQUMsSUFBSSxDQUFDLGdCQUFnQixDQUFDLFNBQVMsQ0FBQyxVQUFVLENBQVcsSUFBSSxJQUFJLENBQUMsZ0JBQWdCLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQztJQUNwRyxDQUFDO0lBVUQsV0FBVyxDQUFDLE9BQXNCO1FBQ2hDLElBQUksT0FBTyxDQUFDLFNBQVMsRUFBRTtZQUNyQixJQUFJLENBQUMsZ0JBQWdCLEdBQUcsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDO1NBQ3pDO1FBRUQsSUFBSSxPQUFPLENBQUMsT0FBTyxJQUFJLElBQUksQ0FBQyxPQUFPLENBQUMsSUFBSSxLQUFLLFdBQVcsRUFBRTtZQUN4RCxJQUFJLENBQUMsVUFBVSxHQUFHLEVBQUUsQ0FBQztZQUNyQixJQUFJLENBQUMsWUFBWSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQzFCLEtBQUssSUFBSSxDQUFDLElBQUksSUFBSSxDQUFDLFlBQVksRUFBRTtnQkFDL0IsSUFBSSxDQUFDLENBQUMsb0JBQW9CLENBQUMsV0FBVyxFQUFFO29CQUN0QyxLQUFLLE1BQU0sVUFBVSxJQUFJLENBQUMsQ0FBQyxvQkFBb0IsQ0FBQyxXQUFXLEVBQUU7d0JBQzNELElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxDQUFDLEVBQUUsR0FBRyxVQUFVLENBQUMsU0FBUyxFQUFFLEVBQUUsR0FBRyxVQUFVLEVBQUUsT0FBTyxFQUFFLFVBQVUsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO3dCQUMzRyxLQUFLLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsVUFBVSxDQUFDLEtBQUssQ0FBQyxNQUFNLEVBQUUsQ0FBQyxFQUFFLEVBQUU7NEJBQ2hELE1BQU0sS0FBSyxHQUFHLEdBQUcsVUFBVSxDQUFDLFNBQVMsSUFBSSxVQUFVLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDOzRCQUN0RSxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsQ0FBQyxLQUFLLEVBQUUsRUFBRSxHQUFHLFVBQVUsRUFBRSxPQUFPLEVBQUUsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDO3lCQUN0RjtxQkFDRjtpQkFDRjthQUNGO1lBRUQsSUFBSSxDQUFDLFNBQVMsR0FBRyxPQUFPLEVBQUU7aUJBQ3ZCLEdBQUcsQ0FBQyxXQUFXLENBQUM7aUJBQ2hCLEdBQUcsQ0FBQyxTQUFTLENBQUM7aUJBQ2QsR0FBRyxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsQ0FBQztZQUVuQyxJQUFJLElBQUksQ0FBQyxTQUFTLEVBQUU7Z0JBQ2xCLElBQUksQ0FBQyxTQUFTLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLENBQUM7YUFDbkU7U0FDRjtJQUNILENBQUM7SUFFRCxlQUFlO1FBQ2IsS0FBSyxFQUFFLGlCQUFpQixFQUFFLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUNwRCxDQUFDO0lBRUQsWUFBWSxDQUFDLE1BQWM7UUFDekIsTUFBTSxHQUFHLEdBQUcsTUFBTSxDQUFDLElBQUksSUFBSSxNQUFNLENBQUMsV0FBVyxDQUFDO1FBQzlDLElBQUksR0FBRyxFQUFFO1lBQ1AsNEJBQTRCO1lBQzVCLE1BQU0sQ0FBQyxJQUFJLENBQUMsR0FBRyxFQUFFLFFBQVEsQ0FBQyxDQUFDO1NBQzVCO1FBQ0QsSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUNyQyxDQUFDO0lBNkVEOztPQUVHO0lBQ0gsa0JBQWtCLENBQUMsT0FBZTtRQUNoQyxPQUFPLE9BQU8sQ0FBQyxPQUFPLENBQUMsbUZBQW1GLEVBQ3hHLENBQUMsR0FBRyxFQUFFLEtBQWEsRUFBRSxFQUFFLENBQUMsSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLFNBQVMsRUFBRSxFQUFFLENBQUMsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQ25GLENBQUM7SUFDSixDQUFDO0lBRUQ7O09BRUc7SUFDSCxtQkFBbUIsQ0FBQyxPQUFlO1FBQ2pDLE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsUUFBUSxDQUFDLDRCQUE0QixDQUFDLENBQUMsQ0FBQztJQUNwRSxDQUFDO0lBRUQsZUFBZSxDQUFDLElBQVk7UUFDMUIsSUFBSSxDQUFDLEVBQUUsQ0FBQyxlQUFlLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDaEMsQ0FBQzs7aUhBMUtVLG9CQUFvQjtxR0FBcEIsb0JBQW9CLHFaQ2xDakMsdTBKQWtIQSxzbkdEbkZZLFlBQVksNlZBQUUsV0FBVyxzUEFBRSxjQUFjLHdIQUFFLFlBQVksbVFBQy9ELHVCQUF1Qiw2RkFBRSxzQkFBc0I7MkZBRXRDLG9CQUFvQjtrQkFUaEMsU0FBUzsrQkFDRSxpQkFBaUIsbUJBR1YsdUJBQXVCLENBQUMsTUFBTSxjQUNuQyxJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsV0FBVyxFQUFFLGNBQWMsRUFBRSxZQUFZO3dCQUMvRCx1QkFBdUIsRUFBRSxzQkFBc0IsQ0FBQzt1TkFHekMsT0FBTztzQkFBZixLQUFLO2dCQUNHLFlBQVk7c0JBQXBCLEtBQUs7Z0JBQ0csYUFBYTtzQkFBckIsS0FBSztnQkFDRyxTQUFTO3NCQUFqQixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFDRyxhQUFhO3NCQUFyQixLQUFLO2dCQUNHLE9BQU87c0JBQWYsS0FBSztnQkFDSSxnQkFBZ0I7c0JBQXpCLE1BQU07Z0JBQ0csSUFBSTtzQkFBYixNQUFNO2dCQUNHLFVBQVU7c0JBQW5CLE1BQU07Z0JBQ0csV0FBVztzQkFBcEIsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQsIEV2ZW50RW1pdHRlciwgSW5wdXQsIE9uQ2hhbmdlcywgT3V0cHV0LCBTaW1wbGVDaGFuZ2VzLCBDaGFuZ2VEZXRlY3RvclJlZiwgQWZ0ZXJWaWV3SW5pdCwgRWxlbWVudFJlZiB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBQcmluY2lwYWxXZWJTZXJ2aWNlLCBSZWNvcmQgfSBmcm9tIFwiQHNpbmVxdWEvY29yZS93ZWItc2VydmljZXNcIjtcbmltcG9ydCB7IFNlYXJjaFNlcnZpY2UgfSBmcm9tIFwiQHNpbmVxdWEvY29tcG9uZW50cy9zZWFyY2hcIjtcbmltcG9ydCB7IENoYXRDb250ZXh0QXR0YWNobWVudCwgQ2hhdE1lc3NhZ2UgfSBmcm9tIFwiLi4vdHlwZXNcIjtcblxuaW1wb3J0IHsgdW5pZmllZCwgUHJvY2Vzc29yIH0gZnJvbSBcInVuaWZpZWRcIjtcbmltcG9ydCByZW1hcmtQYXJzZSBmcm9tIFwicmVtYXJrLXBhcnNlXCI7XG5pbXBvcnQgeyB2aXNpdCwgQ09OVElOVUUsIEVYSVQgfSBmcm9tIFwidW5pc3QtdXRpbC12aXNpdFwiO1xuaW1wb3J0IHsgVGV4dCwgUGFyZW50LCBDb250ZW50IH0gZnJvbSBcIm1kYXN0XCI7XG5pbXBvcnQgeyBOb2RlIH0gZnJvbSBcInVuaXN0XCI7XG5pbXBvcnQgeyBVSVNlcnZpY2UsIFV0aWxzTW9kdWxlIH0gZnJvbSBcIkBzaW5lcXVhL2NvbXBvbmVudHMvdXRpbHNcIjtcbmltcG9ydCByZW1hcmtHZm0gZnJvbSBcInJlbWFyay1nZm1cIjtcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gXCJAYW5ndWxhci9jb21tb25cIjtcbmltcG9ydCB7IENvbGxhcHNlTW9kdWxlIH0gZnJvbSBcIkBzaW5lcXVhL2NvbXBvbmVudHMvY29sbGFwc2VcIjtcbmltcG9ydCB7IFJlbWFya01vZHVsZSB9IGZyb20gXCJuZ3gtcmVtYXJrXCI7XG5pbXBvcnQgeyBJbml0aWFsc0F2YXRhckNvbXBvbmVudCB9IGZyb20gXCIuLi9pbml0aWFscy1hdmF0YXIvaW5pdGlhbHMtYXZhdGFyLmNvbXBvbmVudFwiO1xuaW1wb3J0IHsgQ2hhdFJlZmVyZW5jZUNvbXBvbmVudCB9IGZyb20gXCIuLi9jaGF0LXJlZmVyZW5jZS9jaGF0LXJlZmVyZW5jZS5jb21wb25lbnRcIjtcblxuaW1wb3J0IFwicHJpc21qcy1jb21wb25lbnRzLWltcG9ydGVyL2VzbVwiO1xuaW1wb3J0ICdwcmlzbWpzL3BsdWdpbnMvYXV0b2xvYWRlci9wcmlzbS1hdXRvbG9hZGVyJztcblxuZGVjbGFyZSBtb2R1bGUgUHJpc20ge1xuICBmdW5jdGlvbiBoaWdobGlnaHRBbGxVbmRlcihlbDogSFRNTEVsZW1lbnQpOiB2b2lkO1xufVxuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6IFwic3EtY2hhdC1tZXNzYWdlXCIsXG4gIHRlbXBsYXRlVXJsOiBcIi4vY2hhdC1tZXNzYWdlLmNvbXBvbmVudC5odG1sXCIsXG4gIHN0eWxlVXJsczogW1wiLi9jaGF0LW1lc3NhZ2UuY29tcG9uZW50LnNjc3NcIl0sXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBVdGlsc01vZHVsZSwgQ29sbGFwc2VNb2R1bGUsIFJlbWFya01vZHVsZSxcbiAgICBJbml0aWFsc0F2YXRhckNvbXBvbmVudCwgQ2hhdFJlZmVyZW5jZUNvbXBvbmVudF1cbn0pXG5leHBvcnQgY2xhc3MgQ2hhdE1lc3NhZ2VDb21wb25lbnQgaW1wbGVtZW50cyBPbkNoYW5nZXMsIEFmdGVyVmlld0luaXQge1xuICBASW5wdXQoKSBtZXNzYWdlOiBDaGF0TWVzc2FnZTtcbiAgQElucHV0KCkgY29udmVyc2F0aW9uOiBDaGF0TWVzc2FnZVtdO1xuICBASW5wdXQoKSBhc3Npc3RhbnRJY29uOiBzdHJpbmc7XG4gIEBJbnB1dCgpIHN0cmVhbWluZzogYm9vbGVhbjtcbiAgQElucHV0KCkgY2FuRWRpdDogYm9vbGVhbiA9IGZhbHNlO1xuICBASW5wdXQoKSBjYW5SZWdlbmVyYXRlOiBib29sZWFuID0gZmFsc2U7XG4gIEBJbnB1dCgpIGNhbkNvcHk6IGJvb2xlYW4gPSBmYWxzZTtcbiAgQE91dHB1dCgpIHJlZmVyZW5jZUNsaWNrZWQgPSBuZXcgRXZlbnRFbWl0dGVyPFJlY29yZD4oKTtcbiAgQE91dHB1dCgpIGVkaXQgPSBuZXcgRXZlbnRFbWl0dGVyPENoYXRNZXNzYWdlPigpO1xuICBAT3V0cHV0KCkgcmVnZW5lcmF0ZSA9IG5ldyBFdmVudEVtaXR0ZXI8Q2hhdE1lc3NhZ2U+KCk7XG4gIEBPdXRwdXQoKSBvcGVuUHJldmlldyA9IG5ldyBFdmVudEVtaXR0ZXI8Q2hhdENvbnRleHRBdHRhY2htZW50PigpO1xuXG4gIHByb2Nlc3NvcjogUHJvY2Vzc29yO1xuXG4gIHJlZmVyZW5jZXM6IHN0cmluZ1tdID0gW107XG4gIHJlZmVyZW5jZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBDaGF0Q29udGV4dEF0dGFjaG1lbnQ+KCk7XG4gIHNob3dSZWZlcmVuY2VzOiBib29sZWFuID0gdHJ1ZTtcbiAgY29sbGFwc2VQcm9ncmVzczogYm9vbGVhbjtcblxuICBnZXQgbmFtZSgpOiBzdHJpbmcge1xuICAgIHJldHVybiAhdGhpcy5wcmluY2lwYWxTZXJ2aWNlLnByaW5jaXBhbCA/ICcnXG4gICAgICA6IHRoaXMucHJpbmNpcGFsU2VydmljZS5wcmluY2lwYWxbJ2Z1bGxOYW1lJ10gYXMgc3RyaW5nIHx8IHRoaXMucHJpbmNpcGFsU2VydmljZS5wcmluY2lwYWwubmFtZTtcbiAgfVxuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyBzZWFyY2hTZXJ2aWNlOiBTZWFyY2hTZXJ2aWNlLFxuICAgIHB1YmxpYyB1aTogVUlTZXJ2aWNlLFxuICAgIHB1YmxpYyBwcmluY2lwYWxTZXJ2aWNlOiBQcmluY2lwYWxXZWJTZXJ2aWNlLFxuICAgIHB1YmxpYyBjZHI6IENoYW5nZURldGVjdG9yUmVmLFxuICAgIHB1YmxpYyBlbDogRWxlbWVudFJlZlxuICApIHsgfVxuXG4gIG5nT25DaGFuZ2VzKGNoYW5nZXM6IFNpbXBsZUNoYW5nZXMpOiB2b2lkIHtcbiAgICBpZiAoY2hhbmdlcy5zdHJlYW1pbmcpIHtcbiAgICAgIHRoaXMuY29sbGFwc2VQcm9ncmVzcyA9ICF0aGlzLnN0cmVhbWluZztcbiAgICB9XG5cbiAgICBpZiAoY2hhbmdlcy5tZXNzYWdlICYmIHRoaXMubWVzc2FnZS5yb2xlID09PSBcImFzc2lzdGFudFwiKSB7XG4gICAgICB0aGlzLnJlZmVyZW5jZXMgPSBbXTtcbiAgICAgIHRoaXMucmVmZXJlbmNlTWFwLmNsZWFyKCk7XG4gICAgICBmb3IgKGxldCBtIG9mIHRoaXMuY29udmVyc2F0aW9uKSB7XG4gICAgICAgIGlmIChtLmFkZGl0aW9uYWxQcm9wZXJ0aWVzLiRhdHRhY2htZW50KSB7XG4gICAgICAgICAgZm9yIChjb25zdCBhdHRhY2htZW50IG9mIG0uYWRkaXRpb25hbFByb3BlcnRpZXMuJGF0dGFjaG1lbnQpIHtcbiAgICAgICAgICAgIHRoaXMucmVmZXJlbmNlTWFwLnNldCgnJyArIGF0dGFjaG1lbnQuY29udGV4dElkLCB7IC4uLmF0dGFjaG1lbnQsICRwYXJ0SWQ6IGF0dGFjaG1lbnQucGFydHM/LlswXS5wYXJ0SWQgfSk7XG4gICAgICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGF0dGFjaG1lbnQucGFydHMubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgICAgICAgY29uc3QgcmVmSWQgPSBgJHthdHRhY2htZW50LmNvbnRleHRJZH0uJHthdHRhY2htZW50LnBhcnRzW2ldLnBhcnRJZH1gO1xuICAgICAgICAgICAgICB0aGlzLnJlZmVyZW5jZU1hcC5zZXQocmVmSWQsIHsgLi4uYXR0YWNobWVudCwgJHBhcnRJZDogYXR0YWNobWVudC5wYXJ0c1tpXS5wYXJ0SWQgfSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIHRoaXMucHJvY2Vzc29yID0gdW5pZmllZCgpXG4gICAgICAgIC51c2UocmVtYXJrUGFyc2UpXG4gICAgICAgIC51c2UocmVtYXJrR2ZtKVxuICAgICAgICAudXNlKCgpID0+IHRoaXMucmVmZXJlbmNlUGx1Z2luKTtcblxuICAgICAgaWYgKHRoaXMuc3RyZWFtaW5nKSB7XG4gICAgICAgIHRoaXMucHJvY2Vzc29yID0gdGhpcy5wcm9jZXNzb3IudXNlKCgpID0+IHRoaXMucGxhY2Vob2xkZXJQbHVnaW4pO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIG5nQWZ0ZXJWaWV3SW5pdCgpOiB2b2lkIHtcbiAgICBQcmlzbT8uaGlnaGxpZ2h0QWxsVW5kZXI/Lih0aGlzLmVsLm5hdGl2ZUVsZW1lbnQpO1xuICB9XG5cbiAgb3BlbkRvY3VtZW50KHJlY29yZDogUmVjb3JkKSB7XG4gICAgY29uc3QgdXJsID0gcmVjb3JkLnVybDEgfHwgcmVjb3JkLm9yaWdpbmFsVXJsO1xuICAgIGlmICh1cmwpIHtcbiAgICAgIC8vIE9wZW4gdGhlIFVSTCBpbiBhIG5ldyB0YWJcbiAgICAgIHdpbmRvdy5vcGVuKHVybCwgJ19ibGFuaycpO1xuICAgIH1cbiAgICB0aGlzLnJlZmVyZW5jZUNsaWNrZWQuZW1pdChyZWNvcmQpO1xuICB9XG5cbiAgLyoqXG4gICAqIFRoaXMgVW5pZmllZCBwbHVnaW4gbG9va3MgYSB0ZXh0IG5vZGVzIGFuZCByZXBsYWNlcyBhbnkgcmVmZXJlbmNlIGluIHRoZVxuICAgKiBmb3JtIFsxXSwgWzIuM10sIGV0Yy4gd2l0aCBjdXN0b20gbm9kZXMgb2YgdHlwZSBcImNoYXQtcmVmZXJlbmNlXCIuXG4gICAqL1xuICByZWZlcmVuY2VQbHVnaW4gPSAodHJlZTogTm9kZSkgPT4ge1xuXG4gICAgY29uc3QgcmVmZXJlbmNlcyA9IG5ldyBTZXQ8bnVtYmVyPigpO1xuXG4gICAgLy8gVmlzaXQgYWxsIHRleHQgbm9kZXNcbiAgICB2aXNpdCh0cmVlLCBcInRleHRcIiwgKG5vZGU6IFRleHQsIGluZGV4OiBudW1iZXIsIHBhcmVudDogUGFyZW50KSA9PiB7XG4gICAgICBsZXQgdGV4dCA9IG5vZGUudmFsdWU7XG5cbiAgICAgIHRleHQgPSB0aGlzLnJlZm9ybWF0UmVmZXJlbmNlcyh0ZXh0KTtcbiAgICAgIGNvbnN0IG1hdGNoZXMgPSB0aGlzLmdldFJlZmVyZW5jZU1hdGNoZXModGV4dCk7XG5cbiAgICAgIC8vIFF1aXQgaWYgbm8gcmVmZXJlbmNlcyB3ZXJlIGZvdW5kXG4gICAgICBpZiAobWF0Y2hlcy5sZW5ndGggPT09IDApIHtcbiAgICAgICAgcmV0dXJuIENPTlRJTlVFO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBub2RlczogKENvbnRlbnQgJiB7IGVuZDogbnVtYmVyIH0pW10gPSBbXTtcblxuICAgICAgZm9yIChsZXQgbWF0Y2ggb2YgbWF0Y2hlcykge1xuICAgICAgICBjb25zdCByZWZJZCA9IG1hdGNoWzFdLnRyaW0oKTtcbiAgICAgICAgY29uc3QgW3JlZl0gPSByZWZJZC5zcGxpdChcIi5cIik7XG4gICAgICAgIC8vIFdlIGZpbmQgYSB2YWxpZCByZWZlcmVuY2UgaW4gdGhlIHRleHRcbiAgICAgICAgaWYgKCFpc05hTigrcmVmKSkge1xuICAgICAgICAgIHJlZmVyZW5jZXMuYWRkKCtyZWYpOyAvLyBBZGQgaXQgdG8gdGhlIHNldCBvZiB1c2VkIHJlZmVyZW5jZXNcblxuICAgICAgICAgIC8vIElmIG5lZWRlZCwgaW5zZXJ0IGEgdGV4dCBub2RlIGJlZm9yZSB0aGUgcmVmZXJlbmNlXG4gICAgICAgICAgY29uc3QgY3VycmVudCA9IG5vZGVzLmF0KC0xKSA/PyB7IGVuZDogMCB9O1xuICAgICAgICAgIGlmIChtYXRjaC5pbmRleCEgPiBjdXJyZW50LmVuZCkge1xuICAgICAgICAgICAgbm9kZXMucHVzaCh7IHR5cGU6IFwidGV4dFwiLCB2YWx1ZTogdGV4dC5zdWJzdHJpbmcoY3VycmVudC5lbmQsIG1hdGNoLmluZGV4ISksIGVuZDogbWF0Y2guaW5kZXghIH0pO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIEFkZCBhIGN1c3RvbSByZWZlcmVuY2Ugbm9kZVxuICAgICAgICAgIG5vZGVzLnB1c2goeyB0eXBlOiBcImNoYXQtcmVmZXJlbmNlXCIsIHJlZklkLCBlbmQ6IG1hdGNoLmluZGV4ISArIG1hdGNoWzBdLmxlbmd0aCB9IGFzIGFueSk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgLy8gUXVpdCBpZiBubyByZWZlcmVuY2VzIHdlcmUgZm91bmRcbiAgICAgIGlmIChub2Rlcy5sZW5ndGggPT09IDApIHtcbiAgICAgICAgcmV0dXJuIENPTlRJTlVFO1xuICAgICAgfVxuXG4gICAgICBpZiAobm9kZXMuYXQoLTEpIS5lbmQgPCB0ZXh0Lmxlbmd0aCkge1xuICAgICAgICBub2Rlcy5wdXNoKHsgdHlwZTogXCJ0ZXh0XCIsIHZhbHVlOiB0ZXh0LnN1YnN0cmluZyhub2Rlcy5hdCgtMSkhLmVuZCwgdGV4dC5sZW5ndGgpLCBlbmQ6IHRleHQubGVuZ3RoIH0pO1xuICAgICAgfVxuXG4gICAgICAvLyBEZWxldGUgdGhlIGN1cnJlbnQgdGV4dCBub2RlIGZyb20gdGhlIHBhcmVudCBhbmQgcmVwbGFjZSBpdCB3aXRoIHRoZSBuZXcgbm9kZXNcbiAgICAgIHBhcmVudC5jaGlsZHJlbi5zcGxpY2UoaW5kZXgsIDEsIC4uLm5vZGVzKTtcblxuICAgICAgcmV0dXJuIGluZGV4ICsgbm9kZXMubGVuZ3RoOyAvLyBWaXNpdCB0aGUgbmV4dCBub2RlIGFmdGVyIHRoZSBpbnNlcnRlZCBvbmVzXG4gICAgfSk7XG5cbiAgICBpZiAocmVmZXJlbmNlcy5zaXplID4gMCkge1xuICAgICAgdGhpcy5yZWZlcmVuY2VzID0gQXJyYXkuZnJvbShyZWZlcmVuY2VzLnZhbHVlcygpKVxuICAgICAgICAuc29ydCgoYSwgYikgPT4gYSAtIGIpXG4gICAgICAgIC5tYXAociA9PiAnJyArIHIpO1xuICAgICAgdGhpcy5jZHIuZGV0ZWN0Q2hhbmdlcygpO1xuICAgIH1cblxuICAgIHJldHVybiB0cmVlO1xuICB9XG5cbiAgcGxhY2Vob2xkZXJQbHVnaW4gPSAodHJlZTogTm9kZSkgPT4ge1xuXG4gICAgdmlzaXQodHJlZSwgXCJ0ZXh0XCIsIChub2RlOiBUZXh0LCBpbmRleDogbnVtYmVyLCBwYXJlbnQ6IFBhcmVudCkgPT4ge1xuICAgICAgcGFyZW50LmNoaWxkcmVuLnB1c2goeyB0eXBlOiBcInN0cmVhbWluZy1wbGFjZWhvbGRlclwiIH0gYXMgYW55KTtcbiAgICAgIHJldHVybiBFWElUO1xuICAgIH0sIHRydWUpO1xuXG4gICAgcmV0dXJuIHRyZWU7XG4gIH1cblxuICAvKipcbiAgICogUmVmb3JtYXQgW2lkczogMTIuMiwgNDIuNV0gdG8gWzEyLjJdWzQyLjVdXG4gICAqL1xuICByZWZvcm1hdFJlZmVyZW5jZXMoY29udGVudDogc3RyaW5nKSB7XG4gICAgcmV0dXJuIGNvbnRlbnQucmVwbGFjZSgvXFxbKD86aWRzPzo/XFxzKik/KD86ZG9jdW1lbnRzPzo/XFxzKik/KFxccyooPzosP1xccypcXGQrKD86XFwuXFxkKyk/KD86XFwucGFydCk/XFxzKikrKVxcXS9nLFxuICAgICAgKHN0ciwgbWF0Y2g6IHN0cmluZykgPT4gYFske21hdGNoLnJlcGxhY2UoL1xcLnBhcnQvZywgXCJcIikuc3BsaXQoJywnKS5qb2luKFwiXSBbXCIpfV1gXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBNYXRjaCBhbGwgcmVmZXJlbmNlcyBpbiBhIGdpdmVuIG1lc3NhZ2VcbiAgICovXG4gIGdldFJlZmVyZW5jZU1hdGNoZXMoY29udGVudDogc3RyaW5nKSB7XG4gICAgcmV0dXJuIEFycmF5LmZyb20oY29udGVudC5tYXRjaEFsbCgvXFxbKFxccypcXGQrKD86XFwuXFxkKyk/XFxzKilcXF0vZykpO1xuICB9XG5cbiAgY29weVRvQ2xpcGJvYXJkKHRleHQ6IHN0cmluZykge1xuICAgIHRoaXMudWkuY29weVRvQ2xpcGJvYXJkKHRleHQpO1xuICB9XG59XG4iLCI8IS0tIE1lc3NhZ2UgaWNvbiAtLT5cbjxzcGFuIGNsYXNzPVwibWVzc2FnZS1pY29uXCIgW3RpdGxlXT1cIm1lc3NhZ2Uucm9sZVwiPlxuICA8aSBbbmdDbGFzc109XCJhc3Npc3RhbnRJY29uXCIgW3N0eWxlLi0tc3Etc2l6ZS5weF09XCIyNFwiICpuZ0lmPVwibWVzc2FnZS5yb2xlID09PSAnYXNzaXN0YW50J1wiPjwvaT5cbiAgPHNxLWluaXRpYWxzLWF2YXRhciBbZnVsbE5hbWVdPVwibmFtZVwiICpuZ0lmPVwibWVzc2FnZS5yb2xlICE9PSAnYXNzaXN0YW50J1wiPjwvc3EtaW5pdGlhbHMtYXZhdGFyPlxuPC9zcGFuPlxuXG48IS0tIE1lc3NhZ2UgYm9keSAtLT5cbjxkaXYgY2xhc3M9XCJmbGV4LWdyb3ctMVwiIHN0eWxlPVwibWluLXdpZHRoOiAwO1wiIFtuZ0NsYXNzXT1cIidtZXNzYWdlLScrbWVzc2FnZS5yb2xlXCI+XG5cbiAgPGRpdiAqbmdJZj1cIm1lc3NhZ2UuYWRkaXRpb25hbFByb3BlcnRpZXMuJHByb2dyZXNzIGFzIHByb2dyZXNzXCIgY2xhc3M9XCJzbWFsbCBtcy0zIG1iLTJcIj5cbiAgICA8YSBocmVmPVwiI1wiIChjbGljayk9XCJjb2xsYXBzZVByb2dyZXNzID0gIWNvbGxhcHNlUHJvZ3Jlc3M7IGZhbHNlXCIgY2xhc3M9XCJ0ZXh0LW11dGVkXCI+XG4gICAgICBWaWV3IHByb2dyZXNzXG4gICAgICA8aSBjbGFzcz1cImZhcyBmYS1jaGV2cm9uLXt7Y29sbGFwc2VQcm9ncmVzcz8gJ3JpZ2h0JyA6ICdkb3duJ319XCI+PC9pPlxuICAgIDwvYT5cbiAgICA8c3EtY29sbGFwc2UgW2NvbGxhcHNlZF09XCJjb2xsYXBzZVByb2dyZXNzXCI+XG4gICAgICA8bmctdGVtcGxhdGU+XG4gICAgICAgIDx1bCBjbGFzcz1cImxpc3QtdW5zdHlsZWRcIj5cbiAgICAgICAgICA8bGkgKm5nRm9yPVwibGV0IHN0ZXAgb2YgcHJvZ3Jlc3NcIj5cbiAgICAgICAgICAgIDxpIGNsYXNzPVwiZmFzIGZhLWZ3IGZhLWNoZWNrIHRleHQtc3VjY2Vzc1wiICpuZ0lmPVwic3RlcC5kb25lXCI+PC9pPlxuICAgICAgICAgICAgPGkgY2xhc3M9XCJmYXMgZmEtZncgZmEtc3BpbiBmYS1jaXJjbGUtbm90Y2hcIiAqbmdJZj1cIiFzdGVwLmRvbmVcIj48L2k+XG4gICAgICAgICAgICA8c3BhbiBjbGFzcz1cIm1zLTIgZnctYm9sZFwiPnt7c3RlcC50aXRsZX19PC9zcGFuPlxuICAgICAgICAgICAgPHNwYW4gKm5nSWY9XCJzdGVwLmNvbnRlbnRcIj46IHt7c3RlcC5jb250ZW50fX08L3NwYW4+XG4gICAgICAgICAgPC9saT5cbiAgICAgICAgPC91bD5cbiAgICAgIDwvbmctdGVtcGxhdGU+XG4gICAgPC9zcS1jb2xsYXBzZT5cbiAgPC9kaXY+XG5cbiAgPGRpdiBjbGFzcz1cIm1lc3NhZ2UtY29udGVudFwiICpuZ0lmPVwibWVzc2FnZS5jb250ZW50XCI+XG5cbiAgICA8cmVtYXJrICpuZ0lmPVwibWVzc2FnZS5yb2xlID09PSAnYXNzaXN0YW50J1wiIFttYXJrZG93bl09XCJtZXNzYWdlLmNvbnRlbnRcIiBbcHJvY2Vzc29yXT1cInByb2Nlc3NvclwiPlxuXG4gICAgICA8bmctdGVtcGxhdGUgcmVtYXJrVGVtcGxhdGU9XCJjaGF0LXJlZmVyZW5jZVwiIGxldC1yZWY+XG5cbiAgICAgICAgPGEgKm5nSWY9XCJyZWZlcmVuY2VNYXAuZ2V0KHJlZi5yZWZJZCkgYXMgYXR0YWNobWVudDsgZWxzZSBzdGF0aWNSZWZUcGxcIlxuICAgICAgICAgIChjbGljayk9XCJvcGVuRG9jdW1lbnQoYXR0YWNobWVudC5yZWNvcmQpXCJcbiAgICAgICAgICBjbGFzcz1cInJlZmVyZW5jZVwiXG4gICAgICAgICAgW3NxVG9vbHRpcF09XCJhdHRhY2htZW50XCJcbiAgICAgICAgICBbc3FUb29sdGlwVGVtcGxhdGVdPVwidG9vbHRpcFRwbFwiXG4gICAgICAgICAgW2hvdmVyYWJsZVRvb2x0aXBdPVwidHJ1ZVwiXG4gICAgICAgICAgPnt7cmVmLnJlZklkfX08L2E+XG5cbiAgICAgICAgPG5nLXRlbXBsYXRlICNzdGF0aWNSZWZUcGw+XG4gICAgICAgICAgPHNwYW4gY2xhc3M9XCJyZWZlcmVuY2VcIj57e3JlZi5yZWZJZH19PC9zcGFuPlxuICAgICAgICA8L25nLXRlbXBsYXRlPlxuXG4gICAgICA8L25nLXRlbXBsYXRlPlxuXG4gICAgICA8bmctdGVtcGxhdGUgcmVtYXJrVGVtcGxhdGU9XCJzdHJlYW1pbmctcGxhY2Vob2xkZXJcIj5cbiAgICAgICAgPHNwYW4gY2xhc3M9XCJwbGFjZWhvbGRlci1nbG93XCIgKm5nSWY9XCJzdHJlYW1pbmdcIj5cbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInBsYWNlaG9sZGVyIG1zLTFcIj48L3NwYW4+XG4gICAgICAgIDwvc3Bhbj5cbiAgICAgIDwvbmctdGVtcGxhdGU+XG5cbiAgICAgIDxuZy10ZW1wbGF0ZSByZW1hcmtUZW1wbGF0ZT1cImNvZGVcIiBsZXQtbm9kZT5cbiAgICAgICAgPGRpdiBjbGFzcz1cImNhcmQgbWItMlwiPlxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJjYXJkLWhlYWRlciBkLWZsZXgganVzdGlmeS1jb250ZW50LWJldHdlZW4gYWxpZ24taXRlbXMtY2VudGVyXCI+XG4gICAgICAgICAgICA8c3Bhbj57e25vZGUubGFuZ319PC9zcGFuPlxuICAgICAgICAgICAgPGJ1dHRvbiBjbGFzcz1cImJ0biBidG4tbGlnaHQgYnRuLXNtXCIgKGNsaWNrKT1cImNvcHlUb0NsaXBib2FyZChub2RlLnZhbHVlKVwiPjxpIGNsYXNzPVwiZmFyIGZhLWZ3IGZhLWNsaXBib2FyZFwiPjwvaT4gQ29weSBjb2RlPC9idXR0b24+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPHByZSBjbGFzcz1cImxhbmd1YWdlLXt7bm9kZS5sYW5nfX0gbXktMCByb3VuZGVkLTAgcm91bmRlZC1ib3R0b21cIj48Y29kZSBjbGFzcz1cImxhbmd1YWdlLXt7bm9kZS5sYW5nfX1cIj57e25vZGUudmFsdWV9fTwvY29kZT48L3ByZT5cbiAgICAgICAgPC9kaXY+XG4gICAgICA8L25nLXRlbXBsYXRlPlxuXG4gICAgPC9yZW1hcms+XG5cbiAgICA8cCAqbmdJZj1cIm1lc3NhZ2Uucm9sZSA9PT0gJ3VzZXInXCI+e3ttZXNzYWdlLmNvbnRlbnR9fTwvcD5cblxuICAgIDwhLS0gTGlzdCBvZiByZWZlcmVuY2UsIGlmIGFueSAtLT5cbiAgICA8ZGl2ICpuZ0lmPVwicmVmZXJlbmNlcz8ubGVuZ3RoXCIgY2xhc3M9XCJyZWZlcmVuY2VzXCI+XG4gICAgICA8c3BhbiBjbGFzcz1cInJlZmVyZW5jZXMtdGl0bGVcIj5SZWZlcmVuY2VzPC9zcGFuPiA8aSBjbGFzcz1cImZhcyByZWZlcmVuY2VzLWNvbGxhcHNlXCIgW2NsYXNzLmZhLWNoZXZyb24tZG93bl09XCJzaG93UmVmZXJlbmNlc1wiIFtjbGFzcy5mYS1jaGV2cm9uLXJpZ2h0XT1cIiFzaG93UmVmZXJlbmNlc1wiIChjbGljayk9XCJzaG93UmVmZXJlbmNlcz0hc2hvd1JlZmVyZW5jZXNcIj48L2k+XG4gICAgICA8c3EtY29sbGFwc2UgW2NvbGxhcHNlZF09XCIhc2hvd1JlZmVyZW5jZXNcIj5cbiAgICAgICAgPG5nLXRlbXBsYXRlPlxuICAgICAgICAgIDx1bD5cbiAgICAgICAgICAgIDxuZy1jb250YWluZXIgKm5nRm9yPVwibGV0IHJlZmVyZW5jZSBvZiByZWZlcmVuY2VzXCI+XG4gICAgICAgICAgICAgIDxsaSAqbmdJZj1cInJlZmVyZW5jZU1hcC5nZXQocmVmZXJlbmNlKSBhcyBhdHRhY2htZW50XCIgIGNsYXNzPVwidGV4dC10cnVuY2F0ZVwiPlxuICAgICAgICAgICAgICAgIDxzcS1jaGF0LXJlZmVyZW5jZSBbY2xhc3MuZXhwYW5kZWRdPVwiYXR0YWNobWVudC4kZXhwYW5kZWRcIiBbYXR0YWNobWVudF09XCJhdHRhY2htZW50XCIgW3JlZmVyZW5jZV09XCJyZWZlcmVuY2VcIlxuICAgICAgICAgICAgICAgICAgKG9wZW5QcmV2aWV3KT1cIm9wZW5QcmV2aWV3LmVtaXQoJGV2ZW50KVwiIChvcGVuRG9jdW1lbnQpPVwib3BlbkRvY3VtZW50KCRldmVudClcIj48L3NxLWNoYXQtcmVmZXJlbmNlPlxuICAgICAgICAgICAgICA8L2xpPlxuICAgICAgICAgICAgPC9uZy1jb250YWluZXI+XG4gICAgICAgICAgPC91bD5cbiAgICAgICAgPC9uZy10ZW1wbGF0ZT5cbiAgICAgIDwvc3EtY29sbGFwc2U+XG4gICAgPC9kaXY+XG4gIFxuICA8L2Rpdj5cblxuICAgIDwhLS0gRWRpdCAvIFJlZ2VuZXJhdGUgZmxvYXRpbmcgYWN0aW9ucyAtLT5cbiAgPGRpdiBjbGFzcz1cInNxLWNoYXQtbWVzc2FnZS1hY3Rpb25cIj5cbiAgICA8YnV0dG9uIGNsYXNzPVwiYnRuIGJ0bi1zbVwiICpuZ0lmPVwiY2FuRWRpdFwiIHNxVG9vbHRpcD1cIkVkaXQgbWVzc2FnZVwiXG4gICAgICAgICAgICAoY2xpY2spPVwiZWRpdC5lbWl0KG1lc3NhZ2UpXCI+XG4gICAgICA8aSBjbGFzcz1cImZhcyBmYS1lZGl0XCI+PC9pPlxuICAgIDwvYnV0dG9uPlxuICAgIDxidXR0b24gY2xhc3M9XCJidG4gYnRuLXNtXCIgKm5nSWY9XCJjYW5Db3B5XCIgc3FUb29sdGlwPVwiQ29weSB0byBjbGlwYm9hcmRcIlxuICAgICAgICAgICAgKGNsaWNrKT1cImNvcHlUb0NsaXBib2FyZChtZXNzYWdlLmNvbnRlbnQpXCI+XG4gICAgICA8aSBjbGFzcz1cImZhciBmYS1jbGlwYm9hcmRcIj48L2k+XG4gICAgPC9idXR0b24+XG4gICAgPGJ1dHRvbiBjbGFzcz1cImJ0biBidG4tc21cIiAqbmdJZj1cImNhblJlZ2VuZXJhdGVcIiBzcVRvb2x0aXA9XCJSZWdlbmVyYXRlIG1lc3NhZ2VcIlxuICAgICAgICAgICAgKGNsaWNrKT1cInJlZ2VuZXJhdGUuZW1pdChtZXNzYWdlKVwiPlxuICAgICAgPGkgY2xhc3M9XCJmYXMgZmEtc3luYy1hbHRcIj48L2k+XG4gICAgPC9idXR0b24+XG4gIDwvZGl2PlxuXG4gIDxuZy10ZW1wbGF0ZSAjdG9vbHRpcFRwbCBsZXQtcmVmPlxuICAgIDxzcS1jaGF0LXJlZmVyZW5jZSBjbGFzcz1cImV4cGFuZGVkXCJcbiAgICAgICAgICAgICAgICAgICAgICAgIFthdHRhY2htZW50XT1cInJlZlwiXG4gICAgICAgICAgICAgICAgICAgICAgICBbcmVmZXJlbmNlXT1cInJlZi5jb250ZXh0SWRcIlxuICAgICAgICAgICAgICAgICAgICAgICAgW3BhcnRJZF09XCJyZWYuJHBhcnRJZFwiXG4gICAgICAgICAgICAgICAgICAgICAgICAob3BlblByZXZpZXcpPVwib3BlblByZXZpZXcuZW1pdCgkZXZlbnQpXCJcbiAgICAgICAgICAgICAgICAgICAgICAgIChvcGVuRG9jdW1lbnQpPVwib3BlbkRvY3VtZW50KCRldmVudClcIj5cbiAgICAgIDwvc3EtY2hhdC1yZWZlcmVuY2U+XG4gIDwvbmctdGVtcGxhdGU+XG5cbjwvZGl2PlxuIl19
|
|
203
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC1tZXNzYWdlLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Fzc2lzdGFudC9jaGF0L2NoYXQtbWVzc2FnZS9jaGF0LW1lc3NhZ2UuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvYXNzaXN0YW50L2NoYXQvY2hhdC1tZXNzYWdlL2NoYXQtbWVzc2FnZS5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLFlBQVksRUFBRSxLQUFLLEVBQWEsTUFBTSxFQUErRCxNQUFNLGVBQWUsQ0FBQztBQUt4SyxPQUFPLEVBQUUsT0FBTyxFQUFhLE1BQU0sU0FBUyxDQUFDO0FBQzdDLE9BQU8sV0FBVyxNQUFNLGNBQWMsQ0FBQztBQUN2QyxPQUFPLEVBQUUsS0FBSyxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUd6RCxPQUFPLEVBQWEsV0FBVyxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDbkUsT0FBTyxTQUFTLE1BQU0sWUFBWSxDQUFDO0FBQ25DLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sOEJBQThCLENBQUM7QUFDOUQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLFlBQVksQ0FBQztBQUMxQyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsTUFBTSw4Q0FBOEMsQ0FBQztBQUN2RixPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSw0Q0FBNEMsQ0FBQztBQUVwRixPQUFPLGlDQUFpQyxDQUFDO0FBQ3pDLE9BQU8sNkNBQTZDLENBQUM7Ozs7Ozs7O0FBZXJELE1BQU0sT0FBTyxvQkFBb0I7SUF5Qi9CLFlBQ1MsYUFBNEIsRUFDNUIsRUFBYSxFQUNiLGdCQUFxQyxFQUNyQyxHQUFzQixF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QUM7SUFFRDs7T0FFRztJQUNILG1CQUFtQixDQUFDLE9BQWU7UUFDakMsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsNEJBQTRCLENBQUMsQ0FBQyxDQUFDO0lBQ3BFLENBQUM7SUFFRCxlQUFlLENBQUMsSUFBWTtRQUMxQixJQUFJLENBQUMsRUFBRSxDQUFDLGVBQWUsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNoQyxDQUFDOztpSEEvTFUsb0JBQW9CO3FHQUFwQixvQkFBb0IscVpDbENqQywyL0pBc0hBLGc3R0R2RlksWUFBWSw2VkFBRSxXQUFXLHNQQUFFLGNBQWMsd0hBQUUsWUFBWSxtUUFDL0QsdUJBQXVCLDZGQUFFLHNCQUFzQjsyRkFFdEMsb0JBQW9CO2tCQVRoQyxTQUFTOytCQUNFLGlCQUFpQixtQkFHVix1QkFBdUIsQ0FBQyxNQUFNLGNBQ25DLElBQUksV0FDUCxDQUFDLFlBQVksRUFBRSxXQUFXLEVBQUUsY0FBYyxFQUFFLFlBQVk7d0JBQy9ELHVCQUF1QixFQUFFLHNCQUFzQixDQUFDO3VOQUd6QyxPQUFPO3NCQUFmLEtBQUs7Z0JBQ0csWUFBWTtzQkFBcEIsS0FBSztnQkFDRyxhQUFhO3NCQUFyQixLQUFLO2dCQUNHLFNBQVM7c0JBQWpCLEtBQUs7Z0JBQ0csT0FBTztzQkFBZixLQUFLO2dCQUNHLGFBQWE7c0JBQXJCLEtBQUs7Z0JBQ0csT0FBTztzQkFBZixLQUFLO2dCQUNJLGdCQUFnQjtzQkFBekIsTUFBTTtnQkFDRyxJQUFJO3NCQUFiLE1BQU07Z0JBQ0csVUFBVTtzQkFBbkIsTUFBTTtnQkFDRyxXQUFXO3NCQUFwQixNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3ksIENvbXBvbmVudCwgRXZlbnRFbWl0dGVyLCBJbnB1dCwgT25DaGFuZ2VzLCBPdXRwdXQsIFNpbXBsZUNoYW5nZXMsIENoYW5nZURldGVjdG9yUmVmLCBBZnRlclZpZXdJbml0LCBFbGVtZW50UmVmIH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcbmltcG9ydCB7IFByaW5jaXBhbFdlYlNlcnZpY2UsIFJlY29yZCB9IGZyb20gXCJAc2luZXF1YS9jb3JlL3dlYi1zZXJ2aWNlc1wiO1xuaW1wb3J0IHsgU2VhcmNoU2VydmljZSB9IGZyb20gXCJAc2luZXF1YS9jb21wb25lbnRzL3NlYXJjaFwiO1xuaW1wb3J0IHsgQ2hhdENvbnRleHRBdHRhY2htZW50LCBDaGF0TWVzc2FnZSB9IGZyb20gXCIuLi90eXBlc1wiO1xuXG5pbXBvcnQgeyB1bmlmaWVkLCBQcm9jZXNzb3IgfSBmcm9tIFwidW5pZmllZFwiO1xuaW1wb3J0IHJlbWFya1BhcnNlIGZyb20gXCJyZW1hcmstcGFyc2VcIjtcbmltcG9ydCB7IHZpc2l0LCBDT05USU5VRSwgRVhJVCB9IGZyb20gXCJ1bmlzdC11dGlsLXZpc2l0XCI7XG5pbXBvcnQgeyBUZXh0LCBQYXJlbnQsIENvbnRlbnQgfSBmcm9tIFwibWRhc3RcIjtcbmltcG9ydCB7IE5vZGUgfSBmcm9tIFwidW5pc3RcIjtcbmltcG9ydCB7IFVJU2VydmljZSwgVXRpbHNNb2R1bGUgfSBmcm9tIFwiQHNpbmVxdWEvY29tcG9uZW50cy91dGlsc1wiO1xuaW1wb3J0IHJlbWFya0dmbSBmcm9tIFwicmVtYXJrLWdmbVwiO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSBcIkBhbmd1bGFyL2NvbW1vblwiO1xuaW1wb3J0IHsgQ29sbGFwc2VNb2R1bGUgfSBmcm9tIFwiQHNpbmVxdWEvY29tcG9uZW50cy9jb2xsYXBzZVwiO1xuaW1wb3J0IHsgUmVtYXJrTW9kdWxlIH0gZnJvbSBcIm5neC1yZW1hcmtcIjtcbmltcG9ydCB7IEluaXRpYWxzQXZhdGFyQ29tcG9uZW50IH0gZnJvbSBcIi4uL2luaXRpYWxzLWF2YXRhci9pbml0aWFscy1hdmF0YXIuY29tcG9uZW50XCI7XG5pbXBvcnQgeyBDaGF0UmVmZXJlbmNlQ29tcG9uZW50IH0gZnJvbSBcIi4uL2NoYXQtcmVmZXJlbmNlL2NoYXQtcmVmZXJlbmNlLmNvbXBvbmVudFwiO1xuXG5pbXBvcnQgXCJwcmlzbWpzLWNvbXBvbmVudHMtaW1wb3J0ZXIvZXNtXCI7XG5pbXBvcnQgJ3ByaXNtanMvcGx1Z2lucy9hdXRvbG9hZGVyL3ByaXNtLWF1dG9sb2FkZXInO1xuXG5kZWNsYXJlIG1vZHVsZSBQcmlzbSB7XG4gIGZ1bmN0aW9uIGhpZ2hsaWdodEFsbFVuZGVyKGVsOiBIVE1MRWxlbWVudCk6IHZvaWQ7XG59XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogXCJzcS1jaGF0LW1lc3NhZ2VcIixcbiAgdGVtcGxhdGVVcmw6IFwiLi9jaGF0LW1lc3NhZ2UuY29tcG9uZW50Lmh0bWxcIixcbiAgc3R5bGVVcmxzOiBbXCIuL2NoYXQtbWVzc2FnZS5jb21wb25lbnQuc2Nzc1wiXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIFV0aWxzTW9kdWxlLCBDb2xsYXBzZU1vZHVsZSwgUmVtYXJrTW9kdWxlLFxuICAgIEluaXRpYWxzQXZhdGFyQ29tcG9uZW50LCBDaGF0UmVmZXJlbmNlQ29tcG9uZW50XVxufSlcbmV4cG9ydCBjbGFzcyBDaGF0TWVzc2FnZUNvbXBvbmVudCBpbXBsZW1lbnRzIE9uQ2hhbmdlcywgQWZ0ZXJWaWV3SW5pdCB7XG4gIEBJbnB1dCgpIG1lc3NhZ2U6IENoYXRNZXNzYWdlO1xuICBASW5wdXQoKSBjb252ZXJzYXRpb246IENoYXRNZXNzYWdlW107XG4gIEBJbnB1dCgpIGFzc2lzdGFudEljb246IHN0cmluZztcbiAgQElucHV0KCkgc3RyZWFtaW5nOiBib29sZWFuO1xuICBASW5wdXQoKSBjYW5FZGl0OiBib29sZWFuID0gZmFsc2U7XG4gIEBJbnB1dCgpIGNhblJlZ2VuZXJhdGU6IGJvb2xlYW4gPSBmYWxzZTtcbiAgQElucHV0KCkgY2FuQ29weTogYm9vbGVhbiA9IGZhbHNlO1xuICBAT3V0cHV0KCkgcmVmZXJlbmNlQ2xpY2tlZCA9IG5ldyBFdmVudEVtaXR0ZXI8UmVjb3JkPigpO1xuICBAT3V0cHV0KCkgZWRpdCA9IG5ldyBFdmVudEVtaXR0ZXI8Q2hhdE1lc3NhZ2U+KCk7XG4gIEBPdXRwdXQoKSByZWdlbmVyYXRlID0gbmV3IEV2ZW50RW1pdHRlcjxDaGF0TWVzc2FnZT4oKTtcbiAgQE91dHB1dCgpIG9wZW5QcmV2aWV3ID0gbmV3IEV2ZW50RW1pdHRlcjxDaGF0Q29udGV4dEF0dGFjaG1lbnQ+KCk7XG5cbiAgcHJvY2Vzc29yOiBQcm9jZXNzb3I7XG5cbiAgcmVmZXJlbmNlczogc3RyaW5nW10gPSBbXTtcbiAgcmVmZXJlbmNlTWFwID0gbmV3IE1hcDxzdHJpbmcsIENoYXRDb250ZXh0QXR0YWNobWVudD4oKTtcbiAgc2hvd1JlZmVyZW5jZXM6IGJvb2xlYW4gPSB0cnVlO1xuICBjb2xsYXBzZVByb2dyZXNzOiBib29sZWFuO1xuXG4gIGdldCBuYW1lKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuICF0aGlzLnByaW5jaXBhbFNlcnZpY2UucHJpbmNpcGFsID8gJydcbiAgICAgIDogdGhpcy5wcmluY2lwYWxTZXJ2aWNlLnByaW5jaXBhbFsnZnVsbE5hbWUnXSBhcyBzdHJpbmcgfHwgdGhpcy5wcmluY2lwYWxTZXJ2aWNlLnByaW5jaXBhbC5uYW1lO1xuICB9XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHNlYXJjaFNlcnZpY2U6IFNlYXJjaFNlcnZpY2UsXG4gICAgcHVibGljIHVpOiBVSVNlcnZpY2UsXG4gICAgcHVibGljIHByaW5jaXBhbFNlcnZpY2U6IFByaW5jaXBhbFdlYlNlcnZpY2UsXG4gICAgcHVibGljIGNkcjogQ2hhbmdlRGV0ZWN0b3JSZWYsXG4gICAgcHVibGljIGVsOiBFbGVtZW50UmVmXG4gICkgeyB9XG5cbiAgbmdPbkNoYW5nZXMoY2hhbmdlczogU2ltcGxlQ2hhbmdlcyk6IHZvaWQge1xuICAgIGlmIChjaGFuZ2VzLnN0cmVhbWluZykge1xuICAgICAgdGhpcy5jb2xsYXBzZVByb2dyZXNzID0gIXRoaXMuc3RyZWFtaW5nO1xuICAgIH1cblxuICAgIGlmIChjaGFuZ2VzLm1lc3NhZ2UgJiYgdGhpcy5tZXNzYWdlLnJvbGUgPT09IFwiYXNzaXN0YW50XCIpIHtcbiAgICAgIHRoaXMucmVmZXJlbmNlcyA9IFtdO1xuICAgICAgdGhpcy5yZWZlcmVuY2VNYXAuY2xlYXIoKTtcbiAgICAgIGZvciAobGV0IG0gb2YgdGhpcy5jb252ZXJzYXRpb24pIHtcbiAgICAgICAgaWYgKG0uYWRkaXRpb25hbFByb3BlcnRpZXMuJGF0dGFjaG1lbnQpIHtcbiAgICAgICAgICBmb3IgKGNvbnN0IGF0dGFjaG1lbnQgb2YgbS5hZGRpdGlvbmFsUHJvcGVydGllcy4kYXR0YWNobWVudCkge1xuICAgICAgICAgICAgdGhpcy5yZWZlcmVuY2VNYXAuc2V0KCcnICsgYXR0YWNobWVudC5jb250ZXh0SWQsIHsgLi4uYXR0YWNobWVudCwgJHBhcnRJZDogYXR0YWNobWVudC5wYXJ0cz8uWzBdLnBhcnRJZCB9KTtcbiAgICAgICAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgYXR0YWNobWVudC5wYXJ0cy5sZW5ndGg7IGkrKykge1xuICAgICAgICAgICAgICBjb25zdCByZWZJZCA9IGAke2F0dGFjaG1lbnQuY29udGV4dElkfS4ke2F0dGFjaG1lbnQucGFydHNbaV0ucGFydElkfWA7XG4gICAgICAgICAgICAgIHRoaXMucmVmZXJlbmNlTWFwLnNldChyZWZJZCwgeyAuLi5hdHRhY2htZW50LCAkcGFydElkOiBhdHRhY2htZW50LnBhcnRzW2ldLnBhcnRJZCB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgdGhpcy5wcm9jZXNzb3IgPSB1bmlmaWVkKClcbiAgICAgICAgLnVzZShyZW1hcmtQYXJzZSlcbiAgICAgICAgLnVzZShyZW1hcmtHZm0pXG4gICAgICAgIC51c2UoKCkgPT4gdGhpcy5yZWZlcmVuY2VQbHVnaW4pO1xuXG4gICAgICBpZiAodGhpcy5zdHJlYW1pbmcpIHtcbiAgICAgICAgdGhpcy5wcm9jZXNzb3IgPSB0aGlzLnByb2Nlc3Nvci51c2UoKCkgPT4gdGhpcy5wbGFjZWhvbGRlclBsdWdpbik7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIFByaXNtPy5oaWdobGlnaHRBbGxVbmRlcj8uKHRoaXMuZWwubmF0aXZlRWxlbWVudCk7XG4gIH1cblxuICBvcGVuRG9jdW1lbnQocmVjb3JkOiBSZWNvcmQpIHtcbiAgICBjb25zdCB1cmwgPSByZWNvcmQudXJsMSB8fCByZWNvcmQub3JpZ2luYWxVcmw7XG4gICAgaWYgKHVybCkge1xuICAgICAgLy8gT3BlbiB0aGUgVVJMIGluIGEgbmV3IHRhYlxuICAgICAgd2luZG93Lm9wZW4odXJsLCAnX2JsYW5rJyk7XG4gICAgfVxuICAgIHRoaXMucmVmZXJlbmNlQ2xpY2tlZC5lbWl0KHJlY29yZCk7XG4gIH1cblxuICAvKipcbiAgICogVGhpcyBVbmlmaWVkIHBsdWdpbiBsb29rcyBhIHRleHQgbm9kZXMgYW5kIHJlcGxhY2VzIGFueSByZWZlcmVuY2UgaW4gdGhlXG4gICAqIGZvcm0gWzFdLCBbMi4zXSwgZXRjLiB3aXRoIGN1c3RvbSBub2RlcyBvZiB0eXBlIFwiY2hhdC1yZWZlcmVuY2VcIi5cbiAgICovXG4gIHJlZmVyZW5jZVBsdWdpbiA9ICh0cmVlOiBOb2RlKSA9PiB7XG5cbiAgICBjb25zdCByZWZlcmVuY2VzID0gbmV3IFNldDxudW1iZXI+KCk7XG5cbiAgICAvLyBWaXNpdCBhbGwgdGV4dCBub2Rlc1xuICAgIHZpc2l0KHRyZWUsIFwidGV4dFwiLCAobm9kZTogVGV4dCwgaW5kZXg6IG51bWJlciwgcGFyZW50OiBQYXJlbnQpID0+IHtcbiAgICAgIGxldCB0ZXh0ID0gbm9kZS52YWx1ZTtcblxuICAgICAgdGV4dCA9IHRoaXMucmVmb3JtYXRSZWZlcmVuY2VzKHRleHQpO1xuICAgICAgY29uc3QgbWF0Y2hlcyA9IHRoaXMuZ2V0UmVmZXJlbmNlTWF0Y2hlcyh0ZXh0KTtcblxuICAgICAgLy8gUXVpdCBpZiBubyByZWZlcmVuY2VzIHdlcmUgZm91bmRcbiAgICAgIGlmIChtYXRjaGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICByZXR1cm4gQ09OVElOVUU7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IG5vZGVzOiAoQ29udGVudCAmIHsgZW5kOiBudW1iZXIgfSlbXSA9IFtdO1xuXG4gICAgICBmb3IgKGxldCBtYXRjaCBvZiBtYXRjaGVzKSB7XG4gICAgICAgIGNvbnN0IHJlZklkID0gbWF0Y2hbMV0udHJpbSgpO1xuICAgICAgICBjb25zdCBbcmVmXSA9IHJlZklkLnNwbGl0KFwiLlwiKTtcbiAgICAgICAgLy8gV2UgZmluZCBhIHZhbGlkIHJlZmVyZW5jZSBpbiB0aGUgdGV4dFxuICAgICAgICBpZiAoIWlzTmFOKCtyZWYpKSB7XG4gICAgICAgICAgcmVmZXJlbmNlcy5hZGQoK3JlZik7IC8vIEFkZCBpdCB0byB0aGUgc2V0IG9mIHVzZWQgcmVmZXJlbmNlc1xuXG4gICAgICAgICAgLy8gSWYgbmVlZGVkLCBpbnNlcnQgYSB0ZXh0IG5vZGUgYmVmb3JlIHRoZSByZWZlcmVuY2VcbiAgICAgICAgICBjb25zdCBjdXJyZW50ID0gbm9kZXMuYXQoLTEpID8/IHsgZW5kOiAwIH07XG4gICAgICAgICAgaWYgKG1hdGNoLmluZGV4ISA+IGN1cnJlbnQuZW5kKSB7XG4gICAgICAgICAgICBub2Rlcy5wdXNoKHsgdHlwZTogXCJ0ZXh0XCIsIHZhbHVlOiB0ZXh0LnN1YnN0cmluZyhjdXJyZW50LmVuZCwgbWF0Y2guaW5kZXghKSwgZW5kOiBtYXRjaC5pbmRleCEgfSk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gQWRkIGEgY3VzdG9tIHJlZmVyZW5jZSBub2RlXG4gICAgICAgICAgbm9kZXMucHVzaCh7IHR5cGU6IFwiY2hhdC1yZWZlcmVuY2VcIiwgcmVmSWQsIGVuZDogbWF0Y2guaW5kZXghICsgbWF0Y2hbMF0ubGVuZ3RoIH0gYXMgYW55KTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyBRdWl0IGlmIG5vIHJlZmVyZW5jZXMgd2VyZSBmb3VuZFxuICAgICAgaWYgKG5vZGVzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICByZXR1cm4gQ09OVElOVUU7XG4gICAgICB9XG5cbiAgICAgIGlmIChub2Rlcy5hdCgtMSkhLmVuZCA8IHRleHQubGVuZ3RoKSB7XG4gICAgICAgIG5vZGVzLnB1c2goeyB0eXBlOiBcInRleHRcIiwgdmFsdWU6IHRleHQuc3Vic3RyaW5nKG5vZGVzLmF0KC0xKSEuZW5kLCB0ZXh0Lmxlbmd0aCksIGVuZDogdGV4dC5sZW5ndGggfSk7XG4gICAgICB9XG5cbiAgICAgIC8vIERlbGV0ZSB0aGUgY3VycmVudCB0ZXh0IG5vZGUgZnJvbSB0aGUgcGFyZW50IGFuZCByZXBsYWNlIGl0IHdpdGggdGhlIG5ldyBub2Rlc1xuICAgICAgcGFyZW50LmNoaWxkcmVuLnNwbGljZShpbmRleCwgMSwgLi4ubm9kZXMpO1xuXG4gICAgICByZXR1cm4gaW5kZXggKyBub2Rlcy5sZW5ndGg7IC8vIFZpc2l0IHRoZSBuZXh0IG5vZGUgYWZ0ZXIgdGhlIGluc2VydGVkIG9uZXNcbiAgICB9KTtcblxuICAgIGlmIChyZWZlcmVuY2VzLnNpemUgPiAwKSB7XG4gICAgICB0aGlzLnJlZmVyZW5jZXMgPSBBcnJheS5mcm9tKHJlZmVyZW5jZXMudmFsdWVzKCkpXG4gICAgICAgIC5zb3J0KChhLCBiKSA9PiBhIC0gYilcbiAgICAgICAgLm1hcChyID0+ICcnICsgcik7XG4gICAgICB0aGlzLmNkci5kZXRlY3RDaGFuZ2VzKCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHRyZWU7XG4gIH1cblxuICBwbGFjZWhvbGRlclBsdWdpbiA9ICh0cmVlOiBOb2RlKSA9PiB7XG5cbiAgICB2aXNpdCh0cmVlLCBcInRleHRcIiwgKG5vZGU6IFRleHQsIGluZGV4OiBudW1iZXIsIHBhcmVudDogUGFyZW50KSA9PiB7XG4gICAgICBwYXJlbnQuY2hpbGRyZW4ucHVzaCh7IHR5cGU6IFwic3RyZWFtaW5nLXBsYWNlaG9sZGVyXCIgfSBhcyBhbnkpO1xuICAgICAgcmV0dXJuIEVYSVQ7XG4gICAgfSwgdHJ1ZSk7XG5cbiAgICByZXR1cm4gdHJlZTtcbiAgfVxuXG4gIGdldExpbmtUZXh0KG5vZGU6IGFueSk6IHN0cmluZyB7XG4gICAgaWYgKG5vZGUudGV4dCkge1xuICAgICAgcmV0dXJuIG5vZGUudGV4dDsgLy8gUmV0dXJuIGRpcmVjdGx5IGlmIHRleHQgaXMgcHJvdmlkZWQgaW4gbm9kZS50ZXh0IChbRXhhbXBsZSBsaW5rXShodHRwczovL2V4YW1wbGUuY29tKSlcbiAgICB9IGVsc2UgaWYgKG5vZGUuY2hpbGRyZW4gJiYgbm9kZS5jaGlsZHJlbi5sZW5ndGggPiAwKSB7XG4gICAgICAvLyBSZWN1cnNpdmVseSBzZWFyY2ggZm9yIHRleHQgY29udGVudCBpbiBjaGlsZCBub2Rlc1xuICAgICAgZm9yIChjb25zdCBjaGlsZCBvZiBub2RlLmNoaWxkcmVuKSB7XG4gICAgICAgIGlmIChjaGlsZC50eXBlID09PSAndGV4dCcgJiYgY2hpbGQudmFsdWUpIHtcbiAgICAgICAgICByZXR1cm4gY2hpbGQudmFsdWU7IC8vIFJldHVybiB0aGUgdmFsdWUgb2YgdGhlIGZpcnN0IHRleHQgbm9kZSBmb3VuZCAoWyoqRW1waGFzaXplZCBMaW5rIFRleHQqKl0oaHR0cHM6Ly9leGFtcGxlLmNvbSkpXG4gICAgICAgIH0gZWxzZSBpZiAoY2hpbGQuY2hpbGRyZW4gJiYgY2hpbGQuY2hpbGRyZW4ubGVuZ3RoID4gMCkge1xuICAgICAgICAgIGNvbnN0IHRleHRDb250ZW50ID0gdGhpcy5nZXRMaW5rVGV4dChjaGlsZCk7IC8vIFJlY3Vyc2l2ZWx5IHNlYXJjaCBjaGlsZCBub2RlcyAoWyFbRXhhbXBsZSBpbWFnZV0oaHR0cHM6Ly9leGFtcGxlLmNvbS9pbWFnZS5wbmcpXShodHRwczovL2V4YW1wbGUuY29tKSlcbiAgICAgICAgICBpZiAodGV4dENvbnRlbnQpIHtcbiAgICAgICAgICAgIHJldHVybiB0ZXh0Q29udGVudDsgLy8gUmV0dXJuIHRleHQgY29udGVudCBpZiBmb3VuZFxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gJ2xpbmsnOyAvLyBSZXR1cm4gZW1wdHkgc3RyaW5nIGlmIG5vIHRleHQgY29udGVudCBpcyBmb3VuZFxuICB9XG5cblxuXG4gIC8qKlxuICAgKiBSZWZvcm1hdCBbaWRzOiAxMi4yLCA0Mi41XSB0byBbMTIuMl1bNDIuNV1cbiAgICovXG4gIHJlZm9ybWF0UmVmZXJlbmNlcyhjb250ZW50OiBzdHJpbmcpIHtcbiAgICByZXR1cm4gY29udGVudC5yZXBsYWNlKC9cXFsoPzppZHM/Oj9cXHMqKT8oPzpkb2N1bWVudHM/Oj9cXHMqKT8oXFxzKig/Oiw/XFxzKlxcZCsoPzpcXC5cXGQrKT8oPzpcXC5wYXJ0KT9cXHMqKSspXFxdL2csXG4gICAgICAoc3RyLCBtYXRjaDogc3RyaW5nKSA9PiBgWyR7bWF0Y2gucmVwbGFjZSgvXFwucGFydC9nLCBcIlwiKS5zcGxpdCgnLCcpLmpvaW4oXCJdIFtcIil9XWBcbiAgICApO1xuICB9XG5cbiAgLyoqXG4gICAqIE1hdGNoIGFsbCByZWZlcmVuY2VzIGluIGEgZ2l2ZW4gbWVzc2FnZVxuICAgKi9cbiAgZ2V0UmVmZXJlbmNlTWF0Y2hlcyhjb250ZW50OiBzdHJpbmcpIHtcbiAgICByZXR1cm4gQXJyYXkuZnJvbShjb250ZW50Lm1hdGNoQWxsKC9cXFsoXFxzKlxcZCsoPzpcXC5cXGQrKT9cXHMqKVxcXS9nKSk7XG4gIH1cblxuICBjb3B5VG9DbGlwYm9hcmQodGV4dDogc3RyaW5nKSB7XG4gICAgdGhpcy51aS5jb3B5VG9DbGlwYm9hcmQodGV4dCk7XG4gIH1cbn1cbiIsIjwhLS0gTWVzc2FnZSBpY29uIC0tPlxuPHNwYW4gY2xhc3M9XCJtZXNzYWdlLWljb25cIiBbdGl0bGVdPVwibWVzc2FnZS5yb2xlXCI+XG4gIDxpIFtuZ0NsYXNzXT1cImFzc2lzdGFudEljb25cIiBbc3R5bGUuLS1zcS1zaXplLnB4XT1cIjI0XCIgKm5nSWY9XCJtZXNzYWdlLnJvbGUgPT09ICdhc3Npc3RhbnQnXCI+PC9pPlxuICA8c3EtaW5pdGlhbHMtYXZhdGFyIFtmdWxsTmFtZV09XCJuYW1lXCIgKm5nSWY9XCJtZXNzYWdlLnJvbGUgIT09ICdhc3Npc3RhbnQnXCI+PC9zcS1pbml0aWFscy1hdmF0YXI+XG48L3NwYW4+XG5cbjwhLS0gTWVzc2FnZSBib2R5IC0tPlxuPGRpdiBjbGFzcz1cImZsZXgtZ3Jvdy0xXCIgc3R5bGU9XCJtaW4td2lkdGg6IDA7XCIgW25nQ2xhc3NdPVwiJ21lc3NhZ2UtJyttZXNzYWdlLnJvbGVcIj5cblxuICA8ZGl2ICpuZ0lmPVwibWVzc2FnZS5hZGRpdGlvbmFsUHJvcGVydGllcy4kcHJvZ3Jlc3MgYXMgcHJvZ3Jlc3NcIiBjbGFzcz1cInNtYWxsIG1zLTMgbWItMlwiPlxuICAgIDxhIGhyZWY9XCIjXCIgKGNsaWNrKT1cImNvbGxhcHNlUHJvZ3Jlc3MgPSAhY29sbGFwc2VQcm9ncmVzczsgZmFsc2VcIiBjbGFzcz1cInRleHQtbXV0ZWRcIj5cbiAgICAgIFZpZXcgcHJvZ3Jlc3NcbiAgICAgIDxpIGNsYXNzPVwiZmFzIGZhLWNoZXZyb24te3tjb2xsYXBzZVByb2dyZXNzPyAncmlnaHQnIDogJ2Rvd24nfX1cIj48L2k+XG4gICAgPC9hPlxuICAgIDxzcS1jb2xsYXBzZSBbY29sbGFwc2VkXT1cImNvbGxhcHNlUHJvZ3Jlc3NcIj5cbiAgICAgIDxuZy10ZW1wbGF0ZT5cbiAgICAgICAgPHVsIGNsYXNzPVwibGlzdC11bnN0eWxlZFwiPlxuICAgICAgICAgIDxsaSAqbmdGb3I9XCJsZXQgc3RlcCBvZiBwcm9ncmVzc1wiPlxuICAgICAgICAgICAgPGkgY2xhc3M9XCJmYXMgZmEtZncgZmEtY2hlY2sgdGV4dC1zdWNjZXNzXCIgKm5nSWY9XCJzdGVwLmRvbmVcIj48L2k+XG4gICAgICAgICAgICA8aSBjbGFzcz1cImZhcyBmYS1mdyBmYS1zcGluIGZhLWNpcmNsZS1ub3RjaFwiICpuZ0lmPVwiIXN0ZXAuZG9uZVwiPjwvaT5cbiAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwibXMtMiBmdy1ib2xkXCI+e3tzdGVwLnRpdGxlfX08L3NwYW4+XG4gICAgICAgICAgICA8c3BhbiAqbmdJZj1cInN0ZXAuY29udGVudFwiPjoge3tzdGVwLmNvbnRlbnR9fTwvc3Bhbj5cbiAgICAgICAgICA8L2xpPlxuICAgICAgICA8L3VsPlxuICAgICAgPC9uZy10ZW1wbGF0ZT5cbiAgICA8L3NxLWNvbGxhcHNlPlxuICA8L2Rpdj5cblxuICA8ZGl2IGNsYXNzPVwibWVzc2FnZS1jb250ZW50XCIgKm5nSWY9XCJtZXNzYWdlLmNvbnRlbnRcIj5cblxuICAgIDxyZW1hcmsgKm5nSWY9XCJtZXNzYWdlLnJvbGUgPT09ICdhc3Npc3RhbnQnXCIgW21hcmtkb3duXT1cIm1lc3NhZ2UuY29udGVudFwiIFtwcm9jZXNzb3JdPVwicHJvY2Vzc29yXCI+XG5cbiAgICAgIDxuZy10ZW1wbGF0ZSByZW1hcmtUZW1wbGF0ZT1cImNoYXQtcmVmZXJlbmNlXCIgbGV0LXJlZj5cblxuICAgICAgICA8YSAqbmdJZj1cInJlZmVyZW5jZU1hcC5nZXQocmVmLnJlZklkKSBhcyBhdHRhY2htZW50OyBlbHNlIHN0YXRpY1JlZlRwbFwiXG4gICAgICAgICAgKGNsaWNrKT1cIm9wZW5Eb2N1bWVudChhdHRhY2htZW50LnJlY29yZClcIlxuICAgICAgICAgIGNsYXNzPVwicmVmZXJlbmNlXCJcbiAgICAgICAgICBbc3FUb29sdGlwXT1cImF0dGFjaG1lbnRcIlxuICAgICAgICAgIFtzcVRvb2x0aXBUZW1wbGF0ZV09XCJ0b29sdGlwVHBsXCJcbiAgICAgICAgICBbaG92ZXJhYmxlVG9vbHRpcF09XCJ0cnVlXCJcbiAgICAgICAgICA+e3tyZWYucmVmSWR9fTwvYT5cblxuICAgICAgICA8bmctdGVtcGxhdGUgI3N0YXRpY1JlZlRwbD5cbiAgICAgICAgICA8c3BhbiBjbGFzcz1cInJlZmVyZW5jZVwiPnt7cmVmLnJlZklkfX08L3NwYW4+XG4gICAgICAgIDwvbmctdGVtcGxhdGU+XG5cbiAgICAgIDwvbmctdGVtcGxhdGU+XG5cbiAgICAgIDxuZy10ZW1wbGF0ZSByZW1hcmtUZW1wbGF0ZT1cInN0cmVhbWluZy1wbGFjZWhvbGRlclwiPlxuICAgICAgICA8c3BhbiBjbGFzcz1cInBsYWNlaG9sZGVyLWdsb3dcIiAqbmdJZj1cInN0cmVhbWluZ1wiPlxuICAgICAgICAgIDxzcGFuIGNsYXNzPVwicGxhY2Vob2xkZXIgbXMtMVwiPjwvc3Bhbj5cbiAgICAgICAgPC9zcGFuPlxuICAgICAgPC9uZy10ZW1wbGF0ZT5cblxuICAgICAgPG5nLXRlbXBsYXRlIHJlbWFya1RlbXBsYXRlPVwiY29kZVwiIGxldC1ub2RlPlxuICAgICAgICA8ZGl2IGNsYXNzPVwiY2FyZCBtYi0yXCI+XG4gICAgICAgICAgPGRpdiBjbGFzcz1cImNhcmQtaGVhZGVyIGQtZmxleCBqdXN0aWZ5LWNvbnRlbnQtYmV0d2VlbiBhbGlnbi1pdGVtcy1jZW50ZXJcIj5cbiAgICAgICAgICAgIDxzcGFuPnt7bm9kZS5sYW5nfX08L3NwYW4+XG4gICAgICAgICAgICA8YnV0dG9uIGNsYXNzPVwiYnRuIGJ0bi1saWdodCBidG4tc21cIiAoY2xpY2spPVwiY29weVRvQ2xpcGJvYXJkKG5vZGUudmFsdWUpXCI+PGkgY2xhc3M9XCJmYXIgZmEtZncgZmEtY2xpcGJvYXJkXCI+PC9pPiBDb3B5IGNvZGU8L2J1dHRvbj5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICA8cHJlIGNsYXNzPVwibGFuZ3VhZ2Ute3tub2RlLmxhbmd9fSBteS0wIHJvdW5kZWQtMCByb3VuZGVkLWJvdHRvbVwiPjxjb2RlIGNsYXNzPVwibGFuZ3VhZ2Ute3tub2RlLmxhbmd9fVwiPnt7bm9kZS52YWx1ZX19PC9jb2RlPjwvcHJlPlxuICAgICAgICA8L2Rpdj5cbiAgICAgIDwvbmctdGVtcGxhdGU+XG5cbiAgICAgIDxuZy10ZW1wbGF0ZSByZW1hcmtUZW1wbGF0ZT1cImxpbmtcIiBsZXQtbm9kZT5cbiAgICAgICAgPGEgW2hyZWZdPVwibm9kZS51cmxcIiB0YXJnZXQ9XCJfYmxhbmtcIiByZWw9XCJub29wZW5lciBub3JlZmVycmVyXCI+e3tnZXRMaW5rVGV4dChub2RlKX19PC9hPlxuICAgICAgPC9uZy10ZW1wbGF0ZT5cblxuICAgIDwvcmVtYXJrPlxuXG4gICAgPHAgKm5nSWY9XCJtZXNzYWdlLnJvbGUgPT09ICd1c2VyJ1wiPnt7bWVzc2FnZS5jb250ZW50fX08L3A+XG5cbiAgICA8IS0tIExpc3Qgb2YgcmVmZXJlbmNlLCBpZiBhbnkgLS0+XG4gICAgPGRpdiAqbmdJZj1cInJlZmVyZW5jZXM/Lmxlbmd0aFwiIGNsYXNzPVwicmVmZXJlbmNlc1wiPlxuICAgICAgPHNwYW4gY2xhc3M9XCJyZWZlcmVuY2VzLXRpdGxlXCI+UmVmZXJlbmNlczwvc3Bhbj4gPGkgY2xhc3M9XCJmYXMgcmVmZXJlbmNlcy1jb2xsYXBzZVwiIFtjbGFzcy5mYS1jaGV2cm9uLWRvd25dPVwic2hvd1JlZmVyZW5jZXNcIiBbY2xhc3MuZmEtY2hldnJvbi1yaWdodF09XCIhc2hvd1JlZmVyZW5jZXNcIiAoY2xpY2spPVwic2hvd1JlZmVyZW5jZXM9IXNob3dSZWZlcmVuY2VzXCI+PC9pPlxuICAgICAgPHNxLWNvbGxhcHNlIFtjb2xsYXBzZWRdPVwiIXNob3dSZWZlcmVuY2VzXCI+XG4gICAgICAgIDxuZy10ZW1wbGF0ZT5cbiAgICAgICAgICA8dWw+XG4gICAgICAgICAgICA8bmctY29udGFpbmVyICpuZ0Zvcj1cImxldCByZWZlcmVuY2Ugb2YgcmVmZXJlbmNlc1wiPlxuICAgICAgICAgICAgICA8bGkgKm5nSWY9XCJyZWZlcmVuY2VNYXAuZ2V0KHJlZmVyZW5jZSkgYXMgYXR0YWNobWVudFwiICBjbGFzcz1cInRleHQtdHJ1bmNhdGVcIj5cbiAgICAgICAgICAgICAgICA8c3EtY2hhdC1yZWZlcmVuY2UgW2NsYXNzLmV4cGFuZGVkXT1cImF0dGFjaG1lbnQuJGV4cGFuZGVkXCIgW2F0dGFjaG1lbnRdPVwiYXR0YWNobWVudFwiIFtyZWZlcmVuY2VdPVwicmVmZXJlbmNlXCJcbiAgICAgICAgICAgICAgICAgIChvcGVuUHJldmlldyk9XCJvcGVuUHJldmlldy5lbWl0KCRldmVudClcIiAob3BlbkRvY3VtZW50KT1cIm9wZW5Eb2N1bWVudCgkZXZlbnQpXCI+PC9zcS1jaGF0LXJlZmVyZW5jZT5cbiAgICAgICAgICAgICAgPC9saT5cbiAgICAgICAgICAgIDwvbmctY29udGFpbmVyPlxuICAgICAgICAgIDwvdWw+XG4gICAgICAgIDwvbmctdGVtcGxhdGU+XG4gICAgICA8L3NxLWNvbGxhcHNlPlxuICAgIDwvZGl2PlxuXG4gIDwvZGl2PlxuXG4gICAgPCEtLSBFZGl0IC8gUmVnZW5lcmF0ZSBmbG9hdGluZyBhY3Rpb25zIC0tPlxuICA8ZGl2IGNsYXNzPVwic3EtY2hhdC1tZXNzYWdlLWFjdGlvblwiPlxuICAgIDxidXR0b24gY2xhc3M9XCJidG4gYnRuLXNtXCIgKm5nSWY9XCJjYW5FZGl0XCIgc3FUb29sdGlwPVwiRWRpdCBtZXNzYWdlXCJcbiAgICAgICAgICAgIChjbGljayk9XCJlZGl0LmVtaXQobWVzc2FnZSlcIj5cbiAgICAgIDxpIGNsYXNzPVwiZmFzIGZhLWVkaXRcIj48L2k+XG4gICAgPC9idXR0b24+XG4gICAgPGJ1dHRvbiBjbGFzcz1cImJ0biBidG4tc21cIiAqbmdJZj1cImNhbkNvcHlcIiBzcVRvb2x0aXA9XCJDb3B5IHRvIGNsaXBib2FyZFwiXG4gICAgICAgICAgICAoY2xpY2spPVwiY29weVRvQ2xpcGJvYXJkKG1lc3NhZ2UuY29udGVudClcIj5cbiAgICAgIDxpIGNsYXNzPVwiZmFyIGZhLWNsaXBib2FyZFwiPjwvaT5cbiAgICA8L2J1dHRvbj5cbiAgICA8YnV0dG9uIGNsYXNzPVwiYnRuIGJ0bi1zbVwiICpuZ0lmPVwiY2FuUmVnZW5lcmF0ZVwiIHNxVG9vbHRpcD1cIlJlZ2VuZXJhdGUgbWVzc2FnZVwiXG4gICAgICAgICAgICAoY2xpY2spPVwicmVnZW5lcmF0ZS5lbWl0KG1lc3NhZ2UpXCI+XG4gICAgICA8aSBjbGFzcz1cImZhcyBmYS1zeW5jLWFsdFwiPjwvaT5cbiAgICA8L2J1dHRvbj5cbiAgPC9kaXY+XG5cbiAgPG5nLXRlbXBsYXRlICN0b29sdGlwVHBsIGxldC1yZWY+XG4gICAgPHNxLWNoYXQtcmVmZXJlbmNlIGNsYXNzPVwiZXhwYW5kZWRcIlxuICAgICAgICAgICAgICAgICAgICAgICAgW2F0dGFjaG1lbnRdPVwicmVmXCJcbiAgICAgICAgICAgICAgICAgICAgICAgIFtyZWZlcmVuY2VdPVwicmVmLmNvbnRleHRJZFwiXG4gICAgICAgICAgICAgICAgICAgICAgICBbcGFydElkXT1cInJlZi4kcGFydElkXCJcbiAgICAgICAgICAgICAgICAgICAgICAgIChvcGVuUHJldmlldyk9XCJvcGVuUHJldmlldy5lbWl0KCRldmVudClcIlxuICAgICAgICAgICAgICAgICAgICAgICAgKG9wZW5Eb2N1bWVudCk9XCJvcGVuRG9jdW1lbnQoJGV2ZW50KVwiPlxuICAgICAgPC9zcS1jaGF0LXJlZmVyZW5jZT5cbiAgPC9uZy10ZW1wbGF0ZT5cblxuPC9kaXY+XG4iXX0=
|