stream-chat-angular 5.12.5 → 6.0.0-beta.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.
Files changed (165) hide show
  1. package/assets/i18n/en.d.ts +1 -0
  2. package/assets/version.d.ts +1 -1
  3. package/{esm2020 → esm2022}/assets/i18n/en.mjs +2 -1
  4. package/{esm2020 → esm2022}/assets/version.mjs +2 -2
  5. package/{esm2020 → esm2022}/lib/attachment-configuration.service.mjs +4 -4
  6. package/esm2022/lib/attachment-list/attachment-list.component.mjs +212 -0
  7. package/esm2022/lib/attachment-preview-list/attachment-preview-list.component.mjs +55 -0
  8. package/{esm2020 → esm2022}/lib/attachment.service.mjs +5 -5
  9. package/esm2022/lib/avatar/avatar.component.mjs +157 -0
  10. package/{esm2020 → esm2022}/lib/avatar-placeholder/avatar-placeholder.component.mjs +6 -6
  11. package/esm2022/lib/channel/channel.component.mjs +45 -0
  12. package/esm2022/lib/channel-header/channel-header.component.mjs +72 -0
  13. package/esm2022/lib/channel-list/channel-list.component.mjs +50 -0
  14. package/esm2022/lib/channel-preview/channel-preview.component.mjs +150 -0
  15. package/esm2022/lib/channel.service.mjs +1381 -0
  16. package/esm2022/lib/chat-client.service.mjs +227 -0
  17. package/{esm2020 → esm2022}/lib/custom-templates.service.mjs +5 -5
  18. package/{esm2020 → esm2022}/lib/date-parser.service.mjs +5 -5
  19. package/esm2022/lib/file-utils.mjs +35 -0
  20. package/{esm2020 → esm2022}/lib/get-channel-display-text.mjs +1 -1
  21. package/{esm2020 → esm2022}/lib/get-message-translation.mjs +1 -1
  22. package/{esm2020 → esm2022}/lib/icon/icon-placeholder/icon-placeholder.component.mjs +6 -6
  23. package/{esm2020 → esm2022}/lib/icon/icon.component.mjs +5 -5
  24. package/{esm2020 → esm2022}/lib/icon/icon.module.mjs +11 -11
  25. package/{esm2020 → esm2022}/lib/icon/loading-indicator/loading-indicator.component.mjs +5 -5
  26. package/{esm2020 → esm2022}/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +6 -6
  27. package/{esm2020 → esm2022}/lib/list-users.mjs +1 -1
  28. package/esm2022/lib/message/message.component.mjs +486 -0
  29. package/esm2022/lib/message-actions-box/message-actions-box.component.mjs +120 -0
  30. package/{esm2020 → esm2022}/lib/message-actions.service.mjs +5 -5
  31. package/esm2022/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
  32. package/{esm2020 → esm2022}/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +6 -6
  33. package/{esm2020 → esm2022}/lib/message-input/emoji-input.service.mjs +5 -5
  34. package/{esm2020 → esm2022}/lib/message-input/message-input-config.service.mjs +5 -5
  35. package/esm2022/lib/message-input/message-input.component.mjs +507 -0
  36. package/{esm2020 → esm2022}/lib/message-input/textarea/textarea.component.mjs +5 -5
  37. package/{esm2020 → esm2022}/lib/message-input/textarea.directive.mjs +5 -5
  38. package/{esm2020 → esm2022}/lib/message-input/voice-recorder.service.mjs +5 -5
  39. package/{esm2020 → esm2022}/lib/message-list/group-styles.mjs +1 -1
  40. package/esm2022/lib/message-list/message-list.component.mjs +715 -0
  41. package/{esm2020 → esm2022}/lib/message-preview.mjs +1 -1
  42. package/esm2022/lib/message-reactions/message-reactions.component.mjs +165 -0
  43. package/esm2022/lib/message-reactions-selector/message-reactions-selector.component.mjs +57 -0
  44. package/{esm2020 → esm2022}/lib/message-reactions.service.mjs +5 -5
  45. package/{esm2020 → esm2022}/lib/message-text/message-text.component.mjs +6 -6
  46. package/esm2022/lib/message.service.mjs +43 -0
  47. package/{esm2020 → esm2022}/lib/modal/modal.component.mjs +6 -6
  48. package/{esm2020 → esm2022}/lib/notification/notification.component.mjs +6 -6
  49. package/esm2022/lib/notification-list/notification-list.component.mjs +33 -0
  50. package/{esm2020 → esm2022}/lib/notification.service.mjs +5 -5
  51. package/esm2022/lib/paginated-list/paginated-list.component.mjs +94 -0
  52. package/{esm2020 → esm2022}/lib/parse-date.mjs +1 -1
  53. package/{esm2020 → esm2022}/lib/read-by.mjs +1 -1
  54. package/esm2022/lib/stream-autocomplete-textarea.module.mjs +33 -0
  55. package/{esm2020 → esm2022}/lib/stream-avatar.module.mjs +5 -5
  56. package/{esm2020 → esm2022}/lib/stream-chat.module.mjs +59 -59
  57. package/{esm2020 → esm2022}/lib/stream-i18n.service.mjs +5 -5
  58. package/esm2022/lib/stream-textarea.module.mjs +31 -0
  59. package/{esm2020 → esm2022}/lib/theme.service.mjs +5 -5
  60. package/{esm2020 → esm2022}/lib/thread/thread.component.mjs +6 -6
  61. package/{esm2020 → esm2022}/lib/transliteration.service.mjs +5 -5
  62. package/esm2022/lib/types.mjs +2 -0
  63. package/{esm2020 → esm2022}/lib/user-list/user-list.component.mjs +5 -5
  64. package/esm2022/lib/virtualized-list.service.mjs +273 -0
  65. package/{esm2020 → esm2022}/lib/virtualized-message-list.service.mjs +1 -1
  66. package/{esm2020 → esm2022}/lib/voice-recorder/amplitude-recorder.service.mjs +5 -5
  67. package/{esm2020 → esm2022}/lib/voice-recorder/audio-recorder.service.mjs +5 -5
  68. package/{esm2020 → esm2022}/lib/voice-recorder/media-recorder.mjs +1 -1
  69. package/esm2022/lib/voice-recorder/mp3-transcoder.mjs +61 -0
  70. package/esm2022/lib/voice-recorder/transcoder.service.mjs +121 -0
  71. package/esm2022/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +32 -0
  72. package/esm2022/lib/voice-recorder/voice-recorder.component.mjs +80 -0
  73. package/{esm2020 → esm2022}/lib/voice-recorder/voice-recorder.module.mjs +9 -9
  74. package/esm2022/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +112 -0
  75. package/esm2022/lib/voice-recording/voice-recording.component.mjs +91 -0
  76. package/{esm2020 → esm2022}/lib/voice-recording/voice-recording.module.mjs +5 -5
  77. package/{esm2020 → esm2022}/lib/wave-form-sampler.mjs +1 -1
  78. package/esm2022/public-api.mjs +82 -0
  79. package/{fesm2020 → fesm2022}/stream-chat-angular.mjs +854 -1126
  80. package/fesm2022/stream-chat-angular.mjs.map +1 -0
  81. package/lib/attachment-list/attachment-list.component.d.ts +2 -5
  82. package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +2 -2
  83. package/lib/attachment.service.d.ts +1 -1
  84. package/lib/avatar/avatar.component.d.ts +4 -4
  85. package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +1 -1
  86. package/lib/channel-list/channel-list.component.d.ts +1 -0
  87. package/lib/channel-preview/channel-preview.component.d.ts +3 -4
  88. package/lib/channel.service.d.ts +40 -106
  89. package/lib/chat-client.service.d.ts +1 -4
  90. package/lib/custom-templates.service.d.ts +10 -10
  91. package/lib/icon/icon-placeholder/icon-placeholder.component.d.ts +1 -1
  92. package/lib/icon/icon.component.d.ts +2 -2
  93. package/lib/message/message.component.d.ts +2 -2
  94. package/lib/message-actions-box/message-actions-box.component.d.ts +2 -3
  95. package/lib/message-actions.service.d.ts +1 -1
  96. package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +2 -2
  97. package/lib/message-input/message-input.component.d.ts +2 -2
  98. package/lib/message-input/textarea/textarea.component.d.ts +1 -1
  99. package/lib/message-input/textarea.directive.d.ts +2 -2
  100. package/lib/message-list/group-styles.d.ts +1 -1
  101. package/lib/message-list/message-list.component.d.ts +2 -3
  102. package/lib/message-reactions/message-reactions.component.d.ts +2 -3
  103. package/lib/message-reactions-selector/message-reactions-selector.component.d.ts +1 -2
  104. package/lib/message-text/message-text.component.d.ts +2 -2
  105. package/lib/modal/modal.component.d.ts +1 -1
  106. package/lib/notification/notification.component.d.ts +1 -1
  107. package/lib/notification-list/notification-list.component.d.ts +0 -1
  108. package/lib/paginated-list/paginated-list.component.d.ts +5 -2
  109. package/lib/read-by.d.ts +1 -1
  110. package/lib/types.d.ts +98 -84
  111. package/lib/user-list/user-list.component.d.ts +1 -1
  112. package/lib/voice-recorder/amplitude-recorder.service.d.ts +2 -2
  113. package/lib/voice-recorder/media-recorder.d.ts +2 -2
  114. package/lib/voice-recorder/transcoder.service.d.ts +4 -4
  115. package/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.d.ts +0 -1
  116. package/lib/voice-recorder/voice-recorder.component.d.ts +2 -2
  117. package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +3 -3
  118. package/lib/voice-recording/voice-recording.component.d.ts +1 -1
  119. package/package.json +15 -21
  120. package/public-api.d.ts +0 -1
  121. package/src/assets/i18n/en.ts +1 -0
  122. package/src/assets/version.ts +1 -1
  123. package/esm2020/lib/attachment-list/attachment-list.component.mjs +0 -224
  124. package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +0 -55
  125. package/esm2020/lib/avatar/avatar.component.mjs +0 -160
  126. package/esm2020/lib/channel/channel.component.mjs +0 -45
  127. package/esm2020/lib/channel-header/channel-header.component.mjs +0 -72
  128. package/esm2020/lib/channel-list/channel-list.component.mjs +0 -47
  129. package/esm2020/lib/channel-preview/channel-preview.component.mjs +0 -155
  130. package/esm2020/lib/channel-query.mjs +0 -77
  131. package/esm2020/lib/channel.service.mjs +0 -1546
  132. package/esm2020/lib/chat-client.service.mjs +0 -227
  133. package/esm2020/lib/file-utils.mjs +0 -35
  134. package/esm2020/lib/message/message.component.mjs +0 -486
  135. package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +0 -123
  136. package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +0 -71
  137. package/esm2020/lib/message-input/message-input.component.mjs +0 -507
  138. package/esm2020/lib/message-list/message-list.component.mjs +0 -717
  139. package/esm2020/lib/message-reactions/message-reactions.component.mjs +0 -168
  140. package/esm2020/lib/message-reactions-selector/message-reactions-selector.component.mjs +0 -61
  141. package/esm2020/lib/message.service.mjs +0 -43
  142. package/esm2020/lib/notification-list/notification-list.component.mjs +0 -36
  143. package/esm2020/lib/paginated-list/paginated-list.component.mjs +0 -94
  144. package/esm2020/lib/stream-autocomplete-textarea.module.mjs +0 -33
  145. package/esm2020/lib/stream-textarea.module.mjs +0 -31
  146. package/esm2020/lib/types.mjs +0 -2
  147. package/esm2020/lib/virtualized-list.service.mjs +0 -271
  148. package/esm2020/lib/voice-recorder/mp3-transcoder.mjs +0 -61
  149. package/esm2020/lib/voice-recorder/transcoder.service.mjs +0 -121
  150. package/esm2020/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +0 -35
  151. package/esm2020/lib/voice-recorder/voice-recorder.component.mjs +0 -80
  152. package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +0 -112
  153. package/esm2020/lib/voice-recording/voice-recording.component.mjs +0 -91
  154. package/esm2020/public-api.mjs +0 -83
  155. package/fesm2015/stream-chat-angular.mjs +0 -9141
  156. package/fesm2015/stream-chat-angular.mjs.map +0 -1
  157. package/fesm2020/stream-chat-angular.mjs.map +0 -1
  158. package/lib/channel-query.d.ts +0 -26
  159. /package/{esm2020 → esm2022}/lib/format-duration.mjs +0 -0
  160. /package/{esm2020 → esm2022}/lib/injection-tokens.mjs +0 -0
  161. /package/{esm2020 → esm2022}/lib/is-image-attachment.mjs +0 -0
  162. /package/{esm2020 → esm2022}/lib/is-on-separate-date.mjs +0 -0
  163. /package/{esm2020 → esm2022}/lib/is-safari.mjs +0 -0
  164. /package/{esm2020 → esm2022}/lib/message-input/textarea.interface.mjs +0 -0
  165. /package/{esm2020 → esm2022}/stream-chat-angular.mjs +0 -0
@@ -0,0 +1,715 @@
1
+ import { ChangeDetectionStrategy, Component, HostBinding, Input, ViewChild, } from '@angular/core';
2
+ import { BehaviorSubject, Subject } from 'rxjs';
3
+ import { filter, map, shareReplay, take, tap, throttleTime, } from 'rxjs/operators';
4
+ import { getGroupStyles } from './group-styles';
5
+ import { listUsers } from '../list-users';
6
+ import { isOnSeparateDate } from '../is-on-separate-date';
7
+ import { VirtualizedMessageListService } from '../virtualized-message-list.service';
8
+ import { isSafari } from '../is-safari';
9
+ import * as i0 from "@angular/core";
10
+ import * as i1 from "../channel.service";
11
+ import * as i2 from "../chat-client.service";
12
+ import * as i3 from "../custom-templates.service";
13
+ import * as i4 from "../date-parser.service";
14
+ import * as i5 from "@angular/common";
15
+ import * as i6 from "../icon/icon.component";
16
+ import * as i7 from "../icon/icon-placeholder/icon-placeholder.component";
17
+ import * as i8 from "../icon/loading-indicator-placeholder/loading-indicator-placeholder.component";
18
+ import * as i9 from "../message/message.component";
19
+ import * as i10 from "@ngx-translate/core";
20
+ /**
21
+ * The `MessageList` component renders a scrollable list of messages.
22
+ */
23
+ export class MessageListComponent {
24
+ get class() {
25
+ return `str-chat-angular__main-panel-inner str-chat-angular__message-list-host str-chat__main-panel-inner ${this.isEmpty ? 'str-chat-angular__message-list-host--empty' : ''}`;
26
+ }
27
+ constructor(channelService, chatClientService, customTemplatesService, dateParser, ngZone, cdRef) {
28
+ this.channelService = channelService;
29
+ this.chatClientService = chatClientService;
30
+ this.customTemplatesService = customTemplatesService;
31
+ this.dateParser = dateParser;
32
+ this.ngZone = ngZone;
33
+ this.cdRef = cdRef;
34
+ /**
35
+ * Determines if the message list should display channel messages or [thread messages](/chat/docs/javascript/threads/).
36
+ */
37
+ this.mode = 'main';
38
+ /**
39
+ * The direction of the messages in the list, `bottom-to-top` means newest message is at the bottom of the message list and users scroll upwards to load older messages
40
+ */
41
+ this.direction = 'bottom-to-top';
42
+ /**
43
+ * You can hide the "jump to latest" button while scrolling. A potential use-case for this input would be to [workaround a known issue on iOS Safar webview](https://github.com/GetStream/stream-chat-angular/issues/418)
44
+ *
45
+ */
46
+ this.hideJumpToLatestButtonDuringScroll = false;
47
+ /**
48
+ * If `true` date separators will be displayed
49
+ */
50
+ this.displayDateSeparator = true;
51
+ /**
52
+ * If `true` unread indicator will be displayed
53
+ */
54
+ this.displayUnreadSeparator = true;
55
+ /**
56
+ * If date separators are displayed, you can set the horizontal position of the date text.
57
+ */
58
+ this.dateSeparatorTextPos = 'center';
59
+ /**
60
+ * `last-message` option will open the message list at the last message, `last-read-message` will open the list at the last unread message. This option only works if mode is `main`.
61
+ */
62
+ this.openMessageListAt = 'last-message';
63
+ /**
64
+ * If the user has unread messages when they open the channel the UI shows the unread indicator / notification which features the unread count by default. This count will be increased every time a user receives a new message. If you don't want to show the unread count, you can turn that off.
65
+ *
66
+ * This is only applicable for `main` mode, as threads doesn't have read infromation.
67
+ */
68
+ this.hideUnreadCountForNotificationAndIndicator = false;
69
+ /**
70
+ * You can turn on and off the loading indicator that signals to users that more messages are being loaded to the message list
71
+ */
72
+ this.displayLoadingIndicator = true;
73
+ this.emptyMainMessageListTemplate = null;
74
+ this.emptyThreadMessageListTemplate = null;
75
+ this.enabledMessageActions = [];
76
+ this.isEmpty = true;
77
+ this.newMessageCountWhileBeingScrolled = 0;
78
+ this.groupStyles = [];
79
+ this.isNextMessageOnSeparateDate = [];
80
+ this.loadingState = 'idle';
81
+ this.isUnreadNotificationVisible = true;
82
+ this.isJumpingToLatestUnreadMessage = false;
83
+ this.isJumpToLatestButtonVisible = true;
84
+ this.isJumpingToMessage = false;
85
+ this.scroll$ = new Subject();
86
+ this.isNewMessageSentByUser = false;
87
+ this.subscriptions = [];
88
+ this.isLatestMessageInList = true;
89
+ this.parsedDates = new Map();
90
+ this.isViewInited = false;
91
+ this.jumpToMessageTimeouts = [];
92
+ this.forceRepaintSubject = new Subject();
93
+ this.isSafari = isSafari();
94
+ this.scrollPosition$ = new BehaviorSubject('bottom');
95
+ this.messageNotificationJumpClicked = () => {
96
+ this.jumpToFirstUnreadMessage();
97
+ this.isUnreadNotificationVisible = false;
98
+ };
99
+ this.messageNotificationDismissClicked = () => {
100
+ this.isUnreadNotificationVisible = false;
101
+ };
102
+ this.usersTypingInChannel$ = this.channelService.usersTypingInChannel$;
103
+ this.usersTypingInThread$ = this.channelService.usersTypingInThread$;
104
+ }
105
+ ngOnChanges(changes) {
106
+ if (changes.mode || changes.direction) {
107
+ this.resetScrollState();
108
+ this.setMessages$();
109
+ }
110
+ if (changes.direction) {
111
+ if (this.scrollContainer?.nativeElement) {
112
+ this.jumpToLatestMessage();
113
+ }
114
+ }
115
+ }
116
+ ngOnInit() {
117
+ this.subscriptions.push(this.forceRepaintSubject.pipe(throttleTime(1000)).subscribe(() => {
118
+ this.forceRepaint();
119
+ }));
120
+ this.subscriptions.push(this.channelService.activeChannel$.subscribe((channel) => {
121
+ let wasChannelSwitch = false;
122
+ if (this.channelId !== channel?.id) {
123
+ wasChannelSwitch = true;
124
+ if (this.checkIfUnreadNotificationIsVisibleTimeout) {
125
+ clearTimeout(this.checkIfUnreadNotificationIsVisibleTimeout);
126
+ }
127
+ this.jumpToMessageTimeouts.forEach((timeout) => clearTimeout(timeout));
128
+ this.jumpToMessageTimeouts = [];
129
+ this.highlightedMessageId = undefined;
130
+ this.isUnreadNotificationVisible = false;
131
+ this.parsedDates = new Map();
132
+ this.resetScrollState();
133
+ this.setMessages$();
134
+ this.channelId = channel?.id;
135
+ if (this.isViewInited) {
136
+ this.cdRef.detectChanges();
137
+ }
138
+ }
139
+ if (this.mode === 'main') {
140
+ const lastReadMessageId = this.channelService.activeChannelLastReadMessageId;
141
+ const unreadCount = this.channelService.activeChannelUnreadCount;
142
+ if (lastReadMessageId !== this.lastReadMessageId ||
143
+ unreadCount !== this.unreadCount) {
144
+ this.lastReadMessageId = lastReadMessageId;
145
+ this.unreadCount = unreadCount || 0;
146
+ if (wasChannelSwitch && this.lastReadMessageId) {
147
+ // Delay jumping to last read message in case we need to give precedence to channelService.jumpToMessage
148
+ if (this.openMessageListAt === 'last-read-message') {
149
+ setTimeout(() => {
150
+ // Don't jump if a jump to a message was already started (using channelService.jumpToMessage)
151
+ if (!this.isJumpingToMessage &&
152
+ !this.channelService.isMessageLoadingInProgress) {
153
+ this.jumpToFirstUnreadMessage();
154
+ }
155
+ }, 0);
156
+ }
157
+ else {
158
+ // Wait till messages and the unread banner is rendered
159
+ // If unread banner isn't visible on the screen, we display the unread notificaion
160
+ setTimeout(() => {
161
+ const bannerElement = document.getElementById('stream-chat-new-message-indicator');
162
+ if (!bannerElement ||
163
+ bannerElement?.offsetTop <
164
+ this.scrollContainer?.nativeElement?.scrollHeight -
165
+ this.scrollContainer?.nativeElement?.clientHeight) {
166
+ this.isUnreadNotificationVisible = true;
167
+ if (this.isViewInited) {
168
+ this.cdRef.detectChanges();
169
+ }
170
+ }
171
+ }, 100);
172
+ }
173
+ }
174
+ if (this.isViewInited) {
175
+ this.cdRef.detectChanges();
176
+ }
177
+ }
178
+ }
179
+ else if (this.lastReadMessageId) {
180
+ this.lastReadMessageId = undefined;
181
+ this.unreadCount = 0;
182
+ if (this.isViewInited) {
183
+ this.cdRef.detectChanges();
184
+ }
185
+ }
186
+ const capabilites = channel?.data?.own_capabilities;
187
+ const capabilitesString = [...(capabilites || [])].sort().join('');
188
+ const enabledActionsString = [...(this.enabledMessageActions || [])]
189
+ .sort()
190
+ .join('');
191
+ if (capabilitesString !== enabledActionsString) {
192
+ this.enabledMessageActions = capabilites || [];
193
+ if (this.isViewInited) {
194
+ this.cdRef.detectChanges();
195
+ }
196
+ }
197
+ this.newMessageSubscription?.unsubscribe();
198
+ if (channel) {
199
+ this.newMessageSubscription = channel.on('message.new', (event) => {
200
+ if (!event.message) {
201
+ return;
202
+ }
203
+ else {
204
+ this.newMessageReceived(event.message);
205
+ }
206
+ });
207
+ }
208
+ }));
209
+ this.subscriptions.push(this.channelService.activeParentMessage$.subscribe((message) => {
210
+ if (!message && this.parentMessage && this.mode === 'thread') {
211
+ this.resetScrollState();
212
+ }
213
+ if (message &&
214
+ this.parentMessage &&
215
+ message.id !== this.parentMessage.id &&
216
+ this.mode === 'thread') {
217
+ this.resetScrollState();
218
+ this.setMessages$();
219
+ }
220
+ if (this.parentMessage === message) {
221
+ return;
222
+ }
223
+ this.parentMessage = message;
224
+ if (this.isViewInited) {
225
+ this.cdRef.detectChanges();
226
+ }
227
+ }));
228
+ this.subscriptions.push(this.customTemplatesService.messageTemplate$.subscribe((template) => {
229
+ if (this.messageTemplate === template) {
230
+ return;
231
+ }
232
+ this.messageTemplate = template;
233
+ if (this.isViewInited) {
234
+ this.cdRef.detectChanges();
235
+ }
236
+ }));
237
+ this.subscriptions.push(this.customTemplatesService.dateSeparatorTemplate$.subscribe((template) => {
238
+ if (this.customDateSeparatorTemplate === template) {
239
+ return;
240
+ }
241
+ this.customDateSeparatorTemplate = template;
242
+ if (this.isViewInited) {
243
+ this.cdRef.detectChanges();
244
+ }
245
+ }));
246
+ this.subscriptions.push(this.customTemplatesService.newMessagesIndicatorTemplate$.subscribe((template) => {
247
+ if (this.customnewMessagesIndicatorTemplate === template) {
248
+ return;
249
+ }
250
+ this.customnewMessagesIndicatorTemplate = template;
251
+ if (this.isViewInited) {
252
+ this.cdRef.detectChanges();
253
+ }
254
+ }));
255
+ this.subscriptions.push(this.customTemplatesService.newMessagesNotificationTemplate$.subscribe((template) => {
256
+ if (this.customnewMessagesNotificationTemplate === template) {
257
+ return;
258
+ }
259
+ this.customnewMessagesNotificationTemplate = template;
260
+ if (this.isViewInited) {
261
+ this.cdRef.detectChanges();
262
+ }
263
+ }));
264
+ this.subscriptions.push(this.customTemplatesService.typingIndicatorTemplate$.subscribe((template) => {
265
+ if (this.typingIndicatorTemplate === template) {
266
+ return;
267
+ }
268
+ this.typingIndicatorTemplate = template;
269
+ if (this.isViewInited) {
270
+ this.cdRef.detectChanges();
271
+ }
272
+ }));
273
+ this.subscriptions.push(this.customTemplatesService.emptyMainMessageListPlaceholder$.subscribe((template) => {
274
+ const isChanged = this.emptyMainMessageListTemplate !== template;
275
+ this.emptyMainMessageListTemplate = template || null;
276
+ if (isChanged && this.isViewInited) {
277
+ this.cdRef.detectChanges();
278
+ }
279
+ }));
280
+ this.subscriptions.push(this.customTemplatesService.emptyThreadMessageListPlaceholder$.subscribe((template) => {
281
+ const isChanged = this.emptyThreadMessageListTemplate !== template;
282
+ this.emptyThreadMessageListTemplate = template || null;
283
+ if (isChanged && this.isViewInited) {
284
+ this.cdRef.detectChanges();
285
+ }
286
+ }));
287
+ this.setMessages$();
288
+ }
289
+ ngAfterViewInit() {
290
+ this.isViewInited = true;
291
+ this.ngZone.runOutsideAngular(() => {
292
+ this.scrollContainer?.nativeElement?.addEventListener('scroll', () => this.scrolled());
293
+ });
294
+ }
295
+ ngAfterViewChecked() {
296
+ if (this.isJumpingToMessage) {
297
+ this.isNewMessageSentByUser = false;
298
+ this.messageIdToAnchorTo = undefined;
299
+ this.anchorMessageTopOffset = undefined;
300
+ return;
301
+ }
302
+ if (this.messageIdToAnchorTo && this.loadingState === 'idle') {
303
+ this.preserveScrollbarPosition();
304
+ }
305
+ else if ((!this.isUserScrolled &&
306
+ this.scrollContainer.nativeElement?.scrollHeight >
307
+ this.scrollContainer?.nativeElement.clientHeight &&
308
+ this.getScrollPosition() !==
309
+ (this.direction === 'bottom-to-top' ? 'bottom' : 'top')) ||
310
+ (this.isUserScrolled && this.isNewMessageSentByUser)) {
311
+ this.isNewMessageSentByUser = false;
312
+ this.jumpToLatestMessage();
313
+ }
314
+ }
315
+ ngOnDestroy() {
316
+ this.subscriptions.forEach((s) => s.unsubscribe());
317
+ this.newMessageSubscription?.unsubscribe();
318
+ if (this.scrollEndTimeout) {
319
+ clearTimeout(this.scrollEndTimeout);
320
+ }
321
+ if (this.checkIfUnreadNotificationIsVisibleTimeout) {
322
+ clearTimeout(this.checkIfUnreadNotificationIsVisibleTimeout);
323
+ }
324
+ if (this.jumpToLatestButtonVisibilityTimeout) {
325
+ clearTimeout(this.jumpToLatestButtonVisibilityTimeout);
326
+ }
327
+ this.jumpToMessageTimeouts.forEach((timeout) => clearTimeout(timeout));
328
+ this.disposeVirtualizedList();
329
+ }
330
+ trackByUserId(_, user) {
331
+ return user.id;
332
+ }
333
+ jumpToLatestMessage() {
334
+ if (this.isLatestMessageInList) {
335
+ this.direction === 'bottom-to-top'
336
+ ? this.scrollToBottom()
337
+ : this.scrollToTop();
338
+ }
339
+ else {
340
+ void this.channelService.jumpToMessage('latest', this.mode === 'thread' ? this.parentMessage?.id : undefined);
341
+ }
342
+ }
343
+ scrollToBottom() {
344
+ this.scrollContainer.nativeElement.scrollTop =
345
+ this.scrollContainer.nativeElement.scrollHeight + 0.1;
346
+ if (this.isSafari) {
347
+ this.forceRepaintSubject.next();
348
+ }
349
+ }
350
+ scrollToTop() {
351
+ this.scrollContainer.nativeElement.scrollTop = 0;
352
+ }
353
+ scrolled() {
354
+ if (this.scrollContainer.nativeElement.scrollHeight ===
355
+ this.scrollContainer.nativeElement.clientHeight) {
356
+ if (this.isJumpToLatestButtonVisible) {
357
+ this.isJumpToLatestButtonVisible = false;
358
+ this.newMessageCountWhileBeingScrolled = 0;
359
+ this.cdRef.detectChanges();
360
+ }
361
+ return;
362
+ }
363
+ if (this.scrollContainer.nativeElement.scrollHeight >
364
+ this.scrollContainer.nativeElement.clientHeight) {
365
+ if (!this.isJumpToLatestButtonVisible) {
366
+ this.isJumpToLatestButtonVisible = true;
367
+ }
368
+ }
369
+ this.scroll$.next();
370
+ this.checkIfUserScrolled();
371
+ if (this.hideJumpToLatestButtonDuringScroll) {
372
+ if (this.isJumpToLatestButtonVisible) {
373
+ this.isJumpToLatestButtonVisible = false;
374
+ this.cdRef.detectChanges();
375
+ }
376
+ if (this.jumpToLatestButtonVisibilityTimeout) {
377
+ clearTimeout(this.jumpToLatestButtonVisibilityTimeout);
378
+ }
379
+ this.jumpToLatestButtonVisibilityTimeout = setTimeout(() => {
380
+ if (this.isUserScrolled) {
381
+ this.isJumpToLatestButtonVisible = true;
382
+ this.jumpToLatestButtonVisibilityTimeout = undefined;
383
+ this.cdRef.detectChanges();
384
+ }
385
+ }, 100);
386
+ }
387
+ }
388
+ jumpToFirstUnreadMessage() {
389
+ if (!this.lastReadMessageId) {
390
+ return;
391
+ }
392
+ this.isJumpingToLatestUnreadMessage = true;
393
+ void this.channelService.jumpToMessage(this.lastReadMessageId);
394
+ }
395
+ getTypingIndicatorContext() {
396
+ return {
397
+ usersTyping$: this.usersTyping$,
398
+ };
399
+ }
400
+ getTypingIndicatorText(users) {
401
+ const text = listUsers(users);
402
+ return text;
403
+ }
404
+ isSentByCurrentUser(message) {
405
+ if (!message) {
406
+ return false;
407
+ }
408
+ return message.user?.id === this.chatClientService.chatClient.user?.id;
409
+ }
410
+ parseDate(date) {
411
+ if (this.parsedDates.has(date)) {
412
+ return this.parsedDates.get(date);
413
+ }
414
+ const parsedDate = this.dateParser.parseDate(date);
415
+ this.parsedDates.set(date, parsedDate);
416
+ return parsedDate;
417
+ }
418
+ get replyCountParam() {
419
+ return { replyCount: this.parentMessage?.reply_count };
420
+ }
421
+ get emptyListTemplate() {
422
+ return this.mode === 'main'
423
+ ? this.emptyMainMessageListTemplate
424
+ : this.emptyThreadMessageListTemplate;
425
+ }
426
+ checkIfUserScrolled() {
427
+ let scrollPosition = this.getScrollPosition();
428
+ const isUserScrolled = (this.direction === 'bottom-to-top'
429
+ ? scrollPosition !== 'bottom'
430
+ : scrollPosition !== 'top') || !this.isLatestMessageInList;
431
+ if (this.isUserScrolled !== isUserScrolled) {
432
+ this.ngZone.run(() => {
433
+ this.isUserScrolled = isUserScrolled;
434
+ if (!this.isUserScrolled) {
435
+ this.newMessageCountWhileBeingScrolled = 0;
436
+ }
437
+ this.cdRef.detectChanges();
438
+ });
439
+ }
440
+ const prevScrollPosition = this.scrollPosition$.getValue();
441
+ if (this.direction === 'top-to-bottom') {
442
+ if (scrollPosition === 'top') {
443
+ scrollPosition = 'bottom';
444
+ }
445
+ else if (scrollPosition === 'bottom') {
446
+ scrollPosition = 'top';
447
+ }
448
+ }
449
+ if (prevScrollPosition !== scrollPosition && !this.isJumpingToMessage) {
450
+ if (scrollPosition === 'top' || scrollPosition === 'bottom') {
451
+ this.virtualizedList?.virtualizedItems$
452
+ .pipe(take(1))
453
+ .subscribe((items) => {
454
+ this.messageIdToAnchorTo =
455
+ scrollPosition === 'top'
456
+ ? items[0]?.id
457
+ : items[items.length - 1]?.id;
458
+ this.anchorMessageTopOffset = document
459
+ .getElementById(this.messageIdToAnchorTo)
460
+ ?.getBoundingClientRect()?.top;
461
+ });
462
+ }
463
+ this.ngZone.run(() => {
464
+ this.scrollPosition$.next(scrollPosition);
465
+ });
466
+ }
467
+ }
468
+ preserveScrollbarPosition() {
469
+ if (!this.messageIdToAnchorTo) {
470
+ return;
471
+ }
472
+ const messageToAlignTo = document.getElementById(this.messageIdToAnchorTo);
473
+ this.messageIdToAnchorTo = undefined;
474
+ this.scrollContainer.nativeElement.scrollTop +=
475
+ (messageToAlignTo?.getBoundingClientRect()?.top || 0) -
476
+ (this.anchorMessageTopOffset || 0);
477
+ this.anchorMessageTopOffset = undefined;
478
+ if (this.isSafari) {
479
+ this.forceRepaintSubject.next();
480
+ }
481
+ }
482
+ forceRepaint() {
483
+ // Solves the issue of empty screen on Safari when scrolling
484
+ this.scrollContainer.nativeElement.style.display = 'none';
485
+ // eslint-disable-next-line @typescript-eslint/no-unused-expressions
486
+ this.scrollContainer.nativeElement.offsetHeight; // no need to store this anywhere, the reference is enough
487
+ this.scrollContainer.nativeElement.style.display = '';
488
+ }
489
+ getScrollPosition() {
490
+ let position = 'middle';
491
+ if (Math.ceil(this.scrollContainer.nativeElement.scrollTop) +
492
+ this.scrollContainer.nativeElement.clientHeight +
493
+ 1 >=
494
+ this.scrollContainer.nativeElement.scrollHeight) {
495
+ position = 'bottom';
496
+ }
497
+ else if (Math.floor(this.scrollContainer.nativeElement.scrollTop) <=
498
+ (this.parentMessageElement?.nativeElement?.clientHeight || 0)) {
499
+ position = 'top';
500
+ }
501
+ return position;
502
+ }
503
+ setMessages$() {
504
+ this.disposeVirtualizedList();
505
+ this.virtualizedList = new VirtualizedMessageListService(this.mode, this.scrollPosition$, this.channelService);
506
+ this.queryStateSubscription = this.virtualizedList.queryState$.subscribe((queryState) => {
507
+ let mappedState = 'idle';
508
+ if (queryState.state.includes('loading')) {
509
+ mappedState = queryState.state || 'loading-bottom';
510
+ }
511
+ if (mappedState !== this.loadingState) {
512
+ this.loadingState = mappedState;
513
+ if (this.isViewInited) {
514
+ this.cdRef.detectChanges();
515
+ }
516
+ }
517
+ });
518
+ this.messages$ = this.virtualizedList.virtualizedItems$.pipe(tap(() => {
519
+ if (this.isEmpty) {
520
+ // cdRef.detectChanges() isn't enough here, test will fail
521
+ setTimeout(() => (this.isEmpty = false), 0);
522
+ }
523
+ }), tap((messages) => {
524
+ if (this.isJumpingToLatestUnreadMessage &&
525
+ !this.firstUnreadMessageId &&
526
+ this.lastReadMessageId) {
527
+ const lastReadIndex = messages.findIndex((m) => m.id === this.lastReadMessageId);
528
+ if (lastReadIndex !== -1) {
529
+ this.firstUnreadMessageId = messages[lastReadIndex + 1]?.id;
530
+ }
531
+ }
532
+ }), tap((messages) => (this.lastSentMessageId = [...messages]
533
+ .reverse()
534
+ .find((m) => m.user?.id === this.chatClientService.chatClient?.user?.id &&
535
+ m.status !== 'sending')?.id)), tap((messages) => {
536
+ const latestMessageInList = messages[messages.length - 1];
537
+ const channel = this.channelService.activeChannel;
538
+ const messagesFromState = (this.mode === 'main'
539
+ ? channel?.state.latestMessages
540
+ : channel?.state.threads[this.parentMessage?.id || '']) || [];
541
+ this.isLatestMessageInList =
542
+ !latestMessageInList ||
543
+ latestMessageInList.cid !== channel?.cid ||
544
+ (this.mode === 'thread' &&
545
+ latestMessageInList?.parent_id !== this.parentMessage?.id) ||
546
+ latestMessageInList?.id ===
547
+ messagesFromState[messagesFromState.length - 1]?.id;
548
+ if (!this.isLatestMessageInList) {
549
+ this.isUserScrolled = true;
550
+ }
551
+ }), map((messages) => {
552
+ return this.direction === 'bottom-to-top'
553
+ ? messages
554
+ : [...messages].reverse();
555
+ }), tap((messages) => {
556
+ this.groupStyles = messages.map((m, i) => getGroupStyles(m, messages[i - 1], messages[i + 1], {
557
+ lastReadMessageId: this.lastReadMessageId,
558
+ }));
559
+ this.isNextMessageOnSeparateDate = messages.map((m, i) => this.checkIfOnSeparateDates(m, messages[i + 1]));
560
+ }), shareReplay(1));
561
+ if (this.virtualizedList?.jumpToItem$) {
562
+ this.jumpToItemSubscription = this.virtualizedList.jumpToItem$
563
+ .pipe(filter((jumpToMessage) => !!jumpToMessage.item?.id))
564
+ .subscribe((jumpToMessage) => {
565
+ let messageId = jumpToMessage.item?.id;
566
+ if (messageId) {
567
+ if (this.isJumpingToLatestUnreadMessage) {
568
+ messageId = this.firstUnreadMessageId || messageId;
569
+ }
570
+ if (jumpToMessage.position !== 'bottom') {
571
+ this.highlightedMessageId = messageId;
572
+ }
573
+ else if (this.direction === 'top-to-bottom') {
574
+ jumpToMessage.position = 'top';
575
+ }
576
+ this.isJumpingToMessage = true;
577
+ this.scrollMessageIntoView({
578
+ messageId: this.firstUnreadMessageId || messageId,
579
+ position: jumpToMessage.position || 'middle',
580
+ });
581
+ }
582
+ });
583
+ }
584
+ }
585
+ resetScrollState() {
586
+ this.isEmpty = true;
587
+ this.isUserScrolled = false;
588
+ this.messageIdToAnchorTo = undefined;
589
+ this.anchorMessageTopOffset = undefined;
590
+ this.newMessageCountWhileBeingScrolled = 0;
591
+ this.isNewMessageSentByUser = false;
592
+ this.isLatestMessageInList = true;
593
+ this.isJumpingToMessage = false;
594
+ this.scrollPosition$.next('bottom');
595
+ this.loadingState = 'idle';
596
+ }
597
+ disposeVirtualizedList() {
598
+ this.virtualizedList?.dispose();
599
+ this.jumpToItemSubscription?.unsubscribe();
600
+ this.queryStateSubscription?.unsubscribe();
601
+ }
602
+ get usersTyping$() {
603
+ return this.mode === 'thread'
604
+ ? this.usersTypingInThread$
605
+ : this.usersTypingInChannel$;
606
+ }
607
+ scrollMessageIntoView(options, withRetry = true) {
608
+ const element = document.getElementById(options.messageId);
609
+ this.jumpToMessageTimeouts.forEach((t) => clearTimeout(t));
610
+ this.jumpToMessageTimeouts = [];
611
+ if (!element && withRetry) {
612
+ // If the message was newly inserted into activeChannelMessages$, the message will be rendered after the current change detection cycle -> wait for this cycle to complete
613
+ this.jumpToMessageTimeouts.push(setTimeout(() => this.scrollMessageIntoView(options, false)));
614
+ }
615
+ else if (element) {
616
+ const blockMapping = {
617
+ top: 'start',
618
+ bottom: 'end',
619
+ middle: 'center',
620
+ };
621
+ // We can't know when smooth scrolling ends, so we set the behavior to instant https://github.com/w3c/csswg-drafts/issues/3744
622
+ element.scrollIntoView({
623
+ behavior: 'instant',
624
+ block: blockMapping[options.position],
625
+ });
626
+ if (options.position !== 'middle') {
627
+ options.position === 'bottom'
628
+ ? this.scrollToBottom()
629
+ : this.scrollToTop();
630
+ }
631
+ this.jumpToMessageTimeouts.push(setTimeout(() => {
632
+ this.isJumpingToMessage = false;
633
+ if (!this.isUserScrolled) {
634
+ this.checkIfUserScrolled();
635
+ }
636
+ }, 200));
637
+ this.jumpToMessageTimeouts.push(setTimeout(() => {
638
+ this.highlightedMessageId = undefined;
639
+ this.firstUnreadMessageId = undefined;
640
+ this.isJumpingToLatestUnreadMessage = false;
641
+ this.jumpToMessageTimeouts = [];
642
+ this.cdRef.detectChanges();
643
+ }, 1000));
644
+ }
645
+ else {
646
+ this.isJumpingToMessage = false;
647
+ this.highlightedMessageId = undefined;
648
+ this.firstUnreadMessageId = undefined;
649
+ this.isJumpingToLatestUnreadMessage = false;
650
+ }
651
+ }
652
+ newMessageReceived(message) {
653
+ if ((this.mode === 'main' && message.parent_id) ||
654
+ (this.mode === 'thread' && message.parent_id !== this.parentMessage?.id)) {
655
+ return;
656
+ }
657
+ const isNewMessageSentByCurrentUser = message.user?.id === this.chatClientService.chatClient?.user?.id;
658
+ let shouldDetectChanges = false;
659
+ if (!this.isNewMessageSentByUser && isNewMessageSentByCurrentUser) {
660
+ this.isNewMessageSentByUser = true;
661
+ shouldDetectChanges = true;
662
+ }
663
+ if (this.isUserScrolled) {
664
+ this.newMessageCountWhileBeingScrolled++;
665
+ shouldDetectChanges = true;
666
+ }
667
+ if (!this.isNewMessageSentByUser && this.unreadCount !== undefined) {
668
+ this.unreadCount++;
669
+ shouldDetectChanges = true;
670
+ }
671
+ if (shouldDetectChanges && this.isViewInited) {
672
+ this.cdRef.detectChanges();
673
+ }
674
+ }
675
+ checkIfOnSeparateDates(message, nextMessage) {
676
+ if (!message || !nextMessage) {
677
+ return false;
678
+ }
679
+ return isOnSeparateDate(message.created_at, nextMessage.created_at);
680
+ }
681
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageListComponent, deps: [{ token: i1.ChannelService }, { token: i2.ChatClientService }, { token: i3.CustomTemplatesService }, { token: i4.DateParserService }, { token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
682
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: MessageListComponent, selector: "stream-message-list", inputs: { mode: "mode", direction: "direction", hideJumpToLatestButtonDuringScroll: "hideJumpToLatestButtonDuringScroll", displayDateSeparator: "displayDateSeparator", displayUnreadSeparator: "displayUnreadSeparator", dateSeparatorTextPos: "dateSeparatorTextPos", openMessageListAt: "openMessageListAt", hideUnreadCountForNotificationAndIndicator: "hideUnreadCountForNotificationAndIndicator", displayLoadingIndicator: "displayLoadingIndicator" }, host: { properties: { "class": "this.class" } }, viewQueries: [{ propertyName: "scrollContainer", first: true, predicate: ["scrollContainer"], descendants: true }, { propertyName: "parentMessageElement", first: true, predicate: ["parentMessageElement"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "@if (\n lastReadMessageId &&\n isUnreadNotificationVisible &&\n openMessageListAt === \"last-message\" &&\n displayUnreadSeparator\n) {\n <ng-container\n *ngTemplateOutlet=\"\n customnewMessagesNotificationTemplate ||\n defaultUnreadMessagesNotification;\n context: {\n unreadCount: unreadCount,\n onDismiss: messageNotificationDismissClicked,\n onJump: messageNotificationJumpClicked,\n }\n \"\n />\n}\n<ng-template\n #defaultUnreadMessagesNotification\n let-unreadCount=\"unreadCount\"\n let-onDismiss=\"onDismiss\"\n let-onJump=\"onJump\"\n>\n <div\n class=\"str-chat__unread-messages-notification\"\n data-testid=\"unread-messages-notification\"\n >\n <button\n data-testid=\"unread-messages-notification-jump-to-message\"\n (click)=\"onJump()\"\n >\n @if (unreadCount > 0 && !hideUnreadCountForNotificationAndIndicator) {\n {{\n (unreadCount === 1\n ? \"streamChat.\\{\\{count\\}\\} unread message\"\n : \"streamChat.\\{\\{count\\}\\} unread messages\"\n ) | translate: { count: unreadCount }\n }}\n } @else {\n {{ \"streamChat.Unread messages\" | translate }}\n }\n </button>\n <button\n data-testid=\"unread-messages-notification-dismiss\"\n (click)=\"onDismiss()\"\n >\n <stream-icon-placeholder icon=\"close\" />\n </button>\n </div>\n</ng-template>\n<div #scrollContainer data-testid=\"scroll-container\" class=\"str-chat__list\">\n @if (mode === \"main\" && isEmpty && emptyListTemplate) {\n <ng-container *ngTemplateOutlet=\"emptyListTemplate\" />\n }\n <div class=\"str-chat__reverse-infinite-scroll str-chat__message-list-scroll\">\n <ul class=\"str-chat__ul\">\n @if (mode === \"thread\" && parentMessage) {\n <li\n #parentMessageElement\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage, index: 'parent' }\n \"\n />\n <div data-testid=\"reply-count\" class=\"str-chat__thread-start\">\n {{parentMessage.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n replies' | translate:replyCountParam)}}\n </div>\n </li>\n }\n @if (mode === \"thread\" && isEmpty && emptyListTemplate) {\n <ng-container *ngTemplateOutlet=\"emptyListTemplate\" />\n }\n @if (\n ((loadingState === \"loading-top\" && direction === \"bottom-to-top\") ||\n (loadingState === \"loading-bottom\" &&\n direction === \"top-to-bottom\")) &&\n displayLoadingIndicator\n ) {\n <stream-loading-indicator-placeholder\n data-testid=\"top-loading-indicator\"\n />\n } @else {\n <ng-container *ngTemplateOutlet=\"loadingIndicatorPlaceholder\" />\n }\n @if (messages$ | async; as messages) {\n @for (\n message of messages;\n track message.id;\n let i = $index;\n let isFirst = $first;\n let isLast = $last\n ) {\n @if (isFirst) {\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: message.created_at,\n parsedDate: parseDate(message.created_at),\n }\n \"\n />\n }\n <li\n tabindex=\"0\"\n data-testclass=\"message\"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message, index: i }\n \"\n />\n </li>\n @if (\n (lastReadMessageId === message?.id &&\n direction === \"bottom-to-top\") ||\n (direction === \"top-to-bottom\" &&\n lastReadMessageId === messages[i + 1]?.id)\n ) {\n @if (displayUnreadSeparator) {\n <li\n id=\"stream-chat-new-message-indicator\"\n data-testid=\"new-messages-indicator\"\n class=\"str-chat__li str-chat__unread-messages-separator-wrapper\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n customnewMessagesIndicatorTemplate ||\n defaultNewMessagesIndicator;\n context: { unreadCount: unreadCount }\n \"\n />\n </li>\n }\n }\n @if (isNextMessageOnSeparateDate[i]) {\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: messages[i + 1].created_at,\n parsedDate: parseDate(messages[i + 1].created_at),\n }\n \"\n />\n }\n }\n }\n @if (\n ((loadingState === \"loading-bottom\" && direction === \"bottom-to-top\") ||\n (loadingState === \"loading-top\" && direction === \"top-to-bottom\")) &&\n displayLoadingIndicator\n ) {\n <stream-loading-indicator-placeholder\n data-testid=\"bottom-loading-indicator\"\n />\n } @else {\n <ng-container *ngTemplateOutlet=\"loadingIndicatorPlaceholder\" />\n }\n <ng-template #loadingIndicatorPlaceholder>\n <div class=\"str-chat__loading-indicator-placeholder\"></div>\n </ng-template>\n </ul>\n <ng-template #defaultTypingIndicator let-usersTyping$=\"usersTyping$\">\n <!-- eslint-disable-next-line @angular-eslint/template/no-any -->\n @if ($any(usersTyping$ | async); as users) {\n @if (users.length > 0) {\n <div\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n <div\n data-testid=\"typing-users\"\n class=\"str-chat__typing-indicator__users\"\n >\n {{\n users.length === 1\n ? (\"streamChat.user is typing\"\n | translate: { user: getTypingIndicatorText(users) })\n : (\"streamChat.users are typing\"\n | translate: { users: getTypingIndicatorText(users) })\n }}\n </div>\n </div>\n }\n }\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate || defaultTypingIndicator;\n context: getTypingIndicatorContext()\n \"\n />\n </div>\n</div>\n<div class=\"str-chat__jump-to-latest-message\">\n @if (isUserScrolled && isJumpToLatestButtonVisible) {\n <button\n data-testid=\"scroll-to-latest\"\n class=\"str-chat__message-notification-scroll-to-latest str-chat__message-notification-scroll-to-latest-right str-chat__circle-fab\"\n (keyup.enter)=\"jumpToLatestMessage()\"\n (click)=\"jumpToLatestMessage()\"\n >\n <stream-icon\n class=\"str-chat__jump-to-latest-icon str-chat__circle-fab-icon\"\n [icon]=\"direction === 'bottom-to-top' ? 'arrow-down' : 'arrow-up'\"\n />\n @if (newMessageCountWhileBeingScrolled > 0) {\n <div\n class=\"str-chat__message-notification str-chat__message-notification-scroll-to-latest-unread-count str-chat__jump-to-latest-unread-count\"\n >\n {{ newMessageCountWhileBeingScrolled }}\n </div>\n }\n </button>\n }\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\" let-index=\"index\">\n <ng-template\n #defaultMessageTemplate\n let-messageInput=\"message\"\n let-isLastSentMessage=\"isLastSentMessage\"\n let-enabledMessageActions=\"enabledMessageActions\"\n let-mode=\"mode\"\n let-isHighlighted=\"isHighlighted\"\n let-scroll$=\"scroll$\"\n >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n [scroll$]=\"scroll$\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: {\n message: message,\n isLastSentMessage: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n enabledMessageActions: enabledMessageActions,\n mode: mode,\n isHighlighted: message?.id === highlightedMessageId,\n scroll$: scroll$,\n }\n \"\n />\n</ng-template>\n\n<ng-template #dateSeparator let-date=\"date\" let-parsedDate=\"parsedDate\">\n @if (displayDateSeparator) {\n <ng-container\n *ngTemplateOutlet=\"\n customDateSeparatorTemplate || defaultDateSeparator;\n context: {\n date: date,\n parsedDate: parsedDate,\n }\n \"\n />\n }\n\n <ng-template\n #defaultDateSeparator\n let-date=\"date\"\n let-parsedDate=\"parsedDate\"\n >\n <div data-testid=\"date-separator\" class=\"str-chat__date-separator\">\n @if (\n dateSeparatorTextPos === \"right\" || dateSeparatorTextPos === \"center\"\n ) {\n <hr class=\"str-chat__date-separator-line\" />\n }\n <div class=\"str-chat__date-separator-date\">\n {{ parsedDate }}\n </div>\n @if (\n dateSeparatorTextPos === \"left\" || dateSeparatorTextPos === \"center\"\n ) {\n <hr class=\"str-chat__date-separator-line\" />\n }\n </div>\n </ng-template>\n</ng-template>\n\n<ng-template #defaultNewMessagesIndicator let-unreadCount=\"unreadCount\">\n <div class=\"str-chat__unread-messages-separator\">\n @if (unreadCount > 0 && !hideUnreadCountForNotificationAndIndicator) {\n {{\n (unreadCount === 1\n ? \"streamChat.\\{\\{count\\}\\} unread message\"\n : \"streamChat.\\{\\{count\\}\\} unread messages\"\n ) | translate: { count: unreadCount }\n }}\n } @else {\n {{ \"streamChat.Unread messages\" | translate }}\n }\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i5.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i6.IconComponent, selector: "stream-icon", inputs: ["icon"] }, { kind: "component", type: i7.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon"] }, { kind: "component", type: i8.LoadingIndicatorPlaceholderComponent, selector: "stream-loading-indicator-placeholder" }, { kind: "component", type: i9.MessageComponent, selector: "stream-message", inputs: ["message", "enabledMessageActions", "isLastSentMessage", "mode", "isHighlighted", "scroll$"] }, { kind: "pipe", type: i5.AsyncPipe, name: "async" }, { kind: "pipe", type: i10.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
683
+ }
684
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageListComponent, decorators: [{
685
+ type: Component,
686
+ args: [{ selector: 'stream-message-list', changeDetection: ChangeDetectionStrategy.OnPush, template: "@if (\n lastReadMessageId &&\n isUnreadNotificationVisible &&\n openMessageListAt === \"last-message\" &&\n displayUnreadSeparator\n) {\n <ng-container\n *ngTemplateOutlet=\"\n customnewMessagesNotificationTemplate ||\n defaultUnreadMessagesNotification;\n context: {\n unreadCount: unreadCount,\n onDismiss: messageNotificationDismissClicked,\n onJump: messageNotificationJumpClicked,\n }\n \"\n />\n}\n<ng-template\n #defaultUnreadMessagesNotification\n let-unreadCount=\"unreadCount\"\n let-onDismiss=\"onDismiss\"\n let-onJump=\"onJump\"\n>\n <div\n class=\"str-chat__unread-messages-notification\"\n data-testid=\"unread-messages-notification\"\n >\n <button\n data-testid=\"unread-messages-notification-jump-to-message\"\n (click)=\"onJump()\"\n >\n @if (unreadCount > 0 && !hideUnreadCountForNotificationAndIndicator) {\n {{\n (unreadCount === 1\n ? \"streamChat.\\{\\{count\\}\\} unread message\"\n : \"streamChat.\\{\\{count\\}\\} unread messages\"\n ) | translate: { count: unreadCount }\n }}\n } @else {\n {{ \"streamChat.Unread messages\" | translate }}\n }\n </button>\n <button\n data-testid=\"unread-messages-notification-dismiss\"\n (click)=\"onDismiss()\"\n >\n <stream-icon-placeholder icon=\"close\" />\n </button>\n </div>\n</ng-template>\n<div #scrollContainer data-testid=\"scroll-container\" class=\"str-chat__list\">\n @if (mode === \"main\" && isEmpty && emptyListTemplate) {\n <ng-container *ngTemplateOutlet=\"emptyListTemplate\" />\n }\n <div class=\"str-chat__reverse-infinite-scroll str-chat__message-list-scroll\">\n <ul class=\"str-chat__ul\">\n @if (mode === \"thread\" && parentMessage) {\n <li\n #parentMessageElement\n data-testid=\"parent-message\"\n class=\"str-chat__parent-message-li\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: parentMessage, index: 'parent' }\n \"\n />\n <div data-testid=\"reply-count\" class=\"str-chat__thread-start\">\n {{parentMessage.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n replies' | translate:replyCountParam)}}\n </div>\n </li>\n }\n @if (mode === \"thread\" && isEmpty && emptyListTemplate) {\n <ng-container *ngTemplateOutlet=\"emptyListTemplate\" />\n }\n @if (\n ((loadingState === \"loading-top\" && direction === \"bottom-to-top\") ||\n (loadingState === \"loading-bottom\" &&\n direction === \"top-to-bottom\")) &&\n displayLoadingIndicator\n ) {\n <stream-loading-indicator-placeholder\n data-testid=\"top-loading-indicator\"\n />\n } @else {\n <ng-container *ngTemplateOutlet=\"loadingIndicatorPlaceholder\" />\n }\n @if (messages$ | async; as messages) {\n @for (\n message of messages;\n track message.id;\n let i = $index;\n let isFirst = $first;\n let isLast = $last\n ) {\n @if (isFirst) {\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: message.created_at,\n parsedDate: parseDate(message.created_at),\n }\n \"\n />\n }\n <li\n tabindex=\"0\"\n data-testclass=\"message\"\n class=\"str-chat__li str-chat__li--{{ groupStyles[i] }}\"\n id=\"{{ message.id }}\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplateContainer;\n context: { message: message, index: i }\n \"\n />\n </li>\n @if (\n (lastReadMessageId === message?.id &&\n direction === \"bottom-to-top\") ||\n (direction === \"top-to-bottom\" &&\n lastReadMessageId === messages[i + 1]?.id)\n ) {\n @if (displayUnreadSeparator) {\n <li\n id=\"stream-chat-new-message-indicator\"\n data-testid=\"new-messages-indicator\"\n class=\"str-chat__li str-chat__unread-messages-separator-wrapper\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n customnewMessagesIndicatorTemplate ||\n defaultNewMessagesIndicator;\n context: { unreadCount: unreadCount }\n \"\n />\n </li>\n }\n }\n @if (isNextMessageOnSeparateDate[i]) {\n <ng-container\n *ngTemplateOutlet=\"\n dateSeparator;\n context: {\n date: messages[i + 1].created_at,\n parsedDate: parseDate(messages[i + 1].created_at),\n }\n \"\n />\n }\n }\n }\n @if (\n ((loadingState === \"loading-bottom\" && direction === \"bottom-to-top\") ||\n (loadingState === \"loading-top\" && direction === \"top-to-bottom\")) &&\n displayLoadingIndicator\n ) {\n <stream-loading-indicator-placeholder\n data-testid=\"bottom-loading-indicator\"\n />\n } @else {\n <ng-container *ngTemplateOutlet=\"loadingIndicatorPlaceholder\" />\n }\n <ng-template #loadingIndicatorPlaceholder>\n <div class=\"str-chat__loading-indicator-placeholder\"></div>\n </ng-template>\n </ul>\n <ng-template #defaultTypingIndicator let-usersTyping$=\"usersTyping$\">\n <!-- eslint-disable-next-line @angular-eslint/template/no-any -->\n @if ($any(usersTyping$ | async); as users) {\n @if (users.length > 0) {\n <div\n data-testid=\"typing-indicator\"\n class=\"str-chat__typing-indicator str-chat__typing-indicator--typing\"\n >\n <div class=\"str-chat__typing-indicator__dots\">\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n <span class=\"str-chat__typing-indicator__dot\"></span>\n </div>\n <div\n data-testid=\"typing-users\"\n class=\"str-chat__typing-indicator__users\"\n >\n {{\n users.length === 1\n ? (\"streamChat.user is typing\"\n | translate: { user: getTypingIndicatorText(users) })\n : (\"streamChat.users are typing\"\n | translate: { users: getTypingIndicatorText(users) })\n }}\n </div>\n </div>\n }\n }\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n typingIndicatorTemplate || defaultTypingIndicator;\n context: getTypingIndicatorContext()\n \"\n />\n </div>\n</div>\n<div class=\"str-chat__jump-to-latest-message\">\n @if (isUserScrolled && isJumpToLatestButtonVisible) {\n <button\n data-testid=\"scroll-to-latest\"\n class=\"str-chat__message-notification-scroll-to-latest str-chat__message-notification-scroll-to-latest-right str-chat__circle-fab\"\n (keyup.enter)=\"jumpToLatestMessage()\"\n (click)=\"jumpToLatestMessage()\"\n >\n <stream-icon\n class=\"str-chat__jump-to-latest-icon str-chat__circle-fab-icon\"\n [icon]=\"direction === 'bottom-to-top' ? 'arrow-down' : 'arrow-up'\"\n />\n @if (newMessageCountWhileBeingScrolled > 0) {\n <div\n class=\"str-chat__message-notification str-chat__message-notification-scroll-to-latest-unread-count str-chat__jump-to-latest-unread-count\"\n >\n {{ newMessageCountWhileBeingScrolled }}\n </div>\n }\n </button>\n }\n</div>\n\n<ng-template #messageTemplateContainer let-message=\"message\" let-index=\"index\">\n <ng-template\n #defaultMessageTemplate\n let-messageInput=\"message\"\n let-isLastSentMessage=\"isLastSentMessage\"\n let-enabledMessageActions=\"enabledMessageActions\"\n let-mode=\"mode\"\n let-isHighlighted=\"isHighlighted\"\n let-scroll$=\"scroll$\"\n >\n <stream-message\n [message]=\"messageInput\"\n [isLastSentMessage]=\"isLastSentMessage\"\n [enabledMessageActions]=\"enabledMessageActions\"\n [mode]=\"mode\"\n [isHighlighted]=\"isHighlighted\"\n [scroll$]=\"scroll$\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n messageTemplate || defaultMessageTemplate;\n context: {\n message: message,\n isLastSentMessage: !!(\n lastSentMessageId && message?.id === lastSentMessageId\n ),\n enabledMessageActions: enabledMessageActions,\n mode: mode,\n isHighlighted: message?.id === highlightedMessageId,\n scroll$: scroll$,\n }\n \"\n />\n</ng-template>\n\n<ng-template #dateSeparator let-date=\"date\" let-parsedDate=\"parsedDate\">\n @if (displayDateSeparator) {\n <ng-container\n *ngTemplateOutlet=\"\n customDateSeparatorTemplate || defaultDateSeparator;\n context: {\n date: date,\n parsedDate: parsedDate,\n }\n \"\n />\n }\n\n <ng-template\n #defaultDateSeparator\n let-date=\"date\"\n let-parsedDate=\"parsedDate\"\n >\n <div data-testid=\"date-separator\" class=\"str-chat__date-separator\">\n @if (\n dateSeparatorTextPos === \"right\" || dateSeparatorTextPos === \"center\"\n ) {\n <hr class=\"str-chat__date-separator-line\" />\n }\n <div class=\"str-chat__date-separator-date\">\n {{ parsedDate }}\n </div>\n @if (\n dateSeparatorTextPos === \"left\" || dateSeparatorTextPos === \"center\"\n ) {\n <hr class=\"str-chat__date-separator-line\" />\n }\n </div>\n </ng-template>\n</ng-template>\n\n<ng-template #defaultNewMessagesIndicator let-unreadCount=\"unreadCount\">\n <div class=\"str-chat__unread-messages-separator\">\n @if (unreadCount > 0 && !hideUnreadCountForNotificationAndIndicator) {\n {{\n (unreadCount === 1\n ? \"streamChat.\\{\\{count\\}\\} unread message\"\n : \"streamChat.\\{\\{count\\}\\} unread messages\"\n ) | translate: { count: unreadCount }\n }}\n } @else {\n {{ \"streamChat.Unread messages\" | translate }}\n }\n </div>\n</ng-template>\n" }]
687
+ }], ctorParameters: () => [{ type: i1.ChannelService }, { type: i2.ChatClientService }, { type: i3.CustomTemplatesService }, { type: i4.DateParserService }, { type: i0.NgZone }, { type: i0.ChangeDetectorRef }], propDecorators: { mode: [{
688
+ type: Input
689
+ }], direction: [{
690
+ type: Input
691
+ }], hideJumpToLatestButtonDuringScroll: [{
692
+ type: Input
693
+ }], displayDateSeparator: [{
694
+ type: Input
695
+ }], displayUnreadSeparator: [{
696
+ type: Input
697
+ }], dateSeparatorTextPos: [{
698
+ type: Input
699
+ }], openMessageListAt: [{
700
+ type: Input
701
+ }], hideUnreadCountForNotificationAndIndicator: [{
702
+ type: Input
703
+ }], displayLoadingIndicator: [{
704
+ type: Input
705
+ }], scrollContainer: [{
706
+ type: ViewChild,
707
+ args: ['scrollContainer']
708
+ }], parentMessageElement: [{
709
+ type: ViewChild,
710
+ args: ['parentMessageElement']
711
+ }], class: [{
712
+ type: HostBinding,
713
+ args: ['class']
714
+ }] } });
715
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1saXN0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9tZXNzYWdlLWxpc3QvbWVzc2FnZS1saXN0LmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9tZXNzYWdlLWxpc3QvbWVzc2FnZS1saXN0LmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFHTCx1QkFBdUIsRUFFdkIsU0FBUyxFQUVULFdBQVcsRUFDWCxLQUFLLEVBT0wsU0FBUyxHQUNWLE1BQU0sZUFBZSxDQUFDO0FBRXZCLE9BQU8sRUFBRSxlQUFlLEVBQWMsT0FBTyxFQUFnQixNQUFNLE1BQU0sQ0FBQztBQUMxRSxPQUFPLEVBQ0wsTUFBTSxFQUNOLEdBQUcsRUFDSCxXQUFXLEVBQ1gsSUFBSSxFQUNKLEdBQUcsRUFDSCxZQUFZLEdBQ2IsTUFBTSxnQkFBZ0IsQ0FBQztBQVl4QixPQUFPLEVBQUUsY0FBYyxFQUFjLE1BQU0sZ0JBQWdCLENBQUM7QUFHNUQsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUUxQyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUMxRCxPQUFPLEVBQUUsNkJBQTZCLEVBQUUsTUFBTSxxQ0FBcUMsQ0FBQztBQUNwRixPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sY0FBYyxDQUFDOzs7Ozs7Ozs7Ozs7QUFFeEM7O0dBRUc7QUFPSCxNQUFNLE9BQU8sb0JBQW9CO0lBcUcvQixJQUNZLEtBQUs7UUFDZixPQUFPLHFHQUNMLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLDRDQUE0QyxDQUFDLENBQUMsQ0FBQyxFQUNoRSxFQUFFLENBQUM7SUFDTCxDQUFDO0lBUUQsWUFDVSxjQUE4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RTtZQUNsRSxJQUFJLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDbkIsbUJBQW1CLEdBQUcsSUFBSSxDQUFDO1NBQzVCO1FBRUQsSUFBSSxtQkFBbUIsSUFBSSxJQUFJLENBQUMsWUFBWSxFQUFFO1lBQzVDLElBQUksQ0FBQyxLQUFLLENBQUMsYUFBYSxFQUFFLENBQUM7U0FDNUI7SUFDSCxDQUFDO0lBRU8sc0JBQXNCLENBQzVCLE9BQXVCLEVBQ3ZCLFdBQTJCO1FBRTNCLElBQUksQ0FBQyxPQUFPLElBQUksQ0FBQyxXQUFXLEVBQUU7WUFDNUIsT0FBTyxLQUFLLENBQUM7U0FDZDtRQUNELE9BQU8sZ0JBQWdCLENBQUMsT0FBTyxDQUFDLFVBQVUsRUFBRSxXQUFXLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDdEUsQ0FBQzs4R0E3MEJVLG9CQUFvQjtrR0FBcEIsb0JBQW9CLG95QkN4RGpDLGlrVkE4VEE7OzJGRHRRYSxvQkFBb0I7a0JBTmhDLFNBQVM7K0JBQ0UscUJBQXFCLG1CQUdkLHVCQUF1QixDQUFDLE1BQU07Nk9BUXRDLElBQUk7c0JBQVosS0FBSztnQkFJRyxTQUFTO3NCQUFqQixLQUFLO2dCQUtHLGtDQUFrQztzQkFBMUMsS0FBSztnQkFJRyxvQkFBb0I7c0JBQTVCLEtBQUs7Z0JBSUcsc0JBQXNCO3NCQUE5QixLQUFLO2dCQUlHLG9CQUFvQjtzQkFBNUIsS0FBSztnQkFJRyxpQkFBaUI7c0JBQXpCLEtBQUs7Z0JBT0csMENBQTBDO3NCQUFsRCxLQUFLO2dCQUlHLHVCQUF1QjtzQkFBL0IsS0FBSztnQkFpQ0UsZUFBZTtzQkFEdEIsU0FBUzt1QkFBQyxpQkFBaUI7Z0JBR3BCLG9CQUFvQjtzQkFEM0IsU0FBUzt1QkFBQyxzQkFBc0I7Z0JBMEJyQixLQUFLO3NCQURoQixXQUFXO3VCQUFDLE9BQU8iLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBBZnRlclZpZXdDaGVja2VkLFxuICBBZnRlclZpZXdJbml0LFxuICBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSxcbiAgQ2hhbmdlRGV0ZWN0b3JSZWYsXG4gIENvbXBvbmVudCxcbiAgRWxlbWVudFJlZixcbiAgSG9zdEJpbmRpbmcsXG4gIElucHV0LFxuICBOZ1pvbmUsXG4gIE9uQ2hhbmdlcyxcbiAgT25EZXN0cm95LFxuICBPbkluaXQsXG4gIFNpbXBsZUNoYW5nZXMsXG4gIFRlbXBsYXRlUmVmLFxuICBWaWV3Q2hpbGQsXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ2hhbm5lbFNlcnZpY2UgfSBmcm9tICcuLi9jaGFubmVsLnNlcnZpY2UnO1xuaW1wb3J0IHsgQmVoYXZpb3JTdWJqZWN0LCBPYnNlcnZhYmxlLCBTdWJqZWN0LCBTdWJzY3JpcHRpb24gfSBmcm9tICdyeGpzJztcbmltcG9ydCB7XG4gIGZpbHRlcixcbiAgbWFwLFxuICBzaGFyZVJlcGxheSxcbiAgdGFrZSxcbiAgdGFwLFxuICB0aHJvdHRsZVRpbWUsXG59IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcbmltcG9ydCB7XG4gIE1lc3NhZ2VDb250ZXh0LFxuICBEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzLFxuICBTdHJlYW1NZXNzYWdlLFxuICBUeXBpbmdJbmRpY2F0b3JDb250ZXh0LFxuICBEYXRlU2VwYXJhdG9yQ29udGV4dCxcbiAgVW5yZWFkTWVzc2FnZXNOb3RpZmljYXRpb25Db250ZXh0LFxuICBVbnJlYWRNZXNzYWdlc0luZGljYXRvckNvbnRleHQsXG4gIFZpcnR1YWxpemVkTGlzdFNjcm9sbFBvc2l0aW9uLFxufSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQgeyBDaGF0Q2xpZW50U2VydmljZSB9IGZyb20gJy4uL2NoYXQtY2xpZW50LnNlcnZpY2UnO1xuaW1wb3J0IHsgZ2V0R3JvdXBTdHlsZXMsIEdyb3VwU3R5bGUgfSBmcm9tICcuL2dyb3VwLXN0eWxlcyc7XG5pbXBvcnQgeyBNZXNzYWdlUmVzcG9uc2UsIFVzZXJSZXNwb25zZSB9IGZyb20gJ3N0cmVhbS1jaGF0JztcbmltcG9ydCB7IEN1c3RvbVRlbXBsYXRlc1NlcnZpY2UgfSBmcm9tICcuLi9jdXN0b20tdGVtcGxhdGVzLnNlcnZpY2UnO1xuaW1wb3J0IHsgbGlzdFVzZXJzIH0gZnJvbSAnLi4vbGlzdC11c2Vycyc7XG5pbXBvcnQgeyBEYXRlUGFyc2VyU2VydmljZSB9IGZyb20gJy4uL2RhdGUtcGFyc2VyLnNlcnZpY2UnO1xuaW1wb3J0IHsgaXNPblNlcGFyYXRlRGF0ZSB9IGZyb20gJy4uL2lzLW9uLXNlcGFyYXRlLWRhdGUnO1xuaW1wb3J0IHsgVmlydHVhbGl6ZWRNZXNzYWdlTGlzdFNlcnZpY2UgfSBmcm9tICcuLi92aXJ0dWFsaXplZC1tZXNzYWdlLWxpc3Quc2VydmljZSc7XG5pbXBvcnQgeyBpc1NhZmFyaSB9IGZyb20gJy4uL2lzLXNhZmFyaSc7XG5cbi8qKlxuICogVGhlIGBNZXNzYWdlTGlzdGAgY29tcG9uZW50IHJlbmRlcnMgYSBzY3JvbGxhYmxlIGxpc3Qgb2YgbWVzc2FnZXMuXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3N0cmVhbS1tZXNzYWdlLWxpc3QnLFxuICB0ZW1wbGF0ZVVybDogJy4vbWVzc2FnZS1saXN0LmNvbXBvbmVudC5odG1sJyxcbiAgc3R5bGVzOiBbXSxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG59KVxuZXhwb3J0IGNsYXNzIE1lc3NhZ2VMaXN0Q29tcG9uZW50XG4gIGltcGxlbWVudHMgQWZ0ZXJWaWV3Q2hlY2tlZCwgT25DaGFuZ2VzLCBPbkluaXQsIE9uRGVzdHJveSwgQWZ0ZXJWaWV3SW5pdFxue1xuICAvKipcbiAgICogRGV0ZXJtaW5lcyBpZiB0aGUgbWVzc2FnZSBsaXN0IHNob3VsZCBkaXNwbGF5IGNoYW5uZWwgbWVzc2FnZXMgb3IgW3RocmVhZCBtZXNzYWdlc10oL2NoYXQvZG9jcy9qYXZhc2NyaXB0L3RocmVhZHMvKS5cbiAgICovXG4gIEBJbnB1dCgpIG1vZGU6ICdtYWluJyB8ICd0aHJlYWQnID0gJ21haW4nO1xuICAvKipcbiAgICogVGhlIGRpcmVjdGlvbiBvZiB0aGUgbWVzc2FnZXMgaW4gdGhlIGxpc3QsIGBib3R0b20tdG8tdG9wYCBtZWFucyBuZXdlc3QgbWVzc2FnZSBpcyBhdCB0aGUgYm90dG9tIG9mIHRoZSBtZXNzYWdlIGxpc3QgYW5kIHVzZXJzIHNjcm9sbCB1cHdhcmRzIHRvIGxvYWQgb2xkZXIgbWVzc2FnZXNcbiAgICovXG4gIEBJbnB1dCgpIGRpcmVjdGlvbjogJ2JvdHRvbS10by10b3AnIHwgJ3RvcC10by1ib3R0b20nID0gJ2JvdHRvbS10by10b3AnO1xuICAvKipcbiAgICogWW91IGNhbiBoaWRlIHRoZSBcImp1bXAgdG8gbGF0ZXN0XCIgYnV0dG9uIHdoaWxlIHNjcm9sbGluZy4gQSBwb3RlbnRpYWwgdXNlLWNhc2UgZm9yIHRoaXMgaW5wdXQgd291bGQgYmUgdG8gW3dvcmthcm91bmQgYSBrbm93biBpc3N1ZSBvbiBpT1MgU2FmYXIgd2Vidmlld10oaHR0cHM6Ly9naXRodWIuY29tL0dldFN0cmVhbS9zdHJlYW0tY2hhdC1hbmd1bGFyL2lzc3Vlcy80MTgpXG4gICAqXG4gICAqL1xuICBASW5wdXQoKSBoaWRlSnVtcFRvTGF0ZXN0QnV0dG9uRHVyaW5nU2Nyb2xsID0gZmFsc2U7XG4gIC8qKlxuICAgKiBJZiBgdHJ1ZWAgZGF0ZSBzZXBhcmF0b3JzIHdpbGwgYmUgZGlzcGxheWVkXG4gICAqL1xuICBASW5wdXQoKSBkaXNwbGF5RGF0ZVNlcGFyYXRvciA9IHRydWU7XG4gIC8qKlxuICAgKiBJZiBgdHJ1ZWAgdW5yZWFkIGluZGljYXRvciB3aWxsIGJlIGRpc3BsYXllZFxuICAgKi9cbiAgQElucHV0KCkgZGlzcGxheVVucmVhZFNlcGFyYXRvciA9IHRydWU7XG4gIC8qKlxuICAgKiBJZiBkYXRlIHNlcGFyYXRvcnMgYXJlIGRpc3BsYXllZCwgeW91IGNhbiBzZXQgdGhlIGhvcml6b250YWwgcG9zaXRpb24gb2YgdGhlIGRhdGUgdGV4dC5cbiAgICovXG4gIEBJbnB1dCgpIGRhdGVTZXBhcmF0b3JUZXh0UG9zOiAnY2VudGVyJyB8ICdyaWdodCcgfCAnbGVmdCcgPSAnY2VudGVyJztcbiAgLyoqXG4gICAqIGBsYXN0LW1lc3NhZ2VgIG9wdGlvbiB3aWxsIG9wZW4gdGhlIG1lc3NhZ2UgbGlzdCBhdCB0aGUgbGFzdCBtZXNzYWdlLCBgbGFzdC1yZWFkLW1lc3NhZ2VgIHdpbGwgb3BlbiB0aGUgbGlzdCBhdCB0aGUgbGFzdCB1bnJlYWQgbWVzc2FnZS4gVGhpcyBvcHRpb24gb25seSB3b3JrcyBpZiBtb2RlIGlzIGBtYWluYC5cbiAgICovXG4gIEBJbnB1dCgpIG9wZW5NZXNzYWdlTGlzdEF0OiAnbGFzdC1tZXNzYWdlJyB8ICdsYXN0LXJlYWQtbWVzc2FnZScgPVxuICAgICdsYXN0LW1lc3NhZ2UnO1xuICAvKipcbiAgICogSWYgdGhlIHVzZXIgaGFzIHVucmVhZCBtZXNzYWdlcyB3aGVuIHRoZXkgb3BlbiB0aGUgY2hhbm5lbCB0aGUgVUkgc2hvd3MgdGhlIHVucmVhZCBpbmRpY2F0b3IgLyBub3RpZmljYXRpb24gd2hpY2ggZmVhdHVyZXMgdGhlIHVucmVhZCBjb3VudCBieSBkZWZhdWx0LiBUaGlzIGNvdW50IHdpbGwgYmUgaW5jcmVhc2VkIGV2ZXJ5IHRpbWUgYSB1c2VyIHJlY2VpdmVzIGEgbmV3IG1lc3NhZ2UuIElmIHlvdSBkb24ndCB3YW50IHRvIHNob3cgdGhlIHVucmVhZCBjb3VudCwgeW91IGNhbiB0dXJuIHRoYXQgb2ZmLlxuICAgKlxuICAgKiBUaGlzIGlzIG9ubHkgYXBwbGljYWJsZSBmb3IgYG1haW5gIG1vZGUsIGFzIHRocmVhZHMgZG9lc24ndCBoYXZlIHJlYWQgaW5mcm9tYXRpb24uXG4gICAqL1xuICBASW5wdXQoKSBoaWRlVW5yZWFkQ291bnRGb3JOb3RpZmljYXRpb25BbmRJbmRpY2F0b3IgPSBmYWxzZTtcbiAgLyoqXG4gICAqIFlvdSBjYW4gdHVybiBvbiBhbmQgb2ZmIHRoZSBsb2FkaW5nIGluZGljYXRvciB0aGF0IHNpZ25hbHMgdG8gdXNlcnMgdGhhdCBtb3JlIG1lc3NhZ2VzIGFyZSBiZWluZyBsb2FkZWQgdG8gdGhlIG1lc3NhZ2UgbGlzdFxuICAgKi9cbiAgQElucHV0KCkgZGlzcGxheUxvYWRpbmdJbmRpY2F0b3IgPSB0cnVlO1xuICB0eXBpbmdJbmRpY2F0b3JUZW1wbGF0ZTogVGVtcGxhdGVSZWY8VHlwaW5nSW5kaWNhdG9yQ29udGV4dD4gfCB1bmRlZmluZWQ7XG4gIG1lc3NhZ2VUZW1wbGF0ZTogVGVtcGxhdGVSZWY8TWVzc2FnZUNvbnRleHQ+IHwgdW5kZWZpbmVkO1xuICBjdXN0b21EYXRlU2VwYXJhdG9yVGVtcGxhdGU6IFRlbXBsYXRlUmVmPERhdGVTZXBhcmF0b3JDb250ZXh0PiB8IHVuZGVmaW5lZDtcbiAgY3VzdG9tbmV3TWVzc2FnZXNJbmRpY2F0b3JUZW1wbGF0ZTpcbiAgICB8IFRlbXBsYXRlUmVmPFVucmVhZE1lc3NhZ2VzSW5kaWNhdG9yQ29udGV4dD5cbiAgICB8IHVuZGVmaW5lZDtcbiAgY3VzdG9tbmV3TWVzc2FnZXNOb3RpZmljYXRpb25UZW1wbGF0ZTpcbiAgICB8IFRlbXBsYXRlUmVmPFVucmVhZE1lc3NhZ2VzTm90aWZpY2F0aW9uQ29udGV4dD5cbiAgICB8IHVuZGVmaW5lZDtcbiAgZW1wdHlNYWluTWVzc2FnZUxpc3RUZW1wbGF0ZTogVGVtcGxhdGVSZWY8dm9pZD4gfCBudWxsID0gbnVsbDtcbiAgZW1wdHlUaHJlYWRNZXNzYWdlTGlzdFRlbXBsYXRlOiBUZW1wbGF0ZVJlZjx2b2lkPiB8IG51bGwgPSBudWxsO1xuICBtZXNzYWdlcyQhOiBPYnNlcnZhYmxlPFN0cmVhbU1lc3NhZ2VbXT47XG4gIGVuYWJsZWRNZXNzYWdlQWN0aW9uczogc3RyaW5nW10gPSBbXTtcbiAgaXNFbXB0eSA9IHRydWU7XG4gIG5ld01lc3NhZ2VDb3VudFdoaWxlQmVpbmdTY3JvbGxlZCA9IDA7XG4gIGlzVXNlclNjcm9sbGVkOiBib29sZWFuIHwgdW5kZWZpbmVkO1xuICBncm91cFN0eWxlczogR3JvdXBTdHlsZVtdID0gW107XG4gIGlzTmV4dE1lc3NhZ2VPblNlcGFyYXRlRGF0ZTogYm9vbGVhbltdID0gW107XG4gIGxhc3RTZW50TWVzc2FnZUlkOiBzdHJpbmcgfCB1bmRlZmluZWQ7XG4gIHBhcmVudE1lc3NhZ2U6IFN0cmVhbU1lc3NhZ2UgfCB1bmRlZmluZWQ7XG4gIGhpZ2hsaWdodGVkTWVzc2FnZUlkOiBzdHJpbmcgfCB1bmRlZmluZWQ7XG4gIGxvYWRpbmdTdGF0ZTogJ2lkbGUnIHwgJ2xvYWRpbmctdG9wJyB8ICdsb2FkaW5nLWJvdHRvbScgPSAnaWRsZSc7XG4gIHNjcm9sbEVuZFRpbWVvdXQ/OiBSZXR1cm5UeXBlPHR5cGVvZiBzZXRUaW1lb3V0PjtcbiAgbGFzdFJlYWRNZXNzYWdlSWQ/OiBzdHJpbmc7XG4gIGlzVW5yZWFkTm90aWZpY2F0aW9uVmlzaWJsZSA9IHRydWU7XG4gIGZpcnN0VW5yZWFkTWVzc2FnZUlkPzogc3RyaW5nO1xuICB1bnJlYWRDb3VudD86IG51bWJlcjtcbiAgaXNKdW1waW5nVG9MYXRlc3RVbnJlYWRNZXNzYWdlID0gZmFsc2U7XG4gIGlzSnVtcFRvTGF0ZXN0QnV0dG9uVmlzaWJsZSA9IHRydWU7XG4gIGlzSnVtcGluZ1RvTWVzc2FnZSA9IGZhbHNlO1xuICBzY3JvbGwkID0gbmV3IFN1YmplY3Q8dm9pZD4oKTtcbiAgQFZpZXdDaGlsZCgnc2Nyb2xsQ29udGFpbmVyJylcbiAgcHJpdmF0ZSBzY3JvbGxDb250YWluZXIhOiBFbGVtZW50UmVmPEhUTUxFbGVtZW50PjtcbiAgQFZpZXdDaGlsZCgncGFyZW50TWVzc2FnZUVsZW1lbnQnKVxuICBwcml2YXRlIHBhcmVudE1lc3NhZ2VFbGVtZW50ITogRWxlbWVudFJlZjxIVE1MRWxlbWVudD47XG4gIHByaXZhdGUgaXNOZXdNZXNzYWdlU2VudEJ5VXNlcjogYm9vbGVhbiA9IGZhbHNlO1xuICBwcml2YXRlIHN1YnNjcmlwdGlvbnM6IFN1YnNjcmlwdGlvbltdID0gW107XG4gIHByaXZhdGUgbmV3TWVzc2FnZVN1YnNjcmlwdGlvbjogeyB1bnN1YnNjcmliZTogKCkgPT4gdm9pZCB9IHwgdW5kZWZpbmVkO1xuICBwcml2YXRlIHVzZXJzVHlwaW5nSW5DaGFubmVsJCE6IE9ic2VydmFibGU8XG4gICAgVXNlclJlc3BvbnNlPERlZmF1bHRTdHJlYW1DaGF0R2VuZXJpY3M+W11cbiAgPjtcbiAgcHJpdmF0ZSB1c2Vyc1R5cGluZ0luVGhyZWFkJCE6IE9ic2VydmFibGU8XG4gICAgVXNlclJlc3BvbnNlPERlZmF1bHRTdHJlYW1DaGF0R2VuZXJpY3M+W11cbiAgPjtcbiAgcHJpdmF0ZSBpc0xhdGVzdE1lc3NhZ2VJbkxpc3QgPSB0cnVlO1xuICBwcml2YXRlIGNoYW5uZWxJZD86IHN0cmluZztcbiAgcHJpdmF0ZSBwYXJzZWREYXRlcyA9IG5ldyBNYXA8RGF0ZSwgc3RyaW5nPigpO1xuICBwcml2YXRlIGlzVmlld0luaXRlZCA9IGZhbHNlO1xuICBwcml2YXRlIGNoZWNrSWZVbnJlYWROb3RpZmljYXRpb25Jc1Zpc2libGVUaW1lb3V0PzogUmV0dXJuVHlwZTxcbiAgICB0eXBlb2Ygc2V0VGltZW91dFxuICA+O1xuICBwcml2YXRlIGp1bXBUb01lc3NhZ2VUaW1lb3V0czogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0VGltZW91dD5bXSA9IFtdO1xuICBwcml2YXRlIGp1bXBUb0xhdGVzdEJ1dHRvblZpc2liaWxpdHlUaW1lb3V0PzogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0VGltZW91dD47XG4gIHByaXZhdGUgZm9yY2VSZXBhaW50U3ViamVjdCA9IG5ldyBTdWJqZWN0PHZvaWQ+KCk7XG4gIHByaXZhdGUgbWVzc2FnZUlkVG9BbmNob3JUbz86IHN0cmluZztcbiAgcHJpdmF0ZSBhbmNob3JNZXNzYWdlVG9wT2Zmc2V0PzogbnVtYmVyO1xuICBwcml2YXRlIGlzU2FmYXJpID0gaXNTYWZhcmkoKTtcblxuICBASG9zdEJpbmRpbmcoJ2NsYXNzJylcbiAgcHJpdmF0ZSBnZXQgY2xhc3MoKSB7XG4gICAgcmV0dXJuIGBzdHItY2hhdC1hbmd1bGFyX19tYWluLXBhbmVsLWlubmVyIHN0ci1jaGF0LWFuZ3VsYXJfX21lc3NhZ2UtbGlzdC1ob3N0IHN0ci1jaGF0X19tYWluLXBhbmVsLWlubmVyICR7XG4gICAgICB0aGlzLmlzRW1wdHkgPyAnc3RyLWNoYXQtYW5ndWxhcl9fbWVzc2FnZS1saXN0LWhvc3QtLWVtcHR5JyA6ICcnXG4gICAgfWA7XG4gIH1cbiAgcHJpdmF0ZSB2aXJ0dWFsaXplZExpc3Q/OiBWaXJ0dWFsaXplZE1lc3NhZ2VMaXN0U2VydmljZTtcbiAgcHJpdmF0ZSBzY3JvbGxQb3NpdGlvbiQgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PFZpcnR1YWxpemVkTGlzdFNjcm9sbFBvc2l0aW9uPihcbiAgICAnYm90dG9tJyxcbiAgKTtcbiAgcHJpdmF0ZSBqdW1wVG9JdGVtU3Vic2NyaXB0aW9uPzogU3Vic2NyaXB0aW9uO1xuICBwcml2YXRlIHF1ZXJ5U3RhdGVTdWJzY3JpcHRpb24/OiBTdWJzY3JpcHRpb247XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSBjaGFubmVsU2VydmljZTogQ2hhbm5lbFNlcnZpY2UsXG4gICAgcHJpdmF0ZSBjaGF0Q2xpZW50U2VydmljZTogQ2hhdENsaWVudFNlcnZpY2UsXG4gICAgcHJpdmF0ZSBjdXN0b21UZW1wbGF0ZXNTZXJ2aWNlOiBDdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLFxuICAgIHByaXZhdGUgZGF0ZVBhcnNlcjogRGF0ZVBhcnNlclNlcnZpY2UsXG4gICAgcHJpdmF0ZSBuZ1pvbmU6IE5nWm9uZSxcbiAgICBwcml2YXRlIGNkUmVmOiBDaGFuZ2VEZXRlY3RvclJlZixcbiAgKSB7XG4gICAgdGhpcy51c2Vyc1R5cGluZ0luQ2hhbm5lbCQgPSB0aGlzLmNoYW5uZWxTZXJ2aWNlLnVzZXJzVHlwaW5nSW5DaGFubmVsJDtcbiAgICB0aGlzLnVzZXJzVHlwaW5nSW5UaHJlYWQkID0gdGhpcy5jaGFubmVsU2VydmljZS51c2Vyc1R5cGluZ0luVGhyZWFkJDtcbiAgfVxuXG4gIG1lc3NhZ2VOb3RpZmljYXRpb25KdW1wQ2xpY2tlZCA9ICgpID0+IHtcbiAgICB0aGlzLmp1bXBUb0ZpcnN0VW5yZWFkTWVzc2FnZSgpO1xuICAgIHRoaXMuaXNVbnJlYWROb3RpZmljYXRpb25WaXNpYmxlID0gZmFsc2U7XG4gIH07XG5cbiAgbWVzc2FnZU5vdGlmaWNhdGlvbkRpc21pc3NDbGlja2VkID0gKCkgPT4ge1xuICAgIHRoaXMuaXNVbnJlYWROb3RpZmljYXRpb25WaXNpYmxlID0gZmFsc2U7XG4gIH07XG5cbiAgbmdPbkNoYW5nZXMoY2hhbmdlczogU2ltcGxlQ2hhbmdlcyk6IHZvaWQge1xuICAgIGlmIChjaGFuZ2VzLm1vZGUgfHwgY2hhbmdlcy5kaXJlY3Rpb24pIHtcbiAgICAgIHRoaXMucmVzZXRTY3JvbGxTdGF0ZSgpO1xuICAgICAgdGhpcy5zZXRNZXNzYWdlcyQoKTtcbiAgICB9XG4gICAgaWYgKGNoYW5nZXMuZGlyZWN0aW9uKSB7XG4gICAgICBpZiAodGhpcy5zY3JvbGxDb250YWluZXI/Lm5hdGl2ZUVsZW1lbnQpIHtcbiAgICAgICAgdGhpcy5qdW1wVG9MYXRlc3RNZXNzYWdlKCk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgbmdPbkluaXQoKTogdm9pZCB7XG4gICAgdGhpcy5zdWJzY3JpcHRpb25zLnB1c2goXG4gICAgICB0aGlzLmZvcmNlUmVwYWludFN1YmplY3QucGlwZSh0aHJvdHRsZVRpbWUoMTAwMCkpLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgIHRoaXMuZm9yY2VSZXBhaW50KCk7XG4gICAgICB9KSxcbiAgICApO1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgdGhpcy5jaGFubmVsU2VydmljZS5hY3RpdmVDaGFubmVsJC5zdWJzY3JpYmUoKGNoYW5uZWwpID0+IHtcbiAgICAgICAgbGV0IHdhc0NoYW5uZWxTd2l0Y2ggPSBmYWxzZTtcbiAgICAgICAgaWYgKHRoaXMuY2hhbm5lbElkICE9PSBjaGFubmVsPy5pZCkge1xuICAgICAgICAgIHdhc0NoYW5uZWxTd2l0Y2ggPSB0cnVlO1xuICAgICAgICAgIGlmICh0aGlzLmNoZWNrSWZVbnJlYWROb3RpZmljYXRpb25Jc1Zpc2libGVUaW1lb3V0KSB7XG4gICAgICAgICAgICBjbGVhclRpbWVvdXQodGhpcy5jaGVja0lmVW5yZWFkTm90aWZpY2F0aW9uSXNWaXNpYmxlVGltZW91dCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHRoaXMuanVtcFRvTWVzc2FnZVRpbWVvdXRzLmZvckVhY2goKHRpbWVvdXQpID0+XG4gICAgICAgICAgICBjbGVhclRpbWVvdXQodGltZW91dCksXG4gICAgICAgICAgKTtcbiAgICAgICAgICB0aGlzLmp1bXBUb01lc3NhZ2VUaW1lb3V0cyA9IFtdO1xuICAgICAgICAgIHRoaXMuaGlnaGxpZ2h0ZWRNZXNzYWdlSWQgPSB1bmRlZmluZWQ7XG4gICAgICAgICAgdGhpcy5pc1VucmVhZE5vdGlmaWNhdGlvblZpc2libGUgPSBmYWxzZTtcbiAgICAgICAgICB0aGlzLnBhcnNlZERhdGVzID0gbmV3IE1hcCgpO1xuICAgICAgICAgIHRoaXMucmVzZXRTY3JvbGxTdGF0ZSgpO1xuICAgICAgICAgIHRoaXMuc2V0TWVzc2FnZXMkKCk7XG4gICAgICAgICAgdGhpcy5jaGFubmVsSWQgPSBjaGFubmVsPy5pZDtcbiAgICAgICAgICBpZiAodGhpcy5pc1ZpZXdJbml0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICBpZiAodGhpcy5tb2RlID09PSAnbWFpbicpIHtcbiAgICAgICAgICBjb25zdCBsYXN0UmVhZE1lc3NhZ2VJZCA9XG4gICAgICAgICAgICB0aGlzLmNoYW5uZWxTZXJ2aWNlLmFjdGl2ZUNoYW5uZWxMYXN0UmVhZE1lc3NhZ2VJZDtcbiAgICAgICAgICBjb25zdCB1bnJlYWRDb3VudCA9IHRoaXMuY2hhbm5lbFNlcnZpY2UuYWN0aXZlQ2hhbm5lbFVucmVhZENvdW50O1xuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIGxhc3RSZWFkTWVzc2FnZUlkICE9PSB0aGlzLmxhc3RSZWFkTWVzc2FnZUlkIHx8XG4gICAgICAgICAgICB1bnJlYWRDb3VudCAhPT0gdGhpcy51bnJlYWRDb3VudFxuICAgICAgICAgICkge1xuICAgICAgICAgICAgdGhpcy5sYXN0UmVhZE1lc3NhZ2VJZCA9IGxhc3RSZWFkTWVzc2FnZUlkO1xuICAgICAgICAgICAgdGhpcy51bnJlYWRDb3VudCA9IHVucmVhZENvdW50IHx8IDA7XG4gICAgICAgICAgICBpZiAod2FzQ2hhbm5lbFN3aXRjaCAmJiB0aGlzLmxhc3RSZWFkTWVzc2FnZUlkKSB7XG4gICAgICAgICAgICAgIC8vIERlbGF5IGp1bXBpbmcgdG8gbGFzdCByZWFkIG1lc3NhZ2UgaW4gY2FzZSB3ZSBuZWVkIHRvIGdpdmUgcHJlY2VkZW5jZSB0byBjaGFubmVsU2VydmljZS5qdW1wVG9NZXNzYWdlXG4gICAgICAgICAgICAgIGlmICh0aGlzLm9wZW5NZXNzYWdlTGlzdEF0ID09PSAnbGFzdC1yZWFkLW1lc3NhZ2UnKSB7XG4gICAgICAgICAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAvLyBEb24ndCBqdW1wIGlmIGEganVtcCB0byBhIG1lc3NhZ2Ugd2FzIGFscmVhZHkgc3RhcnRlZCAodXNpbmcgY2hhbm5lbFNlcnZpY2UuanVtcFRvTWVzc2FnZSlcbiAgICAgICAgICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICAgICAgICAgIXRoaXMuaXNKdW1waW5nVG9NZXNzYWdlICYmXG4gICAgICAgICAgICAgICAgICAgICF0aGlzLmNoYW5uZWxTZXJ2aWNlLmlzTWVzc2FnZUxvYWRpbmdJblByb2dyZXNzXG4gICAgICAgICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5qdW1wVG9GaXJzdFVucmVhZE1lc3NhZ2UoKTtcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9LCAwKTtcbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAvLyBXYWl0IHRpbGwgbWVzc2FnZXMgYW5kIHRoZSB1bnJlYWQgYmFubmVyIGlzIHJlbmRlcmVkXG4gICAgICAgICAgICAgICAgLy8gSWYgdW5yZWFkIGJhbm5lciBpc24ndCB2aXNpYmxlIG9uIHRoZSBzY3JlZW4sIHdlIGRpc3BsYXkgdGhlIHVucmVhZCBub3RpZmljYWlvblxuICAgICAgICAgICAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICAgICAgICAgICAgY29uc3QgYmFubmVyRWxlbWVudCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKFxuICAgICAgICAgICAgICAgICAgICAnc3RyZWFtLWNoYXQtbmV3LW1lc3NhZ2UtaW5kaWNhdG9yJyxcbiAgICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgICAgICAgICFiYW5uZXJFbGVtZW50IHx8XG4gICAgICAgICAgICAgICAgICAgIGJhbm5lckVsZW1lbnQ/Lm9mZnNldFRvcCA8XG4gICAgICAgICAgICAgICAgICAgICAgdGhpcy5zY3JvbGxDb250YWluZXI/Lm5hdGl2ZUVsZW1lbnQ/LnNjcm9sbEhlaWdodCAtXG4gICAgICAgICAgICAgICAgICAgICAgICB0aGlzLnNjcm9sbENvbnRhaW5lcj8ubmF0aXZlRWxlbWVudD8uY2xpZW50SGVpZ2h0XG4gICAgICAgICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5pc1VucmVhZE5vdGlmaWNhdGlvblZpc2libGUgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5pc1ZpZXdJbml0ZWQpIHtcbiAgICAgICAgICAgICAgICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH0sIDEwMCk7XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGlmICh0aGlzLmlzVmlld0luaXRlZCkge1xuICAgICAgICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0gZWxzZSBpZiAodGhpcy5sYXN0UmVhZE1lc3NhZ2VJZCkge1xuICAgICAgICAgIHRoaXMubGFzdFJlYWRNZXNzYWdlSWQgPSB1bmRlZmluZWQ7XG4gICAgICAgICAgdGhpcy51bnJlYWRDb3VudCA9IDA7XG4gICAgICAgICAgaWYgKHRoaXMuaXNWaWV3SW5pdGVkKSB7XG4gICAgICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgY29uc3QgY2FwYWJpbGl0ZXMgPSBjaGFubmVsPy5kYXRhPy5vd25fY2FwYWJpbGl0aWVzIGFzIHN0cmluZ1tdO1xuICAgICAgICBjb25zdCBjYXBhYmlsaXRlc1N0cmluZyA9IFsuLi4oY2FwYWJpbGl0ZXMgfHwgW10pXS5zb3J0KCkuam9pbignJyk7XG4gICAgICAgIGNvbnN0IGVuYWJsZWRBY3Rpb25zU3RyaW5nID0gWy4uLih0aGlzLmVuYWJsZWRNZXNzYWdlQWN0aW9ucyB8fCBbXSldXG4gICAgICAgICAgLnNvcnQoKVxuICAgICAgICAgIC5qb2luKCcnKTtcbiAgICAgICAgaWYgKGNhcGFiaWxpdGVzU3RyaW5nICE9PSBlbmFibGVkQWN0aW9uc1N0cmluZykge1xuICAgICAgICAgIHRoaXMuZW5hYmxlZE1lc3NhZ2VBY3Rpb25zID0gY2FwYWJpbGl0ZXMgfHwgW107XG4gICAgICAgICAgaWYgKHRoaXMuaXNWaWV3SW5pdGVkKSB7XG4gICAgICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5uZXdNZXNzYWdlU3Vic2NyaXB0aW9uPy51bnN1YnNjcmliZSgpO1xuICAgICAgICBpZiAoY2hhbm5lbCkge1xuICAgICAgICAgIHRoaXMubmV3TWVzc2FnZVN1YnNjcmlwdGlvbiA9IGNoYW5uZWwub24oJ21lc3NhZ2UubmV3JywgKGV2ZW50KSA9PiB7XG4gICAgICAgICAgICBpZiAoIWV2ZW50Lm1lc3NhZ2UpIHtcbiAgICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgdGhpcy5uZXdNZXNzYWdlUmVjZWl2ZWQoZXZlbnQubWVzc2FnZSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSk7XG4gICAgICAgIH1cbiAgICAgIH0pLFxuICAgICk7XG4gICAgdGhpcy5zdWJzY3JpcHRpb25zLnB1c2goXG4gICAgICB0aGlzLmNoYW5uZWxTZXJ2aWNlLmFjdGl2ZVBhcmVudE1lc3NhZ2UkLnN1YnNjcmliZSgobWVzc2FnZSkgPT4ge1xuICAgICAgICBpZiAoIW1lc3NhZ2UgJiYgdGhpcy5wYXJlbnRNZXNzYWdlICYmIHRoaXMubW9kZSA9PT0gJ3RocmVhZCcpIHtcbiAgICAgICAgICB0aGlzLnJlc2V0U2Nyb2xsU3RhdGUoKTtcbiAgICAgICAgfVxuICAgICAgICBpZiAoXG4gICAgICAgICAgbWVzc2FnZSAmJlxuICAgICAgICAgIHRoaXMucGFyZW50TWVzc2FnZSAmJlxuICAgICAgICAgIG1lc3NhZ2UuaWQgIT09IHRoaXMucGFyZW50TWVzc2FnZS5pZCAmJlxuICAgICAgICAgIHRoaXMubW9kZSA9PT0gJ3RocmVhZCdcbiAgICAgICAgKSB7XG4gICAgICAgICAgdGhpcy5yZXNldFNjcm9sbFN0YXRlKCk7XG4gICAgICAgICAgdGhpcy5zZXRNZXNzYWdlcyQoKTtcbiAgICAgICAgfVxuICAgICAgICBpZiAodGhpcy5wYXJlbnRNZXNzYWdlID09PSBtZXNzYWdlKSB7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMucGFyZW50TWVzc2FnZSA9IG1lc3NhZ2U7XG4gICAgICAgIGlmICh0aGlzLmlzVmlld0luaXRlZCkge1xuICAgICAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgICB9XG4gICAgICB9KSxcbiAgICApO1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgdGhpcy5jdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLm1lc3NhZ2VUZW1wbGF0ZSQuc3Vic2NyaWJlKCh0ZW1wbGF0ZSkgPT4ge1xuICAgICAgICBpZiAodGhpcy5tZXNzYWdlVGVtcGxhdGUgPT09IHRlbXBsYXRlKSB7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMubWVzc2FnZVRlbXBsYXRlID0gdGVtcGxhdGU7XG4gICAgICAgIGlmICh0aGlzLmlzVmlld0luaXRlZCkge1xuICAgICAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgICB9XG4gICAgICB9KSxcbiAgICApO1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgdGhpcy5jdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLmRhdGVTZXBhcmF0b3JUZW1wbGF0ZSQuc3Vic2NyaWJlKFxuICAgICAgICAodGVtcGxhdGUpID0+IHtcbiAgICAgICAgICBpZiAodGhpcy5jdXN0b21EYXRlU2VwYXJhdG9yVGVtcGxhdGUgPT09IHRlbXBsYXRlKSB7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgfVxuICAgICAgICAgIHRoaXMuY3VzdG9tRGF0ZVNlcGFyYXRvclRlbXBsYXRlID0gdGVtcGxhdGU7XG4gICAgICAgICAgaWYgKHRoaXMuaXNWaWV3SW5pdGVkKSB7XG4gICAgICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgICB9XG4gICAgICAgIH0sXG4gICAgICApLFxuICAgICk7XG4gICAgdGhpcy5zdWJzY3JpcHRpb25zLnB1c2goXG4gICAgICB0aGlzLmN1c3RvbVRlbXBsYXRlc1NlcnZpY2UubmV3TWVzc2FnZXNJbmRpY2F0b3JUZW1wbGF0ZSQuc3Vic2NyaWJlKFxuICAgICAgICAodGVtcGxhdGUpID0+IHtcbiAgICAgICAgICBpZiAodGhpcy5jdXN0b21uZXdNZXNzYWdlc0luZGljYXRvclRlbXBsYXRlID09PSB0ZW1wbGF0ZSkge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgIH1cbiAgICAgICAgICB0aGlzLmN1c3RvbW5ld01lc3NhZ2VzSW5kaWNhdG9yVGVtcGxhdGUgPSB0ZW1wbGF0ZTtcbiAgICAgICAgICBpZiAodGhpcy5pc1ZpZXdJbml0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSxcbiAgICAgICksXG4gICAgKTtcbiAgICB0aGlzLnN1YnNjcmlwdGlvbnMucHVzaChcbiAgICAgIHRoaXMuY3VzdG9tVGVtcGxhdGVzU2VydmljZS5uZXdNZXNzYWdlc05vdGlmaWNhdGlvblRlbXBsYXRlJC5zdWJzY3JpYmUoXG4gICAgICAgICh0ZW1wbGF0ZSkgPT4ge1xuICAgICAgICAgIGlmICh0aGlzLmN1c3RvbW5ld01lc3NhZ2VzTm90aWZpY2F0aW9uVGVtcGxhdGUgPT09IHRlbXBsYXRlKSB7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgfVxuICAgICAgICAgIHRoaXMuY3VzdG9tbmV3TWVzc2FnZXNOb3RpZmljYXRpb25UZW1wbGF0ZSA9IHRlbXBsYXRlO1xuICAgICAgICAgIGlmICh0aGlzLmlzVmlld0luaXRlZCkge1xuICAgICAgICAgICAgdGhpcy5jZFJlZi5kZXRlY3RDaGFuZ2VzKCk7XG4gICAgICAgICAgfVxuICAgICAgICB9LFxuICAgICAgKSxcbiAgICApO1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgdGhpcy5jdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLnR5cGluZ0luZGljYXRvclRlbXBsYXRlJC5zdWJzY3JpYmUoXG4gICAgICAgICh0ZW1wbGF0ZSkgPT4ge1xuICAgICAgICAgIGlmICh0aGlzLnR5cGluZ0luZGljYXRvclRlbXBsYXRlID09PSB0ZW1wbGF0ZSkge1xuICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgIH1cbiAgICAgICAgICB0aGlzLnR5cGluZ0luZGljYXRvclRlbXBsYXRlID0gdGVtcGxhdGU7XG4gICAgICAgICAgaWYgKHRoaXMuaXNWaWV3SW5pdGVkKSB7XG4gICAgICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgICB9XG4gICAgICAgIH0sXG4gICAgICApLFxuICAgICk7XG4gICAgdGhpcy5zdWJzY3JpcHRpb25zLnB1c2goXG4gICAgICB0aGlzLmN1c3RvbVRlbXBsYXRlc1NlcnZpY2UuZW1wdHlNYWluTWVzc2FnZUxpc3RQbGFjZWhvbGRlciQuc3Vic2NyaWJlKFxuICAgICAgICAodGVtcGxhdGUpID0+IHtcbiAgICAgICAgICBjb25zdCBpc0NoYW5nZWQgPSB0aGlzLmVtcHR5TWFpbk1lc3NhZ2VMaXN0VGVtcGxhdGUgIT09IHRlbXBsYXRlO1xuICAgICAgICAgIHRoaXMuZW1wdHlNYWluTWVzc2FnZUxpc3RUZW1wbGF0ZSA9IHRlbXBsYXRlIHx8IG51bGw7XG4gICAgICAgICAgaWYgKGlzQ2hhbmdlZCAmJiB0aGlzLmlzVmlld0luaXRlZCkge1xuICAgICAgICAgICAgdGhpcy5jZFJlZi5kZXRlY3RDaGFuZ2VzKCk7XG4gICAgICAgICAgfVxuICAgICAgICB9LFxuICAgICAgKSxcbiAgICApO1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5wdXNoKFxuICAgICAgdGhpcy5jdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLmVtcHR5VGhyZWFkTWVzc2FnZUxpc3RQbGFjZWhvbGRlciQuc3Vic2NyaWJlKFxuICAgICAgICAodGVtcGxhdGUpID0+IHtcbiAgICAgICAgICBjb25zdCBpc0NoYW5nZWQgPSB0aGlzLmVtcHR5VGhyZWFkTWVzc2FnZUxpc3RUZW1wbGF0ZSAhPT0gdGVtcGxhdGU7XG4gICAgICAgICAgdGhpcy5lbXB0eVRocmVhZE1lc3NhZ2VMaXN0VGVtcGxhdGUgPSB0ZW1wbGF0ZSB8fCBudWxsO1xuICAgICAgICAgIGlmIChpc0NoYW5nZWQgJiYgdGhpcy5pc1ZpZXdJbml0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSxcbiAgICAgICksXG4gICAgKTtcbiAgICB0aGlzLnNldE1lc3NhZ2VzJCgpO1xuICB9XG5cbiAgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIHRoaXMuaXNWaWV3SW5pdGVkID0gdHJ1ZTtcbiAgICB0aGlzLm5nWm9uZS5ydW5PdXRzaWRlQW5ndWxhcigoKSA9PiB7XG4gICAgICB0aGlzLnNjcm9sbENvbnRhaW5lcj8ubmF0aXZlRWxlbWVudD8uYWRkRXZlbnRMaXN0ZW5lcignc2Nyb2xsJywgKCkgPT5cbiAgICAgICAgdGhpcy5zY3JvbGxlZCgpLFxuICAgICAgKTtcbiAgICB9KTtcbiAgfVxuXG4gIG5nQWZ0ZXJWaWV3Q2hlY2tlZCgpIHtcbiAgICBpZiAodGhpcy5pc0p1bXBpbmdUb01lc3NhZ2UpIHtcbiAgICAgIHRoaXMuaXNOZXdNZXNzYWdlU2VudEJ5VXNlciA9IGZhbHNlO1xuICAgICAgdGhpcy5tZXNzYWdlSWRUb0FuY2hvclRvID0gdW5kZWZpbmVkO1xuICAgICAgdGhpcy5hbmNob3JNZXNzYWdlVG9wT2Zmc2V0ID0gdW5kZWZpbmVkO1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBpZiAodGhpcy5tZXNzYWdlSWRUb0FuY2hvclRvICYmIHRoaXMubG9hZGluZ1N0YXRlID09PSAnaWRsZScpIHtcbiAgICAgIHRoaXMucHJlc2VydmVTY3JvbGxiYXJQb3NpdGlvbigpO1xuICAgIH0gZWxzZSBpZiAoXG4gICAgICAoIXRoaXMuaXNVc2VyU2Nyb2xsZWQgJiZcbiAgICAgICAgdGhpcy5zY3JvbGxDb250YWluZXIubmF0aXZlRWxlbWVudD8uc2Nyb2xsSGVpZ2h0ID5cbiAgICAgICAgICB0aGlzLnNjcm9sbENvbnRhaW5lcj8ubmF0aXZlRWxlbWVudC5jbGllbnRIZWlnaHQgJiZcbiAgICAgICAgdGhpcy5nZXRTY3JvbGxQb3NpdGlvbigpICE9PVxuICAgICAgICAgICh0aGlzLmRpcmVjdGlvbiA9PT0gJ2JvdHRvbS10by10b3AnID8gJ2JvdHRvbScgOiAndG9wJykpIHx8XG4gICAgICAodGhpcy5pc1VzZXJTY3JvbGxlZCAmJiB0aGlzLmlzTmV3TWVzc2FnZVNlbnRCeVVzZXIpXG4gICAgKSB7XG4gICAgICB0aGlzLmlzTmV3TWVzc2FnZVNlbnRCeVVzZXIgPSBmYWxzZTtcbiAgICAgIHRoaXMuanVtcFRvTGF0ZXN0TWVzc2FnZSgpO1xuICAgIH1cbiAgfVxuXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5mb3JFYWNoKChzKSA9PiBzLnVuc3Vic2NyaWJlKCkpO1xuICAgIHRoaXMubmV3TWVzc2FnZVN1YnNjcmlwdGlvbj8udW5zdWJzY3JpYmUoKTtcbiAgICBpZiAodGhpcy5zY3JvbGxFbmRUaW1lb3V0KSB7XG4gICAgICBjbGVhclRpbWVvdXQodGhpcy5zY3JvbGxFbmRUaW1lb3V0KTtcbiAgICB9XG4gICAgaWYgKHRoaXMuY2hlY2tJZlVucmVhZE5vdGlmaWNhdGlvbklzVmlzaWJsZVRpbWVvdXQpIHtcbiAgICAgIGNsZWFyVGltZW91dCh0aGlzLmNoZWNrSWZVbnJlYWROb3RpZmljYXRpb25Jc1Zpc2libGVUaW1lb3V0KTtcbiAgICB9XG4gICAgaWYgKHRoaXMuanVtcFRvTGF0ZXN0QnV0dG9uVmlzaWJpbGl0eVRpbWVvdXQpIHtcbiAgICAgIGNsZWFyVGltZW91dCh0aGlzLmp1bXBUb0xhdGVzdEJ1dHRvblZpc2liaWxpdHlUaW1lb3V0KTtcbiAgICB9XG4gICAgdGhpcy5qdW1wVG9NZXNzYWdlVGltZW91dHMuZm9yRWFjaCgodGltZW91dCkgPT4gY2xlYXJUaW1lb3V0KHRpbWVvdXQpKTtcbiAgICB0aGlzLmRpc3Bvc2VWaXJ0dWFsaXplZExpc3QoKTtcbiAgfVxuXG4gIHRyYWNrQnlVc2VySWQoXzogbnVtYmVyLCB1c2VyOiBVc2VyUmVzcG9uc2UpIHtcbiAgICByZXR1cm4gdXNlci5pZDtcbiAgfVxuXG4gIGp1bXBUb0xhdGVzdE1lc3NhZ2UoKSB7XG4gICAgaWYgKHRoaXMuaXNMYXRlc3RNZXNzYWdlSW5MaXN0KSB7XG4gICAgICB0aGlzLmRpcmVjdGlvbiA9PT0gJ2JvdHRvbS10by10b3AnXG4gICAgICAgID8gdGhpcy5zY3JvbGxUb0JvdHRvbSgpXG4gICAgICAgIDogdGhpcy5zY3JvbGxUb1RvcCgpO1xuICAgIH0gZWxzZSB7XG4gICAgICB2b2lkIHRoaXMuY2hhbm5lbFNlcnZpY2UuanVtcFRvTWVzc2FnZShcbiAgICAgICAgJ2xhdGVzdCcsXG4gICAgICAgIHRoaXMubW9kZSA9PT0gJ3RocmVhZCcgPyB0aGlzLnBhcmVudE1lc3NhZ2U/LmlkIDogdW5kZWZpbmVkLFxuICAgICAgKTtcbiAgICB9XG4gIH1cblxuICBzY3JvbGxUb0JvdHRvbSgpOiB2b2lkIHtcbiAgICB0aGlzLnNjcm9sbENvbnRhaW5lci5uYXRpdmVFbGVtZW50LnNjcm9sbFRvcCA9XG4gICAgICB0aGlzLnNjcm9sbENvbnRhaW5lci5uYXRpdmVFbGVtZW50LnNjcm9sbEhlaWdodCArIDAuMTtcbiAgICBpZiAodGhpcy5pc1NhZmFyaSkge1xuICAgICAgdGhpcy5mb3JjZVJlcGFpbnRTdWJqZWN0Lm5leHQoKTtcbiAgICB9XG4gIH1cblxuICBzY3JvbGxUb1RvcCgpIHtcbiAgICB0aGlzLnNjcm9sbENvbnRhaW5lci5uYXRpdmVFbGVtZW50LnNjcm9sbFRvcCA9IDA7XG4gIH1cblxuICBzY3JvbGxlZCgpIHtcbiAgICBpZiAoXG4gICAgICB0aGlzLnNjcm9sbENvbnRhaW5lci5uYXRpdmVFbGVtZW50LnNjcm9sbEhlaWdodCA9PT1cbiAgICAgIHRoaXMuc2Nyb2xsQ29udGFpbmVyLm5hdGl2ZUVsZW1lbnQuY2xpZW50SGVpZ2h0XG4gICAgKSB7XG4gICAgICBpZiAodGhpcy5pc0p1bXBUb0xhdGVzdEJ1dHRvblZpc2libGUpIHtcbiAgICAgICAgdGhpcy5pc0p1bXBUb0xhdGVzdEJ1dHRvblZpc2libGUgPSBmYWxzZTtcbiAgICAgICAgdGhpcy5uZXdNZXNzYWdlQ291bnRXaGlsZUJlaW5nU2Nyb2xsZWQgPSAwO1xuICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgIH1cbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKFxuICAgICAgdGhpcy5zY3JvbGxDb250YWluZXIubmF0aXZlRWxlbWVudC5zY3JvbGxIZWlnaHQgPlxuICAgICAgdGhpcy5zY3JvbGxDb250YWluZXIubmF0aXZlRWxlbWVudC5jbGllbnRIZWlnaHRcbiAgICApIHtcbiAgICAgIGlmICghdGhpcy5pc0p1bXBUb0xhdGVzdEJ1dHRvblZpc2libGUpIHtcbiAgICAgICAgdGhpcy5pc0p1bXBUb0xhdGVzdEJ1dHRvblZpc2libGUgPSB0cnVlO1xuICAgICAgfVxuICAgIH1cblxuICAgIHRoaXMuc2Nyb2xsJC5uZXh0KCk7XG5cbiAgICB0aGlzLmNoZWNrSWZVc2VyU2Nyb2xsZWQoKTtcblxuICAgIGlmICh0aGlzLmhpZGVKdW1wVG9MYXRlc3RCdXR0b25EdXJpbmdTY3JvbGwpIHtcbiAgICAgIGlmICh0aGlzLmlzSnVtcFRvTGF0ZXN0QnV0dG9uVmlzaWJsZSkge1xuICAgICAgICB0aGlzLmlzSnVtcFRvTGF0ZXN0QnV0dG9uVmlzaWJsZSA9IGZhbHNlO1xuICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgIH1cbiAgICAgIGlmICh0aGlzLmp1bXBUb0xhdGVzdEJ1dHRvblZpc2liaWxpdHlUaW1lb3V0KSB7XG4gICAgICAgIGNsZWFyVGltZW91dCh0aGlzLmp1bXBUb0xhdGVzdEJ1dHRvblZpc2liaWxpdHlUaW1lb3V0KTtcbiAgICAgIH1cbiAgICAgIHRoaXMuanVtcFRvTGF0ZXN0QnV0dG9uVmlzaWJpbGl0eVRpbWVvdXQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgaWYgKHRoaXMuaXNVc2VyU2Nyb2xsZWQpIHtcbiAgICAgICAgICB0aGlzLmlzSnVtcFRvTGF0ZXN0QnV0dG9uVmlzaWJsZSA9IHRydWU7XG4gICAgICAgICAgdGhpcy5qdW1wVG9MYXRlc3RCdXR0b25WaXNpYmlsaXR5VGltZW91dCA9IHVuZGVmaW5lZDtcbiAgICAgICAgICB0aGlzLmNkUmVmLmRldGVjdENoYW5nZXMoKTtcbiAgICAgICAgfVxuICAgICAgfSwgMTAwKTtcbiAgICB9XG4gIH1cblxuICBqdW1wVG9GaXJzdFVucmVhZE1lc3NhZ2UoKSB7XG4gICAgaWYgKCF0aGlzLmxhc3RSZWFkTWVzc2FnZUlkKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIHRoaXMuaXNKdW1waW5nVG9MYXRlc3RVbnJlYWRNZXNzYWdlID0gdHJ1ZTtcbiAgICB2b2lkIHRoaXMuY2hhbm5lbFNlcnZpY2UuanVtcFRvTWVzc2FnZSh0aGlzLmxhc3RSZWFkTWVzc2FnZUlkKTtcbiAgfVxuXG4gIGdldFR5cGluZ0luZGljYXRvckNvbnRleHQoKTogVHlwaW5nSW5kaWNhdG9yQ29udGV4dCB7XG4gICAgcmV0dXJuIHtcbiAgICAgIHVzZXJzVHlwaW5nJDogdGhpcy51c2Vyc1R5cGluZyQsXG4gICAgfTtcbiAgfVxuXG4gIGdldFR5cGluZ0luZGljYXRvclRleHQodXNlcnM6IFVzZXJSZXNwb25zZVtdKSB7XG4gICAgY29uc3QgdGV4dCA9IGxpc3RVc2Vycyh1c2Vycyk7XG5cbiAgICByZXR1cm4gdGV4dDtcbiAgfVxuXG4gIGlzU2VudEJ5Q3VycmVudFVzZXIobWVzc2FnZT86IFN0cmVhbU1lc3NhZ2UpIHtcbiAgICBpZiAoIW1lc3NhZ2UpIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIG1lc3NhZ2UudXNlcj8uaWQgPT09IHRoaXMuY2hhdENsaWVudFNlcnZpY2UuY2hhdENsaWVudC51c2VyPy5pZDtcbiAgfVxuXG4gIHBhcnNlRGF0ZShkYXRlOiBEYXRlKSB7XG4gICAgaWYgKHRoaXMucGFyc2VkRGF0ZXMuaGFzKGRhdGUpKSB7XG4gICAgICByZXR1cm4gdGhpcy5wYXJzZWREYXRlcy5nZXQoZGF0ZSk7XG4gICAgfVxuICAgIGNvbnN0IHBhcnNlZERhdGUgPSB0aGlzLmRhdGVQYXJzZXIucGFyc2VEYXRlKGRhdGUpO1xuICAgIHRoaXMucGFyc2VkRGF0ZXMuc2V0KGRhdGUsIHBhcnNlZERhdGUpO1xuICAgIHJldHVybiBwYXJzZWREYXRlO1xuICB9XG5cbiAgZ2V0IHJlcGx5Q291bnRQYXJhbSgpIHtcbiAgICByZXR1cm4geyByZXBseUNvdW50OiB0aGlzLnBhcmVudE1lc3NhZ2U/LnJlcGx5X2NvdW50IH07XG4gIH1cblxuICBnZXQgZW1wdHlMaXN0VGVtcGxhdGUoKSB7XG4gICAgcmV0dXJuIHRoaXMubW9kZSA9PT0gJ21haW4nXG4gICAgICA/IHRoaXMuZW1wdHlNYWluTWVzc2FnZUxpc3RUZW1wbGF0ZVxuICAgICAgOiB0aGlzLmVtcHR5VGhyZWFkTWVzc2FnZUxpc3RUZW1wbGF0ZTtcbiAgfVxuXG4gIHByaXZhdGUgY2hlY2tJZlVzZXJTY3JvbGxlZCgpIHtcbiAgICBsZXQgc2Nyb2xsUG9zaXRpb24gPSB0aGlzLmdldFNjcm9sbFBvc2l0aW9uKCk7XG5cbiAgICBjb25zdCBpc1VzZXJTY3JvbGxlZCA9XG4gICAgICAodGhpcy5kaXJlY3Rpb24gPT09ICdib3R0b20tdG8tdG9wJ1xuICAgICAgICA/IHNjcm9sbFBvc2l0aW9uICE9PSAnYm90dG9tJ1xuICAgICAgICA6IHNjcm9sbFBvc2l0aW9uICE9PSAndG9wJykgfHwgIXRoaXMuaXNMYXRlc3RNZXNzYWdlSW5MaXN0O1xuICAgIGlmICh0aGlzLmlzVXNlclNjcm9sbGVkICE9PSBpc1VzZXJTY3JvbGxlZCkge1xuICAgICAgdGhpcy5uZ1pvbmUucnVuKCgpID0+IHtcbiAgICAgICAgdGhpcy5pc1VzZXJTY3JvbGxlZCA9IGlzVXNlclNjcm9sbGVkO1xuICAgICAgICBpZiAoIXRoaXMuaXNVc2VyU2Nyb2xsZWQpIHtcbiAgICAgICAgICB0aGlzLm5ld01lc3NhZ2VDb3VudFdoaWxlQmVpbmdTY3JvbGxlZCA9IDA7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5jZFJlZi5kZXRlY3RDaGFuZ2VzKCk7XG4gICAgICB9KTtcbiAgICB9XG5cbiAgICBjb25zdCBwcmV2U2Nyb2xsUG9zaXRpb24gPSB0aGlzLnNjcm9sbFBvc2l0aW9uJC5nZXRWYWx1ZSgpO1xuXG4gICAgaWYgKHRoaXMuZGlyZWN0aW9uID09PSAndG9wLXRvLWJvdHRvbScpIHtcbiAgICAgIGlmIChzY3JvbGxQb3NpdGlvbiA9PT0gJ3RvcCcpIHtcbiAgICAgICAgc2Nyb2xsUG9zaXRpb24gPSAnYm90dG9tJztcbiAgICAgIH0gZWxzZSBpZiAoc2Nyb2xsUG9zaXRpb24gPT09ICdib3R0b20nKSB7XG4gICAgICAgIHNjcm9sbFBvc2l0aW9uID0gJ3RvcCc7XG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYgKHByZXZTY3JvbGxQb3NpdGlvbiAhPT0gc2Nyb2xsUG9zaXRpb24gJiYgIXRoaXMuaXNKdW1waW5nVG9NZXNzYWdlKSB7XG4gICAgICBpZiAoc2Nyb2xsUG9zaXRpb24gPT09ICd0b3AnIHx8IHNjcm9sbFBvc2l0aW9uID09PSAnYm90dG9tJykge1xuICAgICAgICB0aGlzLnZpcnR1YWxpemVkTGlzdD8udmlydHVhbGl6ZWRJdGVtcyRcbiAgICAgICAgICAucGlwZSh0YWtlKDEpKVxuICAgICAgICAgIC5zdWJzY3JpYmUoKGl0ZW1zKSA9PiB7XG4gICAgICAgICAgICB0aGlzLm1lc3NhZ2VJZFRvQW5jaG9yVG8gPVxuICAgICAgICAgICAgICBzY3JvbGxQb3NpdGlvbiA9PT0gJ3RvcCdcbiAgICAgICAgICAgICAgICA/IGl0ZW1zWzBdPy5pZFxuICAgICAgICAgICAgICAgIDogaXRlbXNbaXRlbXMubGVuZ3RoIC0gMV0/LmlkO1xuICAgICAgICAgICAgdGhpcy5hbmNob3JNZXNzYWdlVG9wT2Zmc2V0ID0gZG9jdW1lbnRcbiAgICAgICAgICAgICAgLmdldEVsZW1lbnRCeUlkKHRoaXMubWVzc2FnZUlkVG9BbmNob3JUbylcbiAgICAgICAgICAgICAgPy5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKT8udG9wO1xuICAgICAgICAgIH0pO1xuICAgICAgfVxuICAgICAgdGhpcy5uZ1pvbmUucnVuKCgpID0+IHtcbiAgICAgICAgdGhpcy5zY3JvbGxQb3NpdGlvbiQubmV4dChzY3JvbGxQb3NpdGlvbik7XG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIHByZXNlcnZlU2Nyb2xsYmFyUG9zaXRpb24oKSB7XG4gICAgaWYgKCF0aGlzLm1lc3NhZ2VJZFRvQW5jaG9yVG8pIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgY29uc3QgbWVzc2FnZVRvQWxpZ25UbyA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKHRoaXMubWVzc2FnZUlkVG9BbmNob3JUbyk7XG4gICAgdGhpcy5tZXNzYWdlSWRUb0FuY2hvclRvID0gdW5kZWZpbmVkO1xuICAgIHRoaXMuc2Nyb2xsQ29udGFpbmVyLm5hdGl2ZUVsZW1lbnQuc2Nyb2xsVG9wICs9XG4gICAgICAobWVzc2FnZVRvQWxpZ25Ubz8uZ2V0Qm91bmRpbmdDbGllbnRSZWN0KCk/LnRvcCB8fCAwKSAtXG4gICAgICAodGhpcy5hbmNob3JNZXNzYWdlVG9wT2Zmc2V0IHx8IDApO1xuICAgIHRoaXMuYW5jaG9yTWVzc2FnZVRvcE9mZnNldCA9IHVuZGVmaW5lZDtcbiAgICBpZiAodGhpcy5pc1NhZmFyaSkge1xuICAgICAgdGhpcy5mb3JjZVJlcGFpbnRTdWJqZWN0Lm5leHQoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGZvcmNlUmVwYWludCgpIHtcbiAgICAvLyBTb2x2ZXMgdGhlIGlzc3VlIG9mIGVtcHR5IHNjcmVlbiBvbiBTYWZhcmkgd2hlbiBzY3JvbGxpbmdcbiAgICB0aGlzLnNjcm9sbENvbnRhaW5lci5uYXRpdmVFbGVtZW50LnN0eWxlLmRpc3BsYXkgPSAnbm9uZSc7XG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtZXhwcmVzc2lvbnNcbiAgICB0aGlzLnNjcm9sbENvbnRhaW5lci5uYXRpdmVFbGVtZW50Lm9mZnNldEhlaWdodDsgLy8gbm8gbmVlZCB0byBzdG9yZSB0aGlzIGFueXdoZXJlLCB0aGUgcmVmZXJlbmNlIGlzIGVub3VnaFxuICAgIHRoaXMuc2Nyb2xsQ29udGFpbmVyLm5hdGl2ZUVsZW1lbnQuc3R5bGUuZGlzcGxheSA9ICcnO1xuICB9XG5cbiAgcHJpdmF0ZSBnZXRTY3JvbGxQb3NpdGlvbigpOiAndG9wJyB8ICdib3R0b20nIHwgJ21pZGRsZScge1xuICAgIGxldCBwb3NpdGlvbjogJ3RvcCcgfCAnYm90dG9tJyB8ICdtaWRkbGUnID0gJ21pZGRsZSc7XG4gICAgaWYgKFxuICAgICAgTWF0aC5jZWlsKHRoaXMuc2Nyb2xsQ29udGFpbmVyLm5hdGl2ZUVsZW1lbnQuc2Nyb2xsVG9wKSArXG4gICAgICAgIHRoaXMuc2Nyb2xsQ29udGFpbmVyLm5hdGl2ZUVsZW1lbnQuY2xpZW50SGVpZ2h0ICtcbiAgICAgICAgMSA+PVxuICAgICAgdGhpcy5zY3JvbGxDb250YWluZXIubmF0aXZlRWxlbWVudC5zY3JvbGxIZWlnaHRcbiAgICApIHtcbiAgICAgIHBvc2l0aW9uID0gJ2JvdHRvbSc7XG4gICAgfSBlbHNlIGlmIChcbiAgICAgIE1hdGguZmxvb3IodGhpcy5zY3JvbGxDb250YWluZXIubmF0aXZlRWxlbWVudC5zY3JvbGxUb3ApIDw9XG4gICAgICAodGhpcy5wYXJlbnRNZXNzYWdlRWxlbWVudD8ubmF0aXZlRWxlbWVudD8uY2xpZW50SGVpZ2h0IHx8IDApXG4gICAgKSB7XG4gICAgICBwb3NpdGlvbiA9ICd0b3AnO1xuICAgIH1cblxuICAgIHJldHVybiBwb3NpdGlvbjtcbiAgfVxuXG4gIHByaXZhdGUgc2V0TWVzc2FnZXMkKCkge1xuICAgIHRoaXMuZGlzcG9zZVZpcnR1YWxpemVkTGlzdCgpO1xuICAgIHRoaXMudmlydHVhbGl6ZWRMaXN0ID0gbmV3IFZpcnR1YWxpemVkTWVzc2FnZUxpc3RTZXJ2aWNlKFxuICAgICAgdGhpcy5tb2RlLFxuICAgICAgdGhpcy5zY3JvbGxQb3NpdGlvbiQsXG4gICAgICB0aGlzLmNoYW5uZWxTZXJ2aWNlLFxuICAgICk7XG4gICAgdGhpcy5xdWVyeVN0YXRlU3Vic2NyaXB0aW9uID0gdGhpcy52aXJ0dWFsaXplZExpc3QucXVlcnlTdGF0ZSQuc3Vic2NyaWJlKFxuICAgICAgKHF1ZXJ5U3RhdGUpID0+IHtcbiAgICAgICAgbGV0IG1hcHBlZFN0YXRlOiAnaWRsZScgfCAnbG9hZGluZy10b3AnIHwgJ2xvYWRpbmctYm90dG9tJyA9ICdpZGxlJztcbiAgICAgICAgaWYgKHF1ZXJ5U3RhdGUuc3RhdGUuaW5jbHVkZXMoJ2xvYWRpbmcnKSkge1xuICAgICAgICAgIG1hcHBlZFN0YXRlID0gKHF1ZXJ5U3RhdGUuc3RhdGUgYXMgJ2xvYWRpbmctdG9wJykgfHwgJ2xvYWRpbmctYm90dG9tJztcbiAgICAgICAgfVxuICAgICAgICBpZiAobWFwcGVkU3RhdGUgIT09IHRoaXMubG9hZGluZ1N0YXRlKSB7XG4gICAgICAgICAgdGhpcy5sb2FkaW5nU3RhdGUgPSBtYXBwZWRTdGF0ZTtcbiAgICAgICAgICBpZiAodGhpcy5pc1ZpZXdJbml0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSxcbiAgICApO1xuICAgIHRoaXMubWVzc2FnZXMkID0gdGhpcy52aXJ0dWFsaXplZExpc3QudmlydHVhbGl6ZWRJdGVtcyQucGlwZShcbiAgICAgIHRhcCgoKSA9PiB7XG4gICAgICAgIGlmICh0aGlzLmlzRW1wdHkpIHtcbiAgICAgICAgICAvLyBjZFJlZi5kZXRlY3RDaGFuZ2VzKCkgaXNuJ3QgZW5vdWdoIGhlcmUsIHRlc3Qgd2lsbCBmYWlsXG4gICAgICAgICAgc2V0VGltZW91dCgoKSA9PiAodGhpcy5pc0VtcHR5ID0gZmFsc2UpLCAwKTtcbiAgICAgICAgfVxuICAgICAgfSksXG4gICAgICB0YXAoKG1lc3NhZ2VzKSA9PiB7XG4gICAgICAgIGlmIChcbiAgICAgICAgICB0aGlzLmlzSnVtcGluZ1RvTGF0ZXN0VW5yZWFkTWVzc2FnZSAmJlxuICAgICAgICAgICF0aGlzLmZpcnN0VW5yZWFkTWVzc2FnZUlkICYmXG4gICAgICAgICAgdGhpcy5sYXN0UmVhZE1lc3NhZ2VJZFxuICAgICAgICApIHtcbiAgICAgICAgICBjb25zdCBsYXN0UmVhZEluZGV4ID0gbWVzc2FnZXMuZmluZEluZGV4KFxuICAgICAgICAgICAgKG0pID0+IG0uaWQgPT09IHRoaXMubGFzdFJlYWRNZXNzYWdlSWQsXG4gICAgICAgICAgKTtcbiAgICAgICAgICBpZiAobGFzdFJlYWRJbmRleCAhPT0gLTEpIHtcbiAgICAgICAgICAgIHRoaXMuZmlyc3RVbnJlYWRNZXNzYWdlSWQgPSBtZXNzYWdlc1tsYXN0UmVhZEluZGV4ICsgMV0/LmlkO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSksXG4gICAgICB0YXAoXG4gICAgICAgIChtZXNzYWdlcykgPT5cbiAgICAgICAgICAodGhpcy5sYXN0U2VudE1lc3NhZ2VJZCA9IFsuLi5tZXNzYWdlc11cbiAgICAgICAgICAgIC5yZXZlcnNlKClcbiAgICAgICAgICAgIC5maW5kKFxuICAgICAgICAgICAgICAobSkgPT5cbiAgICAgICAgICAgICAgICBtLnVzZXI/LmlkID09PSB0aGlzLmNoYXRDbGllbnRTZXJ2aWNlLmNoYXRDbGllbnQ/LnVzZXI/LmlkICYmXG4gICAgICAgICAgICAgICAgbS5zdGF0dXMgIT09ICdzZW5kaW5nJyxcbiAgICAgICAgICAgICk/LmlkKSxcbiAgICAgICksXG4gICAgICB0YXAoKG1lc3NhZ2VzKSA9PiB7XG4gICAgICAgIGNvbnN0IGxhdGVzdE1lc3NhZ2VJbkxpc3QgPSBtZXNzYWdlc1ttZXNzYWdlcy5sZW5ndGggLSAxXTtcbiAgICAgICAgY29uc3QgY2hhbm5lbCA9IHRoaXMuY2hhbm5lbFNlcnZpY2UuYWN0aXZlQ2hhbm5lbDtcbiAgICAgICAgY29uc3QgbWVzc2FnZXNGcm9tU3RhdGUgPVxuICAgICAgICAgICh0aGlzLm1vZGUgPT09ICdtYWluJ1xuICAgICAgICAgICAgPyBjaGFubmVsPy5zdGF0ZS5sYXRlc3RNZXNzYWdlc1xuICAgICAgICAgICAgOiBjaGFubmVsPy5zdGF0ZS50aHJlYWRzW3RoaXMucGFyZW50TWVzc2FnZT8uaWQgfHwgJyddKSB8fCBbXTtcbiAgICAgICAgdGhpcy5pc0xhdGVzdE1lc3NhZ2VJbkxpc3QgPVxuICAgICAgICAgICFsYXRlc3RNZXNzYWdlSW5MaXN0IHx8XG4gICAgICAgICAgbGF0ZXN0TWVzc2FnZUluTGlzdC5jaWQgIT09IGNoYW5uZWw/LmNpZCB8fFxuICAgICAgICAgICh0aGlzLm1vZGUgPT09ICd0aHJlYWQnICYmXG4gICAgICAgICAgICBsYXRlc3RNZXNzYWdlSW5MaXN0Py5wYXJlbnRfaWQgIT09IHRoaXMucGFyZW50TWVzc2FnZT8uaWQpIHx8XG4gICAgICAgICAgbGF0ZXN0TWVzc2FnZUluTGlzdD8uaWQgPT09XG4gICAgICAgICAgICBtZXNzYWdlc0Zyb21TdGF0ZVttZXNzYWdlc0Zyb21TdGF0ZS5sZW5ndGggLSAxXT8uaWQ7XG4gICAgICAgIGlmICghdGhpcy5pc0xhdGVzdE1lc3NhZ2VJbkxpc3QpIHtcbiAgICAgICAgICB0aGlzLmlzVXNlclNjcm9sbGVkID0gdHJ1ZTtcbiAgICAgICAgfVxuICAgICAgfSksXG4gICAgICBtYXAoKG1lc3NhZ2VzKSA9PiB7XG4gICAgICAgIHJldHVybiB0aGlzLmRpcmVjdGlvbiA9PT0gJ2JvdHRvbS10by10b3AnXG4gICAgICAgICAgPyBtZXNzYWdlc1xuICAgICAgICAgIDogWy4uLm1lc3NhZ2VzXS5yZXZlcnNlKCk7XG4gICAgICB9KSxcbiAgICAgIHRhcCgobWVzc2FnZXMpID0+IHtcbiAgICAgICAgdGhpcy5ncm91cFN0eWxlcyA9IG1lc3NhZ2VzLm1hcCgobSwgaSkgPT5cbiAgICAgICAgICBnZXRHcm91cFN0eWxlcyhtLCBtZXNzYWdlc1tpIC0gMV0sIG1lc3NhZ2VzW2kgKyAxXSwge1xuICAgICAgICAgICAgbGFzdFJlYWRNZXNzYWdlSWQ6IHRoaXMubGFzdFJlYWRNZXNzYWdlSWQsXG4gICAgICAgICAgfSksXG4gICAgICAgICk7XG4gICAgICAgIHRoaXMuaXNOZXh0TWVzc2FnZU9uU2VwYXJhdGVEYXRlID0gbWVzc2FnZXMubWFwKChtLCBpKSA9PlxuICAgICAgICAgIHRoaXMuY2hlY2tJZk9uU2VwYXJhdGVEYXRlcyhtLCBtZXNzYWdlc1tpICsgMV0pLFxuICAgICAgICApO1xuICAgICAgfSksXG4gICAgICBzaGFyZVJlcGxheSgxKSxcbiAgICApO1xuICAgIGlmICh0aGlzLnZpcnR1YWxpemVkTGlzdD8uanVtcFRvSXRlbSQpIHtcbiAgICAgIHRoaXMuanVtcFRvSXRlbVN1YnNjcmlwdGlvbiA9IHRoaXMudmlydHVhbGl6ZWRMaXN0Lmp1bXBUb0l0ZW0kXG4gICAgICAgIC5waXBlKGZpbHRlcigoanVtcFRvTWVzc2FnZSkgPT4gISFqdW1wVG9NZXNzYWdlLml0ZW0/LmlkKSlcbiAgICAgICAgLnN1YnNjcmliZSgoanVtcFRvTWVzc2FnZSkgPT4ge1xuICAgICAgICAgIGxldCBtZXNzYWdlSWQgPSBqdW1wVG9NZXNzYWdlLml0ZW0/LmlkO1xuICAgICAgICAgIGlmIChtZXNzYWdlSWQpIHtcbiAgICAgICAgICAgIGlmICh0aGlzLmlzSnVtcGluZ1RvTGF0ZXN0VW5yZWFkTWVzc2FnZSkge1xuICAgICAgICAgICAgICBtZXNzYWdlSWQgPSB0aGlzLmZpcnN0VW5yZWFkTWVzc2FnZUlkIHx8IG1lc3NhZ2VJZDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGlmIChqdW1wVG9NZXNzYWdlLnBvc2l0aW9uICE9PSAnYm90dG9tJykge1xuICAgICAgICAgICAgICB0aGlzLmhpZ2hsaWdodGVkTWVzc2FnZUlkID0gbWVzc2FnZUlkO1xuICAgICAgICAgICAgfSBlbHNlIGlmICh0aGlzLmRpcmVjdGlvbiA9PT0gJ3RvcC10by1ib3R0b20nKSB7XG4gICAgICAgICAgICAgIGp1bXBUb01lc3NhZ2UucG9zaXRpb24gPSAndG9wJztcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHRoaXMuaXNKdW1waW5nVG9NZXNzYWdlID0gdHJ1ZTtcbiAgICAgICAgICAgIHRoaXMuc2Nyb2xsTWVzc2FnZUludG9WaWV3KHtcbiAgICAgICAgICAgICAgbWVzc2FnZUlkOiB0aGlzLmZpcnN0VW5yZWFkTWVzc2FnZUlkIHx8IG1lc3NhZ2VJZCxcbiAgICAgICAgICAgICAgcG9zaXRpb246IGp1bXBUb01lc3NhZ2UucG9zaXRpb24gfHwgJ21pZGRsZScsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgcmVzZXRTY3JvbGxTdGF0ZSgpIHtcbiAgICB0aGlzLmlzRW1wdHkgPSB0cnVlO1xuICAgIHRoaXMuaXNVc2VyU2Nyb2xsZWQgPSBmYWxzZTtcbiAgICB0aGlzLm1lc3NhZ2VJZFRvQW5jaG9yVG8gPSB1bmRlZmluZWQ7XG4gICAgdGhpcy5hbmNob3JNZXNzYWdlVG9wT2Zmc2V0ID0gdW5kZWZpbmVkO1xuICAgIHRoaXMubmV3TWVzc2FnZUNvdW50V2hpbGVCZWluZ1Njcm9sbGVkID0gMDtcbiAgICB0aGlzLmlzTmV3TWVzc2FnZVNlbnRCeVVzZXIgPSBmYWxzZTtcbiAgICB0aGlzLmlzTGF0ZXN0TWVzc2FnZUluTGlzdCA9IHRydWU7XG4gICAgdGhpcy5pc0p1bXBpbmdUb01lc3NhZ2UgPSBmYWxzZTtcbiAgICB0aGlzLnNjcm9sbFBvc2l0aW9uJC5uZXh0KCdib3R0b20nKTtcbiAgICB0aGlzLmxvYWRpbmdTdGF0ZSA9ICdpZGxlJztcbiAgfVxuXG4gIHByaXZhdGUgZGlzcG9zZVZpcnR1YWxpemVkTGlzdCgpIHtcbiAgICB0aGlzLnZpcnR1YWxpemVkTGlzdD8uZGlzcG9zZSgpO1xuICAgIHRoaXMuanVtcFRvSXRlbVN1YnNjcmlwdGlvbj8udW5zdWJzY3JpYmUoKTtcbiAgICB0aGlzLnF1ZXJ5U3RhdGVTdWJzY3JpcHRpb24/LnVuc3Vic2NyaWJlKCk7XG4gIH1cblxuICBwcml2YXRlIGdldCB1c2Vyc1R5cGluZyQoKSB7XG4gICAgcmV0dXJuIHRoaXMubW9kZSA9PT0gJ3RocmVhZCdcbiAgICAgID8gdGhpcy51c2Vyc1R5cGluZ0luVGhyZWFkJFxuICAgICAgOiB0aGlzLnVzZXJzVHlwaW5nSW5DaGFubmVsJDtcbiAgfVxuXG4gIHByaXZhdGUgc2Nyb2xsTWVzc2FnZUludG9WaWV3KFxuICAgIG9wdGlvbnM6IHsgbWVzc2FnZUlkOiBzdHJpbmc7IHBvc2l0aW9uOiAndG9wJyB8ICdib3R0b20nIHwgJ21pZGRsZScgfSxcbiAgICB3aXRoUmV0cnk6IGJvb2xlYW4gPSB0cnVlLFxuICApIHtcbiAgICBjb25zdCBlbGVtZW50ID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQob3B0aW9ucy5tZXNzYWdlSWQpO1xuICAgIHRoaXMuanVtcFRvTWVzc2FnZVRpbWVvdXRzLmZvckVhY2goKHQpID0+IGNsZWFyVGltZW91dCh0KSk7XG4gICAgdGhpcy5qdW1wVG9NZXNzYWdlVGltZW91dHMgPSBbXTtcbiAgICBpZiAoIWVsZW1lbnQgJiYgd2l0aFJldHJ5KSB7XG4gICAgICAvLyBJZiB0aGUgbWVzc2FnZSB3YXMgbmV3bHkgaW5zZXJ0ZWQgaW50byBhY3RpdmVDaGFubmVsTWVzc2FnZXMkLCB0aGUgbWVzc2FnZSB3aWxsIGJlIHJlbmRlcmVkIGFmdGVyIHRoZSBjdXJyZW50IGNoYW5nZSBkZXRlY3Rpb24gY3ljbGUgLT4gd2FpdCBmb3IgdGhpcyBjeWNsZSB0byBjb21wbGV0ZVxuICAgICAgdGhpcy5qdW1wVG9NZXNzYWdlVGltZW91dHMucHVzaChcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiB0aGlzLnNjcm9sbE1lc3NhZ2VJbnRvVmlldyhvcHRpb25zLCBmYWxzZSkpLFxuICAgICAgKTtcbiAgICB9IGVsc2UgaWYgKGVsZW1lbnQpIHtcbiAgICAgIGNvbnN0IGJsb2NrTWFwcGluZzogeyBba2V5OiBzdHJpbmddOiBTY3JvbGxMb2dpY2FsUG9zaXRpb24gfSA9IHtcbiAgICAgICAgdG9wOiAnc3RhcnQnLFxuICAgICAgICBib3R0b206ICdlbmQnLFxuICAgICAgICBtaWRkbGU6ICdjZW50ZXInLFxuICAgICAgfTtcbiAgICAgIC8vIFdlIGNhbid0IGtub3cgd2hlbiBzbW9vdGggc2Nyb2xsaW5nIGVuZHMsIHNvIHdlIHNldCB0aGUgYmVoYXZpb3IgdG8gaW5zdGFudCBodHRwczovL2dpdGh1Yi5jb20vdzNjL2Nzc3dnLWRyYWZ0cy9pc3N1ZXMvMzc0NFxuICAgICAgZWxlbWVudC5zY3JvbGxJbnRvVmlldyh7XG4gICAgICAgIGJlaGF2aW9yOiAnaW5zdGFudCcgYXMgU2Nyb2xsQmVoYXZpb3IsXG4gICAgICAgIGJsb2NrOiBibG9ja01hcHBpbmdbb3B0aW9ucy5wb3NpdGlvbl0sXG4gICAgICB9KTtcbiAgICAgIGlmIChvcHRpb25zLnBvc2l0aW9uICE9PSAnbWlkZGxlJykge1xuICAgICAgICBvcHRpb25zLnBvc2l0aW9uID09PSAnYm90dG9tJ1xuICAgICAgICAgID8gdGhpcy5zY3JvbGxUb0JvdHRvbSgpXG4gICAgICAgICAgOiB0aGlzLnNjcm9sbFRvVG9wKCk7XG4gICAgICB9XG4gICAgICB0aGlzLmp1bXBUb01lc3NhZ2VUaW1lb3V0cy5wdXNoKFxuICAgICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICB0aGlzLmlzSnVtcGluZ1RvTWVzc2FnZSA9IGZhbHNlO1xuICAgICAgICAgIGlmICghdGhpcy5pc1VzZXJTY3JvbGxlZCkge1xuICAgICAgICAgICAgdGhpcy5jaGVja0lmVXNlclNjcm9sbGVkKCk7XG4gICAgICAgICAgfVxuICAgICAgICB9LCAyMDApLFxuICAgICAgKTtcbiAgICAgIHRoaXMuanVtcFRvTWVzc2FnZVRpbWVvdXRzLnB1c2goXG4gICAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICAgIHRoaXMuaGlnaGxpZ2h0ZWRNZXNzYWdlSWQgPSB1bmRlZmluZWQ7XG4gICAgICAgICAgdGhpcy5maXJzdFVucmVhZE1lc3NhZ2VJZCA9IHVuZGVmaW5lZDtcbiAgICAgICAgICB0aGlzLmlzSnVtcGluZ1RvTGF0ZXN0VW5yZWFkTWVzc2FnZSA9IGZhbHNlO1xuICAgICAgICAgIHRoaXMuanVtcFRvTWVzc2FnZVRpbWVvdXRzID0gW107XG4gICAgICAgICAgdGhpcy5jZFJlZi5kZXRlY3RDaGFuZ2VzKCk7XG4gICAgICAgIH0sIDEwMDApLFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy5pc0p1bXBpbmdUb01lc3NhZ2UgPSBmYWxzZTtcbiAgICAgIHRoaXMuaGlnaGxpZ2h0ZWRNZXNzYWdlSWQgPSB1bmRlZmluZWQ7XG4gICAgICB0aGlzLmZpcnN0VW5yZWFkTWVzc2FnZUlkID0gdW5kZWZpbmVkO1xuICAgICAgdGhpcy5pc0p1bXBpbmdUb0xhdGVzdFVucmVhZE1lc3NhZ2UgPSBmYWxzZTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIG5ld01lc3NhZ2VSZWNlaXZlZChtZXNzYWdlOiBNZXNzYWdlUmVzcG9uc2UpIHtcbiAgICBpZiAoXG4gICAgICAodGhpcy5tb2RlID09PSAnbWFpbicgJiYgbWVzc2FnZS5wYXJlbnRfaWQpIHx8XG4gICAgICAodGhpcy5tb2RlID09PSAndGhyZWFkJyAmJiBtZXNzYWdlLnBhcmVudF9pZCAhPT0gdGhpcy5wYXJlbnRNZXNzYWdlPy5pZClcbiAgICApIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgY29uc3QgaXNOZXdNZXNzYWdlU2VudEJ5Q3VycmVudFVzZXIgPVxuICAgICAgbWVzc2FnZS51c2VyPy5pZCA9PT0gdGhpcy5jaGF0Q2xpZW50U2VydmljZS5jaGF0Q2xpZW50Py51c2VyPy5pZDtcblxuICAgIGxldCBzaG91bGREZXRlY3RDaGFuZ2VzID0gZmFsc2U7XG5cbiAgICBpZiAoIXRoaXMuaXNOZXdNZXNzYWdlU2VudEJ5VXNlciAmJiBpc05ld01lc3NhZ2VTZW50QnlDdXJyZW50VXNlcikge1xuICAgICAgdGhpcy5pc05ld01lc3NhZ2VTZW50QnlVc2VyID0gdHJ1ZTtcbiAgICAgIHNob3VsZERldGVjdENoYW5nZXMgPSB0cnVlO1xuICAgIH1cblxuICAgIGlmICh0aGlzLmlzVXNlclNjcm9sbGVkKSB7XG4gICAgICB0aGlzLm5ld01lc3NhZ2VDb3VudFdoaWxlQmVpbmdTY3JvbGxlZCsrO1xuICAgICAgc2hvdWxkRGV0ZWN0Q2hhbmdlcyA9IHRydWU7XG4gICAgfVxuICAgIGlmICghdGhpcy5pc05ld01lc3NhZ2VTZW50QnlVc2VyICYmIHRoaXMudW5yZWFkQ291bnQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgdGhpcy51bnJlYWRDb3VudCsrO1xuICAgICAgc2hvdWxkRGV0ZWN0Q2hhbmdlcyA9IHRydWU7XG4gICAgfVxuXG4gICAgaWYgKHNob3VsZERldGVjdENoYW5nZXMgJiYgdGhpcy5pc1ZpZXdJbml0ZWQpIHtcbiAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgY2hlY2tJZk9uU2VwYXJhdGVEYXRlcyhcbiAgICBtZXNzYWdlPzogU3RyZWFtTWVzc2FnZSxcbiAgICBuZXh0TWVzc2FnZT86IFN0cmVhbU1lc3NhZ2UsXG4gICkge1xuICAgIGlmICghbWVzc2FnZSB8fCAhbmV4dE1lc3NhZ2UpIHtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIGlzT25TZXBhcmF0ZURhdGUobWVzc2FnZS5jcmVhdGVkX2F0LCBuZXh0TWVzc2FnZS5jcmVhdGVkX2F0KTtcbiAgfVxufVxuIiwiQGlmIChcbiAgbGFzdFJlYWRNZXNzYWdlSWQgJiZcbiAgaXNVbnJlYWROb3RpZmljYXRpb25WaXNpYmxlICYmXG4gIG9wZW5NZXNzYWdlTGlzdEF0ID09PSBcImxhc3QtbWVzc2FnZVwiICYmXG4gIGRpc3BsYXlVbnJlYWRTZXBhcmF0b3Jcbikge1xuICA8bmctY29udGFpbmVyXG4gICAgKm5nVGVtcGxhdGVPdXRsZXQ9XCJcbiAgICAgIGN1c3RvbW5ld01lc3NhZ2VzTm90aWZpY2F0aW9uVGVtcGxhdGUgfHxcbiAgICAgICAgZGVmYXVsdFVucmVhZE1lc3NhZ2VzTm90aWZpY2F0aW9uO1xuICAgICAgY29udGV4dDoge1xuICAgICAgICB1bnJlYWRDb3VudDogdW5yZWFkQ291bnQsXG4gICAgICAgIG9uRGlzbWlzczogbWVzc2FnZU5vdGlmaWNhdGlvbkRpc21pc3NDbGlja2VkLFxuICAgICAgICBvbkp1bXA6IG1lc3NhZ2VOb3RpZmljYXRpb25KdW1wQ2xpY2tlZCxcbiAgICAgIH1cbiAgICBcIlxuICAvPlxufVxuPG5nLXRlbXBsYXRlXG4gICNkZWZhdWx0VW5yZWFkTWVzc2FnZXNOb3RpZmljYXRpb25cbiAgbGV0LXVucmVhZENvdW50PVwidW5yZWFkQ291bnRcIlxuICBsZXQtb25EaXNtaXNzPVwib25EaXNtaXNzXCJcbiAgbGV0LW9uSnVtcD1cIm9uSnVtcFwiXG4+XG4gIDxkaXZcbiAgICBjbGFzcz1cInN0ci1jaGF0X191bnJlYWQtbWVzc2FnZXMtbm90aWZpY2F0aW9uXCJcbiAgICBkYXRhLXRlc3RpZD1cInVucmVhZC1tZXNzYWdlcy1ub3RpZmljYXRpb25cIlxuICA+XG4gICAgPGJ1dHRvblxuICAgICAgZGF0YS10ZXN0aWQ9XCJ1bnJlYWQtbWVzc2FnZXMtbm90aWZpY2F0aW9uLWp1bXAtdG8tbWVzc2FnZVwiXG4gICAgICAoY2xpY2spPVwib25KdW1wKClcIlxuICAgID5cbiAgICAgIEBpZiAodW5yZWFkQ291bnQgPiAwICYmICFoaWRlVW5yZWFkQ291bnRGb3JOb3RpZmljYXRpb25BbmRJbmRpY2F0b3IpIHtcbiAgICAgICAge3tcbiAgICAgICAgICAodW5yZWFkQ291bnQgPT09IDFcbiAgICAgICAgICAgID8gXCJzdHJlYW1DaGF0Llxce1xce2NvdW50XFx9XFx9IHVucmVhZCBtZXNzYWdlXCJcbiAgICAgICAgICAgIDogXCJzdHJlYW1DaGF0Llxce1xce2NvdW50XFx9XFx9IHVucmVhZCBtZXNzYWdlc1wiXG4gICAgICAgICAgKSB8IHRyYW5zbGF0ZTogeyBjb3VudDogdW5yZWFkQ291bnQgfVxuICAgICAgICB9fVxuICAgICAgfSBAZWxzZSB7XG4gICAgICAgIHt7IFwic3RyZWFtQ2hhdC5VbnJlYWQgbWVzc2FnZXNcIiB8IHRyYW5zbGF0ZSB9fVxuICAgICAgfVxuICAgIDwvYnV0dG9uPlxuICAgIDxidXR0b25cbiAgICAgIGRhdGEtdGVzdGlkPVwidW5yZWFkLW1lc3NhZ2VzLW5vdGlmaWNhdGlvbi1kaXNtaXNzXCJcbiAgICAgIChjbGljayk9XCJvbkRpc21pc3MoKVwiXG4gICAgPlxuICAgICAgPHN0cmVhbS1pY29uLXBsYWNlaG9sZGVyIGljb249XCJjbG9zZVwiIC8+XG4gICAgPC9idXR0b24+XG4gIDwvZGl2PlxuPC9uZy10ZW1wbGF0ZT5cbjxkaXYgI3Njcm9sbENvbnRhaW5lciBkYXRhLXRlc3RpZD1cInNjcm9sbC1jb250YWluZXJcIiBjbGFzcz1cInN0ci1jaGF0X19saXN0XCI+XG4gIEBpZiAobW9kZSA9PT0gXCJtYWluXCIgJiYgaXNFbXB0eSAmJiBlbXB0eUxpc3RUZW1wbGF0ZSkge1xuICAgIDxuZy1jb250YWluZXIgKm5nVGVtcGxhdGVPdXRsZXQ9XCJlbXB0eUxpc3RUZW1wbGF0ZVwiIC8+XG4gIH1cbiAgPGRpdiBjbGFzcz1cInN0ci1jaGF0X19yZXZlcnNlLWluZmluaXRlLXNjcm9sbCBzdHItY2hhdF9fbWVzc2FnZS1saXN0LXNjcm9sbFwiPlxuICAgIDx1bCBjbGFzcz1cInN0ci1jaGF0X191bFwiPlxuICAgICAgQGlmIChtb2RlID09PSBcInRocmVhZFwiICYmIHBhcmVudE1lc3NhZ2UpIHtcbiAgICAgICAgPGxpXG4gICAgICAgICAgI3BhcmVudE1lc3NhZ2VFbGVtZW50XG4gICAgICAgICAgZGF0YS10ZXN0aWQ9XCJwYXJlbnQtbWVzc2FnZVwiXG4gICAgICAgICAgY2xhc3M9XCJzdHItY2hhdF9fcGFyZW50LW1lc3NhZ2UtbGlcIlxuICAgICAgICA+XG4gICAgICAgICAgPG5nLWNvbnRhaW5lclxuICAgICAgICAgICAgKm5nVGVtcGxhdGVPdXRsZXQ9XCJcbiAgICAgICAgICAgICAgbWVzc2FnZVRlbXBsYXRlQ29udGFpbmVyO1xuICAgICAgICAgICAgICBjb250ZXh0OiB7IG1lc3NhZ2U6IHBhcmVudE1lc3NhZ2UsIGluZGV4OiAncGFyZW50JyB9XG4gICAgICAgICAgICBcIlxuICAgICAgICAgIC8+XG4gICAgICAgICAgPGRpdiBkYXRhLXRlc3RpZD1cInJlcGx5LWNvdW50XCIgY2xhc3M9XCJzdHItY2hhdF9fdGhyZWFkLXN0YXJ0XCI+XG4gICAgICAgICAgICB7e3BhcmVudE1lc3NhZ2UucmVwbHlfY291bnQgPT09IDEgPyAoJ3N0cmVhbUNoYXQuMSByZXBseScgfCB0cmFuc2xhdGUpIDogKCdzdHJlYW1DaGF0Lnt7IHJlcGx5Q291bnQgfX1cbiAgICAgICAgICAgIHJlcGxpZXMnIHwgdHJhbnNsYXRlOnJlcGx5Q291bnRQYXJhbSl9fVxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICA8L2xpPlxuICAgICAgfVxuICAgICAgQGlmIChtb2RlID09PSBcInRocmVhZFwiICYmIGlzRW1wdHkgJiYgZW1wdHlMaXN0VGVtcGxhdGUpIHtcbiAgICAgICAgPG5nLWNvbnRhaW5lciAqbmdUZW1wbGF0ZU91dGxldD1cImVtcHR5TGlzdFRlbXBsYXRlXCIgLz5cbiAgICAgIH1cbiAgICAgIEBpZiAoXG4gICAgICAgICgobG9hZGluZ1N0YXRlID09PSBcImxvYWRpbmctdG9wXCIgJiYgZGlyZWN0aW9uID09PSBcImJvdHRvbS10by10b3BcIikgfHxcbiAgICAgICAgICAobG9hZGluZ1N0YXRlID09PSBcImxvYWRpbmctYm90dG9tXCIgJiZcbiAgICAgICAgICAgIGRpcmVjdGlvbiA9PT0gXCJ0b3AtdG8tYm90dG9tXCIpKSAmJlxuICAgICAgICBkaXNwbGF5TG9hZGluZ0luZGljYXRvclxuICAgICAgKSB7XG4gICAgICAgIDxzdHJlYW0tbG9hZGluZy1pbmRpY2F0b3ItcGxhY2Vob2xkZXJcbiAgICAgICAgICBkYXRhLXRlc3RpZD1cInRvcC1sb2FkaW5nLWluZGljYXRvclwiXG4gICAgICAgIC8+XG4gICAgICB9IEBlbHNlIHtcbiAgICAgICAgPG5nLWNvbnRhaW5lciAqbmdUZW1wbGF0ZU91dGxldD1cImxvYWRpbmdJbmRpY2F0b3JQbGFjZWhvbGRlclwiIC8+XG4gICAgICB9XG4gICAgICBAaWYgKG1lc3NhZ2VzJCB8IGFzeW5jOyBhcyBtZXNzYWdlcykge1xuICAgICAgICBAZm9yIChcbiAgICAgICAgICBtZXNzYWdlIG9mIG1lc3NhZ2VzO1xuICAgICAgICAgIHRyYWNrIG1lc3NhZ2UuaWQ7XG4gICAgICAgICAgbGV0IGkgPSAkaW5kZXg7XG4gICAgICAgICAgbGV0IGlzRmlyc3QgPSAkZmlyc3Q7XG4gICAgICAgICAgbGV0IGlzTGFzdCA9ICRsYXN0XG4gICAgICAgICkge1xuICAgICAgICAgIEBpZiAoaXNGaXJzdCkge1xuICAgICAgICAgICAgPG5nLWNvbnRhaW5lclxuICAgICAgICAgICAgICAqbmdUZW1wbGF0ZU91dGxldD1cIlxuICAgICAgICAgICAgICAgIGRhdGVTZXBhcmF0b3I7XG4gICAgICAgICAgICAgICAgY29udGV4dDoge1xuICAgICAgICAgICAgICAgICAgZGF0ZTogbWVzc2FnZS5jcmVhdGVkX2F0LFxuICAgICAgICAgICAgICAgICAgcGFyc2VkRGF0ZTogcGFyc2VEYXRlKG1lc3NhZ2UuY3JlYXRlZF9hdCksXG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICBcIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICB9XG4gICAgICAgICAgPGxpXG4gICAgICAgICAgICB0YWJpbmRleD1cIjBcIlxuICAgICAgICAgICAgZGF0YS10ZXN0Y2xhc3M9XCJtZXNzYWdlXCJcbiAgICAgICAgICAgIGNsYXNzPVwic3RyLWNoYXRfX2xpIHN0ci1jaGF0X19saS0te3sgZ3JvdXBTdHlsZXNbaV0gfX1cIlxuICAgICAgICAgICAgaWQ9XCJ7eyBtZXNzYWdlLmlkIH19XCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICA8bmctY29udGFpbmVyXG4gICAgICAgICAgICAgICpuZ1RlbXBsYXRlT3V0bGV0PVwiXG4gICAgICAgICAgICAgICAgbWVzc2FnZVRlbXBsYXRlQ29udGFpbmVyO1xuICAgICAgICAgICAgICAgIGNvbnRleHQ6IHsgbWVzc2FnZTogbWVzc2FnZSwgaW5kZXg6IGkgfVxuICAgICAgICAgICAgICBcIlxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L2xpPlxuICAgICAgICAgIEBpZiAoXG4gICAgICAgICAgICAobGFzdFJlYWRNZXNzYWdlSWQgPT09IG1lc3NhZ2U/LmlkICYmXG4gICAgICAgICAgICAgIGRpcmVjdGlvbiA9PT0gXCJib3R0b20tdG8tdG9wXCIpIHx8XG4gICAgICAgICAgICAoZGlyZWN0aW9uID09PSBcInRvcC10by1ib3R0b21cIiAmJlxuICAgICAgICAgICAgICBsYXN0UmVhZE1lc3NhZ2VJZCA9PT0gbWVzc2FnZXNbaSArIDFdPy5pZClcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIEBpZiAoZGlzcGxheVVucmVhZFNlcGFyYXRvcikge1xuICAgICAgICAgICAgICA8bGlcbiAgICAgICAgICAgICAgICBpZD1cInN0cmVhbS1jaGF0LW5ldy1tZXNzYWdlLWluZGljYXRvclwiXG4gICAgICAgICAgICAgICAgZGF0YS10ZXN0aWQ9XCJuZXctbWVzc2FnZXMtaW5kaWNhdG9yXCJcbiAgICAgICAgICAgICAgICBjbGFzcz1cInN0ci1jaGF0X19saSBzdHItY2hhdF9fdW5yZWFkLW1lc3NhZ2VzLXNlcGFyYXRvci13cmFwcGVyXCJcbiAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgIDxuZy1jb250YWluZXJcbiAgICAgICAgICAgICAgICAgICpuZ1RlbXBsYXRlT3V0bGV0PVwiXG4gICAgICAgICAgICAgICAgICAgIGN1c3RvbW5ld01lc3NhZ2VzSW5kaWNhdG9yVGVtcGxhdGUgfHxcbiAgICAgICAgICAgICAgICAgICAgICBkZWZhdWx0TmV3TWVzc2FnZXNJbmRpY2F0b3I7XG4gICAgICAgICAgICAgICAgICAgIGNvbnRleHQ6IHsgdW5yZWFkQ291bnQ6IHVucmVhZENvdW50IH1cbiAgICAgICAgICAgICAgICAgIFwiXG4gICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgPC9saT5cbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgICAgQGlmIChpc05leHRNZXNzYWdlT25TZXBhcmF0ZURhdGVbaV0pIHtcbiAgICAgICAgICAgIDxuZy1jb250YWluZXJcbiAgICAgICAgICAgICAgKm5nVGVtcGxhdGVPdXRsZXQ9XCJcbiAgICAgICAgICAgICAgICBkYXRlU2VwYXJhdG9yO1xuICAgICAgICAgICAgICAgIGNvbnRleHQ6IHtcbiAgICAgICAgICAgICAgICAgIGRhdGU6IG1lc3NhZ2VzW2kgKyAxXS5jcmVhdGVkX2F0LFxuICAgICAgICAgICAgICAgICAgcGFyc2VkRGF0ZTogcGFyc2VEYXRlKG1lc3NhZ2VzW2kgKyAxXS5jcmVhdGVkX2F0KSxcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIFwiXG4gICAgICAgICAgICAvPlxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgQGlmIChcbiAgICAgICAgKChsb2FkaW5nU3RhdGUgPT09IFwibG9hZGluZy1ib3R0b21cIiAmJiBkaXJlY3Rpb24gPT09IFwiYm90dG9tLXRvLXRvcFwiKSB8fFxuICAgICAgICAgIChsb2FkaW5nU3RhdGUgPT09IFwibG9hZGluZy10b3BcIiAmJiBkaXJlY3Rpb24gPT09IFwidG9wLXRvLWJvdHRvbVwiKSkgJiZcbiAgICAgICAgZGlzcGxheUxvYWRpbmdJbmRpY2F0b3JcbiAgICAgICkge1xuICAgICAgICA8c3RyZWFtLWxvYWRpbmctaW5kaWNhdG9yLXBsYWNlaG9sZGVyXG4gICAgICAgICAgZGF0YS10ZXN0aWQ9XCJib3R0b20tbG9hZGluZy1pbmRpY2F0b3JcIlxuICAgICAgICAvPlxuICAgICAgfSBAZWxzZSB7XG4gICAgICAgIDxuZy1jb250YWluZXIgKm5nVGVtcGxhdGVPdXRsZXQ9XCJsb2FkaW5nSW5kaWNhdG9yUGxhY2Vob2xkZXJcIiAvPlxuICAgICAgfVxuICAgICAgPG5nLXRlbXBsYXRlICNsb2FkaW5nSW5kaWNhdG9yUGxhY2Vob2xkZXI+XG4gICAgICAgIDxkaXYgY2xhc3M9XCJzdHItY2hhdF9fbG9hZGluZy1pbmRpY2F0b3ItcGxhY2Vob2xkZXJcIj48L2Rpdj5cbiAgICAgIDwvbmctdGVtcGxhdGU+XG4gICAgPC91bD5cbiAgICA8bmctdGVtcGxhdGUgI2RlZmF1bHRUeXBpbmdJbmRpY2F0b3IgbGV0LXVzZXJzVHlwaW5nJD1cInVzZXJzVHlwaW5nJFwiPlxuICAgICAgPCEtLSBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQGFuZ3VsYXItZXNsaW50L3RlbXBsYXRlL25vLWFueSAtLT5cbiAgICAgIEBpZiAoJGFueSh1c2Vyc1R5cGluZyQgfCBhc3luYyk7IGFzIHVzZXJzKSB7XG4gICAgICAgIEBpZiAodXNlcnMubGVuZ3RoID4gMCkge1xuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIGRhdGEtdGVzdGlkPVwidHlwaW5nLWluZGljYXRvclwiXG4gICAgICAgICAgICBjbGFzcz1cInN0ci1jaGF0X190eXBpbmctaW5kaWNhdG9yIHN0ci1jaGF0X190eXBpbmctaW5kaWNhdG9yLS10eXBpbmdcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIDxkaXYgY2xhc3M9XCJzdHItY2hhdF9fdHlwaW5nLWluZGljYXRvcl9fZG90c1wiPlxuICAgICAgICAgICAgICA8c3BhbiBjbGFzcz1cInN0ci1jaGF0X190eXBpbmctaW5kaWNhdG9yX19kb3RcIj48L3NwYW4+XG4gICAgICAgICAgICAgIDxzcGFuIGNsYXNzPVwic3RyLWNoYXRfX3R5cGluZy1pbmRpY2F0b3JfX2RvdFwiPjwvc3Bhbj5cbiAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9XCJzdHItY2hhdF9fdHlwaW5nLWluZGljYXRvcl9fZG90XCI+PC9zcGFuPlxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgICA8ZGl2XG4gICAgICAgICAgICAgIGRhdGEtdGVzdGlkPVwidHlwaW5nLXVzZXJzXCJcbiAgICAgICAgICAgICAgY2xhc3M9XCJzdHItY2hhdF9fdHlwaW5nLWluZGljYXRvcl9fdXNlcnNcIlxuICAgICAgICAgICAgPlxuICAgICAgICAgICAgICB7e1xuICAgICAgICAgICAgICAgIHVzZXJzLmxlbmd0aCA9PT0gMVxuICAgICAgICAgICAgICAgICAgPyAoXCJzdHJlYW1DaGF0LnVzZXIgaXMgdHlwaW5nXCJcbiAgICAgICAgICAgICAgICAgICAgfCB0cmFuc2xhdGU6IHsgdXNlcjogZ2V0VHlwaW5nSW5kaWNhdG9yVGV4dCh1c2VycykgfSlcbiAgICAgICAgICAgICAgICAgIDogKFwic3RyZWFtQ2hhdC51c2VycyBhcmUgdHlwaW5nXCJcbiAgICAgICAgICAgICAgICAgICAgfCB0cmFuc2xhdGU6IHsgdXNlcnM6IGdldFR5cGluZ0luZGljYXRvclRleHQodXNlcnMpIH0pXG4gICAgICAgICAgICAgIH19XG4gICAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgfVxuICAgICAgfVxuICAgIDwvbmctdGVtcGxhdGU+XG4gICAgPG5nLWNvbnRhaW5lclxuICAgICAgKm5nVGVtcGxhdGVPdXRsZXQ9XCJcbiAgICAgICAgdHlwaW5nSW5kaWNhdG9yVGVtcGxhdGUgfHwgZGVmYXVsdFR5cGluZ0luZGljYXRvcjtcbiAgICAgICAgY29udGV4dDogZ2V0VHlwaW5nSW5kaWNhdG9yQ29udGV4dCgpXG4gICAgICBcIlxuICAgIC8+XG4gIDwvZGl2PlxuPC9kaXY+XG48ZGl2IGNsYXNzPVwic3RyLWNoYXRfX2p1bXAtdG8tbGF0ZXN0LW1lc3NhZ2VcIj5cbiAgQGlmIChpc1VzZXJTY3JvbGxlZCAmJiBpc0p1bXBUb0xhdGVzdEJ1dHRvblZpc2libGUpIHtcbiAgICA8YnV0dG9uXG4gICAgICBkYXRhLXRlc3RpZD1cInNjcm9sbC10by1sYXRlc3RcIlxuICAgICAgY2xhc3M9XCJzdHItY2hhdF9fbWVzc2FnZS1ub3RpZmljYXRpb24tc2Nyb2xsLXRvLWxhdGVzdCBzdHItY2hhdF9fbWVzc2FnZS1ub3RpZmljYXRpb24tc2Nyb2xsLXRvLWxhdGVzdC1yaWdodCBzdHItY2hhdF9fY2lyY2xlLWZhYlwiXG4gICAgICAoa2V5dXAuZW50ZXIpPVwianVtcFRvTGF0ZXN0TWVzc2FnZSgpXCJcbiAgICAgIChjbGljayk9XCJqdW1wVG9MYXRlc3RNZXNzYWdlKClcIlxuICAgID5cbiAgICAgIDxzdHJlYW0taWNvblxuICAgICAgICBjbGFzcz1cInN0ci1jaGF0X19qdW1wLXRvLWxhdGVzdC1pY29uIHN0ci1jaGF0X19jaXJjbGUtZmFiLWljb25cIlxuICAgICAgICBbaWNvbl09XCJkaXJlY3Rpb24gPT09ICdib3R0b20tdG8tdG9wJyA/ICdhcnJvdy1kb3duJyA6ICdhcnJvdy11cCdcIlxuICAgICAgLz5cbiAgICAgIEBpZiAobmV3TWVzc2FnZUNvdW50V2hpbGVCZWluZ1Njcm9sbGVkID4gMCkge1xuICAgICAgICA8ZGl2XG4gICAgICAgICAgY2xhc3M9XCJzdHItY2hhdF9fbWVzc2FnZS1ub3RpZmljYXRpb24gc3RyLWNoYXRfX21lc3NhZ2Utbm90aWZpY2F0aW9uLXNjcm9sbC10by1sYXRlc3QtdW5yZWFkLWNvdW50IHN0ci1jaGF0X19qdW1wLXRvLWxhdGVzdC11bnJlYWQtY291bnRcIlxuICAgICAgICA+XG4gICAgICAgICAge3sgbmV3TWVzc2FnZUNvdW50V2hpbGVCZWluZ1Njcm9sbGVkIH19XG4gICAgICAgIDwvZGl2PlxuICAgICAgfVxuICAgIDwvYnV0dG9uPlxuICB9XG48L2Rpdj5cblxuPG5nLXRlbXBsYXRlICNtZXNzYWdlVGVtcGxhdGVDb250YWluZXIgbGV0LW1lc3NhZ2U9XCJtZXNzYWdlXCIgbGV0LWluZGV4PVwiaW5kZXhcIj5cbiAgPG5nLXRlbXBsYXRlXG4gICAgI2RlZmF1bHRNZXNzYWdlVGVtcGxhdGVcbiAgICBsZXQtbWVzc2FnZUlucHV0PVwibWVzc2FnZVwiXG4gICAgbGV0LWlzTGFzdFNlbnRNZXNzYWdlPVwiaXNMYXN0U2VudE1lc3NhZ2VcIlxuICAgIGxldC1lbmFibGVkTWVzc2FnZUFjdGlvbnM9XCJlbmFibGVkTWVzc2FnZUFjdGlvbnNcIlxuICAgIGxldC1tb2RlPVwibW9kZVwiXG4gICAgbGV0LWlzSGlnaGxpZ2h0ZWQ9XCJpc0hpZ2hsaWdodGVkXCJcbiAgICBsZXQtc2Nyb2xsJD1cInNjcm9sbCRcIlxuICA+XG4gICAgPHN0cmVhbS1tZXNzYWdlXG4gICAgICBbbWVzc2FnZV09XCJtZXNzYWdlSW5wdXRcIlxuICAgICAgW2lzTGFzdFNlbnRNZXNzYWdlXT1cImlzTGFzdFNlbnRNZXNzYWdlXCJcbiAgICAgIFtlbmFibGVkTWVzc2FnZUFjdGlvbnNdPVwiZW5hYmxlZE1lc3NhZ2VBY3Rpb25zXCJcbiAgICAgIFttb2RlXT1cIm1vZGVcIlxuICAgICAgW2lzSGlnaGxpZ2h0ZWRdPVwiaXNIaWdobGlnaHRlZFwiXG4gICAgICBbc2Nyb2xsJF09XCJzY3JvbGwkXCJcbiAgICAvPlxuICA8L25nLXRlbXBsYXRlPlxuICA8bmctY29udGFpbmVyXG4gICAgKm5nVGVtcGxhdGVPdXRsZXQ9XCJcbiAgICAgIG1lc3NhZ2VUZW1wbGF0ZSB8fCBkZWZhdWx0TWVzc2FnZVRlbXBsYXRlO1xuICAgICAgY29udGV4dDoge1xuICAgICAgICBtZXNzYWdlOiBtZXNzYWdlLFxuICAgICAgICBpc0xhc3RTZW50TWVzc2FnZTogISEoXG4gICAgICAgICAgbGFzdFNlbnRNZXNzYWdlSWQgJiYgbWVzc2FnZT8uaWQgPT09IGxhc3RTZW50TWVzc2FnZUlkXG4gICAgICAgICksXG4gICAgICAgIGVuYWJsZWRNZXNzYWdlQWN0aW9uczogZW5hYmxlZE1lc3NhZ2VBY3Rpb25zLFxuICAgICAgICBtb2RlOiBtb2RlLFxuICAgICAgICBpc0hpZ2hsaWdodGVkOiBtZXNzYWdlPy5pZCA9PT0gaGlnaGxpZ2h0ZWRNZXNzYWdlSWQsXG4gICAgICAgIHNjcm9sbCQ6IHNjcm9sbCQsXG4gICAgICB9XG4gICAgXCJcbiAgLz5cbjwvbmctdGVtcGxhdGU+XG5cbjxuZy10ZW1wbGF0ZSAjZGF0ZVNlcGFyYXRvciBsZXQtZGF0ZT1cImRhdGVcIiBsZXQtcGFyc2VkRGF0ZT1cInBhcnNlZERhdGVcIj5cbiAgQGlmIChkaXNwbGF5RGF0ZVNlcGFyYXRvcikge1xuICAgIDxuZy1jb250YWluZXJcbiAgICAgICpuZ1RlbXBsYXRlT3V0bGV0PVwiXG4gICAgICAgIGN1c3RvbURhdGVTZXBhcmF0b3JUZW1wbGF0ZSB8fCBkZWZhdWx0RGF0ZVNlcGFyYXRvcjtcbiAgICAgICAgY29udGV4dDoge1xuICAgICAgICAgIGRhdGU6IGRhdGUsXG4gICAgICAgICAgcGFyc2VkRGF0ZTogcGFyc2VkRGF0ZSxcbiAgICAgICAgfVxuICAgICAgXCJcbiAgICAvPlxuICB9XG5cbiAgPG5nLXRlbXBsYXRlXG4gICAgI2RlZmF1bHREYXRlU2VwYXJhdG9yXG4gICAgbGV0LWRhdGU9XCJkYXRlXCJcbiAgICBsZXQtcGFyc2VkRGF0ZT1cInBhcnNlZERhdGVcIlxuICA+XG4gICAgPGRpdiBkYXRhLXRlc3RpZD1cImRhdGUtc2VwYXJhdG9yXCIgY2xhc3M9XCJzdHItY2hhdF9fZGF0ZS1zZXBhcmF0b3JcIj5cbiAgICAgIEBpZiAoXG4gICAgICAgIGRhdGVTZXBhcmF0b3JUZXh0UG9zID09PSBcInJpZ2h0XCIgfHwgZGF0ZVNlcGFyYXRvclRleHRQb3MgPT09IFwiY2VudGVyXCJcbiAgICAgICkge1xuICAgICAgICA8aHIgY2xhc3M9XCJzdHItY2hhdF9fZGF0ZS1zZXBhcmF0b3ItbGluZVwiIC8+XG4gICAgICB9XG4gICAgICA8ZGl2IGNsYXNzPVwic3RyLWNoYXRfX2RhdGUtc2VwYXJhdG9yLWRhdGVcIj5cbiAgICAgICAge3sgcGFyc2VkRGF0ZSB9fVxuICAgICAgPC9kaXY+XG4gICAgICBAaWYgKFxuICAgICAgICBkYXRlU2VwYXJhdG9yVGV4dFBvcyA9PT0gXCJsZWZ0XCIgfHwgZGF0ZVNlcGFyYXRvclRleHRQb3MgPT09IFwiY2VudGVyXCJcbiAgICAgICkge1xuICAgICAgICA8aHIgY2xhc3M9XCJzdHItY2hhdF9fZGF0ZS1zZXBhcmF0b3ItbGluZVwiIC8+XG4gICAgICB9XG4gICAgPC9kaXY+XG4gIDwvbmctdGVtcGxhdGU+XG48L25nLXRlbXBsYXRlPlxuXG48bmctdGVtcGxhdGUgI2RlZmF1bHROZXdNZXNzYWdlc0luZGljYXRvciBsZXQtdW5yZWFkQ291bnQ9XCJ1bnJlYWRDb3VudFwiPlxuICA8ZGl2IGNsYXNzPVwic3RyLWNoYXRfX3VucmVhZC1tZXNzYWdlcy1zZXBhcmF0b3JcIj5cbiAgICBAaWYgKHVucmVhZENvdW50ID4gMCAmJiAhaGlkZVVucmVhZENvdW50Rm9yTm90aWZpY2F0aW9uQW5kSW5kaWNhdG9yKSB7XG4gICAgICB7e1xuICAgICAgICAodW5yZWFkQ291bnQgPT09IDFcbiAgICAgICAgICA/IFwic3RyZWFtQ2hhdC5cXHtcXHtjb3VudFxcfVxcfSB1bnJlYWQgbWVzc2FnZVwiXG4gICAgICAgICAgOiBcInN0cmVhbUNoYXQuXFx7XFx7Y291bnRcXH1cXH0gdW5yZWFkIG1lc3NhZ2VzXCJcbiAgICAgICAgKSB8IHRyYW5zbGF0ZTogeyBjb3VudDogdW5yZWFkQ291bnQgfVxuICAgICAgfX1cbiAgICB9IEBlbHNlIHtcbiAgICAgIHt7IFwic3RyZWFtQ2hhdC5VbnJlYWQgbWVzc2FnZXNcIiB8IHRyYW5zbGF0ZSB9fVxuICAgIH1cbiAgPC9kaXY+XG48L25nLXRlbXBsYXRlPlxuIl19