http-request-manager 18.4.5 → 18.4.8

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 (204) hide show
  1. package/fesm2022/http-request-manager.mjs +372 -126
  2. package/fesm2022/http-request-manager.mjs.map +1 -1
  3. package/http-request-manager-18.4.8.tgz +0 -0
  4. package/package.json +3 -5
  5. package/types/http-request-manager.d.ts +1732 -0
  6. package/esm2022/http-request-manager.mjs +0 -5
  7. package/esm2022/lib/http-request-manager.module.mjs +0 -162
  8. package/esm2022/lib/http-request-services-demo/database-data-demo/database-data-demo.component.mjs +0 -152
  9. package/esm2022/lib/http-request-services-demo/http-request-services-demo.component.mjs +0 -61
  10. package/esm2022/lib/http-request-services-demo/local-storage-demo/local-storage-demo.component.mjs +0 -173
  11. package/esm2022/lib/http-request-services-demo/local-storage-signals-demo/local-storage-signals-demo.component.mjs +0 -184
  12. package/esm2022/lib/http-request-services-demo/request-manager-demo/file-downloader/download-file/download-file.component.mjs +0 -80
  13. package/esm2022/lib/http-request-services-demo/request-manager-demo/file-downloader/file-download.module.mjs +0 -42
  14. package/esm2022/lib/http-request-services-demo/request-manager-demo/file-downloader/file-downloader.component.mjs +0 -88
  15. package/esm2022/lib/http-request-services-demo/request-manager-demo/file-downloader/models/download-labels-model.mjs +0 -11
  16. package/esm2022/lib/http-request-services-demo/request-manager-demo/file-downloader/spinner/spinner.component.mjs +0 -29
  17. package/esm2022/lib/http-request-services-demo/request-manager-demo/models/sample-ai-prompt.mjs +0 -9
  18. package/esm2022/lib/http-request-services-demo/request-manager-demo/models/sample-client-info.mjs +0 -12
  19. package/esm2022/lib/http-request-services-demo/request-manager-demo/models/sample-mapper-client-info.mjs +0 -14
  20. package/esm2022/lib/http-request-services-demo/request-manager-demo/request-manager-demo.component.mjs +0 -325
  21. package/esm2022/lib/http-request-services-demo/request-manager-state-demo/request-manager-state-demo.component.mjs +0 -277
  22. package/esm2022/lib/http-request-services-demo/request-manager-state-demo/services/state-manager-demo.service.mjs +0 -64
  23. package/esm2022/lib/http-request-services-demo/request-manager-ws-demo/models/oidc-client.model.mjs +0 -13
  24. package/esm2022/lib/http-request-services-demo/request-manager-ws-demo/models/user-data.model.mjs +0 -13
  25. package/esm2022/lib/http-request-services-demo/request-manager-ws-demo/request-manager-ws-demo.component.mjs +0 -136
  26. package/esm2022/lib/http-request-services-demo/request-manager-ws-demo/services/state-request-demo.service.mjs +0 -97
  27. package/esm2022/lib/http-request-services-demo/request-signals-manager-demo/models/sample-ai-prompt.mjs +0 -9
  28. package/esm2022/lib/http-request-services-demo/request-signals-manager-demo/models/sample-client-info.mjs +0 -12
  29. package/esm2022/lib/http-request-services-demo/request-signals-manager-demo/models/sample-mapper-client-info.mjs +0 -14
  30. package/esm2022/lib/http-request-services-demo/request-signals-manager-demo/request-signals-manager-demo.component.mjs +0 -336
  31. package/esm2022/lib/http-request-services-demo/store-state-manager-demo/models/settings.model.mjs +0 -12
  32. package/esm2022/lib/http-request-services-demo/store-state-manager-demo/services/settings-state.service.mjs +0 -36
  33. package/esm2022/lib/http-request-services-demo/store-state-manager-demo/store-state-manager-demo.component.mjs +0 -34
  34. package/esm2022/lib/index.mjs +0 -4
  35. package/esm2022/lib/interceptors/credentials.interceptor.mjs +0 -14
  36. package/esm2022/lib/interceptors/index.mjs +0 -5
  37. package/esm2022/lib/interceptors/models/error-settings.model.mjs +0 -10
  38. package/esm2022/lib/interceptors/models/index.mjs +0 -2
  39. package/esm2022/lib/interceptors/proxy-debugger.interceptor.mjs +0 -47
  40. package/esm2022/lib/interceptors/request-error.interceptor.mjs +0 -49
  41. package/esm2022/lib/interceptors/request-header.interceptor.mjs +0 -41
  42. package/esm2022/lib/models/config-http-options.model.mjs +0 -18
  43. package/esm2022/lib/models/config-local-storage-options.model.mjs +0 -12
  44. package/esm2022/lib/models/config-options.model.mjs +0 -12
  45. package/esm2022/lib/models/config-token.model.mjs +0 -8
  46. package/esm2022/lib/models/data-type.enum.mjs +0 -7
  47. package/esm2022/lib/models/database-storage.model.mjs +0 -10
  48. package/esm2022/lib/models/index.mjs +0 -8
  49. package/esm2022/lib/models/retry-options.model.mjs +0 -10
  50. package/esm2022/lib/services/database-manager-service/database.manager.service.mjs +0 -171
  51. package/esm2022/lib/services/database-manager-service/db.storage.service.mjs +0 -183
  52. package/esm2022/lib/services/database-manager-service/index.mjs +0 -4
  53. package/esm2022/lib/services/database-manager-service/models/index.mjs +0 -2
  54. package/esm2022/lib/services/database-manager-service/models/table-schema.mjs +0 -19
  55. package/esm2022/lib/services/index.mjs +0 -9
  56. package/esm2022/lib/services/local-storage-manager-service/index.mjs +0 -4
  57. package/esm2022/lib/services/local-storage-manager-service/local-storage-manager.service.mjs +0 -308
  58. package/esm2022/lib/services/local-storage-manager-service/local-storage-signals-manager.service.mjs +0 -277
  59. package/esm2022/lib/services/local-storage-manager-service/models/global-store-options.model.mjs +0 -13
  60. package/esm2022/lib/services/local-storage-manager-service/models/index.mjs +0 -6
  61. package/esm2022/lib/services/local-storage-manager-service/models/setting-options.model.mjs +0 -17
  62. package/esm2022/lib/services/local-storage-manager-service/models/storage-data.model.mjs +0 -10
  63. package/esm2022/lib/services/local-storage-manager-service/models/storage-option.model.mjs +0 -13
  64. package/esm2022/lib/services/local-storage-manager-service/models/storage-type.enum.mjs +0 -7
  65. package/esm2022/lib/services/request-manager-services/http-manager-signals.service.mjs +0 -199
  66. package/esm2022/lib/services/request-manager-services/http-manager.service.mjs +0 -207
  67. package/esm2022/lib/services/request-manager-services/index.mjs +0 -7
  68. package/esm2022/lib/services/request-manager-services/request-signals.service.mjs +0 -170
  69. package/esm2022/lib/services/request-manager-services/request.service.mjs +0 -192
  70. package/esm2022/lib/services/request-manager-services/rxjs-operators/countdown.mjs +0 -9
  71. package/esm2022/lib/services/request-manager-services/rxjs-operators/delay-retry.mjs +0 -10
  72. package/esm2022/lib/services/request-manager-services/rxjs-operators/index.mjs +0 -5
  73. package/esm2022/lib/services/request-manager-services/rxjs-operators/request-polling.mjs +0 -33
  74. package/esm2022/lib/services/request-manager-services/rxjs-operators/request-streaming.mjs +0 -19
  75. package/esm2022/lib/services/request-manager-state-service/http-manager-state.store.mjs +0 -592
  76. package/esm2022/lib/services/request-manager-state-service/index.mjs +0 -3
  77. package/esm2022/lib/services/request-manager-state-service/models/api-request.model.mjs +0 -22
  78. package/esm2022/lib/services/request-manager-state-service/models/index.mjs +0 -5
  79. package/esm2022/lib/services/request-manager-state-service/models/request-options.model.mjs +0 -10
  80. package/esm2022/lib/services/request-manager-state-service/models/ws-options.model.mjs +0 -15
  81. package/esm2022/lib/services/request-manager-state-service/models/ws-user.model.mjs +0 -10
  82. package/esm2022/lib/services/store-state-manager-service/index.mjs +0 -3
  83. package/esm2022/lib/services/store-state-manager-service/models/index.mjs +0 -2
  84. package/esm2022/lib/services/store-state-manager-service/models/state-storage-options.model.mjs +0 -12
  85. package/esm2022/lib/services/store-state-manager-service/store-state-manager.service.mjs +0 -71
  86. package/esm2022/lib/services/utils/app.service.mjs +0 -26
  87. package/esm2022/lib/services/utils/encryption/asymmetrical-encryption.service.mjs +0 -186
  88. package/esm2022/lib/services/utils/encryption/encryption-test.service.mjs +0 -35
  89. package/esm2022/lib/services/utils/encryption/index.mjs +0 -4
  90. package/esm2022/lib/services/utils/encryption/random.mjs +0 -52
  91. package/esm2022/lib/services/utils/encryption/symmetrical-encryption.service.mjs +0 -77
  92. package/esm2022/lib/services/utils/headers.service.mjs +0 -21
  93. package/esm2022/lib/services/utils/index.mjs +0 -6
  94. package/esm2022/lib/services/utils/object-merger.service.mjs +0 -70
  95. package/esm2022/lib/services/utils/path-query.service.mjs +0 -54
  96. package/esm2022/lib/services/utils/utils.service.mjs +0 -155
  97. package/esm2022/lib/services/ws-manager-service/index.mjs +0 -3
  98. package/esm2022/lib/services/ws-manager-service/models/channel-info.model.mjs +0 -10
  99. package/esm2022/lib/services/ws-manager-service/models/channel-notification.model.mjs +0 -14
  100. package/esm2022/lib/services/ws-manager-service/models/communication-type.enum.mjs +0 -7
  101. package/esm2022/lib/services/ws-manager-service/models/index.mjs +0 -5
  102. package/esm2022/lib/services/ws-manager-service/models/user-message.model.mjs +0 -12
  103. package/esm2022/lib/services/ws-manager-service/services/websocket.service.mjs +0 -178
  104. package/esm2022/public-api.mjs +0 -11
  105. package/http-request-manager-18.4.5.tgz +0 -0
  106. package/index.d.ts +0 -5
  107. package/lib/http-request-manager.module.d.ts +0 -39
  108. package/lib/http-request-services-demo/database-data-demo/database-data-demo.component.d.ts +0 -46
  109. package/lib/http-request-services-demo/http-request-services-demo.component.d.ts +0 -37
  110. package/lib/http-request-services-demo/local-storage-demo/local-storage-demo.component.d.ts +0 -56
  111. package/lib/http-request-services-demo/local-storage-signals-demo/local-storage-signals-demo.component.d.ts +0 -55
  112. package/lib/http-request-services-demo/request-manager-demo/file-downloader/download-file/download-file.component.d.ts +0 -26
  113. package/lib/http-request-services-demo/request-manager-demo/file-downloader/file-download.module.d.ts +0 -13
  114. package/lib/http-request-services-demo/request-manager-demo/file-downloader/file-downloader.component.d.ts +0 -27
  115. package/lib/http-request-services-demo/request-manager-demo/file-downloader/models/download-labels-model.d.ts +0 -12
  116. package/lib/http-request-services-demo/request-manager-demo/file-downloader/spinner/spinner.component.d.ts +0 -16
  117. package/lib/http-request-services-demo/request-manager-demo/models/sample-ai-prompt.d.ts +0 -8
  118. package/lib/http-request-services-demo/request-manager-demo/models/sample-client-info.d.ts +0 -14
  119. package/lib/http-request-services-demo/request-manager-demo/models/sample-mapper-client-info.d.ts +0 -14
  120. package/lib/http-request-services-demo/request-manager-demo/request-manager-demo.component.d.ts +0 -110
  121. package/lib/http-request-services-demo/request-manager-state-demo/request-manager-state-demo.component.d.ts +0 -124
  122. package/lib/http-request-services-demo/request-manager-state-demo/services/state-manager-demo.service.d.ts +0 -15
  123. package/lib/http-request-services-demo/request-manager-ws-demo/models/oidc-client.model.d.ts +0 -16
  124. package/lib/http-request-services-demo/request-manager-ws-demo/models/user-data.model.d.ts +0 -14
  125. package/lib/http-request-services-demo/request-manager-ws-demo/request-manager-ws-demo.component.d.ts +0 -42
  126. package/lib/http-request-services-demo/request-manager-ws-demo/services/state-request-demo.service.d.ts +0 -15
  127. package/lib/http-request-services-demo/request-signals-manager-demo/models/sample-ai-prompt.d.ts +0 -8
  128. package/lib/http-request-services-demo/request-signals-manager-demo/models/sample-client-info.d.ts +0 -14
  129. package/lib/http-request-services-demo/request-signals-manager-demo/models/sample-mapper-client-info.d.ts +0 -14
  130. package/lib/http-request-services-demo/request-signals-manager-demo/request-signals-manager-demo.component.d.ts +0 -106
  131. package/lib/http-request-services-demo/store-state-manager-demo/models/settings.model.d.ts +0 -14
  132. package/lib/http-request-services-demo/store-state-manager-demo/services/settings-state.service.d.ts +0 -11
  133. package/lib/http-request-services-demo/store-state-manager-demo/store-state-manager-demo.component.d.ts +0 -14
  134. package/lib/index.d.ts +0 -3
  135. package/lib/interceptors/credentials.interceptor.d.ts +0 -8
  136. package/lib/interceptors/index.d.ts +0 -4
  137. package/lib/interceptors/models/error-settings.model.d.ts +0 -10
  138. package/lib/interceptors/models/index.d.ts +0 -1
  139. package/lib/interceptors/proxy-debugger.interceptor.d.ts +0 -12
  140. package/lib/interceptors/request-error.interceptor.d.ts +0 -10
  141. package/lib/interceptors/request-header.interceptor.d.ts +0 -15
  142. package/lib/models/config-http-options.model.d.ts +0 -21
  143. package/lib/models/config-local-storage-options.model.d.ts +0 -13
  144. package/lib/models/config-options.model.d.ts +0 -12
  145. package/lib/models/config-token.model.d.ts +0 -8
  146. package/lib/models/data-type.enum.d.ts +0 -5
  147. package/lib/models/database-storage.model.d.ts +0 -10
  148. package/lib/models/index.d.ts +0 -7
  149. package/lib/models/retry-options.model.d.ts +0 -10
  150. package/lib/services/database-manager-service/database.manager.service.d.ts +0 -33
  151. package/lib/services/database-manager-service/db.storage.service.d.ts +0 -26
  152. package/lib/services/database-manager-service/index.d.ts +0 -3
  153. package/lib/services/database-manager-service/models/index.d.ts +0 -1
  154. package/lib/services/database-manager-service/models/table-schema.d.ts +0 -11
  155. package/lib/services/index.d.ts +0 -7
  156. package/lib/services/local-storage-manager-service/index.d.ts +0 -3
  157. package/lib/services/local-storage-manager-service/local-storage-manager.service.d.ts +0 -86
  158. package/lib/services/local-storage-manager-service/local-storage-signals-manager.service.d.ts +0 -64
  159. package/lib/services/local-storage-manager-service/models/global-store-options.model.d.ts +0 -15
  160. package/lib/services/local-storage-manager-service/models/index.d.ts +0 -5
  161. package/lib/services/local-storage-manager-service/models/setting-options.model.d.ts +0 -15
  162. package/lib/services/local-storage-manager-service/models/storage-data.model.d.ts +0 -10
  163. package/lib/services/local-storage-manager-service/models/storage-option.model.d.ts +0 -15
  164. package/lib/services/local-storage-manager-service/models/storage-type.enum.d.ts +0 -5
  165. package/lib/services/request-manager-services/http-manager-signals.service.d.ts +0 -38
  166. package/lib/services/request-manager-services/http-manager.service.d.ts +0 -41
  167. package/lib/services/request-manager-services/index.d.ts +0 -6
  168. package/lib/services/request-manager-services/request-signals.service.d.ts +0 -26
  169. package/lib/services/request-manager-services/request.service.d.ts +0 -28
  170. package/lib/services/request-manager-services/rxjs-operators/countdown.d.ts +0 -2
  171. package/lib/services/request-manager-services/rxjs-operators/delay-retry.d.ts +0 -2
  172. package/lib/services/request-manager-services/rxjs-operators/index.d.ts +0 -4
  173. package/lib/services/request-manager-services/rxjs-operators/request-polling.d.ts +0 -2
  174. package/lib/services/request-manager-services/rxjs-operators/request-streaming.d.ts +0 -2
  175. package/lib/services/request-manager-state-service/http-manager-state.store.d.ts +0 -90
  176. package/lib/services/request-manager-state-service/index.d.ts +0 -2
  177. package/lib/services/request-manager-state-service/models/api-request.model.d.ts +0 -32
  178. package/lib/services/request-manager-state-service/models/index.d.ts +0 -4
  179. package/lib/services/request-manager-state-service/models/request-options.model.d.ts +0 -10
  180. package/lib/services/request-manager-state-service/models/ws-options.model.d.ts +0 -22
  181. package/lib/services/request-manager-state-service/models/ws-user.model.d.ts +0 -10
  182. package/lib/services/store-state-manager-service/index.d.ts +0 -2
  183. package/lib/services/store-state-manager-service/models/index.d.ts +0 -1
  184. package/lib/services/store-state-manager-service/models/state-storage-options.model.d.ts +0 -13
  185. package/lib/services/store-state-manager-service/store-state-manager.service.d.ts +0 -23
  186. package/lib/services/utils/app.service.d.ts +0 -8
  187. package/lib/services/utils/encryption/asymmetrical-encryption.service.d.ts +0 -17
  188. package/lib/services/utils/encryption/encryption-test.service.d.ts +0 -10
  189. package/lib/services/utils/encryption/index.d.ts +0 -3
  190. package/lib/services/utils/encryption/random.d.ts +0 -7
  191. package/lib/services/utils/encryption/symmetrical-encryption.service.d.ts +0 -14
  192. package/lib/services/utils/headers.service.d.ts +0 -10
  193. package/lib/services/utils/index.d.ts +0 -5
  194. package/lib/services/utils/object-merger.service.d.ts +0 -14
  195. package/lib/services/utils/path-query.service.d.ts +0 -11
  196. package/lib/services/utils/utils.service.d.ts +0 -24
  197. package/lib/services/ws-manager-service/index.d.ts +0 -2
  198. package/lib/services/ws-manager-service/models/channel-info.model.d.ts +0 -10
  199. package/lib/services/ws-manager-service/models/channel-notification.model.d.ts +0 -17
  200. package/lib/services/ws-manager-service/models/communication-type.enum.d.ts +0 -5
  201. package/lib/services/ws-manager-service/models/index.d.ts +0 -4
  202. package/lib/services/ws-manager-service/models/user-message.model.d.ts +0 -14
  203. package/lib/services/ws-manager-service/services/websocket.service.d.ts +0 -24
  204. package/public-api.d.ts +0 -7
@@ -1,592 +0,0 @@
1
- import { inject, Inject, Injectable, InjectionToken } from '@angular/core';
2
- import { ComponentStore } from '@ngrx/component-store';
3
- import { BehaviorSubject, of, Subject, timer, merge } from 'rxjs';
4
- import { tap, switchMap, concatMap, scan, delay, take, map, distinctUntilChanged, takeUntil, takeWhile, filter } from 'rxjs/operators';
5
- import { DatabaseStorage } from '../../models/database-storage.model';
6
- import { ApiRequest, RequestOptions, WSOptions } from './models';
7
- import { ChannelMessage } from '../ws-manager-service/models';
8
- import { HTTPManagerService, DataType } from '../request-manager-services';
9
- import { DatabaseManagerService } from '../database-manager-service';
10
- import { TableSchemaDef } from '../database-manager-service/models/table-schema';
11
- import { LocalStorageManagerService, StorageType } from '../local-storage-manager-service';
12
- import { SettingOptions } from '../local-storage-manager-service/models/setting-options.model';
13
- import { UtilsService } from '../..';
14
- import * as i0 from "@angular/core";
15
- const API_OPTS = new InjectionToken('API_OPTS');
16
- const defaultState = {
17
- data: [],
18
- dataObject: null,
19
- };
20
- export class HTTPManagerStateService extends ComponentStore {
21
- constructor(apiOptions = ApiRequest.adapt(), dataType, database) {
22
- super(defaultState);
23
- this.apiOptions = apiOptions;
24
- this.dataType = dataType;
25
- this.database = database;
26
- this.httpManagerService = inject(HTTPManagerService);
27
- this.dbManagerService = inject(DatabaseManagerService);
28
- this.localStorageManagerService = inject(LocalStorageManagerService);
29
- this.utils = inject(UtilsService);
30
- this.error$ = this.httpManagerService.error$;
31
- this.isPending$ = this.httpManagerService.isPending$.pipe(delay(1));
32
- // PAGINATION
33
- this.page = new BehaviorSubject(0);
34
- this.page$ = this.page.asObservable();
35
- this.totalPages = new BehaviorSubject(0);
36
- this.totalPages$ = this.totalPages.asObservable();
37
- this.percentage = new BehaviorSubject(0);
38
- this.percentage$ = this.percentage.asObservable();
39
- this.hasDatabase = false;
40
- this.streamedResponse = [];
41
- this.shouldRetry = true;
42
- this.wsRetryAttempts = new BehaviorSubject(0);
43
- this.wsRetryAttempts$ = this.wsRetryAttempts.asObservable();
44
- this.messages$ = null;
45
- this.connectionStatus$ = null;
46
- this.userList = new BehaviorSubject([]);
47
- this.userList$ = this.userList.asObservable();
48
- this.user = new BehaviorSubject(null);
49
- this.user$ = this.user.asObservable();
50
- this.communicationMessages = new BehaviorSubject([]);
51
- this.communicationMessages$ = this.communicationMessages.asObservable();
52
- this.latestCommunicationMessages = new BehaviorSubject(null);
53
- this.latestCommunicationMessages$ = this.latestCommunicationMessages.asObservable();
54
- this.channels = [];
55
- this.channelList = [];
56
- this.messages = [];
57
- this.wsConnection = false;
58
- this.wsOptions = WSOptions.adapt();
59
- // WebSocket
60
- this.initWS = this.effect((wsOptions$) => wsOptions$.pipe(tap((wsOptions) => {
61
- this.wsOptions = wsOptions;
62
- if (wsOptions?.wsServer)
63
- this.httpManagerService.connect(wsOptions.wsServer, wsOptions.jwtToken);
64
- }), switchMap((wsOptions) => merge(this.httpManagerService.connectionStatus$.pipe(tap((isConnected) => {
65
- this.wsConnection = isConnected;
66
- if (isConnected) {
67
- console.log('🟢 WebSocket connection is open.');
68
- }
69
- else {
70
- console.log('🔴 WebSocket connection is closed.');
71
- }
72
- })), this.httpManagerService.messages$.pipe(tap((message) => {
73
- if (!message)
74
- return;
75
- console.log('Received:', message.type);
76
- if (message.error === 'JWT_INVALID') {
77
- this.shouldRetry = false;
78
- this.httpManagerService.disconnect();
79
- }
80
- if (message.type === 'success') {
81
- const currentUser = this.apiOptions.ws?.userAdapter?.adapter?.(message.data) || message.data;
82
- this.user.next(currentUser);
83
- }
84
- switch (message.type) {
85
- case 'channelsList':
86
- console.log('💬 Channels:', message.channels);
87
- this.channelList = message.channels;
88
- if (this.channelList.includes(wsOptions.id)) {
89
- this.httpManagerService.subscribeToChannel(wsOptions.id);
90
- }
91
- else {
92
- this.httpManagerService.createChannel(wsOptions.id);
93
- }
94
- break;
95
- case 'stateMangerMessage':
96
- console.log('💬 Message:', message.content);
97
- this.fetchRecord(RequestOptions.adapt({ path: message.content.path }), message.content.method);
98
- break;
99
- case 'channelMessage':
100
- console.log('💬 Message:', message.content);
101
- this.fetchRecord(RequestOptions.adapt({ path: message.content.path }), message.content.method);
102
- break;
103
- case 'channelCommunication':
104
- console.log('💬 Message Communication:', message.content);
105
- this.appendMessages(message.content);
106
- break;
107
- case 'channelAlerts':
108
- console.log('💬 Message Alerts:', message.content);
109
- break;
110
- case 'usersInChannel':
111
- const data = message.data;
112
- const wsUsers = data.users.map((item) => this.apiOptions.ws?.userAdapter?.adapter?.(item) || item);
113
- const userList = wsUsers.filter((item) => item.username !== this.user.value[this.wsOptions.userAdapter?.id || 'id']);
114
- this.userList.next(userList);
115
- console.log('👥 Users:', userList);
116
- break;
117
- default:
118
- if (message)
119
- this.messages.push(message);
120
- break;
121
- }
122
- }))))));
123
- this.initDBStorage = this.effect((trigger$) => trigger$.pipe(tap(() => {
124
- if (this.dataType !== DataType.ARRAY)
125
- console.warn('Database storage requires dataType to be ARRAY');
126
- if (!this.apiOptions.adapter)
127
- console.warn('Database storage requires an adapter to define the data shape');
128
- if (!this.database?.table)
129
- console.warn('Database storage requires a table name');
130
- }), filter(() => this.dataType === DataType.ARRAY && !!this.apiOptions.adapter && !!this.database?.table), switchMap(() => {
131
- const sampleData = this.apiOptions.adapter?.({}) || {};
132
- const schemaKeys = Object.keys(sampleData).filter(key => sampleData[key] !== undefined);
133
- let schema = '++id';
134
- if (schemaKeys.length > 0) {
135
- const otherKeys = schemaKeys.filter(k => k !== 'id');
136
- if (otherKeys.length > 0) {
137
- schema += ', ' + otherKeys.join(', ');
138
- }
139
- }
140
- const tableDef = TableSchemaDef.adapt({
141
- table: this.database?.table,
142
- schema: schema
143
- });
144
- return this.dbManagerService.createDatabaseTable(tableDef);
145
- })));
146
- // --------------------------------------------------------------------------------------------------
147
- // SELECTORS
148
- this.data$ = this.select(({ data, dataObject }) => {
149
- const isArray = (this.dataType === DataType.ARRAY) ? true : false;
150
- return (isArray) ? data : dataObject;
151
- });
152
- this.selectRecord$ = (id) => this.select(this.data$, (data) => {
153
- if (this.dataType === DataType.ARRAY && Array.isArray(data)) {
154
- return data.find(item => item.id === id);
155
- }
156
- else {
157
- return data.id === id ? data : null;
158
- }
159
- });
160
- // --------------------------------------------------------------------------------------------------
161
- // UPDATERS
162
- this.setData$ = this.updater((state, data) => {
163
- if (!data)
164
- return state;
165
- if (this.dataType === DataType.ARRAY) {
166
- const dataArray = Array.isArray(data) ? data : [data];
167
- const stateDataSample = (state.data.length > 0) ? Object.keys(state.data[0]) : [];
168
- const newDataSample = (dataArray.length > 0) ? Object.keys(dataArray[0]) : [];
169
- const isSame = (state.data.length === 0) ? false : stateDataSample.every((value, index) => value === newDataSample[index]);
170
- const updatedData = (!isSame && dataArray.length !== 0) ? this.updateArrayState([], dataArray) : this.updateArrayState(state.data, dataArray);
171
- return { ...state, data: updatedData, dataObject: null };
172
- }
173
- else {
174
- const dataObject = this.isEmpty(data) ? null : data;
175
- return { ...state, data: [], dataObject: dataObject };
176
- }
177
- });
178
- this.addData$ = this.updater((state, data) => {
179
- if (this.dataType === DataType.ARRAY) {
180
- const exists = state.data.some(item => item.id === data.id);
181
- if (exists) {
182
- const updatedData = state.data.map(item => item.id === data.id ? data : item);
183
- return { ...state, data: updatedData };
184
- }
185
- else {
186
- const newState = [...state.data, data];
187
- return { ...state, data: newState };
188
- }
189
- }
190
- else {
191
- return { ...state, dataObject: data };
192
- }
193
- });
194
- this.deleteData$ = this.updater((state, data) => {
195
- if (this.dataType === DataType.ARRAY) {
196
- const newState = state.data.filter(item => item.id !== data.id);
197
- return { ...state, ...{ data: newState } };
198
- }
199
- else {
200
- return { ...state, ...{ dataObject: null } };
201
- }
202
- });
203
- this.updateData$ = this.updater((state, data) => {
204
- if (this.dataType === DataType.ARRAY) {
205
- const objIndex = state.data.findIndex(item => item.id === data.id);
206
- if (objIndex > -1) {
207
- const newState = [...state.data];
208
- newState[objIndex] = data;
209
- return { ...state, ...{ data: newState } };
210
- }
211
- return state;
212
- }
213
- else {
214
- return { ...state, ...{ dataObject: data } };
215
- }
216
- });
217
- // --------------------------------------------------------------------------------------------------
218
- // EFFECTS
219
- this.clearRecords = this.effect(data => data.pipe(tap(() => {
220
- if (this.dataType === DataType.ARRAY) {
221
- this.setData$([]);
222
- }
223
- else {
224
- this.setData$({});
225
- }
226
- }), concatMap(() => {
227
- if (this.hasDatabase && this.database?.table) {
228
- const currentData = this.get()?.data;
229
- const idsToDelete = Array.isArray(currentData) ? currentData.map((r) => r.id) : [];
230
- if (idsToDelete.length > 0) {
231
- return this.dbManagerService.deleteTableRecords(this.database.table, idsToDelete);
232
- }
233
- }
234
- return of(null);
235
- })));
236
- // --------------------------------------------------------------------------------------------------
237
- // CRUD OPERATIONS
238
- // FETCH RECORDS
239
- this.fetchRecords = (options) => this.effect(() => of(RequestOptions.adapt(options)).pipe(switchMap(() => {
240
- this.streamedResponse = [];
241
- const requestOptions = this.updateRequestOptions(options?.headers);
242
- const fetchFromAPI = () => {
243
- return this.httpManagerService.getRequest(requestOptions, options?.path).pipe(tap((data) => {
244
- data = (!data) ? (this.dataType === DataType.ARRAY) ? [] : {} : data;
245
- this.setData$(data);
246
- }), concatMap((data) => {
247
- if (this.hasDatabase && this.database?.table && Array.isArray(data) && data.length > 0) {
248
- this.localStorageManagerService.updateStore({
249
- name: this.database.table,
250
- data: { ...this.database, ...{ expires: this.utils.expires(this.database.expiresIn) } }
251
- });
252
- return this.dbManagerService.createTableRecords(this.database.table, data);
253
- }
254
- return of(data);
255
- }));
256
- };
257
- if (this.hasDatabase && this.database?.table) {
258
- return this.dbManagerService.databaseExists().pipe(switchMap((dbExists) => {
259
- if (!dbExists) {
260
- const initObs = this.initDBStorageAsync();
261
- return initObs.pipe(switchMap(() => fetchFromAPI()));
262
- }
263
- return this.dbManagerService.hasDatabaseTable(this.database.table).pipe(switchMap((tableExists) => {
264
- if (!tableExists) {
265
- const initObs = this.initDBStorageAsync();
266
- return initObs.pipe(switchMap(() => fetchFromAPI()));
267
- }
268
- return this.localStorageManagerService.store$(this.database.table).pipe(take(1), switchMap((storeData) => {
269
- const expires = storeData?.expires || 0;
270
- const hasExpired = expires > 0 && this.utils.hasExpired(expires);
271
- if (hasExpired) {
272
- return this.dbManagerService.clearTable(this.database.table).pipe(switchMap(() => fetchFromAPI()), tap(() => {
273
- this.localStorageManagerService.updateStore({
274
- name: this.database.table,
275
- data: { ...this.database, ...{ expires: this.utils.expires(this.database.expiresIn) } }
276
- });
277
- }));
278
- }
279
- return this.dbManagerService.getTableRecords(this.database.table).pipe(switchMap((dbData) => {
280
- if (Array.isArray(dbData) && dbData.length > 0) {
281
- this.setData$(dbData);
282
- return of(dbData);
283
- }
284
- return fetchFromAPI();
285
- }));
286
- }));
287
- }));
288
- }));
289
- }
290
- return fetchFromAPI();
291
- })));
292
- // FETCH RECORD
293
- this.fetchRecord = (options, method) => this.effect(() => of(RequestOptions.adapt(options)).pipe(switchMap((options) => {
294
- this.streamedResponse = [];
295
- const requestOptions = this.updateRequestOptions(options?.headers);
296
- return this.httpManagerService.getRequest(requestOptions, options?.path)
297
- .pipe(tap((data) => {
298
- data = (!data) ? (this.dataType === DataType.ARRAY) ? [] : {} : data;
299
- const id = options.path?.length ? options.path[options.path.length - 1] : null;
300
- if (method === 'DELETE')
301
- this.deleteData$({ id });
302
- if (method === 'UPDATE')
303
- this.updateData$(data);
304
- if (method === 'CREATE')
305
- this.addData$(data);
306
- }), concatMap((data) => {
307
- if (this.hasDatabase && this.database?.table) {
308
- const id = options.path?.length ? options.path[options.path.length - 1] : null;
309
- if (method === 'DELETE' && id)
310
- return this.dbManagerService.deleteTableRecord(this.database.table, id);
311
- if (method === 'UPDATE' && data)
312
- return this.dbManagerService.updateTableRecord(this.database.table, data);
313
- if (method === 'CREATE' && data)
314
- return this.dbManagerService.createTableRecord(this.database.table, data);
315
- }
316
- return of(data);
317
- }));
318
- })));
319
- // CREATE RECORD
320
- this.createRecord = (data, options) => this.effect(() => of(data).pipe(switchMap((data) => {
321
- this.streamedResponse = [];
322
- const requestOptions = this.updateRequestOptions(options?.headers);
323
- return this.httpManagerService.postRequest(data, requestOptions, options?.path)
324
- .pipe(tap((data) => {
325
- data = (!data) ? (this.dataType === DataType.ARRAY) ? [] : {} : data;
326
- this.addData$(data);
327
- if (this.wsConnection)
328
- this.wsCommunication('CREATE', [...options?.path || [], data.id]);
329
- }), concatMap((data) => {
330
- if (this.hasDatabase && this.database?.table && data?.id) {
331
- return this.dbManagerService.createTableRecord(this.database.table, data);
332
- }
333
- return of(data);
334
- }));
335
- })));
336
- // UPDATE RECORD
337
- this.updateRecord = (data, options) => this.effect(() => of(data).pipe(concatMap((data) => {
338
- this.streamedResponse = [];
339
- const requestOptions = this.updateRequestOptions(options?.headers);
340
- return this.httpManagerService.putRequest(data, requestOptions, options?.path)
341
- .pipe(tap((data) => {
342
- data = (!data) ? (this.dataType === DataType.ARRAY) ? [] : {} : data;
343
- this.updateData$(data);
344
- if (this.wsConnection)
345
- this.wsCommunication('UPDATE', [...options?.path || []]);
346
- }), concatMap((data) => {
347
- if (this.hasDatabase && this.database?.table && data?.id) {
348
- return this.dbManagerService.updateTableRecord(this.database.table, data);
349
- }
350
- return of(data);
351
- }));
352
- })));
353
- // DELETE RECORD
354
- this.deleteRecord = (options) => this.effect(() => of(options).pipe(concatMap((data) => {
355
- this.streamedResponse = [];
356
- const requestOptions = this.updateRequestOptions(options?.headers);
357
- return this.httpManagerService.deleteRequest(requestOptions, options?.path)
358
- .pipe(tap((data) => {
359
- data = (!data) ? (this.dataType === DataType.ARRAY) ? [] : {} : data;
360
- this.deleteData$(data);
361
- if (this.wsConnection)
362
- this.wsCommunication('DELETE', [...options?.path || []]);
363
- }), concatMap((data) => {
364
- if (this.hasDatabase && this.database?.table && data?.id) {
365
- return this.dbManagerService.deleteTableRecord(this.database.table, data.id);
366
- }
367
- return of(data);
368
- }));
369
- })));
370
- // --------------------------------------------------------------------------------------------------
371
- // FETCH STREAM
372
- this.createStream = (data, options) => this.effect(() => of(data).pipe(tap(() => this.httpManagerService.isPending.next(true)), switchMap((data) => {
373
- const requestOptions = this.updateRequestOptions(options?.headers);
374
- return this.httpManagerService.postRequest(data, requestOptions, options?.path)
375
- .pipe(tap((res) => {
376
- if (res.length > 0)
377
- this.setData$(res);
378
- this.streamedResponse = res;
379
- }), scan((acc, res) => {
380
- const previous = acc.current;
381
- const current = res;
382
- return { previous, current };
383
- }, { previous: null, current: null }), tap(({ previous, current }) => {
384
- if (previous && JSON.stringify(previous) === JSON.stringify(current)) {
385
- this.httpManagerService.isPending.next(false);
386
- this.setData$([]);
387
- }
388
- else {
389
- this.httpManagerService.isPending.next(true);
390
- }
391
- }));
392
- })));
393
- this.fetchStream = (options) => this.effect(() => of(options).pipe(tap(() => this.httpManagerService.isPending.next(true)), switchMap((options) => {
394
- const requestOptions = this.updateRequestOptions(options?.headers);
395
- requestOptions.stream = true;
396
- return this.httpManagerService.getRequest(requestOptions, options?.path)
397
- .pipe(tap((res) => {
398
- if (res.length > 0)
399
- this.setData$(res);
400
- this.streamedResponse = res;
401
- }), scan((acc, res) => {
402
- const previous = acc.current;
403
- const current = res;
404
- return { previous, current };
405
- }, { previous: null, current: null }), tap(({ previous, current }) => {
406
- if (previous && JSON.stringify(previous) === JSON.stringify(current)) {
407
- this.httpManagerService.isPending.next(false);
408
- this.setData$([]);
409
- }
410
- else {
411
- this.httpManagerService.isPending.next(true);
412
- }
413
- }));
414
- })));
415
- this.maxRetries = this.apiOptions.ws?.retry?.times || 3;
416
- this.retryDelay = (this.apiOptions.ws?.retry?.delay && this.apiOptions.ws.retry.delay * 1000) || 5 * 1000;
417
- this.wsNextRetry = new BehaviorSubject(this.retryDelay);
418
- this.wsNextRetry$ = this.wsNextRetry.asObservable();
419
- this.setApiRequestOptions(apiOptions, dataType, database);
420
- this.status$ = this.setupConnectionStatus();
421
- if (this.database && this.database.table) {
422
- this.localStorageManagerService.createStore({
423
- name: this.database.table,
424
- data: { ...this.database, ...{ expires: this.utils.expires(this.database.expiresIn) } },
425
- options: SettingOptions.adapt({
426
- storage: StorageType.GLOBAL,
427
- encrypted: false,
428
- })
429
- });
430
- }
431
- }
432
- setApiRequestOptions(apiOptions, dataType, database) {
433
- this.apiOptions = ApiRequest.adapt(apiOptions);
434
- this.dataType = (dataType) ? dataType : DataType.ARRAY;
435
- this.hasDatabase = (this.database?.table) ? true : false;
436
- this.database = (this.hasDatabase) ? DatabaseStorage.adapt(database) : undefined;
437
- if (this.database)
438
- this.initDBStorage();
439
- if (this.apiOptions.ws)
440
- this.initWS(this.apiOptions.ws);
441
- }
442
- // WebSocket
443
- setupConnectionStatus() {
444
- return this.httpManagerService.connectionStatus$.pipe(distinctUntilChanged(), switchMap(status => {
445
- if (status === true) {
446
- this.shouldRetry = true;
447
- this.wsRetryAttempts.next(0);
448
- this.wsNextRetry.next(0);
449
- return of(true);
450
- }
451
- if (!this.shouldRetry)
452
- return of(false);
453
- const countdownEnder$ = new Subject();
454
- return timer(0, this.retryDelay)
455
- .pipe(take(this.maxRetries), tap(i => {
456
- const attempt = i + 1;
457
- this.wsRetryAttempts.next(attempt);
458
- countdownEnder$.next();
459
- if (this.apiOptions.ws?.jwtToken !== '' && this.apiOptions.ws?.wsServer !== '')
460
- this.initWS(this.apiOptions.ws);
461
- if (attempt === this.maxRetries) {
462
- this.wsNextRetry.next(0);
463
- // console.error(`🚨 FAILED CONNECTION: Tried #${attempt} times`);
464
- }
465
- else {
466
- // console.log(`⚠️ Retry Attempt #${attempt}: Retrying in ${this.retryDelay / 1000}s`);
467
- const seconds = this.retryDelay / 1000;
468
- timer(0, 1000).pipe(map(tick => seconds - tick), takeWhile(val => val >= 0), takeUntil(countdownEnder$)).subscribe(remaining => {
469
- this.wsNextRetry.next(remaining);
470
- });
471
- }
472
- }), map(() => false));
473
- }));
474
- }
475
- appendMessages(message) {
476
- const currentMessages = this.communicationMessages.value;
477
- this.communicationMessages.next([...currentMessages, message]);
478
- this.latestMessage();
479
- }
480
- latestMessage() {
481
- const messages = this.communicationMessages.value;
482
- const latestMessage = messages[messages.length - 1];
483
- this.latestCommunicationMessages.next(latestMessage);
484
- }
485
- clearMessages() {
486
- this.communicationMessages.next([]);
487
- }
488
- get ApiRequestOptions() {
489
- return this.apiOptions;
490
- }
491
- initializeState(data) {
492
- this.setData$(data);
493
- }
494
- updateArrayState(currentData, newData) {
495
- const filterCurrentData = () => {
496
- const ids = this.streamedResponse.map((obj) => obj.id);
497
- return currentData.filter(obj => (obj.id) ? ids.includes(obj.id) : obj);
498
- };
499
- const filteredCurrentData = (this.httpManagerService.isPending.value) ? currentData : filterCurrentData();
500
- const updatedData = filteredCurrentData.map(item => {
501
- const newItem = newData.find(newItem => {
502
- const hasId = (newItem?.id && item?.id) ? true : false;
503
- return (hasId) ? newItem.id === item.id : JSON.stringify(newItem) === JSON.stringify(item);
504
- });
505
- return (newItem) ? { ...item, ...newItem } : item;
506
- });
507
- const addedData = newData.filter(newItem => {
508
- return !filteredCurrentData.some(item => {
509
- const hasId = (newItem?.id && item?.id) ? true : false;
510
- return (hasId) ? item.id === newItem.id : JSON.stringify(newItem) === JSON.stringify(item);
511
- });
512
- });
513
- return [...updatedData, ...addedData];
514
- }
515
- initDBStorageAsync() {
516
- if (this.dataType !== DataType.ARRAY) {
517
- console.warn('Database storage requires dataType to be ARRAY');
518
- return of(null);
519
- }
520
- if (!this.apiOptions.adapter) {
521
- console.warn('Database storage requires an adapter to define the data shape');
522
- return of(null);
523
- }
524
- if (!this.database?.table) {
525
- console.warn('Database storage requires a table name');
526
- return of(null);
527
- }
528
- const sampleData = this.apiOptions.adapter?.({}) || {};
529
- const schemaKeys = Object.keys(sampleData).filter(key => sampleData[key] !== undefined);
530
- let schema = '++id';
531
- if (schemaKeys.length > 0) {
532
- const otherKeys = schemaKeys.filter(k => k !== 'id');
533
- if (otherKeys.length > 0) {
534
- schema += ', ' + otherKeys.join(', ');
535
- }
536
- }
537
- const tableDef = TableSchemaDef.adapt({
538
- table: this.database?.table,
539
- schema: schema
540
- });
541
- return this.dbManagerService.createDatabaseTable(tableDef);
542
- }
543
- wsCommunication(method, path) {
544
- if (this.wsConnection && this.apiOptions.ws) {
545
- const wsServer = this.apiOptions.ws.id;
546
- this.httpManagerService.sendMessageInChannel(wsServer, { method, path });
547
- }
548
- }
549
- wsMessaging(message, channel) {
550
- const user = this.wsOptions?.userAdapter?.adapter?.(this.user.value) || this.user.value;
551
- const messageInfo = ChannelMessage.adapt({ ...message, fromUser: user });
552
- if (this.wsConnection && this.apiOptions.ws) {
553
- const wsServer = (channel) ? channel : this.apiOptions.ws.id;
554
- if (messageInfo.toUser === 'allChannels') {
555
- this.httpManagerService.sendBroadcast(messageInfo);
556
- }
557
- else if (messageInfo.toUser === 'allInChannel') {
558
- this.httpManagerService.sendMessageInChannel(wsServer, messageInfo);
559
- }
560
- else {
561
- this.httpManagerService.sendMessageToUser(wsServer, messageInfo);
562
- }
563
- }
564
- }
565
- // --------------------------------------------------------------------------------------------------
566
- // MISC
567
- isEmpty(obj) {
568
- return Object.keys(obj).length === 0;
569
- }
570
- updateRequestOptions(headers) {
571
- const options = ApiRequest.adapt({ ...this.apiOptions });
572
- options.headers = (headers)
573
- ? { ...options.headers, ...headers }
574
- : { ...options.headers };
575
- return options;
576
- }
577
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: HTTPManagerStateService, deps: [{ token: API_OPTS }, { token: "dataType" }, { token: "database" }], target: i0.ɵɵFactoryTarget.Injectable }); }
578
- static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: HTTPManagerStateService }); }
579
- }
580
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: HTTPManagerStateService, decorators: [{
581
- type: Injectable
582
- }], ctorParameters: () => [{ type: undefined, decorators: [{
583
- type: Inject,
584
- args: [API_OPTS]
585
- }] }, { type: undefined, decorators: [{
586
- type: Inject,
587
- args: ["dataType"]
588
- }] }, { type: undefined, decorators: [{
589
- type: Inject,
590
- args: ["database"]
591
- }] }] });
592
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaHR0cC1tYW5hZ2VyLXN0YXRlLnN0b3JlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvaHR0cC1yZXF1ZXN0LW1hbmFnZXIvc3JjL2xpYi9zZXJ2aWNlcy9yZXF1ZXN0LW1hbmFnZXItc3RhdGUtc2VydmljZS9odHRwLW1hbmFnZXItc3RhdGUuc3RvcmUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsVUFBVSxFQUFFLGNBQWMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzRSxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFFdkQsT0FBTyxFQUFFLGVBQWUsRUFBYyxFQUFFLEVBQUUsT0FBTyxFQUFnQixLQUFLLEVBQUUsS0FBSyxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQzVGLE9BQU8sRUFBRSxHQUFHLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBRSxHQUFHLEVBQUUsb0JBQW9CLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxNQUFNLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUN2SSxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0scUNBQXFDLENBQUM7QUFDdEUsT0FBTyxFQUFFLFVBQVUsRUFBRSxjQUFjLEVBQUUsU0FBUyxFQUFFLE1BQU0sVUFBVSxDQUFDO0FBQ2pFLE9BQU8sRUFBZSxjQUFjLEVBQWUsTUFBTSw4QkFBOEIsQ0FBQztBQUN4RixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsUUFBUSxFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDM0UsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDckUsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLGlEQUFpRCxDQUFDO0FBQ2pGLE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxXQUFXLEVBQUUsTUFBTSxr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QVU7OzBCQThETixNQUFNOzJCQUFDLFFBQVE7OzBCQUNmLE1BQU07MkJBQUMsVUFBVTs7MEJBQ2pCLE1BQU07MkJBQUMsVUFBVSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGluamVjdCwgSW5qZWN0LCBJbmplY3RhYmxlLCBJbmplY3Rpb25Ub2tlbiB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tcG9uZW50U3RvcmUgfSBmcm9tICdAbmdyeC9jb21wb25lbnQtc3RvcmUnO1xuXG5pbXBvcnQgeyBCZWhhdmlvclN1YmplY3QsIE9ic2VydmFibGUsIG9mLCBTdWJqZWN0LCBTdWJzY3JpcHRpb24sIHRpbWVyLCBtZXJnZSB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgdGFwLCBzd2l0Y2hNYXAsIGNvbmNhdE1hcCwgc2NhbiwgZGVsYXksIHRha2UsIG1hcCwgZGlzdGluY3RVbnRpbENoYW5nZWQsIHRha2VVbnRpbCwgdGFrZVdoaWxlLCBmaWx0ZXIgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5pbXBvcnQgeyBEYXRhYmFzZVN0b3JhZ2UgfSBmcm9tICcuLi8uLi9tb2RlbHMvZGF0YWJhc2Utc3RvcmFnZS5tb2RlbCc7XG5pbXBvcnQgeyBBcGlSZXF1ZXN0LCBSZXF1ZXN0T3B0aW9ucywgV1NPcHRpb25zIH0gZnJvbSAnLi9tb2RlbHMnO1xuaW1wb3J0IHsgQ2hhbm5lbEluZm8sIENoYW5uZWxNZXNzYWdlLCBVc2VyTWVzc2FnZSB9IGZyb20gJy4uL3dzLW1hbmFnZXItc2VydmljZS9tb2RlbHMnO1xuaW1wb3J0IHsgSFRUUE1hbmFnZXJTZXJ2aWNlLCBEYXRhVHlwZSB9IGZyb20gJy4uL3JlcXVlc3QtbWFuYWdlci1zZXJ2aWNlcyc7XG5pbXBvcnQgeyBEYXRhYmFzZU1hbmFnZXJTZXJ2aWNlIH0gZnJvbSAnLi4vZGF0YWJhc2UtbWFuYWdlci1zZXJ2aWNlJztcbmltcG9ydCB7IFRhYmxlU2NoZW1hRGVmIH0gZnJvbSAnLi4vZGF0YWJhc2UtbWFuYWdlci1zZXJ2aWNlL21vZGVscy90YWJsZS1zY2hlbWEnO1xuaW1wb3J0IHsgTG9jYWxTdG9yYWdlTWFuYWdlclNlcnZpY2UsIFN0b3JhZ2VUeXBlIH0gZnJvbSAnLi4vbG9jYWwtc3RvcmFnZS1tYW5hZ2VyLXNlcnZpY2UnO1xuaW1wb3J0IHsgU2V0dGluZ09wdGlvbnMgfSBmcm9tICcuLi9sb2NhbC1zdG9yYWdlLW1hbmFnZXItc2VydmljZS9tb2RlbHMvc2V0dGluZy1vcHRpb25zLm1vZGVsJztcbmltcG9ydCB7IFV0aWxzU2VydmljZSB9IGZyb20gJy4uLy4uJztcblxuY29uc3QgQVBJX09QVFMgPSBuZXcgSW5qZWN0aW9uVG9rZW48QXBpUmVxdWVzdD4oJ0FQSV9PUFRTJyk7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQVBJU3RhdGVNYW5hZ2VyRGF0YTxUPiB7XG4gIGRhdGE6IFRbXVxuICBkYXRhT2JqZWN0OiBUIHwgbnVsbFxufVxuXG5jb25zdCBkZWZhdWx0U3RhdGU6IEFQSVN0YXRlTWFuYWdlckRhdGE8YW55PiA9IHtcbiAgZGF0YTogW10sXG4gIGRhdGFPYmplY3Q6IG51bGwsXG59O1xuXG5ASW5qZWN0YWJsZSgpXG5leHBvcnQgY2xhc3MgSFRUUE1hbmFnZXJTdGF0ZVNlcnZpY2U8VCBleHRlbmRzIHsgaWQ6IG51bWJlciB9PiBleHRlbmRzIENvbXBvbmVudFN0b3JlPEFQSVN0YXRlTWFuYWdlckRhdGE8VD4+IHtcblxuICBodHRwTWFuYWdlclNlcnZpY2UgPSBpbmplY3QoSFRUUE1hbmFnZXJTZXJ2aWNlKVxuICBkYk1hbmFnZXJTZXJ2aWNlID0gaW5qZWN0KERhdGFiYXNlTWFuYWdlclNlcnZpY2UpXG4gIGxvY2FsU3RvcmFnZU1hbmFnZXJTZXJ2aWNlID0gaW5qZWN0KExvY2FsU3RvcmFnZU1hbmFnZXJTZXJ2aWNlKVxuICB1dGlscyA9IGluamVjdChVdGlsc1NlcnZpY2UpXG5cbiAgZXJyb3IkID0gdGhpcy5odHRwTWFuYWdlclNlcnZpY2UuZXJyb3IkXG4gIGlzUGVuZGluZyQgPSB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5pc1BlbmRpbmckLnBpcGUoZGVsYXkoMSkpXG5cbiAgLy8gUEFHSU5BVElPTlxuICBwcml2YXRlIHBhZ2UgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PG51bWJlcj4oMClcbiAgcGFnZSQgPSB0aGlzLnBhZ2UuYXNPYnNlcnZhYmxlKClcblxuICBwcml2YXRlIHRvdGFsUGFnZXMgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PG51bWJlcj4oMClcbiAgdG90YWxQYWdlcyQgPSB0aGlzLnRvdGFsUGFnZXMuYXNPYnNlcnZhYmxlKClcblxuICBwcml2YXRlIHBlcmNlbnRhZ2UgPSBuZXcgQmVoYXZpb3JTdWJqZWN0PG51bWJlcj4oMClcbiAgcGVyY2VudGFnZSQgPSB0aGlzLnBlcmNlbnRhZ2UuYXNPYnNlcnZhYmxlKClcblxuICBwcml2YXRlIGhhc0RhdGFiYXNlID0gZmFsc2VcblxuICBzdHJlYW1lZFJlc3BvbnNlID0gW11cblxuICAvLyBXU1xuICBwcml2YXRlIG1heFJldHJpZXM6IG51bWJlclxuICBwcml2YXRlIHJldHJ5RGVsYXk6IG51bWJlclxuICBwcml2YXRlIHNob3VsZFJldHJ5ID0gdHJ1ZVxuXG4gIHByaXZhdGUgd3NSZXRyeUF0dGVtcHRzID0gbmV3IEJlaGF2aW9yU3ViamVjdDxudW1iZXI+KDApXG4gIHdzUmV0cnlBdHRlbXB0cyQgPSB0aGlzLndzUmV0cnlBdHRlbXB0cy5hc09ic2VydmFibGUoKVxuXG4gIHByaXZhdGUgd3NOZXh0UmV0cnk6IEJlaGF2aW9yU3ViamVjdDxudW1iZXI+XG4gIHdzTmV4dFJldHJ5JDogT2JzZXJ2YWJsZTxudW1iZXI+XG5cblxuICBtZXNzYWdlcyQ6IFN1YnNjcmlwdGlvbiB8IG51bGwgPSBudWxsO1xuICBjb25uZWN0aW9uU3RhdHVzJDogU3Vic2NyaXB0aW9uIHwgbnVsbCA9IG51bGw7XG5cbiAgcHJpdmF0ZSB1c2VyTGlzdCA9IG5ldyBCZWhhdmlvclN1YmplY3Q8YW55W10+KFtdKVxuICB1c2VyTGlzdCQgPSB0aGlzLnVzZXJMaXN0LmFzT2JzZXJ2YWJsZSgpXG5cbiAgcHJpdmF0ZSB1c2VyID0gbmV3IEJlaGF2aW9yU3ViamVjdDxhbnl8bnVsbD4obnVsbClcbiAgdXNlciQgPSB0aGlzLnVzZXIuYXNPYnNlcnZhYmxlKClcblxuICBwcml2YXRlIGNvbW11bmljYXRpb25NZXNzYWdlcyA9IG5ldyBCZWhhdmlvclN1YmplY3Q8VXNlck1lc3NhZ2VbXT4oW10pXG4gIGNvbW11bmljYXRpb25NZXNzYWdlcyQgPSB0aGlzLmNvbW11bmljYXRpb25NZXNzYWdlcy5hc09ic2VydmFibGUoKVxuXG4gIHByaXZhdGUgbGF0ZXN0Q29tbXVuaWNhdGlvbk1lc3NhZ2VzID0gbmV3IEJlaGF2aW9yU3ViamVjdDxVc2VyTWVzc2FnZXxudWxsPihudWxsKVxuICBsYXRlc3RDb21tdW5pY2F0aW9uTWVzc2FnZXMkID0gdGhpcy5sYXRlc3RDb21tdW5pY2F0aW9uTWVzc2FnZXMuYXNPYnNlcnZhYmxlKClcblxuICBjaGFubmVsczogQ2hhbm5lbEluZm9bXSA9IFtdXG4gIGNoYW5uZWxMaXN0OiBzdHJpbmdbXSA9IFtdXG4gIG1lc3NhZ2VzOiBhbnlbXSA9IFtdXG5cbiAgd3NDb25uZWN0aW9uID0gZmFsc2VcbiAgd3NPcHRpb25zID0gV1NPcHRpb25zLmFkYXB0KClcblxuICBwdWJsaWMgc3RhdHVzJDogT2JzZXJ2YWJsZTxib29sZWFuPjtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBASW5qZWN0KEFQSV9PUFRTKSBwcml2YXRlIGFwaU9wdGlvbnMgPSBBcGlSZXF1ZXN0LmFkYXB0KCksXG4gICAgQEluamVjdChcImRhdGFUeXBlXCIpIHByaXZhdGUgZGF0YVR5cGU6IERhdGFUeXBlIHwgdW5kZWZpbmVkLFxuICAgIEBJbmplY3QoXCJkYXRhYmFzZVwiKSBwcml2YXRlIGRhdGFiYXNlPzogRGF0YWJhc2VTdG9yYWdlIHwgdW5kZWZpbmVkXG4gICkge1xuXG4gICAgc3VwZXIoZGVmYXVsdFN0YXRlKTtcblxuICAgIHRoaXMubWF4UmV0cmllcyA9IHRoaXMuYXBpT3B0aW9ucy53cz8ucmV0cnk/LnRpbWVzIHx8IDNcbiAgICB0aGlzLnJldHJ5RGVsYXkgPSAodGhpcy5hcGlPcHRpb25zLndzPy5yZXRyeT8uZGVsYXkgJiYgdGhpcy5hcGlPcHRpb25zLndzLnJldHJ5LmRlbGF5ICogMTAwMCkgfHwgNSAqIDEwMDBcbiAgICB0aGlzLndzTmV4dFJldHJ5ID0gbmV3IEJlaGF2aW9yU3ViamVjdDxudW1iZXI+KHRoaXMucmV0cnlEZWxheSlcbiAgICB0aGlzLndzTmV4dFJldHJ5JCA9IHRoaXMud3NOZXh0UmV0cnkuYXNPYnNlcnZhYmxlKClcblxuICAgIHRoaXMuc2V0QXBpUmVxdWVzdE9wdGlvbnMoYXBpT3B0aW9ucywgZGF0YVR5cGUsIGRhdGFiYXNlKVxuICAgIHRoaXMuc3RhdHVzJCA9IHRoaXMuc2V0dXBDb25uZWN0aW9uU3RhdHVzKCk7XG5cbiAgICBpZiAodGhpcy5kYXRhYmFzZSAmJiB0aGlzLmRhdGFiYXNlLnRhYmxlKSB7XG5cbiAgICAgIHRoaXMubG9jYWxTdG9yYWdlTWFuYWdlclNlcnZpY2UuY3JlYXRlU3RvcmUoe1xuICAgICAgICBuYW1lOiB0aGlzLmRhdGFiYXNlLnRhYmxlLFxuICAgICAgICBkYXRhOiB7IC4uLnRoaXMuZGF0YWJhc2UsIC4uLnsgZXhwaXJlczogdGhpcy51dGlscy5leHBpcmVzKHRoaXMuZGF0YWJhc2UuZXhwaXJlc0luKX0gfSxcbiAgICAgICAgb3B0aW9uczogU2V0dGluZ09wdGlvbnMuYWRhcHQoe1xuICAgICAgICAgIHN0b3JhZ2U6IFN0b3JhZ2VUeXBlLkdMT0JBTCxcbiAgICAgICAgICBlbmNyeXB0ZWQ6IGZhbHNlLFxuICAgICAgICB9KVxuICAgICAgfSlcblxuICAgIH1cblxuICB9XG5cbiAgc2V0QXBpUmVxdWVzdE9wdGlvbnMoYXBpT3B0aW9ucz86IEFwaVJlcXVlc3QsIGRhdGFUeXBlPzogRGF0YVR5cGUsIGRhdGFiYXNlPzogRGF0YWJhc2VTdG9yYWdlKSB7XG5cbiAgICB0aGlzLmFwaU9wdGlvbnMgPSBBcGlSZXF1ZXN0LmFkYXB0KGFwaU9wdGlvbnMpXG4gICAgdGhpcy5kYXRhVHlwZSA9IChkYXRhVHlwZSkgPyBkYXRhVHlwZSA6IERhdGFUeXBlLkFSUkFZXG5cbiAgICB0aGlzLmhhc0RhdGFiYXNlID0gKHRoaXMuZGF0YWJhc2U/LnRhYmxlKSA/IHRydWUgOiBmYWxzZVxuICAgIHRoaXMuZGF0YWJhc2UgPSAodGhpcy5oYXNEYXRhYmFzZSkgPyBEYXRhYmFzZVN0b3JhZ2UuYWRhcHQoZGF0YWJhc2UpIDogdW5kZWZpbmVkXG5cbiAgICBpZih0aGlzLmRhdGFiYXNlKSB0aGlzLmluaXREQlN0b3JhZ2UoKVxuICAgIGlmKHRoaXMuYXBpT3B0aW9ucy53cykgdGhpcy5pbml0V1ModGhpcy5hcGlPcHRpb25zLndzIGFzIFdTT3B0aW9ucylcblxuICB9XG5cbiAgLy8gV2ViU29ja2V0XG4gIHByaXZhdGUgc2V0dXBDb25uZWN0aW9uU3RhdHVzKCk6IE9ic2VydmFibGU8Ym9vbGVhbj4ge1xuICAgIHJldHVybiB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5jb25uZWN0aW9uU3RhdHVzJC5waXBlKFxuICAgICAgZGlzdGluY3RVbnRpbENoYW5nZWQoKSxcbiAgICAgIHN3aXRjaE1hcChzdGF0dXMgPT4ge1xuXG4gICAgICAgIGlmIChzdGF0dXMgPT09IHRydWUpIHtcbiAgICAgICAgICB0aGlzLnNob3VsZFJldHJ5ID0gdHJ1ZVxuICAgICAgICAgIHRoaXMud3NSZXRyeUF0dGVtcHRzLm5leHQoMCk7XG4gICAgICAgICAgdGhpcy53c05leHRSZXRyeS5uZXh0KDApO1xuICAgICAgICAgIHJldHVybiBvZih0cnVlKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICghdGhpcy5zaG91bGRSZXRyeSkgcmV0dXJuIG9mKGZhbHNlKVxuXG4gICAgICAgIGNvbnN0IGNvdW50ZG93bkVuZGVyJCA9IG5ldyBTdWJqZWN0PHZvaWQ+KCk7XG5cbiAgICAgICAgcmV0dXJuIHRpbWVyKDAsIHRoaXMucmV0cnlEZWxheSlcbiAgICAgICAgLnBpcGUoXG4gICAgICAgICAgdGFrZSh0aGlzLm1heFJldHJpZXMpLFxuICAgICAgICAgIHRhcChpID0+IHtcblxuICAgICAgICAgICAgY29uc3QgYXR0ZW1wdCA9IGkgKyAxO1xuICAgICAgICAgICAgdGhpcy53c1JldHJ5QXR0ZW1wdHMubmV4dChhdHRlbXB0KTtcbiAgICAgICAgICAgIGNvdW50ZG93bkVuZGVyJC5uZXh0KCk7XG5cbiAgICAgICAgICAgIGlmKHRoaXMuYXBpT3B0aW9ucy53cz8uand0VG9rZW4gIT09ICcnICYmIHRoaXMuYXBpT3B0aW9ucy53cz8ud3NTZXJ2ZXIgIT09ICcnKSB0aGlzLmluaXRXUyh0aGlzLmFwaU9wdGlvbnMud3MgYXMgV1NPcHRpb25zKTtcblxuICAgICAgICAgICAgaWYgKGF0dGVtcHQgPT09IHRoaXMubWF4UmV0cmllcykge1xuICAgICAgICAgICAgICB0aGlzLndzTmV4dFJldHJ5Lm5leHQoMClcbiAgICAgICAgICAgICAgLy8gICBjb25zb2xlLmVycm9yKGDwn5qoIEZBSUxFRCBDT05ORUNUSU9OOiBUcmllZCAjJHthdHRlbXB0fSB0aW1lc2ApO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgLy8gICBjb25zb2xlLmxvZyhg4pqg77iPIFJldHJ5IEF0dGVtcHQgIyR7YXR0ZW1wdH06IFJldHJ5aW5nIGluICR7dGhpcy5yZXRyeURlbGF5IC8gMTAwMH1zYCk7XG4gICAgICAgICAgICAgIGNvbnN0IHNlY29uZHMgPSB0aGlzLnJldHJ5RGVsYXkgLyAxMDAwO1xuXG4gICAgICAgICAgICAgIHRpbWVyKDAsIDEwMDApLnBpcGUoXG4gICAgICAgICAgICAgICAgbWFwKHRpY2sgPT4gc2Vjb25kcyAtIHRpY2spLFxuICAgICAgICAgICAgICAgIHRha2VXaGlsZSh2YWwgPT4gdmFsID49IDApLFxuICAgICAgICAgICAgICAgIHRha2VVbnRpbChjb3VudGRvd25FbmRlciQpXG4gICAgICAgICAgICAgICkuc3Vic2NyaWJlKHJlbWFpbmluZyA9PiB7XG4gICAgICAgICAgICAgICAgdGhpcy53c05leHRSZXRyeS5uZXh0KHJlbWFpbmluZyk7XG4gICAgICAgICAgICAgIH0pXG5cbiAgICAgICAgICAgIH1cblxuICAgICAgICAgIH0pLFxuICAgICAgICAgIG1hcCgoKSA9PiBmYWxzZSlcbiAgICAgICAgKVxuICAgICAgfSlcbiAgICApO1xuICB9XG5cbiAgLy8gV2ViU29ja2V0XG4gIHJlYWRvbmx5IGluaXRXUyA9IHRoaXMuZWZmZWN0PFdTT3B0aW9ucz4oKHdzT3B0aW9ucyQpID0+XG4gICAgd3NPcHRpb25zJC5waXBlKFxuICAgICAgdGFwKCh3c09wdGlvbnMpID0+IHtcbiAgICAgICAgdGhpcy53c09wdGlvbnMgPSB3c09wdGlvbnNcbiAgICAgICAgaWYgKHdzT3B0aW9ucz8ud3NTZXJ2ZXIpIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLmNvbm5lY3Qod3NPcHRpb25zLndzU2VydmVyLCB3c09wdGlvbnMuand0VG9rZW4pXG4gICAgICB9KSxcbiAgICAgIHN3aXRjaE1hcCgod3NPcHRpb25zKSA9PlxuICAgICAgICBtZXJnZShcbiAgICAgICAgICB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5jb25uZWN0aW9uU3RhdHVzJC5waXBlKFxuICAgICAgICAgICAgdGFwKChpc0Nvbm5lY3RlZDogYW55KSA9PiB7XG5cbiAgICAgICAgICAgICAgdGhpcy53c0Nvbm5lY3Rpb24gPSBpc0Nvbm5lY3RlZFxuXG4gICAgICAgICAgICAgIGlmIChpc0Nvbm5lY3RlZCkge1xuICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKCfwn5+iIFdlYlNvY2tldCBjb25uZWN0aW9uIGlzIG9wZW4uJylcbiAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZygn8J+UtCBXZWJTb2NrZXQgY29ubmVjdGlvbiBpcyBjbG9zZWQuJylcbiAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICB9KVxuICAgICAgICAgICksXG4gICAgICAgICAgdGhpcy5odHRwTWFuYWdlclNlcnZpY2UubWVzc2FnZXMkLnBpcGUoXG4gICAgICAgICAgICB0YXAoKG1lc3NhZ2U6IGFueSkgPT4ge1xuICAgICAgICAgICAgICBpZiAoIW1lc3NhZ2UpIHJldHVyblxuXG4gICAgICAgICAgICAgIGNvbnNvbGUubG9nKCdSZWNlaXZlZDonLCBtZXNzYWdlLnR5cGUpXG5cbiAgICAgICAgICAgICAgaWYgKG1lc3NhZ2UuZXJyb3IgPT09ICdKV1RfSU5WQUxJRCcpIHtcbiAgICAgICAgICAgICAgICB0aGlzLnNob3VsZFJldHJ5ID0gZmFsc2VcbiAgICAgICAgICAgICAgICB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5kaXNjb25uZWN0KClcbiAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgIGlmIChtZXNzYWdlLnR5cGUgPT09ICdzdWNjZXNzJykge1xuICAgICAgICAgICAgICAgIGNvbnN0IGN1cnJlbnRVc2VyID0gdGhpcy5hcGlPcHRpb25zLndzPy51c2VyQWRhcHRlcj8uYWRhcHRlcj8uKG1lc3NhZ2UuZGF0YSkgfHwgbWVzc2FnZS5kYXRhXG4gICAgICAgICAgICAgICAgdGhpcy51c2VyLm5leHQoY3VycmVudFVzZXIpXG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICBzd2l0Y2ggKG1lc3NhZ2UudHlwZSkge1xuICAgICAgICAgICAgICAgIGNhc2UgJ2NoYW5uZWxzTGlzdCc6XG5cbiAgICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKCfwn5KsIENoYW5uZWxzOicsIG1lc3NhZ2UuY2hhbm5lbHMpXG5cbiAgICAgICAgICAgICAgICAgIHRoaXMuY2hhbm5lbExpc3QgPSBtZXNzYWdlLmNoYW5uZWxzXG5cbiAgICAgICAgICAgICAgICAgIGlmICh0aGlzLmNoYW5uZWxMaXN0LmluY2x1ZGVzKHdzT3B0aW9ucy5pZCkpIHtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5odHRwTWFuYWdlclNlcnZpY2Uuc3Vic2NyaWJlVG9DaGFubmVsKHdzT3B0aW9ucy5pZClcbiAgICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLmNyZWF0ZUNoYW5uZWwod3NPcHRpb25zLmlkKVxuICAgICAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgICAgICBicmVhaztcblxuICAgICAgICAgICAgICAgIGNhc2UgJ3N0YXRlTWFuZ2VyTWVzc2FnZSc6XG4gICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZygn8J+SrCBNZXNzYWdlOicsIG1lc3NhZ2UuY29udGVudClcbiAgICAgICAgICAgICAgICAgIHRoaXMuZmV0Y2hSZWNvcmQoUmVxdWVzdE9wdGlvbnMuYWRhcHQoeyBwYXRoOiBtZXNzYWdlLmNvbnRlbnQucGF0aCB9KSwgbWVzc2FnZS5jb250ZW50Lm1ldGhvZClcbiAgICAgICAgICAgICAgICAgIGJyZWFrO1xuXG4gICAgICAgICAgICAgICAgY2FzZSAnY2hhbm5lbE1lc3NhZ2UnOlxuICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coJ/CfkqwgTWVzc2FnZTonLCBtZXNzYWdlLmNvbnRlbnQpXG4gICAgICAgICAgICAgICAgICB0aGlzLmZldGNoUmVjb3JkKFJlcXVlc3RPcHRpb25zLmFkYXB0KHsgcGF0aDogbWVzc2FnZS5jb250ZW50LnBhdGggfSksIG1lc3NhZ2UuY29udGVudC5tZXRob2QpXG4gICAgICAgICAgICAgICAgICBicmVhaztcblxuICAgICAgICAgICAgICAgIGNhc2UgJ2NoYW5uZWxDb21tdW5pY2F0aW9uJzpcbiAgICAgICAgICAgICAgICAgIGNvbnNvbGUubG9nKCfwn5KsIE1lc3NhZ2UgQ29tbXVuaWNhdGlvbjonLCBtZXNzYWdlLmNvbnRlbnQpXG4gICAgICAgICAgICAgICAgICB0aGlzLmFwcGVuZE1lc3NhZ2VzKG1lc3NhZ2UuY29udGVudClcbiAgICAgICAgICAgICAgICAgIGJyZWFrO1xuXG4gICAgICAgICAgICAgICAgY2FzZSAnY2hhbm5lbEFsZXJ0cyc6XG4gICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZygn8J+SrCBNZXNzYWdlIEFsZXJ0czonLCBtZXNzYWdlLmNvbnRlbnQpXG4gICAgICAgICAgICAgICAgICBicmVhaztcblxuICAgICAgICAgICAgICAgIGNhc2UgJ3VzZXJzSW5DaGFubmVsJzpcblxuICAgICAgICAgICAgICAgIGNvbnN0IGRhdGEgPSBtZXNzYWdlLmRhdGFcbiAgICAgICAgICAgICAgICBjb25zdCB3c1VzZXJzID0gZGF0YS51c2Vycy5tYXAoKGl0ZW06IGFueSkgPT4gdGhpcy5hcGlPcHRpb25zLndzPy51c2VyQWRhcHRlcj8uYWRhcHRlcj8uKGl0ZW0pIHx8IGl0ZW0pXG4gICAgICAgICAgICAgICAgY29uc3QgdXNlckxpc3QgPSB3c1VzZXJzLmZpbHRlcigoaXRlbTogYW55KSA9PiBpdGVtLnVzZXJuYW1lICE9PSB0aGlzLnVzZXIudmFsdWVbdGhpcy53c09wdGlvbnMudXNlckFkYXB0ZXI/LmlkIHx8ICdpZCddKVxuXG4gICAgICAgICAgICAgICAgICB0aGlzLnVzZXJMaXN0Lm5leHQodXNlckxpc3QpXG4gICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZygn8J+RpSBVc2VyczonLCB1c2VyTGlzdClcbiAgICAgICAgICAgICAgICAgIGJyZWFrO1xuXG4gICAgICAgICAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICAgICAgICAgIGlmIChtZXNzYWdlKSB0aGlzLm1lc3NhZ2VzLnB1c2gobWVzc2FnZSlcbiAgICAgICAgICAgICAgICAgIGJyZWFrXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgKVxuICAgICAgICApXG4gICAgICApXG4gICAgKVxuICApO1xuXG4gIGFwcGVuZE1lc3NhZ2VzKG1lc3NhZ2U6IFVzZXJNZXNzYWdlKSB7XG4gICAgY29uc3QgY3VycmVudE1lc3NhZ2VzID0gdGhpcy5jb21tdW5pY2F0aW9uTWVzc2FnZXMudmFsdWVcbiAgICB0aGlzLmNvbW11bmljYXRpb25NZXNzYWdlcy5uZXh0KFsuLi5jdXJyZW50TWVzc2FnZXMsIG1lc3NhZ2VdKVxuICAgIHRoaXMubGF0ZXN0TWVzc2FnZSgpXG4gIH1cblxuICBsYXRlc3RNZXNzYWdlKCkge1xuICAgIGNvbnN0IG1lc3NhZ2VzID0gdGhpcy5jb21tdW5pY2F0aW9uTWVzc2FnZXMudmFsdWVcbiAgICBjb25zdCBsYXRlc3RNZXNzYWdlID0gbWVzc2FnZXNbbWVzc2FnZXMubGVuZ3RoIC0xXVxuICAgIHRoaXMubGF0ZXN0Q29tbXVuaWNhdGlvbk1lc3NhZ2VzLm5leHQobGF0ZXN0TWVzc2FnZSlcbiAgfVxuXG4gIGNsZWFyTWVzc2FnZXMoKSB7XG4gICAgdGhpcy5jb21tdW5pY2F0aW9uTWVzc2FnZXMubmV4dChbXSlcbiAgfVxuXG4gIGdldCBBcGlSZXF1ZXN0T3B0aW9ucygpIHtcbiAgICByZXR1cm4gdGhpcy5hcGlPcHRpb25zXG4gIH1cblxuICByZWFkb25seSBpbml0REJTdG9yYWdlID0gdGhpcy5lZmZlY3Q8dm9pZD4oKHRyaWdnZXIkKSA9PlxuICAgIHRyaWdnZXIkLnBpcGUoXG4gICAgICB0YXAoKCkgPT4ge1xuICAgICAgICBpZiAodGhpcy5kYXRhVHlwZSAhPT0gRGF0YVR5cGUuQVJSQVkpIGNvbnNvbGUud2FybignRGF0YWJhc2Ugc3RvcmFnZSByZXF1aXJlcyBkYXRhVHlwZSB0byBiZSBBUlJBWScpO1xuICAgICAgICBpZiAoIXRoaXMuYXBpT3B0aW9ucy5hZGFwdGVyKSBjb25zb2xlLndhcm4oJ0RhdGFiYXNlIHN0b3JhZ2UgcmVxdWlyZXMgYW4gYWRhcHRlciB0byBkZWZpbmUgdGhlIGRhdGEgc2hhcGUnKTtcbiAgICAgICAgaWYgKCF0aGlzLmRhdGFiYXNlPy50YWJsZSkgY29uc29sZS53YXJuKCdEYXRhYmFzZSBzdG9yYWdlIHJlcXVpcmVzIGEgdGFibGUgbmFtZScpO1xuICAgICAgfSksXG4gICAgICBmaWx0ZXIoKCkgPT4gdGhpcy5kYXRhVHlwZSA9PT0gRGF0YVR5cGUuQVJSQVkgJiYgISF0aGlzLmFwaU9wdGlvbnMuYWRhcHRlciAmJiAhIXRoaXMuZGF0YWJhc2U/LnRhYmxlKSxcbiAgICAgIHN3aXRjaE1hcCgoKSA9PiB7XG4gICAgICAgIGNvbnN0IHNhbXBsZURhdGEgPSB0aGlzLmFwaU9wdGlvbnMuYWRhcHRlcj8uKHt9KSB8fCB7fTtcbiAgICAgICAgY29uc3Qgc2NoZW1hS2V5cyA9IE9iamVjdC5rZXlzKHNhbXBsZURhdGEpLmZpbHRlcihrZXkgPT4gc2FtcGxlRGF0YVtrZXldICE9PSB1bmRlZmluZWQpO1xuXG4gICAgICAgIGxldCBzY2hlbWEgPSAnKytpZCc7XG5cbiAgICAgICAgaWYgKHNjaGVtYUtleXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgIGNvbnN0IG90aGVyS2V5cyA9IHNjaGVtYUtleXMuZmlsdGVyKGsgPT4gayAhPT0gJ2lkJyk7XG4gICAgICAgICAgaWYgKG90aGVyS2V5cy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICBzY2hlbWEgKz0gJywgJyArIG90aGVyS2V5cy5qb2luKCcsICcpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHRhYmxlRGVmID0gVGFibGVTY2hlbWFEZWYuYWRhcHQoe1xuICAgICAgICAgIHRhYmxlOiB0aGlzLmRhdGFiYXNlPy50YWJsZSxcbiAgICAgICAgICBzY2hlbWE6IHNjaGVtYVxuICAgICAgICB9KTtcblxuICAgICAgICByZXR1cm4gdGhpcy5kYk1hbmFnZXJTZXJ2aWNlLmNyZWF0ZURhdGFiYXNlVGFibGUodGFibGVEZWYpXG4gICAgICB9KVxuICAgIClcbiAgKTtcblxuICBpbml0aWFsaXplU3RhdGUoZGF0YTogYW55KSB7XG4gICAgdGhpcy5zZXREYXRhJChkYXRhKVxuICB9XG5cbiAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbiAgLy8gU0VMRUNUT1JTXG5cbiAgcmVhZG9ubHkgZGF0YSQgPSB0aGlzLnNlbGVjdCgoeyBkYXRhLCBkYXRhT2JqZWN0IH0pID0+IHtcbiAgICBjb25zdCBpc0FycmF5ID0oIHRoaXMuZGF0YVR5cGUgPT09IERhdGFUeXBlLkFSUkFZKSA/IHRydWUgOiBmYWxzZVxuICAgIHJldHVybiAoaXNBcnJheSkgPyBkYXRhIDogZGF0YU9iamVjdFxuICB9KVxuXG4gIHJlYWRvbmx5IHNlbGVjdFJlY29yZCQgPSAoaWQ6IG51bWJlcikgPT4gdGhpcy5zZWxlY3QoXG4gICAgdGhpcy5kYXRhJCxcbiAgICAoZGF0YSkgPT4ge1xuICAgICAgaWYgKHRoaXMuZGF0YVR5cGUgPT09IERhdGFUeXBlLkFSUkFZICYmIEFycmF5LmlzQXJyYXkoZGF0YSkpIHtcbiAgICAgICAgcmV0dXJuIGRhdGEuZmluZChpdGVtID0+IGl0ZW0uaWQgPT09IGlkKSBhcyBUIHwgbnVsbFxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgcmV0dXJuIChkYXRhIGFzIFQpLmlkID09PSBpZCA/IGRhdGEgOiBudWxsXG4gICAgICB9XG4gICAgfVxuICApO1xuXG4gIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG5cbiAgLy8gVVBEQVRFUlNcblxuICBwcml2YXRlIHJlYWRvbmx5IHNldERhdGEkID0gdGhpcy51cGRhdGVyKChzdGF0ZTogQVBJU3RhdGVNYW5hZ2VyRGF0YTxUPiwgZGF0YTogVCB8IFRbXSB8IGFueSkgPT4ge1xuXG4gICAgaWYgKCFkYXRhKSByZXR1cm4gc3RhdGU7XG5cbiAgICBpZiAodGhpcy5kYXRhVHlwZSA9PT0gRGF0YVR5cGUuQVJSQVkpIHtcblxuICAgICAgY29uc3QgZGF0YUFycmF5ID0gQXJyYXkuaXNBcnJheShkYXRhKSA/IGRhdGEgOiBbZGF0YV1cblxuICAgICAgY29uc3Qgc3RhdGVEYXRhU2FtcGxlID0gKHN0YXRlLmRhdGEubGVuZ3RoID4gMCkgPyBPYmplY3Qua2V5cyhzdGF0ZS5kYXRhWzBdKSA6IFtdXG4gICAgICBjb25zdCBuZXdEYXRhU2FtcGxlID0gKGRhdGFBcnJheS5sZW5ndGggPiAwKSA/IE9iamVjdC5rZXlzKGRhdGFBcnJheVswXSkgOiBbXVxuXG4gICAgICBjb25zdCBpc1NhbWUgPSAoc3RhdGUuZGF0YS5sZW5ndGggPT09IDApID8gZmFsc2UgOiBzdGF0ZURhdGFTYW1wbGUuZXZlcnkoKHZhbHVlLCBpbmRleCkgPT4gdmFsdWUgPT09IG5ld0RhdGFTYW1wbGVbaW5kZXhdKVxuICAgICAgY29uc3QgdXBkYXRlZERhdGEgPSAoIWlzU2FtZSAmJiBkYXRhQXJyYXkubGVuZ3RoICE9PSAwKSA/IHRoaXMudXBkYXRlQXJyYXlTdGF0ZShbXSwgZGF0YUFycmF5KSA6IHRoaXMudXBkYXRlQXJyYXlTdGF0ZShzdGF0ZS5kYXRhLCBkYXRhQXJyYXkpXG5cbiAgICAgIHJldHVybiB7IC4uLnN0YXRlLCBkYXRhOiB1cGRhdGVkRGF0YSwgZGF0YU9iamVjdDogbnVsbCB9IGFzIEFQSVN0YXRlTWFuYWdlckRhdGE8VD47XG5cbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3QgZGF0YU9iamVjdCA9IHRoaXMuaXNFbXB0eShkYXRhKSA/IG51bGwgOiBkYXRhO1xuICAgICAgcmV0dXJuIHsgLi4uc3RhdGUsIGRhdGE6IFtdLCBkYXRhT2JqZWN0OiBkYXRhT2JqZWN0IH0gYXMgQVBJU3RhdGVNYW5hZ2VyRGF0YTxUPjtcbiAgICB9XG5cbiAgfSk7XG5cbiAgcHJpdmF0ZSB1cGRhdGVBcnJheVN0YXRlKGN1cnJlbnREYXRhOiBhbnlbXSwgbmV3RGF0YTogYW55W10pOiBhbnlbXSB7XG5cbiAgICBjb25zdCBmaWx0ZXJDdXJyZW50RGF0YSA9ICgpID0+IHtcbiAgICAgIGNvbnN0IGlkcyA9IHRoaXMuc3RyZWFtZWRSZXNwb25zZS5tYXAoKG9iajogYW55KSA9PiBvYmouaWQpXG4gICAgICByZXR1cm4gY3VycmVudERhdGEuZmlsdGVyKG9iaiA9PiAob2JqLmlkKSA/IGlkcy5pbmNsdWRlcyhvYmouaWQpIDogb2JqKVxuICAgIH1cblxuICAgIGNvbnN0IGZpbHRlcmVkQ3VycmVudERhdGEgPSAodGhpcy5odHRwTWFuYWdlclNlcnZpY2UuaXNQZW5kaW5nLnZhbHVlKSA/IGN1cnJlbnREYXRhIDogZmlsdGVyQ3VycmVudERhdGEoKVxuXG4gICAgY29uc3QgdXBkYXRlZERhdGEgPSBmaWx0ZXJlZEN1cnJlbnREYXRhLm1hcChpdGVtID0+IHtcbiAgICAgICAgY29uc3QgbmV3SXRlbSA9IG5ld0RhdGEuZmluZChuZXdJdGVtID0+IHtcbiAgICAgICAgICBjb25zdCBoYXNJZCA9IChuZXdJdGVtPy5pZCAmJiBpdGVtPy5pZCkgPyB0cnVlIDogZmFsc2VcbiAgICAgICAgICByZXR1cm4gKGhhc0lkKSA/IG5ld0l0ZW0uaWQgPT09IGl0ZW0uaWQgOiBKU09OLnN0cmluZ2lmeShuZXdJdGVtKSA9PT0gSlNPTi5zdHJpbmdpZnkoaXRlbSlcbiAgICAgICAgfSlcbiAgICAgICAgcmV0dXJuIChuZXdJdGVtKSA/IHsgLi4uaXRlbSwgLi4ubmV3SXRlbSB9IDogaXRlbVxuICAgIH0pXG5cbiAgICBjb25zdCBhZGRlZERhdGEgPSBuZXdEYXRhLmZpbHRlcihuZXdJdGVtID0+IHtcbiAgICAgIHJldHVybiAhZmlsdGVyZWRDdXJyZW50RGF0YS5zb21lKGl0ZW0gPT4ge1xuICAgICAgICAgIGNvbnN0IGhhc0lkID0gKG5ld0l0ZW0/LmlkICYmIGl0ZW0/LmlkKSA/IHRydWUgOiBmYWxzZVxuICAgICAgICAgIHJldHVybiAoaGFzSWQpID8gaXRlbS5pZCA9PT0gbmV3SXRlbS5pZCA6IEpTT04uc3RyaW5naWZ5KG5ld0l0ZW0pID09PSBKU09OLnN0cmluZ2lmeShpdGVtKVxuICAgICAgfSlcbiAgICB9KVxuXG4gICAgcmV0dXJuIFsuLi51cGRhdGVkRGF0YSwgLi4uYWRkZWREYXRhXVxuXG59XG5cbiAgcHJpdmF0ZSByZWFkb25seSBhZGREYXRhJCA9IHRoaXMudXBkYXRlcigoc3RhdGU6IEFQSVN0YXRlTWFuYWdlckRhdGE8VD4sIGRhdGE6IFQpID0+IHtcblxuICBpZiAodGhpcy5kYXRhVHlwZSA9PT0gRGF0YVR5cGUuQVJSQVkpIHtcbiAgICAgIGNvbnN0IGV4aXN0cyA9IHN0YXRlLmRhdGEuc29tZShpdGVtID0+IGl0ZW0uaWQgPT09IGRhdGEuaWQpO1xuXG4gICAgICBpZiAoZXhpc3RzKSB7XG4gICAgICAgICAgY29uc3QgdXBkYXRlZERhdGEgPSBzdGF0ZS5kYXRhLm1hcChpdGVtID0+XG4gICAgICAgICAgaXRlbS5pZCA9PT0gZGF0YS5pZCA/IGRhdGEgOiBpdGVtXG4gICAgICAgICk7XG5cbiAgICAgICAgcmV0dXJuIHsgLi4uc3RhdGUsIGRhdGE6IHVwZGF0ZWREYXRhIH07XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zdCBuZXdTdGF0ZSA9IFsuLi5zdGF0ZS5kYXRhLCBkYXRhXTtcbiAgICAgICAgcmV0dXJuIHsgLi4uc3RhdGUsIGRhdGE6IG5ld1N0YXRlIH07XG4gICAgICB9XG4gICAgfSBlbHNlIHtcbiAgICAgIHJldHVybiB7IC4uLnN0YXRlLCBkYXRhT2JqZWN0OiBkYXRhIH07XG4gICAgfVxuXG4gIH0pXG5cbiAgcHJpdmF0ZSByZWFkb25seSBkZWxldGVEYXRhJCA9IHRoaXMudXBkYXRlcigoc3RhdGU6IEFQSVN0YXRlTWFuYWdlckRhdGE8VD4sIGRhdGE6IFQgJiB7IGlkOiBudW1iZXIgfSkgPT4ge1xuICAgIGlmICh0aGlzLmRhdGFUeXBlID09PSBEYXRhVHlwZS5BUlJBWSkge1xuICAgIGNvbnN0IG5ld1N0YXRlID0gc3RhdGUuZGF0YS5maWx0ZXIoaXRlbSA9PiBpdGVtLmlkICE9PSBkYXRhLmlkKVxuICAgIHJldHVybiB7IC4uLnN0YXRlLCAuLi57IGRhdGE6IG5ld1N0YXRlIH0gfVxuICAgIH0gZWxzZSB7XG4gICAgICByZXR1cm4geyAuLi5zdGF0ZSwgLi4ueyBkYXRhT2JqZWN0OiBudWxsIH0gfVxuICAgIH1cbiAgfSlcblxuICBwcml2YXRlIHJlYWRvbmx5IHVwZGF0ZURhdGEkID0gdGhpcy51cGRhdGVyKChzdGF0ZTogQVBJU3RhdGVNYW5hZ2VyRGF0YTxUPiwgZGF0YTogVCkgPT4ge1xuXG4gICAgaWYgKHRoaXMuZGF0YVR5cGUgPT09IERhdGFUeXBlLkFSUkFZKSB7XG5cbiAgICAgIGNvbnN0IG9iakluZGV4ID0gc3RhdGUuZGF0YS5maW5kSW5kZXgoaXRlbSA9PiBpdGVtLmlkID09PSBkYXRhLmlkKVxuXG4gICAgICBpZiAob2JqSW5kZXggPiAtMSkge1xuICAgICAgICBjb25zdCBuZXdTdGF0ZSA9IFsuLi5zdGF0ZS5kYXRhXVxuICAgICAgICBuZXdTdGF0ZVtvYmpJbmRleF0gPSBkYXRhXG4gICAgICAgIHJldHVybiB7IC4uLnN0YXRlLCAuLi57IGRhdGE6IG5ld1N0YXRlIH0gfVxuICAgICAgfVxuXG4gICAgICByZXR1cm4gc3RhdGVcblxuICAgIH0gZWxzZSB7XG4gICAgICByZXR1cm4geyAuLi5zdGF0ZSwgLi4ueyBkYXRhT2JqZWN0OiBkYXRhIH0gfVxuICAgIH1cblxuICB9KVxuXG4gIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gIC8vIEVGRkVDVFNcblxuICByZWFkb25seSBjbGVhclJlY29yZHMgPSB0aGlzLmVmZmVjdChkYXRhID0+XG4gICAgZGF0YS5waXBlKFxuXG4gICAgICB0YXAoKCkgPT4ge1xuXG4gICAgICAgIGlmICh0aGlzLmRhdGFUeXBlID09PSBEYXRhVHlwZS5BUlJBWSkge1xuICAgICAgICAgIHRoaXMuc2V0RGF0YSQoW10pXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdGhpcy5zZXREYXRhJCh7fSlcbiAgICAgICAgfVxuXG4gICAgICB9KSxcbiAgICAgIGNvbmNhdE1hcCgoKSA9PiB7XG4gICAgICAgIGlmICh0aGlzLmhhc0RhdGFiYXNlICYmIHRoaXMuZGF0YWJhc2U/LnRhYmxlKSB7XG5cbiAgICAgICAgICBjb25zdCBjdXJyZW50RGF0YSA9IHRoaXMuZ2V0KCk/LmRhdGE7XG4gICAgICAgICAgY29uc3QgaWRzVG9EZWxldGUgPSBBcnJheS5pc0FycmF5KGN1cnJlbnREYXRhKSA/IGN1cnJlbnREYXRhLm1hcCgocjogYW55KSA9PiByLmlkKSA6IFtdO1xuXG4gICAgICAgICAgaWYgKGlkc1RvRGVsZXRlLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UuZGVsZXRlVGFibGVSZWNvcmRzKHRoaXMuZGF0YWJhc2UudGFibGUsIGlkc1RvRGVsZXRlKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gb2YobnVsbCk7XG4gICAgICB9KVxuICApKVxuXG5cbiAgLy8gLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbiAgLy8gQ1JVRCBPUEVSQVRJT05TXG5cbiAgLy8gRkVUQ0ggUkVDT1JEU1xuICByZWFkb25seSBmZXRjaFJlY29yZHMgPSAob3B0aW9ucz86IFJlcXVlc3RPcHRpb25zKSA9PlxuICAgIHRoaXMuZWZmZWN0PGFueT4oKCkgPT5cbiAgICBvZihSZXF1ZXN0T3B0aW9ucy5hZGFwdChvcHRpb25zKSkucGlwZShcbiAgICAgIHN3aXRjaE1hcCgoKSA9PiB7XG5cbiAgICAgICAgdGhpcy5zdHJlYW1lZFJlc3BvbnNlID0gW11cbiAgICAgICAgY29uc3QgcmVxdWVzdE9wdGlvbnMgPSB0aGlzLnVwZGF0ZVJlcXVlc3RPcHRpb25zKG9wdGlvbnM/LmhlYWRlcnMpXG5cbiAgICAgICAgY29uc3QgZmV0Y2hGcm9tQVBJID0gKCkgPT4ge1xuXG4gICAgICAgICAgcmV0dXJuIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLmdldFJlcXVlc3QocmVxdWVzdE9wdGlvbnMsIG9wdGlvbnM/LnBhdGgpLnBpcGUoXG4gICAgICAgICAgICB0YXAoKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICAgICAgICBkYXRhID0gKCFkYXRhKSA/ICh0aGlzLmRhdGFUeXBlID09PSBEYXRhVHlwZS5BUlJBWSkgPyBbXSA6IHt9IDogZGF0YVxuICAgICAgICAgICAgICB0aGlzLnNldERhdGEkKGRhdGEpXG4gICAgICAgICAgICB9KSxcbiAgICAgICAgICAgIGNvbmNhdE1hcCgoZGF0YTogYW55KSA9PiB7XG4gICAgICAgICAgICAgIGlmICh0aGlzLmhhc0RhdGFiYXNlICYmIHRoaXMuZGF0YWJhc2U/LnRhYmxlICYmIEFycmF5LmlzQXJyYXkoZGF0YSkgJiYgZGF0YS5sZW5ndGggPiAwKSB7XG5cbiAgICAgICAgICAgICAgICB0aGlzLmxvY2FsU3RvcmFnZU1hbmFnZXJTZXJ2aWNlLnVwZGF0ZVN0b3JlKHtcbiAgICAgICAgICAgICAgICAgIG5hbWU6IHRoaXMuZGF0YWJhc2UhLnRhYmxlLFxuICAgICAgICAgICAgICAgICAgZGF0YTogeyAuLi50aGlzLmRhdGFiYXNlLCAuLi57IGV4cGlyZXM6IHRoaXMudXRpbHMuZXhwaXJlcyh0aGlzLmRhdGFiYXNlIS5leHBpcmVzSW4pfSB9XG4gICAgICAgICAgICAgICAgfSlcblxuICAgICAgICAgICAgICAgIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UuY3JlYXRlVGFibGVSZWNvcmRzKHRoaXMuZGF0YWJhc2UudGFibGUsIGRhdGEpO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHJldHVybiBvZihkYXRhKTtcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgKTtcblxuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHRoaXMuaGFzRGF0YWJhc2UgJiYgdGhpcy5kYXRhYmFzZT8udGFibGUpIHtcblxuICAgICAgICAgIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UuZGF0YWJhc2VFeGlzdHMoKS5waXBlKFxuICAgICAgICAgICAgc3dpdGNoTWFwKChkYkV4aXN0czogYm9vbGVhbikgPT4ge1xuXG4gICAgICAgICAgICAgIGlmICghZGJFeGlzdHMpIHtcbiAgICAgICAgICAgICAgICBjb25zdCBpbml0T2JzOiBPYnNlcnZhYmxlPGFueT4gPSB0aGlzLmluaXREQlN0b3JhZ2VBc3luYygpO1xuICAgICAgICAgICAgICAgIHJldHVybiBpbml0T2JzLnBpcGUoXG4gICAgICAgICAgICAgICAgICBzd2l0Y2hNYXAoKCkgPT4gZmV0Y2hGcm9tQVBJKCkpXG4gICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UuaGFzRGF0YWJhc2VUYWJsZSh0aGlzLmRhdGFiYXNlIS50YWJsZSkucGlwZShcbiAgICAgICAgICAgICAgICBzd2l0Y2hNYXAoKHRhYmxlRXhpc3RzOiBib29sZWFuKTogT2JzZXJ2YWJsZTxhbnk+ID0+IHtcblxuICAgICAgICAgICAgICAgICAgaWYgKCF0YWJsZUV4aXN0cykge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBpbml0T2JzOiBPYnNlcnZhYmxlPGFueT4gPSB0aGlzLmluaXREQlN0b3JhZ2VBc3luYygpO1xuICAgICAgICAgICAgICAgICAgICByZXR1cm4gaW5pdE9icy5waXBlKFxuICAgICAgICAgICAgICAgICAgICAgIHN3aXRjaE1hcCgoKSA9PiBmZXRjaEZyb21BUEkoKSlcbiAgICAgICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAgICAgcmV0dXJuIHRoaXMubG9jYWxTdG9yYWdlTWFuYWdlclNlcnZpY2Uuc3RvcmUkKHRoaXMuZGF0YWJhc2UhLnRhYmxlKS5waXBlKFxuICAgICAgICAgICAgICAgICAgICB0YWtlKDEpLFxuICAgICAgICAgICAgICAgICAgICBzd2l0Y2hNYXAoKHN0b3JlRGF0YTogYW55KSA9PiB7XG5cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBleHBpcmVzID0gc3RvcmVEYXRhPy5leHBpcmVzIHx8IDA7XG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgaGFzRXhwaXJlZCA9IGV4cGlyZXMgPiAwICYmIHRoaXMudXRpbHMuaGFzRXhwaXJlZChleHBpcmVzKTtcblxuICAgICAgICAgICAgICAgICAgICAgIGlmIChoYXNFeHBpcmVkKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gdGhpcy5kYk1hbmFnZXJTZXJ2aWNlLmNsZWFyVGFibGUodGhpcy5kYXRhYmFzZSEudGFibGUpLnBpcGUoXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHN3aXRjaE1hcCgoKSA9PiBmZXRjaEZyb21BUEkoKSksXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHRhcCgoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5sb2NhbFN0b3JhZ2VNYW5hZ2VyU2VydmljZS51cGRhdGVTdG9yZSh7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYW1lOiB0aGlzLmRhdGFiYXNlIS50YWJsZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRhdGE6IHsgLi4udGhpcy5kYXRhYmFzZSwgLi4ueyBleHBpcmVzOiB0aGlzLnV0aWxzLmV4cGlyZXModGhpcy5kYXRhYmFzZSEuZXhwaXJlc0luKX0gfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAgICAgICAgIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UuZ2V0VGFibGVSZWNvcmRzKHRoaXMuZGF0YWJhc2UhLnRhYmxlKS5waXBlKFxuICAgICAgICAgICAgICAgICAgICAgICAgc3dpdGNoTWFwKChkYkRhdGE6IGFueSkgPT4ge1xuXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChBcnJheS5pc0FycmF5KGRiRGF0YSkgJiYgZGJEYXRhLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aGlzLnNldERhdGEkKGRiRGF0YSk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIG9mKGRiRGF0YSk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gZmV0Y2hGcm9tQVBJKCk7XG5cbiAgICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgICAgKTtcbiAgICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgICk7XG5cbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgfSlcbiAgICAgICAgICApO1xuXG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gZmV0Y2hGcm9tQVBJKCk7XG5cbiAgICAgIH0pXG4gICAgKVxuICApXG5cbiAgcHJpdmF0ZSBpbml0REJTdG9yYWdlQXN5bmMoKSB7XG5cbiAgICBpZiAodGhpcy5kYXRhVHlwZSAhPT0gRGF0YVR5cGUuQVJSQVkpIHtcbiAgICAgIGNvbnNvbGUud2FybignRGF0YWJhc2Ugc3RvcmFnZSByZXF1aXJlcyBkYXRhVHlwZSB0byBiZSBBUlJBWScpO1xuICAgICAgcmV0dXJuIG9mKG51bGwpO1xuICAgIH1cblxuICAgIGlmICghdGhpcy5hcGlPcHRpb25zLmFkYXB0ZXIpIHtcbiAgICAgIGNvbnNvbGUud2FybignRGF0YWJhc2Ugc3RvcmFnZSByZXF1aXJlcyBhbiBhZGFwdGVyIHRvIGRlZmluZSB0aGUgZGF0YSBzaGFwZScpO1xuICAgICAgcmV0dXJuIG9mKG51bGwpO1xuICAgIH1cblxuICAgIGlmICghdGhpcy5kYXRhYmFzZT8udGFibGUpIHtcbiAgICAgIGNvbnNvbGUud2FybignRGF0YWJhc2Ugc3RvcmFnZSByZXF1aXJlcyBhIHRhYmxlIG5hbWUnKTtcbiAgICAgIHJldHVybiBvZihudWxsKTtcbiAgICB9XG5cbiAgICBjb25zdCBzYW1wbGVEYXRhID0gdGhpcy5hcGlPcHRpb25zLmFkYXB0ZXI/Lih7fSkgfHwge307XG4gICAgY29uc3Qgc2NoZW1hS2V5cyA9IE9iamVjdC5rZXlzKHNhbXBsZURhdGEpLmZpbHRlcihrZXkgPT4gc2FtcGxlRGF0YVtrZXldICE9PSB1bmRlZmluZWQpO1xuXG4gICAgbGV0IHNjaGVtYSA9ICcrK2lkJztcblxuICAgIGlmIChzY2hlbWFLZXlzLmxlbmd0aCA+IDApIHtcbiAgICAgIGNvbnN0IG90aGVyS2V5cyA9IHNjaGVtYUtleXMuZmlsdGVyKGsgPT4gayAhPT0gJ2lkJyk7XG4gICAgICBpZiAob3RoZXJLZXlzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgc2NoZW1hICs9ICcsICcgKyBvdGhlcktleXMuam9pbignLCAnKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBjb25zdCB0YWJsZURlZiA9IFRhYmxlU2NoZW1hRGVmLmFkYXB0KHtcbiAgICAgIHRhYmxlOiB0aGlzLmRhdGFiYXNlPy50YWJsZSxcbiAgICAgIHNjaGVtYTogc2NoZW1hXG4gICAgfSk7XG5cbiAgICByZXR1cm4gdGhpcy5kYk1hbmFnZXJTZXJ2aWNlLmNyZWF0ZURhdGFiYXNlVGFibGUodGFibGVEZWYpO1xuXG4gIH1cblxuICAvLyBGRVRDSCBSRUNPUkRcbiAgcmVhZG9ubHkgZmV0Y2hSZWNvcmQgPSAob3B0aW9uczogUmVxdWVzdE9wdGlvbnMsIG1ldGhvZDogc3RyaW5nKSA9PlxuICAgIHRoaXMuZWZmZWN0PGFueT4oKCkgPT5cbiAgICBvZihSZXF1ZXN0T3B0aW9ucy5hZGFwdChvcHRpb25zKSkucGlwZShcbiAgICAgIHN3aXRjaE1hcCgob3B0aW9ucykgPT4ge1xuXG4gICAgICAgIHRoaXMuc3RyZWFtZWRSZXNwb25zZSA9IFtdXG4gICAgICAgIGNvbnN0IHJlcXVlc3RPcHRpb25zID0gdGhpcy51cGRhdGVSZXF1ZXN0T3B0aW9ucyhvcHRpb25zPy5oZWFkZXJzKVxuXG4gICAgICAgIHJldHVybiB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5nZXRSZXF1ZXN0KHJlcXVlc3RPcHRpb25zLCBvcHRpb25zPy5wYXRoKVxuICAgICAgICAucGlwZShcbiAgICAgICAgICB0YXAoKGRhdGE6IGFueSkgPT4ge1xuXG4gICAgICAgICAgICBkYXRhID0gKCFkYXRhKSA/ICh0aGlzLmRhdGFUeXBlID09PSBEYXRhVHlwZS5BUlJBWSkgPyBbXSA6IHt9IDogZGF0YVxuXG4gICAgICAgICAgICBjb25zdCBpZCA9IChvcHRpb25zIGFzIGFueSkucGF0aD8ubGVuZ3RoID8gb3B0aW9ucy5wYXRoW29wdGlvbnMucGF0aC5sZW5ndGggLTFdIDogbnVsbFxuXG4gICAgICAgICAgICBpZihtZXRob2QgPT09ICdERUxFVEUnKSB0aGlzLmRlbGV0ZURhdGEkKHsgaWQgfSBhcyBUICYgeyBpZDogbnVtYmVyIH0pXG4gICAgICAgICAgICBpZihtZXRob2QgPT09ICdVUERBVEUnKSB0aGlzLnVwZGF0ZURhdGEkKGRhdGEpXG4gICAgICAgICAgICBpZihtZXRob2QgPT09ICdDUkVBVEUnKSB0aGlzLmFkZERhdGEkKGRhdGEpXG5cbiAgICAgICAgICB9KSxcbiAgICAgICAgICBjb25jYXRNYXAoKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICAgICAgaWYgKHRoaXMuaGFzRGF0YWJhc2UgJiYgdGhpcy5kYXRhYmFzZT8udGFibGUpIHtcbiAgICAgICAgICAgICAgY29uc3QgaWQgPSAob3B0aW9ucyBhcyBhbnkpLnBhdGg/Lmxlbmd0aCA/IG9wdGlvbnMucGF0aFtvcHRpb25zLnBhdGgubGVuZ3RoIC0xXSA6IG51bGxcblxuICAgICAgICAgICAgICBpZihtZXRob2QgPT09ICdERUxFVEUnICYmIGlkKSByZXR1cm4gdGhpcy5kYk1hbmFnZXJTZXJ2aWNlLmRlbGV0ZVRhYmxlUmVjb3JkKHRoaXMuZGF0YWJhc2UudGFibGUsIGlkKVxuICAgICAgICAgICAgICBpZihtZXRob2QgPT09ICdVUERBVEUnICYmIGRhdGEpIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UudXBkYXRlVGFibGVSZWNvcmQodGhpcy5kYXRhYmFzZS50YWJsZSwgZGF0YSlcbiAgICAgICAgICAgICAgaWYobWV0aG9kID09PSAnQ1JFQVRFJyAmJiBkYXRhKSByZXR1cm4gdGhpcy5kYk1hbmFnZXJTZXJ2aWNlLmNyZWF0ZVRhYmxlUmVjb3JkKHRoaXMuZGF0YWJhc2UudGFibGUsIGRhdGEpXG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gb2YoZGF0YSlcbiAgICAgICAgICB9KVxuICAgICAgICApXG5cbiAgICAgIH0pXG4gICAgKVxuICApXG5cbiAgLy8gQ1JFQVRFIFJFQ09SRFxuICByZWFkb25seSBjcmVhdGVSZWNvcmQgPSAoZGF0YTogYW55fG51bGwsIG9wdGlvbnM/OiBSZXF1ZXN0T3B0aW9ucykgPT5cbiAgICB0aGlzLmVmZmVjdDxhbnk+KCgpID0+XG4gICAgICBvZihkYXRhKS5waXBlKFxuICAgICAgICBzd2l0Y2hNYXAoKGRhdGE6IGFueSkgPT4ge1xuXG4gICAgICAgIHRoaXMuc3RyZWFtZWRSZXNwb25zZSA9IFtdXG4gICAgICAgIGNvbnN0IHJlcXVlc3RPcHRpb25zID0gdGhpcy51cGRhdGVSZXF1ZXN0T3B0aW9ucyhvcHRpb25zPy5oZWFkZXJzKVxuXG4gICAgICAgIHJldHVybiB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5wb3N0UmVxdWVzdChkYXRhLCByZXF1ZXN0T3B0aW9ucywgb3B0aW9ucz8ucGF0aClcbiAgICAgICAgLnBpcGUoXG4gICAgICAgICAgdGFwKChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgICAgIGRhdGEgPSAoIWRhdGEpID8gKHRoaXMuZGF0YVR5cGUgPT09IERhdGFUeXBlLkFSUkFZKSA/IFtdIDoge30gOiBkYXRhXG4gICAgICAgICAgICB0aGlzLmFkZERhdGEkKGRhdGEpXG4gICAgICAgICAgICBpZih0aGlzLndzQ29ubmVjdGlvbikgdGhpcy53c0NvbW11bmljYXRpb24oJ0NSRUFURScsIFsuLi5vcHRpb25zPy5wYXRoIHx8IFtdLCBkYXRhLmlkXSlcbiAgICAgICAgICB9KSxcbiAgICAgICAgICBjb25jYXRNYXAoKGRhdGE6IGFueSkgPT4ge1xuICAgICAgICAgICAgaWYgKHRoaXMuaGFzRGF0YWJhc2UgJiYgdGhpcy5kYXRhYmFzZT8udGFibGUgJiYgZGF0YT8uaWQpIHtcbiAgICAgICAgICAgICAgcmV0dXJuIHRoaXMuZGJNYW5hZ2VyU2VydmljZS5jcmVhdGVUYWJsZVJlY29yZCh0aGlzLmRhdGFiYXNlLnRhYmxlLCBkYXRhKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJldHVybiBvZihkYXRhKTtcbiAgICAgICAgICB9KVxuICAgICAgICApXG5cbiAgICAgIH0pXG4gICAgKVxuICApXG5cbiAgLy8gVVBEQVRFIFJFQ09SRFxuICByZWFkb25seSB1cGRhdGVSZWNvcmQgPSAoZGF0YTogYW55fG51bGwsIG9wdGlvbnM/OiBSZXF1ZXN0T3B0aW9ucykgPT5cbiAgICB0aGlzLmVmZmVjdDxhbnk+KCgpID0+XG4gICAgb2YoZGF0YSkucGlwZShcbiAgICAgIGNvbmNhdE1hcCgoZGF0YTogYW55KSA9PiB7XG5cbiAgICAgICAgdGhpcy5zdHJlYW1lZFJlc3BvbnNlID0gW11cbiAgICAgICAgY29uc3QgcmVxdWVzdE9wdGlvbnMgPSB0aGlzLnVwZGF0ZVJlcXVlc3RPcHRpb25zKG9wdGlvbnM/LmhlYWRlcnMpXG5cbiAgICAgICAgcmV0dXJuIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLnB1dFJlcXVlc3QoZGF0YSwgcmVxdWVzdE9wdGlvbnMsIG9wdGlvbnM/LnBhdGgpXG4gICAgICAgIC5waXBlKFxuICAgICAgICAgIHRhcCgoZGF0YTogYW55KSA9PiB7XG4gICAgICAgICAgICBkYXRhID0gKCFkYXRhKSA/ICh0aGlzLmRhdGFUeXBlID09PSBEYXRhVHlwZS5BUlJBWSkgPyBbXSA6IHt9IDogZGF0YVxuICAgICAgICAgICAgdGhpcy51cGRhdGVEYXRhJChkYXRhKVxuICAgICAgICAgICAgaWYodGhpcy53c0Nvbm5lY3Rpb24pIHRoaXMud3NDb21tdW5pY2F0aW9uKCdVUERBVEUnLCBbLi4ub3B0aW9ucz8ucGF0aCB8fCBbXV0pXG4gICAgICAgICAgfSksXG4gICAgICAgICAgY29uY2F0TWFwKChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgICAgIGlmICh0aGlzLmhhc0RhdGFiYXNlICYmIHRoaXMuZGF0YWJhc2U/LnRhYmxlICYmIGRhdGE/LmlkKSB7XG4gICAgICAgICAgICAgIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UudXBkYXRlVGFibGVSZWNvcmQodGhpcy5kYXRhYmFzZS50YWJsZSwgZGF0YSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gb2YoZGF0YSk7XG4gICAgICAgICAgfSlcbiAgICAgICAgKVxuXG4gICAgICB9KVxuICAgIClcbiAgKVxuXG4gIC8vIERFTEVURSBSRUNPUkRcbiAgcmVhZG9ubHkgZGVsZXRlUmVjb3JkID0gKG9wdGlvbnM/OiBSZXF1ZXN0T3B0aW9ucykgPT5cbiAgICB0aGlzLmVmZmVjdDxhbnk+KCgpID0+XG4gICAgb2Yob3B0aW9ucykucGlwZShcbiAgICAgIGNvbmNhdE1hcCgoZGF0YTogYW55KSA9PiB7XG5cbiAgICAgICAgdGhpcy5zdHJlYW1lZFJlc3BvbnNlID0gW11cbiAgICAgICAgY29uc3QgcmVxdWVzdE9wdGlvbnMgPSB0aGlzLnVwZGF0ZVJlcXVlc3RPcHRpb25zKG9wdGlvbnM/LmhlYWRlcnMpXG5cbiAgICAgICAgcmV0dXJuIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLmRlbGV0ZVJlcXVlc3QocmVxdWVzdE9wdGlvbnMsIG9wdGlvbnM/LnBhdGgpXG4gICAgICAgIC5waXBlKFxuICAgICAgICAgIHRhcCgoZGF0YTogYW55KSA9PiB7XG4gICAgICAgICAgICBkYXRhID0gKCFkYXRhKSA/ICh0aGlzLmRhdGFUeXBlID09PSBEYXRhVHlwZS5BUlJBWSkgPyBbXSA6IHt9IDogZGF0YVxuICAgICAgICAgICAgdGhpcy5kZWxldGVEYXRhJChkYXRhKVxuICAgICAgICAgICAgaWYodGhpcy53c0Nvbm5lY3Rpb24pIHRoaXMud3NDb21tdW5pY2F0aW9uKCdERUxFVEUnLCBbLi4ub3B0aW9ucz8ucGF0aCB8fCBbXV0pXG4gICAgICAgICAgfSksXG4gICAgICAgICAgY29uY2F0TWFwKChkYXRhOiBhbnkpID0+IHtcbiAgICAgICAgICAgIGlmICh0aGlzLmhhc0RhdGFiYXNlICYmIHRoaXMuZGF0YWJhc2U/LnRhYmxlICYmIGRhdGE/LmlkKSB7XG4gICAgICAgICAgICAgIHJldHVybiB0aGlzLmRiTWFuYWdlclNlcnZpY2UuZGVsZXRlVGFibGVSZWNvcmQodGhpcy5kYXRhYmFzZS50YWJsZSwgZGF0YS5pZCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXR1cm4gb2YoZGF0YSk7XG4gICAgICAgICAgfSlcbiAgICAgICAgKVxuXG4gICAgICB9KVxuICAgIClcbiAgKVxuXG4gIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gIC8vIEZFVENIIFNUUkVBTVxuXG4gIHJlYWRvbmx5IGNyZWF0ZVN0cmVhbSA9IChkYXRhOiBhbnl8bnVsbCwgb3B0aW9ucz86IFJlcXVlc3RPcHRpb25zKSA9PlxuICAgIHRoaXMuZWZmZWN0PGFueT4oKCkgPT5cbiAgICBvZihkYXRhKS5waXBlKFxuICAgICAgdGFwKCgpID0+IHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLmlzUGVuZGluZy5uZXh0KHRydWUpKSxcbiAgICAgIHN3aXRjaE1hcCgoZGF0YTogYW55KSA9PiB7XG5cbiAgICAgICAgY29uc3QgcmVxdWVzdE9wdGlvbnMgPSB0aGlzLnVwZGF0ZVJlcXVlc3RPcHRpb25zKG9wdGlvbnM/LmhlYWRlcnMpXG5cbiAgICAgICAgcmV0dXJuIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLnBvc3RSZXF1ZXN0KGRhdGEsIHJlcXVlc3RPcHRpb25zLCBvcHRpb25zPy5wYXRoKVxuICAgICAgICAucGlwZShcbiAgICAgICAgICB0YXAoKHJlczogYW55KSA9PiB7XG4gICAgICAgICAgICBpZihyZXMubGVuZ3RoID4gMCkgdGhpcy5zZXREYXRhJChyZXMpXG4gICAgICAgICAgICB0aGlzLnN0cmVhbWVkUmVzcG9uc2UgPSByZXNcbiAgICAgICAgICB9KSxcbiAgICAgICAgICBzY2FuKChhY2MsIHJlczogYW55KSA9PiB7XG5cbiAgICAgICAgICAgIGNvbnN0IHByZXZpb3VzID0gYWNjLmN1cnJlbnRcbiAgICAgICAgICAgIGNvbnN0IGN1cnJlbnQgPSByZXNcbiAgICAgICAgICAgIHJldHVybiB7IHByZXZpb3VzLCBjdXJyZW50IH07XG5cbiAgICAgICAgICB9LCB7IHByZXZpb3VzOiBudWxsLCBjdXJyZW50OiBudWxsIH0pLFxuICAgICAgICAgIHRhcCgoeyBwcmV2aW91cywgY3VycmVudCB9KSA9PiB7XG5cbiAgICAgICAgICAgIGlmKHByZXZpb3VzICYmIEpTT04uc3RyaW5naWZ5KHByZXZpb3VzKSA9PT0gSlNPTi5zdHJpbmdpZnkoY3VycmVudCkpIHtcbiAgICAgICAgICAgICAgdGhpcy5odHRwTWFuYWdlclNlcnZpY2UuaXNQZW5kaW5nLm5leHQoZmFsc2UpXG4gICAgICAgICAgICAgIHRoaXMuc2V0RGF0YSQoW10pXG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5pc1BlbmRpbmcubmV4dCh0cnVlKVxuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgfSlcbiAgICAgICAgKVxuXG4gICAgICB9KVxuICAgIClcblxuICApXG5cbiAgcmVhZG9ubHkgZmV0Y2hTdHJlYW0gPSAob3B0aW9ucz86IFJlcXVlc3RPcHRpb25zKSA9PlxuICAgIHRoaXMuZWZmZWN0PGFueT4oKCkgPT5cbiAgICBvZihvcHRpb25zKS5waXBlKFxuICAgICAgdGFwKCgpID0+IHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLmlzUGVuZGluZy5uZXh0KHRydWUpKSxcbiAgICAgIHN3aXRjaE1hcCgob3B0aW9uczogYW55KSA9PiB7XG5cbiAgICAgICAgY29uc3QgcmVxdWVzdE9wdGlvbnMgPSB0aGlzLnVwZGF0ZVJlcXVlc3RPcHRpb25zKG9wdGlvbnM/LmhlYWRlcnMpXG4gICAgICAgIHJlcXVlc3RPcHRpb25zLnN0cmVhbSA9IHRydWVcblxuICAgICAgICByZXR1cm4gdGhpcy5odHRwTWFuYWdlclNlcnZpY2UuZ2V0UmVxdWVzdChyZXF1ZXN0T3B0aW9ucywgb3B0aW9ucz8ucGF0aClcbiAgICAgICAgLnBpcGUoXG4gICAgICAgICAgdGFwKChyZXM6IGFueSkgPT4ge1xuXG4gICAgICAgICAgICBpZihyZXMubGVuZ3RoID4gMCkgdGhpcy5zZXREYXRhJChyZXMpXG4gICAgICAgICAgICB0aGlzLnN0cmVhbWVkUmVzcG9uc2UgPSByZXNcbiAgICAgICAgICB9KSxcbiAgICAgICAgICBzY2FuKChhY2MsIHJlczogYW55KSA9PiB7XG5cbiAgICAgICAgICAgIGNvbnN0IHByZXZpb3VzID0gYWNjLmN1cnJlbnRcbiAgICAgICAgICAgIGNvbnN0IGN1cnJlbnQgPSByZXNcbiAgICAgICAgICAgIHJldHVybiB7IHByZXZpb3VzLCBjdXJyZW50IH1cblxuICAgICAgICAgIH0sIHsgcHJldmlvdXM6IG51bGwsIGN1cnJlbnQ6IG51bGwgfSksXG4gICAgICAgICAgdGFwKCh7IHByZXZpb3VzLCBjdXJyZW50IH0pID0+IHtcblxuICAgICAgICAgICAgaWYocHJldmlvdXMgJiYgSlNPTi5zdHJpbmdpZnkocHJldmlvdXMpID09PSBKU09OLnN0cmluZ2lmeShjdXJyZW50KSkge1xuICAgICAgICAgICAgICB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5pc1BlbmRpbmcubmV4dChmYWxzZSlcbiAgICAgICAgICAgICAgdGhpcy5zZXREYXRhJChbXSlcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLmlzUGVuZGluZy5uZXh0KHRydWUpXG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICB9KVxuICAgICAgICApXG5cbiAgICAgIH0pLFxuICAgIClcbiAgKVxuXG4gIHByaXZhdGUgd3NDb21tdW5pY2F0aW9uKG1ldGhvZDogc3RyaW5nLCBwYXRoPzogc3RyaW5nfG51bWJlcltdKSB7XG5cbiAgICBpZih0aGlzLndzQ29ubmVjdGlvbiAmJiB0aGlzLmFwaU9wdGlvbnMud3MpIHtcbiAgICAgIGNvbnN0IHdzU2VydmVyID0gdGhpcy5hcGlPcHRpb25zLndzLmlkXG4gICAgICB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5zZW5kTWVzc2FnZUluQ2hhbm5lbCh3c1NlcnZlciwgeyBtZXRob2QsIHBhdGggfSlcbiAgICB9XG5cbiAgfVxuXG4gIHdzTWVzc2FnaW5nKG1lc3NhZ2U6IENoYW5uZWxNZXNzYWdlLCBjaGFubmVsPzogc3RyaW5nKSB7XG5cbiAgICBjb25zdCB1c2VyID0gdGhpcy53c09wdGlvbnM/LnVzZXJBZGFwdGVyPy5hZGFwdGVyPy4odGhpcy51c2VyLnZhbHVlKSB8fCB0aGlzLnVzZXIudmFsdWVcbiAgICBjb25zdCBtZXNzYWdlSW5mbyA9IENoYW5uZWxNZXNzYWdlLmFkYXB0KHsgLi4ubWVzc2FnZSwgZnJvbVVzZXI6IHVzZXIgfSlcblxuICAgIGlmKHRoaXMud3NDb25uZWN0aW9uICYmIHRoaXMuYXBpT3B0aW9ucy53cykge1xuXG4gICAgICBjb25zdCB3c1NlcnZlciA9IChjaGFubmVsKSA/IGNoYW5uZWwgOiB0aGlzLmFwaU9wdGlvbnMud3MuaWRcblxuICAgICAgaWYobWVzc2FnZUluZm8udG9Vc2VyID09PSAnYWxsQ2hhbm5lbHMnKSB7XG4gICAgICAgIHRoaXMuaHR0cE1hbmFnZXJTZXJ2aWNlLnNlbmRCcm9hZGNhc3QobWVzc2FnZUluZm8pXG4gICAgICB9IGVsc2UgaWYobWVzc2FnZUluZm8udG9Vc2VyID09PSAnYWxsSW5DaGFubmVsJykge1xuICAgICAgICB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5zZW5kTWVzc2FnZUluQ2hhbm5lbCh3c1NlcnZlciwgbWVzc2FnZUluZm8pXG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0aGlzLmh0dHBNYW5hZ2VyU2VydmljZS5zZW5kTWVzc2FnZVRvVXNlcih3c1NlcnZlciwgbWVzc2FnZUluZm8pXG4gICAgICB9XG5cbiAgICB9XG5cbiAgfVxuXG4gIC8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gIC8vIE1JU0NcblxuICBwcml2YXRlIGlzRW1wdHkob2JqOiBhbnkpIHtcbiAgICByZXR1cm4gT2JqZWN0LmtleXMob2JqKS5sZW5ndGggPT09IDBcbiAgfVxuXG4gIHByaXZhdGUgdXBkYXRlUmVxdWVzdE9wdGlvbnMoaGVhZGVycz86IGFueSk6IEFwaVJlcXVlc3Qge1xuXG4gICAgY29uc3Qgb3B0aW9ucyA9IEFwaVJlcXVlc3QuYWRhcHQoeyAuLi50aGlzLmFwaU9wdGlvbnMgfSlcblxuICAgIG9wdGlvbnMuaGVhZGVycyA9IChoZWFkZXJzKVxuICAgICAgICAgID8geyAuLi5vcHRpb25zLmhlYWRlcnMsIC4uLmhlYWRlcnMgfVxuICAgICAgICAgIDogeyAuLi5vcHRpb25zLmhlYWRlcnMgfVxuXG4gICAgcmV0dXJuIG9wdGlvbnNcbiAgfVxuXG59XG4iXX0=