mobx-keystone-yjs 1.5.4 → 1.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (31) hide show
  1. package/CHANGELOG.md +57 -45
  2. package/dist/mobx-keystone-yjs.esm.js +475 -299
  3. package/dist/mobx-keystone-yjs.esm.mjs +475 -299
  4. package/dist/mobx-keystone-yjs.umd.js +475 -299
  5. package/dist/types/binding/YjsTextModel.d.ts +5 -4
  6. package/dist/types/binding/applyMobxChangeToYjsObject.d.ts +3 -0
  7. package/dist/types/binding/applyYjsEventToMobx.d.ts +8 -0
  8. package/dist/types/binding/bindYjsToMobxKeystone.d.ts +1 -1
  9. package/dist/types/binding/convertJsonToYjsData.d.ts +23 -4
  10. package/dist/types/binding/convertYjsDataToJson.d.ts +1 -1
  11. package/dist/types/binding/resolveYjsPath.d.ts +14 -1
  12. package/dist/types/binding/yjsBindingContext.d.ts +2 -2
  13. package/dist/types/binding/yjsSnapshotTracking.d.ts +24 -0
  14. package/dist/types/index.d.ts +7 -6
  15. package/dist/types/utils/isYjsValueDeleted.d.ts +7 -0
  16. package/package.json +90 -78
  17. package/src/binding/YjsTextModel.ts +280 -247
  18. package/src/binding/applyMobxChangeToYjsObject.ts +77 -0
  19. package/src/binding/applyYjsEventToMobx.ts +173 -0
  20. package/src/binding/bindYjsToMobxKeystone.ts +300 -192
  21. package/src/binding/convertJsonToYjsData.ts +218 -76
  22. package/src/binding/convertYjsDataToJson.ts +1 -1
  23. package/src/binding/resolveYjsPath.ts +51 -27
  24. package/src/binding/yjsSnapshotTracking.ts +40 -0
  25. package/src/index.ts +11 -10
  26. package/src/utils/getOrCreateYjsCollectionAtom.ts +27 -27
  27. package/src/utils/isYjsValueDeleted.ts +14 -0
  28. package/dist/types/binding/applyMobxKeystonePatchToYjsObject.d.ts +0 -2
  29. package/dist/types/binding/convertYjsEventToPatches.d.ts +0 -3
  30. package/src/binding/applyMobxKeystonePatchToYjsObject.ts +0 -98
  31. package/src/binding/convertYjsEventToPatches.ts +0 -92
@@ -1,19 +1,9 @@
1
1
  var __defProp = Object.defineProperty;
2
2
  var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
3
3
  var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
4
- import { createAtom, reaction, observe, computed, action, runInAction } from "mobx";
5
- import { createContext, types, Model, tProp, frozen, getParentToChildPath, onSnapshot, model, modelSnapshotOutWithMetadata, applyPatches, onPatches, onGlobalPatches, fromSnapshot } from "mobx-keystone";
4
+ import { createAtom, reaction, observe, computed, action, remove } from "mobx";
5
+ import { createContext, Model, tProp, types, frozen, getParentToChildPath, onSnapshot, model, frozenKey, modelTypeKey, DeepChangeType, modelSnapshotOutWithMetadata, resolvePath, runUnprotected, isModel, getSnapshot, getSnapshotModelId, isFrozenSnapshot, fromSnapshot, onGlobalDeepChange, onDeepChange, isTreeNode } from "mobx-keystone";
6
6
  import * as Y from "yjs";
7
- function __decorate(decorators, target, key, desc) {
8
- var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
9
- if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
10
- else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
11
- return c > 3 && r && Object.defineProperty(target, key, r), r;
12
- }
13
- typeof SuppressedError === "function" ? SuppressedError : function(error, suppressed, message) {
14
- var e = new Error(message);
15
- return e.name = "SuppressedError", e.error = error, e.suppressed = suppressed, e;
16
- };
17
7
  class MobxKeystoneYjsError extends Error {
18
8
  constructor(msg) {
19
9
  super(msg);
@@ -23,7 +13,6 @@ class MobxKeystoneYjsError extends Error {
23
13
  function failure(msg) {
24
14
  return new MobxKeystoneYjsError(msg);
25
15
  }
26
- const yjsBindingContext = createContext(void 0);
27
16
  const yjsCollectionAtoms = /* @__PURE__ */ new WeakMap();
28
17
  const getYjsCollectionAtom = (yjsCollection) => {
29
18
  return yjsCollectionAtoms.get(yjsCollection);
@@ -36,9 +25,21 @@ const getOrCreateYjsCollectionAtom = (yjsCollection) => {
36
25
  }
37
26
  return atom;
38
27
  };
28
+ function isYjsValueDeleted(yjsValue) {
29
+ var _a, _b;
30
+ if (yjsValue instanceof Y.AbstractType) {
31
+ return !!((_a = yjsValue._item) == null ? void 0 : _a.deleted) || !!((_b = yjsValue.doc) == null ? void 0 : _b.isDestroyed);
32
+ }
33
+ return false;
34
+ }
39
35
  function resolveYjsPath(yjsObject, path) {
40
36
  let currentYjsObject = yjsObject;
41
- path.forEach((pathPart, i) => {
37
+ let i = -1;
38
+ for (const pathPart of path) {
39
+ i++;
40
+ if (currentYjsObject instanceof Y.Text) {
41
+ return currentYjsObject;
42
+ }
42
43
  if (currentYjsObject instanceof Y.Map) {
43
44
  getOrCreateYjsCollectionAtom(currentYjsObject).reportObserved();
44
45
  const key = String(pathPart);
@@ -48,14 +49,29 @@ function resolveYjsPath(yjsObject, path) {
48
49
  const key = Number(pathPart);
49
50
  currentYjsObject = currentYjsObject.get(key);
50
51
  } else {
51
- throw failure(`Y.Map or Y.Array was expected at path ${JSON.stringify(path.slice(0, i))} in order to resolve path ${JSON.stringify(path)}, but got ${currentYjsObject} instead`);
52
+ throw failure(
53
+ `Y.Map or Y.Array was expected at path ${JSON.stringify(
54
+ path.slice(0, i)
55
+ )} in order to resolve path ${JSON.stringify(path)}, but got ${currentYjsObject} instead`
56
+ );
52
57
  }
53
- });
58
+ }
54
59
  return currentYjsObject;
55
60
  }
61
+ const yjsBindingContext = createContext(void 0);
62
+ var __defProp2 = Object.defineProperty;
63
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
64
+ var __decorateClass = (decorators, target, key, kind) => {
65
+ var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target;
66
+ for (var i = decorators.length - 1, decorator; i >= 0; i--)
67
+ if (decorator = decorators[i])
68
+ result = (kind ? decorator(target, key, result) : decorator(result)) || result;
69
+ if (kind && result) __defProp2(target, key, result);
70
+ return result;
71
+ };
56
72
  const deltaListType = types.array(types.frozen(types.unchecked()));
57
73
  const yjsTextModelId = "mobx-keystone-yjs/YjsTextModel";
58
- let YjsTextModel = class YjsTextModel2 extends Model({
74
+ let YjsTextModel = class extends Model({
59
75
  deltaList: tProp(deltaListType, () => [])
60
76
  }) {
61
77
  constructor() {
@@ -79,13 +95,12 @@ let YjsTextModel = class YjsTextModel2 extends Model({
79
95
  ]
80
96
  });
81
97
  }
82
- /**
83
- * The Y.js path from the bound object to the YjsTextModel instance.
84
- */
85
98
  get _yjsObjectPath() {
86
99
  const ctx = yjsBindingContext.get(this);
87
100
  if ((ctx == null ? void 0 : ctx.boundObject) == null) {
88
- throw failure("the YjsTextModel instance must be part of a bound object before it can be accessed");
101
+ throw failure(
102
+ "the YjsTextModel instance must be part of a bound object before it can be accessed"
103
+ );
89
104
  }
90
105
  const path = getParentToChildPath(ctx.boundObject, this);
91
106
  if (!path) {
@@ -93,17 +108,11 @@ let YjsTextModel = class YjsTextModel2 extends Model({
93
108
  }
94
109
  return path;
95
110
  }
96
- /**
97
- * The Yjs.Text object present at this mobx-keystone node's path.
98
- */
99
111
  get _yjsObjectAtPath() {
100
112
  const path = this._yjsObjectPath;
101
113
  const ctx = yjsBindingContext.get(this);
102
114
  return resolveYjsPath(ctx.yjsObject, path);
103
115
  }
104
- /**
105
- * The Yjs.Text object represented by this mobx-keystone node.
106
- */
107
116
  get yjsText() {
108
117
  const yjsObject = this._yjsObjectAtPath;
109
118
  if (!(yjsObject instanceof Y.Text)) {
@@ -111,13 +120,27 @@ let YjsTextModel = class YjsTextModel2 extends Model({
111
120
  }
112
121
  return yjsObject;
113
122
  }
114
- /**
115
- * The text value of the Yjs.Text object.
116
- * Shortcut for `yjsText.toString()`, but computed.
117
- */
118
123
  get text() {
119
124
  this.yjsTextChangedAtom.reportObserved();
120
- return this.yjsText.toString();
125
+ const ctx = yjsBindingContext.get(this);
126
+ if ((ctx == null ? void 0 : ctx.boundObject) != null) {
127
+ try {
128
+ const yjsTextString = this.yjsText.toString();
129
+ if (yjsTextString !== "" || this.deltaList.length === 0) {
130
+ return yjsTextString;
131
+ }
132
+ } catch {
133
+ }
134
+ }
135
+ return this.deltaListToText();
136
+ }
137
+ deltaListToText() {
138
+ const doc = new Y.Doc();
139
+ const text = doc.getText();
140
+ this.deltaList.forEach((d) => {
141
+ text.applyDelta(d.data);
142
+ });
143
+ return text.toString();
121
144
  }
122
145
  onInit() {
123
146
  const shouldReplicateToYjs = (ctx) => {
@@ -126,29 +149,36 @@ let YjsTextModel = class YjsTextModel2 extends Model({
126
149
  let reapplyDeltasToYjsText = false;
127
150
  const newDeltas = [];
128
151
  let disposeObserveDeltaList;
129
- const disposeReactionToDeltaListRefChange = reaction(() => this.$.deltaList, (deltaList) => {
130
- disposeObserveDeltaList == null ? void 0 : disposeObserveDeltaList();
131
- disposeObserveDeltaList = void 0;
132
- disposeObserveDeltaList = observe(deltaList, (change) => {
133
- if (reapplyDeltasToYjsText) {
134
- return;
135
- }
136
- if (!shouldReplicateToYjs(yjsBindingContext.get(this))) {
137
- return;
138
- }
139
- if (change.type === "splice" && change.removedCount === 0 && change.addedCount > 0 && change.index === this.deltaList.length) {
140
- newDeltas.push(...change.added);
141
- } else {
142
- reapplyDeltasToYjsText = true;
143
- }
144
- });
145
- }, { fireImmediately: true });
152
+ const disposeReactionToDeltaListRefChange = reaction(
153
+ () => this.$.deltaList,
154
+ (deltaList) => {
155
+ disposeObserveDeltaList == null ? void 0 : disposeObserveDeltaList();
156
+ disposeObserveDeltaList = void 0;
157
+ disposeObserveDeltaList = observe(deltaList, (change) => {
158
+ if (reapplyDeltasToYjsText) {
159
+ return;
160
+ }
161
+ if (!shouldReplicateToYjs(yjsBindingContext.get(this))) {
162
+ return;
163
+ }
164
+ if (change.type === "splice" && change.removedCount === 0 && change.addedCount > 0 && change.index === this.deltaList.length) {
165
+ newDeltas.push(...change.added);
166
+ } else {
167
+ reapplyDeltasToYjsText = true;
168
+ }
169
+ });
170
+ },
171
+ { fireImmediately: true }
172
+ );
146
173
  const disposeOnSnapshot = onSnapshot(this, () => {
147
174
  try {
148
175
  if (reapplyDeltasToYjsText) {
149
176
  const ctx = yjsBindingContext.get(this);
150
177
  if (shouldReplicateToYjs(ctx)) {
151
178
  const { yjsText } = this;
179
+ if (isYjsValueDeleted(yjsText)) {
180
+ throw failure("cannot reapply deltas to deleted Yjs.Text");
181
+ }
152
182
  ctx.yjsDoc.transact(() => {
153
183
  if (yjsText.length > 0) {
154
184
  yjsText.delete(0, yjsText.length);
@@ -162,6 +192,9 @@ let YjsTextModel = class YjsTextModel2 extends Model({
162
192
  const ctx = yjsBindingContext.get(this);
163
193
  if (shouldReplicateToYjs(ctx)) {
164
194
  const { yjsText } = this;
195
+ if (isYjsValueDeleted(yjsText)) {
196
+ throw failure("cannot reapply deltas to deleted Yjs.Text");
197
+ }
165
198
  ctx.yjsDoc.transact(() => {
166
199
  newDeltas.forEach((frozenDeltas) => {
167
200
  yjsText.applyDelta(frozenDeltas.data);
@@ -174,7 +207,10 @@ let YjsTextModel = class YjsTextModel2 extends Model({
174
207
  newDeltas.length = 0;
175
208
  }
176
209
  });
177
- const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(() => this.yjsText, this.yjsTextChangedAtom);
210
+ const diposeYjsTextChangedAtom = hookYjsTextChangedAtom(
211
+ () => this.yjsText,
212
+ this.yjsTextChangedAtom
213
+ );
178
214
  return () => {
179
215
  disposeOnSnapshot();
180
216
  disposeReactionToDeltaListRefChange();
@@ -184,19 +220,19 @@ let YjsTextModel = class YjsTextModel2 extends Model({
184
220
  };
185
221
  }
186
222
  };
187
- __decorate([
223
+ __decorateClass([
188
224
  computed
189
- ], YjsTextModel.prototype, "_yjsObjectPath", null);
190
- __decorate([
225
+ ], YjsTextModel.prototype, "_yjsObjectPath", 1);
226
+ __decorateClass([
191
227
  computed
192
- ], YjsTextModel.prototype, "_yjsObjectAtPath", null);
193
- __decorate([
228
+ ], YjsTextModel.prototype, "_yjsObjectAtPath", 1);
229
+ __decorateClass([
194
230
  computed
195
- ], YjsTextModel.prototype, "yjsText", null);
196
- __decorate([
231
+ ], YjsTextModel.prototype, "yjsText", 1);
232
+ __decorateClass([
197
233
  computed
198
- ], YjsTextModel.prototype, "text", null);
199
- YjsTextModel = __decorate([
234
+ ], YjsTextModel.prototype, "text", 1);
235
+ YjsTextModel = __decorateClass([
200
236
  model(yjsTextModelId)
201
237
  ], YjsTextModel);
202
238
  const DecoratedYjsTextModel = YjsTextModel;
@@ -205,31 +241,43 @@ function hookYjsTextChangedAtom(getYjsText, textChangedAtom) {
205
241
  const observeFn = () => {
206
242
  textChangedAtom.reportChanged();
207
243
  };
208
- const disposeReactionToYTextChange = reaction(() => {
209
- try {
210
- return getYjsText();
211
- } catch {
212
- return void 0;
213
- }
214
- }, (yjsText) => {
215
- disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
216
- disposeObserveYjsText = void 0;
217
- if (yjsText) {
218
- yjsText.observe(observeFn);
219
- disposeObserveYjsText = () => {
220
- yjsText.unobserve(observeFn);
221
- };
244
+ const disposeReactionToYTextChange = reaction(
245
+ () => {
246
+ try {
247
+ const yjsText = getYjsText();
248
+ return isYjsValueDeleted(yjsText) ? void 0 : yjsText;
249
+ } catch {
250
+ return void 0;
251
+ }
252
+ },
253
+ (yjsText) => {
254
+ disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
255
+ disposeObserveYjsText = void 0;
256
+ if (yjsText) {
257
+ yjsText.observe(observeFn);
258
+ disposeObserveYjsText = () => {
259
+ yjsText.unobserve(observeFn);
260
+ };
261
+ }
262
+ textChangedAtom.reportChanged();
263
+ },
264
+ {
265
+ fireImmediately: true
222
266
  }
223
- textChangedAtom.reportChanged();
224
- }, {
225
- fireImmediately: true
226
- });
267
+ );
227
268
  return () => {
228
269
  disposeReactionToYTextChange();
229
270
  disposeObserveYjsText == null ? void 0 : disposeObserveYjsText();
230
271
  disposeObserveYjsText = void 0;
231
272
  };
232
273
  }
274
+ const yjsContainerToSnapshot = /* @__PURE__ */ new WeakMap();
275
+ function setYjsContainerSnapshot(container, snapshot) {
276
+ yjsContainerToSnapshot.set(container, snapshot);
277
+ }
278
+ function isYjsContainerUpToDate(container, snapshot) {
279
+ return yjsContainerToSnapshot.get(container) === snapshot;
280
+ }
233
281
  function isPlainPrimitive(v) {
234
282
  const t = typeof v;
235
283
  return t === "string" || t === "number" || t === "boolean" || v === null || v === void 0;
@@ -238,118 +286,163 @@ function isPlainArray(v) {
238
286
  return Array.isArray(v);
239
287
  }
240
288
  function isPlainObject(v) {
241
- return !isPlainArray(v) && typeof v === "object" && v !== null;
289
+ return typeof v === "object" && v !== null && !Array.isArray(v);
242
290
  }
243
291
  function convertJsonToYjsData(v) {
244
- return runInAction(() => {
245
- if (isPlainPrimitive(v)) {
292
+ if (isPlainPrimitive(v)) {
293
+ return v;
294
+ }
295
+ if (isPlainArray(v)) {
296
+ const arr = new Y.Array();
297
+ applyJsonArrayToYArray(arr, v);
298
+ return arr;
299
+ }
300
+ if (isPlainObject(v)) {
301
+ if (v[frozenKey] === true) {
246
302
  return v;
247
303
  }
248
- if (isPlainArray(v)) {
249
- const arr = new Y.Array();
250
- applyJsonArrayToYArray(arr, v);
251
- return arr;
304
+ if (v[modelTypeKey] === yjsTextModelId) {
305
+ const text = new Y.Text();
306
+ const yjsTextModel = v;
307
+ yjsTextModel.deltaList.forEach((frozenDeltas) => {
308
+ text.applyDelta(frozenDeltas.data);
309
+ });
310
+ return text;
252
311
  }
253
- if (isPlainObject(v)) {
254
- if (v.$frozen === true) {
255
- return v;
256
- }
257
- if (v.$modelType === yjsTextModelId) {
258
- const text = new Y.Text();
259
- const yjsTextModel = v;
260
- yjsTextModel.deltaList.forEach((frozenDeltas) => {
261
- text.applyDelta(frozenDeltas.data);
262
- });
263
- return text;
312
+ const map = new Y.Map();
313
+ applyJsonObjectToYMap(map, v);
314
+ return map;
315
+ }
316
+ throw new Error(`unsupported value type: ${v}`);
317
+ }
318
+ const applyJsonArrayToYArray = (dest, source, options = {}) => {
319
+ const { mode = "add" } = options;
320
+ if (mode === "merge" && isYjsContainerUpToDate(dest, source)) {
321
+ return;
322
+ }
323
+ const srcLen = source.length;
324
+ if (mode === "add") {
325
+ for (let i = 0; i < srcLen; i++) {
326
+ dest.push([convertJsonToYjsData(source[i])]);
327
+ }
328
+ return;
329
+ }
330
+ const destLen = dest.length;
331
+ if (destLen > srcLen) {
332
+ dest.delete(srcLen, destLen - srcLen);
333
+ }
334
+ const minLen = Math.min(destLen, srcLen);
335
+ for (let i = 0; i < minLen; i++) {
336
+ const srcItem = source[i];
337
+ const destItem = dest.get(i);
338
+ if (isPlainObject(srcItem) && destItem instanceof Y.Map) {
339
+ applyJsonObjectToYMap(destItem, srcItem, options);
340
+ continue;
341
+ }
342
+ if (isPlainArray(srcItem) && destItem instanceof Y.Array) {
343
+ applyJsonArrayToYArray(destItem, srcItem, options);
344
+ continue;
345
+ }
346
+ if (isPlainPrimitive(srcItem) && destItem === srcItem) {
347
+ continue;
348
+ }
349
+ dest.delete(i, 1);
350
+ dest.insert(i, [convertJsonToYjsData(srcItem)]);
351
+ }
352
+ for (let i = destLen; i < srcLen; i++) {
353
+ dest.push([convertJsonToYjsData(source[i])]);
354
+ }
355
+ setYjsContainerSnapshot(dest, source);
356
+ };
357
+ const applyJsonObjectToYMap = (dest, source, options = {}) => {
358
+ const { mode = "add" } = options;
359
+ if (mode === "merge" && isYjsContainerUpToDate(dest, source)) {
360
+ return;
361
+ }
362
+ if (mode === "add") {
363
+ for (const k of Object.keys(source)) {
364
+ const v = source[k];
365
+ if (v !== void 0) {
366
+ dest.set(k, convertJsonToYjsData(v));
264
367
  }
265
- const map = new Y.Map();
266
- applyJsonObjectToYMap(map, v);
267
- return map;
268
368
  }
269
- throw new Error(`unsupported value type: ${v}`);
270
- });
271
- }
272
- const applyJsonArrayToYArray = action((dest, source) => {
273
- dest.push(source.map(convertJsonToYjsData));
274
- });
275
- const applyJsonObjectToYMap = action((dest, source) => {
276
- Object.entries(source).forEach(([k, v]) => {
369
+ return;
370
+ }
371
+ const sourceKeysWithValues = new Set(Object.keys(source).filter((k) => source[k] !== void 0));
372
+ for (const key of dest.keys()) {
373
+ if (!sourceKeysWithValues.has(key)) {
374
+ dest.delete(key);
375
+ }
376
+ }
377
+ for (const k of Object.keys(source)) {
378
+ const v = source[k];
379
+ if (v === void 0) {
380
+ continue;
381
+ }
382
+ const existing = dest.get(k);
383
+ if (isPlainObject(v) && existing instanceof Y.Map) {
384
+ applyJsonObjectToYMap(existing, v, options);
385
+ continue;
386
+ }
387
+ if (isPlainArray(v) && existing instanceof Y.Array) {
388
+ applyJsonArrayToYArray(existing, v, options);
389
+ continue;
390
+ }
391
+ if (isPlainPrimitive(v) && existing === v) {
392
+ continue;
393
+ }
277
394
  dest.set(k, convertJsonToYjsData(v));
278
- });
279
- });
280
- function applyMobxKeystonePatchToYjsObject(patch, yjs) {
281
- if (patch.path.length > 1) {
282
- const [key, ...rest] = patch.path;
283
- if (yjs instanceof Y.Map) {
284
- const child = yjs.get(String(key));
285
- if (child === void 0) {
286
- throw failure(`invalid patch path, key "${key}" not found in Yjs map - patch: ${JSON.stringify(patch)}`);
287
- }
288
- applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child);
289
- } else if (yjs instanceof Y.Array) {
290
- const child = yjs.get(Number(key));
291
- if (child === void 0) {
292
- throw failure(`invalid patch path, key "${key}" not found in Yjs array - patch: ${JSON.stringify(patch)}`);
293
- }
294
- applyMobxKeystonePatchToYjsObject({ ...patch, path: rest }, child);
295
- } else if (yjs instanceof Y.Text) ;
296
- else {
297
- throw failure(`invalid patch path, key "${key}" not found in unknown Yjs object - patch: ${JSON.stringify(patch)}`);
298
- }
299
- } else if (patch.path.length === 1) {
300
- if (yjs instanceof Y.Map) {
301
- const key = String(patch.path[0]);
302
- switch (patch.op) {
303
- case "add":
304
- case "replace": {
305
- yjs.set(key, convertJsonToYjsData(patch.value));
306
- break;
307
- }
308
- case "remove": {
309
- yjs.delete(key);
310
- break;
311
- }
312
- default: {
313
- throw failure(`invalid patch operation for map`);
314
- }
395
+ }
396
+ setYjsContainerSnapshot(dest, source);
397
+ };
398
+ function convertValue(v) {
399
+ if (v === null || v === void 0 || typeof v !== "object") {
400
+ return v;
401
+ }
402
+ if (Array.isArray(v) && v.length === 0) {
403
+ return new Y.Array();
404
+ }
405
+ return convertJsonToYjsData(v);
406
+ }
407
+ function applyMobxChangeToYjsObject(change, yjsObject) {
408
+ if (isYjsValueDeleted(yjsObject)) {
409
+ throw failure("cannot apply patch to deleted Yjs value");
410
+ }
411
+ const yjsContainer = resolveYjsPath(yjsObject, change.path);
412
+ if (!yjsContainer) {
413
+ return;
414
+ }
415
+ if (yjsContainer instanceof Y.Array) {
416
+ if (change.type === DeepChangeType.ArraySplice) {
417
+ yjsContainer.delete(change.index, change.removedValues.length);
418
+ if (change.addedValues.length > 0) {
419
+ const valuesToInsert = change.addedValues.map(convertValue);
420
+ yjsContainer.insert(change.index, valuesToInsert);
315
421
  }
316
- } else if (yjs instanceof Y.Array) {
317
- const key = patch.path[0];
318
- switch (patch.op) {
319
- case "replace": {
320
- if (key === "length") {
321
- const newLength = patch.value;
322
- if (yjs.length > newLength) {
323
- const toDelete = yjs.length - newLength;
324
- yjs.delete(newLength, toDelete);
325
- } else if (yjs.length < patch.value) {
326
- const toInsert = patch.value - yjs.length;
327
- yjs.insert(yjs.length, Array.from({ length: toInsert }).fill(void 0));
328
- }
329
- } else {
330
- yjs.delete(Number(key));
331
- yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
332
- }
333
- break;
334
- }
335
- case "add": {
336
- yjs.insert(Number(key), [convertJsonToYjsData(patch.value)]);
337
- break;
338
- }
339
- case "remove": {
340
- yjs.delete(Number(key));
341
- break;
342
- }
343
- default: {
344
- throw failure(`invalid patch operation for array`);
345
- }
422
+ } else if (change.type === DeepChangeType.ArrayUpdate) {
423
+ yjsContainer.delete(change.index, 1);
424
+ yjsContainer.insert(change.index, [convertValue(change.newValue)]);
425
+ } else {
426
+ throw failure(`unsupported array change type: ${change.type}`);
427
+ }
428
+ } else if (yjsContainer instanceof Y.Map) {
429
+ if (change.type === DeepChangeType.ObjectAdd || change.type === DeepChangeType.ObjectUpdate) {
430
+ const key = change.key;
431
+ if (change.newValue === void 0) {
432
+ yjsContainer.delete(key);
433
+ } else {
434
+ yjsContainer.set(key, convertValue(change.newValue));
346
435
  }
347
- } else if (yjs instanceof Y.Text) ;
348
- else {
349
- throw failure(`invalid patch path, the Yjs object is of an unkown type, so key "${String(patch.path[0])}" cannot be found in it`);
436
+ } else if (change.type === DeepChangeType.ObjectRemove) {
437
+ const key = change.key;
438
+ yjsContainer.delete(key);
439
+ } else {
440
+ throw failure(`unsupported object change type: ${change.type}`);
350
441
  }
442
+ } else if (yjsContainer instanceof Y.Text) {
443
+ return;
351
444
  } else {
352
- throw failure(`invalid patch path, it cannot be empty`);
445
+ throw failure(`unsupported Yjs container type: ${yjsContainer}`);
353
446
  }
354
447
  }
355
448
  const convertYjsDataToJson = action((yjsData) => {
@@ -371,89 +464,126 @@ const convertYjsDataToJson = action((yjsData) => {
371
464
  }
372
465
  return yjsData;
373
466
  });
374
- function convertYjsEventToPatches(event) {
375
- const patches = [];
376
- if (event instanceof Y.YMapEvent) {
377
- const source = event.target;
378
- event.changes.keys.forEach((change, key) => {
379
- const path = [...event.path, key];
380
- switch (change.action) {
381
- case "add":
382
- patches.push({
383
- op: "add",
384
- path,
385
- value: toPlainValue(source.get(key))
386
- });
387
- break;
388
- case "update":
389
- patches.push({
390
- op: "replace",
391
- path,
392
- value: toPlainValue(source.get(key))
393
- });
394
- break;
395
- case "delete":
396
- patches.push({
397
- op: "remove",
398
- path
399
- });
400
- break;
401
- default:
402
- throw failure(`unsupported Yjs map event action: ${change.action}`);
403
- }
404
- });
405
- } else if (event instanceof Y.YArrayEvent) {
406
- let retain = 0;
407
- event.changes.delta.forEach((change) => {
408
- if (change.retain) {
409
- retain += change.retain;
467
+ function applyYjsEventToMobx(event, boundObject, reconciliationMap) {
468
+ const path = event.path;
469
+ const { value: target } = resolvePath(boundObject, path);
470
+ if (!target) {
471
+ throw failure(`cannot resolve path ${JSON.stringify(path)}`);
472
+ }
473
+ runUnprotected(() => {
474
+ if (event instanceof Y.YMapEvent) {
475
+ applyYMapEventToMobx(event, target, reconciliationMap);
476
+ } else if (event instanceof Y.YArrayEvent) {
477
+ applyYArrayEventToMobx(event, target, reconciliationMap);
478
+ } else if (event instanceof Y.YTextEvent) {
479
+ applyYTextEventToMobx(event, target);
480
+ }
481
+ });
482
+ }
483
+ function processDeletedValue(val, reconciliationMap) {
484
+ if (val && typeof val === "object" && isModel(val)) {
485
+ const sn = getSnapshot(val);
486
+ const id = getSnapshotModelId(sn);
487
+ if (id) {
488
+ reconciliationMap.set(id, val);
489
+ }
490
+ }
491
+ }
492
+ function reviveValue(jsonValue, reconciliationMap) {
493
+ if (jsonValue === null || typeof jsonValue !== "object") {
494
+ return jsonValue;
495
+ }
496
+ if (isFrozenSnapshot(jsonValue)) {
497
+ return frozen(jsonValue.data);
498
+ }
499
+ if (reconciliationMap && jsonValue && typeof jsonValue === "object") {
500
+ const modelId = getSnapshotModelId(jsonValue);
501
+ if (modelId) {
502
+ const existing = reconciliationMap.get(modelId);
503
+ if (existing) {
504
+ reconciliationMap.delete(modelId);
505
+ return existing;
410
506
  }
411
- if (change.delete) {
412
- const path = [...event.path, retain];
413
- for (let i = 0; i < change.delete; i++) {
414
- patches.push({
415
- op: "remove",
416
- path
417
- });
507
+ }
508
+ }
509
+ return fromSnapshot(jsonValue);
510
+ }
511
+ function applyYMapEventToMobx(event, target, reconciliationMap) {
512
+ const source = event.target;
513
+ event.changes.keys.forEach((change, key) => {
514
+ switch (change.action) {
515
+ case "add":
516
+ case "update": {
517
+ const yjsValue = source.get(key);
518
+ const jsonValue = convertYjsDataToJson(yjsValue);
519
+ if (change.action === "update") {
520
+ processDeletedValue(target[key], reconciliationMap);
418
521
  }
522
+ target[key] = reviveValue(jsonValue, reconciliationMap);
523
+ break;
419
524
  }
420
- if (change.insert) {
421
- const newValues = Array.isArray(change.insert) ? change.insert : [change.insert];
422
- newValues.forEach((v) => {
423
- const path = [...event.path, retain];
424
- patches.push({
425
- op: "add",
426
- path,
427
- value: toPlainValue(v)
428
- });
429
- retain++;
430
- });
525
+ case "delete": {
526
+ processDeletedValue(target[key], reconciliationMap);
527
+ if (isModel(target)) {
528
+ remove(target.$, key);
529
+ } else {
530
+ remove(target, key);
531
+ }
532
+ break;
431
533
  }
432
- });
433
- } else if (event instanceof Y.YTextEvent) {
434
- const path = [
435
- ...event.path,
436
- "deltaList",
437
- -1
438
- /* last item */
439
- ];
440
- patches.push({
441
- op: "add",
442
- path,
443
- value: { $frozen: true, data: event.delta }
444
- });
534
+ default:
535
+ throw failure(`unsupported Yjs map event action: ${change.action}`);
536
+ }
537
+ });
538
+ }
539
+ function applyYArrayEventToMobx(event, target, reconciliationMap) {
540
+ let currentIndex = 0;
541
+ for (const change of event.changes.delta) {
542
+ if (change.retain) {
543
+ currentIndex += change.retain;
544
+ }
545
+ if (change.delete) {
546
+ const deletedItems = target.slice(currentIndex, currentIndex + change.delete);
547
+ deletedItems.forEach((item) => {
548
+ processDeletedValue(item, reconciliationMap);
549
+ });
550
+ target.splice(currentIndex, change.delete);
551
+ }
552
+ if (change.insert) {
553
+ const insertedItems = Array.isArray(change.insert) ? change.insert : [change.insert];
554
+ const values = insertedItems.map((yjsValue) => {
555
+ const jsonValue = convertYjsDataToJson(yjsValue);
556
+ return reviveValue(jsonValue, reconciliationMap);
557
+ });
558
+ target.splice(currentIndex, 0, ...values);
559
+ currentIndex += values.length;
560
+ }
445
561
  }
446
- return patches;
447
562
  }
448
- function toPlainValue(v) {
449
- if (v instanceof Y.Map || v instanceof Y.Array) {
450
- return v.toJSON();
451
- } else {
452
- return v;
563
+ function applyYTextEventToMobx(event, target) {
564
+ if (target == null ? void 0 : target.deltaList) {
565
+ target.deltaList.push(frozen(event.delta));
566
+ }
567
+ }
568
+ function captureChangeSnapshots(change) {
569
+ if (change.type === DeepChangeType.ArraySplice && change.addedValues.length > 0) {
570
+ const snapshots = change.addedValues.map((v) => isTreeNode(v) ? getSnapshot(v) : v);
571
+ return { ...change, addedValues: snapshots };
572
+ } else if (change.type === DeepChangeType.ArrayUpdate) {
573
+ const snapshot = isTreeNode(change.newValue) ? getSnapshot(change.newValue) : change.newValue;
574
+ return { ...change, newValue: snapshot };
575
+ } else if (change.type === DeepChangeType.ObjectAdd || change.type === DeepChangeType.ObjectUpdate) {
576
+ const snapshot = isTreeNode(change.newValue) ? getSnapshot(change.newValue) : change.newValue;
577
+ return { ...change, newValue: snapshot };
453
578
  }
579
+ return change;
454
580
  }
455
- function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
456
- const yjsOrigin = Symbol("bindYjsToMobxKeystoneTransactionOrigin");
581
+ function bindYjsToMobxKeystone({
582
+ yjsDoc,
583
+ yjsObject,
584
+ mobxKeystoneType
585
+ }) {
586
+ const yjsOrigin = /* @__PURE__ */ Symbol("bindYjsToMobxKeystoneTransactionOrigin");
457
587
  let applyingYjsChangesToMobxKeystone = 0;
458
588
  const bindingContext = {
459
589
  yjsDoc,
@@ -466,90 +596,136 @@ function bindYjsToMobxKeystone({ yjsDoc, yjsObject, mobxKeystoneType }) {
466
596
  return applyingYjsChangesToMobxKeystone > 0;
467
597
  }
468
598
  };
599
+ if (isYjsValueDeleted(yjsObject)) {
600
+ throw failure("cannot apply patch to deleted Yjs value");
601
+ }
469
602
  const yjsJson = convertYjsDataToJson(yjsObject);
470
- const initializationGlobalPatches = [];
603
+ let boundObject;
604
+ let hasInitChanges = false;
471
605
  const createBoundObject = () => {
472
- const disposeOnGlobalPatches = onGlobalPatches((target, patches) => {
473
- initializationGlobalPatches.push({ target, patches });
606
+ const disposeGlobalListener = onGlobalDeepChange((_target, change) => {
607
+ if (change.isInit) {
608
+ hasInitChanges = true;
609
+ }
474
610
  });
475
611
  try {
476
- const boundObject2 = yjsBindingContext.apply(() => fromSnapshot(mobxKeystoneType, yjsJson), bindingContext);
477
- yjsBindingContext.set(boundObject2, { ...bindingContext, boundObject: boundObject2 });
478
- return boundObject2;
612
+ const result = yjsBindingContext.apply(
613
+ () => fromSnapshot(mobxKeystoneType, yjsJson),
614
+ bindingContext
615
+ );
616
+ yjsBindingContext.set(result, { ...bindingContext, boundObject: result });
617
+ return result;
479
618
  } finally {
480
- disposeOnGlobalPatches();
619
+ disposeGlobalListener();
481
620
  }
482
621
  };
483
- const boundObject = createBoundObject();
622
+ boundObject = createBoundObject();
484
623
  const observeDeepCb = action((events) => {
485
- const patches = [];
624
+ const eventsToApply = [];
486
625
  events.forEach((event) => {
487
626
  var _a;
488
627
  if (event.transaction.origin !== yjsOrigin) {
489
- patches.push(...convertYjsEventToPatches(event));
628
+ eventsToApply.push(event);
490
629
  }
491
630
  if (event.target instanceof Y.Map || event.target instanceof Y.Array) {
492
631
  (_a = getYjsCollectionAtom(event.target)) == null ? void 0 : _a.reportChanged();
493
632
  }
494
633
  });
495
- if (patches.length > 0) {
634
+ if (eventsToApply.length > 0) {
496
635
  applyingYjsChangesToMobxKeystone++;
497
636
  try {
498
- applyPatches(boundObject, patches);
637
+ const reconciliationMap = /* @__PURE__ */ new Map();
638
+ const initChanges = [];
639
+ const disposeGlobalListener = onGlobalDeepChange((target, change) => {
640
+ if (change.isInit) {
641
+ initChanges.push({ target, change: captureChangeSnapshots(change) });
642
+ }
643
+ });
644
+ try {
645
+ eventsToApply.forEach((event) => {
646
+ applyYjsEventToMobx(event, boundObject, reconciliationMap);
647
+ });
648
+ } finally {
649
+ disposeGlobalListener();
650
+ }
651
+ if (initChanges.length > 0 && !isYjsValueDeleted(yjsObject)) {
652
+ yjsDoc.transact(() => {
653
+ for (const { target, change } of initChanges) {
654
+ const pathToTarget = getParentToChildPath(boundObject, target);
655
+ if (pathToTarget !== void 0) {
656
+ const changeWithCorrectPath = {
657
+ ...change,
658
+ path: [...pathToTarget, ...change.path]
659
+ };
660
+ applyMobxChangeToYjsObject(changeWithCorrectPath, yjsObject);
661
+ }
662
+ }
663
+ }, yjsOrigin);
664
+ }
665
+ if (yjsObject instanceof Y.Map || yjsObject instanceof Y.Array) {
666
+ setYjsContainerSnapshot(yjsObject, getSnapshot(boundObject));
667
+ }
499
668
  } finally {
500
669
  applyingYjsChangesToMobxKeystone--;
501
670
  }
502
671
  }
503
672
  });
504
673
  yjsObject.observeDeep(observeDeepCb);
505
- let pendingArrayOfArrayOfPatches = [];
506
- const disposeOnPatches = onPatches(boundObject, (patches) => {
674
+ let pendingChanges = [];
675
+ const disposeOnDeepChange = onDeepChange(boundObject, (change) => {
507
676
  if (applyingYjsChangesToMobxKeystone > 0) {
508
677
  return;
509
678
  }
510
- pendingArrayOfArrayOfPatches.push(patches);
679
+ if (change.isInit) {
680
+ return;
681
+ }
682
+ pendingChanges.push(captureChangeSnapshots(change));
511
683
  });
512
- const disposeOnSnapshot = onSnapshot(boundObject, () => {
513
- if (pendingArrayOfArrayOfPatches.length === 0) {
684
+ const disposeOnSnapshot = onSnapshot(boundObject, (boundObjectSnapshot) => {
685
+ if (pendingChanges.length === 0) {
686
+ return;
687
+ }
688
+ const changesToApply = pendingChanges;
689
+ pendingChanges = [];
690
+ if (isYjsValueDeleted(yjsObject)) {
514
691
  return;
515
692
  }
516
- const arrayOfArrayOfPatches = pendingArrayOfArrayOfPatches;
517
- pendingArrayOfArrayOfPatches = [];
518
693
  yjsDoc.transact(() => {
519
- arrayOfArrayOfPatches.forEach((arrayOfPatches) => {
520
- arrayOfPatches.forEach((patch) => {
521
- applyMobxKeystonePatchToYjsObject(patch, yjsObject);
522
- });
694
+ changesToApply.forEach((change) => {
695
+ applyMobxChangeToYjsObject(change, yjsObject);
523
696
  });
524
697
  }, yjsOrigin);
698
+ if (yjsObject instanceof Y.Map || yjsObject instanceof Y.Array) {
699
+ setYjsContainerSnapshot(yjsObject, boundObjectSnapshot);
700
+ }
525
701
  });
526
- yjsDoc.transact(() => {
527
- let boundObjectFound = false;
528
- initializationGlobalPatches.forEach(({ target, patches }) => {
529
- if (!boundObjectFound) {
530
- if (target !== boundObject) {
531
- return;
532
- }
533
- boundObjectFound = true;
534
- }
535
- const parentToChildPath = getParentToChildPath(boundObject, target);
536
- if (parentToChildPath !== void 0) {
537
- patches.forEach((patch) => {
538
- applyMobxKeystonePatchToYjsObject({
539
- ...patch,
540
- path: [...parentToChildPath, ...patch.path]
541
- }, yjsObject);
702
+ const finalSnapshot = getSnapshot(boundObject);
703
+ if (hasInitChanges) {
704
+ yjsDoc.transact(() => {
705
+ if (yjsObject instanceof Y.Map) {
706
+ applyJsonObjectToYMap(yjsObject, finalSnapshot, {
707
+ mode: "merge"
708
+ });
709
+ } else if (yjsObject instanceof Y.Array) {
710
+ applyJsonArrayToYArray(yjsObject, finalSnapshot, {
711
+ mode: "merge"
542
712
  });
543
713
  }
544
- });
545
- }, yjsOrigin);
714
+ }, yjsOrigin);
715
+ }
716
+ if (yjsObject instanceof Y.Map || yjsObject instanceof Y.Array) {
717
+ setYjsContainerSnapshot(yjsObject, finalSnapshot);
718
+ }
719
+ const dispose = () => {
720
+ yjsDoc.off("destroy", dispose);
721
+ disposeOnDeepChange();
722
+ disposeOnSnapshot();
723
+ yjsObject.unobserveDeep(observeDeepCb);
724
+ };
725
+ yjsDoc.on("destroy", dispose);
546
726
  return {
547
727
  boundObject,
548
- dispose: () => {
549
- disposeOnPatches();
550
- disposeOnSnapshot();
551
- yjsObject.unobserveDeep(observeDeepCb);
552
- },
728
+ dispose,
553
729
  yjsOrigin
554
730
  };
555
731
  }
@@ -563,4 +739,4 @@ export {
563
739
  yjsBindingContext,
564
740
  yjsTextModelId
565
741
  };
566
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9ieC1rZXlzdG9uZS15anMuZXNtLm1qcyIsInNvdXJjZXMiOlsiLi4vc3JjL3V0aWxzL2Vycm9yLnRzIiwiLi4vc3JjL2JpbmRpbmcveWpzQmluZGluZ0NvbnRleHQudHMiLCIuLi9zcmMvdXRpbHMvZ2V0T3JDcmVhdGVZanNDb2xsZWN0aW9uQXRvbS50cyIsIi4uL3NyYy9iaW5kaW5nL3Jlc29sdmVZanNQYXRoLnRzIiwiLi4vc3JjL2JpbmRpbmcvWWpzVGV4dE1vZGVsLnRzIiwiLi4vc3JjL2JpbmRpbmcvY29udmVydEpzb25Ub1lqc0RhdGEudHMiLCIuLi9zcmMvYmluZGluZy9hcHBseU1vYnhLZXlzdG9uZVBhdGNoVG9ZanNPYmplY3QudHMiLCIuLi9zcmMvYmluZGluZy9jb252ZXJ0WWpzRGF0YVRvSnNvbi50cyIsIi4uL3NyYy9iaW5kaW5nL2NvbnZlcnRZanNFdmVudFRvUGF0Y2hlcy50cyIsIi4uL3NyYy9iaW5kaW5nL2JpbmRZanNUb01vYnhLZXlzdG9uZS50cyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogQSBtb2J4LWtleXN0b25lLXlqcyBlcnJvci5cclxuICovXHJcbmV4cG9ydCBjbGFzcyBNb2J4S2V5c3RvbmVZanNFcnJvciBleHRlbmRzIEVycm9yIHtcclxuICBjb25zdHJ1Y3Rvcihtc2c6IHN0cmluZykge1xyXG4gICAgc3VwZXIobXNnKVxyXG5cclxuICAgIC8vIFNldCB0aGUgcHJvdG90eXBlIGV4cGxpY2l0bHkuXHJcbiAgICBPYmplY3Quc2V0UHJvdG90eXBlT2YodGhpcywgTW9ieEtleXN0b25lWWpzRXJyb3IucHJvdG90eXBlKVxyXG4gIH1cclxufVxyXG5cclxuLyoqXHJcbiAqIEBpbnRlcm5hbFxyXG4gKi9cclxuZXhwb3J0IGZ1bmN0aW9uIGZhaWx1cmUobXNnOiBzdHJpbmcpIHtcclxuICByZXR1cm4gbmV3IE1vYnhLZXlzdG9uZVlqc0Vycm9yKG1zZylcclxufVxyXG4iLCJpbXBvcnQgeyBBbnlUeXBlLCBjcmVhdGVDb250ZXh0IH0gZnJvbSBcIm1vYngta2V5c3RvbmVcIlxuaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcblxuLyoqXG4gKiBDb250ZXh0IHdpdGggaW5mbyBvbiBob3cgYSBtb2J4LWtleXN0b25lIG1vZGVsIGlzIGJvdW5kIHRvIGEgWS5qcyBkYXRhIHN0cnVjdHVyZS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBZanNCaW5kaW5nQ29udGV4dCB7XG4gIC8qKlxuICAgKiBUaGUgWS5qcyBkb2N1bWVudC5cbiAgICovXG4gIHlqc0RvYzogWS5Eb2NcblxuICAvKipcbiAgICogVGhlIGJvdW5kIFkuanMgZGF0YSBzdHJ1Y3R1cmUuXG4gICAqL1xuICB5anNPYmplY3Q6IFkuTWFwPHVua25vd24+IHwgWS5BcnJheTx1bmtub3duPiB8IFkuVGV4dFxuXG4gIC8qKlxuICAgKiBUaGUgbW9ieC1rZXlzdG9uZSBtb2RlbCB0eXBlLlxuICAgKi9cbiAgbW9ieEtleXN0b25lVHlwZTogQW55VHlwZVxuXG4gIC8qKlxuICAgKiBUaGUgb3JpZ2luIHN5bWJvbCB1c2VkIGZvciB0cmFuc2FjdGlvbnMuXG4gICAqL1xuICB5anNPcmlnaW46IHN5bWJvbFxuXG4gIC8qKlxuICAgKiBUaGUgYm91bmQgbW9ieC1rZXlzdG9uZSBpbnN0YW5jZS5cbiAgICovXG4gIGJvdW5kT2JqZWN0OiB1bmtub3duXG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgd2UgYXJlIGN1cnJlbnRseSBhcHBseWluZyBZLmpzIGNoYW5nZXMgdG8gdGhlIG1vYngta2V5c3RvbmUgbW9kZWwuXG4gICAqL1xuICBpc0FwcGx5aW5nWWpzQ2hhbmdlc1RvTW9ieEtleXN0b25lOiBib29sZWFuXG59XG5cbi8qKlxuICogQ29udGV4dCB3aXRoIGluZm8gb24gaG93IGEgbW9ieC1rZXlzdG9uZSBtb2RlbCBpcyBib3VuZCB0byBhIFkuanMgZGF0YSBzdHJ1Y3R1cmUuXG4gKi9cbmV4cG9ydCBjb25zdCB5anNCaW5kaW5nQ29udGV4dCA9IGNyZWF0ZUNvbnRleHQ8WWpzQmluZGluZ0NvbnRleHQgfCB1bmRlZmluZWQ+KHVuZGVmaW5lZClcbiIsImltcG9ydCB7IElBdG9tLCBjcmVhdGVBdG9tIH0gZnJvbSBcIm1vYnhcIlxuaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcblxuY29uc3QgeWpzQ29sbGVjdGlvbkF0b21zID0gbmV3IFdlYWtNYXA8WS5NYXA8dW5rbm93bj4gfCBZLkFycmF5PHVua25vd24+LCBJQXRvbT4oKVxuXG4vKipcbiAqIEBpbnRlcm5hbFxuICovXG5leHBvcnQgY29uc3QgZ2V0WWpzQ29sbGVjdGlvbkF0b20gPSAoXG4gIHlqc0NvbGxlY3Rpb246IFkuTWFwPHVua25vd24+IHwgWS5BcnJheTx1bmtub3duPlxuKTogSUF0b20gfCB1bmRlZmluZWQgPT4ge1xuICByZXR1cm4geWpzQ29sbGVjdGlvbkF0b21zLmdldCh5anNDb2xsZWN0aW9uKVxufVxuXG4vKipcbiAqIEBpbnRlcm5hbFxuICovXG5leHBvcnQgY29uc3QgZ2V0T3JDcmVhdGVZanNDb2xsZWN0aW9uQXRvbSA9IChcbiAgeWpzQ29sbGVjdGlvbjogWS5NYXA8dW5rbm93bj4gfCBZLkFycmF5PHVua25vd24+XG4pOiBJQXRvbSA9PiB7XG4gIGxldCBhdG9tID0geWpzQ29sbGVjdGlvbkF0b21zLmdldCh5anNDb2xsZWN0aW9uKVxuICBpZiAoIWF0b20pIHtcbiAgICBhdG9tID0gY3JlYXRlQXRvbShgeWpzQ29sbGVjdGlvbkF0b21gKVxuICAgIHlqc0NvbGxlY3Rpb25BdG9tcy5zZXQoeWpzQ29sbGVjdGlvbiwgYXRvbSlcbiAgfVxuICByZXR1cm4gYXRvbVxufVxuIiwiaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcbmltcG9ydCB7IGZhaWx1cmUgfSBmcm9tIFwiLi4vdXRpbHMvZXJyb3JcIlxuaW1wb3J0IHsgZ2V0T3JDcmVhdGVZanNDb2xsZWN0aW9uQXRvbSB9IGZyb20gXCIuLi91dGlscy9nZXRPckNyZWF0ZVlqc0NvbGxlY3Rpb25BdG9tXCJcblxuZXhwb3J0IGZ1bmN0aW9uIHJlc29sdmVZanNQYXRoKHlqc09iamVjdDogdW5rbm93biwgcGF0aDogcmVhZG9ubHkgKHN0cmluZyB8IG51bWJlcilbXSk6IHVua25vd24ge1xuICBsZXQgY3VycmVudFlqc09iamVjdDogdW5rbm93biA9IHlqc09iamVjdFxuXG4gIHBhdGguZm9yRWFjaCgocGF0aFBhcnQsIGkpID0+IHtcbiAgICBpZiAoY3VycmVudFlqc09iamVjdCBpbnN0YW5jZW9mIFkuTWFwKSB7XG4gICAgICBnZXRPckNyZWF0ZVlqc0NvbGxlY3Rpb25BdG9tKGN1cnJlbnRZanNPYmplY3QpLnJlcG9ydE9ic2VydmVkKClcbiAgICAgIGNvbnN0IGtleSA9IFN0cmluZyhwYXRoUGFydClcbiAgICAgIGN1cnJlbnRZanNPYmplY3QgPSBjdXJyZW50WWpzT2JqZWN0LmdldChrZXkpXG4gICAgfSBlbHNlIGlmIChjdXJyZW50WWpzT2JqZWN0IGluc3RhbmNlb2YgWS5BcnJheSkge1xuICAgICAgZ2V0T3JDcmVhdGVZanNDb2xsZWN0aW9uQXRvbShjdXJyZW50WWpzT2JqZWN0KS5yZXBvcnRPYnNlcnZlZCgpXG4gICAgICBjb25zdCBrZXkgPSBOdW1iZXIocGF0aFBhcnQpXG4gICAgICBjdXJyZW50WWpzT2JqZWN0ID0gY3VycmVudFlqc09iamVjdC5nZXQoa2V5KVxuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBmYWlsdXJlKFxuICAgICAgICBgWS5NYXAgb3IgWS5BcnJheSB3YXMgZXhwZWN0ZWQgYXQgcGF0aCAke0pTT04uc3RyaW5naWZ5KFxuICAgICAgICAgIHBhdGguc2xpY2UoMCwgaSlcbiAgICAgICAgKX0gaW4gb3JkZXIgdG8gcmVzb2x2ZSBwYXRoICR7SlNPTi5zdHJpbmdpZnkocGF0aCl9LCBidXQgZ290ICR7Y3VycmVudFlqc09iamVjdH0gaW5zdGVhZGBcbiAgICAgIClcbiAgICB9XG4gIH0pXG5cbiAgcmV0dXJuIGN1cnJlbnRZanNPYmplY3Rcbn1cbiIsImltcG9ydCB7IElBdG9tLCBjb21wdXRlZCwgY3JlYXRlQXRvbSwgb2JzZXJ2ZSwgcmVhY3Rpb24gfSBmcm9tIFwibW9ieFwiXHJcbmltcG9ydCB7XHJcbiAgRnJvemVuLFxyXG4gIE1vZGVsLFxyXG4gIGZyb3plbixcclxuICBnZXRQYXJlbnRUb0NoaWxkUGF0aCxcclxuICBtb2RlbCxcclxuICBvblNuYXBzaG90LFxyXG4gIHRQcm9wLFxyXG4gIHR5cGVzLFxyXG59IGZyb20gXCJtb2J4LWtleXN0b25lXCJcclxuaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcclxuaW1wb3J0IHsgZmFpbHVyZSB9IGZyb20gXCIuLi91dGlscy9lcnJvclwiXHJcbmltcG9ydCB7IFlqc0JpbmRpbmdDb250ZXh0LCB5anNCaW5kaW5nQ29udGV4dCB9IGZyb20gXCIuL3lqc0JpbmRpbmdDb250ZXh0XCJcclxuaW1wb3J0IHsgcmVzb2x2ZVlqc1BhdGggfSBmcm9tIFwiLi9yZXNvbHZlWWpzUGF0aFwiXHJcblxyXG4vLyBEZWx0YVtdW10sIHNpbmNlIGVhY2ggc2luZ2xlIGNoYW5nZSBpcyBhIERlbHRhW11cclxuLy8gd2UgdXNlIGZyb3plbiBzbyB0aGF0IHdlIGNhbiByZXVzZSBlYWNoIGRlbHRhIGNoYW5nZVxyXG5jb25zdCBkZWx0YUxpc3RUeXBlID0gdHlwZXMuYXJyYXkodHlwZXMuZnJvemVuKHR5cGVzLnVuY2hlY2tlZDx1bmtub3duW10+KCkpKVxyXG5cclxuZXhwb3J0IGNvbnN0IHlqc1RleHRNb2RlbElkID0gXCJtb2J4LWtleXN0b25lLXlqcy9ZanNUZXh0TW9kZWxcIlxyXG5cclxuLyoqXHJcbiAqIEEgbW9ieC1rZXlzdG9uZSBtb2RlbCB0aGF0IHJlcHJlc2VudHMgYSBZanMuVGV4dCBvYmplY3QuXHJcbiAqL1xyXG5AbW9kZWwoeWpzVGV4dE1vZGVsSWQpXHJcbmV4cG9ydCBjbGFzcyBZanNUZXh0TW9kZWwgZXh0ZW5kcyBNb2RlbCh7XHJcbiAgZGVsdGFMaXN0OiB0UHJvcChkZWx0YUxpc3RUeXBlLCAoKSA9PiBbXSksXHJcbn0pIHtcclxuICAvKipcclxuICAgKiBIZWxwZXIgZnVuY3Rpb24gdG8gY3JlYXRlIGEgWWpzVGV4dE1vZGVsIGluc3RhbmNlIHdpdGggYSBzaW1wbGUgdGV4dC5cclxuICAgKi9cclxuICBzdGF0aWMgd2l0aFRleHQodGV4dDogc3RyaW5nKTogWWpzVGV4dE1vZGVsIHtcclxuICAgIHJldHVybiBuZXcgRGVjb3JhdGVkWWpzVGV4dE1vZGVsKHtcclxuICAgICAgZGVsdGFMaXN0OiBbXHJcbiAgICAgICAgZnJvemVuKFtcclxuICAgICAgICAgIHtcclxuICAgICAgICAgICAgaW5zZXJ0OiB0ZXh0LFxyXG4gICAgICAgICAgfSxcclxuICAgICAgICBdKSxcclxuICAgICAgXSxcclxuICAgIH0pXHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBUaGUgWS5qcyBwYXRoIGZyb20gdGhlIGJvdW5kIG9iamVjdCB0byB0aGUgWWpzVGV4dE1vZGVsIGluc3RhbmNlLlxyXG4gICAqL1xyXG4gIEBjb21wdXRlZFxyXG4gIHByaXZhdGUgZ2V0IF95anNPYmplY3RQYXRoKCkge1xyXG4gICAgY29uc3QgY3R4ID0geWpzQmluZGluZ0NvbnRleHQuZ2V0KHRoaXMpXHJcbiAgICBpZiAoY3R4Py5ib3VuZE9iamVjdCA9PSBudWxsKSB7XHJcbiAgICAgIHRocm93IGZhaWx1cmUoXHJcbiAgICAgICAgXCJ0aGUgWWpzVGV4dE1vZGVsIGluc3RhbmNlIG11c3QgYmUgcGFydCBvZiBhIGJvdW5kIG9iamVjdCBiZWZvcmUgaXQgY2FuIGJlIGFjY2Vzc2VkXCJcclxuICAgICAgKVxyXG4gICAgfVxyXG5cclxuICAgIGNvbnN0IHBhdGggPSBnZXRQYXJlbnRUb0NoaWxkUGF0aChjdHguYm91bmRPYmplY3QsIHRoaXMpXHJcbiAgICBpZiAoIXBhdGgpIHtcclxuICAgICAgdGhyb3cgZmFpbHVyZShcImEgcGF0aCBmcm9tIHRoZSBib3VuZCBvYmplY3QgdG8gdGhlIFlqc1RleHRNb2RlbCBpbnN0YW5jZSBpcyBub3QgYXZhaWxhYmxlXCIpXHJcbiAgICB9XHJcblxyXG4gICAgcmV0dXJuIHBhdGhcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIFRoZSBZanMuVGV4dCBvYmplY3QgcHJlc2VudCBhdCB0aGlzIG1vYngta2V5c3RvbmUgbm9kZSdzIHBhdGguXHJcbiAgICovXHJcbiAgQGNvbXB1dGVkXHJcbiAgcHJpdmF0ZSBnZXQgX3lqc09iamVjdEF0UGF0aCgpOiB1bmtub3duIHtcclxuICAgIGNvbnN0IHBhdGggPSB0aGlzLl95anNPYmplY3RQYXRoXHJcblxyXG4gICAgY29uc3QgY3R4ID0geWpzQmluZGluZ0NvbnRleHQuZ2V0KHRoaXMpIVxyXG5cclxuICAgIHJldHVybiByZXNvbHZlWWpzUGF0aChjdHgueWpzT2JqZWN0LCBwYXRoKVxyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogVGhlIFlqcy5UZXh0IG9iamVjdCByZXByZXNlbnRlZCBieSB0aGlzIG1vYngta2V5c3RvbmUgbm9kZS5cclxuICAgKi9cclxuICBAY29tcHV0ZWRcclxuICBnZXQgeWpzVGV4dCgpOiBZLlRleHQge1xyXG4gICAgY29uc3QgeWpzT2JqZWN0ID0gdGhpcy5feWpzT2JqZWN0QXRQYXRoXHJcblxyXG4gICAgaWYgKCEoeWpzT2JqZWN0IGluc3RhbmNlb2YgWS5UZXh0KSkge1xyXG4gICAgICB0aHJvdyBmYWlsdXJlKGBZLlRleHQgd2FzIGV4cGVjdGVkIGF0IHBhdGggJHtKU09OLnN0cmluZ2lmeSh0aGlzLl95anNPYmplY3RQYXRoKX1gKVxyXG4gICAgfVxyXG5cclxuICAgIHJldHVybiB5anNPYmplY3RcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIEF0b20gdGhhdCBnZXRzIGNoYW5nZWQgd2hlbiB0aGUgYXNzb2NpYXRlZCBZLmpzIHRleHQgY2hhbmdlcy5cclxuICAgKi9cclxuICB5anNUZXh0Q2hhbmdlZEF0b20gPSBjcmVhdGVBdG9tKFwieWpzVGV4dENoYW5nZWRBdG9tXCIpXHJcblxyXG4gIC8qKlxyXG4gICAqIFRoZSB0ZXh0IHZhbHVlIG9mIHRoZSBZanMuVGV4dCBvYmplY3QuXHJcbiAgICogU2hvcnRjdXQgZm9yIGB5anNUZXh0LnRvU3RyaW5nKClgLCBidXQgY29tcHV0ZWQuXHJcbiAgICovXHJcbiAgQGNvbXB1dGVkXHJcbiAgZ2V0IHRleHQoKTogc3RyaW5nIHtcclxuICAgIHRoaXMueWpzVGV4dENoYW5nZWRBdG9tLnJlcG9ydE9ic2VydmVkKClcclxuICAgIHJldHVybiB0aGlzLnlqc1RleHQudG9TdHJpbmcoKVxyXG4gIH1cclxuXHJcbiAgcHJvdGVjdGVkIG9uSW5pdCgpIHtcclxuICAgIGNvbnN0IHNob3VsZFJlcGxpY2F0ZVRvWWpzID0gKGN0eDogWWpzQmluZGluZ0NvbnRleHQgfCB1bmRlZmluZWQpOiBjdHggaXMgWWpzQmluZGluZ0NvbnRleHQgPT4ge1xyXG4gICAgICByZXR1cm4gISFjdHggJiYgISFjdHguYm91bmRPYmplY3QgJiYgIWN0eC5pc0FwcGx5aW5nWWpzQ2hhbmdlc1RvTW9ieEtleXN0b25lXHJcbiAgICB9XHJcblxyXG4gICAgbGV0IHJlYXBwbHlEZWx0YXNUb1lqc1RleHQgPSBmYWxzZVxyXG4gICAgY29uc3QgbmV3RGVsdGFzOiBGcm96ZW48dW5rbm93bltdPltdID0gW11cclxuXHJcbiAgICBsZXQgZGlzcG9zZU9ic2VydmVEZWx0YUxpc3Q6ICgoKSA9PiB2b2lkKSB8IHVuZGVmaW5lZFxyXG5cclxuICAgIGNvbnN0IGRpc3Bvc2VSZWFjdGlvblRvRGVsdGFMaXN0UmVmQ2hhbmdlID0gcmVhY3Rpb24oXHJcbiAgICAgICgpID0+IHRoaXMuJC5kZWx0YUxpc3QsXHJcbiAgICAgIChkZWx0YUxpc3QpID0+IHtcclxuICAgICAgICBkaXNwb3NlT2JzZXJ2ZURlbHRhTGlzdD8uKClcclxuICAgICAgICBkaXNwb3NlT2JzZXJ2ZURlbHRhTGlzdCA9IHVuZGVmaW5lZFxyXG5cclxuICAgICAgICBkaXNwb3NlT2JzZXJ2ZURlbHRhTGlzdCA9IG9ic2VydmUoZGVsdGFMaXN0LCAoY2hhbmdlKSA9PiB7XHJcbiAgICAgICAgICBpZiAocmVhcHBseURlbHRhc1RvWWpzVGV4dCkge1xyXG4gICAgICAgICAgICAvLyBhbHJlYWR5IGdvbm5hIHJlcGxhY2UgdGhlbSBhbGxcclxuICAgICAgICAgICAgcmV0dXJuXHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgICBpZiAoIXNob3VsZFJlcGxpY2F0ZVRvWWpzKHlqc0JpbmRpbmdDb250ZXh0LmdldCh0aGlzKSkpIHtcclxuICAgICAgICAgICAgLy8geWpzIHRleHQgaXMgYWxyZWFkeSB1cCB0byBkYXRlIHdpdGggdGhlc2UgY2hhbmdlc1xyXG4gICAgICAgICAgICByZXR1cm5cclxuICAgICAgICAgIH1cclxuXHJcbiAgICAgICAgICBpZiAoXHJcbiAgICAgICAgICAgIGNoYW5nZS50eXBlID09PSBcInNwbGljZVwiICYmXHJcbiAgICAgICAgICAgIGNoYW5nZS5yZW1vdmVkQ291bnQgPT09IDAgJiZcclxuICAgICAgICAgICAgY2hhbmdlLmFkZGVkQ291bnQgPiAwICYmXHJcbiAgICAgICAgICAgIGNoYW5nZS5pbmRleCA9PT0gdGhpcy5kZWx0YUxpc3QubGVuZ3RoXHJcbiAgICAgICAgICApIHtcclxuICAgICAgICAgICAgLy8gb3B0aW1pemF0aW9uLCBqdXN0IGFkZGluZyBuZXcgb25lcyB0byB0aGUgZW5kXHJcbiAgICAgICAgICAgIG5ld0RlbHRhcy5wdXNoKC4uLmNoYW5nZS5hZGRlZClcclxuICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIC8vIGFueSBvdGhlciBjaGFuZ2UsIHdlIG5lZWQgdG8gcmVhcHBseSBhbGwgZGVsdGFzXHJcbiAgICAgICAgICAgIHJlYXBwbHlEZWx0YXNUb1lqc1RleHQgPSB0cnVlXHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgfSlcclxuICAgICAgfSxcclxuICAgICAgeyBmaXJlSW1tZWRpYXRlbHk6IHRydWUgfVxyXG4gICAgKVxyXG5cclxuICAgIGNvbnN0IGRpc3Bvc2VPblNuYXBzaG90ID0gb25TbmFwc2hvdCh0aGlzLCAoKSA9PiB7XHJcbiAgICAgIHRyeSB7XHJcbiAgICAgICAgaWYgKHJlYXBwbHlEZWx0YXNUb1lqc1RleHQpIHtcclxuICAgICAgICAgIGNvbnN0IGN0eCA9IHlqc0JpbmRpbmdDb250ZXh0LmdldCh0aGlzKVxyXG5cclxuICAgICAgICAgIGlmIChzaG91bGRSZXBsaWNhdGVUb1lqcyhjdHgpKSB7XHJcbiAgICAgICAgICAgIGNvbnN0IHsgeWpzVGV4dCB9ID0gdGhpc1xyXG5cclxuICAgICAgICAgICAgY3R4Lnlqc0RvYy50cmFuc2FjdCgoKSA9PiB7XHJcbiAgICAgICAgICAgICAgLy8gZGlkbid0IGZpbmQgYSBiZXR0ZXIgd2F5IHRoYW4gdGhpcyB0byByZWFwcGx5IGFsbCBkZWx0YXNcclxuICAgICAgICAgICAgICAvLyB3aXRob3V0IGhhdmluZyB0byByZS1jcmVhdGUgdGhlIFkuVGV4dCBvYmplY3RcclxuICAgICAgICAgICAgICBpZiAoeWpzVGV4dC5sZW5ndGggPiAwKSB7XHJcbiAgICAgICAgICAgICAgICB5anNUZXh0LmRlbGV0ZSgwLCB5anNUZXh0Lmxlbmd0aClcclxuICAgICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICAgIHRoaXMuZGVsdGFMaXN0LmZvckVhY2goKGZyb3plbkRlbHRhcykgPT4ge1xyXG4gICAgICAgICAgICAgICAgeWpzVGV4dC5hcHBseURlbHRhKGZyb3plbkRlbHRhcy5kYXRhKVxyXG4gICAgICAgICAgICAgIH0pXHJcbiAgICAgICAgICAgIH0sIGN0eC55anNPcmlnaW4pXHJcbiAgICAgICAgICB9XHJcbiAgICAgICAgfSBlbHNlIGlmIChuZXdEZWx0YXMubGVuZ3RoID4gMCkge1xyXG4gICAgICAgICAgY29uc3QgY3R4ID0geWpzQmluZGluZ0NvbnRleHQuZ2V0KHRoaXMpXHJcblxyXG4gICAgICAgICAgaWYgKHNob3VsZFJlcGxpY2F0ZVRvWWpzKGN0eCkpIHtcclxuICAgICAgICAgICAgY29uc3QgeyB5anNUZXh0IH0gPSB0aGlzXHJcblxyXG4gICAgICAgICAgICBjdHgueWpzRG9jLnRyYW5zYWN0KCgpID0+IHtcclxuICAgICAgICAgICAgICBuZXdEZWx0YXMuZm9yRWFjaCgoZnJvemVuRGVsdGFzKSA9PiB7XHJcbiAgICAgICAgICAgICAgICB5anNUZXh0LmFwcGx5RGVsdGEoZnJvemVuRGVsdGFzLmRhdGEpXHJcbiAgICAgICAgICAgICAgfSlcclxuICAgICAgICAgICAgfSwgY3R4Lnlqc09yaWdpbilcclxuICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICAgIH0gZmluYWxseSB7XHJcbiAgICAgICAgcmVhcHBseURlbHRhc1RvWWpzVGV4dCA9IGZhbHNlXHJcbiAgICAgICAgbmV3RGVsdGFzLmxlbmd0aCA9IDBcclxuICAgICAgfVxyXG4gICAgfSlcclxuXHJcbiAgICBjb25zdCBkaXBvc2VZanNUZXh0Q2hhbmdlZEF0b20gPSBob29rWWpzVGV4dENoYW5nZWRBdG9tKFxyXG4gICAgICAoKSA9PiB0aGlzLnlqc1RleHQsXHJcbiAgICAgIHRoaXMueWpzVGV4dENoYW5nZWRBdG9tXHJcbiAgICApXHJcblxyXG4gICAgcmV0dXJuICgpID0+IHtcclxuICAgICAgZGlzcG9zZU9uU25hcHNob3QoKVxyXG4gICAgICBkaXNwb3NlUmVhY3Rpb25Ub0RlbHRhTGlzdFJlZkNoYW5nZSgpXHJcbiAgICAgIGRpc3Bvc2VPYnNlcnZlRGVsdGFMaXN0Py4oKVxyXG4gICAgICBkaXNwb3NlT2JzZXJ2ZURlbHRhTGlzdCA9IHVuZGVmaW5lZFxyXG5cclxuICAgICAgZGlwb3NlWWpzVGV4dENoYW5nZWRBdG9tKClcclxuICAgIH1cclxuICB9XHJcbn1cclxuXHJcbi8vIHdlIHVzZSB0aGlzIHRyaWNrIGp1c3QgdG8gYXZvaWQgYSBiYWJlbCBidWcgdGhhdCBjYXVzZXMgY2xhc3NlcyB1c2VkIGluc2lkZSBjbGFzc2VzIG5vdCB0byBiZSBvdmVycmlkZW5cclxuLy8gYnkgdGhlIGRlY29yYXRvclxyXG5jb25zdCBEZWNvcmF0ZWRZanNUZXh0TW9kZWwgPSBZanNUZXh0TW9kZWxcclxuXHJcbmZ1bmN0aW9uIGhvb2tZanNUZXh0Q2hhbmdlZEF0b20oZ2V0WWpzVGV4dDogKCkgPT4gWS5UZXh0LCB0ZXh0Q2hhbmdlZEF0b206IElBdG9tKSB7XHJcbiAgbGV0IGRpc3Bvc2VPYnNlcnZlWWpzVGV4dDogKCgpID0+IHZvaWQpIHwgdW5kZWZpbmVkXHJcblxyXG4gIGNvbnN0IG9ic2VydmVGbiA9ICgpID0+IHtcclxuICAgIHRleHRDaGFuZ2VkQXRvbS5yZXBvcnRDaGFuZ2VkKClcclxuICB9XHJcblxyXG4gIGNvbnN0IGRpc3Bvc2VSZWFjdGlvblRvWVRleHRDaGFuZ2UgPSByZWFjdGlvbihcclxuICAgICgpID0+IHtcclxuICAgICAgdHJ5IHtcclxuICAgICAgICByZXR1cm4gZ2V0WWpzVGV4dCgpXHJcbiAgICAgIH0gY2F0Y2gge1xyXG4gICAgICAgIHJldHVybiB1bmRlZmluZWRcclxuICAgICAgfVxyXG4gICAgfSxcclxuICAgICh5anNUZXh0KSA9PiB7XHJcbiAgICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dD8uKClcclxuICAgICAgZGlzcG9zZU9ic2VydmVZanNUZXh0ID0gdW5kZWZpbmVkXHJcblxyXG4gICAgICBpZiAoeWpzVGV4dCkge1xyXG4gICAgICAgIHlqc1RleHQub2JzZXJ2ZShvYnNlcnZlRm4pXHJcblxyXG4gICAgICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dCA9ICgpID0+IHtcclxuICAgICAgICAgIHlqc1RleHQudW5vYnNlcnZlKG9ic2VydmVGbilcclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIHRleHRDaGFuZ2VkQXRvbS5yZXBvcnRDaGFuZ2VkKClcclxuICAgIH0sXHJcbiAgICB7XHJcbiAgICAgIGZpcmVJbW1lZGlhdGVseTogdHJ1ZSxcclxuICAgIH1cclxuICApXHJcblxyXG4gIHJldHVybiAoKSA9PiB7XHJcbiAgICBkaXNwb3NlUmVhY3Rpb25Ub1lUZXh0Q2hhbmdlKClcclxuICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dD8uKClcclxuICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dCA9IHVuZGVmaW5lZFxyXG4gIH1cclxufVxyXG4iLCJpbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuaW1wb3J0IHsgWWpzVGV4dE1vZGVsLCB5anNUZXh0TW9kZWxJZCB9IGZyb20gXCIuL1lqc1RleHRNb2RlbFwiXG5pbXBvcnQgeyBTbmFwc2hvdE91dE9mIH0gZnJvbSBcIm1vYngta2V5c3RvbmVcIlxuaW1wb3J0IHsgWWpzRGF0YSB9IGZyb20gXCIuL2NvbnZlcnRZanNEYXRhVG9Kc29uXCJcbmltcG9ydCB7IFBsYWluQXJyYXksIFBsYWluT2JqZWN0LCBQbGFpblByaW1pdGl2ZSwgUGxhaW5WYWx1ZSB9IGZyb20gXCIuLi9wbGFpblR5cGVzXCJcbmltcG9ydCB7IGFjdGlvbiwgcnVuSW5BY3Rpb24gfSBmcm9tIFwibW9ieFwiXG5cbmZ1bmN0aW9uIGlzUGxhaW5QcmltaXRpdmUodjogUGxhaW5WYWx1ZSk6IHYgaXMgUGxhaW5QcmltaXRpdmUge1xuICBjb25zdCB0ID0gdHlwZW9mIHZcbiAgcmV0dXJuIHQgPT09IFwic3RyaW5nXCIgfHwgdCA9PT0gXCJudW1iZXJcIiB8fCB0ID09PSBcImJvb2xlYW5cIiB8fCB2ID09PSBudWxsIHx8IHYgPT09IHVuZGVmaW5lZFxufVxuXG5mdW5jdGlvbiBpc1BsYWluQXJyYXkodjogUGxhaW5WYWx1ZSk6IHYgaXMgUGxhaW5BcnJheSB7XG4gIHJldHVybiBBcnJheS5pc0FycmF5KHYpXG59XG5cbmZ1bmN0aW9uIGlzUGxhaW5PYmplY3QodjogUGxhaW5WYWx1ZSk6IHYgaXMgUGxhaW5PYmplY3Qge1xuICByZXR1cm4gIWlzUGxhaW5BcnJheSh2KSAmJiB0eXBlb2YgdiA9PT0gXCJvYmplY3RcIiAmJiB2ICE9PSBudWxsXG59XG5cbi8qKlxuICogQ29udmVydHMgYSBwbGFpbiB2YWx1ZSB0byBhIFkuanMgZGF0YSBzdHJ1Y3R1cmUuXG4gKiBPYmplY3RzIGFyZSBjb252ZXJ0ZWQgdG8gWS5NYXBzLCBhcnJheXMgdG8gWS5BcnJheXMsIHByaW1pdGl2ZXMgYXJlIHVudG91Y2hlZC5cbiAqIEZyb3plbiB2YWx1ZXMgYXJlIGEgc3BlY2lhbCBjYXNlIGFuZCB0aGV5IGFyZSBrZXB0IGFzIGltbXV0YWJsZSBwbGFpbiB2YWx1ZXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb252ZXJ0SnNvblRvWWpzRGF0YSh2OiBQbGFpblZhbHVlKTogWWpzRGF0YSB7XG4gIHJldHVybiBydW5JbkFjdGlvbigoKSA9PiB7XG4gICAgaWYgKGlzUGxhaW5QcmltaXRpdmUodikpIHtcbiAgICAgIHJldHVybiB2XG4gICAgfVxuXG4gICAgaWYgKGlzUGxhaW5BcnJheSh2KSkge1xuICAgICAgY29uc3QgYXJyID0gbmV3IFkuQXJyYXkoKVxuICAgICAgYXBwbHlKc29uQXJyYXlUb1lBcnJheShhcnIsIHYpXG4gICAgICByZXR1cm4gYXJyXG4gICAgfVxuXG4gICAgaWYgKGlzUGxhaW5PYmplY3QodikpIHtcbiAgICAgIGlmICh2LiRmcm96ZW4gPT09IHRydWUpIHtcbiAgICAgICAgLy8gZnJvemVuIHZhbHVlLCBzYXZlIGFzIGltbXV0YWJsZSBvYmplY3RcbiAgICAgICAgcmV0dXJuIHZcbiAgICAgIH1cblxuICAgICAgaWYgKHYuJG1vZGVsVHlwZSA9PT0geWpzVGV4dE1vZGVsSWQpIHtcbiAgICAgICAgY29uc3QgdGV4dCA9IG5ldyBZLlRleHQoKVxuICAgICAgICBjb25zdCB5anNUZXh0TW9kZWwgPSB2IGFzIHVua25vd24gYXMgU25hcHNob3RPdXRPZjxZanNUZXh0TW9kZWw+XG4gICAgICAgIHlqc1RleHRNb2RlbC5kZWx0YUxpc3QuZm9yRWFjaCgoZnJvemVuRGVsdGFzKSA9PiB7XG4gICAgICAgICAgdGV4dC5hcHBseURlbHRhKGZyb3plbkRlbHRhcy5kYXRhKVxuICAgICAgICB9KVxuICAgICAgICByZXR1cm4gdGV4dFxuICAgICAgfVxuXG4gICAgICBjb25zdCBtYXAgPSBuZXcgWS5NYXAoKVxuICAgICAgYXBwbHlKc29uT2JqZWN0VG9ZTWFwKG1hcCwgdilcbiAgICAgIHJldHVybiBtYXBcbiAgICB9XG5cbiAgICB0aHJvdyBuZXcgRXJyb3IoYHVuc3VwcG9ydGVkIHZhbHVlIHR5cGU6ICR7dn1gKVxuICB9KVxufVxuXG4vKipcbiAqIEFwcGxpZXMgYSBKU09OIGFycmF5IHRvIGEgWS5BcnJheSwgdXNpbmcgdGhlIGNvbnZlcnRKc29uVG9ZanNEYXRhIHRvIGNvbnZlcnQgdGhlIHZhbHVlcy5cbiAqL1xuZXhwb3J0IGNvbnN0IGFwcGx5SnNvbkFycmF5VG9ZQXJyYXkgPSBhY3Rpb24oKGRlc3Q6IFkuQXJyYXk8YW55Piwgc291cmNlOiBQbGFpbkFycmF5KSA9PiB7XG4gIGRlc3QucHVzaChzb3VyY2UubWFwKGNvbnZlcnRKc29uVG9ZanNEYXRhKSlcbn0pXG5cbi8qKlxuICogQXBwbGllcyBhIEpTT04gb2JqZWN0IHRvIGEgWS5NYXAsIHVzaW5nIHRoZSBjb252ZXJ0SnNvblRvWWpzRGF0YSB0byBjb252ZXJ0IHRoZSB2YWx1ZXMuXG4gKi9cbmV4cG9ydCBjb25zdCBhcHBseUpzb25PYmplY3RUb1lNYXAgPSBhY3Rpb24oKGRlc3Q6IFkuTWFwPGFueT4sIHNvdXJjZTogUGxhaW5PYmplY3QpID0+IHtcbiAgT2JqZWN0LmVudHJpZXMoc291cmNlKS5mb3JFYWNoKChbaywgdl0pID0+IHtcbiAgICBkZXN0LnNldChrLCBjb252ZXJ0SnNvblRvWWpzRGF0YSh2KSlcbiAgfSlcbn0pXG4iLCJpbXBvcnQgeyBQYXRjaCB9IGZyb20gXCJtb2J4LWtleXN0b25lXCJcbmltcG9ydCAqIGFzIFkgZnJvbSBcInlqc1wiXG5pbXBvcnQgeyBmYWlsdXJlIH0gZnJvbSBcIi4uL3V0aWxzL2Vycm9yXCJcbmltcG9ydCB7IGNvbnZlcnRKc29uVG9ZanNEYXRhIH0gZnJvbSBcIi4vY29udmVydEpzb25Ub1lqc0RhdGFcIlxuaW1wb3J0IHsgUGxhaW5WYWx1ZSB9IGZyb20gXCIuLi9wbGFpblR5cGVzXCJcblxuZXhwb3J0IGZ1bmN0aW9uIGFwcGx5TW9ieEtleXN0b25lUGF0Y2hUb1lqc09iamVjdChwYXRjaDogUGF0Y2gsIHlqczogdW5rbm93bik6IHZvaWQge1xuICBpZiAocGF0Y2gucGF0aC5sZW5ndGggPiAxKSB7XG4gICAgY29uc3QgW2tleSwgLi4ucmVzdF0gPSBwYXRjaC5wYXRoXG5cbiAgICBpZiAoeWpzIGluc3RhbmNlb2YgWS5NYXApIHtcbiAgICAgIGNvbnN0IGNoaWxkID0geWpzLmdldChTdHJpbmcoa2V5KSkgYXMgdW5rbm93blxuICAgICAgaWYgKGNoaWxkID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgdGhyb3cgZmFpbHVyZShcbiAgICAgICAgICBgaW52YWxpZCBwYXRjaCBwYXRoLCBrZXkgXCIke2tleX1cIiBub3QgZm91bmQgaW4gWWpzIG1hcCAtIHBhdGNoOiAke0pTT04uc3RyaW5naWZ5KHBhdGNoKX1gXG4gICAgICAgIClcbiAgICAgIH1cbiAgICAgIGFwcGx5TW9ieEtleXN0b25lUGF0Y2hUb1lqc09iamVjdCh7IC4uLnBhdGNoLCBwYXRoOiByZXN0IH0sIGNoaWxkKVxuICAgIH0gZWxzZSBpZiAoeWpzIGluc3RhbmNlb2YgWS5BcnJheSkge1xuICAgICAgY29uc3QgY2hpbGQgPSB5anMuZ2V0KE51bWJlcihrZXkpKSBhcyB1bmtub3duXG4gICAgICBpZiAoY2hpbGQgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICB0aHJvdyBmYWlsdXJlKFxuICAgICAgICAgIGBpbnZhbGlkIHBhdGNoIHBhdGgsIGtleSBcIiR7a2V5fVwiIG5vdCBmb3VuZCBpbiBZanMgYXJyYXkgLSBwYXRjaDogJHtKU09OLnN0cmluZ2lmeShcbiAgICAgICAgICAgIHBhdGNoXG4gICAgICAgICAgKX1gXG4gICAgICAgIClcbiAgICAgIH1cbiAgICAgIGFwcGx5TW9ieEtleXN0b25lUGF0Y2hUb1lqc09iamVjdCh7IC4uLnBhdGNoLCBwYXRoOiByZXN0IH0sIGNoaWxkKVxuICAgIH0gZWxzZSBpZiAoeWpzIGluc3RhbmNlb2YgWS5UZXh0KSB7XG4gICAgICAvLyBjaGFuZ2VzIHRvIGRlbHRhTGlzdCB3aWxsIGJlIGhhbmRsZWQgYnkgdGhlIGFycmF5IG9ic2VydmUgaW4gdGhlIFlqc1RleHRNb2RlbCBjbGFzc1xuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBmYWlsdXJlKFxuICAgICAgICBgaW52YWxpZCBwYXRjaCBwYXRoLCBrZXkgXCIke2tleX1cIiBub3QgZm91bmQgaW4gdW5rbm93biBZanMgb2JqZWN0IC0gcGF0Y2g6ICR7SlNPTi5zdHJpbmdpZnkoXG4gICAgICAgICAgcGF0Y2hcbiAgICAgICAgKX1gXG4gICAgICApXG4gICAgfVxuICB9IGVsc2UgaWYgKHBhdGNoLnBhdGgubGVuZ3RoID09PSAxKSB7XG4gICAgaWYgKHlqcyBpbnN0YW5jZW9mIFkuTWFwKSB7XG4gICAgICBjb25zdCBrZXkgPSBTdHJpbmcocGF0Y2gucGF0aFswXSlcblxuICAgICAgc3dpdGNoIChwYXRjaC5vcCkge1xuICAgICAgICBjYXNlIFwiYWRkXCI6XG4gICAgICAgIGNhc2UgXCJyZXBsYWNlXCI6IHtcbiAgICAgICAgICB5anMuc2V0KGtleSwgY29udmVydEpzb25Ub1lqc0RhdGEocGF0Y2gudmFsdWUgYXMgUGxhaW5WYWx1ZSkpXG4gICAgICAgICAgYnJlYWtcbiAgICAgICAgfVxuICAgICAgICBjYXNlIFwicmVtb3ZlXCI6IHtcbiAgICAgICAgICB5anMuZGVsZXRlKGtleSlcbiAgICAgICAgICBicmVha1xuICAgICAgICB9XG4gICAgICAgIGRlZmF1bHQ6IHtcbiAgICAgICAgICB0aHJvdyBmYWlsdXJlKGBpbnZhbGlkIHBhdGNoIG9wZXJhdGlvbiBmb3IgbWFwYClcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoeWpzIGluc3RhbmNlb2YgWS5BcnJheSkge1xuICAgICAgY29uc3Qga2V5ID0gcGF0Y2gucGF0aFswXVxuXG4gICAgICBzd2l0Y2ggKHBhdGNoLm9wKSB7XG4gICAgICAgIGNhc2UgXCJyZXBsYWNlXCI6IHtcbiAgICAgICAgICBpZiAoa2V5ID09PSBcImxlbmd0aFwiKSB7XG4gICAgICAgICAgICBjb25zdCBuZXdMZW5ndGggPSBwYXRjaC52YWx1ZSBhcyBudW1iZXJcbiAgICAgICAgICAgIGlmICh5anMubGVuZ3RoID4gbmV3TGVuZ3RoKSB7XG4gICAgICAgICAgICAgIGNvbnN0IHRvRGVsZXRlID0geWpzLmxlbmd0aCAtIG5ld0xlbmd0aFxuICAgICAgICAgICAgICB5anMuZGVsZXRlKG5ld0xlbmd0aCwgdG9EZWxldGUpXG4gICAgICAgICAgICB9IGVsc2UgaWYgKHlqcy5sZW5ndGggPCBwYXRjaC52YWx1ZSkge1xuICAgICAgICAgICAgICBjb25zdCB0b0luc2VydCA9IHBhdGNoLnZhbHVlIC0geWpzLmxlbmd0aFxuICAgICAgICAgICAgICB5anMuaW5zZXJ0KHlqcy5sZW5ndGgsIEFycmF5LmZyb20oeyBsZW5ndGg6IHRvSW5zZXJ0IH0pLmZpbGwodW5kZWZpbmVkKSlcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgeWpzLmRlbGV0ZShOdW1iZXIoa2V5KSlcbiAgICAgICAgICAgIHlqcy5pbnNlcnQoTnVtYmVyKGtleSksIFtjb252ZXJ0SnNvblRvWWpzRGF0YShwYXRjaC52YWx1ZSBhcyBQbGFpblZhbHVlKV0pXG4gICAgICAgICAgfVxuICAgICAgICAgIGJyZWFrXG4gICAgICAgIH1cbiAgICAgICAgY2FzZSBcImFkZFwiOiB7XG4gICAgICAgICAgeWpzLmluc2VydChOdW1iZXIoa2V5KSwgW2NvbnZlcnRKc29uVG9ZanNEYXRhKHBhdGNoLnZhbHVlIGFzIFBsYWluVmFsdWUpXSlcbiAgICAgICAgICBicmVha1xuICAgICAgICB9XG4gICAgICAgIGNhc2UgXCJyZW1vdmVcIjoge1xuICAgICAgICAgIHlqcy5kZWxldGUoTnVtYmVyKGtleSkpXG4gICAgICAgICAgYnJlYWtcbiAgICAgICAgfVxuICAgICAgICBkZWZhdWx0OiB7XG4gICAgICAgICAgdGhyb3cgZmFpbHVyZShgaW52YWxpZCBwYXRjaCBvcGVyYXRpb24gZm9yIGFycmF5YClcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoeWpzIGluc3RhbmNlb2YgWS5UZXh0KSB7XG4gICAgICAvLyBpbml0aWFsaXphdGlvbiBvZiBhIFlqc1RleHRNb2RlbCwgZG8gbm90aGluZ1xuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBmYWlsdXJlKFxuICAgICAgICBgaW52YWxpZCBwYXRjaCBwYXRoLCB0aGUgWWpzIG9iamVjdCBpcyBvZiBhbiB1bmtvd24gdHlwZSwgc28ga2V5IFwiJHtTdHJpbmcocGF0Y2gucGF0aFswXSl9XCIgY2Fubm90IGJlIGZvdW5kIGluIGl0YFxuICAgICAgKVxuICAgIH1cbiAgfSBlbHNlIHtcbiAgICB0aHJvdyBmYWlsdXJlKGBpbnZhbGlkIHBhdGNoIHBhdGgsIGl0IGNhbm5vdCBiZSBlbXB0eWApXG4gIH1cbn1cbiIsImltcG9ydCB7IG1vZGVsU25hcHNob3RPdXRXaXRoTWV0YWRhdGEgfSBmcm9tIFwibW9ieC1rZXlzdG9uZVwiXG5pbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuaW1wb3J0IHsgUGxhaW5PYmplY3QsIFBsYWluVmFsdWUgfSBmcm9tIFwiLi4vcGxhaW5UeXBlc1wiXG5pbXBvcnQgeyBZanNUZXh0TW9kZWwgfSBmcm9tIFwiLi9ZanNUZXh0TW9kZWxcIlxuaW1wb3J0IHsgYWN0aW9uIH0gZnJvbSBcIm1vYnhcIlxuXG5leHBvcnQgdHlwZSBZanNEYXRhID0gWS5BcnJheTxhbnk+IHwgWS5NYXA8YW55PiB8IFkuVGV4dCB8IFBsYWluVmFsdWVcblxuZXhwb3J0IGNvbnN0IGNvbnZlcnRZanNEYXRhVG9Kc29uID0gYWN0aW9uKCh5anNEYXRhOiBZanNEYXRhKTogUGxhaW5WYWx1ZSA9PiB7XG4gIGlmICh5anNEYXRhIGluc3RhbmNlb2YgWS5BcnJheSkge1xuICAgIHJldHVybiB5anNEYXRhLm1hcCgodikgPT4gY29udmVydFlqc0RhdGFUb0pzb24odikpXG4gIH1cblxuICBpZiAoeWpzRGF0YSBpbnN0YW5jZW9mIFkuTWFwKSB7XG4gICAgY29uc3Qgb2JqOiBQbGFpbk9iamVjdCA9IHt9XG4gICAgeWpzRGF0YS5mb3JFYWNoKCh2LCBrKSA9PiB7XG4gICAgICBvYmpba10gPSBjb252ZXJ0WWpzRGF0YVRvSnNvbih2KVxuICAgIH0pXG4gICAgcmV0dXJuIG9ialxuICB9XG5cbiAgaWYgKHlqc0RhdGEgaW5zdGFuY2VvZiBZLlRleHQpIHtcbiAgICBjb25zdCBkZWx0YXMgPSB5anNEYXRhLnRvRGVsdGEoKSBhcyB1bmtub3duW11cblxuICAgIHJldHVybiBtb2RlbFNuYXBzaG90T3V0V2l0aE1ldGFkYXRhKFlqc1RleHRNb2RlbCwge1xuICAgICAgZGVsdGFMaXN0OiBkZWx0YXMubGVuZ3RoID4gMCA/IFt7ICRmcm96ZW46IHRydWUsIGRhdGE6IGRlbHRhcyB9XSA6IFtdLFxuICAgIH0pIGFzIHVua25vd24gYXMgUGxhaW5WYWx1ZVxuICB9XG5cbiAgLy8gYXNzdW1lIGl0J3MgYSBwcmltaXRpdmVcbiAgcmV0dXJuIHlqc0RhdGFcbn0pXG4iLCJpbXBvcnQgeyBQYXRjaCB9IGZyb20gXCJtb2J4LWtleXN0b25lXCJcbmltcG9ydCAqIGFzIFkgZnJvbSBcInlqc1wiXG5pbXBvcnQgeyBQbGFpbkFycmF5LCBQbGFpbk9iamVjdCwgUGxhaW5WYWx1ZSB9IGZyb20gXCIuLi9wbGFpblR5cGVzXCJcbmltcG9ydCB7IGZhaWx1cmUgfSBmcm9tIFwiLi4vdXRpbHMvZXJyb3JcIlxuXG5leHBvcnQgZnVuY3Rpb24gY29udmVydFlqc0V2ZW50VG9QYXRjaGVzKGV2ZW50OiBZLllFdmVudDxhbnk+KTogUGF0Y2hbXSB7XG4gIGNvbnN0IHBhdGNoZXM6IFBhdGNoW10gPSBbXVxuXG4gIGlmIChldmVudCBpbnN0YW5jZW9mIFkuWU1hcEV2ZW50KSB7XG4gICAgY29uc3Qgc291cmNlID0gZXZlbnQudGFyZ2V0XG5cbiAgICBldmVudC5jaGFuZ2VzLmtleXMuZm9yRWFjaCgoY2hhbmdlLCBrZXkpID0+IHtcbiAgICAgIGNvbnN0IHBhdGggPSBbLi4uZXZlbnQucGF0aCwga2V5XVxuXG4gICAgICBzd2l0Y2ggKGNoYW5nZS5hY3Rpb24pIHtcbiAgICAgICAgY2FzZSBcImFkZFwiOlxuICAgICAgICAgIHBhdGNoZXMucHVzaCh7XG4gICAgICAgICAgICBvcDogXCJhZGRcIixcbiAgICAgICAgICAgIHBhdGgsXG4gICAgICAgICAgICB2YWx1ZTogdG9QbGFpblZhbHVlKHNvdXJjZS5nZXQoa2V5KSksXG4gICAgICAgICAgfSlcbiAgICAgICAgICBicmVha1xuXG4gICAgICAgIGNhc2UgXCJ1cGRhdGVcIjpcbiAgICAgICAgICBwYXRjaGVzLnB1c2goe1xuICAgICAgICAgICAgb3A6IFwicmVwbGFjZVwiLFxuICAgICAgICAgICAgcGF0aCxcbiAgICAgICAgICAgIHZhbHVlOiB0b1BsYWluVmFsdWUoc291cmNlLmdldChrZXkpKSxcbiAgICAgICAgICB9KVxuICAgICAgICAgIGJyZWFrXG5cbiAgICAgICAgY2FzZSBcImRlbGV0ZVwiOlxuICAgICAgICAgIHBhdGNoZXMucHVzaCh7XG4gICAgICAgICAgICBvcDogXCJyZW1vdmVcIixcbiAgICAgICAgICAgIHBhdGgsXG4gICAgICAgICAgfSlcbiAgICAgICAgICBicmVha1xuXG4gICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgdGhyb3cgZmFpbHVyZShgdW5zdXBwb3J0ZWQgWWpzIG1hcCBldmVudCBhY3Rpb246ICR7Y2hhbmdlLmFjdGlvbn1gKVxuICAgICAgfVxuICAgIH0pXG4gIH0gZWxzZSBpZiAoZXZlbnQgaW5zdGFuY2VvZiBZLllBcnJheUV2ZW50KSB7XG4gICAgbGV0IHJldGFpbiA9IDBcbiAgICBldmVudC5jaGFuZ2VzLmRlbHRhLmZvckVhY2goKGNoYW5nZSkgPT4ge1xuICAgICAgaWYgKGNoYW5nZS5yZXRhaW4pIHtcbiAgICAgICAgcmV0YWluICs9IGNoYW5nZS5yZXRhaW5cbiAgICAgIH1cblxuICAgICAgaWYgKGNoYW5nZS5kZWxldGUpIHtcbiAgICAgICAgLy8gcmVtb3ZlIFggaXRlbXMgYXQgcmV0YWluIHBvc2l0aW9uXG4gICAgICAgIGNvbnN0IHBhdGggPSBbLi4uZXZlbnQucGF0aCwgcmV0YWluXVxuICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGNoYW5nZS5kZWxldGU7IGkrKykge1xuICAgICAgICAgIHBhdGNoZXMucHVzaCh7XG4gICAgICAgICAgICBvcDogXCJyZW1vdmVcIixcbiAgICAgICAgICAgIHBhdGgsXG4gICAgICAgICAgfSlcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZiAoY2hhbmdlLmluc2VydCkge1xuICAgICAgICBjb25zdCBuZXdWYWx1ZXMgPSBBcnJheS5pc0FycmF5KGNoYW5nZS5pbnNlcnQpID8gY2hhbmdlLmluc2VydCA6IFtjaGFuZ2UuaW5zZXJ0XVxuICAgICAgICBuZXdWYWx1ZXMuZm9yRWFjaCgodikgPT4ge1xuICAgICAgICAgIGNvbnN0IHBhdGggPSBbLi4uZXZlbnQucGF0aCwgcmV0YWluXVxuICAgICAgICAgIHBhdGNoZXMucHVzaCh7XG4gICAgICAgICAgICBvcDogXCJhZGRcIixcbiAgICAgICAgICAgIHBhdGgsXG4gICAgICAgICAgICB2YWx1ZTogdG9QbGFpblZhbHVlKHYpLFxuICAgICAgICAgIH0pXG4gICAgICAgICAgcmV0YWluKytcbiAgICAgICAgfSlcbiAgICAgIH1cbiAgICB9KVxuICB9IGVsc2UgaWYgKGV2ZW50IGluc3RhbmNlb2YgWS5ZVGV4dEV2ZW50KSB7XG4gICAgY29uc3QgcGF0aCA9IFsuLi5ldmVudC5wYXRoLCBcImRlbHRhTGlzdFwiLCAtMSAvKiBsYXN0IGl0ZW0gKi9dXG4gICAgcGF0Y2hlcy5wdXNoKHtcbiAgICAgIG9wOiBcImFkZFwiLFxuICAgICAgcGF0aCxcbiAgICAgIHZhbHVlOiB7ICRmcm96ZW46IHRydWUsIGRhdGE6IGV2ZW50LmRlbHRhIH0sXG4gICAgfSlcbiAgfVxuXG4gIHJldHVybiBwYXRjaGVzXG59XG5cbmZ1bmN0aW9uIHRvUGxhaW5WYWx1ZSh2OiBZLk1hcDxhbnk+IHwgWS5BcnJheTxhbnk+IHwgUGxhaW5WYWx1ZSkge1xuICBpZiAodiBpbnN0YW5jZW9mIFkuTWFwIHx8IHYgaW5zdGFuY2VvZiBZLkFycmF5KSB7XG4gICAgcmV0dXJuIHYudG9KU09OKCkgYXMgUGxhaW5PYmplY3QgfCBQbGFpbkFycmF5XG4gIH0gZWxzZSB7XG4gICAgcmV0dXJuIHZcbiAgfVxufVxuIiwiaW1wb3J0IHsgYWN0aW9uIH0gZnJvbSBcIm1vYnhcIlxuaW1wb3J0IHtcbiAgQW55RGF0YU1vZGVsLFxuICBBbnlNb2RlbCxcbiAgQW55U3RhbmRhcmRUeXBlLFxuICBNb2RlbENsYXNzLFxuICBQYXRjaCxcbiAgU25hcHNob3RJbk9mLFxuICBUeXBlVG9EYXRhLFxuICBhcHBseVBhdGNoZXMsXG4gIGZyb21TbmFwc2hvdCxcbiAgZ2V0UGFyZW50VG9DaGlsZFBhdGgsXG4gIG9uR2xvYmFsUGF0Y2hlcyxcbiAgb25QYXRjaGVzLFxuICBvblNuYXBzaG90LFxufSBmcm9tIFwibW9ieC1rZXlzdG9uZVwiXG5pbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuaW1wb3J0IHsgZ2V0WWpzQ29sbGVjdGlvbkF0b20gfSBmcm9tIFwiLi4vdXRpbHMvZ2V0T3JDcmVhdGVZanNDb2xsZWN0aW9uQXRvbVwiXG5pbXBvcnQgeyBhcHBseU1vYnhLZXlzdG9uZVBhdGNoVG9ZanNPYmplY3QgfSBmcm9tIFwiLi9hcHBseU1vYnhLZXlzdG9uZVBhdGNoVG9ZanNPYmplY3RcIlxuaW1wb3J0IHsgY29udmVydFlqc0RhdGFUb0pzb24gfSBmcm9tIFwiLi9jb252ZXJ0WWpzRGF0YVRvSnNvblwiXG5pbXBvcnQgeyBjb252ZXJ0WWpzRXZlbnRUb1BhdGNoZXMgfSBmcm9tIFwiLi9jb252ZXJ0WWpzRXZlbnRUb1BhdGNoZXNcIlxuaW1wb3J0IHsgWWpzQmluZGluZ0NvbnRleHQsIHlqc0JpbmRpbmdDb250ZXh0IH0gZnJvbSBcIi4veWpzQmluZGluZ0NvbnRleHRcIlxuXG4vKipcbiAqIENyZWF0ZXMgYSBiaWRpcmVjdGlvbmFsIGJpbmRpbmcgYmV0d2VlbiBhIFkuanMgZGF0YSBzdHJ1Y3R1cmUgYW5kIGEgbW9ieC1rZXlzdG9uZSBtb2RlbC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGJpbmRZanNUb01vYnhLZXlzdG9uZTxcbiAgVFR5cGUgZXh0ZW5kcyBBbnlTdGFuZGFyZFR5cGUgfCBNb2RlbENsYXNzPEFueU1vZGVsPiB8IE1vZGVsQ2xhc3M8QW55RGF0YU1vZGVsPixcbj4oe1xuICB5anNEb2MsXG4gIHlqc09iamVjdCxcbiAgbW9ieEtleXN0b25lVHlwZSxcbn06IHtcbiAgLyoqXG4gICAqIFRoZSBZLmpzIGRvY3VtZW50LlxuICAgKi9cbiAgeWpzRG9jOiBZLkRvY1xuICAvKipcbiAgICogVGhlIGJvdW5kIFkuanMgZGF0YSBzdHJ1Y3R1cmUuXG4gICAqL1xuICB5anNPYmplY3Q6IFkuTWFwPGFueT4gfCBZLkFycmF5PGFueT4gfCBZLlRleHRcbiAgLyoqXG4gICAqIFRoZSBtb2J4LWtleXN0b25lIG1vZGVsIHR5cGUuXG4gICAqL1xuICBtb2J4S2V5c3RvbmVUeXBlOiBUVHlwZVxufSk6IHtcbiAgLyoqXG4gICAqIFRoZSBib3VuZCBtb2J4LWtleXN0b25lIGluc3RhbmNlLlxuICAgKi9cbiAgYm91bmRPYmplY3Q6IFR5cGVUb0RhdGE8VFR5cGU+XG4gIC8qKlxuICAgKiBEaXNwb3NlcyB0aGUgYmluZGluZy5cbiAgICovXG4gIGRpc3Bvc2U6ICgpID0+IHZvaWRcbiAgLyoqXG4gICAqIFRoZSBZLmpzIG9yaWdpbiBzeW1ib2wgdXNlZCBmb3IgYmluZGluZyB0cmFuc2FjdGlvbnMuXG4gICAqL1xuICB5anNPcmlnaW46IHN5bWJvbFxufSB7XG4gIGNvbnN0IHlqc09yaWdpbiA9IFN5bWJvbChcImJpbmRZanNUb01vYnhLZXlzdG9uZVRyYW5zYWN0aW9uT3JpZ2luXCIpXG5cbiAgbGV0IGFwcGx5aW5nWWpzQ2hhbmdlc1RvTW9ieEtleXN0b25lID0gMFxuXG4gIGNvbnN0IGJpbmRpbmdDb250ZXh0OiBZanNCaW5kaW5nQ29udGV4dCA9IHtcbiAgICB5anNEb2MsXG4gICAgeWpzT2JqZWN0LFxuICAgIG1vYnhLZXlzdG9uZVR5cGUsXG4gICAgeWpzT3JpZ2luLFxuICAgIGJvdW5kT2JqZWN0OiB1bmRlZmluZWQsIC8vIG5vdCB5ZXQgY3JlYXRlZFxuXG4gICAgZ2V0IGlzQXBwbHlpbmdZanNDaGFuZ2VzVG9Nb2J4S2V5c3RvbmUoKSB7XG4gICAgICByZXR1cm4gYXBwbHlpbmdZanNDaGFuZ2VzVG9Nb2J4S2V5c3RvbmUgPiAwXG4gICAgfSxcbiAgfVxuXG4gIGNvbnN0IHlqc0pzb24gPSBjb252ZXJ0WWpzRGF0YVRvSnNvbih5anNPYmplY3QpXG5cbiAgY29uc3QgaW5pdGlhbGl6YXRpb25HbG9iYWxQYXRjaGVzOiB7IHRhcmdldDogb2JqZWN0OyBwYXRjaGVzOiBQYXRjaFtdIH1bXSA9IFtdXG5cbiAgY29uc3QgY3JlYXRlQm91bmRPYmplY3QgPSAoKSA9PiB7XG4gICAgY29uc3QgZGlzcG9zZU9uR2xvYmFsUGF0Y2hlcyA9IG9uR2xvYmFsUGF0Y2hlcygodGFyZ2V0LCBwYXRjaGVzKSA9PiB7XG4gICAgICBpbml0aWFsaXphdGlvbkdsb2JhbFBhdGNoZXMucHVzaCh7IHRhcmdldCwgcGF0Y2hlcyB9KVxuICAgIH0pXG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgYm91bmRPYmplY3QgPSB5anNCaW5kaW5nQ29udGV4dC5hcHBseShcbiAgICAgICAgKCkgPT4gZnJvbVNuYXBzaG90KG1vYnhLZXlzdG9uZVR5cGUsIHlqc0pzb24gYXMgdW5rbm93biBhcyBTbmFwc2hvdEluT2Y8VHlwZVRvRGF0YTxUVHlwZT4+KSxcbiAgICAgICAgYmluZGluZ0NvbnRleHRcbiAgICAgIClcbiAgICAgIHlqc0JpbmRpbmdDb250ZXh0LnNldChib3VuZE9iamVjdCwgeyAuLi5iaW5kaW5nQ29udGV4dCwgYm91bmRPYmplY3QgfSlcbiAgICAgIHJldHVybiBib3VuZE9iamVjdFxuICAgIH0gZmluYWxseSB7XG4gICAgICBkaXNwb3NlT25HbG9iYWxQYXRjaGVzKClcbiAgICB9XG4gIH1cblxuICBjb25zdCBib3VuZE9iamVjdCA9IGNyZWF0ZUJvdW5kT2JqZWN0KClcblxuICAvLyBiaW5kIGFueSBjaGFuZ2VzIGZyb20geWpzIHRvIG1vYngta2V5c3RvbmVcbiAgY29uc3Qgb2JzZXJ2ZURlZXBDYiA9IGFjdGlvbigoZXZlbnRzOiBZLllFdmVudDxhbnk+W10pID0+IHtcbiAgICBjb25zdCBwYXRjaGVzOiBQYXRjaFtdID0gW11cbiAgICBldmVudHMuZm9yRWFjaCgoZXZlbnQpID0+IHtcbiAgICAgIGlmIChldmVudC50cmFuc2FjdGlvbi5vcmlnaW4gIT09IHlqc09yaWdpbikge1xuICAgICAgICBwYXRjaGVzLnB1c2goLi4uY29udmVydFlqc0V2ZW50VG9QYXRjaGVzKGV2ZW50KSlcbiAgICAgIH1cblxuICAgICAgaWYgKGV2ZW50LnRhcmdldCBpbnN0YW5jZW9mIFkuTWFwIHx8IGV2ZW50LnRhcmdldCBpbnN0YW5jZW9mIFkuQXJyYXkpIHtcbiAgICAgICAgZ2V0WWpzQ29sbGVjdGlvbkF0b20oZXZlbnQudGFyZ2V0KT8ucmVwb3J0Q2hhbmdlZCgpXG4gICAgICB9XG4gICAgfSlcblxuICAgIGlmIChwYXRjaGVzLmxlbmd0aCA+IDApIHtcbiAgICAgIGFwcGx5aW5nWWpzQ2hhbmdlc1RvTW9ieEtleXN0b25lKytcbiAgICAgIHRyeSB7XG4gICAgICAgIGFwcGx5UGF0Y2hlcyhib3VuZE9iamVjdCwgcGF0Y2hlcylcbiAgICAgIH0gZmluYWxseSB7XG4gICAgICAgIGFwcGx5aW5nWWpzQ2hhbmdlc1RvTW9ieEtleXN0b25lLS1cbiAgICAgIH1cbiAgICB9XG4gIH0pXG5cbiAgeWpzT2JqZWN0Lm9ic2VydmVEZWVwKG9ic2VydmVEZWVwQ2IpXG5cbiAgLy8gYmluZCBhbnkgY2hhbmdlcyBmcm9tIG1vYngta2V5c3RvbmUgdG8geWpzXG4gIGxldCBwZW5kaW5nQXJyYXlPZkFycmF5T2ZQYXRjaGVzOiBQYXRjaFtdW10gPSBbXVxuICBjb25zdCBkaXNwb3NlT25QYXRjaGVzID0gb25QYXRjaGVzKGJvdW5kT2JqZWN0LCAocGF0Y2hlcykgPT4ge1xuICAgIGlmIChhcHBseWluZ1lqc0NoYW5nZXNUb01vYnhLZXlzdG9uZSA+IDApIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIHBlbmRpbmdBcnJheU9mQXJyYXlPZlBhdGNoZXMucHVzaChwYXRjaGVzKVxuICB9KVxuXG4gIC8vIHRoaXMgaXMgb25seSB1c2VkIHNvIHdlIGNhbiB0cmFuc2FjdCBhbGwgcGF0Y2hlcyB0byB0aGUgc25hcHNob3QgYm91bmRhcnlcbiAgY29uc3QgZGlzcG9zZU9uU25hcHNob3QgPSBvblNuYXBzaG90KGJvdW5kT2JqZWN0LCAoKSA9PiB7XG4gICAgaWYgKHBlbmRpbmdBcnJheU9mQXJyYXlPZlBhdGNoZXMubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICBjb25zdCBhcnJheU9mQXJyYXlPZlBhdGNoZXMgPSBwZW5kaW5nQXJyYXlPZkFycmF5T2ZQYXRjaGVzXG4gICAgcGVuZGluZ0FycmF5T2ZBcnJheU9mUGF0Y2hlcyA9IFtdXG5cbiAgICB5anNEb2MudHJhbnNhY3QoKCkgPT4ge1xuICAgICAgYXJyYXlPZkFycmF5T2ZQYXRjaGVzLmZvckVhY2goKGFycmF5T2ZQYXRjaGVzKSA9PiB7XG4gICAgICAgIGFycmF5T2ZQYXRjaGVzLmZvckVhY2goKHBhdGNoKSA9PiB7XG4gICAgICAgICAgYXBwbHlNb2J4S2V5c3RvbmVQYXRjaFRvWWpzT2JqZWN0KHBhdGNoLCB5anNPYmplY3QpXG4gICAgICAgIH0pXG4gICAgICB9KVxuICAgIH0sIHlqc09yaWdpbilcbiAgfSlcblxuICAvLyBzeW5jIGluaXRpYWwgcGF0Y2hlcywgdGhhdCBtaWdodCBpbmNsdWRlIHNldHRpbmcgZGVmYXVsdHMsIElEcywgZXRjXG4gIHlqc0RvYy50cmFuc2FjdCgoKSA9PiB7XG4gICAgLy8gd2UgbmVlZCB0byBza2lwIGluaXRpYWxpemF0aW9ucyB1bnRpbCB3ZSBoaXQgdGhlIGluaXRpYWxpemF0aW9uIG9mIHRoZSBib3VuZCBvYmplY3RcbiAgICAvLyB0aGlzIGlzIGJlY2F1c2UgZGVmYXVsdCBvYmplY3RzIG1pZ2h0IGJlIGNyZWF0ZWQgYW5kIGluaXRpYWxpemVkIGJlZm9yZSB0aGUgbWFpbiBvYmplY3RcbiAgICAvLyBidXQgd2UganVzdCBuZWVkIHRvIGNhdGNoIHdoZW4gdGhvc2UgYXJlIGFjdHVhbGx5IGFzc2lnbmVkIHRvIHRoZSBib3VuZCBvYmplY3RcbiAgICBsZXQgYm91bmRPYmplY3RGb3VuZCA9IGZhbHNlXG5cbiAgICBpbml0aWFsaXphdGlvbkdsb2JhbFBhdGNoZXMuZm9yRWFjaCgoeyB0YXJnZXQsIHBhdGNoZXMgfSkgPT4ge1xuICAgICAgaWYgKCFib3VuZE9iamVjdEZvdW5kKSB7XG4gICAgICAgIGlmICh0YXJnZXQgIT09IGJvdW5kT2JqZWN0KSB7XG4gICAgICAgICAgcmV0dXJuIC8vIHNraXBcbiAgICAgICAgfVxuICAgICAgICBib3VuZE9iamVjdEZvdW5kID0gdHJ1ZVxuICAgICAgfVxuXG4gICAgICBjb25zdCBwYXJlbnRUb0NoaWxkUGF0aCA9IGdldFBhcmVudFRvQ2hpbGRQYXRoKGJvdW5kT2JqZWN0LCB0YXJnZXQpXG4gICAgICAvLyB0aGlzIGlzIHVuZGVmaW5lZCBvbmx5IGlmIHRhcmdldCBpcyBub3QgYSBjaGlsZCBvZiBib3VuZE1vZGVsXG4gICAgICBpZiAocGFyZW50VG9DaGlsZFBhdGggIT09IHVuZGVmaW5lZCkge1xuICAgICAgICBwYXRjaGVzLmZvckVhY2goKHBhdGNoKSA9PiB7XG4gICAgICAgICAgYXBwbHlNb2J4S2V5c3RvbmVQYXRjaFRvWWpzT2JqZWN0KFxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAuLi5wYXRjaCxcbiAgICAgICAgICAgICAgcGF0aDogWy4uLnBhcmVudFRvQ2hpbGRQYXRoLCAuLi5wYXRjaC5wYXRoXSxcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB5anNPYmplY3RcbiAgICAgICAgICApXG4gICAgICAgIH0pXG4gICAgICB9XG4gICAgfSlcbiAgfSwgeWpzT3JpZ2luKVxuXG4gIHJldHVybiB7XG4gICAgYm91bmRPYmplY3QsXG4gICAgZGlzcG9zZTogKCkgPT4ge1xuICAgICAgZGlzcG9zZU9uUGF0Y2hlcygpXG4gICAgICBkaXNwb3NlT25TbmFwc2hvdCgpXG4gICAgICB5anNPYmplY3QudW5vYnNlcnZlRGVlcChvYnNlcnZlRGVlcENiKVxuICAgIH0sXG4gICAgeWpzT3JpZ2luLFxuICB9XG59XG4iXSwibmFtZXMiOlsiWWpzVGV4dE1vZGVsIiwiYm91bmRPYmplY3QiXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFHTSxNQUFPLDZCQUE2QixNQUFLO0FBQUEsRUFDN0MsWUFBWSxLQUFXO0FBQ3JCLFVBQU0sR0FBRztBQUdGLFdBQUEsZUFBZSxNQUFNLHFCQUFxQixTQUFTO0FBQUEsRUFBQTtBQUU3RDtBQUtLLFNBQVUsUUFBUSxLQUFXO0FBQzFCLFNBQUEsSUFBSSxxQkFBcUIsR0FBRztBQUNyQztBQ3dCYSxNQUFBLG9CQUFvQixjQUE2QyxNQUFTO0FDdEN2RixNQUFNLHlDQUF5QjtBQUtsQixNQUFBLHVCQUF1QixDQUNsQyxrQkFDcUI7QUFDZCxTQUFBLG1CQUFtQixJQUFJLGFBQWE7QUFDN0M7QUFLYSxNQUFBLCtCQUErQixDQUMxQyxrQkFDUztBQUNMLE1BQUEsT0FBTyxtQkFBbUIsSUFBSSxhQUFhO0FBQy9DLE1BQUksQ0FBQyxNQUFNO0FBQ1QsV0FBTyxXQUFXLG1CQUFtQjtBQUNsQix1QkFBQSxJQUFJLGVBQWUsSUFBSTtBQUFBLEVBQUE7QUFFckMsU0FBQTtBQUNUO0FDdEJnQixTQUFBLGVBQWUsV0FBb0IsTUFBa0M7QUFDbkYsTUFBSSxtQkFBNEI7QUFFM0IsT0FBQSxRQUFRLENBQUMsVUFBVSxNQUFLO0FBQ3ZCLFFBQUEs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
742
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9ieC1rZXlzdG9uZS15anMuZXNtLm1qcyIsInNvdXJjZXMiOlsiLi4vc3JjL3V0aWxzL2Vycm9yLnRzIiwiLi4vc3JjL3V0aWxzL2dldE9yQ3JlYXRlWWpzQ29sbGVjdGlvbkF0b20udHMiLCIuLi9zcmMvdXRpbHMvaXNZanNWYWx1ZURlbGV0ZWQudHMiLCIuLi9zcmMvYmluZGluZy9yZXNvbHZlWWpzUGF0aC50cyIsIi4uL3NyYy9iaW5kaW5nL3lqc0JpbmRpbmdDb250ZXh0LnRzIiwiLi4vc3JjL2JpbmRpbmcvWWpzVGV4dE1vZGVsLnRzIiwiLi4vc3JjL2JpbmRpbmcveWpzU25hcHNob3RUcmFja2luZy50cyIsIi4uL3NyYy9iaW5kaW5nL2NvbnZlcnRKc29uVG9ZanNEYXRhLnRzIiwiLi4vc3JjL2JpbmRpbmcvYXBwbHlNb2J4Q2hhbmdlVG9ZanNPYmplY3QudHMiLCIuLi9zcmMvYmluZGluZy9jb252ZXJ0WWpzRGF0YVRvSnNvbi50cyIsIi4uL3NyYy9iaW5kaW5nL2FwcGx5WWpzRXZlbnRUb01vYngudHMiLCIuLi9zcmMvYmluZGluZy9iaW5kWWpzVG9Nb2J4S2V5c3RvbmUudHMiXSwic291cmNlc0NvbnRlbnQiOlsiLyoqXHJcbiAqIEEgbW9ieC1rZXlzdG9uZS15anMgZXJyb3IuXHJcbiAqL1xyXG5leHBvcnQgY2xhc3MgTW9ieEtleXN0b25lWWpzRXJyb3IgZXh0ZW5kcyBFcnJvciB7XHJcbiAgY29uc3RydWN0b3IobXNnOiBzdHJpbmcpIHtcclxuICAgIHN1cGVyKG1zZylcclxuXHJcbiAgICAvLyBTZXQgdGhlIHByb3RvdHlwZSBleHBsaWNpdGx5LlxyXG4gICAgT2JqZWN0LnNldFByb3RvdHlwZU9mKHRoaXMsIE1vYnhLZXlzdG9uZVlqc0Vycm9yLnByb3RvdHlwZSlcclxuICB9XHJcbn1cclxuXHJcbi8qKlxyXG4gKiBAaW50ZXJuYWxcclxuICovXHJcbmV4cG9ydCBmdW5jdGlvbiBmYWlsdXJlKG1zZzogc3RyaW5nKSB7XHJcbiAgcmV0dXJuIG5ldyBNb2J4S2V5c3RvbmVZanNFcnJvcihtc2cpXHJcbn1cclxuIiwiaW1wb3J0IHsgY3JlYXRlQXRvbSwgSUF0b20gfSBmcm9tIFwibW9ieFwiXG5pbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuXG5jb25zdCB5anNDb2xsZWN0aW9uQXRvbXMgPSBuZXcgV2Vha01hcDxZLk1hcDx1bmtub3duPiB8IFkuQXJyYXk8dW5rbm93bj4sIElBdG9tPigpXG5cbi8qKlxuICogQGludGVybmFsXG4gKi9cbmV4cG9ydCBjb25zdCBnZXRZanNDb2xsZWN0aW9uQXRvbSA9IChcbiAgeWpzQ29sbGVjdGlvbjogWS5NYXA8dW5rbm93bj4gfCBZLkFycmF5PHVua25vd24+XG4pOiBJQXRvbSB8IHVuZGVmaW5lZCA9PiB7XG4gIHJldHVybiB5anNDb2xsZWN0aW9uQXRvbXMuZ2V0KHlqc0NvbGxlY3Rpb24pXG59XG5cbi8qKlxuICogQGludGVybmFsXG4gKi9cbmV4cG9ydCBjb25zdCBnZXRPckNyZWF0ZVlqc0NvbGxlY3Rpb25BdG9tID0gKFxuICB5anNDb2xsZWN0aW9uOiBZLk1hcDx1bmtub3duPiB8IFkuQXJyYXk8dW5rbm93bj5cbik6IElBdG9tID0+IHtcbiAgbGV0IGF0b20gPSB5anNDb2xsZWN0aW9uQXRvbXMuZ2V0KHlqc0NvbGxlY3Rpb24pXG4gIGlmICghYXRvbSkge1xuICAgIGF0b20gPSBjcmVhdGVBdG9tKGB5anNDb2xsZWN0aW9uQXRvbWApXG4gICAgeWpzQ29sbGVjdGlvbkF0b21zLnNldCh5anNDb2xsZWN0aW9uLCBhdG9tKVxuICB9XG4gIHJldHVybiBhdG9tXG59XG4iLCJpbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuXG4vKipcbiAqIENoZWNrcyBpZiBhIFkuanMgdmFsdWUgaGFzIGJlZW4gZGVsZXRlZCBvciBpdHMgZG9jdW1lbnQgZGVzdHJveWVkLlxuICpcbiAqIEBwYXJhbSB5anNWYWx1ZSBUaGUgWS5qcyB2YWx1ZSB0byBjaGVjay5cbiAqIEByZXR1cm5zIGB0cnVlYCBpZiB0aGUgdmFsdWUgaXMgZGVsZXRlZCBvciBkZXN0cm95ZWQsIGBmYWxzZWAgb3RoZXJ3aXNlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gaXNZanNWYWx1ZURlbGV0ZWQoeWpzVmFsdWU6IHVua25vd24pOiBib29sZWFuIHtcbiAgaWYgKHlqc1ZhbHVlIGluc3RhbmNlb2YgWS5BYnN0cmFjdFR5cGUpIHtcbiAgICByZXR1cm4gISEoeWpzVmFsdWUgYXMgYW55KS5faXRlbT8uZGVsZXRlZCB8fCAhIXlqc1ZhbHVlLmRvYz8uaXNEZXN0cm95ZWRcbiAgfVxuICByZXR1cm4gZmFsc2Vcbn1cbiIsImltcG9ydCAqIGFzIFkgZnJvbSBcInlqc1wiXG5pbXBvcnQgeyBmYWlsdXJlIH0gZnJvbSBcIi4uL3V0aWxzL2Vycm9yXCJcbmltcG9ydCB7IGdldE9yQ3JlYXRlWWpzQ29sbGVjdGlvbkF0b20gfSBmcm9tIFwiLi4vdXRpbHMvZ2V0T3JDcmVhdGVZanNDb2xsZWN0aW9uQXRvbVwiXG5cbi8qKlxuICogUmVzb2x2ZXMgYSBwYXRoIHdpdGhpbiBhIFlqcyBvYmplY3Qgc3RydWN0dXJlLlxuICogUmV0dXJucyB0aGUgWWpzIGNvbnRhaW5lciBhdCB0aGUgc3BlY2lmaWVkIHBhdGguXG4gKlxuICogV2hlbiBhIFkuVGV4dCBpcyBlbmNvdW50ZXJlZCBkdXJpbmcgcGF0aCByZXNvbHV0aW9uIChlaXRoZXIgYXQgdGhlIHN0YXJ0XG4gKiBvciBtaWQtcGF0aCksIGl0IGlzIHJldHVybmVkIGltbWVkaWF0ZWx5IHNpbmNlIFkuVGV4dCBkb2Vzbid0IHN1cHBvcnRcbiAqIG5lc3RlZCBwYXRoIHRyYXZlcnNhbC5cbiAqXG4gKiBAcGFyYW0geWpzT2JqZWN0IFRoZSByb290IFlqcyBvYmplY3RcbiAqIEBwYXJhbSBwYXRoIEFycmF5IG9mIGtleXMvaW5kaWNlcyB0byB0cmF2ZXJzZVxuICogQHJldHVybnMgVGhlIFlqcyBjb250YWluZXIgYXQgdGhlIHBhdGgsIG9yIFkuVGV4dCBpZiBlbmNvdW50ZXJlZCBkdXJpbmcgdHJhdmVyc2FsXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiByZXNvbHZlWWpzUGF0aChcbiAgeWpzT2JqZWN0OiBZLk1hcDx1bmtub3duPiB8IFkuQXJyYXk8dW5rbm93bj4gfCBZLlRleHQsXG4gIHBhdGg6IHJlYWRvbmx5IChzdHJpbmcgfCBudW1iZXIpW11cbik6IHVua25vd24ge1xuICBsZXQgY3VycmVudFlqc09iamVjdDogdW5rbm93biA9IHlqc09iamVjdFxuXG4gIGxldCBpID0gLTFcbiAgZm9yIChjb25zdCBwYXRoUGFydCBvZiBwYXRoKSB7XG4gICAgaSsrXG4gICAgLy8gSWYgd2UgZW5jb3VudGVyIGEgWS5UZXh0IGR1cmluZyBwYXRoIHJlc29sdXRpb24sIHJldHVybiBpdCBpbW1lZGlhdGVseS5cbiAgICAvLyBZLlRleHQgb2JqZWN0cyBkb24ndCBzdXBwb3J0IG5lc3RlZCBwYXRoIHRyYXZlcnNhbCwgYW5kIHRoZWlyIHVwZGF0ZXNcbiAgICAvLyBhcmUgaGFuZGxlZCBzZXBhcmF0ZWx5IGJ5IFlqc1RleHRNb2RlbCdzIG93biBzeW5jaHJvbml6YXRpb24gbWVjaGFuaXNtLlxuICAgIGlmIChjdXJyZW50WWpzT2JqZWN0IGluc3RhbmNlb2YgWS5UZXh0KSB7XG4gICAgICByZXR1cm4gY3VycmVudFlqc09iamVjdFxuICAgIH1cblxuICAgIGlmIChjdXJyZW50WWpzT2JqZWN0IGluc3RhbmNlb2YgWS5NYXApIHtcbiAgICAgIGdldE9yQ3JlYXRlWWpzQ29sbGVjdGlvbkF0b20oY3VycmVudFlqc09iamVjdCkucmVwb3J0T2JzZXJ2ZWQoKVxuICAgICAgY29uc3Qga2V5ID0gU3RyaW5nKHBhdGhQYXJ0KVxuICAgICAgY3VycmVudFlqc09iamVjdCA9IGN1cnJlbnRZanNPYmplY3QuZ2V0KGtleSlcbiAgICB9IGVsc2UgaWYgKGN1cnJlbnRZanNPYmplY3QgaW5zdGFuY2VvZiBZLkFycmF5KSB7XG4gICAgICBnZXRPckNyZWF0ZVlqc0NvbGxlY3Rpb25BdG9tKGN1cnJlbnRZanNPYmplY3QpLnJlcG9ydE9ic2VydmVkKClcbiAgICAgIGNvbnN0IGtleSA9IE51bWJlcihwYXRoUGFydClcbiAgICAgIGN1cnJlbnRZanNPYmplY3QgPSBjdXJyZW50WWpzT2JqZWN0LmdldChrZXkpXG4gICAgfSBlbHNlIHtcbiAgICAgIHRocm93IGZhaWx1cmUoXG4gICAgICAgIGBZLk1hcCBvciBZLkFycmF5IHdhcyBleHBlY3RlZCBhdCBwYXRoICR7SlNPTi5zdHJpbmdpZnkoXG4gICAgICAgICAgcGF0aC5zbGljZSgwLCBpKVxuICAgICAgICApfSBpbiBvcmRlciB0byByZXNvbHZlIHBhdGggJHtKU09OLnN0cmluZ2lmeShwYXRoKX0sIGJ1dCBnb3QgJHtjdXJyZW50WWpzT2JqZWN0fSBpbnN0ZWFkYFxuICAgICAgKVxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBjdXJyZW50WWpzT2JqZWN0XG59XG4iLCJpbXBvcnQgeyBBbnlUeXBlLCBjcmVhdGVDb250ZXh0IH0gZnJvbSBcIm1vYngta2V5c3RvbmVcIlxuaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcblxuLyoqXG4gKiBDb250ZXh0IHdpdGggaW5mbyBvbiBob3cgYSBtb2J4LWtleXN0b25lIG1vZGVsIGlzIGJvdW5kIHRvIGEgWS5qcyBkYXRhIHN0cnVjdHVyZS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBZanNCaW5kaW5nQ29udGV4dCB7XG4gIC8qKlxuICAgKiBUaGUgWS5qcyBkb2N1bWVudC5cbiAgICovXG4gIHlqc0RvYzogWS5Eb2NcblxuICAvKipcbiAgICogVGhlIGJvdW5kIFkuanMgZGF0YSBzdHJ1Y3R1cmUuXG4gICAqL1xuICB5anNPYmplY3Q6IFkuTWFwPHVua25vd24+IHwgWS5BcnJheTx1bmtub3duPiB8IFkuVGV4dFxuXG4gIC8qKlxuICAgKiBUaGUgbW9ieC1rZXlzdG9uZSBtb2RlbCB0eXBlLlxuICAgKi9cbiAgbW9ieEtleXN0b25lVHlwZTogQW55VHlwZVxuXG4gIC8qKlxuICAgKiBUaGUgb3JpZ2luIHN5bWJvbCB1c2VkIGZvciB0cmFuc2FjdGlvbnMuXG4gICAqL1xuICB5anNPcmlnaW46IHN5bWJvbFxuXG4gIC8qKlxuICAgKiBUaGUgYm91bmQgbW9ieC1rZXlzdG9uZSBpbnN0YW5jZS5cbiAgICovXG4gIGJvdW5kT2JqZWN0OiB1bmtub3duXG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgd2UgYXJlIGN1cnJlbnRseSBhcHBseWluZyBZLmpzIGNoYW5nZXMgdG8gdGhlIG1vYngta2V5c3RvbmUgbW9kZWwuXG4gICAqL1xuICBpc0FwcGx5aW5nWWpzQ2hhbmdlc1RvTW9ieEtleXN0b25lOiBib29sZWFuXG59XG5cbi8qKlxuICogQ29udGV4dCB3aXRoIGluZm8gb24gaG93IGEgbW9ieC1rZXlzdG9uZSBtb2RlbCBpcyBib3VuZCB0byBhIFkuanMgZGF0YSBzdHJ1Y3R1cmUuXG4gKi9cbmV4cG9ydCBjb25zdCB5anNCaW5kaW5nQ29udGV4dCA9IGNyZWF0ZUNvbnRleHQ8WWpzQmluZGluZ0NvbnRleHQgfCB1bmRlZmluZWQ+KHVuZGVmaW5lZClcbiIsImltcG9ydCB7IGNvbXB1dGVkLCBjcmVhdGVBdG9tLCBJQXRvbSwgb2JzZXJ2ZSwgcmVhY3Rpb24gfSBmcm9tIFwibW9ieFwiXG5pbXBvcnQge1xuICBGcm96ZW4sXG4gIGZyb3plbixcbiAgZ2V0UGFyZW50VG9DaGlsZFBhdGgsXG4gIE1vZGVsLFxuICBtb2RlbCxcbiAgb25TbmFwc2hvdCxcbiAgdFByb3AsXG4gIHR5cGVzLFxufSBmcm9tIFwibW9ieC1rZXlzdG9uZVwiXG5pbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuaW1wb3J0IHsgZmFpbHVyZSB9IGZyb20gXCIuLi91dGlscy9lcnJvclwiXG5pbXBvcnQgeyBpc1lqc1ZhbHVlRGVsZXRlZCB9IGZyb20gXCIuLi91dGlscy9pc1lqc1ZhbHVlRGVsZXRlZFwiXG5pbXBvcnQgeyByZXNvbHZlWWpzUGF0aCB9IGZyb20gXCIuL3Jlc29sdmVZanNQYXRoXCJcbmltcG9ydCB7IFlqc0JpbmRpbmdDb250ZXh0LCB5anNCaW5kaW5nQ29udGV4dCB9IGZyb20gXCIuL3lqc0JpbmRpbmdDb250ZXh0XCJcblxuLy8gRGVsdGFbXVtdLCBzaW5jZSBlYWNoIHNpbmdsZSBjaGFuZ2UgaXMgYSBEZWx0YVtdXG4vLyB3ZSB1c2UgZnJvemVuIHNvIHRoYXQgd2UgY2FuIHJldXNlIGVhY2ggZGVsdGEgY2hhbmdlXG5jb25zdCBkZWx0YUxpc3RUeXBlID0gdHlwZXMuYXJyYXkodHlwZXMuZnJvemVuKHR5cGVzLnVuY2hlY2tlZDx1bmtub3duW10+KCkpKVxuXG5leHBvcnQgY29uc3QgeWpzVGV4dE1vZGVsSWQgPSBcIm1vYngta2V5c3RvbmUteWpzL1lqc1RleHRNb2RlbFwiXG5cbi8qKlxuICogQSBtb2J4LWtleXN0b25lIG1vZGVsIHRoYXQgcmVwcmVzZW50cyBhIFlqcy5UZXh0IG9iamVjdC5cbiAqL1xuQG1vZGVsKHlqc1RleHRNb2RlbElkKVxuZXhwb3J0IGNsYXNzIFlqc1RleHRNb2RlbCBleHRlbmRzIE1vZGVsKHtcbiAgZGVsdGFMaXN0OiB0UHJvcChkZWx0YUxpc3RUeXBlLCAoKSA9PiBbXSksXG59KSB7XG4gIC8qKlxuICAgKiBIZWxwZXIgZnVuY3Rpb24gdG8gY3JlYXRlIGEgWWpzVGV4dE1vZGVsIGluc3RhbmNlIHdpdGggYSBzaW1wbGUgdGV4dC5cbiAgICovXG4gIHN0YXRpYyB3aXRoVGV4dCh0ZXh0OiBzdHJpbmcpOiBZanNUZXh0TW9kZWwge1xuICAgIHJldHVybiBuZXcgRGVjb3JhdGVkWWpzVGV4dE1vZGVsKHtcbiAgICAgIGRlbHRhTGlzdDogW1xuICAgICAgICBmcm96ZW4oW1xuICAgICAgICAgIHtcbiAgICAgICAgICAgIGluc2VydDogdGV4dCxcbiAgICAgICAgICB9LFxuICAgICAgICBdKSxcbiAgICAgIF0sXG4gICAgfSlcbiAgfVxuXG4gIC8qKlxuICAgKiBUaGUgWS5qcyBwYXRoIGZyb20gdGhlIGJvdW5kIG9iamVjdCB0byB0aGUgWWpzVGV4dE1vZGVsIGluc3RhbmNlLlxuICAgKi9cbiAgQGNvbXB1dGVkXG4gIHByaXZhdGUgZ2V0IF95anNPYmplY3RQYXRoKCkge1xuICAgIGNvbnN0IGN0eCA9IHlqc0JpbmRpbmdDb250ZXh0LmdldCh0aGlzKVxuICAgIGlmIChjdHg/LmJvdW5kT2JqZWN0ID09IG51bGwpIHtcbiAgICAgIHRocm93IGZhaWx1cmUoXG4gICAgICAgIFwidGhlIFlqc1RleHRNb2RlbCBpbnN0YW5jZSBtdXN0IGJlIHBhcnQgb2YgYSBib3VuZCBvYmplY3QgYmVmb3JlIGl0IGNhbiBiZSBhY2Nlc3NlZFwiXG4gICAgICApXG4gICAgfVxuXG4gICAgY29uc3QgcGF0aCA9IGdldFBhcmVudFRvQ2hpbGRQYXRoKGN0eC5ib3VuZE9iamVjdCwgdGhpcylcbiAgICBpZiAoIXBhdGgpIHtcbiAgICAgIHRocm93IGZhaWx1cmUoXCJhIHBhdGggZnJvbSB0aGUgYm91bmQgb2JqZWN0IHRvIHRoZSBZanNUZXh0TW9kZWwgaW5zdGFuY2UgaXMgbm90IGF2YWlsYWJsZVwiKVxuICAgIH1cblxuICAgIHJldHVybiBwYXRoXG4gIH1cblxuICAvKipcbiAgICogVGhlIFlqcy5UZXh0IG9iamVjdCBwcmVzZW50IGF0IHRoaXMgbW9ieC1rZXlzdG9uZSBub2RlJ3MgcGF0aC5cbiAgICovXG4gIEBjb21wdXRlZFxuICBwcml2YXRlIGdldCBfeWpzT2JqZWN0QXRQYXRoKCk6IHVua25vd24ge1xuICAgIGNvbnN0IHBhdGggPSB0aGlzLl95anNPYmplY3RQYXRoXG5cbiAgICBjb25zdCBjdHggPSB5anNCaW5kaW5nQ29udGV4dC5nZXQodGhpcykhXG5cbiAgICByZXR1cm4gcmVzb2x2ZVlqc1BhdGgoY3R4Lnlqc09iamVjdCwgcGF0aClcbiAgfVxuXG4gIC8qKlxuICAgKiBUaGUgWWpzLlRleHQgb2JqZWN0IHJlcHJlc2VudGVkIGJ5IHRoaXMgbW9ieC1rZXlzdG9uZSBub2RlLlxuICAgKi9cbiAgQGNvbXB1dGVkXG4gIGdldCB5anNUZXh0KCk6IFkuVGV4dCB7XG4gICAgY29uc3QgeWpzT2JqZWN0ID0gdGhpcy5feWpzT2JqZWN0QXRQYXRoXG5cbiAgICBpZiAoISh5anNPYmplY3QgaW5zdGFuY2VvZiBZLlRleHQpKSB7XG4gICAgICB0aHJvdyBmYWlsdXJlKGBZLlRleHQgd2FzIGV4cGVjdGVkIGF0IHBhdGggJHtKU09OLnN0cmluZ2lmeSh0aGlzLl95anNPYmplY3RQYXRoKX1gKVxuICAgIH1cblxuICAgIHJldHVybiB5anNPYmplY3RcbiAgfVxuXG4gIC8qKlxuICAgKiBBdG9tIHRoYXQgZ2V0cyBjaGFuZ2VkIHdoZW4gdGhlIGFzc29jaWF0ZWQgWS5qcyB0ZXh0IGNoYW5nZXMuXG4gICAqL1xuICB5anNUZXh0Q2hhbmdlZEF0b20gPSBjcmVhdGVBdG9tKFwieWpzVGV4dENoYW5nZWRBdG9tXCIpXG5cbiAgLyoqXG4gICAqIFRoZSB0ZXh0IHZhbHVlIG9mIHRoZSBZanMuVGV4dCBvYmplY3QuXG4gICAqIFNob3J0Y3V0IGZvciBgeWpzVGV4dC50b1N0cmluZygpYCwgYnV0IGNvbXB1dGVkLlxuICAgKi9cbiAgQGNvbXB1dGVkXG4gIGdldCB0ZXh0KCk6IHN0cmluZyB7XG4gICAgdGhpcy55anNUZXh0Q2hhbmdlZEF0b20ucmVwb3J0T2JzZXJ2ZWQoKVxuXG4gICAgY29uc3QgY3R4ID0geWpzQmluZGluZ0NvbnRleHQuZ2V0KHRoaXMpXG4gICAgaWYgKGN0eD8uYm91bmRPYmplY3QgIT0gbnVsbCkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgY29uc3QgeWpzVGV4dFN0cmluZyA9IHRoaXMueWpzVGV4dC50b1N0cmluZygpXG4gICAgICAgIC8vIGlmIHRoZSB5anNUZXh0IGlzIGRldGFjaGVkLCB0b1N0cmluZygpIHJldHVybnMgYW4gZW1wdHkgc3RyaW5nXG4gICAgICAgIC8vIGluIHRoYXQgY2FzZSB3ZSBzaG91bGQgdXNlIHRoZSBkZWx0YUxpc3QgYXMgYSBmYWxsYmFja1xuICAgICAgICBpZiAoeWpzVGV4dFN0cmluZyAhPT0gXCJcIiB8fCB0aGlzLmRlbHRhTGlzdC5sZW5ndGggPT09IDApIHtcbiAgICAgICAgICByZXR1cm4geWpzVGV4dFN0cmluZ1xuICAgICAgICB9XG4gICAgICB9IGNhdGNoIHtcbiAgICAgICAgLy8gZmFsbCBiYWNrXG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gZmFsbCBiYWNrIHRvIGRlbHRhTGlzdFxuICAgIHJldHVybiB0aGlzLmRlbHRhTGlzdFRvVGV4dCgpXG4gIH1cblxuICBwcml2YXRlIGRlbHRhTGlzdFRvVGV4dCgpOiBzdHJpbmcge1xuICAgIGNvbnN0IGRvYyA9IG5ldyBZLkRvYygpXG4gICAgY29uc3QgdGV4dCA9IGRvYy5nZXRUZXh0KClcbiAgICB0aGlzLmRlbHRhTGlzdC5mb3JFYWNoKChkKSA9PiB7XG4gICAgICB0ZXh0LmFwcGx5RGVsdGEoZC5kYXRhKVxuICAgIH0pXG4gICAgcmV0dXJuIHRleHQudG9TdHJpbmcoKVxuICB9XG5cbiAgcHJvdGVjdGVkIG9uSW5pdCgpIHtcbiAgICBjb25zdCBzaG91bGRSZXBsaWNhdGVUb1lqcyA9IChjdHg6IFlqc0JpbmRpbmdDb250ZXh0IHwgdW5kZWZpbmVkKTogY3R4IGlzIFlqc0JpbmRpbmdDb250ZXh0ID0+IHtcbiAgICAgIHJldHVybiAhIWN0eCAmJiAhIWN0eC5ib3VuZE9iamVjdCAmJiAhY3R4LmlzQXBwbHlpbmdZanNDaGFuZ2VzVG9Nb2J4S2V5c3RvbmVcbiAgICB9XG5cbiAgICBsZXQgcmVhcHBseURlbHRhc1RvWWpzVGV4dCA9IGZhbHNlXG4gICAgY29uc3QgbmV3RGVsdGFzOiBGcm96ZW48dW5rbm93bltdPltdID0gW11cblxuICAgIGxldCBkaXNwb3NlT2JzZXJ2ZURlbHRhTGlzdDogKCgpID0+IHZvaWQpIHwgdW5kZWZpbmVkXG5cbiAgICBjb25zdCBkaXNwb3NlUmVhY3Rpb25Ub0RlbHRhTGlzdFJlZkNoYW5nZSA9IHJlYWN0aW9uKFxuICAgICAgKCkgPT4gdGhpcy4kLmRlbHRhTGlzdCxcbiAgICAgIChkZWx0YUxpc3QpID0+IHtcbiAgICAgICAgZGlzcG9zZU9ic2VydmVEZWx0YUxpc3Q/LigpXG4gICAgICAgIGRpc3Bvc2VPYnNlcnZlRGVsdGFMaXN0ID0gdW5kZWZpbmVkXG5cbiAgICAgICAgZGlzcG9zZU9ic2VydmVEZWx0YUxpc3QgPSBvYnNlcnZlKGRlbHRhTGlzdCwgKGNoYW5nZSkgPT4ge1xuICAgICAgICAgIGlmIChyZWFwcGx5RGVsdGFzVG9ZanNUZXh0KSB7XG4gICAgICAgICAgICAvLyBhbHJlYWR5IGdvbm5hIHJlcGxhY2UgdGhlbSBhbGxcbiAgICAgICAgICAgIHJldHVyblxuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoIXNob3VsZFJlcGxpY2F0ZVRvWWpzKHlqc0JpbmRpbmdDb250ZXh0LmdldCh0aGlzKSkpIHtcbiAgICAgICAgICAgIC8vIHlqcyB0ZXh0IGlzIGFscmVhZHkgdXAgdG8gZGF0ZSB3aXRoIHRoZXNlIGNoYW5nZXNcbiAgICAgICAgICAgIHJldHVyblxuICAgICAgICAgIH1cblxuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIGNoYW5nZS50eXBlID09PSBcInNwbGljZVwiICYmXG4gICAgICAgICAgICBjaGFuZ2UucmVtb3ZlZENvdW50ID09PSAwICYmXG4gICAgICAgICAgICBjaGFuZ2UuYWRkZWRDb3VudCA+IDAgJiZcbiAgICAgICAgICAgIGNoYW5nZS5pbmRleCA9PT0gdGhpcy5kZWx0YUxpc3QubGVuZ3RoXG4gICAgICAgICAgKSB7XG4gICAgICAgICAgICAvLyBvcHRpbWl6YXRpb24sIGp1c3QgYWRkaW5nIG5ldyBvbmVzIHRvIHRoZSBlbmRcbiAgICAgICAgICAgIG5ld0RlbHRhcy5wdXNoKC4uLmNoYW5nZS5hZGRlZClcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgLy8gYW55IG90aGVyIGNoYW5nZSwgd2UgbmVlZCB0byByZWFwcGx5IGFsbCBkZWx0YXNcbiAgICAgICAgICAgIHJlYXBwbHlEZWx0YXNUb1lqc1RleHQgPSB0cnVlXG4gICAgICAgICAgfVxuICAgICAgICB9KVxuICAgICAgfSxcbiAgICAgIHsgZmlyZUltbWVkaWF0ZWx5OiB0cnVlIH1cbiAgICApXG5cbiAgICBjb25zdCBkaXNwb3NlT25TbmFwc2hvdCA9IG9uU25hcHNob3QodGhpcywgKCkgPT4ge1xuICAgICAgdHJ5IHtcbiAgICAgICAgaWYgKHJlYXBwbHlEZWx0YXNUb1lqc1RleHQpIHtcbiAgICAgICAgICBjb25zdCBjdHggPSB5anNCaW5kaW5nQ29udGV4dC5nZXQodGhpcylcblxuICAgICAgICAgIGlmIChzaG91bGRSZXBsaWNhdGVUb1lqcyhjdHgpKSB7XG4gICAgICAgICAgICBjb25zdCB7IHlqc1RleHQgfSA9IHRoaXNcbiAgICAgICAgICAgIGlmIChpc1lqc1ZhbHVlRGVsZXRlZCh5anNUZXh0KSkge1xuICAgICAgICAgICAgICB0aHJvdyBmYWlsdXJlKFwiY2Fubm90IHJlYXBwbHkgZGVsdGFzIHRvIGRlbGV0ZWQgWWpzLlRleHRcIilcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgY3R4Lnlqc0RvYy50cmFuc2FjdCgoKSA9PiB7XG4gICAgICAgICAgICAgIC8vIGRpZG4ndCBmaW5kIGEgYmV0dGVyIHdheSB0aGFuIHRoaXMgdG8gcmVhcHBseSBhbGwgZGVsdGFzXG4gICAgICAgICAgICAgIC8vIHdpdGhvdXQgaGF2aW5nIHRvIHJlLWNyZWF0ZSB0aGUgWS5UZXh0IG9iamVjdFxuICAgICAgICAgICAgICBpZiAoeWpzVGV4dC5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICAgICAgeWpzVGV4dC5kZWxldGUoMCwgeWpzVGV4dC5sZW5ndGgpXG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICB0aGlzLmRlbHRhTGlzdC5mb3JFYWNoKChmcm96ZW5EZWx0YXMpID0+IHtcbiAgICAgICAgICAgICAgICB5anNUZXh0LmFwcGx5RGVsdGEoZnJvemVuRGVsdGFzLmRhdGEpXG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICB9LCBjdHgueWpzT3JpZ2luKVxuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIGlmIChuZXdEZWx0YXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgIGNvbnN0IGN0eCA9IHlqc0JpbmRpbmdDb250ZXh0LmdldCh0aGlzKVxuXG4gICAgICAgICAgaWYgKHNob3VsZFJlcGxpY2F0ZVRvWWpzKGN0eCkpIHtcbiAgICAgICAgICAgIGNvbnN0IHsgeWpzVGV4dCB9ID0gdGhpc1xuICAgICAgICAgICAgaWYgKGlzWWpzVmFsdWVEZWxldGVkKHlqc1RleHQpKSB7XG4gICAgICAgICAgICAgIHRocm93IGZhaWx1cmUoXCJjYW5ub3QgcmVhcHBseSBkZWx0YXMgdG8gZGVsZXRlZCBZanMuVGV4dFwiKVxuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBjdHgueWpzRG9jLnRyYW5zYWN0KCgpID0+IHtcbiAgICAgICAgICAgICAgbmV3RGVsdGFzLmZvckVhY2goKGZyb3plbkRlbHRhcykgPT4ge1xuICAgICAgICAgICAgICAgIHlqc1RleHQuYXBwbHlEZWx0YShmcm96ZW5EZWx0YXMuZGF0YSlcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIH0sIGN0eC55anNPcmlnaW4pXG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9IGZpbmFsbHkge1xuICAgICAgICByZWFwcGx5RGVsdGFzVG9ZanNUZXh0ID0gZmFsc2VcbiAgICAgICAgbmV3RGVsdGFzLmxlbmd0aCA9IDBcbiAgICAgIH1cbiAgICB9KVxuXG4gICAgY29uc3QgZGlwb3NlWWpzVGV4dENoYW5nZWRBdG9tID0gaG9va1lqc1RleHRDaGFuZ2VkQXRvbShcbiAgICAgICgpID0+IHRoaXMueWpzVGV4dCxcbiAgICAgIHRoaXMueWpzVGV4dENoYW5nZWRBdG9tXG4gICAgKVxuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIGRpc3Bvc2VPblNuYXBzaG90KClcbiAgICAgIGRpc3Bvc2VSZWFjdGlvblRvRGVsdGFMaXN0UmVmQ2hhbmdlKClcbiAgICAgIGRpc3Bvc2VPYnNlcnZlRGVsdGFMaXN0Py4oKVxuICAgICAgZGlzcG9zZU9ic2VydmVEZWx0YUxpc3QgPSB1bmRlZmluZWRcblxuICAgICAgZGlwb3NlWWpzVGV4dENoYW5nZWRBdG9tKClcbiAgICB9XG4gIH1cbn1cblxuLy8gd2UgdXNlIHRoaXMgdHJpY2sganVzdCB0byBhdm9pZCBhIGJhYmVsIGJ1ZyB0aGF0IGNhdXNlcyBjbGFzc2VzIHVzZWQgaW5zaWRlIGNsYXNzZXMgbm90IHRvIGJlIG92ZXJyaWRlblxuLy8gYnkgdGhlIGRlY29yYXRvclxuY29uc3QgRGVjb3JhdGVkWWpzVGV4dE1vZGVsID0gWWpzVGV4dE1vZGVsXG5cbmZ1bmN0aW9uIGhvb2tZanNUZXh0Q2hhbmdlZEF0b20oZ2V0WWpzVGV4dDogKCkgPT4gWS5UZXh0LCB0ZXh0Q2hhbmdlZEF0b206IElBdG9tKSB7XG4gIGxldCBkaXNwb3NlT2JzZXJ2ZVlqc1RleHQ6ICgoKSA9PiB2b2lkKSB8IHVuZGVmaW5lZFxuXG4gIGNvbnN0IG9ic2VydmVGbiA9ICgpID0+IHtcbiAgICB0ZXh0Q2hhbmdlZEF0b20ucmVwb3J0Q2hhbmdlZCgpXG4gIH1cblxuICBjb25zdCBkaXNwb3NlUmVhY3Rpb25Ub1lUZXh0Q2hhbmdlID0gcmVhY3Rpb24oXG4gICAgKCkgPT4ge1xuICAgICAgdHJ5IHtcbiAgICAgICAgY29uc3QgeWpzVGV4dCA9IGdldFlqc1RleHQoKVxuICAgICAgICByZXR1cm4gaXNZanNWYWx1ZURlbGV0ZWQoeWpzVGV4dCkgPyB1bmRlZmluZWQgOiB5anNUZXh0XG4gICAgICB9IGNhdGNoIHtcbiAgICAgICAgcmV0dXJuIHVuZGVmaW5lZFxuICAgICAgfVxuICAgIH0sXG4gICAgKHlqc1RleHQpID0+IHtcbiAgICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dD8uKClcbiAgICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dCA9IHVuZGVmaW5lZFxuXG4gICAgICBpZiAoeWpzVGV4dCkge1xuICAgICAgICB5anNUZXh0Lm9ic2VydmUob2JzZXJ2ZUZuKVxuXG4gICAgICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dCA9ICgpID0+IHtcbiAgICAgICAgICB5anNUZXh0LnVub2JzZXJ2ZShvYnNlcnZlRm4pXG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgdGV4dENoYW5nZWRBdG9tLnJlcG9ydENoYW5nZWQoKVxuICAgIH0sXG4gICAge1xuICAgICAgZmlyZUltbWVkaWF0ZWx5OiB0cnVlLFxuICAgIH1cbiAgKVxuXG4gIHJldHVybiAoKSA9PiB7XG4gICAgZGlzcG9zZVJlYWN0aW9uVG9ZVGV4dENoYW5nZSgpXG4gICAgZGlzcG9zZU9ic2VydmVZanNUZXh0Py4oKVxuICAgIGRpc3Bvc2VPYnNlcnZlWWpzVGV4dCA9IHVuZGVmaW5lZFxuICB9XG59XG4iLCJpbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuXG4vKipcbiAqIFdlYWtNYXAgdGhhdCB0cmFja3Mgd2hpY2ggc25hcHNob3QgZWFjaCBZLmpzIGNvbnRhaW5lciB3YXMgbGFzdCBzeW5jZWQgZnJvbS5cbiAqIFRoaXMgaXMgdXNlZCBkdXJpbmcgcmVjb25jaWxpYXRpb24gdG8gc2tpcCBjb250YWluZXJzIHRoYXQgYXJlIGFscmVhZHkgdXAtdG8tZGF0ZS5cbiAqXG4gKiBUaGUga2V5IGlzIHRoZSBZLmpzIGNvbnRhaW5lciAoWS5NYXAgb3IgWS5BcnJheSkuXG4gKiBUaGUgdmFsdWUgaXMgdGhlIHNuYXBzaG90IChwbGFpbiBvYmplY3Qgb3IgYXJyYXkpIHRoYXQgd2FzIGxhc3Qgc3luY2VkIHRvIGl0LlxuICovXG5leHBvcnQgY29uc3QgeWpzQ29udGFpbmVyVG9TbmFwc2hvdCA9IG5ldyBXZWFrTWFwPFkuTWFwPGFueT4gfCBZLkFycmF5PGFueT4sIHVua25vd24+KClcblxuLyoqXG4gKiBVcGRhdGVzIHRoZSBzbmFwc2hvdCB0cmFja2luZyBmb3IgYSBZLmpzIGNvbnRhaW5lci5cbiAqIENhbGwgdGhpcyBhZnRlciBzeW5jaW5nIGEgc25hcHNob3QgdG8gYSBZLmpzIGNvbnRhaW5lci5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHNldFlqc0NvbnRhaW5lclNuYXBzaG90KFxuICBjb250YWluZXI6IFkuTWFwPGFueT4gfCBZLkFycmF5PGFueT4sXG4gIHNuYXBzaG90OiB1bmtub3duXG4pOiB2b2lkIHtcbiAgeWpzQ29udGFpbmVyVG9TbmFwc2hvdC5zZXQoY29udGFpbmVyLCBzbmFwc2hvdClcbn1cblxuLyoqXG4gKiBHZXRzIHRoZSBsYXN0IHN5bmNlZCBzbmFwc2hvdCBmb3IgYSBZLmpzIGNvbnRhaW5lci5cbiAqIFJldHVybnMgdW5kZWZpbmVkIGlmIHRoZSBjb250YWluZXIgaGFzIG5ldmVyIGJlZW4gc3luY2VkLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2V0WWpzQ29udGFpbmVyU25hcHNob3QoY29udGFpbmVyOiBZLk1hcDxhbnk+IHwgWS5BcnJheTxhbnk+KTogdW5rbm93biB7XG4gIHJldHVybiB5anNDb250YWluZXJUb1NuYXBzaG90LmdldChjb250YWluZXIpXG59XG5cbi8qKlxuICogQ2hlY2tzIGlmIGEgWS5qcyBjb250YWluZXIgaXMgdXAtdG8tZGF0ZSB3aXRoIHRoZSBnaXZlbiBzbmFwc2hvdC5cbiAqIFVzZXMgcmVmZXJlbmNlIGVxdWFsaXR5IHRvIGNoZWNrIGlmIHRoZSBzbmFwc2hvdCBpcyB0aGUgc2FtZS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGlzWWpzQ29udGFpbmVyVXBUb0RhdGUoXG4gIGNvbnRhaW5lcjogWS5NYXA8YW55PiB8IFkuQXJyYXk8YW55PixcbiAgc25hcHNob3Q6IHVua25vd25cbik6IGJvb2xlYW4ge1xuICByZXR1cm4geWpzQ29udGFpbmVyVG9TbmFwc2hvdC5nZXQoY29udGFpbmVyKSA9PT0gc25hcHNob3Rcbn1cbiIsImltcG9ydCB7IGZyb3plbktleSwgbW9kZWxUeXBlS2V5LCBTbmFwc2hvdE91dE9mIH0gZnJvbSBcIm1vYngta2V5c3RvbmVcIlxuaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcbmltcG9ydCB7IFBsYWluQXJyYXksIFBsYWluT2JqZWN0LCBQbGFpblByaW1pdGl2ZSwgUGxhaW5WYWx1ZSB9IGZyb20gXCIuLi9wbGFpblR5cGVzXCJcbmltcG9ydCB7IFlqc0RhdGEgfSBmcm9tIFwiLi9jb252ZXJ0WWpzRGF0YVRvSnNvblwiXG5pbXBvcnQgeyBZanNUZXh0TW9kZWwsIHlqc1RleHRNb2RlbElkIH0gZnJvbSBcIi4vWWpzVGV4dE1vZGVsXCJcbmltcG9ydCB7IGlzWWpzQ29udGFpbmVyVXBUb0RhdGUsIHNldFlqc0NvbnRhaW5lclNuYXBzaG90IH0gZnJvbSBcIi4veWpzU25hcHNob3RUcmFja2luZ1wiXG5cbi8qKlxuICogT3B0aW9ucyBmb3IgYXBwbHlpbmcgSlNPTiBkYXRhIHRvIFkuanMgZGF0YSBzdHJ1Y3R1cmVzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEFwcGx5SnNvblRvWWpzT3B0aW9ucyB7XG4gIC8qKlxuICAgKiBUaGUgbW9kZSB0byB1c2Ugd2hlbiBhcHBseWluZyBKU09OIGRhdGEgdG8gWS5qcyBkYXRhIHN0cnVjdHVyZXMuXG4gICAqIC0gYGFkZGA6IENyZWF0ZXMgbmV3IFkuanMgY29udGFpbmVycyBmb3Igb2JqZWN0cy9hcnJheXMgKGRlZmF1bHQsIGJhY2t3YXJkcyBjb21wYXRpYmxlKVxuICAgKiAtIGBtZXJnZWA6IFJlY3Vyc2l2ZWx5IG1lcmdlcyB2YWx1ZXMsIHByZXNlcnZpbmcgZXhpc3RpbmcgY29udGFpbmVyIHJlZmVyZW5jZXMgd2hlcmUgcG9zc2libGVcbiAgICovXG4gIG1vZGU/OiBcImFkZFwiIHwgXCJtZXJnZVwiXG59XG5cbmZ1bmN0aW9uIGlzUGxhaW5QcmltaXRpdmUodjogUGxhaW5WYWx1ZSk6IHYgaXMgUGxhaW5QcmltaXRpdmUge1xuICBjb25zdCB0ID0gdHlwZW9mIHZcbiAgcmV0dXJuIHQgPT09IFwic3RyaW5nXCIgfHwgdCA9PT0gXCJudW1iZXJcIiB8fCB0ID09PSBcImJvb2xlYW5cIiB8fCB2ID09PSBudWxsIHx8IHYgPT09IHVuZGVmaW5lZFxufVxuXG5mdW5jdGlvbiBpc1BsYWluQXJyYXkodjogUGxhaW5WYWx1ZSk6IHYgaXMgUGxhaW5BcnJheSB7XG4gIHJldHVybiBBcnJheS5pc0FycmF5KHYpXG59XG5cbmZ1bmN0aW9uIGlzUGxhaW5PYmplY3QodjogUGxhaW5WYWx1ZSk6IHYgaXMgUGxhaW5PYmplY3Qge1xuICByZXR1cm4gdHlwZW9mIHYgPT09IFwib2JqZWN0XCIgJiYgdiAhPT0gbnVsbCAmJiAhQXJyYXkuaXNBcnJheSh2KVxufVxuXG4vKipcbiAqIENvbnZlcnRzIGEgcGxhaW4gdmFsdWUgdG8gYSBZLmpzIGRhdGEgc3RydWN0dXJlLlxuICogT2JqZWN0cyBhcmUgY29udmVydGVkIHRvIFkuTWFwcywgYXJyYXlzIHRvIFkuQXJyYXlzLCBwcmltaXRpdmVzIGFyZSB1bnRvdWNoZWQuXG4gKiBGcm96ZW4gdmFsdWVzIGFyZSBhIHNwZWNpYWwgY2FzZSBhbmQgdGhleSBhcmUga2VwdCBhcyBpbW11dGFibGUgcGxhaW4gdmFsdWVzLlxuICovXG5leHBvcnQgZnVuY3Rpb24gY29udmVydEpzb25Ub1lqc0RhdGEodjogUGxhaW5WYWx1ZSk6IFlqc0RhdGEge1xuICBpZiAoaXNQbGFpblByaW1pdGl2ZSh2KSkge1xuICAgIHJldHVybiB2XG4gIH1cblxuICBpZiAoaXNQbGFpbkFycmF5KHYpKSB7XG4gICAgY29uc3QgYXJyID0gbmV3IFkuQXJyYXkoKVxuICAgIGFwcGx5SnNvbkFycmF5VG9ZQXJyYXkoYXJyLCB2KVxuICAgIHJldHVybiBhcnJcbiAgfVxuXG4gIGlmIChpc1BsYWluT2JqZWN0KHYpKSB7XG4gICAgaWYgKHZbZnJvemVuS2V5XSA9PT0gdHJ1ZSkge1xuICAgICAgLy8gZnJvemVuIHZhbHVlLCBzYXZlIGFzIGltbXV0YWJsZSBvYmplY3RcbiAgICAgIHJldHVybiB2XG4gICAgfVxuXG4gICAgaWYgKHZbbW9kZWxUeXBlS2V5XSA9PT0geWpzVGV4dE1vZGVsSWQpIHtcbiAgICAgIGNvbnN0IHRleHQgPSBuZXcgWS5UZXh0KClcbiAgICAgIGNvbnN0IHlqc1RleHRNb2RlbCA9IHYgYXMgdW5rbm93biBhcyBTbmFwc2hvdE91dE9mPFlqc1RleHRNb2RlbD5cbiAgICAgIHlqc1RleHRNb2RlbC5kZWx0YUxpc3QuZm9yRWFjaCgoZnJvemVuRGVsdGFzKSA9PiB7XG4gICAgICAgIHRleHQuYXBwbHlEZWx0YShmcm96ZW5EZWx0YXMuZGF0YSlcbiAgICAgIH0pXG4gICAgICByZXR1cm4gdGV4dFxuICAgIH1cblxuICAgIGNvbnN0IG1hcCA9IG5ldyBZLk1hcCgpXG4gICAgYXBwbHlKc29uT2JqZWN0VG9ZTWFwKG1hcCwgdilcbiAgICByZXR1cm4gbWFwXG4gIH1cblxuICB0aHJvdyBuZXcgRXJyb3IoYHVuc3VwcG9ydGVkIHZhbHVlIHR5cGU6ICR7dn1gKVxufVxuXG4vKipcbiAqIEFwcGxpZXMgYSBKU09OIGFycmF5IHRvIGEgWS5BcnJheSwgdXNpbmcgdGhlIGNvbnZlcnRKc29uVG9ZanNEYXRhIHRvIGNvbnZlcnQgdGhlIHZhbHVlcy5cbiAqXG4gKiBAcGFyYW0gZGVzdCBUaGUgZGVzdGluYXRpb24gWS5BcnJheS5cbiAqIEBwYXJhbSBzb3VyY2UgVGhlIHNvdXJjZSBKU09OIGFycmF5LlxuICogQHBhcmFtIG9wdGlvbnMgT3B0aW9ucyBmb3IgYXBwbHlpbmcgdGhlIEpTT04gZGF0YS5cbiAqL1xuZXhwb3J0IGNvbnN0IGFwcGx5SnNvbkFycmF5VG9ZQXJyYXkgPSAoXG4gIGRlc3Q6IFkuQXJyYXk8YW55PixcbiAgc291cmNlOiBQbGFpbkFycmF5LFxuICBvcHRpb25zOiBBcHBseUpzb25Ub1lqc09wdGlvbnMgPSB7fVxuKSA9PiB7XG4gIGNvbnN0IHsgbW9kZSA9IFwiYWRkXCIgfSA9IG9wdGlvbnNcblxuICAvLyBJbiBtZXJnZSBtb2RlLCBjaGVjayBpZiB0aGUgY29udGFpbmVyIGlzIGFscmVhZHkgdXAtdG8tZGF0ZSB3aXRoIHRoaXMgc25hcHNob3RcbiAgaWYgKG1vZGUgPT09IFwibWVyZ2VcIiAmJiBpc1lqc0NvbnRhaW5lclVwVG9EYXRlKGRlc3QsIHNvdXJjZSkpIHtcbiAgICByZXR1cm5cbiAgfVxuXG4gIGNvbnN0IHNyY0xlbiA9IHNvdXJjZS5sZW5ndGhcblxuICBpZiAobW9kZSA9PT0gXCJhZGRcIikge1xuICAgIC8vIEFkZCBtb2RlOiBqdXN0IHB1c2ggYWxsIGl0ZW1zIHRvIHRoZSBlbmRcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IHNyY0xlbjsgaSsrKSB7XG4gICAgICBkZXN0LnB1c2goW2NvbnZlcnRKc29uVG9ZanNEYXRhKHNvdXJjZVtpXSldKVxuICAgIH1cbiAgICByZXR1cm5cbiAgfVxuXG4gIC8vIE1lcmdlIG1vZGU6IHJlY3Vyc2l2ZWx5IG1lcmdlIHZhbHVlcywgcHJlc2VydmluZyBleGlzdGluZyBjb250YWluZXIgcmVmZXJlbmNlc1xuICBjb25zdCBkZXN0TGVuID0gZGVzdC5sZW5ndGhcblxuICAvLyBSZW1vdmUgZXh0cmEgaXRlbXMgZnJvbSB0aGUgZW5kXG4gIGlmIChkZXN0TGVuID4gc3JjTGVuKSB7XG4gICAgZGVzdC5kZWxldGUoc3JjTGVuLCBkZXN0TGVuIC0gc3JjTGVuKVxuICB9XG5cbiAgLy8gVXBkYXRlIGV4aXN0aW5nIGl0ZW1zXG4gIGNvbnN0IG1pbkxlbiA9IE1hdGgubWluKGRlc3RMZW4sIHNyY0xlbilcbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBtaW5MZW47IGkrKykge1xuICAgIGNvbnN0IHNyY0l0ZW0gPSBzb3VyY2VbaV1cbiAgICBjb25zdCBkZXN0SXRlbSA9IGRlc3QuZ2V0KGkpXG5cbiAgICAvLyBJZiBib3RoIGFyZSBvYmplY3RzLCBtZXJnZSByZWN1cnNpdmVseVxuICAgIGlmIChpc1BsYWluT2JqZWN0KHNyY0l0ZW0pICYmIGRlc3RJdGVtIGluc3RhbmNlb2YgWS5NYXApIHtcbiAgICAgIGFwcGx5SnNvbk9iamVjdFRvWU1hcChkZXN0SXRlbSwgc3JjSXRlbSwgb3B0aW9ucylcbiAgICAgIGNvbnRpbnVlXG4gICAgfVxuXG4gICAgLy8gSWYgYm90aCBhcmUgYXJyYXlzLCBtZXJnZSByZWN1cnNpdmVseVxuICAgIGlmIChpc1BsYWluQXJyYXkoc3JjSXRlbSkgJiYgZGVzdEl0ZW0gaW5zdGFuY2VvZiBZLkFycmF5KSB7XG4gICAgICBhcHBseUpzb25BcnJheVRvWUFycmF5KGRlc3RJdGVtLCBzcmNJdGVtLCBvcHRpb25zKVxuICAgICAgY29udGludWVcbiAgICB9XG5cbiAgICAvLyBTa2lwIGlmIHByaW1pdGl2ZSB2YWx1ZSBpcyB1bmNoYW5nZWQgKG9wdGltaXphdGlvbilcbiAgICBpZiAoaXNQbGFpblByaW1pdGl2ZShzcmNJdGVtKSAmJiBkZXN0SXRlbSA9PT0gc3JjSXRlbSkge1xuICAgICAgY29udGludWVcbiAgICB9XG5cbiAgICAvLyBPdGhlcndpc2UsIHJlcGxhY2UgdGhlIGl0ZW1cbiAgICBkZXN0LmRlbGV0ZShpLCAxKVxuICAgIGRlc3QuaW5zZXJ0KGksIFtjb252ZXJ0SnNvblRvWWpzRGF0YShzcmNJdGVtKV0pXG4gIH1cblxuICAvLyBBZGQgbmV3IGl0ZW1zIGF0IHRoZSBlbmRcbiAgZm9yIChsZXQgaSA9IGRlc3RMZW47IGkgPCBzcmNMZW47IGkrKykge1xuICAgIGRlc3QucHVzaChbY29udmVydEpzb25Ub1lqc0RhdGEoc291cmNlW2ldKV0pXG4gIH1cblxuICAvLyBVcGRhdGUgc25hcHNob3QgdHJhY2tpbmcgYWZ0ZXIgc3VjY2Vzc2Z1bCBtZXJnZVxuICBzZXRZanNDb250YWluZXJTbmFwc2hvdChkZXN0LCBzb3VyY2UpXG59XG5cbi8qKlxuICogQXBwbGllcyBhIEpTT04gb2JqZWN0IHRvIGEgWS5NYXAsIHVzaW5nIHRoZSBjb252ZXJ0SnNvblRvWWpzRGF0YSB0byBjb252ZXJ0IHRoZSB2YWx1ZXMuXG4gKlxuICogQHBhcmFtIGRlc3QgVGhlIGRlc3RpbmF0aW9uIFkuTWFwLlxuICogQHBhcmFtIHNvdXJjZSBUaGUgc291cmNlIEpTT04gb2JqZWN0LlxuICogQHBhcmFtIG9wdGlvbnMgT3B0aW9ucyBmb3IgYXBwbHlpbmcgdGhlIEpTT04gZGF0YS5cbiAqL1xuZXhwb3J0IGNvbnN0IGFwcGx5SnNvbk9iamVjdFRvWU1hcCA9IChcbiAgZGVzdDogWS5NYXA8YW55PixcbiAgc291cmNlOiBQbGFpbk9iamVjdCxcbiAgb3B0aW9uczogQXBwbHlKc29uVG9ZanNPcHRpb25zID0ge31cbikgPT4ge1xuICBjb25zdCB7IG1vZGUgPSBcImFkZFwiIH0gPSBvcHRpb25zXG5cbiAgLy8gSW4gbWVyZ2UgbW9kZSwgY2hlY2sgaWYgdGhlIGNvbnRhaW5lciBpcyBhbHJlYWR5IHVwLXRvLWRhdGUgd2l0aCB0aGlzIHNuYXBzaG90XG4gIGlmIChtb2RlID09PSBcIm1lcmdlXCIgJiYgaXNZanNDb250YWluZXJVcFRvRGF0ZShkZXN0LCBzb3VyY2UpKSB7XG4gICAgcmV0dXJuXG4gIH1cblxuICBpZiAobW9kZSA9PT0gXCJhZGRcIikge1xuICAgIC8vIEFkZCBtb2RlOiBqdXN0IHNldCBhbGwgdmFsdWVzXG4gICAgZm9yIChjb25zdCBrIG9mIE9iamVjdC5rZXlzKHNvdXJjZSkpIHtcbiAgICAgIGNvbnN0IHYgPSBzb3VyY2Vba11cbiAgICAgIGlmICh2ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgZGVzdC5zZXQoaywgY29udmVydEpzb25Ub1lqc0RhdGEodikpXG4gICAgICB9XG4gICAgfVxuICAgIHJldHVyblxuICB9XG5cbiAgLy8gTWVyZ2UgbW9kZTogcmVjdXJzaXZlbHkgbWVyZ2UgdmFsdWVzLCBwcmVzZXJ2aW5nIGV4aXN0aW5nIGNvbnRhaW5lciByZWZlcmVuY2VzXG5cbiAgLy8gRGVsZXRlIGtleXMgdGhhdCBhcmUgbm90IHByZXNlbnQgaW4gc291cmNlIChvciBoYXZlIHVuZGVmaW5lZCB2YWx1ZSlcbiAgY29uc3Qgc291cmNlS2V5c1dpdGhWYWx1ZXMgPSBuZXcgU2V0KE9iamVjdC5rZXlzKHNvdXJjZSkuZmlsdGVyKChrKSA9PiBzb3VyY2Vba10gIT09IHVuZGVmaW5lZCkpXG4gIGZvciAoY29uc3Qga2V5IG9mIGRlc3Qua2V5cygpKSB7XG4gICAgaWYgKCFzb3VyY2VLZXlzV2l0aFZhbHVlcy5oYXMoa2V5KSkge1xuICAgICAgZGVzdC5kZWxldGUoa2V5KVxuICAgIH1cbiAgfVxuXG4gIGZvciAoY29uc3QgayBvZiBPYmplY3Qua2V5cyhzb3VyY2UpKSB7XG4gICAgY29uc3QgdiA9IHNvdXJjZVtrXVxuICAgIC8vIFNraXAgdW5kZWZpbmVkIHZhbHVlcyAtIFkuanMgbWFwcyBjYW5ub3Qgc3RvcmUgdW5kZWZpbmVkXG4gICAgaWYgKHYgPT09IHVuZGVmaW5lZCkge1xuICAgICAgY29udGludWVcbiAgICB9XG5cbiAgICBjb25zdCBleGlzdGluZyA9IGRlc3QuZ2V0KGspXG5cbiAgICAvLyBJZiBzb3VyY2UgaXMgYW4gb2JqZWN0IGFuZCBkZXN0IGhhcyBhIFkuTWFwLCBtZXJnZSByZWN1cnNpdmVseVxuICAgIGlmIChpc1BsYWluT2JqZWN0KHYpICYmIGV4aXN0aW5nIGluc3RhbmNlb2YgWS5NYXApIHtcbiAgICAgIGFwcGx5SnNvbk9iamVjdFRvWU1hcChleGlzdGluZywgdiwgb3B0aW9ucylcbiAgICAgIGNvbnRpbnVlXG4gICAgfVxuXG4gICAgLy8gSWYgc291cmNlIGlzIGFuIGFycmF5IGFuZCBkZXN0IGhhcyBhIFkuQXJyYXksIG1lcmdlIHJlY3Vyc2l2ZWx5XG4gICAgaWYgKGlzUGxhaW5BcnJheSh2KSAmJiBleGlzdGluZyBpbnN0YW5jZW9mIFkuQXJyYXkpIHtcbiAgICAgIGFwcGx5SnNvbkFycmF5VG9ZQXJyYXkoZXhpc3RpbmcsIHYsIG9wdGlvbnMpXG4gICAgICBjb250aW51ZVxuICAgIH1cblxuICAgIC8vIFNraXAgaWYgcHJpbWl0aXZlIHZhbHVlIGlzIHVuY2hhbmdlZCAob3B0aW1pemF0aW9uKVxuICAgIGlmIChpc1BsYWluUHJpbWl0aXZlKHYpICYmIGV4aXN0aW5nID09PSB2KSB7XG4gICAgICBjb250aW51ZVxuICAgIH1cblxuICAgIC8vIE90aGVyd2lzZSwgY29udmVydCBhbmQgc2V0IHRoZSB2YWx1ZSAodGhpcyBjcmVhdGVzIG5ldyBjb250YWluZXJzIGlmIG5lZWRlZClcbiAgICBkZXN0LnNldChrLCBjb252ZXJ0SnNvblRvWWpzRGF0YSh2KSlcbiAgfVxuXG4gIC8vIFVwZGF0ZSBzbmFwc2hvdCB0cmFja2luZyBhZnRlciBzdWNjZXNzZnVsIG1lcmdlXG4gIHNldFlqc0NvbnRhaW5lclNuYXBzaG90KGRlc3QsIHNvdXJjZSlcbn1cbiIsImltcG9ydCB7IERlZXBDaGFuZ2UsIERlZXBDaGFuZ2VUeXBlIH0gZnJvbSBcIm1vYngta2V5c3RvbmVcIlxuaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcbmltcG9ydCB7IGZhaWx1cmUgfSBmcm9tIFwiLi4vdXRpbHMvZXJyb3JcIlxuaW1wb3J0IHsgaXNZanNWYWx1ZURlbGV0ZWQgfSBmcm9tIFwiLi4vdXRpbHMvaXNZanNWYWx1ZURlbGV0ZWRcIlxuaW1wb3J0IHsgY29udmVydEpzb25Ub1lqc0RhdGEgfSBmcm9tIFwiLi9jb252ZXJ0SnNvblRvWWpzRGF0YVwiXG5pbXBvcnQgeyByZXNvbHZlWWpzUGF0aCB9IGZyb20gXCIuL3Jlc29sdmVZanNQYXRoXCJcblxuLyoqXG4gKiBDb252ZXJ0cyBhIHNuYXBzaG90IHZhbHVlIHRvIGEgWWpzLWNvbXBhdGlibGUgdmFsdWUuXG4gKiBOb3RlOiBBbGwgdmFsdWVzIHBhc3NlZCBoZXJlIGFyZSBhbHJlYWR5IHNuYXBzaG90cyAoY2FwdHVyZWQgYXQgY2hhbmdlIHRpbWUpLlxuICovXG5mdW5jdGlvbiBjb252ZXJ0VmFsdWUodjogdW5rbm93bik6IGFueSB7XG4gIC8vIEhhbmRsZSBwcmltaXRpdmVzIGRpcmVjdGx5XG4gIGlmICh2ID09PSBudWxsIHx8IHYgPT09IHVuZGVmaW5lZCB8fCB0eXBlb2YgdiAhPT0gXCJvYmplY3RcIikge1xuICAgIHJldHVybiB2XG4gIH1cbiAgLy8gSGFuZGxlIHBsYWluIGFycmF5cyAtIHVzZWQgZm9yIGVtcHR5IGFycmF5IGluaXRcbiAgaWYgKEFycmF5LmlzQXJyYXkodikgJiYgdi5sZW5ndGggPT09IDApIHtcbiAgICByZXR1cm4gbmV3IFkuQXJyYXkoKVxuICB9XG4gIC8vIFZhbHVlIGlzIGFscmVhZHkgYSBzbmFwc2hvdCwgY29udmVydCB0byBZanMgZGF0YVxuICByZXR1cm4gY29udmVydEpzb25Ub1lqc0RhdGEodiBhcyBhbnkpXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBhcHBseU1vYnhDaGFuZ2VUb1lqc09iamVjdChcbiAgY2hhbmdlOiBEZWVwQ2hhbmdlLFxuICB5anNPYmplY3Q6IFkuTWFwPGFueT4gfCBZLkFycmF5PGFueT4gfCBZLlRleHRcbik6IHZvaWQge1xuICAvLyBDaGVjayBpZiB0aGUgWUpTIG9iamVjdCBpcyBkZWxldGVkXG4gIGlmIChpc1lqc1ZhbHVlRGVsZXRlZCh5anNPYmplY3QpKSB7XG4gICAgdGhyb3cgZmFpbHVyZShcImNhbm5vdCBhcHBseSBwYXRjaCB0byBkZWxldGVkIFlqcyB2YWx1ZVwiKVxuICB9XG5cbiAgY29uc3QgeWpzQ29udGFpbmVyID0gcmVzb2x2ZVlqc1BhdGgoeWpzT2JqZWN0LCBjaGFuZ2UucGF0aClcblxuICBpZiAoIXlqc0NvbnRhaW5lcikge1xuICAgIC8vIENvbnRhaW5lciBub3QgZm91bmQsIHNraXAgdGhpcyBjaGFuZ2VcbiAgICByZXR1cm5cbiAgfVxuXG4gIGlmICh5anNDb250YWluZXIgaW5zdGFuY2VvZiBZLkFycmF5KSB7XG4gICAgaWYgKGNoYW5nZS50eXBlID09PSBEZWVwQ2hhbmdlVHlwZS5BcnJheVNwbGljZSkge1xuICAgICAgLy8gc3BsaWNlXG4gICAgICB5anNDb250YWluZXIuZGVsZXRlKGNoYW5nZS5pbmRleCwgY2hhbmdlLnJlbW92ZWRWYWx1ZXMubGVuZ3RoKVxuICAgICAgaWYgKGNoYW5nZS5hZGRlZFZhbHVlcy5sZW5ndGggPiAwKSB7XG4gICAgICAgIGNvbnN0IHZhbHVlc1RvSW5zZXJ0ID0gY2hhbmdlLmFkZGVkVmFsdWVzLm1hcChjb252ZXJ0VmFsdWUpXG4gICAgICAgIHlqc0NvbnRhaW5lci5pbnNlcnQoY2hhbmdlLmluZGV4LCB2YWx1ZXNUb0luc2VydClcbiAgICAgIH1cbiAgICB9IGVsc2UgaWYgKGNoYW5nZS50eXBlID09PSBEZWVwQ2hhbmdlVHlwZS5BcnJheVVwZGF0ZSkge1xuICAgICAgLy8gdXBkYXRlXG4gICAgICB5anNDb250YWluZXIuZGVsZXRlKGNoYW5nZS5pbmRleCwgMSlcbiAgICAgIHlqc0NvbnRhaW5lci5pbnNlcnQoY2hhbmdlLmluZGV4LCBbY29udmVydFZhbHVlKGNoYW5nZS5uZXdWYWx1ZSldKVxuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBmYWlsdXJlKGB1bnN1cHBvcnRlZCBhcnJheSBjaGFuZ2UgdHlwZTogJHtjaGFuZ2UudHlwZX1gKVxuICAgIH1cbiAgfSBlbHNlIGlmICh5anNDb250YWluZXIgaW5zdGFuY2VvZiBZLk1hcCkge1xuICAgIGlmIChjaGFuZ2UudHlwZSA9PT0gRGVlcENoYW5nZVR5cGUuT2JqZWN0QWRkIHx8IGNoYW5nZS50eXBlID09PSBEZWVwQ2hhbmdlVHlwZS5PYmplY3RVcGRhdGUpIHtcbiAgICAgIGNvbnN0IGtleSA9IGNoYW5nZS5rZXlcbiAgICAgIGlmIChjaGFuZ2UubmV3VmFsdWUgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICB5anNDb250YWluZXIuZGVsZXRlKGtleSlcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHlqc0NvbnRhaW5lci5zZXQoa2V5LCBjb252ZXJ0VmFsdWUoY2hhbmdlLm5ld1ZhbHVlKSlcbiAgICAgIH1cbiAgICB9IGVsc2UgaWYgKGNoYW5nZS50eXBlID09PSBEZWVwQ2hhbmdlVHlwZS5PYmplY3RSZW1vdmUpIHtcbiAgICAgIGNvbnN0IGtleSA9IGNoYW5nZS5rZXlcbiAgICAgIHlqc0NvbnRhaW5lci5kZWxldGUoa2V5KVxuICAgIH0gZWxzZSB7XG4gICAgICB0aHJvdyBmYWlsdXJlKGB1bnN1cHBvcnRlZCBvYmplY3QgY2hhbmdlIHR5cGU6ICR7Y2hhbmdlLnR5cGV9YClcbiAgICB9XG4gIH0gZWxzZSBpZiAoeWpzQ29udGFpbmVyIGluc3RhbmNlb2YgWS5UZXh0KSB7XG4gICAgLy8gWS5UZXh0IGlzIGhhbmRsZWQgZGlmZmVyZW50bHkgLSBpbml0IGNoYW5nZXMgZm9yIHRleHQgYXJlIG1hbmFnZWQgYnkgWWpzVGV4dE1vZGVsXG4gICAgLy8gU2tpcCBpbml0IGNoYW5nZXMgZm9yIFkuVGV4dCBjb250YWluZXJzXG4gICAgcmV0dXJuXG4gIH0gZWxzZSB7XG4gICAgdGhyb3cgZmFpbHVyZShgdW5zdXBwb3J0ZWQgWWpzIGNvbnRhaW5lciB0eXBlOiAke3lqc0NvbnRhaW5lcn1gKVxuICB9XG59XG4iLCJpbXBvcnQgeyBhY3Rpb24gfSBmcm9tIFwibW9ieFwiXG5pbXBvcnQgeyBtb2RlbFNuYXBzaG90T3V0V2l0aE1ldGFkYXRhIH0gZnJvbSBcIm1vYngta2V5c3RvbmVcIlxuaW1wb3J0ICogYXMgWSBmcm9tIFwieWpzXCJcbmltcG9ydCB7IFBsYWluT2JqZWN0LCBQbGFpblZhbHVlIH0gZnJvbSBcIi4uL3BsYWluVHlwZXNcIlxuaW1wb3J0IHsgWWpzVGV4dE1vZGVsIH0gZnJvbSBcIi4vWWpzVGV4dE1vZGVsXCJcblxuZXhwb3J0IHR5cGUgWWpzRGF0YSA9IFkuQXJyYXk8YW55PiB8IFkuTWFwPGFueT4gfCBZLlRleHQgfCBQbGFpblZhbHVlXG5cbmV4cG9ydCBjb25zdCBjb252ZXJ0WWpzRGF0YVRvSnNvbiA9IGFjdGlvbigoeWpzRGF0YTogWWpzRGF0YSk6IFBsYWluVmFsdWUgPT4ge1xuICBpZiAoeWpzRGF0YSBpbnN0YW5jZW9mIFkuQXJyYXkpIHtcbiAgICByZXR1cm4geWpzRGF0YS5tYXAoKHYpID0+IGNvbnZlcnRZanNEYXRhVG9Kc29uKHYpKVxuICB9XG5cbiAgaWYgKHlqc0RhdGEgaW5zdGFuY2VvZiBZLk1hcCkge1xuICAgIGNvbnN0IG9iajogUGxhaW5PYmplY3QgPSB7fVxuICAgIHlqc0RhdGEuZm9yRWFjaCgodiwgaykgPT4ge1xuICAgICAgb2JqW2tdID0gY29udmVydFlqc0RhdGFUb0pzb24odilcbiAgICB9KVxuICAgIHJldHVybiBvYmpcbiAgfVxuXG4gIGlmICh5anNEYXRhIGluc3RhbmNlb2YgWS5UZXh0KSB7XG4gICAgY29uc3QgZGVsdGFzID0geWpzRGF0YS50b0RlbHRhKCkgYXMgdW5rbm93bltdXG5cbiAgICByZXR1cm4gbW9kZWxTbmFwc2hvdE91dFdpdGhNZXRhZGF0YShZanNUZXh0TW9kZWwsIHtcbiAgICAgIGRlbHRhTGlzdDogZGVsdGFzLmxlbmd0aCA+IDAgPyBbeyAkZnJvemVuOiB0cnVlLCBkYXRhOiBkZWx0YXMgfV0gOiBbXSxcbiAgICB9KSBhcyB1bmtub3duIGFzIFBsYWluVmFsdWVcbiAgfVxuXG4gIC8vIGFzc3VtZSBpdCdzIGEgcHJpbWl0aXZlXG4gIHJldHVybiB5anNEYXRhXG59KVxuIiwiaW1wb3J0IHsgcmVtb3ZlIH0gZnJvbSBcIm1vYnhcIlxuaW1wb3J0IHtcbiAgRnJvemVuLFxuICBmcm9tU25hcHNob3QsXG4gIGZyb3plbixcbiAgZ2V0U25hcHNob3QsXG4gIGdldFNuYXBzaG90TW9kZWxJZCxcbiAgaXNGcm96ZW5TbmFwc2hvdCxcbiAgaXNNb2RlbCxcbiAgUGF0aCxcbiAgcmVzb2x2ZVBhdGgsXG4gIHJ1blVucHJvdGVjdGVkLFxufSBmcm9tIFwibW9ieC1rZXlzdG9uZVwiXG5pbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuaW1wb3J0IHsgZmFpbHVyZSB9IGZyb20gXCIuLi91dGlscy9lcnJvclwiXG5pbXBvcnQgeyBjb252ZXJ0WWpzRGF0YVRvSnNvbiB9IGZyb20gXCIuL2NvbnZlcnRZanNEYXRhVG9Kc29uXCJcblxuLy8gUmVwcmVzZW50cyB0aGUgbWFwIG9mIHBvdGVudGlhbCBvYmplY3RzIHRvIHJlY29uY2lsZSAoSUQgLT4gT2JqZWN0KVxuZXhwb3J0IHR5cGUgUmVjb25jaWxpYXRpb25NYXAgPSBNYXA8c3RyaW5nLCBvYmplY3Q+XG5cbi8qKlxuICogQXBwbGllcyBhIFkuanMgZXZlbnQgZGlyZWN0bHkgdG8gdGhlIE1vYlggbW9kZWwgdHJlZSB1c2luZyBwcm9wZXIgbXV0YXRpb25zXG4gKiAoc3BsaWNlIGZvciBhcnJheXMsIHByb3BlcnR5IGFzc2lnbm1lbnQgZm9yIG9iamVjdHMpLlxuICogVGhpcyBpcyBtb3JlIGVmZmljaWVudCB0aGFuIGNvbnZlcnRpbmcgdG8gcGF0Y2hlcyBmaXJzdC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGFwcGx5WWpzRXZlbnRUb01vYngoXG4gIGV2ZW50OiBZLllFdmVudDxhbnk+LFxuICBib3VuZE9iamVjdDogb2JqZWN0LFxuICByZWNvbmNpbGlhdGlvbk1hcDogUmVjb25jaWxpYXRpb25NYXBcbik6IHZvaWQge1xuICBjb25zdCBwYXRoID0gZXZlbnQucGF0aCBhcyBQYXRoXG4gIGNvbnN0IHsgdmFsdWU6IHRhcmdldCB9ID0gcmVzb2x2ZVBhdGgoYm91bmRPYmplY3QsIHBhdGgpXG5cbiAgaWYgKCF0YXJnZXQpIHtcbiAgICB0aHJvdyBmYWlsdXJlKGBjYW5ub3QgcmVzb2x2ZSBwYXRoICR7SlNPTi5zdHJpbmdpZnkocGF0aCl9YClcbiAgfVxuXG4gIC8vIFdyYXAgaW4gcnVuVW5wcm90ZWN0ZWQgc2luY2Ugd2UncmUgbW9kaWZ5aW5nIHRoZSB0cmVlIGZyb20gb3V0c2lkZSBhIG1vZGVsIGFjdGlvblxuICBydW5VbnByb3RlY3RlZCgoKSA9PiB7XG4gICAgaWYgKGV2ZW50IGluc3RhbmNlb2YgWS5ZTWFwRXZlbnQpIHtcbiAgICAgIGFwcGx5WU1hcEV2ZW50VG9Nb2J4KGV2ZW50LCB0YXJnZXQsIHJlY29uY2lsaWF0aW9uTWFwKVxuICAgIH0gZWxzZSBpZiAoZXZlbnQgaW5zdGFuY2VvZiBZLllBcnJheUV2ZW50KSB7XG4gICAgICBhcHBseVlBcnJheUV2ZW50VG9Nb2J4KGV2ZW50LCB0YXJnZXQsIHJlY29uY2lsaWF0aW9uTWFwKVxuICAgIH0gZWxzZSBpZiAoZXZlbnQgaW5zdGFuY2VvZiBZLllUZXh0RXZlbnQpIHtcbiAgICAgIGFwcGx5WVRleHRFdmVudFRvTW9ieChldmVudCwgdGFyZ2V0KVxuICAgIH1cbiAgfSlcbn1cblxuZnVuY3Rpb24gcHJvY2Vzc0RlbGV0ZWRWYWx1ZSh2YWw6IHVua25vd24sIHJlY29uY2lsaWF0aW9uTWFwOiBSZWNvbmNpbGlhdGlvbk1hcCkge1xuICBpZiAodmFsICYmIHR5cGVvZiB2YWwgPT09IFwib2JqZWN0XCIgJiYgaXNNb2RlbCh2YWwpKSB7XG4gICAgY29uc3Qgc24gPSBnZXRTbmFwc2hvdCh2YWwpXG4gICAgY29uc3QgaWQgPSBnZXRTbmFwc2hvdE1vZGVsSWQoc24pXG4gICAgaWYgKGlkKSB7XG4gICAgICByZWNvbmNpbGlhdGlvbk1hcC5zZXQoaWQsIHZhbClcbiAgICB9XG4gIH1cbn1cblxuZnVuY3Rpb24gcmV2aXZlVmFsdWUoanNvblZhbHVlOiBhbnksIHJlY29uY2lsaWF0aW9uTWFwOiBSZWNvbmNpbGlhdGlvbk1hcCk6IGFueSB7XG4gIC8vIEhhbmRsZSBwcmltaXRpdmVzXG4gIGlmIChqc29uVmFsdWUgPT09IG51bGwgfHwgdHlwZW9mIGpzb25WYWx1ZSAhPT0gXCJvYmplY3RcIikge1xuICAgIHJldHVybiBqc29uVmFsdWVcbiAgfVxuXG4gIC8vIEhhbmRsZSBmcm96ZW5cbiAgaWYgKGlzRnJvemVuU25hcHNob3QoanNvblZhbHVlKSkge1xuICAgIHJldHVybiBmcm96ZW4oanNvblZhbHVlLmRhdGEpXG4gIH1cblxuICAvLyBJZiB3ZSBoYXZlIGEgcmVjb25jaWxpYXRpb24gbWFwIGFuZCB0aGUgdmFsdWUgbG9va3MgbGlrZSBhIG1vZGVsIHdpdGggYW4gSUQsIGNoZWNrIGlmIHdlIGhhdmUgaXRcbiAgaWYgKHJlY29uY2lsaWF0aW9uTWFwICYmIGpzb25WYWx1ZSAmJiB0eXBlb2YganNvblZhbHVlID09PSBcIm9iamVjdFwiKSB7XG4gICAgY29uc3QgbW9kZWxJZCA9IGdldFNuYXBzaG90TW9kZWxJZChqc29uVmFsdWUpXG4gICAgaWYgKG1vZGVsSWQpIHtcbiAgICAgIGNvbnN0IGV4aXN0aW5nID0gcmVjb25jaWxpYXRpb25NYXAuZ2V0KG1vZGVsSWQpXG4gICAgICBpZiAoZXhpc3RpbmcpIHtcbiAgICAgICAgcmVjb25jaWxpYXRpb25NYXAuZGVsZXRlKG1vZGVsSWQpXG4gICAgICAgIHJldHVybiBleGlzdGluZ1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBmcm9tU25hcHNob3QoanNvblZhbHVlKVxufVxuXG5mdW5jdGlvbiBhcHBseVlNYXBFdmVudFRvTW9ieChcbiAgZXZlbnQ6IFkuWU1hcEV2ZW50PGFueT4sXG4gIHRhcmdldDogUmVjb3JkPHN0cmluZywgYW55PixcbiAgcmVjb25jaWxpYXRpb25NYXA6IFJlY29uY2lsaWF0aW9uTWFwXG4pOiB2b2lkIHtcbiAgY29uc3Qgc291cmNlID0gZXZlbnQudGFyZ2V0XG5cbiAgZXZlbnQuY2hhbmdlcy5rZXlzLmZvckVhY2goKGNoYW5nZSwga2V5KSA9PiB7XG4gICAgc3dpdGNoIChjaGFuZ2UuYWN0aW9uKSB7XG4gICAgICBjYXNlIFwiYWRkXCI6XG4gICAgICBjYXNlIFwidXBkYXRlXCI6IHtcbiAgICAgICAgY29uc3QgeWpzVmFsdWUgPSBzb3VyY2UuZ2V0KGtleSlcbiAgICAgICAgY29uc3QganNvblZhbHVlID0gY29udmVydFlqc0RhdGFUb0pzb24oeWpzVmFsdWUpXG5cbiAgICAgICAgLy8gSWYgdXBkYXRpbmcsIHRoZSBvbGQgdmFsdWUgaXMgb3ZlcndyaXR0ZW4gKGRlbGV0ZWQgY29uY2VwdHVhbGx5KVxuICAgICAgICBpZiAoY2hhbmdlLmFjdGlvbiA9PT0gXCJ1cGRhdGVcIikge1xuICAgICAgICAgIHByb2Nlc3NEZWxldGVkVmFsdWUodGFyZ2V0W2tleV0sIHJlY29uY2lsaWF0aW9uTWFwKVxuICAgICAgICB9XG5cbiAgICAgICAgdGFyZ2V0W2tleV0gPSByZXZpdmVWYWx1ZShqc29uVmFsdWUsIHJlY29uY2lsaWF0aW9uTWFwKVxuICAgICAgICBicmVha1xuICAgICAgfVxuXG4gICAgICBjYXNlIFwiZGVsZXRlXCI6IHtcbiAgICAgICAgcHJvY2Vzc0RlbGV0ZWRWYWx1ZSh0YXJnZXRba2V5XSwgcmVjb25jaWxpYXRpb25NYXApXG4gICAgICAgIC8vIFVzZSBNb2JYJ3MgcmVtb3ZlIHRvIHByb3Blcmx5IGRlbGV0ZSB0aGUga2V5IGZyb20gdGhlIG9ic2VydmFibGUgb2JqZWN0XG4gICAgICAgIC8vIFRoaXMgdHJpZ2dlcnMgdGhlIFwicmVtb3ZlXCIgaW50ZXJjZXB0b3IgaW4gbW9ieC1rZXlzdG9uZSdzIHR3ZWFrZXJcbiAgICAgICAgaWYgKGlzTW9kZWwodGFyZ2V0KSkge1xuICAgICAgICAgIHJlbW92ZSh0YXJnZXQuJCwga2V5KVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHJlbW92ZSh0YXJnZXQsIGtleSlcbiAgICAgICAgfVxuICAgICAgICBicmVha1xuICAgICAgfVxuXG4gICAgICBkZWZhdWx0OlxuICAgICAgICB0aHJvdyBmYWlsdXJlKGB1bnN1cHBvcnRlZCBZanMgbWFwIGV2ZW50IGFjdGlvbjogJHtjaGFuZ2UuYWN0aW9ufWApXG4gICAgfVxuICB9KVxufVxuXG5mdW5jdGlvbiBhcHBseVlBcnJheUV2ZW50VG9Nb2J4KFxuICBldmVudDogWS5ZQXJyYXlFdmVudDxhbnk+LFxuICB0YXJnZXQ6IGFueVtdLFxuICByZWNvbmNpbGlhdGlvbk1hcDogUmVjb25jaWxpYXRpb25NYXBcbik6IHZvaWQge1xuICAvLyBQcm9jZXNzIGRlbHRhIG9wZXJhdGlvbnMgaW4gb3JkZXJcbiAgbGV0IGN1cnJlbnRJbmRleCA9IDBcblxuICBmb3IgKGNvbnN0IGNoYW5nZSBvZiBldmVudC5jaGFuZ2VzLmRlbHRhKSB7XG4gICAgaWYgKGNoYW5nZS5yZXRhaW4pIHtcbiAgICAgIGN1cnJlbnRJbmRleCArPSBjaGFuZ2UucmV0YWluXG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZS5kZWxldGUpIHtcbiAgICAgIC8vIENhcHR1cmUgZGVsZXRlZCBpdGVtcyBmb3IgcmVjb25jaWxpYXRpb25cbiAgICAgIGNvbnN0IGRlbGV0ZWRJdGVtcyA9IHRhcmdldC5zbGljZShjdXJyZW50SW5kZXgsIGN1cnJlbnRJbmRleCArIGNoYW5nZS5kZWxldGUpXG4gICAgICBkZWxldGVkSXRlbXMuZm9yRWFjaCgoaXRlbSkgPT4ge1xuICAgICAgICBwcm9jZXNzRGVsZXRlZFZhbHVlKGl0ZW0sIHJlY29uY2lsaWF0aW9uTWFwKVxuICAgICAgfSlcblxuICAgICAgLy8gRGVsZXRlIGl0ZW1zIGF0IGN1cnJlbnQgcG9zaXRpb25cbiAgICAgIHRhcmdldC5zcGxpY2UoY3VycmVudEluZGV4LCBjaGFuZ2UuZGVsZXRlKVxuICAgIH1cblxuICAgIGlmIChjaGFuZ2UuaW5zZXJ0KSB7XG4gICAgICAvLyBJbnNlcnQgaXRlbXMgYXQgY3VycmVudCBwb3NpdGlvblxuICAgICAgY29uc3QgaW5zZXJ0ZWRJdGVtcyA9IEFycmF5LmlzQXJyYXkoY2hhbmdlLmluc2VydCkgPyBjaGFuZ2UuaW5zZXJ0IDogW2NoYW5nZS5pbnNlcnRdXG4gICAgICBjb25zdCB2YWx1ZXMgPSBpbnNlcnRlZEl0ZW1zLm1hcCgoeWpzVmFsdWUpID0+IHtcbiAgICAgICAgY29uc3QganNvblZhbHVlID0gY29udmVydFlqc0RhdGFUb0pzb24oeWpzVmFsdWUpXG4gICAgICAgIHJldHVybiByZXZpdmVWYWx1ZShqc29uVmFsdWUsIHJlY29uY2lsaWF0aW9uTWFwKVxuICAgICAgfSlcblxuICAgICAgdGFyZ2V0LnNwbGljZShjdXJyZW50SW5kZXgsIDAsIC4uLnZhbHVlcylcbiAgICAgIGN1cnJlbnRJbmRleCArPSB2YWx1ZXMubGVuZ3RoXG4gICAgfVxuICB9XG59XG5cbmZ1bmN0aW9uIGFwcGx5WVRleHRFdmVudFRvTW9ieChcbiAgZXZlbnQ6IFkuWVRleHRFdmVudCxcbiAgdGFyZ2V0OiB7IGRlbHRhTGlzdD86IEZyb3plbjx1bmtub3duW10+W10gfVxuKTogdm9pZCB7XG4gIC8vIFlqc1RleHRNb2RlbCBoYW5kbGVzIHRleHQgZXZlbnRzIGJ5IGFwcGVuZGluZyBkZWx0YSB0byBkZWx0YUxpc3RcbiAgaWYgKHRhcmdldD8uZGVsdGFMaXN0KSB7XG4gICAgdGFyZ2V0LmRlbHRhTGlzdC5wdXNoKGZyb3plbihldmVudC5kZWx0YSkpXG4gIH1cbn1cbiIsImltcG9ydCB7IGFjdGlvbiB9IGZyb20gXCJtb2J4XCJcbmltcG9ydCB7XG4gIEFueURhdGFNb2RlbCxcbiAgQW55TW9kZWwsXG4gIEFueVN0YW5kYXJkVHlwZSxcbiAgRGVlcENoYW5nZSxcbiAgRGVlcENoYW5nZVR5cGUsXG4gIGZyb21TbmFwc2hvdCxcbiAgZ2V0UGFyZW50VG9DaGlsZFBhdGgsXG4gIGdldFNuYXBzaG90LFxuICBpc1RyZWVOb2RlLFxuICBNb2RlbENsYXNzLFxuICBvbkRlZXBDaGFuZ2UsXG4gIG9uR2xvYmFsRGVlcENoYW5nZSxcbiAgb25TbmFwc2hvdCxcbiAgU25hcHNob3RJbk9mLFxuICBUeXBlVG9EYXRhLFxufSBmcm9tIFwibW9ieC1rZXlzdG9uZVwiXG5pbXBvcnQgKiBhcyBZIGZyb20gXCJ5anNcIlxuaW1wb3J0IHR5cGUgeyBQbGFpbkFycmF5LCBQbGFpbk9iamVjdCB9IGZyb20gXCIuLi9wbGFpblR5cGVzXCJcbmltcG9ydCB7IGZhaWx1cmUgfSBmcm9tIFwiLi4vdXRpbHMvZXJyb3JcIlxuaW1wb3J0IHsgZ2V0WWpzQ29sbGVjdGlvbkF0b20gfSBmcm9tIFwiLi4vdXRpbHMvZ2V0T3JDcmVhdGVZanNDb2xsZWN0aW9uQXRvbVwiXG5pbXBvcnQgeyBpc1lqc1ZhbHVlRGVsZXRlZCB9IGZyb20gXCIuLi91dGlscy9pc1lqc1ZhbHVlRGVsZXRlZFwiXG5pbXBvcnQgeyBhcHBseU1vYnhDaGFuZ2VUb1lqc09iamVjdCB9IGZyb20gXCIuL2FwcGx5TW9ieENoYW5nZVRvWWpzT2JqZWN0XCJcbmltcG9ydCB7IGFwcGx5WWpzRXZlbnRUb01vYngsIFJlY29uY2lsaWF0aW9uTWFwIH0gZnJvbSBcIi4vYXBwbHlZanNFdmVudFRvTW9ieFwiXG5pbXBvcnQgeyBhcHBseUpzb25BcnJheVRvWUFycmF5LCBhcHBseUpzb25PYmplY3RUb1lNYXAgfSBmcm9tIFwiLi9jb252ZXJ0SnNvblRvWWpzRGF0YVwiXG5pbXBvcnQgeyBjb252ZXJ0WWpzRGF0YVRvSnNvbiB9IGZyb20gXCIuL2NvbnZlcnRZanNEYXRhVG9Kc29uXCJcbmltcG9ydCB7IFlqc0JpbmRpbmdDb250ZXh0LCB5anNCaW5kaW5nQ29udGV4dCB9IGZyb20gXCIuL3lqc0JpbmRpbmdDb250ZXh0XCJcbmltcG9ydCB7IHNldFlqc0NvbnRhaW5lclNuYXBzaG90IH0gZnJvbSBcIi4veWpzU25hcHNob3RUcmFja2luZ1wiXG5cbi8qKlxuICogQ2FwdHVyZXMgc25hcHNob3RzIG9mIHRyZWUgbm9kZXMgaW4gYSBEZWVwQ2hhbmdlLlxuICogVGhpcyBlbnN1cmVzIHZhbHVlcyBhcmUgY2FwdHVyZWQgYXQgY2hhbmdlIHRpbWUsIG5vdCBhdCBhcHBseSB0aW1lLFxuICogcHJldmVudGluZyBpc3N1ZXMgd2hlbiB2YWx1ZXMgYXJlIG11dGF0ZWQgYWZ0ZXIgYmVpbmcgYWRkZWQgdG8gYSBjb2xsZWN0aW9uLlxuICovXG5mdW5jdGlvbiBjYXB0dXJlQ2hhbmdlU25hcHNob3RzKGNoYW5nZTogRGVlcENoYW5nZSk6IERlZXBDaGFuZ2Uge1xuICBpZiAoY2hhbmdlLnR5cGUgPT09IERlZXBDaGFuZ2VUeXBlLkFycmF5U3BsaWNlICYmIGNoYW5nZS5hZGRlZFZhbHVlcy5sZW5ndGggPiAwKSB7XG4gICAgY29uc3Qgc25hcHNob3RzID0gY2hhbmdlLmFkZGVkVmFsdWVzLm1hcCgodikgPT4gKGlzVHJlZU5vZGUodikgPyBnZXRTbmFwc2hvdCh2KSA6IHYpKVxuICAgIHJldHVybiB7IC4uLmNoYW5nZSwgYWRkZWRWYWx1ZXM6IHNuYXBzaG90cyB9XG4gIH0gZWxzZSBpZiAoY2hhbmdlLnR5cGUgPT09IERlZXBDaGFuZ2VUeXBlLkFycmF5VXBkYXRlKSB7XG4gICAgY29uc3Qgc25hcHNob3QgPSBpc1RyZWVOb2RlKGNoYW5nZS5uZXdWYWx1ZSkgPyBnZXRTbmFwc2hvdChjaGFuZ2UubmV3VmFsdWUpIDogY2hhbmdlLm5ld1ZhbHVlXG4gICAgcmV0dXJuIHsgLi4uY2hhbmdlLCBuZXdWYWx1ZTogc25hcHNob3QgfVxuICB9IGVsc2UgaWYgKFxuICAgIGNoYW5nZS50eXBlID09PSBEZWVwQ2hhbmdlVHlwZS5PYmplY3RBZGQgfHxcbiAgICBjaGFuZ2UudHlwZSA9PT0gRGVlcENoYW5nZVR5cGUuT2JqZWN0VXBkYXRlXG4gICkge1xuICAgIGNvbnN0IHNuYXBzaG90ID0gaXNUcmVlTm9kZShjaGFuZ2UubmV3VmFsdWUpID8gZ2V0U25hcHNob3QoY2hhbmdlLm5ld1ZhbHVlKSA6IGNoYW5nZS5uZXdWYWx1ZVxuICAgIHJldHVybiB7IC4uLmNoYW5nZSwgbmV3VmFsdWU6IHNuYXBzaG90IH1cbiAgfVxuICByZXR1cm4gY2hhbmdlXG59XG5cbi8qKlxuICogQ3JlYXRlcyBhIGJpZGlyZWN0aW9uYWwgYmluZGluZyBiZXR3ZWVuIGEgWS5qcyBkYXRhIHN0cnVjdHVyZSBhbmQgYSBtb2J4LWtleXN0b25lIG1vZGVsLlxuICovXG5leHBvcnQgZnVuY3Rpb24gYmluZFlqc1RvTW9ieEtleXN0b25lPFxuICBUVHlwZSBleHRlbmRzIEFueVN0YW5kYXJkVHlwZSB8IE1vZGVsQ2xhc3M8QW55TW9kZWw+IHwgTW9kZWxDbGFzczxBbnlEYXRhTW9kZWw+LFxuPih7XG4gIHlqc0RvYyxcbiAgeWpzT2JqZWN0LFxuICBtb2J4S2V5c3RvbmVUeXBlLFxufToge1xuICAvKipcbiAgICogVGhlIFkuanMgZG9jdW1lbnQuXG4gICAqL1xuICB5anNEb2M6IFkuRG9jXG4gIC8qKlxuICAgKiBUaGUgYm91bmQgWS5qcyBkYXRhIHN0cnVjdHVyZS5cbiAgICovXG4gIHlqc09iamVjdDogWS5NYXA8YW55PiB8IFkuQXJyYXk8YW55PiB8IFkuVGV4dFxuICAvKipcbiAgICogVGhlIG1vYngta2V5c3RvbmUgbW9kZWwgdHlwZS5cbiAgICovXG4gIG1vYnhLZXlzdG9uZVR5cGU6IFRUeXBlXG59KToge1xuICAvKipcbiAgICogVGhlIGJvdW5kIG1vYngta2V5c3RvbmUgaW5zdGFuY2UuXG4gICAqL1xuICBib3VuZE9iamVjdDogVHlwZVRvRGF0YTxUVHlwZT5cbiAgLyoqXG4gICAqIERpc3Bvc2VzIHRoZSBiaW5kaW5nLlxuICAgKi9cbiAgZGlzcG9zZTogKCkgPT4gdm9pZFxuICAvKipcbiAgICogVGhlIFkuanMgb3JpZ2luIHN5bWJvbCB1c2VkIGZvciBiaW5kaW5nIHRyYW5zYWN0aW9ucy5cbiAgICovXG4gIHlqc09yaWdpbjogc3ltYm9sXG59IHtcbiAgY29uc3QgeWpzT3JpZ2luID0gU3ltYm9sKFwiYmluZFlqc1RvTW9ieEtleXN0b25lVHJhbnNhY3Rpb25PcmlnaW5cIilcblxuICBsZXQgYXBwbHlpbmdZanNDaGFuZ2VzVG9Nb2J4S2V5c3RvbmUgPSAwXG5cbiAgY29uc3QgYmluZGluZ0NvbnRleHQ6IFlqc0JpbmRpbmdDb250ZXh0ID0ge1xuICAgIHlqc0RvYyxcbiAgICB5anNPYmplY3QsXG4gICAgbW9ieEtleXN0b25lVHlwZSxcbiAgICB5anNPcmlnaW4sXG4gICAgYm91bmRPYmplY3Q6IHVuZGVmaW5lZCwgLy8gbm90IHlldCBjcmVhdGVkXG5cbiAgICBnZXQgaXNBcHBseWluZ1lqc0NoYW5nZXNUb01vYnhLZXlzdG9uZSgpIHtcbiAgICAgIHJldHVybiBhcHBseWluZ1lqc0NoYW5nZXNUb01vYnhLZXlzdG9uZSA+IDBcbiAgICB9LFxuICB9XG5cbiAgaWYgKGlzWWpzVmFsdWVEZWxldGVkKHlqc09iamVjdCkpIHtcbiAgICB0aHJvdyBmYWlsdXJlKFwiY2Fubm90IGFwcGx5IHBhdGNoIHRvIGRlbGV0ZWQgWWpzIHZhbHVlXCIpXG4gIH1cblxuICBjb25zdCB5anNKc29uID0gY29udmVydFlqc0RhdGFUb0pzb24oeWpzT2JqZWN0KVxuXG4gIGxldCBib3VuZE9iamVjdDogVHlwZVRvRGF0YTxUVHlwZT5cblxuICAvLyBUcmFjayBpZiBhbnkgaW5pdCBjaGFuZ2VzIG9jY3VyIGR1cmluZyBmcm9tU25hcHNob3RcbiAgLy8gKGUuZy4sIGRlZmF1bHRzIGJlaW5nIGFwcGxpZWQsIG9uSW5pdCBob29rcyBtdXRhdGluZyB0aGUgbW9kZWwpXG4gIGxldCBoYXNJbml0Q2hhbmdlcyA9IGZhbHNlXG5cbiAgY29uc3QgY3JlYXRlQm91bmRPYmplY3QgPSAoKSA9PiB7XG4gICAgLy8gU2V0IHVwIGEgdGVtcG9yYXJ5IGdsb2JhbCBsaXN0ZW5lciB0byBkZXRlY3QgaWYgYW55IGluaXQgY2hhbmdlcyBvY2N1ciBkdXJpbmcgZnJvbVNuYXBzaG90XG4gICAgY29uc3QgZGlzcG9zZUdsb2JhbExpc3RlbmVyID0gb25HbG9iYWxEZWVwQ2hhbmdlKChfdGFyZ2V0LCBjaGFuZ2UpID0+IHtcbiAgICAgIGlmIChjaGFuZ2UuaXNJbml0KSB7XG4gICAgICAgIGhhc0luaXRDaGFuZ2VzID0gdHJ1ZVxuICAgICAgfVxuICAgIH0pXG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgcmVzdWx0ID0geWpzQmluZGluZ0NvbnRleHQuYXBwbHkoXG4gICAgICAgICgpID0+IGZyb21TbmFwc2hvdChtb2J4S2V5c3RvbmVUeXBlLCB5anNKc29uIGFzIHVua25vd24gYXMgU25hcHNob3RJbk9mPFR5cGVUb0RhdGE8VFR5cGU+PiksXG4gICAgICAgIGJpbmRpbmdDb250ZXh0XG4gICAgICApXG4gICAgICB5anNCaW5kaW5nQ29udGV4dC5zZXQocmVzdWx0LCB7IC4uLmJpbmRpbmdDb250ZXh0LCBib3VuZE9iamVjdDogcmVzdWx0IH0pXG4gICAgICByZXR1cm4gcmVzdWx0XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIGRpc3Bvc2VHbG9iYWxMaXN0ZW5lcigpXG4gICAgfVxuICB9XG5cbiAgYm91bmRPYmplY3QgPSBjcmVhdGVCb3VuZE9iamVjdCgpXG5cbiAgLy8gYmluZCBhbnkgY2hhbmdlcyBmcm9tIHlqcyB0byBtb2J4LWtleXN0b25lXG4gIGNvbnN0IG9ic2VydmVEZWVwQ2IgPSBhY3Rpb24oKGV2ZW50czogWS5ZRXZlbnQ8YW55PltdKSA9PiB7XG4gICAgY29uc3QgZXZlbnRzVG9BcHBseTogWS5ZRXZlbnQ8YW55PltdID0gW11cblxuICAgIGV2ZW50cy5mb3JFYWNoKChldmVudCkgPT4ge1xuICAgICAgaWYgKGV2ZW50LnRyYW5zYWN0aW9uLm9yaWdpbiAhPT0geWpzT3JpZ2luKSB7XG4gICAgICAgIGV2ZW50c1RvQXBwbHkucHVzaChldmVudClcbiAgICAgIH1cblxuICAgICAgaWYgKGV2ZW50LnRhcmdldCBpbnN0YW5jZW9mIFkuTWFwIHx8IGV2ZW50LnRhcmdldCBpbnN0YW5jZW9mIFkuQXJyYXkpIHtcbiAgICAgICAgZ2V0WWpzQ29sbGVjdGlvbkF0b20oZXZlbnQudGFyZ2V0KT8ucmVwb3J0Q2hhbmdlZCgpXG4gICAgICB9XG4gICAgfSlcblxuICAgIGlmIChldmVudHNUb0FwcGx5Lmxlbmd0aCA+IDApIHtcbiAgICAgIGFwcGx5aW5nWWpzQ2hhbmdlc1RvTW9ieEtleXN0b25lKytcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IHJlY29uY2lsaWF0aW9uTWFwOiBSZWNvbmNpbGlhdGlvbk1hcCA9IG5ldyBNYXAoKVxuXG4gICAgICAgIC8vIENvbGxlY3QgaW5pdCBjaGFuZ2VzIHRoYXQgb2NjdXIgZHVyaW5nIGV2ZW50IGFwcGxpY2F0aW9uXG4gICAgICAgIC8vIChlLmcuLCBmcm9tU25hcHNob3QgY2FsbHMgdGhhdCB0cmlnZ2VyIG9uSW5pdCBob29rcylcbiAgICAgICAgLy8gV2Ugc3RvcmUgYm90aCB0YXJnZXQgYW5kIGNoYW5nZSBzbyB3ZSBjYW4gY29tcHV0ZSB0aGUgY29ycmVjdCBwYXRoIGxhdGVyXG4gICAgICAgIC8vIFNuYXBzaG90cyBhcmUgY2FwdHVyZWQgaW1tZWRpYXRlbHkgdG8gcHJlc2VydmUgdmFsdWVzIGF0IGluaXQgdGltZVxuICAgICAgICBjb25zdCBpbml0Q2hhbmdlczogeyB0YXJnZXQ6IG9iamVjdDsgY2hhbmdlOiBEZWVwQ2hhbmdlIH1bXSA9IFtdXG4gICAgICAgIGNvbnN0IGRpc3Bvc2VHbG9iYWxMaXN0ZW5lciA9IG9uR2xvYmFsRGVlcENoYW5nZSgodGFyZ2V0LCBjaGFuZ2UpID0+IHtcbiAgICAgICAgICBpZiAoY2hhbmdlLmlzSW5pdCkge1xuICAgICAgICAgICAgaW5pdENoYW5nZXMucHVzaCh7IHRhcmdldCwgY2hhbmdlOiBjYXB0dXJlQ2hhbmdlU25hcHNob3RzKGNoYW5nZSkgfSlcbiAgICAgICAgICB9XG4gICAgICAgIH0pXG5cbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBldmVudHNUb0FwcGx5LmZvckVhY2goKGV2ZW50KSA9PiB7XG4gICAgICAgICAgICBhcHBseVlqc0V2ZW50VG9Nb2J4KGV2ZW50LCBib3VuZE9iamVjdCwgcmVjb25jaWxpYXRpb25NYXApXG4gICAgICAgICAgfSlcbiAgICAgICAgfSBmaW5hbGx5IHtcbiAgICAgICAgICBkaXNwb3NlR2xvYmFsTGlzdGVuZXIoKVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gU3luYyBiYWNrIGFueSBpbml0LXRpbWUgbXV0YXRpb25zIGZyb20gZnJvbVNuYXBzaG90IGNhbGxzXG4gICAgICAgIC8vIChlLmcuLCBvbkluaXQgaG9va3MgdGhhdCBtb2RpZnkgdGhlIG1vZGVsKVxuICAgICAgICAvLyBUaGlzIGlzIG5lZWRlZCBiZWNhdXNlIGluaXQgY2hhbmdlcyBkdXJpbmcgWWpzIGV2ZW50IGhhbmRsaW5nIGFyZSBub3RcbiAgICAgICAgLy8gY2FwdHVyZWQgYnkgdGhlIG1haW4gb25EZWVwQ2hhbmdlIChpdCBza2lwcyBjaGFuZ2VzIHdoZW4gYXBwbHlpbmdZanNDaGFuZ2VzVG9Nb2J4S2V5c3RvbmUgPiAwKVxuICAgICAgICBpZiAoaW5pdENoYW5nZXMubGVuZ3RoID4gMCAmJiAhaXNZanNWYWx1ZURlbGV0ZWQoeWpzT2JqZWN0KSkge1xuICAgICAgICAgIHlqc0RvYy50cmFuc2FjdCgoKSA9PiB7XG4gICAgICAgICAgICBmb3IgKGNvbnN0IHsgdGFyZ2V0LCBjaGFuZ2UgfSBvZiBpbml0Q2hhbmdlcykge1xuICAgICAgICAgICAgICAvLyBDb21wdXRlIHRoZSBwYXRoIGZyb20gYm91bmRPYmplY3QgdG8gdGhlIHRhcmdldFxuICAgICAgICAgICAgICBjb25zdCBwYXRoVG9UYXJnZXQgPSBnZXRQYXJlbnRUb0NoaWxkUGF0aChib3VuZE9iamVjdCwgdGFyZ2V0KVxuICAgICAgICAgICAgICBpZiAocGF0aFRvVGFyZ2V0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAvLyBDcmVhdGUgYSBuZXcgY2hhbmdlIHdpdGggdGhlIGNvcnJlY3QgcGF0aCBmcm9tIHRoZSByb290XG4gICAgICAgICAgICAgICAgY29uc3QgY2hhbmdlV2l0aENvcnJlY3RQYXRoOiBEZWVwQ2hhbmdlID0ge1xuICAgICAgICAgICAgICAgICAgLi4uY2hhbmdlLFxuICAgICAgICAgICAgICAgICAgcGF0aDogWy4uLnBhdGhUb1RhcmdldCwgLi4uY2hhbmdlLnBhdGhdLFxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBhcHBseU1vYnhDaGFuZ2VUb1lqc09iamVjdChjaGFuZ2VXaXRoQ29ycmVjdFBhdGgsIHlqc09iamVjdClcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0sIHlqc09yaWdpbilcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFVwZGF0ZSBzbmFwc2hvdCB0cmFja2luZzogdGhlIFkuanMgY29udGFpbmVyIGlzIG5vdyBpbiBzeW5jIHdpdGggdGhlIGN1cnJlbnQgTW9iWCBzbmFwc2hvdFxuICAgICAgICAvLyBUaGlzIGVuYWJsZXMgdGhlIG1lcmdlIG9wdGltaXphdGlvbiB0byBza2lwIHVuY2hhbmdlZCBzdWJ0cmVlcyBkdXJpbmcgcmVjb25jaWxpYXRpb25cbiAgICAgICAgaWYgKHlqc09iamVjdCBpbnN0YW5jZW9mIFkuTWFwIHx8IHlqc09iamVjdCBpbnN0YW5jZW9mIFkuQXJyYXkpIHtcbiAgICAgICAgICBzZXRZanNDb250YWluZXJTbmFwc2hvdCh5anNPYmplY3QsIGdldFNuYXBzaG90KGJvdW5kT2JqZWN0KSlcbiAgICAgICAgfVxuICAgICAgfSBmaW5hbGx5IHtcbiAgICAgICAgYXBwbHlpbmdZanNDaGFuZ2VzVG9Nb2J4S2V5c3RvbmUtLVxuICAgICAgfVxuICAgIH1cbiAgfSlcblxuICB5anNPYmplY3Qub2JzZXJ2ZURlZXAob2JzZXJ2ZURlZXBDYilcblxuICAvLyBiaW5kIGFueSBjaGFuZ2VzIGZyb20gbW9ieC1rZXlzdG9uZSB0byB5anMgdXNpbmcgZGVlcCBjaGFuZ2Ugb2JzZXJ2YXRpb25cbiAgLy8gVGhpcyBwcm92aWRlcyBwcm9wZXIgc3BsaWNlIGRldGVjdGlvbiBmb3IgYXJyYXkgb3BlcmF0aW9uc1xuICBsZXQgcGVuZGluZ0NoYW5nZXM6IERlZXBDaGFuZ2VbXSA9IFtdXG5cbiAgY29uc3QgZGlzcG9zZU9uRGVlcENoYW5nZSA9IG9uRGVlcENoYW5nZShib3VuZE9iamVjdCwgKGNoYW5nZSkgPT4ge1xuICAgIGlmIChhcHBseWluZ1lqc0NoYW5nZXNUb01vYnhLZXlzdG9uZSA+IDApIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIC8vIFNraXAgaW5pdCBjaGFuZ2VzIC0gdGhleSBhcmUgaGFuZGxlZCBieSB0aGUgZ2V0U25hcHNob3QgKyBtZXJnZSBhdCB0aGUgZW5kIG9mIGJpbmRpbmdcbiAgICBpZiAoY2hhbmdlLmlzSW5pdCkge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgLy8gQ2FwdHVyZSBzbmFwc2hvdHMgbm93IGJlZm9yZSB0aGUgdmFsdWVzIGNhbiBiZSBtdXRhdGVkIHdpdGhpbiB0aGUgc2FtZSB0cmFuc2FjdGlvbi5cbiAgICAvLyBUaGlzIGlzIG5lY2Vzc2FyeSBiZWNhdXNlIGNoYW5nZXMgYXJlIGNvbGxlY3RlZCBhbmQgYXBwbGllZCBhZnRlciB0aGUgYWN0aW9uIGNvbXBsZXRlcyxcbiAgICAvLyBieSB3aGljaCB0aW1lIHRoZSBvcmlnaW5hbCB2YWx1ZXMgbWF5IGhhdmUgYmVlbiBtb2RpZmllZC5cbiAgICAvLyBFeGFtcGxlOiBgb2JqLml0ZW1zID0gW2EsIGJdOyBvYmouaXRlbXMuc3BsaWNlKDAsIDEpYCAtIHdpdGhvdXQgZWFybHkgY2FwdHVyZSxcbiAgICAvLyB0aGUgT2JqZWN0VXBkYXRlIGZvciBgaXRlbXNgIHdvdWxkIGdldCB0aGUgcG9zdC1zcGxpY2UgYXJyYXkgc3RhdGUuXG4gICAgcGVuZGluZ0NoYW5nZXMucHVzaChjYXB0dXJlQ2hhbmdlU25hcHNob3RzKGNoYW5nZSkpXG4gIH0pXG5cbiAgLy8gdGhpcyBpcyBvbmx5IHVzZWQgc28gd2UgY2FuIHRyYW5zYWN0IGFsbCBjaGFuZ2VzIHRvIHRoZSBzbmFwc2hvdCBib3VuZGFyeVxuICBjb25zdCBkaXNwb3NlT25TbmFwc2hvdCA9IG9uU25hcHNob3QoYm91bmRPYmplY3QsIChib3VuZE9iamVjdFNuYXBzaG90KSA9PiB7XG4gICAgaWYgKHBlbmRpbmdDaGFuZ2VzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgY29uc3QgY2hhbmdlc1RvQXBwbHkgPSBwZW5kaW5nQ2hhbmdlc1xuICAgIHBlbmRpbmdDaGFuZ2VzID0gW11cblxuICAgIC8vIFNraXAgc3luY2luZyB0byBZanMgaWYgdGhlIFlqcyBvYmplY3QgaGFzIGJlZW4gZGVsZXRlZC9kZXRhY2hlZFxuICAgIGlmIChpc1lqc1ZhbHVlRGVsZXRlZCh5anNPYmplY3QpKSB7XG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICB5anNEb2MudHJhbnNhY3QoKCkgPT4ge1xuICAgICAgY2hhbmdlc1RvQXBwbHkuZm9yRWFjaCgoY2hhbmdlKSA9PiB7XG4gICAgICAgIGFwcGx5TW9ieENoYW5nZVRvWWpzT2JqZWN0KGNoYW5nZSwgeWpzT2JqZWN0KVxuICAgICAgfSlcbiAgICB9LCB5anNPcmlnaW4pXG5cbiAgICAvLyBVcGRhdGUgc25hcHNob3QgdHJhY2tpbmc6IHRoZSBZLmpzIGNvbnRhaW5lciBpcyBub3cgaW4gc3luYyB3aXRoIHRoZSBjdXJyZW50IE1vYlggc25hcHNob3RcbiAgICBpZiAoeWpzT2JqZWN0IGluc3RhbmNlb2YgWS5NYXAgfHwgeWpzT2JqZWN0IGluc3RhbmNlb2YgWS5BcnJheSkge1xuICAgICAgc2V0WWpzQ29udGFpbmVyU25hcHNob3QoeWpzT2JqZWN0LCBib3VuZE9iamVjdFNuYXBzaG90KVxuICAgIH1cbiAgfSlcblxuICAvLyBTeW5jIHRoZSBpbml0IGNoYW5nZXMgdG8gdGhlIENSRFQuXG4gIC8vIEluaXQgY2hhbmdlcyBpbmNsdWRlOiBkZWZhdWx0cyBiZWluZyBhcHBsaWVkLCBvbkluaXQgaG9va3MgbXV0YXRpbmcgdGhlIG1vZGVsLlxuICAvLyBXZSB1c2UgZ2V0U25hcHNob3QgKyBtZXJnZSBiZWNhdXNlIHRoZSBwZXItY2hhbmdlIGFwcHJvYWNoIGhhcyBpc3N1ZXMgd2l0aCByZWZlcmVuY2UgbXV0YXRpb25cbiAgLy8gKHZhbHVlcyBjYXB0dXJlZCBpbiBEZWVwQ2hhbmdlIGNhbiBiZSBtdXRhdGVkIGJlZm9yZSB3ZSBhcHBseSB0aGVtKS5cbiAgLy8gVGhlIHNuYXBzaG90IHRyYWNraW5nIG9wdGltaXphdGlvbiBlbnN1cmVzIHVuY2hhbmdlZCBzdWJ0cmVlcyBhcmUgc2tpcHBlZCBkdXJpbmcgbWVyZ2UuXG4gIGNvbnN0IGZpbmFsU25hcHNob3QgPSBnZXRTbmFwc2hvdChib3VuZE9iamVjdClcblxuICBpZiAoaGFzSW5pdENoYW5nZXMpIHtcbiAgICB5anNEb2MudHJhbnNhY3QoKCkgPT4ge1xuICAgICAgaWYgKHlqc09iamVjdCBpbnN0YW5jZW9mIFkuTWFwKSB7XG4gICAgICAgIGFwcGx5SnNvbk9iamVjdFRvWU1hcCh5anNPYmplY3QsIGZpbmFsU25hcHNob3QgYXMgdW5rbm93biBhcyBQbGFpbk9iamVjdCwge1xuICAgICAgICAgIG1vZGU6IFwibWVyZ2VcIixcbiAgICAgICAgfSlcbiAgICAgIH0gZWxzZSBpZiAoeWpzT2JqZWN0IGluc3RhbmNlb2YgWS5BcnJheSkge1xuICAgICAgICBhcHBseUpzb25BcnJheVRvWUFycmF5KHlqc09iamVjdCwgZmluYWxTbmFwc2hvdCBhcyB1bmtub3duIGFzIFBsYWluQXJyYXksIHtcbiAgICAgICAgICBtb2RlOiBcIm1lcmdlXCIsXG4gICAgICAgIH0pXG4gICAgICB9XG4gICAgfSwgeWpzT3JpZ2luKVxuICB9XG5cbiAgLy8gQWx3YXlzIHVwZGF0ZSBzbmFwc2hvdCB0cmFja2luZyBhZnRlciBiaW5kaW5nIGluaXRpYWxpemF0aW9uXG4gIC8vIFRoaXMgZW5zdXJlcyB0aGUgbWVyZ2Ugb3B0aW1pemF0aW9uIGNhbiBza2lwIHVuY2hhbmdlZCBzdWJ0cmVlcyBpbiBmdXR1cmUgcmVjb25jaWxpYXRpb25zXG4gIGlmICh5anNPYmplY3QgaW5zdGFuY2VvZiBZLk1hcCB8fCB5anNPYmplY3QgaW5zdGFuY2VvZiBZLkFycmF5KSB7XG4gICAgc2V0WWpzQ29udGFpbmVyU25hcHNob3QoeWpzT2JqZWN0LCBmaW5hbFNuYXBzaG90KVxuICB9XG5cbiAgY29uc3QgZGlzcG9zZSA9ICgpID0+IHtcbiAgICB5anNEb2Mub2ZmKFwiZGVzdHJveVwiLCBkaXNwb3NlKVxuICAgIGRpc3Bvc2VPbkRlZXBDaGFuZ2UoKVxuICAgIGRpc3Bvc2VPblNuYXBzaG90KClcbiAgICB5anNPYmplY3QudW5vYnNlcnZlRGVlcChvYnNlcnZlRGVlcENiKVxuICB9XG5cbiAgeWpzRG9jLm9uKFwiZGVzdHJveVwiLCBkaXNwb3NlKVxuXG4gIHJldHVybiB7XG4gICAgYm91bmRPYmplY3QsXG4gICAgZGlzcG9zZSxcbiAgICB5anNPcmlnaW4sXG4gIH1cbn1cbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFHTyxNQUFNLDZCQUE2QixNQUFNO0FBQUEsRUFDOUMsWUFBWSxLQUFhO0FBQ3ZCLFVBQU0sR0FBRztBQUdULFdBQU8sZUFBZSxNQUFNLHFCQUFxQixTQUFTO0FBQUEsRUFDNUQ7QUFDRjtBQUtPLFNBQVMsUUFBUSxLQUFhO0FBQ25DLFNBQU8sSUFBSSxxQkFBcUIsR0FBRztBQUNyQztBQ2RBLE1BQU0seUNBQXlCLFFBQUE7QUFLeEIsTUFBTSx1QkFBdUIsQ0FDbEMsa0JBQ3NCO0FBQ3RCLFNBQU8s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