@datagrok-libraries/bio 5.3.0 → 5.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,1095 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ /* Do not change these import lines to match external modules in webpack configuration */
11
+ import * as grok from 'datagrok-api/grok';
12
+ import * as DG from 'datagrok-api/dg';
13
+ import { getSplitter } from '../../index';
14
+ import { HELM_CORE_FIELDS } from './const';
15
+ import { NotationConverter } from './notation-converter';
16
+ // constants for parsing molfile V2000
17
+ const V2K_RGP_SHIFT = 8;
18
+ const V2K_RGP_LINE = 'M RGP';
19
+ const V2K_A_LINE = 'A ';
20
+ // constants for parsing/reconstruction of molfile V3000
21
+ const V3K_COUNTS_SHIFT = 14;
22
+ const V3K_IDX_SHIFT = 7;
23
+ const V3K_HEADER_FIRST_LINE = '\nDatagrok macromolecule handler\n\n';
24
+ const V3K_HEADER_SECOND_LINE = ' 0 0 0 0 0 0 999 V3000\n';
25
+ const V3K_BEGIN_CTAB_BLOCK = 'M V30 BEGIN CTAB\n';
26
+ const V3K_END_CTAB_BLOCK = 'M V30 END CTAB\n';
27
+ const V3K_BEGIN_COUNTS_LINE = 'M V30 COUNTS ';
28
+ const V3K_COUNTS_LINE_ENDING = ' 0 0 0\n';
29
+ const V3K_BEGIN_ATOM_BLOCK = 'M V30 BEGIN ATOM\n';
30
+ const V3K_END_ATOM_BLOCK = 'M V30 END ATOM\n';
31
+ const V3K_BEGIN_BOND_BLOCK = 'M V30 BEGIN BOND\n';
32
+ const V3K_END_BOND_BLOCK = 'M V30 END BOND\n';
33
+ const V3K_BOND_CONFIG = ' CFG=';
34
+ const V3K_BEGIN_DATA_LINE = 'M V30 ';
35
+ const V3K_END = 'M END\n';
36
+ const PRECISION_FACTOR = 10000; // HELMCoreLibrary has 4 significant digits after decimal point in atom coordinates
37
+ // symbols for the corresponding monomers in HELM library
38
+ const DEOXYRIBOSE = 'd';
39
+ const RIBOSE = 'r';
40
+ const PHOSPHATE = 'p';
41
+ const OXYGEN = 'O';
42
+ const HYDROGEN = 'H';
43
+ // todo: verify that all functions have return types
44
+ /* Convert Macromolecule column into Molecule column storing molfile V3000 with the help of a monomer library */
45
+ export function _toAtomicLevel(df, macroMolCol, monomersLibList) {
46
+ return __awaiter(this, void 0, void 0, function* () {
47
+ if (DG.Func.find({ package: 'Chem', name: 'getRdKitModule' }).length === 0) {
48
+ grok.shell.warning('Transformation to atomic level requires package "Chem" installed.');
49
+ return;
50
+ }
51
+ if (macroMolCol.semType !== DG.SEMTYPE.MACROMOLECULE) {
52
+ grok.shell.warning(`Only the ${DG.SEMTYPE.MACROMOLECULE} columns can be converted to atomic
53
+ level, the chosen column has semType ${macroMolCol.semType}`);
54
+ return;
55
+ }
56
+ // convert 'helm' to 'separator' units
57
+ if (macroMolCol.getTag(DG.TAGS.UNITS) === "helm" /* NOTATION.HELM */) {
58
+ const converter = new NotationConverter(macroMolCol);
59
+ const separator = '/';
60
+ macroMolCol = converter.convert("separator" /* NOTATION.SEPARATOR */, separator);
61
+ }
62
+ const alphabet = macroMolCol.getTag("alphabet" /* TAGS.alphabet */);
63
+ // determine the polymer type according to HELM specifications
64
+ let polymerType;
65
+ // todo: an exception from dart comes before this check if the alphabet is UN
66
+ if (alphabet === "PT" /* ALPHABET.PT */) {
67
+ polymerType = "PEPTIDE" /* HELM_POLYMER_TYPE.PEPTIDE */;
68
+ }
69
+ else if (alphabet === "RNA" /* ALPHABET.RNA */ || alphabet === "DNA" /* ALPHABET.DNA */) {
70
+ polymerType = "RNA" /* HELM_POLYMER_TYPE.RNA */;
71
+ }
72
+ else {
73
+ grok.shell.warning(`Only PT, DNA and RNA alphabets are supported, while the selected column has ${polymerType} alphabet`);
74
+ return;
75
+ }
76
+ const monomerSequencesArray = getMonomerSequencesArray(macroMolCol);
77
+ // todo: consider separately backbone, terminal, branch monomer types
78
+ const monomersDict = yield getMonomersDict(monomerSequencesArray, monomersLibList, polymerType, alphabet);
79
+ const columnLength = macroMolCol.length;
80
+ const reconstructed = new Array(columnLength);
81
+ for (let row = 0; row < columnLength; ++row) {
82
+ const monomerSeq = monomerSequencesArray[row];
83
+ reconstructed[row] = monomerSeqToMolfile(monomerSeq, monomersDict, alphabet, polymerType);
84
+ console.log(reconstructed[row]);
85
+ }
86
+ // exclude name collisions
87
+ const name = 'molfile(' + macroMolCol.name + ')';
88
+ const newColName = df.columns.getUnusedName(name);
89
+ const newCol = DG.Column.fromStrings(newColName, reconstructed);
90
+ newCol.semType = DG.SEMTYPE.MOLECULE;
91
+ newCol.setTag(DG.TAGS.UNITS, DG.UNITS.Molecule.MOLBLOCK);
92
+ df.columns.add(newCol, true);
93
+ yield grok.data.detectSemanticTypes(df);
94
+ });
95
+ }
96
+ /* Get a mapping of peptide symbols to HELM monomer library objects with
97
+ * selectted fields */
98
+ function getFormattedMonomerLib(monomersLibList, polymerType, alphabet) {
99
+ const map = new Map();
100
+ monomersLibList.forEach((it) => {
101
+ if (it["polymerType" /* HELM_FIELDS.POLYMER_TYPE */] === polymerType) {
102
+ if (polymerType === "RNA" /* HELM_POLYMER_TYPE.RNA */ &&
103
+ (it["monomerType" /* HELM_FIELDS.MONOMER_TYPE */] === "Branch" /* HELM_MONOMER_TYPE.BRANCH */ ||
104
+ alphabet === "DNA" /* ALPHABET.DNA */ && it["symbol" /* HELM_FIELDS.SYMBOL */] === DEOXYRIBOSE ||
105
+ alphabet === "RNA" /* ALPHABET.RNA */ && it["symbol" /* HELM_FIELDS.SYMBOL */] === RIBOSE ||
106
+ it["symbol" /* HELM_FIELDS.SYMBOL */] === PHOSPHATE) ||
107
+ polymerType === "PEPTIDE" /* HELM_POLYMER_TYPE.PEPTIDE */ &&
108
+ it["monomerType" /* HELM_FIELDS.MONOMER_TYPE */] !== "Branch" /* HELM_MONOMER_TYPE.BRANCH */) {
109
+ const monomerObject = {};
110
+ HELM_CORE_FIELDS.forEach((field) => {
111
+ monomerObject[field] = it[field];
112
+ });
113
+ map.set(it["symbol" /* HELM_FIELDS.SYMBOL */], monomerObject);
114
+ }
115
+ }
116
+ });
117
+ return map;
118
+ }
119
+ /* Get jagged array of monomer symbols for the dataframe */
120
+ function getMonomerSequencesArray(macroMolCol) {
121
+ const columnLength = macroMolCol.length;
122
+ const result = new Array(columnLength);
123
+ // split the string into monomers
124
+ const colUnits = macroMolCol.getTag(DG.TAGS.UNITS);
125
+ const separator = macroMolCol.getTag("separator" /* TAGS.separator */);
126
+ const splitterFunc = getSplitter(colUnits, separator);
127
+ for (let row = 0; row < columnLength; ++row) {
128
+ const macroMolecule = macroMolCol.get(row);
129
+ // todo: handle the exception case when macroMolecule is null
130
+ result[row] = macroMolecule ? splitterFunc(macroMolecule) : [];
131
+ }
132
+ return result;
133
+ }
134
+ /* Get a mapping of monomer symbols to MolGraph objects */
135
+ function getMonomersDict(monomerSequencesArray, monomersLibList, polymerType, alphabet) {
136
+ return __awaiter(this, void 0, void 0, function* () {
137
+ // todo: exception - no gaps, no empty string monomers
138
+ const formattedMonomerLib = getFormattedMonomerLib(monomersLibList, polymerType, alphabet);
139
+ const monomersDict = new Map();
140
+ const moduleRdkit = yield grok.functions.call('Chem:getRdKitModule');
141
+ // add deoxyribose/ribose and phosphate for nucleotide sequences
142
+ if (polymerType === "RNA" /* HELM_POLYMER_TYPE.RNA */) {
143
+ const symbols = (alphabet === "RNA" /* ALPHABET.RNA */) ?
144
+ [RIBOSE, PHOSPHATE] : [DEOXYRIBOSE, PHOSPHATE];
145
+ for (const sym of symbols)
146
+ updateMonomersDict(monomersDict, sym, formattedMonomerLib, moduleRdkit, polymerType);
147
+ }
148
+ for (let row = 0; row < monomerSequencesArray.length; ++row) {
149
+ const monomerSeq = monomerSequencesArray[row];
150
+ for (const sym of monomerSeq)
151
+ updateMonomersDict(monomersDict, sym, formattedMonomerLib, moduleRdkit, polymerType);
152
+ }
153
+ // console.log(monomersDict);
154
+ return monomersDict;
155
+ });
156
+ }
157
+ /* Adds MolGraph object for 'sym' to the monomers dict when necessary */
158
+ function updateMonomersDict(monomersDict, sym, formattedMonomerLib, moduleRdkit, polymerType) {
159
+ if (!monomersDict.has(sym)) {
160
+ const monomerData = getMolGraph(sym, formattedMonomerLib, moduleRdkit, polymerType);
161
+ if (monomerData)
162
+ monomersDict.set(sym, monomerData);
163
+ else
164
+ throw new Error(`Monomer with symbol '${sym}' is absent the monomer library`);
165
+ // todo: handle exception
166
+ }
167
+ }
168
+ /* Construct the MolGraph object for specified monomerSymbol: the associated
169
+ * graph is adjusted in XY plane and filled with default R-groups */
170
+ function getMolGraph(monomerSymbol, formattedMonomerLib, moduleRdkit, polymerType // todo: specify type for moduleRdkit
171
+ ) {
172
+ if (!formattedMonomerLib.has(monomerSymbol)) {
173
+ return null;
174
+ }
175
+ else {
176
+ const libObject = formattedMonomerLib.get(monomerSymbol);
177
+ const capGroups = parseCapGroups(libObject["rgroups" /* HELM_FIELDS.RGROUPS */]);
178
+ const capGroupIdxMap = parseCapGroupIdxMap(libObject["molfile" /* HELM_FIELDS.MOLFILE */]);
179
+ const molfileV3K = convertMolfileToV3K(removeRGroupLines(libObject["molfile" /* HELM_FIELDS.MOLFILE */]), moduleRdkit);
180
+ const counts = parseAtomAndBondCounts(molfileV3K);
181
+ const atoms = parseAtomBlock(molfileV3K, counts.atomCount);
182
+ const bonds = parseBondBlock(molfileV3K, counts.bondCount);
183
+ const meta = getMonomerMetadata(atoms, bonds, capGroups, capGroupIdxMap);
184
+ const monomerGraph = { atoms: atoms, bonds: bonds, meta: meta };
185
+ if (polymerType === "PEPTIDE" /* HELM_POLYMER_TYPE.PEPTIDE */) {
186
+ adjustPeptideMonomerGraph(monomerGraph);
187
+ }
188
+ else { // nucleotides
189
+ if (monomerSymbol === RIBOSE || monomerSymbol === DEOXYRIBOSE)
190
+ adjustSugarMonomerGraph(monomerGraph);
191
+ else if (monomerSymbol === PHOSPHATE)
192
+ adjustPhosphateMonomerGraph(monomerGraph);
193
+ else
194
+ adjustBaseMonomerGraph(monomerGraph);
195
+ }
196
+ // remove the 'rightmost' chain-extending r-group node in the backbone
197
+ if (polymerType === "PEPTIDE" /* HELM_POLYMER_TYPE.PEPTIDE */) {
198
+ setShifts(monomerGraph, polymerType);
199
+ removeNodeAndBonds(monomerGraph, monomerGraph.meta.rNodes[1]);
200
+ }
201
+ else { // nucleotides
202
+ if (monomerSymbol === RIBOSE || monomerSymbol === DEOXYRIBOSE) {
203
+ // remove R2
204
+ removeNodeAndBonds(monomerGraph, monomerGraph.meta.rNodes[1]);
205
+ // set terminalNode2 (oxygen) as new R2
206
+ monomerGraph.meta.rNodes[1] = monomerGraph.meta.terminalNodes[1];
207
+ setTerminalNodes(monomerGraph.bonds, monomerGraph.meta); // set terminal nodes anew
208
+ setShifts(monomerGraph, polymerType);
209
+ // remove 'new' R2 (oxygen)
210
+ removeNodeAndBonds(monomerGraph, monomerGraph.meta.rNodes[1]);
211
+ // remove R1
212
+ removeNodeAndBonds(monomerGraph, monomerGraph.meta.rNodes[0]);
213
+ // remove the branching r-group
214
+ removeNodeAndBonds(monomerGraph, monomerGraph.meta.rNodes[2]);
215
+ }
216
+ else if (monomerSymbol === PHOSPHATE) {
217
+ monomerGraph.meta.terminalNodes[0] = monomerGraph.meta.rNodes[0];
218
+ shiftCoordinates(monomerGraph, -monomerGraph.atoms.x[monomerGraph.meta.terminalNodes[0] - 1], -monomerGraph.atoms.y[monomerGraph.meta.terminalNodes[0] - 1]);
219
+ setShifts(monomerGraph, polymerType);
220
+ removeNodeAndBonds(monomerGraph, monomerGraph.meta.rNodes[1]);
221
+ }
222
+ else { // nucleobases
223
+ removeNodeAndBonds(monomerGraph, monomerGraph.meta.rNodes[0]);
224
+ }
225
+ }
226
+ removeHydrogen(monomerGraph);
227
+ return monomerGraph;
228
+ }
229
+ }
230
+ // todo: doc
231
+ function getMonomerMetadata(atoms, bonds, capGroups, capGroupIdxMap) {
232
+ const meta = {
233
+ backboneShift: null,
234
+ branchShift: null,
235
+ terminalNodes: [],
236
+ rNodes: [],
237
+ };
238
+ substituteCapGroups(atoms, capGroups, capGroupIdxMap);
239
+ setRNodes(capGroupIdxMap, meta);
240
+ setTerminalNodes(bonds, meta);
241
+ return meta;
242
+ }
243
+ /* Parse element symbols for R-groups from the HELM monomer library R-groups
244
+ * field */
245
+ function parseCapGroups(rGroupObjList) {
246
+ // specifically for HELMCoreLibrary
247
+ // considered only monoatomic rgroups
248
+ // supposing that elements in rGroupObjList are sorted w.r.t. the rgroups idx
249
+ // todo: possible generalizations
250
+ const capGroupsArray = [];
251
+ for (const obj of rGroupObjList) {
252
+ let capGroup = obj["capGroupSmiles" /* RGROUP_FIELDS.CAP_GROUP_SMILES */];
253
+ // in some cases the smiles field is written with uppercase
254
+ if (!capGroup)
255
+ capGroup = obj["capGroupSMILES" /* RGROUP_FIELDS.CAP_GROUP_SMILES_UPPERCASE */];
256
+ // todo: verify that there are no multi-element cap groups, or consider how to
257
+ // transform them
258
+ capGroup = capGroup.replace(/(\[|\]|\*|:|\d)/g, '');
259
+ if (capGroup.length > 1) // todo: check if such cases are possible, remove if not
260
+ throw new Error('Default cap group has length more than one');
261
+ capGroupsArray.push(capGroup);
262
+ }
263
+ return capGroupsArray;
264
+ }
265
+ /* Substitute the cap group elements instead of R# */
266
+ function substituteCapGroups(atoms, capGroups, capGroupIdxMap) {
267
+ for (const [node, capIdx] of capGroupIdxMap)
268
+ atoms.atomTypes[node - 1] = capGroups[capIdx - 1]; // -1 because molfile indexing starts from 1
269
+ }
270
+ //todo: doc
271
+ function setRNodes(capGroupIdxMap, meta) {
272
+ meta.rNodes = Array.from(capGroupIdxMap.keys());
273
+ for (let i = 0; i < meta.rNodes.length; i++) {
274
+ for (const j of [1, 2]) { // 1 and 2 by def. correspond to 'left/rightmost' r-nodes
275
+ // swap the values if necessary, so that the "leftmost" r-node is at 0,
276
+ // and the 'rightmost', at 1
277
+ if (capGroupIdxMap.get(meta.rNodes[i]) === j) {
278
+ const tmp = meta.rNodes[j - 1];
279
+ meta.rNodes[j - 1] = meta.rNodes[i];
280
+ meta.rNodes[i] = tmp;
281
+ }
282
+ }
283
+ }
284
+ }
285
+ //todo: doc
286
+ function setTerminalNodes(bonds, meta) {
287
+ const rNodes = meta.rNodes;
288
+ meta.terminalNodes = new Array(rNodes.length).fill(0);
289
+ const terminalNodes = meta.terminalNodes;
290
+ const atomPairs = bonds.atomPairs;
291
+ let i = 0;
292
+ let j = 0;
293
+ while ((i < atomPairs.length) && j < terminalNodes.length) {
294
+ // rNodes array is sorted so that its 0th and 1st elements (if both
295
+ // present) correspond to the chain extending (i.e. not branching) r-groups
296
+ for (let k = 0; k < terminalNodes.length; ++k) {
297
+ for (let l = 0; l < 2; ++l) {
298
+ if (atomPairs[i][l] === rNodes[k]) {
299
+ terminalNodes[k] = atomPairs[i][(l + 1) % 2];
300
+ if (rNodes.length > 2) {
301
+ }
302
+ ++j;
303
+ }
304
+ }
305
+ }
306
+ ++i;
307
+ }
308
+ }
309
+ //todo: doc
310
+ function setShifts(molGraph, polymerType) {
311
+ if (molGraph.meta.rNodes.length > 1) {
312
+ molGraph.meta.backboneShift = [
313
+ keepPrecision(molGraph.atoms.x[molGraph.meta.rNodes[1] - 1] -
314
+ molGraph.atoms.x[molGraph.meta.terminalNodes[0] - 1]),
315
+ keepPrecision(molGraph.atoms.y[molGraph.meta.rNodes[1] - 1] -
316
+ molGraph.atoms.y[molGraph.meta.terminalNodes[0] - 1]),
317
+ ];
318
+ }
319
+ if (polymerType === "RNA" /* HELM_POLYMER_TYPE.RNA */ && molGraph.meta.rNodes.length > 2) {
320
+ molGraph.meta.branchShift = [
321
+ keepPrecision(molGraph.atoms.x[molGraph.meta.rNodes[2] - 1] -
322
+ molGraph.atoms.x[molGraph.meta.terminalNodes[0] - 1]),
323
+ keepPrecision(molGraph.atoms.y[molGraph.meta.rNodes[2] - 1] -
324
+ molGraph.atoms.y[molGraph.meta.terminalNodes[0] - 1]),
325
+ ];
326
+ }
327
+ }
328
+ /* Helper function necessary to build a correct V3000 molfile out of V2000 with
329
+ * specified r-groups*/
330
+ function removeRGroupLines(molfileV2K) {
331
+ let begin = molfileV2K.indexOf(V2K_A_LINE, 0);
332
+ if (begin === -1)
333
+ begin = molfileV2K.indexOf(V2K_RGP_LINE);
334
+ const end = molfileV2K.indexOf(V3K_END, begin);
335
+ return molfileV2K.substring(0, begin) + molfileV2K.substring(end);
336
+ }
337
+ /* V2000 to V3000 converter */
338
+ function convertMolfileToV3K(molfileV2K, moduleRdkit) {
339
+ // todo: type of moduleRdkit
340
+ // todo: consider the use of standard Chem converter (relies on creation of moduleRdkit on each iteration, though)
341
+ const molObj = moduleRdkit.get_mol(molfileV2K);
342
+ const molfileV3K = molObj.get_v3Kmolblock();
343
+ molObj.delete();
344
+ return molfileV3K;
345
+ }
346
+ /* Parse V3000 bond block and construct the Bonds object */
347
+ function parseBondBlock(molfileV3K, bondCount) {
348
+ // todo: consider the case when there is no simple leftmost/rightmost choice
349
+ // todo: consider the case when there are multiple consequent M RGP lines,
350
+ // like in HELMCoreLibrary nucleotides
351
+ const bondTypes = new Array(bondCount);
352
+ const atomPairs = new Array(bondCount);
353
+ const bondConfiguration = new Map();
354
+ const kwargs = new Map;
355
+ let begin = molfileV3K.indexOf(V3K_BEGIN_BOND_BLOCK);
356
+ begin = molfileV3K.indexOf('\n', begin);
357
+ let end = begin;
358
+ for (let i = 0; i < bondCount; ++i) {
359
+ // parse bond type and atom pair
360
+ const parsedValues = new Array(3);
361
+ begin = molfileV3K.indexOf(V3K_BEGIN_DATA_LINE, end) + V3K_IDX_SHIFT;
362
+ end = molfileV3K.indexOf(' ', begin);
363
+ for (let k = 0; k < 3; ++k) {
364
+ begin = end + 1;
365
+ end = Math.min(molfileV3K.indexOf('\n', begin), molfileV3K.indexOf(' ', begin));
366
+ parsedValues[k] = parseInt(molfileV3K.slice(begin, end));
367
+ }
368
+ bondTypes[i] = parsedValues[0];
369
+ atomPairs[i] = parsedValues.slice(1);
370
+ // parse keyword arguments
371
+ const endOfLine = molfileV3K.indexOf('\n', begin);
372
+ let lineRemainder = molfileV3K.slice(end, endOfLine);
373
+ let beginCfg = lineRemainder.indexOf(V3K_BOND_CONFIG);
374
+ if (beginCfg !== -1) {
375
+ beginCfg = lineRemainder.indexOf('=', beginCfg) + 1;
376
+ let endCfg = lineRemainder.indexOf(' ', beginCfg);
377
+ if (endCfg === -1)
378
+ endCfg = lineRemainder.length;
379
+ const bondConfig = parseInt(lineRemainder.slice(beginCfg, endCfg));
380
+ bondConfiguration.set(i, bondConfig);
381
+ const removedSubstring = V3K_BOND_CONFIG + bondConfig.toString();
382
+ lineRemainder = lineRemainder.replace(removedSubstring, '');
383
+ }
384
+ if (!lineRemainder)
385
+ kwargs.set(i, lineRemainder);
386
+ }
387
+ return {
388
+ bondTypes: bondTypes,
389
+ atomPairs: atomPairs,
390
+ bondConfiguration: bondConfiguration,
391
+ kwargs: kwargs,
392
+ };
393
+ }
394
+ /* Constructs mapping of r-group nodes to default capGroups, all numeration starting from 1.
395
+ * According to https://pubs.acs.org/doi/10.1021/ci3001925, R1 and R2 are the chain extending attachment points,
396
+ * while R3 is the branching attachment point. */
397
+ function parseCapGroupIdxMap(molfileV2K) {
398
+ const capGroupIdxMap = new Map();
399
+ // parse A-lines (RNA)
400
+ let begin = molfileV2K.indexOf(V2K_A_LINE, 0);
401
+ let end = begin;
402
+ while (begin !== -1) {
403
+ // parse the rNode to which the cap group is attached
404
+ end = molfileV2K.indexOf('\n', begin);
405
+ const rNode = parseInt(molfileV2K.substring(begin, end).replace(/^A\s+/, ''));
406
+ // parse the capGroup index
407
+ begin = molfileV2K.indexOf('R', end);
408
+ end = molfileV2K.indexOf('\n', begin);
409
+ const capGroup = parseInt(molfileV2K.substring(begin, end).replace(/^R/, ''));
410
+ capGroupIdxMap.set(rNode, capGroup);
411
+ begin = molfileV2K.indexOf(V2K_A_LINE, end);
412
+ }
413
+ // parse RGP lines (may be more than one in RNA monomers)
414
+ begin = molfileV2K.indexOf(V2K_RGP_LINE, 0);
415
+ end = molfileV2K.indexOf('\n', begin);
416
+ while (begin !== -1) {
417
+ begin += V2K_RGP_SHIFT;
418
+ end = molfileV2K.indexOf('\n', begin);
419
+ const rgpStringParsed = molfileV2K.substring(begin, end)
420
+ .replaceAll(/\s+/g, ' ')
421
+ .split(' ');
422
+ const rgpIndicesArray = rgpStringParsed.map((el) => parseInt(el))
423
+ .slice(1); // slice from 1 because the 1st value is the number of pairs in the line
424
+ for (let i = 0; i < rgpIndicesArray.length; i += 2) {
425
+ // notice: there may be conflicting cap group definitions, like 3-O-Methylribose (2,5 connectivity)
426
+ // (the last monomer in HELMCoreLibrary)
427
+ // there the indices of cap groups are self-contradictory
428
+ // todo: clarify why such situations occur in principle
429
+ if (capGroupIdxMap.has(rgpIndicesArray[i]) && capGroupIdxMap.get(rgpIndicesArray[i]) !== rgpIndicesArray[i + 1])
430
+ throw new Error(`r-group index ${rgpIndicesArray[i]} has already been added with a different value`);
431
+ else
432
+ capGroupIdxMap.set(rgpIndicesArray[i], rgpIndicesArray[i + 1]);
433
+ }
434
+ begin = molfileV2K.indexOf(V2K_RGP_LINE, end);
435
+ }
436
+ return capGroupIdxMap;
437
+ }
438
+ function parseAtomAndBondCounts(molfileV3K) {
439
+ molfileV3K = molfileV3K.replaceAll('\r', ''); // to handle old and new sdf standards
440
+ // parse atom count
441
+ let begin = molfileV3K.indexOf(V3K_BEGIN_COUNTS_LINE) + V3K_COUNTS_SHIFT;
442
+ let end = molfileV3K.indexOf(' ', begin);
443
+ const numOfAtoms = parseInt(molfileV3K.substring(begin, end));
444
+ // parse bond count
445
+ begin = end + 1;
446
+ end = molfileV3K.indexOf(' ', begin);
447
+ const numOfBonds = parseInt(molfileV3K.substring(begin, end));
448
+ return { atomCount: numOfAtoms, bondCount: numOfBonds };
449
+ }
450
+ /* Parse V3000 atom block and return Atoms object. NOTICE: only atomTypes, x, y
451
+ * and kwargs fields are set in the return value, with other fields dummy */
452
+ function parseAtomBlock(molfileV3K, atomCount) {
453
+ const atomTypes = new Array(atomCount);
454
+ const x = new Array(atomCount);
455
+ const y = new Array(atomCount);
456
+ const kwargs = new Array(atomCount);
457
+ let begin = molfileV3K.indexOf(V3K_BEGIN_ATOM_BLOCK); // V3000 atoms block
458
+ begin = molfileV3K.indexOf('\n', begin);
459
+ let end = begin;
460
+ for (let i = 0; i < atomCount; i++) {
461
+ begin = molfileV3K.indexOf(V3K_BEGIN_DATA_LINE, begin) + V3K_IDX_SHIFT;
462
+ end = molfileV3K.indexOf(' ', begin); // skip the idx row
463
+ // parse atom type
464
+ begin = end + 1;
465
+ end = molfileV3K.indexOf(' ', begin);
466
+ atomTypes[i] = molfileV3K.substring(begin, end);
467
+ // parse X and Y coordinates of the atom
468
+ const coordinate = new Array(2);
469
+ for (let k = 0; k < 2; ++k) {
470
+ begin = end + 1;
471
+ end = molfileV3K.indexOf(' ', begin);
472
+ coordinate[k] = parseFloat(molfileV3K.substring(begin, end));
473
+ }
474
+ x[i] = coordinate[0];
475
+ y[i] = coordinate[1];
476
+ // parse the remaining possible keyword arguments
477
+ begin = end;
478
+ end = molfileV3K.indexOf('\n', begin) + 1;
479
+ kwargs[i] = molfileV3K.slice(begin, end);
480
+ begin = end;
481
+ }
482
+ return {
483
+ atomTypes: atomTypes,
484
+ x: x,
485
+ y: y,
486
+ kwargs: kwargs,
487
+ };
488
+ }
489
+ /* Remove hydrogen nodes */
490
+ function removeHydrogen(monomerGraph) {
491
+ let i = 0;
492
+ while (i < monomerGraph.atoms.atomTypes.length) {
493
+ if (monomerGraph.atoms.atomTypes[i] === HYDROGEN) {
494
+ removeNodeAndBonds(monomerGraph, i + 1); // i + 1 because molfile node indexing starts from 1
495
+ --i;
496
+ // monomerGraph.atoms.atomTypes[i] = 'Li';
497
+ }
498
+ ++i;
499
+ }
500
+ }
501
+ /* Remove node 'removedNode' and the associated bonds. Notice, numeration of
502
+ * nodes in molfiles starts from 1, not 0 */
503
+ function removeNodeAndBonds(monomerGraph, removedNode) {
504
+ if (typeof removedNode !== 'undefined') {
505
+ const removedNodeIdx = removedNode - 1;
506
+ const atoms = monomerGraph.atoms;
507
+ const bonds = monomerGraph.bonds;
508
+ const meta = monomerGraph.meta;
509
+ // remove the node from atoms
510
+ atoms.atomTypes.splice(removedNodeIdx, 1);
511
+ atoms.x.splice(removedNodeIdx, 1);
512
+ atoms.y.splice(removedNodeIdx, 1);
513
+ atoms.kwargs.splice(removedNodeIdx, 1);
514
+ // update the values of terminal and r-group nodes if necessary
515
+ for (let i = 0; i < meta.terminalNodes.length; ++i) {
516
+ if (meta.terminalNodes[i] > removedNode)
517
+ --meta.terminalNodes[i];
518
+ else if (meta.terminalNodes[i] === removedNode)
519
+ meta.terminalNodes[i] = -1; // sentinel to mark the value as removed
520
+ }
521
+ for (let i = 0; i < meta.rNodes.length; ++i) {
522
+ if (meta.rNodes[i] > removedNode)
523
+ --meta.rNodes[i];
524
+ else if (meta.rNodes[i] === removedNode)
525
+ meta.rNodes[i] = -1; // sentinel to mark the value as removed
526
+ }
527
+ // update indices of atoms in bonds
528
+ let i = 0;
529
+ while (i < bonds.atomPairs.length) {
530
+ const firstAtom = bonds.atomPairs[i][0];
531
+ const secondAtom = bonds.atomPairs[i][1];
532
+ if (firstAtom === removedNode || secondAtom === removedNode) {
533
+ bonds.atomPairs.splice(i, 1);
534
+ bonds.bondTypes.splice(i, 1);
535
+ if (bonds.bondConfiguration.has(i))
536
+ bonds.bondConfiguration.delete(i);
537
+ if (bonds.kwargs.has(i))
538
+ bonds.kwargs.delete(i);
539
+ --i;
540
+ }
541
+ else {
542
+ bonds.atomPairs[i][0] = (firstAtom > removedNode) ? firstAtom - 1 : firstAtom;
543
+ bonds.atomPairs[i][1] = (secondAtom > removedNode) ? secondAtom - 1 : secondAtom;
544
+ }
545
+ ++i;
546
+ }
547
+ // update bondConfiguration and kwargs keys
548
+ let keys = Array.from(bonds.bondConfiguration.keys());
549
+ keys.forEach((key) => {
550
+ if (bonds.bondConfiguration.has(key) && key > removedNodeIdx) {
551
+ const value = bonds.bondConfiguration.get(key);
552
+ bonds.bondConfiguration.delete(key);
553
+ bonds.bondConfiguration.set(key - 1, value);
554
+ }
555
+ });
556
+ keys = Array.from(bonds.kwargs.keys());
557
+ keys.forEach((key) => {
558
+ if (bonds.kwargs.has(key) && key > removedNodeIdx) {
559
+ const value = bonds.kwargs.get(key);
560
+ bonds.kwargs.delete(key);
561
+ bonds.kwargs.set(key - 1, value);
562
+ }
563
+ });
564
+ }
565
+ }
566
+ // todo: rewrite description
567
+ /* Adjust the (peptide) monomer graph so that it has standard form */
568
+ function adjustPeptideMonomerGraph(monomer) {
569
+ const nodeOneIdx = monomer.meta.terminalNodes[0] - 1; // node indexing in molfiles starts from 1
570
+ const nodeTwoIdx = monomer.meta.rNodes[0] - 1;
571
+ const x = monomer.atoms.x;
572
+ const y = monomer.atoms.y;
573
+ // place nodeOne at origin
574
+ shiftCoordinates(monomer, -x[nodeOneIdx], -y[nodeOneIdx]);
575
+ // angle is measured between OY and the rotated node
576
+ const angle = findAngleWithOY(x[nodeTwoIdx], y[nodeTwoIdx]);
577
+ // rotate the centered graph, so that 'nodeTwo' ends up on the positive ray of OY
578
+ rotateCenteredGraph(monomer.atoms, -angle);
579
+ if (x[monomer.meta.rNodes[1] - 1] < 0)
580
+ flipMonomerAroundOY(monomer);
581
+ const doubleBondedOxygen = findDoubleBondedCarbonylOxygen(monomer);
582
+ // flip carboxyl and R if necessary
583
+ flipCarboxylAndRadical(monomer, doubleBondedOxygen);
584
+ // flip hydroxyl group with double-bound O inside carboxyl group if necessary
585
+ flipHydroxilGroup(monomer, doubleBondedOxygen);
586
+ }
587
+ function adjustPhosphateMonomerGraph(monomer) {
588
+ const nodeOneIdx = monomer.meta.terminalNodes[0] - 1; // node indexing in molfiles starts from 1
589
+ const nodeTwoIdx = monomer.meta.rNodes[0] - 1;
590
+ const x = monomer.atoms.x;
591
+ const y = monomer.atoms.y;
592
+ // place nodeOne at origin
593
+ shiftCoordinates(monomer, -x[nodeOneIdx], -y[nodeOneIdx]);
594
+ // // angle is measured between OY and the rotated node
595
+ // const angle = findAngleWithOY(x[nodeTwoIdx], y[nodeTwoIdx]);
596
+ // // rotate the centered graph, so that 'nodeTwo' ends up on the positive ray of OY
597
+ // rotateCenteredGraph(monomer.atoms, -angle);
598
+ // if (x[monomer.meta.rNodes[1] - 1] < 0)
599
+ // flipMonomerAroundOY(monomer);
600
+ // const doubleBondedOxygen = findDoubleBondedCarbonylOxygen(monomer);
601
+ // // flip carboxyl and R if necessary
602
+ // flipCarboxylAndRadical(monomer, doubleBondedOxygen);
603
+ // // flip hydroxyl group with double-bound O inside carboxyl group if necessary
604
+ // flipHydroxilGroup(monomer, doubleBondedOxygen);
605
+ }
606
+ function adjustSugarMonomerGraph(monomer) {
607
+ const nodeOneIdx = monomer.meta.terminalNodes[0] - 1; // node indexing in molfiles starts from 1
608
+ const nodeTwoIdx = monomer.meta.rNodes[0] - 1;
609
+ const x = monomer.atoms.x;
610
+ const y = monomer.atoms.y;
611
+ // place nodeOne at origin
612
+ shiftCoordinates(monomer, -x[nodeOneIdx], -y[nodeOneIdx]);
613
+ // // angle is measured between OY and the rotated node
614
+ // const angle = findAngleWithOY(x[nodeTwoIdx], y[nodeTwoIdx]);
615
+ // // rotate the centered graph, so that 'nodeTwo' ends up on the positive ray of OY
616
+ // rotateCenteredGraph(monomer.atoms, -angle);
617
+ // if (x[monomer.meta.rNodes[1] - 1] < 0)
618
+ // flipMonomerAroundOY(monomer);
619
+ // const doubleBondedOxygen = findDoubleBondedCarbonylOxygen(monomer);
620
+ // // flip carboxyl and R if necessary
621
+ // flipCarboxylAndRadical(monomer, doubleBondedOxygen);
622
+ // // flip hydroxyl group with double-bound O inside carboxyl group if necessary
623
+ // flipHydroxilGroup(monomer, doubleBondedOxygen);
624
+ }
625
+ function adjustBaseMonomerGraph(monomer) {
626
+ const nodeOneIdx = monomer.meta.terminalNodes[0] - 1; // node indexing in molfiles starts from 1
627
+ const nodeTwoIdx = monomer.meta.rNodes[0] - 1;
628
+ const x = monomer.atoms.x;
629
+ const y = monomer.atoms.y;
630
+ // place nodeOne at origin
631
+ shiftCoordinates(monomer, -x[nodeOneIdx], -y[nodeOneIdx]);
632
+ // // angle is measured between OY and the rotated node
633
+ // const angle = findAngleWithOY(x[nodeTwoIdx], y[nodeTwoIdx]);
634
+ // // rotate the centered graph, so that 'nodeTwo' ends up on the positive ray of OY
635
+ // rotateCenteredGraph(monomer.atoms, -angle);
636
+ // if (x[monomer.meta.rNodes[1] - 1] < 0)
637
+ // flipMonomerAroundOY(monomer);
638
+ // const doubleBondedOxygen = findDoubleBondedCarbonylOxygen(monomer);
639
+ // // flip carboxyl and R if necessary
640
+ // flipCarboxylAndRadical(monomer, doubleBondedOxygen);
641
+ // // flip hydroxyl group with double-bound O inside carboxyl group if necessary
642
+ // flipHydroxilGroup(monomer, doubleBondedOxygen);
643
+ }
644
+ /* Flip carboxyl group with the radical in a peptide monomer in case the
645
+ * carboxyl group is in the lower half-plane */
646
+ function flipCarboxylAndRadical(monomer, doubleBondedOxygen) {
647
+ // verify that the carboxyl group is in the lower half-plane
648
+ if (monomer.atoms.y[monomer.meta.rNodes[1] - 1] < 0 &&
649
+ monomer.atoms.y[doubleBondedOxygen - 1] < 0) {
650
+ flipMonomerAroundOX(monomer);
651
+ rotateCenteredGraph(monomer.atoms, -findAngleWithOX(monomer.atoms.x[monomer.meta.terminalNodes[1] - 1], monomer.atoms.y[monomer.meta.terminalNodes[1] - 1]));
652
+ }
653
+ }
654
+ /* Finds angle between OY and the ray joining origin with (x, y) */
655
+ function findAngleWithOY(x, y) {
656
+ let angle;
657
+ if (x === 0) {
658
+ angle = y > 0 ? 0 : Math.PI;
659
+ }
660
+ else if (y === 0) {
661
+ angle = x > 0 ? -Math.PI / 2 : Math.PI / 2;
662
+ }
663
+ else {
664
+ const tan = y / x;
665
+ const atan = Math.atan(tan);
666
+ angle = (x < 0) ? Math.PI / 2 + atan : -Math.PI / 2 + atan;
667
+ }
668
+ return angle;
669
+ }
670
+ /* Finds angle between OX and the ray joining origin with (x, y) */
671
+ function findAngleWithOX(x, y) {
672
+ return findAngleWithOY(x, y) + Math.PI / 2;
673
+ }
674
+ /* Rotate the graph around the origin by 'angle' */
675
+ function rotateCenteredGraph(atoms, angle) {
676
+ if (angle !== 0) {
677
+ const x = atoms.x;
678
+ const y = atoms.y;
679
+ const cos = Math.cos(angle);
680
+ const sin = Math.sin(angle);
681
+ for (let i = 0; i < x.length; ++i) {
682
+ const tmp = x[i];
683
+ x[i] = keepPrecision(tmp * cos - y[i] * sin);
684
+ y[i] = keepPrecision(tmp * sin + y[i] * cos);
685
+ }
686
+ }
687
+ }
688
+ /* Flip monomer graph around OX axis preserving stereometry */
689
+ function flipMonomerAroundOX(monomer) {
690
+ flipMolGraph(monomer, true);
691
+ }
692
+ /* Flip monomer graph around OY axis preserving stereometry */
693
+ function flipMonomerAroundOY(monomer) {
694
+ flipMolGraph(monomer, false);
695
+ }
696
+ /* Flip graph around a specified axis: 'true' corresponds to OX, 'false' to OY */
697
+ function flipMolGraph(molGraph, axis) {
698
+ if (axis) { // flipping around OX
699
+ const y = molGraph.atoms.y;
700
+ for (let i = 0; i < y.length; i++)
701
+ y[i] = -y[i];
702
+ }
703
+ else { // flipping around OY
704
+ const x = molGraph.atoms.x;
705
+ for (let i = 0; i < x.length; i++)
706
+ x[i] = -x[i];
707
+ }
708
+ // preserve the stereometry
709
+ const orientation = molGraph.bonds.bondConfiguration;
710
+ for (const [key, value] of orientation) {
711
+ const newValue = value === 1 ? 3 : 1;
712
+ orientation.set(key, newValue);
713
+ }
714
+ }
715
+ /* Flips double-bonded 'O' in carbonyl group with 'OH' in order for the monomers
716
+ * to have standard representation simplifying their concatenation. The
717
+ * monomer must already be adjusted with adjustPeptideMonomerGraph in order for this function to be implemented */
718
+ function flipHydroxilGroup(monomer, doubleBondedOxygen) {
719
+ const x = monomer.atoms.x;
720
+ // -1 below because indexing of nodes in molfiles starts from 1, unlike arrays
721
+ if (x[monomer.meta.rNodes[1] - 1] > x[doubleBondedOxygen - 1])
722
+ swapNodes(monomer, doubleBondedOxygen, monomer.meta.rNodes[1]);
723
+ }
724
+ /* Determine the number of node (starting from 1) corresponding to the
725
+ * double-bonded oxygen of the carbonyl group */
726
+ function findDoubleBondedCarbonylOxygen(monomer) {
727
+ const bondsMap = constructBondsMap(monomer);
728
+ let doubleBondedOxygen = 0;
729
+ let i = 0;
730
+ // iterate over the nodes bonded to the carbon and find the double one
731
+ while (doubleBondedOxygen === 0) {
732
+ const node = bondsMap.get(monomer.meta.terminalNodes[1])[i];
733
+ if (monomer.atoms.atomTypes[node - 1] === OXYGEN && node !== monomer.meta.rNodes[1])
734
+ doubleBondedOxygen = node;
735
+ i++;
736
+ }
737
+ return doubleBondedOxygen;
738
+ }
739
+ /* Swap the Cartesian coordinates of the two specified nodes in MolGraph */
740
+ function swapNodes(monomer, nodeOne, nodeTwo) {
741
+ const nodeOneIdx = nodeOne - 1;
742
+ const nodeTwoIdx = nodeTwo - 1;
743
+ const x = monomer.atoms.x;
744
+ const y = monomer.atoms.y;
745
+ const tmpX = x[nodeOneIdx];
746
+ const tmpY = y[nodeOneIdx];
747
+ x[nodeOneIdx] = x[nodeTwoIdx];
748
+ y[nodeOneIdx] = y[nodeTwoIdx];
749
+ x[nodeTwoIdx] = tmpX;
750
+ y[nodeTwoIdx] = tmpY;
751
+ }
752
+ // todo: doc
753
+ function constructBondsMap(monomer) {
754
+ var _a;
755
+ const map = new Map();
756
+ for (const atomPairs of monomer.bonds.atomPairs) {
757
+ for (let i = 0; i < 2; i++) {
758
+ const key = atomPairs[i];
759
+ const value = atomPairs[(i + 1) % 2];
760
+ if (map.has(key))
761
+ (_a = map.get(key)) === null || _a === void 0 ? void 0 : _a.push(value);
762
+ else
763
+ map.set(key, new Array(1).fill(value));
764
+ }
765
+ }
766
+ return map;
767
+ }
768
+ /* Shift molGraph in the XOY plane */
769
+ function shiftCoordinates(molGraph, xShift, yShift) {
770
+ const x = molGraph.atoms.x;
771
+ const y = molGraph.atoms.y;
772
+ for (let i = 0; i < x.length; ++i) {
773
+ x[i] = keepPrecision(x[i] + xShift);
774
+ if (typeof yShift !== 'undefined')
775
+ y[i] = keepPrecision(y[i] + yShift);
776
+ }
777
+ }
778
+ /* Translate a sequence of monomer symbols into Molfile V3000 */
779
+ function monomerSeqToMolfile(monomerSeq, monomersDict, alphabet, polymerType) {
780
+ // todo: handle the case when the polymer is empty
781
+ if (monomerSeq.length === 0)
782
+ throw new Error('monomerSeq is empty');
783
+ // define atom and bond counts, taking into account the bond type
784
+ const { atomCount, bondCount } = getResultingAtomBondCounts(monomerSeq, monomersDict, alphabet, polymerType);
785
+ // create arrays to store lines of the resulting molfile
786
+ const molfileAtomBlock = new Array(atomCount);
787
+ const molfileBondBlock = new Array(bondCount);
788
+ let addMonomerToMolblock;
789
+ let capMolblock;
790
+ let nodeShiftInitValue;
791
+ let bondShiftInitValue;
792
+ let sugar = null;
793
+ let phosphate = null;
794
+ if (polymerType === "PEPTIDE" /* HELM_POLYMER_TYPE.PEPTIDE */) {
795
+ addMonomerToMolblock = addAminoAcidToMolblock;
796
+ capMolblock = capPeptideMolblock;
797
+ nodeShiftInitValue = bondShiftInitValue = 0;
798
+ }
799
+ else { // nucleotides
800
+ addMonomerToMolblock = addNucleotideToMolblock;
801
+ capMolblock = capPeptideMolblock; // todo: cleanup & refactor
802
+ nodeShiftInitValue = 0;
803
+ bondShiftInitValue = 0;
804
+ sugar = (alphabet === "DNA" /* ALPHABET.DNA */) ? monomersDict.get(DEOXYRIBOSE) : monomersDict.get(RIBOSE);
805
+ phosphate = monomersDict.get(PHOSPHATE);
806
+ }
807
+ const v = {
808
+ i: 0,
809
+ nodeShift: nodeShiftInitValue,
810
+ bondShift: bondShiftInitValue,
811
+ backbonePositionShift: new Array(2).fill(0),
812
+ branchPositionShift: new Array(2).fill(0),
813
+ backboneAttachNode: 0,
814
+ branchAttachNode: 0,
815
+ flipFactor: 1,
816
+ };
817
+ const C = {
818
+ sugar: sugar,
819
+ phosphate: phosphate,
820
+ seqLength: monomerSeq.length,
821
+ atomCount: atomCount,
822
+ bondCount: bondCount,
823
+ };
824
+ for (v.i = 0; v.i < C.seqLength; ++v.i) {
825
+ const monomer = monomersDict.get(monomerSeq[v.i]);
826
+ addMonomerToMolblock(monomer, molfileAtomBlock, molfileBondBlock, v, C);
827
+ }
828
+ capMolblock(molfileAtomBlock, molfileBondBlock, v, C);
829
+ const molfileCountsLine = V3K_BEGIN_COUNTS_LINE + atomCount + ' ' + bondCount + V3K_COUNTS_LINE_ENDING;
830
+ // todo: optimize concatenation using Alexander's hint
831
+ const molfileParts = [
832
+ V3K_HEADER_FIRST_LINE,
833
+ V3K_HEADER_SECOND_LINE,
834
+ V3K_BEGIN_CTAB_BLOCK,
835
+ molfileCountsLine,
836
+ V3K_BEGIN_ATOM_BLOCK,
837
+ molfileAtomBlock.join(''),
838
+ V3K_END_ATOM_BLOCK,
839
+ V3K_BEGIN_BOND_BLOCK,
840
+ molfileBondBlock.join(''),
841
+ V3K_END_BOND_BLOCK,
842
+ V3K_END_CTAB_BLOCK,
843
+ V3K_END,
844
+ ];
845
+ return molfileParts.join('');
846
+ }
847
+ // todo: doc
848
+ function capPeptideMolblock(molfileAtomBlock, molfileBondBlock, v, C) {
849
+ // add terminal oxygen
850
+ const atomIdx = v.nodeShift + 1;
851
+ molfileAtomBlock[C.atomCount] = V3K_BEGIN_DATA_LINE + atomIdx + ' ' +
852
+ OXYGEN + ' ' + keepPrecision(v.backbonePositionShift[0]) + ' ' +
853
+ v.flipFactor * keepPrecision(v.backbonePositionShift[1]) + ' ' + '0.000000 0' + '\n';
854
+ // add terminal bond
855
+ const firstAtom = v.backboneAttachNode;
856
+ const secondAtom = atomIdx;
857
+ molfileBondBlock[C.bondCount] = V3K_BEGIN_DATA_LINE + v.bondShift + ' ' +
858
+ 1 + ' ' + firstAtom + ' ' + secondAtom + '\n';
859
+ }
860
+ // todo: doc
861
+ function addAminoAcidToMolblock(monomer, molfileAtomBlock, molfileBondBlock, v, C) {
862
+ v.flipFactor = Math.pow((-1), (v.i % 2)); // to flip every even monomer over OX
863
+ addBackboneMonomerToMolblock(monomer, molfileAtomBlock, molfileBondBlock, v, C);
864
+ }
865
+ function addBackboneMonomerToMolblock(monomer, molfileAtomBlock, molfileBondBlock, v, C) {
866
+ // todo: remove these comments to the docstrings of the corr. functions
867
+ // construnct the lines of V3K molfile atom block
868
+ fillAtomLines(monomer, molfileAtomBlock, v);
869
+ // construct the lines of V3K molfile bond block
870
+ fillBondLines(monomer, molfileBondBlock, v);
871
+ // peptide bond
872
+ fillChainExtendingBond(monomer, molfileBondBlock, v);
873
+ // update branch variables if necessary
874
+ if (monomer.meta.branchShift !== null && monomer.meta.terminalNodes.length > 2)
875
+ updateBranchVariables(monomer, v);
876
+ // update loop variables
877
+ updateChainExtendingVariables(monomer, v, C);
878
+ }
879
+ // todo: doc
880
+ function addNucleotideToMolblock(nucleobase, molfileAtomBlock, molfileBondBlock, v, C) {
881
+ // construnct the lines of V3K molfile atom block corresponding to phosphate
882
+ // and sugar
883
+ for (const monomer of [C.phosphate, C.sugar])
884
+ addBackboneMonomerToMolblock(monomer, molfileAtomBlock, molfileBondBlock, v, C);
885
+ addBranchMonomerToMolblock(nucleobase, molfileAtomBlock, molfileBondBlock, v, C);
886
+ }
887
+ function addBranchMonomerToMolblock(monomer, molfileAtomBlock, molfileBondBlock, v, C) {
888
+ fillBranchAtomLines(monomer, molfileAtomBlock, v);
889
+ fillBondLines(monomer, molfileBondBlock, v);
890
+ fillBackboneToBranchBond(monomer, molfileBondBlock, v);
891
+ // C-N bond
892
+ const bondIdx = v.bondShift;
893
+ const firstAtom = v.branchAttachNode;
894
+ const secondAtom = monomer.meta.terminalNodes[0] + v.nodeShift;
895
+ molfileBondBlock[bondIdx - 1] = V3K_BEGIN_DATA_LINE + bondIdx + ' ' +
896
+ 1 + ' ' + firstAtom + ' ' + secondAtom + '\n';
897
+ // update loop variables
898
+ v.bondShift += monomer.bonds.atomPairs.length + 1;
899
+ v.nodeShift += monomer.atoms.atomTypes.length;
900
+ }
901
+ function updateChainExtendingVariables(monomer, v, C) {
902
+ v.backboneAttachNode = v.nodeShift + monomer.meta.terminalNodes[1];
903
+ v.bondShift += monomer.bonds.atomPairs.length + 1;
904
+ v.nodeShift += monomer.atoms.atomTypes.length;
905
+ v.backbonePositionShift[0] += monomer.meta.backboneShift[0]; // todo: non-null check
906
+ v.backbonePositionShift[1] += v.flipFactor * monomer.meta.backboneShift[1];
907
+ }
908
+ function updateBranchVariables(monomer, v) {
909
+ v.branchAttachNode = v.nodeShift + monomer.meta.terminalNodes[2];
910
+ for (let i = 0; i < 2; ++i)
911
+ v.branchPositionShift[i] = v.backbonePositionShift[i] + monomer.meta.branchShift[i];
912
+ }
913
+ function fillAtomLines(monomer, molfileAtomBlock, v) {
914
+ for (let j = 0; j < monomer.atoms.atomTypes.length; ++j) {
915
+ const atomIdx = v.nodeShift + j + 1;
916
+ molfileAtomBlock[v.nodeShift + j] = V3K_BEGIN_DATA_LINE + atomIdx + ' ' +
917
+ monomer.atoms.atomTypes[j] + ' ' +
918
+ keepPrecision(v.backbonePositionShift[0] + monomer.atoms.x[j]) + ' ' +
919
+ keepPrecision(v.backbonePositionShift[1] + v.flipFactor * monomer.atoms.y[j]) +
920
+ ' ' + monomer.atoms.kwargs[j];
921
+ }
922
+ }
923
+ // todo: remove as quickfix
924
+ function fillBranchAtomLines(monomer, molfileAtomBlock, v) {
925
+ for (let j = 0; j < monomer.atoms.atomTypes.length; ++j) {
926
+ const atomIdx = v.nodeShift + j + 1;
927
+ molfileAtomBlock[v.nodeShift + j] = V3K_BEGIN_DATA_LINE + atomIdx + ' ' +
928
+ monomer.atoms.atomTypes[j] + ' ' +
929
+ keepPrecision(v.branchPositionShift[0] + monomer.atoms.x[j]) + ' ' +
930
+ keepPrecision(v.branchPositionShift[1] + v.flipFactor * monomer.atoms.y[j]) +
931
+ ' ' + monomer.atoms.kwargs[j];
932
+ }
933
+ }
934
+ function fillBondLines(monomer, molfileBondBlock, v) {
935
+ // construct the lines of V3K molfile bond block
936
+ for (let j = 0; j < monomer.bonds.atomPairs.length; ++j) {
937
+ const bondIdx = v.bondShift + j + 1;
938
+ const firstAtom = monomer.bonds.atomPairs[j][0] + v.nodeShift;
939
+ const secondAtom = monomer.bonds.atomPairs[j][1] + v.nodeShift;
940
+ let bondCfg = '';
941
+ if (monomer.bonds.bondConfiguration.has(j)) {
942
+ // flip orientation when necessary
943
+ let orientation = monomer.bonds.bondConfiguration.get(j);
944
+ if (v.flipFactor < 0)
945
+ orientation = (orientation === 1) ? 3 : 1;
946
+ bondCfg = ' CFG=' + orientation;
947
+ }
948
+ const kwargs = monomer.bonds.kwargs.has(j) ?
949
+ ' ' + monomer.bonds.kwargs.get(j) : '';
950
+ molfileBondBlock[v.bondShift + j] = V3K_BEGIN_DATA_LINE + bondIdx + ' ' +
951
+ monomer.bonds.bondTypes[j] + ' ' +
952
+ firstAtom + ' ' + secondAtom + bondCfg + kwargs + '\n';
953
+ }
954
+ }
955
+ function fillChainExtendingBond(monomer, molfileBondBlock, v) {
956
+ if (v.backboneAttachNode !== 0) {
957
+ const bondIdx = v.bondShift;
958
+ const firstAtom = v.backboneAttachNode;
959
+ const secondAtom = monomer.meta.terminalNodes[0] + v.nodeShift;
960
+ molfileBondBlock[v.bondShift - 1] = V3K_BEGIN_DATA_LINE + bondIdx + ' ' +
961
+ 1 + ' ' + firstAtom + ' ' + secondAtom + '\n';
962
+ }
963
+ }
964
+ // todo: remove
965
+ function fillBackboneToBranchBond(branchMonomer, molfileBondBlock, v) {
966
+ const bondIdx = v.bondShift;
967
+ const firstAtom = v.branchAttachNode;
968
+ const secondAtom = branchMonomer.meta.terminalNodes[0] + v.nodeShift;
969
+ molfileBondBlock[bondIdx - 1] = V3K_BEGIN_DATA_LINE + bondIdx + ' ' +
970
+ 1 + ' ' + firstAtom + ' ' + secondAtom + '\n';
971
+ }
972
+ /* Compute the atom/bond counts for the resulting molfile, depending on the
973
+ * type of polymer (peptide/nucleotide) */
974
+ function getResultingAtomBondCounts(monomerSeq, monomersDict, alphabet, polymerType) {
975
+ let atomCount = 0;
976
+ let bondCount = 0;
977
+ // sum up all the atoms/nodes provided by the sequence
978
+ for (const monomerSymbol of monomerSeq) {
979
+ const monomer = monomersDict.get(monomerSymbol);
980
+ atomCount += monomer.atoms.x.length;
981
+ bondCount += monomer.bonds.bondTypes.length;
982
+ }
983
+ // add extra values depending on the polymer type
984
+ if (polymerType === "PEPTIDE" /* HELM_POLYMER_TYPE.PEPTIDE */) {
985
+ // add the rightmost/terminating cap group 'OH' (i.e. 'O')
986
+ atomCount += 1;
987
+ // add chain-extending bonds (C-NH per each monomer pair and terminal C-OH)
988
+ bondCount += monomerSeq.length;
989
+ }
990
+ else { // nucleotides
991
+ const sugar = (alphabet === "DNA" /* ALPHABET.DNA */) ?
992
+ monomersDict.get(DEOXYRIBOSE) : monomersDict.get(RIBOSE);
993
+ const phosphate = monomersDict.get(PHOSPHATE);
994
+ // add phosphate and sugar per each nucleobase symbol
995
+ atomCount += monomerSeq.length * (phosphate.atoms.x.length + sugar.atoms.x.length);
996
+ // add the leftmost cap group 'OH' (i.e. 'O') to the first phosphate
997
+ atomCount += 1;
998
+ // add bonds from phosphate and sugar
999
+ bondCount += monomerSeq.length * (phosphate.bonds.bondTypes.length + sugar.bonds.bondTypes.length);
1000
+ // add chain-extending and branch bonds (O-P, C-O and C-N per each nucleotide)
1001
+ bondCount += monomerSeq.length * 3;
1002
+ }
1003
+ return { atomCount, bondCount };
1004
+ }
1005
+ /* Keep precision upon floating point operations over atom coordinates */
1006
+ function keepPrecision(x) {
1007
+ return Math.round(PRECISION_FACTOR * x) / PRECISION_FACTOR;
1008
+ }
1009
+ function convertMolGraphToMolfileV3K(molGraph) {
1010
+ // counts line
1011
+ const atomType = molGraph.atoms.atomTypes;
1012
+ const x = molGraph.atoms.x;
1013
+ const y = molGraph.atoms.y;
1014
+ const atomKwargs = molGraph.atoms.kwargs;
1015
+ const bondType = molGraph.bonds.bondTypes;
1016
+ const atomPair = molGraph.bonds.atomPairs;
1017
+ const bondKwargs = molGraph.bonds.kwargs;
1018
+ const bondConfig = molGraph.bonds.bondConfiguration;
1019
+ const atomCount = atomType.length;
1020
+ const bondCount = molGraph.bonds.bondTypes.length;
1021
+ // todo rewrite using constants
1022
+ const molfileCountsLine = V3K_BEGIN_COUNTS_LINE + atomCount + ' ' + bondCount + V3K_COUNTS_LINE_ENDING;
1023
+ // atom block
1024
+ let molfileAtomBlock = '';
1025
+ for (let i = 0; i < atomCount; ++i) {
1026
+ const atomIdx = i + 1;
1027
+ const coordinate = [x[i].toString(), y[i].toString()];
1028
+ // format coordinates so that they have 6 digits after decimal point
1029
+ // for (let k = 0; k < 2; ++k) {
1030
+ // const formatted = coordinate[k].toString().split('.');
1031
+ // if (formatted.length === 1)
1032
+ // formatted.push('0');
1033
+ // formatted[1] = formatted[1].padEnd(V3K_ATOM_COORDINATE_PRECISION, '0');
1034
+ // coordinate[k] = formatted.join('.');
1035
+ // }
1036
+ const atomLine = V3K_BEGIN_DATA_LINE + atomIdx + ' ' + atomType[i] + ' ' +
1037
+ coordinate[0] + ' ' + coordinate[1] + ' ' + atomKwargs[i];
1038
+ molfileAtomBlock += atomLine;
1039
+ }
1040
+ // bond block
1041
+ let molfileBondBlock = '';
1042
+ for (let i = 0; i < bondCount; ++i) {
1043
+ const bondIdx = i + 1;
1044
+ const firstAtom = atomPair[i][0];
1045
+ const secondAtom = atomPair[i][1];
1046
+ const kwargs = bondKwargs.has(i) ? ' ' + bondKwargs.get(i) : '';
1047
+ const bondCfg = bondConfig.has(i) ? ' CFG=' + bondConfig.get(i) : '';
1048
+ const bondLine = V3K_BEGIN_DATA_LINE + bondIdx + ' ' + bondType[i] + ' ' +
1049
+ firstAtom + ' ' + secondAtom + bondCfg + kwargs + '\n';
1050
+ molfileBondBlock += bondLine;
1051
+ }
1052
+ const molfileParts = [
1053
+ V3K_HEADER_FIRST_LINE,
1054
+ V3K_HEADER_SECOND_LINE,
1055
+ V3K_BEGIN_CTAB_BLOCK,
1056
+ molfileCountsLine,
1057
+ V3K_BEGIN_ATOM_BLOCK,
1058
+ molfileAtomBlock,
1059
+ V3K_END_ATOM_BLOCK,
1060
+ V3K_BEGIN_BOND_BLOCK,
1061
+ molfileBondBlock,
1062
+ V3K_END_BOND_BLOCK,
1063
+ V3K_END_CTAB_BLOCK,
1064
+ V3K_END,
1065
+ ];
1066
+ const resultingMolfile = molfileParts.join('');
1067
+ // console.log(resultingMolfile);
1068
+ return resultingMolfile;
1069
+ }
1070
+ export function getSymbolToCappedMolfileMap(monomersLibList) {
1071
+ return __awaiter(this, void 0, void 0, function* () {
1072
+ if (DG.Func.find({ package: 'Chem', name: 'getRdKitModule' }).length === 0) {
1073
+ grok.shell.warning('Transformation to atomic level requires package "Chem" installed.');
1074
+ return;
1075
+ }
1076
+ const symbolToCappedMolfileMap = new Map;
1077
+ const moduleRdkit = yield grok.functions.call('Chem:getRdKitModule');
1078
+ for (const monomerLibObject of monomersLibList) {
1079
+ const monomerSymbol = monomerLibObject["symbol" /* HELM_FIELDS.SYMBOL */];
1080
+ const capGroups = parseCapGroups(monomerLibObject["rgroups" /* HELM_FIELDS.RGROUPS */]);
1081
+ const capGroupIdxMap = parseCapGroupIdxMap(monomerLibObject["molfile" /* HELM_FIELDS.MOLFILE */]);
1082
+ const molfileV3K = convertMolfileToV3K(removeRGroupLines(monomerLibObject["molfile" /* HELM_FIELDS.MOLFILE */]), moduleRdkit);
1083
+ const counts = parseAtomAndBondCounts(molfileV3K);
1084
+ const atoms = parseAtomBlock(molfileV3K, counts.atomCount);
1085
+ const bonds = parseBondBlock(molfileV3K, counts.bondCount);
1086
+ const meta = getMonomerMetadata(atoms, bonds, capGroups, capGroupIdxMap);
1087
+ const monomerGraph = { atoms: atoms, bonds: bonds, meta: meta };
1088
+ removeHydrogen(monomerGraph);
1089
+ const molfile = convertMolGraphToMolfileV3K(monomerGraph);
1090
+ symbolToCappedMolfileMap.set(monomerSymbol, molfile);
1091
+ }
1092
+ return symbolToCappedMolfileMap;
1093
+ });
1094
+ }
1095
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG8tYXRvbWljLWxldmVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsidG8tYXRvbWljLWxldmVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUFBLHlGQUF5RjtBQUN6RixPQUFPLEtBQUssSUFBSSxNQUFNLG1CQUFtQixDQUFDO0FBQzFDLE9BQU8sS0FBSyxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDdEMsT0FBTyxFQUFxQixXQUFXLEVBQUMsTUFBTSxhQUFhLENBQUM7QUFDNUQsT0FBTyxFQUFjLGdCQUFnQixFQUFzRCxNQUFNLFNBQVMsQ0FBQztBQUczRyxPQUFPLEVBQUMsaUJBQWlCLEVBQUMsTUFBTSxzQkFBc0IsQ0FBQztBQUV2RCxzQ0FBc0M7QUFDdEMsTUFBTSxhQUFhLEdBQUcsQ0FBQyxDQUFDO0FBQ3hCLE1BQU0sWUFBWSxHQUFHLFFBQVEsQ0FBQztBQUM5QixNQUFNLFVBQVUsR0FBRyxLQUFLLENBQUM7QUFFekIsd0RBQXdEO0FBQ3hELE1BQU0sZ0JBQWdCLEdBQUcsRUFBRSxDQUFDO0FBQzVCLE1BQU0sYUFBYSxHQUFHLENBQUMsQ0FBQztBQUN4QixNQUFNLHFCQUFxQixHQUFHLHNDQUFzQyxDQUFDO0FBQ3JFLE1BQU0sc0JBQXNCLEdBQUcsMkNBQTJDLENBQUM7QUFDM0UsTUFBTSxvQkFBb0IsR0FBRyxxQkFBcUIsQ0FBQztBQUNuRCxNQUFNLGtCQUFrQixHQUFHLG1CQUFtQixDQUFDO0FBQy9DLE1BQU0scUJBQXFCLEdBQUcsZ0JBQWdCLENBQUM7QUFDL0MsTUFBTSxzQkFBc0IsR0FBRyxVQUFVLENBQUM7QUFDMUMsTUFBTSxvQkFBb0IsR0FBRyxxQkFBcUIsQ0FBQztBQUNuRCxNQUFNLGtCQUFrQixHQUFHLG1CQUFtQixDQUFDO0FBQy9DLE1BQU0sb0JBQW9CLEdBQUcscUJBQXFCLENBQUM7QUFDbkQsTUFBTSxrQkFBa0IsR0FBRyxtQkFBbUIsQ0FBQztBQUMvQyxNQUFNLGVBQWUsR0FBRyxPQUFPLENB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RTtRQUNsQixLQUFLLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsRUFBRSxHQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEVBQUUsR0FBQyxDQUFDLENBQUM7S0FDeEM7U0FBTTtRQUNMLE1BQU0sR0FBRyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDbEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM1QixLQUFLLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxFQUFFLEdBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsRUFBRSxHQUFDLENBQUMsR0FBRyxJQUFJLENBQUM7S0FDeEQ7SUFDRCxPQUFPLEtBQUssQ0FBQztBQUNmLENBQUM7QUFFRCxtRUFBbUU7QUFDbkUsU0FBUyxlQUFlLENBQUMsQ0FBUyxFQUFFLENBQVM7SUFDM0MsT0FBTyxlQUFlLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxFQUFFLEdBQUMsQ0FBQyxDQUFDO0FBQzNDLENBQUM7QUFFRCxvREFBb0Q7QUFDcEQsU0FBUyxtQkFBbUIsQ0FBQyxLQUFZLEVBQUUsS0FBYTtJQUN0RCxJQUFJLEtBQUssS0FBSyxDQUFDLEVBQUU7UUFDZixNQUFNLENBQUMsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQ2xCLE1BQU0sQ0FBQyxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUM7UUFFbEIsTUFBTSxHQUFHLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM1QixNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBRTVCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FB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YW5nZSB0aGVzZSBpbXBvcnQgbGluZXMgdG8gbWF0Y2ggZXh0ZXJuYWwgbW9kdWxlcyBpbiB3ZWJwYWNrIGNvbmZpZ3VyYXRpb24gKi9cbmltcG9ydCAqIGFzIGdyb2sgZnJvbSAnZGF0YWdyb2stYXBpL2dyb2snO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcbmltcG9ydCB7U3BsaXR0ZXJGdW5jLCBUQUdTLCBnZXRTcGxpdHRlcn0gZnJvbSAnLi4vLi4vaW5kZXgnO1xuaW1wb3J0IHtIRUxNX0ZJRUxEUywgSEVMTV9DT1JFX0ZJRUxEUywgSEVMTV9QT0xZTUVSX1RZUEUsIEhFTE1fTU9OT01FUl9UWVBFLCBSR1JPVVBfRklFTERTfSBmcm9tICcuL2NvbnN0JztcbmltcG9ydCB7QUxQSEFCRVQsIE5PVEFUSU9OfSBmcm9tICcuL21hY3JvbW9sZWN1bGUnO1xuaW1wb3J0IHtVbml0c0hhbmRsZXJ9IGZyb20gJy4vdW5pdHMtaGFuZGxlcic7XG5pbXBvcnQge05vdGF0aW9uQ29udmVydGVyfSBmcm9tICcuL25vdGF0aW9uLWNvbnZlcnRlcic7XG5cbi8vIGNvbnN0YW50cyBmb3IgcGFyc2luZyBtb2xmaWxlIFYyMDAwXG5jb25zdCBWMktfUkdQX1NISUZUID0gODtcbmNvbnN0IFYyS19SR1BfTElORSA9ICdNICBSR1AnO1xuY29uc3QgVjJLX0FfTElORSA9ICdBICAnO1xuXG4vLyBjb25zdGFudHMgZm9yIHBhcnNpbmcvcmVjb25zdHJ1Y3Rpb24gb2YgbW9sZmlsZSBWMzAwMFxuY29uc3QgVjNLX0NPVU5UU19TSElGVCA9IDE0O1xuY29uc3QgVjNLX0lEWF9TSElGVCA9IDc7XG5jb25zdCBWM0tfSEVBREVSX0ZJUlNUX0xJTkUgPSAnXFxuRGF0YWdyb2sgbWFjcm9tb2xlY3VsZSBoYW5kbGVyXFxuXFxuJztcbmNvbnN0IFYzS19IRUFERVJfU0VDT05EX0xJTkUgPSAnICAwICAwICAwICAwICAwICAwICAgICAgICAgICAgOTk5IFYzMDAwXFxuJztcbmNvbnN0IFYzS19CRUdJTl9DVEFCX0JMT0NLID0gJ00gIFYzMCBCRUdJTiBDVEFCXFxuJztcbmNvbnN0IFYzS19FTkRfQ1RBQl9CTE9DSyA9ICdNICBWMzAgRU5EIENUQUJcXG4nO1xuY29uc3QgVjNLX0JFR0lOX0NPVU5UU19MSU5FID0gJ00gIFYzMCBDT1VOVFMgJztcbmNvbnN0IFYzS19DT1VOVFNfTElORV9FTkRJTkcgPSAnIDAgMCAwXFxuJztcbmNvbnN0IFYzS19CRUdJTl9BVE9NX0JMT0NLID0gJ00gIFYzMCBCRUdJTiBBVE9NXFxuJztcbmNvbnN0IFYzS19FTkRfQVRPTV9CTE9DSyA9ICdNICBWMzAgRU5EIEFUT01cXG4nO1xuY29uc3QgVjNLX0JFR0lOX0JPTkRfQkxPQ0sgPSAnTSAgVjMwIEJFR0lOIEJPTkRcXG4nO1xuY29uc3QgVjNLX0VORF9CT05EX0JMT0NLID0gJ00gIFYzMCBFTkQgQk9ORFxcbic7XG5jb25zdCBWM0tfQk9ORF9DT05GSUcgPSAnIENGRz0nO1xuY29uc3QgVjNLX0JFR0lOX0RBVEFfTElORSA9ICdNICBWMzAgJztcbmNvbnN0IFYzS19FTkQgPSAnTSAgRU5EXFxuJztcblxuY29uc3QgUFJFQ0lTSU9OX0ZBQ1RPUiA9IDEwXzAwMDsgLy8gSEVMTUNvcmVMaWJyYXJ5IGhhcyA0IHNpZ25pZmljYW50IGRpZ2l0cyBhZnRlciBkZWNpbWFsIHBvaW50IGluIGF0b20gY29vcmRpbmF0ZXNcblxuLy8gc3ltYm9scyBmb3IgdGhlIGNvcnJlc3BvbmRpbmcgbW9ub21lcnMgaW4gSEVMTSBsaWJyYXJ5XG5jb25zdCBERU9YWVJJQk9TRSA9ICdkJztcbmNvbnN0IFJJQk9TRSA9ICdyJztcbmNvbnN0IFBIT1NQSEFURSA9ICdwJztcblxuY29uc3QgT1hZR0VOID0gJ08nO1xuY29uc3QgSFlEUk9HRU4gPSAnSCc7XG5cbi8qIFN0b3JlcyBuZWNlc3NhcnkgZGF0YSBhYm91dCBhdG9tcyBvZiBhIG1vbm9tZXIgcGFyc2VkIGZyb20gTW9sZmlsZSAqL1xudHlwZSBBdG9tcyA9IHtcbiAgLyogZWxlbWVudCBzeW1ib2xzIGZvciBtb25vbWVyJ3MgYXRvbXMgKi9cbiAgYXRvbVR5cGVzOiBzdHJpbmdbXSxcbiAgLyogQ2FydGVzaWFuIGNvb3JkaWFudGVzIG9mIG1vbm9tZXIncyBhdG9tcyAqL1xuICB4OiBudW1iZXJbXSwgLy8gdG9kbzogY29udmVydCB0byBGbG9hdDMyXG4gIHk6IG51bWJlcltdLFxuICAvKiBWM0sgYXRvbSBsaW5lIG1heSBjb250YWluIGtleXdvcmQgYXJncyAqL1xuICBrd2FyZ3M6IHN0cmluZ1tdLFxufVxuXG4vKiBTdG9yZXMgbmVjZXNzYXJ5IGRhdGEgYWJvdXQgYm9uZHMgb2YgYSBtb25vbWVyIHBhcnNlZCBmcm9tIE1vbGZpbGUgKi9cbnR5cGUgQm9uZHMgPSB7XG4gIC8qIGJvbmQgdHlwZXMgZm9yIGFsbCBsaW5lcyBvZiBNb2xmaWxlIGJvbmQgYmxvY2sgKi9cbiAgYm9uZFR5cGVzOiBudW1iZXJbXSwgLy8gdG9kbzogY29udmVydCB0byBJbmQzMlxuICAvKiBJbmRpY2VzIG9mIGFsbCBhdG9tIHBhaXJzLCBpbmRleGluZyBzdGFydGluZyBmcm9tIDEgICovXG4gIGF0b21QYWlyczogbnVtYmVyW11bXSxcbiAgLyogSWYgYSBib25kIGhhcyBDRkc9Li4uIGtleXdvcmQgYXJndW1lbnQsIGl0IGlzIHBhcnNlZCBhbmQgc290cmVkIGFzIGFcbiAgICogdmFsdWUgb2YgdGhlIG1hcCwgd2l0aCB0aGUga2V5IGJlaW5nIHRoZSBib25kJ3MgaW5kZXggICovXG4gIGJvbmRDb25maWd1cmF0aW9uOiBNYXA8bnVtYmVyLCBudW1iZXI+LFxuICAvKiBWM0sgYm9uZCBsaW5lIG1heSBjb250YWluIGtleXdvcmQgYXJncyAqL1xuICBrd2FyZ3M6IE1hcDxudW1iZXIsIHN0cmluZz4sXG59XG5cbi8qIE1ldGFkYXRhIGFzc29jaWF0ZWQgd2l0aCB0aGUgbW9ub21lciBuZWNlc3NhcnkgdG8gcmVzdG9yZSB0aGUgcmVzdWx0aW5nXG4gKiBtb2xmaWxlICAqL1xudHlwZSBNb25vbWVyTWV0YWRhdGEgPSB7XG4gIC8qIHRlcm1pbmFsIG5vZGVzOiAwLXRoIGNvcnJlc3BvbmRzIHRvIHRoZSBcImxlZnRtb3N0XCIgb25lLCAxc3QsIHRvIHRoZSBcInJpZ2h0bW9zdFwiLFxuICAgKiBlLmcuIE4tdGVybWludXMgYW5kIEMtdGVybWludXMgaW4gcGVwdGlkZXMgKi9cbiAgdGVybWluYWxOb2RlczogbnVtYmVyW10sXG4gIC8qIHItZ3JvdXAgbm9kZXM6IDAtdGggY29ycmVzcG9uZHMgdG8gdGhlIFwibGVmdG1vc3RcIiBvbmUsIDFzdCwgdG8gdGhlIFwicmlnaHRtb3N0XCIgKi9cbiAgck5vZGVzOiBudW1iZXJbXSxcbiAgLyogc2hpZnQgZnJvbSB0aGUgb3JpZ2luIHRvIHRoZSBuZXh0IGJhY2tib25lLCBudWxsIGZvciBicmFuY2ggbW9ub21lcnMgKi9cbiAgYmFja2JvbmVTaGlmdDogbnVtYmVyW10gfCBudWxsLFxuICAvKiBzaGlmdCBmcm9tIHRoZSBvcmlnaW4gdG8gdGhlIG5leHQgYnJhbmNoLCBudWxsIGZvciBicmFuY2ggbW9ub21lcnMgKi9cbiAgYnJhbmNoU2hpZnQ6IG51bWJlcltdIHwgbnVsbFxufVxuXG50eXBlIE1vbEdyYXBoID0ge1xuICBhdG9tczogQXRvbXMsXG4gIGJvbmRzOiBCb25kcyxcbiAgbWV0YTogTW9ub21lck1ldGFkYXRhLFxufVxuXG4vKiBIZWxwZXIgc3RydWN0dXJlIHdyYXBwaW5nIGNvbW1vbiBhcmd1bWVudHMgdG8gc2V2ZXJhbCBmdW5jdGlvbnMgKi9cbnR5cGUgTG9vcFZhcmlhYmxlcyA9IHtcbiAgaTogbnVtYmVyLFxuICBub2RlU2hpZnQ6IG51bWJlcixcbiAgYm9uZFNoaWZ0OiBudW1iZXIsXG4gIGJhY2tib25lUG9zaXRpb25TaGlmdDogbnVtYmVyW10sXG4gIGJhY2tib25lQXR0YWNoTm9kZTogbnVtYmVyOyAvLyBub2RlIHRvIHdoaWNoIHRoZSBuZXh0IGJhY2tib25lIGlzIGF0dGFjaGVkXG4gIGJyYW5jaFBvc2l0aW9uU2hpZnQ6IG51bWJlcltdLFxuICBicmFuY2hBdHRhY2hOb2RlOiBudW1iZXIsXG4gIGZsaXBGYWN0b3I6IG51bWJlcixcbiAgLy8gdG9kbzogc2hvdWxkIHdlIGNvbnNpZGVyIHJlcHJlc2VudGF0aW9ucyBvdGhlciB0aGFuIHBsYW5hcj9cbn1cblxuLyogSGVscGVyIHN0cnVjdHVyZSB3cmFwcGluZyBjb21tb24gYXJndW1lbnRzIHRvIHNldmVyYWwgZnVuY3Rpb25zICovXG50eXBlIExvb3BDb25zdGFudHMgPSB7XG4gIHN1Z2FyOiBNb2xHcmFwaCB8IG51bGwsXG4gIHBob3NwaGF0ZTogTW9sR3JhcGggfCBudWxsLFxuICBzZXFMZW5ndGg6IG51bWJlcixcbiAgYXRvbUNvdW50OiBudW1iZXIsXG4gIGJvbmRDb3VudDogbnVtYmVyLFxufVxuXG4vLyB0b2RvOiB2ZXJpZnkgdGhhdCBhbGwgZnVuY3Rpb25zIGhhdmUgcmV0dXJuIHR5cGVzXG5cbi8qIENvbnZlcnQgTWFjcm9tb2xlY3VsZSBjb2x1bW4gaW50byBNb2xlY3VsZSBjb2x1bW4gc3RvcmluZyBtb2xmaWxlIFYzMDAwIHdpdGggdGhlIGhlbHAgb2YgYSBtb25vbWVyIGxpYnJhcnkgICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gX3RvQXRvbWljTGV2ZWwoXG4gIGRmOiBERy5EYXRhRnJhbWUsIG1hY3JvTW9sQ29sOiBERy5Db2x1bW48c3RyaW5nPiwgbW9ub21lcnNMaWJMaXN0OiBhbnlbXVxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIGlmIChERy5GdW5jLmZpbmQoe3BhY2thZ2U6ICdDaGVtJywgbmFtZTogJ2dldFJkS2l0TW9kdWxlJ30pLmxlbmd0aCA9PT0gMCkge1xuICAgIGdyb2suc2hlbGwud2FybmluZygnVHJhbnNmb3JtYXRpb24gdG8gYXRvbWljIGxldmVsIHJlcXVpcmVzIHBhY2thZ2UgXCJDaGVtXCIgaW5zdGFsbGVkLicpO1xuICAgIHJldHVybjtcbiAgfVxuXG4gIGlmIChtYWNyb01vbENvbC5zZW1UeXBlICE9PSBERy5TRU1UWVBFLk1BQ1JPTU9MRUNVTEUpIHtcbiAgICBncm9rLnNoZWxsLndhcm5pbmcoXG4gICAgICBgT25seSB0aGUgJHtERy5TRU1UWVBFLk1BQ1JPTU9MRUNVTEV9IGNvbHVtbnMgY2FuIGJlIGNvbnZlcnRlZCB0byBhdG9taWNcbiAgICAgIGxldmVsLCB0aGUgY2hvc2VuIGNvbHVtbiBoYXMgc2VtVHlwZSAke21hY3JvTW9sQ29sLnNlbVR5cGV9YFxuICAgICk7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgLy8gY29udmVydCAnaGVsbScgdG8gJ3NlcGFyYXRvcicgdW5pdHNcbiAgaWYgKG1hY3JvTW9sQ29sLmdldFRhZyhERy5UQUdTLlVOSVRTKSA9PT0gTk9UQVRJT04uSEVMTSkge1xuICAgIGNvbnN0IGNvbnZlcnRlciA9IG5ldyBOb3RhdGlvbkNvbnZlcnRlcihtYWNyb01vbENvbCk7XG4gICAgY29uc3Qgc2VwYXJhdG9yID0gJy8nO1xuICAgIG1hY3JvTW9sQ29sID0gY29udmVydGVyLmNvbnZlcnQoTk9UQVRJT04uU0VQQVJBVE9SLCBzZXBhcmF0b3IpO1xuICB9XG5cbiAgY29uc3QgYWxwaGFiZXQgPSBtYWNyb01vbENvbC5nZXRUYWcoVEFHUy5hbHBoYWJldCk7XG5cbiAgLy8gZGV0ZXJtaW5lIHRoZSBwb2x5bWVyIHR5cGUgYWNjb3JkaW5nIHRvIEhFTE0gc3BlY2lmaWNhdGlvbnNcbiAgbGV0IHBvbHltZXJUeXBlO1xuICAvLyB0b2RvOiBhbiBleGNlcHRpb24gZnJvbSBkYXJ0IGNvbWVzIGJlZm9yZSB0aGlzIGNoZWNrIGlmIHRoZSBhbHBoYWJldCBpcyBVTlxuICBpZiAoYWxwaGFiZXQgPT09IEFMUEhBQkVULlBUKSB7XG4gICAgcG9seW1lclR5cGUgPSBIRUxNX1BPTFlNRVJfVFlQRS5QRVBUSURFO1xuICB9IGVsc2UgaWYgKGFscGhhYmV0ID09PSBBTFBIQUJFVC5STkEgfHwgYWxwaGFiZXQgPT09IEFMUEhBQkVULkROQSkge1xuICAgIHBvbHltZXJUeXBlID0gSEVMTV9QT0xZTUVSX1RZUEUuUk5BO1xuICB9IGVsc2Uge1xuICAgIGdyb2suc2hlbGwud2FybmluZyhcbiAgICAgIGBPbmx5IFBULCBETkEgYW5kIFJOQSBhbHBoYWJldHMgYXJlIHN1cHBvcnRlZCwgd2hpbGUgdGhlIHNlbGVjdGVkIGNvbHVtbiBoYXMgJHtwb2x5bWVyVHlwZX0gYWxwaGFiZXRgXG4gICAgKTtcbiAgICByZXR1cm47XG4gIH1cblxuICBjb25zdCBtb25vbWVyU2VxdWVuY2VzQXJyYXk6IHN0cmluZ1tdW10gPSBnZXRNb25vbWVyU2VxdWVuY2VzQXJyYXkobWFjcm9Nb2xDb2wpO1xuICAvLyB0b2RvOiBjb25zaWRlciBzZXBhcmF0ZWx5IGJhY2tib25lLCB0ZXJtaW5hbCwgYnJhbmNoIG1vbm9tZXIgdHlwZXNcbiAgY29uc3QgbW9ub21lcnNEaWN0ID0gYXdhaXQgZ2V0TW9ub21lcnNEaWN0KG1vbm9tZXJTZXF1ZW5jZXNBcnJheSwgbW9ub21lcnNMaWJMaXN0LCBwb2x5bWVyVHlwZSwgYWxwaGFiZXQpO1xuICBjb25zdCBjb2x1bW5MZW5ndGggPSBtYWNyb01vbENvbC5sZW5ndGg7XG4gIGNvbnN0IHJlY29uc3RydWN0ZWQ6IHN0cmluZ1tdID0gbmV3IEFycmF5KGNvbHVtbkxlbmd0aCk7XG4gIGZvciAobGV0IHJvdyA9IDA7IHJvdyA8IGNvbHVtbkxlbmd0aDsgKytyb3cpIHtcbiAgICBjb25zdCBtb25vbWVyU2VxID0gbW9ub21lclNlcXVlbmNlc0FycmF5W3Jvd107XG4gICAgcmVjb25zdHJ1Y3RlZFtyb3ddID0gbW9ub21lclNlcVRvTW9sZmlsZShtb25vbWVyU2VxLCBtb25vbWVyc0RpY3QsIGFscGhhYmV0LCBwb2x5bWVyVHlwZSk7XG4gICAgY29uc29sZS5sb2cocmVjb25zdHJ1Y3RlZFtyb3ddKTtcbiAgfVxuXG4gIC8vIGV4Y2x1ZGUgbmFtZSBjb2xsaXNpb25zXG4gIGNvbnN0IG5hbWUgPSAnbW9sZmlsZSgnICsgbWFjcm9Nb2xDb2wubmFtZSArICcpJztcbiAgY29uc3QgbmV3Q29sTmFtZSA9IGRmLmNvbHVtbnMuZ2V0VW51c2VkTmFtZShuYW1lKTtcbiAgY29uc3QgbmV3Q29sID0gREcuQ29sdW1uLmZyb21TdHJpbmdzKG5ld0NvbE5hbWUsIHJlY29uc3RydWN0ZWQpO1xuXG4gIG5ld0NvbC5zZW1UeXBlID0gREcuU0VNVFlQRS5NT0xFQ1VMRTtcbiAgbmV3Q29sLnNldFRhZyhERy5UQUdTLlVOSVRTLCBERy5VTklUUy5Nb2xlY3VsZS5NT0xCTE9DSyk7XG4gIGRmLmNvbHVtbnMuYWRkKG5ld0NvbCwgdHJ1ZSk7XG4gIGF3YWl0IGdyb2suZGF0YS5kZXRlY3RTZW1hbnRpY1R5cGVzKGRmKTtcbn1cblxuLyogR2V0IGEgbWFwcGluZyBvZiBwZXB0aWRlIHN5bWJvbHMgdG8gSEVMTSBtb25vbWVyIGxpYnJhcnkgb2JqZWN0cyB3aXRoXG4gKiBzZWxlY3R0ZWQgZmllbGRzICAqL1xuZnVuY3Rpb24gZ2V0Rm9ybWF0dGVkTW9ub21lckxpYihcbiAgbW9ub21lcnNMaWJMaXN0OiBhbnlbXSwgcG9seW1lclR5cGU6IEhFTE1fUE9MWU1FUl9UWVBFLCBhbHBoYWJldDogQUxQSEFCRVRcbik6IE1hcDxzdHJpbmcsIGFueT4ge1xuICBjb25zdCBtYXAgPSBuZXcgTWFwPHN0cmluZywgYW55PigpO1xuICBtb25vbWVyc0xpYkxpc3QuZm9yRWFjaChcbiAgICAoaXQpID0+IHtcbiAgICAgIGlmIChpdFtIRUxNX0ZJRUxEUy5QT0xZTUVSX1RZUEVdID09PSBwb2x5bWVyVHlwZSkge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgcG9seW1lclR5cGUgPT09IEhFTE1fUE9MWU1FUl9UWVBFLlJOQSAmJlxuICAgICAgICAgIChpdFtIRUxNX0ZJRUxEUy5NT05PTUVSX1RZUEVdID09PSBIRUxNX01PTk9NRVJfVFlQRS5CUkFOQ0ggfHxcbiAgICAgICAgICBhbHBoYWJldCA9PT0gQUxQSEFCRVQuRE5BICYmIGl0W0hFTE1fRklFTERTLlNZTUJPTF0gPT09IERFT1hZUklCT1NFIHx8XG4gICAgICAgICAgYWxwaGFiZXQgPT09IEFMUEhBQkVULlJOQSAmJiBpdFtIRUxNX0ZJRUxEUy5TWU1CT0xdID09PSBSSUJPU0UgfHxcbiAgICAgICAgICBpdFtIRUxNX0ZJRUxEUy5TWU1CT0xdID09PSBQSE9TUEhBVEUpIHx8XG4gICAgICAgICAgcG9seW1lclR5cGUgPT09IEhFTE1fUE9MWU1FUl9UWVBFLlBFUFRJREUgJiZcbiAgICAgICAgICBpdFtIRUxNX0ZJRUxEUy5NT05PTUVSX1RZUEVdICE9PSBIRUxNX01PTk9NRVJfVFlQRS5CUkFOQ0hcbiAgICAgICAgKSB7XG4gICAgICAgICAgY29uc3QgbW9ub21lck9iamVjdDogeyBba2V5OiBzdHJpbmddOiBhbnkgfSA9IHt9O1xuICAgICAgICAgIEhFTE1fQ09SRV9GSUVMRFMuZm9yRWFjaCgoZmllbGQpID0+IHtcbiAgICAgICAgICAgIG1vbm9tZXJPYmplY3RbZmllbGRdID0gaXRbZmllbGRdO1xuICAgICAgICAgIH0pO1xuICAgICAgICAgIG1hcC5zZXQoaXRbSEVMTV9GSUVMRFMuU1lNQk9MXSwgbW9ub21lck9iamVjdCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9KTtcbiAgcmV0dXJuIG1hcDtcbn1cblxuLyogR2V0IGphZ2dlZCBhcnJheSBvZiBtb25vbWVyIHN5bWJvbHMgZm9yIHRoZSBkYXRhZnJhbWUgICovXG5mdW5jdGlvbiBnZXRNb25vbWVyU2VxdWVuY2VzQXJyYXkobWFjcm9Nb2xDb2w6IERHLkNvbHVtbjxzdHJpbmc+KTogc3RyaW5nW11bXSB7XG4gIGNvbnN0IGNvbHVtbkxlbmd0aCA9IG1hY3JvTW9sQ29sLmxlbmd0aDtcbiAgY29uc3QgcmVzdWx0OiBzdHJpbmdbXVtdID0gbmV3IEFycmF5KGNvbHVtbkxlbmd0aCk7XG5cbiAgLy8gc3BsaXQgdGhlIHN0cmluZyBpbnRvIG1vbm9tZXJzXG4gIGNvbnN0IGNvbFVuaXRzID0gbWFjcm9Nb2xDb2wuZ2V0VGFnKERHLlRBR1MuVU5JVFMpO1xuICBjb25zdCBzZXBhcmF0b3IgPSBtYWNyb01vbENvbC5nZXRUYWcoVEFHUy5zZXBhcmF0b3IpO1xuICBjb25zdCBzcGxpdHRlckZ1bmM6IFNwbGl0dGVyRnVuYyA9IGdldFNwbGl0dGVyKGNvbFVuaXRzLCBzZXBhcmF0b3IpO1xuXG4gIGZvciAobGV0IHJvdyA9IDA7IHJvdyA8IGNvbHVtbkxlbmd0aDsgKytyb3cpIHtcbiAgICBjb25zdCBtYWNyb01vbGVjdWxlID0gbWFjcm9Nb2xDb2wuZ2V0KHJvdyk7XG4gICAgLy8gdG9kbzogaGFuZGxlIHRoZSBleGNlcHRpb24gY2FzZSB3aGVuIG1hY3JvTW9sZWN1bGUgaXMgbnVsbFxuICAgIHJlc3VsdFtyb3ddID0gbWFjcm9Nb2xlY3VsZSA/IHNwbGl0dGVyRnVuYyhtYWNyb01vbGVjdWxlKSA6IFtdO1xuICB9XG4gIHJldHVybiByZXN1bHQ7XG59XG5cbi8qIEdldCBhIG1hcHBpbmcgb2YgbW9ub21lciBzeW1ib2xzIHRvIE1vbEdyYXBoIG9iamVjdHMgKi9cbmFzeW5jIGZ1bmN0aW9uIGdldE1vbm9tZXJzRGljdChcbiAgbW9ub21lclNlcXVlbmNlc0FycmF5OiBzdHJpbmdbXVtdLCBtb25vbWVyc0xpYkxpc3Q6IGFueVtdLCBwb2x5bWVyVHlwZTogSEVMTV9QT0xZTUVSX1RZUEUsIGFscGhhYmV0OiBBTFBIQUJFVFxuKTogUHJvbWlzZTxNYXA8c3RyaW5nLCBNb2xHcmFwaD4+IHtcbiAgLy8gdG9kbzogZXhjZXB0aW9uIC0gbm8gZ2Fwcywgbm8gZW1wdHkgc3RyaW5nIG1vbm9tZXJzXG4gIGNvbnN0IGZvcm1hdHRlZE1vbm9tZXJMaWIgPSBnZXRGb3JtYXR0ZWRNb25vbWVyTGliKG1vbm9tZXJzTGliTGlzdCwgcG9seW1lclR5cGUsIGFscGhhYmV0KTtcbiAgY29uc3QgbW9ub21lcnNEaWN0ID0gbmV3IE1hcDxzdHJpbmcsIE1vbEdyYXBoPigpO1xuXG4gIGNvbnN0IG1vZHVsZVJka2l0ID0gYXdhaXQgZ3Jvay5mdW5jdGlvbnMuY2FsbCgnQ2hlbTpnZXRSZEtpdE1vZHVsZScpO1xuXG4gIC8vIGFkZCBkZW94eXJpYm9zZS9yaWJvc2UgYW5kIHBob3NwaGF0ZSBmb3IgbnVjbGVvdGlkZSBzZXF1ZW5jZXNcbiAgaWYgKHBvbHltZXJUeXBlID09PSBIRUxNX1BPTFlNRVJfVFlQRS5STkEpIHtcbiAgICBjb25zdCBzeW1ib2xzID0gKGFscGhhYmV0ID09PSBBTFBIQUJFVC5STkEpID9cbiAgICAgIFtSSUJPU0UsIFBIT1NQSEFURV0gOiBbREVPWFlSSUJPU0UsIFBIT1NQSEFURV07XG4gICAgZm9yIChjb25zdCBzeW0gb2Ygc3ltYm9scylcbiAgICAgIHVwZGF0ZU1vbm9tZXJzRGljdChtb25vbWVyc0RpY3QsIHN5bSwgZm9ybWF0dGVkTW9ub21lckxpYiwgbW9kdWxlUmRraXQsIHBvbHltZXJUeXBlKTtcbiAgfVxuXG4gIGZvciAobGV0IHJvdyA9IDA7IHJvdyA8IG1vbm9tZXJTZXF1ZW5jZXNBcnJheS5sZW5ndGg7ICsrcm93KSB7XG4gICAgY29uc3QgbW9ub21lclNlcTogc3RyaW5nW10gPSBtb25vbWVyU2VxdWVuY2VzQXJyYXlbcm93XTtcbiAgICBmb3IgKGNvbnN0IHN5bSBvZiBtb25vbWVyU2VxKVxuICAgICAgdXBkYXRlTW9ub21lcnNEaWN0KG1vbm9tZXJzRGljdCwgc3ltLCBmb3JtYXR0ZWRNb25vbWVyTGliLCBtb2R1bGVSZGtpdCwgcG9seW1lclR5cGUpO1xuICB9XG4gIC8vIGNvbnNvbGUubG9nKG1vbm9tZXJzRGljdCk7XG5cbiAgcmV0dXJuIG1vbm9tZXJzRGljdDtcbn1cblxuLyogQWRkcyBNb2xHcmFwaCBvYmplY3QgZm9yICdzeW0nIHRvIHRoZSBtb25vbWVycyBkaWN0IHdoZW4gbmVjZXNzYXJ5ICAqL1xuZnVuY3Rpb24gdXBkYXRlTW9ub21lcnNEaWN0KFxuICBtb25vbWVyc0RpY3Q6IE1hcDxzdHJpbmcsIE1vbEdyYXBoPiwgc3ltOiBzdHJpbmcsXG4gIGZvcm1hdHRlZE1vbm9tZXJMaWI6IE1hcDxzdHJpbmcsIGFueT4sIG1vZHVsZVJka2l0OiBhbnksIHBvbHltZXJUeXBlOiBIRUxNX1BPTFlNRVJfVFlQRVxuKTogdm9pZCB7XG4gIGlmICghbW9ub21lcnNEaWN0LmhhcyhzeW0pKSB7XG4gICAgY29uc3QgbW9ub21lckRhdGE6IE1vbEdyYXBoIHwgbnVsbCA9IGdldE1vbEdyYXBoKHN5bSwgZm9ybWF0dGVkTW9ub21lckxpYiwgbW9kdWxlUmRraXQsIHBvbHltZXJUeXBlKTtcbiAgICBpZiAobW9ub21lckRhdGEpXG4gICAgICBtb25vbWVyc0RpY3Quc2V0KHN5bSwgbW9ub21lckRhdGEpO1xuICAgIGVsc2VcbiAgICAgIHRocm93IG5ldyBFcnJvcihgTW9ub21lciB3aXRoIHN5bWJvbCAnJHtzeW19JyBpcyBhYnNlbnQgdGhlIG1vbm9tZXIgbGlicmFyeWApO1xuICAgIC8vIHRvZG86IGhhbmRsZSBleGNlcHRpb25cbiAgfVxufVxuXG4vKiBDb25zdHJ1Y3QgdGhlIE1vbEdyYXBoIG9iamVjdCBmb3Igc3BlY2lmaWVkIG1vbm9tZXJTeW1ib2w6IHRoZSBhc3NvY2lhdGVkXG4gKiBncmFwaCBpcyBhZGp1c3RlZCBpbiBYWSBwbGFuZSBhbmQgZmlsbGVkIHdpdGggZGVmYXVsdCBSLWdyb3VwcyAqL1xuZnVuY3Rpb24gZ2V0TW9sR3JhcGgoXG4gIG1vbm9tZXJTeW1ib2w6IHN0cmluZywgZm9ybWF0dGVkTW9ub21lckxpYjogTWFwPHN0cmluZywgYW55PixcbiAgbW9kdWxlUmRraXQ6IGFueSwgcG9seW1lclR5cGU6IEhFTE1fUE9MWU1FUl9UWVBFIC8vIHRvZG86IHNwZWNpZnkgdHlwZSBmb3IgbW9kdWxlUmRraXRcbik6IE1vbEdyYXBoIHwgbnVsbCB7XG4gIGlmICghZm9ybWF0dGVkTW9ub21lckxpYi5oYXMobW9ub21lclN5bWJvbCkpIHtcbiAgICByZXR1cm4gbnVsbDtcbiAgfSBlbHNlIHtcbiAgICBjb25zdCBsaWJPYmplY3QgPSBmb3JtYXR0ZWRNb25vbWVyTGliLmdldChtb25vbWVyU3ltYm9sKTtcbiAgICBjb25zdCBjYXBHcm91cHMgPSBwYXJzZUNhcEdyb3VwcyhsaWJPYmplY3RbSEVMTV9GSUVMRFMuUkdST1VQU10pO1xuICAgIGNvbnN0IGNhcEdyb3VwSWR4TWFwID0gcGFyc2VDYXBHcm91cElkeE1hcChsaWJPYmplY3RbSEVMTV9GSUVMRFMuTU9MRklMRV0pO1xuICAgIGNvbnN0IG1vbGZpbGVWM0sgPSBjb252ZXJ0TW9sZmlsZVRvVjNLKHJlbW92ZVJHcm91cExpbmVzKGxpYk9iamVjdFtIRUxNX0ZJRUxEUy5NT0xGSUxFXSksIG1vZHVsZVJka2l0KTtcbiAgICBjb25zdCBjb3VudHMgPSBwYXJzZUF0b21BbmRCb25kQ291bnRzKG1vbGZpbGVWM0spO1xuXG4gICAgY29uc3QgYXRvbXMgPSBwYXJzZUF0b21CbG9jayhtb2xmaWxlVjNLLCBjb3VudHMuYXRvbUNvdW50KTtcbiAgICBjb25zdCBib25kcyA9IHBhcnNlQm9uZEJsb2NrKG1vbGZpbGVWM0ssIGNvdW50cy5ib25kQ291bnQpO1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNb25vbWVyTWV0YWRhdGEoYXRvbXMsIGJvbmRzLCBjYXBHcm91cHMsIGNhcEdyb3VwSWR4TWFwKTtcblxuICAgIGNvbnN0IG1vbm9tZXJHcmFwaDogTW9sR3JhcGggPSB7YXRvbXM6IGF0b21zLCBib25kczogYm9uZHMsIG1ldGE6IG1ldGF9O1xuXG4gICAgaWYgKHBvbHltZXJUeXBlID09PSBIRUxNX1BPTFlNRVJfVFlQRS5QRVBUSURFKSB7XG4gICAgICBhZGp1c3RQZXB0aWRlTW9ub21lckdyYXBoKG1vbm9tZXJHcmFwaCk7XG4gICAgfSBlbHNlIHsgLy8gbnVjbGVvdGlkZXNcbiAgICAgIGlmIChtb25vbWVyU3ltYm9sID09PSBSSUJPU0UgfHwgbW9ub21lclN5bWJvbCA9PT0gREVPWFlSSUJPU0UpXG4gICAgICAgIGFkanVzdFN1Z2FyTW9ub21lckdyYXBoKG1vbm9tZXJHcmFwaCk7XG4gICAgICBlbHNlIGlmIChtb25vbWVyU3ltYm9sID09PSBQSE9TUEhBVEUpXG4gICAgICAgIGFkanVzdFBob3NwaGF0ZU1vbm9tZXJHcmFwaChtb25vbWVyR3JhcGgpO1xuICAgICAgZWxzZVxuICAgICAgICBhZGp1c3RCYXNlTW9ub21lckdyYXBoKG1vbm9tZXJHcmFwaCk7XG4gICAgfVxuXG4gICAgLy8gcmVtb3ZlIHRoZSAncmlnaHRtb3N0JyBjaGFpbi1leHRlbmRpbmcgci1ncm91cCBub2RlIGluIHRoZSBiYWNrYm9uZVxuICAgIGlmIChwb2x5bWVyVHlwZSA9PT0gSEVMTV9QT0xZTUVSX1RZUEUuUEVQVElERSkge1xuICAgICAgc2V0U2hpZnRzKG1vbm9tZXJHcmFwaCwgcG9seW1lclR5cGUpO1xuICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzFdKTtcbiAgICB9IGVsc2UgeyAvLyBudWNsZW90aWRlc1xuICAgICAgaWYgKG1vbm9tZXJTeW1ib2wgPT09IFJJQk9TRSB8fCBtb25vbWVyU3ltYm9sID09PSBERU9YWVJJQk9TRSkge1xuICAgICAgICAvLyByZW1vdmUgUjJcbiAgICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzFdKTtcbiAgICAgICAgLy8gc2V0IHRlcm1pbmFsTm9kZTIgKG94eWdlbikgYXMgbmV3IFIyXG4gICAgICAgIG1vbm9tZXJHcmFwaC5tZXRhLnJOb2Rlc1sxXSA9IG1vbm9tZXJHcmFwaC5tZXRhLnRlcm1pbmFsTm9kZXNbMV07XG4gICAgICAgIHNldFRlcm1pbmFsTm9kZXMobW9ub21lckdyYXBoLmJvbmRzLCBtb25vbWVyR3JhcGgubWV0YSk7IC8vIHNldCB0ZXJtaW5hbCBub2RlcyBhbmV3XG4gICAgICAgIHNldFNoaWZ0cyhtb25vbWVyR3JhcGgsIHBvbHltZXJUeXBlKTtcbiAgICAgICAgLy8gcmVtb3ZlICduZXcnIFIyIChveHlnZW4pXG4gICAgICAgIHJlbW92ZU5vZGVBbmRCb25kcyhtb25vbWVyR3JhcGgsIG1vbm9tZXJHcmFwaC5tZXRhLnJOb2Rlc1sxXSk7XG4gICAgICAgIC8vIHJlbW92ZSBSMVxuICAgICAgICByZW1vdmVOb2RlQW5kQm9uZHMobW9ub21lckdyYXBoLCBtb25vbWVyR3JhcGgubWV0YS5yTm9kZXNbMF0pO1xuICAgICAgICAvLyByZW1vdmUgdGhlIGJyYW5jaGluZyByLWdyb3VwXG4gICAgICAgIHJlbW92ZU5vZGVBbmRCb25kcyhtb25vbWVyR3JhcGgsIG1vbm9tZXJHcmFwaC5tZXRhLnJOb2Rlc1syXSk7XG4gICAgICB9IGVsc2UgaWYgKG1vbm9tZXJTeW1ib2wgPT09IFBIT1NQSEFURSkge1xuICAgICAgICBtb25vbWVyR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdID0gbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzBdO1xuICAgICAgICBzaGlmdENvb3JkaW5hdGVzKFxuICAgICAgICAgIG1vbm9tZXJHcmFwaCxcbiAgICAgICAgICAtbW9ub21lckdyYXBoLmF0b21zLnhbbW9ub21lckdyYXBoLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDFdLFxuICAgICAgICAgIC1tb25vbWVyR3JhcGguYXRvbXMueVttb25vbWVyR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMV1cbiAgICAgICAgKTtcbiAgICAgICAgc2V0U2hpZnRzKG1vbm9tZXJHcmFwaCwgcG9seW1lclR5cGUpO1xuICAgICAgICByZW1vdmVOb2RlQW5kQm9uZHMobW9ub21lckdyYXBoLCBtb25vbWVyR3JhcGgubWV0YS5yTm9kZXNbMV0pO1xuICAgICAgfSBlbHNlIHsgLy8gbnVjbGVvYmFzZXNcbiAgICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzBdKTtcbiAgICAgIH1cbiAgICB9XG4gICAgcmVtb3ZlSHlkcm9nZW4obW9ub21lckdyYXBoKTtcblxuICAgIHJldHVybiBtb25vbWVyR3JhcGg7XG4gIH1cbn1cblxuLy8gdG9kbzogZG9jXG5mdW5jdGlvbiBnZXRNb25vbWVyTWV0YWRhdGEoXG4gIGF0b21zOiBBdG9tcywgYm9uZHM6IEJvbmRzLCBjYXBHcm91cHM6IHN0cmluZ1tdLCBjYXBHcm91cElkeE1hcDogTWFwPG51bWJlciwgbnVtYmVyPlxuKTogTW9ub21lck1ldGFkYXRhIHtcbiAgY29uc3QgbWV0YTogTW9ub21lck1ldGFkYXRhID0ge1xuICAgIGJhY2tib25lU2hpZnQ6IG51bGwsXG4gICAgYnJhbmNoU2hpZnQ6IG51bGwsXG4gICAgdGVybWluYWxOb2RlczogW10sXG4gICAgck5vZGVzOiBbXSxcbiAgfTtcblxuICBzdWJzdGl0dXRlQ2FwR3JvdXBzKGF0b21zLCBjYXBHcm91cHMsIGNhcEdyb3VwSWR4TWFwKTtcbiAgc2V0Uk5vZGVzKGNhcEdyb3VwSWR4TWFwLCBtZXRhKTtcbiAgc2V0VGVybWluYWxOb2Rlcyhib25kcywgbWV0YSk7XG4gIHJldHVybiBtZXRhO1xufVxuXG4vKiBQYXJzZSBlbGVtZW50IHN5bWJvbHMgZm9yIFItZ3JvdXBzIGZyb20gdGhlIEhFTE0gbW9ub21lciBsaWJyYXJ5IFItZ3JvdXBzXG4gKiBmaWVsZCAgKi9cbmZ1bmN0aW9uIHBhcnNlQ2FwR3JvdXBzKHJHcm91cE9iakxpc3Q6IGFueVtdKTogc3RyaW5nW10ge1xuICAvLyBzcGVjaWZpY2FsbHkgZm9yIEhFTE1Db3JlTGlicmFyeVxuICAvLyBjb25zaWRlcmVkIG9ubHkgbW9ub2F0b21pYyByZ3JvdXBzXG4gIC8vIHN1cHBvc2luZyB0aGF0IGVsZW1lbnRzIGluIHJHcm91cE9iakxpc3QgYXJlIHNvcnRlZCB3LnIudC4gdGhlIHJncm91cHMgaWR4XG4gIC8vIHRvZG86IHBvc3NpYmxlIGdlbmVyYWxpemF0aW9uc1xuICBjb25zdCBjYXBHcm91cHNBcnJheTogc3RyaW5nW10gPSBbXTtcbiAgZm9yIChjb25zdCBvYmogb2Ygckdyb3VwT2JqTGlzdCkge1xuICAgIGxldCBjYXBHcm91cDogc3RyaW5nID0gb2JqW1JHUk9VUF9GSUVMRFMuQ0FQX0dST1VQX1NNSUxFU107XG5cbiAgICAvLyBpbiBzb21lIGNhc2VzIHRoZSBzbWlsZXMgZmllbGQgaXMgd3JpdHRlbiB3aXRoIHVwcGVyY2FzZVxuICAgIGlmICghY2FwR3JvdXApXG4gICAgICBjYXBHcm91cCA9IG9ialtSR1JPVVBfRklFTERTLkNBUF9HUk9VUF9TTUlMRVNfVVBQRVJDQVNFXTtcbiAgICAvLyB0b2RvOiB2ZXJpZnkgdGhhdCB0aGVyZSBhcmUgbm8gbXVsdGktZWxlbWVudCBjYXAgZ3JvdXBzLCBvciBjb25zaWRlciBob3cgdG9cbiAgICAvLyB0cmFuc2Zvcm0gdGhlbVxuICAgIGNhcEdyb3VwID0gY2FwR3JvdXAucmVwbGFjZSgvKFxcW3xcXF18XFwqfDp8XFxkKS9nLCAnJyk7XG4gICAgaWYgKGNhcEdyb3VwLmxlbmd0aCA+IDEpIC8vIHRvZG86IGNoZWNrIGlmIHN1Y2ggY2FzZXMgYXJlIHBvc3NpYmxlLCByZW1vdmUgaWYgbm90XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0RlZmF1bHQgY2FwIGdyb3VwIGhhcyBsZW5ndGggbW9yZSB0aGFuIG9uZScpO1xuICAgIGNhcEdyb3Vwc0FycmF5LnB1c2goY2FwR3JvdXApO1xuICB9XG4gIHJldHVybiBjYXBHcm91cHNBcnJheTtcbn1cblxuLyogU3Vic3RpdHV0ZSB0aGUgY2FwIGdyb3VwIGVsZW1lbnRzIGluc3RlYWQgb2YgUiMgKi9cbmZ1bmN0aW9uIHN1YnN0aXR1dGVDYXBHcm91cHMoXG4gIGF0b21zOiBBdG9tcywgY2FwR3JvdXBzOiBzdHJpbmdbXSwgY2FwR3JvdXBJZHhNYXA6IE1hcDxudW1iZXIsIG51bWJlcj5cbik6IHZvaWQge1xuICBmb3IgKGNvbnN0IFtub2RlLCBjYXBJZHhdIG9mIGNhcEdyb3VwSWR4TWFwKVxuICAgIGF0b21zLmF0b21UeXBlc1tub2RlIC0gMV0gPSBjYXBHcm91cHNbY2FwSWR4IC0gMV07IC8vIC0xIGJlY2F1c2UgbW9sZmlsZSBpbmRleGluZyBzdGFydHMgZnJvbSAxXG59XG5cbi8vdG9kbzogZG9jXG5mdW5jdGlvbiBzZXRSTm9kZXMoY2FwR3JvdXBJZHhNYXA6IE1hcDxudW1iZXIsIG51bWJlcj4sIG1ldGE6IE1vbm9tZXJNZXRhZGF0YSk6IHZvaWQge1xuICBtZXRhLnJOb2RlcyA9IEFycmF5LmZyb20oY2FwR3JvdXBJZHhNYXAua2V5cygpKTtcbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBtZXRhLnJOb2Rlcy5sZW5ndGg7IGkrKykge1xuICAgIGZvciAoY29uc3QgaiBvZiBbMSwgMl0pIHsgLy8gMSBhbmQgMiBieSBkZWYuIGNvcnJlc3BvbmQgdG8gJ2xlZnQvcmlnaHRtb3N0JyByLW5vZGVzXG4gICAgICAvLyBzd2FwIHRoZSB2YWx1ZXMgaWYgbmVjZXNzYXJ5LCBzbyB0aGF0IHRoZSBcImxlZnRtb3N0XCIgci1ub2RlIGlzIGF0IDAsXG4gICAgICAvLyBhbmQgdGhlICdyaWdodG1vc3QnLCBhdCAxXG4gICAgICBpZiAoY2FwR3JvdXBJZHhNYXAuZ2V0KG1ldGEuck5vZGVzW2ldKSA9PT0gaikge1xuICAgICAgICBjb25zdCB0bXAgPSBtZXRhLnJOb2Rlc1tqIC0gMV07XG4gICAgICAgIG1ldGEuck5vZGVzW2ogLSAxXSA9IG1ldGEuck5vZGVzW2ldO1xuICAgICAgICBtZXRhLnJOb2Rlc1tpXSA9IHRtcDtcbiAgICAgIH1cbiAgICB9XG4gIH1cbn1cblxuLy90b2RvOiBkb2NcbmZ1bmN0aW9uIHNldFRlcm1pbmFsTm9kZXMoYm9uZHM6IEJvbmRzLCBtZXRhOiBNb25vbWVyTWV0YWRhdGEpOiB2b2lkIHtcbiAgY29uc3Qgck5vZGVzID0gbWV0YS5yTm9kZXM7XG4gIG1ldGEudGVybWluYWxOb2RlcyA9IG5ldyBBcnJheTxudW1iZXI+KHJOb2Rlcy5sZW5ndGgpLmZpbGwoMCk7XG4gIGNvbnN0IHRlcm1pbmFsTm9kZXMgPSBtZXRhLnRlcm1pbmFsTm9kZXM7XG4gIGNvbnN0IGF0b21QYWlycyA9IGJvbmRzLmF0b21QYWlycztcbiAgbGV0IGkgPSAwO1xuICBsZXQgaiA9IDA7XG4gIHdoaWxlICgoaSA8IGF0b21QYWlycy5sZW5ndGgpICYmIGogPCB0ZXJtaW5hbE5vZGVzLmxlbmd0aCkge1xuICAgIC8vIHJOb2RlcyBhcnJheSBpcyBzb3J0ZWQgc28gdGhhdCBpdHMgMHRoIGFuZCAxc3QgZWxlbWVudHMgKGlmIGJvdGhcbiAgICAvLyBwcmVzZW50KSBjb3JyZXNwb25kIHRvIHRoZSBjaGFpbiBleHRlbmRpbmcgKGkuZS4gbm90IGJyYW5jaGluZykgci1ncm91cHNcbiAgICBmb3IgKGxldCBrID0gMDsgayA8IHRlcm1pbmFsTm9kZXMubGVuZ3RoOyArK2spIHtcbiAgICAgIGZvciAobGV0IGwgPSAwOyBsIDwgMjsgKytsKSB7XG4gICAgICAgIGlmIChhdG9tUGFpcnNbaV1bbF0gPT09IHJOb2Rlc1trXSkge1xuICAgICAgICAgIHRlcm1pbmFsTm9kZXNba10gPSBhdG9tUGFpcnNbaV1bKGwgKyAxKSUyXTtcbiAgICAgICAgICBpZiAock5vZGVzLmxlbmd0aCA+IDIpIHtcbiAgICAgICAgICB9XG4gICAgICAgICAgKytqO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICAgICsraTtcbiAgfVxufVxuXG4vL3RvZG86IGRvY1xuZnVuY3Rpb24gc2V0U2hpZnRzKG1vbEdyYXBoOiBNb2xHcmFwaCwgcG9seW1lclR5cGU6IEhFTE1fUE9MWU1FUl9UWVBFKTogdm9pZCB7XG4gIGlmIChtb2xHcmFwaC5tZXRhLnJOb2Rlcy5sZW5ndGggPiAxKSB7XG4gICAgbW9sR3JhcGgubWV0YS5iYWNrYm9uZVNoaWZ0ID0gW1xuICAgICAga2VlcFByZWNpc2lvbihcbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnJOb2Rlc1sxXSAtIDFdIC1cbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnRlcm1pbmFsTm9kZXNbMF0gLSAxXVxuICAgICAgKSxcbiAgICAgIGtlZXBQcmVjaXNpb24oXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS5yTm9kZXNbMV0gLSAxXSAtXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMV1cbiAgICAgICksXG4gICAgXTtcbiAgfVxuXG4gIGlmIChwb2x5bWVyVHlwZSA9PT0gSEVMTV9QT0xZTUVSX1RZUEUuUk5BICYmIG1vbEdyYXBoLm1ldGEuck5vZGVzLmxlbmd0aCA+IDIpIHtcbiAgICBtb2xHcmFwaC5tZXRhLmJyYW5jaFNoaWZ0ID0gW1xuICAgICAga2VlcFByZWNpc2lvbihcbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnJOb2Rlc1syXSAtIDFdIC1cbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnRlcm1pbmFsTm9kZXNbMF0gLSAxXVxuICAgICAgKSxcbiAgICAgIGtlZXBQcmVjaXNpb24oXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS5yTm9kZXNbMl0gLSAxXSAtXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMV1cbiAgICAgICksXG4gICAgXTtcbiAgfVxufVxuXG4vKiBIZWxwZXIgZnVuY3Rpb24gbmVjZXNzYXJ5IHRvIGJ1aWxkIGEgY29ycmVjdCBWMzAwMCBtb2xmaWxlIG91dCBvZiBWMjAwMCB3aXRoXG4gKiBzcGVjaWZpZWQgci1ncm91cHMqL1xuZnVuY3Rpb24gcmVtb3ZlUkdyb3VwTGluZXMobW9sZmlsZVYySzogc3RyaW5nKTogc3RyaW5nIHtcbiAgbGV0IGJlZ2luID0gbW9sZmlsZVYySy5pbmRleE9mKFYyS19BX0xJTkUsIDApO1xuICBpZiAoYmVnaW4gPT09IC0xKVxuICAgIGJlZ2luID0gbW9sZmlsZVYySy5pbmRleE9mKFYyS19SR1BfTElORSk7XG4gIGNvbnN0IGVuZCA9IG1vbGZpbGVWMksuaW5kZXhPZihWM0tfRU5ELCBiZWdpbik7XG4gIHJldHVybiBtb2xmaWxlVjJLLnN1YnN0cmluZygwLCBiZWdpbikgKyBtb2xmaWxlVjJLLnN1YnN0cmluZyhlbmQpO1xufVxuXG4vKiBWMjAwMCB0byBWMzAwMCBjb252ZXJ0ZXIgICovXG5mdW5jdGlvbiBjb252ZXJ0TW9sZmlsZVRvVjNLKG1vbGZpbGVWMks6IHN0cmluZywgbW9kdWxlUmRraXQ6IGFueSk6IHN0cmluZyB7XG4gIC8vIHRvZG86IHR5cGUgb2YgbW9kdWxlUmRraXRcbiAgLy8gdG9kbzogY29uc2lkZXIgdGhlIHVzZSBvZiBzdGFuZGFyZCBDaGVtIGNvbnZlcnRlciAocmVsaWVzIG9uIGNyZWF0aW9uIG9mIG1vZHVsZVJka2l0IG9uIGVhY2ggaXRlcmF0aW9uLCB0aG91Z2gpXG4gIGNvbnN0IG1vbE9iaiA9IG1vZHVsZVJka2l0LmdldF9tb2wobW9sZmlsZVYySyk7XG4gIGNvbnN0IG1vbGZpbGVWM0sgPSBtb2xPYmouZ2V0X3YzS21vbGJsb2NrKCk7XG4gIG1vbE9iai5kZWxldGUoKTtcbiAgcmV0dXJuIG1vbGZpbGVWM0s7XG59XG5cbi8qIFBhcnNlIFYzMDAwIGJvbmQgYmxvY2sgYW5kIGNvbnN0cnVjdCB0aGUgQm9uZHMgb2JqZWN0ICovXG5mdW5jdGlvbiBwYXJzZUJvbmRCbG9jayhtb2xmaWxlVjNLOiBzdHJpbmcsIGJvbmRDb3VudDogbnVtYmVyKTogQm9uZHMge1xuICAvLyB0b2RvOiBjb25zaWRlciB0aGUgY2FzZSB3aGVuIHRoZXJlIGlzIG5vIHNpbXBsZSBsZWZ0bW9zdC9yaWdodG1vc3QgY2hvaWNlXG4gIC8vIHRvZG86IGNvbnNpZGVyIHRoZSBjYXNlIHdoZW4gdGhlcmUgYXJlIG11bHRpcGxlIGNvbnNlcXVlbnQgTSAgUkdQIGxpbmVzLFxuICAvLyBsaWtlIGluIEhFTE1Db3JlTGlicmFyeSBudWNsZW90aWRlc1xuXG4gIGNvbnN0IGJvbmRUeXBlczogbnVtYmVyW10gPSBuZXcgQXJyYXkoYm9uZENvdW50KTtcbiAgY29uc3QgYXRvbVBhaXJzOiBudW1iZXJbXVtdID0gbmV3IEFycmF5KGJvbmRDb3VudCk7XG4gIGNvbnN0IGJvbmRDb25maWd1cmF0aW9uID0gbmV3IE1hcDxudW1iZXIsIG51bWJlcj4oKTtcbiAgY29uc3Qga3dhcmdzID0gbmV3IE1hcDxudW1iZXIsIHN0cmluZz47XG5cbiAgbGV0IGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKFYzS19CRUdJTl9CT05EX0JMT0NLKTtcbiAgYmVnaW4gPSBtb2xmaWxlVjNLLmluZGV4T2YoJ1xcbicsIGJlZ2luKTtcbiAgbGV0IGVuZCA9IGJlZ2luO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IGJvbmRDb3VudDsgKytpKSB7XG4gICAgLy8gcGFyc2UgYm9uZCB0eXBlIGFuZCBhdG9tIHBhaXJcbiAgICBjb25zdCBwYXJzZWRWYWx1ZXM6IG51bWJlcltdID0gbmV3IEFycmF5KDMpO1xuICAgIGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKFYzS19CRUdJTl9EQVRBX0xJTkUsIGVuZCkgKyBWM0tfSURYX1NISUZUO1xuICAgIGVuZCA9IG1vbGZpbGVWM0suaW5kZXhPZignICcsIGJlZ2luKTtcbiAgICBmb3IgKGxldCBrID0gMDsgayA8IDM7ICsraykge1xuICAgICAgYmVnaW4gPSBlbmQgKyAxO1xuICAgICAgZW5kID0gTWF0aC5taW4obW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbiksIG1vbGZpbGVWM0suaW5kZXhPZignICcsIGJlZ2luKSk7XG4gICAgICBwYXJzZWRWYWx1ZXNba10gPSBwYXJzZUludChtb2xmaWxlVjNLLnNsaWNlKGJlZ2luLCBlbmQpKTtcbiAgICB9XG4gICAgYm9uZFR5cGVzW2ldID0gcGFyc2VkVmFsdWVzWzBdO1xuICAgIGF0b21QYWlyc1tpXSA9IHBhcnNlZFZhbHVlcy5zbGljZSgxKTtcblxuICAgIC8vIHBhcnNlIGtleXdvcmQgYXJndW1lbnRzXG4gICAgY29uc3QgZW5kT2ZMaW5lID0gbW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gICAgbGV0IGxpbmVSZW1haW5kZXIgPSBtb2xmaWxlVjNLLnNsaWNlKGVuZCwgZW5kT2ZMaW5lKTtcbiAgICBsZXQgYmVnaW5DZmcgPSBsaW5lUmVtYWluZGVyLmluZGV4T2YoVjNLX0JPTkRfQ09ORklHKTtcbiAgICBpZiAoYmVnaW5DZmcgIT09IC0xKSB7XG4gICAgICBiZWdpbkNmZyA9IGxpbmVSZW1haW5kZXIuaW5kZXhPZignPScsIGJlZ2luQ2ZnKSArIDE7XG4gICAgICBsZXQgZW5kQ2ZnID0gbGluZVJlbWFpbmRlci5pbmRleE9mKCcgJywgYmVnaW5DZmcpO1xuICAgICAgaWYgKGVuZENmZyA9PT0gLTEpXG4gICAgICAgIGVuZENmZyA9IGxpbmVSZW1haW5kZXIubGVuZ3RoO1xuICAgICAgY29uc3QgYm9uZENvbmZpZyA9IHBhcnNlSW50KGxpbmVSZW1haW5kZXIuc2xpY2UoYmVnaW5DZmcsIGVuZENmZykpO1xuICAgICAgYm9uZENvbmZpZ3VyYXRpb24uc2V0KGksIGJvbmRDb25maWcpO1xuICAgICAgY29uc3QgcmVtb3ZlZFN1YnN0cmluZyA9IFYzS19CT05EX0NPTkZJRyArIGJvbmRDb25maWcudG9TdHJpbmcoKTtcbiAgICAgIGxpbmVSZW1haW5kZXIgPSBsaW5lUmVtYWluZGVyLnJlcGxhY2UocmVtb3ZlZFN1YnN0cmluZywgJycpO1xuICAgIH1cbiAgICBpZiAoIWxpbmVSZW1haW5kZXIpXG4gICAgICBrd2FyZ3Muc2V0KGksIGxpbmVSZW1haW5kZXIpO1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBib25kVHlwZXM6IGJvbmRUeXBlcyxcbiAgICBhdG9tUGFpcnM6IGF0b21QYWlycyxcbiAgICBib25kQ29uZmlndXJhdGlvbjogYm9uZENvbmZpZ3VyYXRpb24sXG4gICAga3dhcmdzOiBrd2FyZ3MsXG4gIH07XG59XG5cbi8qIENvbnN0cnVjdHMgbWFwcGluZyBvZiByLWdyb3VwIG5vZGVzIHRvIGRlZmF1bHQgY2FwR3JvdXBzLCBhbGwgbnVtZXJhdGlvbiBzdGFydGluZyBmcm9tIDEuXG4gKiBBY2NvcmRpbmcgdG8gaHR0cHM6Ly9wdWJzLmFjcy5vcmcvZG9pLzEwLjEwMjEvY2kzMDAxOTI1LCBSMSBhbmQgUjIgYXJlIHRoZSBjaGFpbiBleHRlbmRpbmcgYXR0YWNobWVudCBwb2ludHMsXG4gKiB3aGlsZSBSMyBpcyB0aGUgYnJhbmNoaW5nIGF0dGFjaG1lbnQgcG9pbnQuICovXG5mdW5jdGlvbiBwYXJzZUNhcEdyb3VwSWR4TWFwKG1vbGZpbGVWMks6IHN0cmluZyk6IE1hcDxudW1iZXIsIG51bWJlcj4ge1xuICBjb25zdCBjYXBHcm91cElkeE1hcCA9IG5ldyBNYXA8bnVtYmVyLCBudW1iZXI+KCk7XG5cbiAgLy8gcGFyc2UgQS1saW5lcyAoUk5BKVxuICBsZXQgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoVjJLX0FfTElORSwgMCk7XG4gIGxldCBlbmQgPSBiZWdpbjtcbiAgd2hpbGUgKGJlZ2luICE9PSAtMSkge1xuICAgIC8vIHBhcnNlIHRoZSByTm9kZSB0byB3aGljaCB0aGUgY2FwIGdyb3VwIGlzIGF0dGFjaGVkXG4gICAgZW5kID0gbW9sZmlsZVYySy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gICAgY29uc3Qgck5vZGUgPSBwYXJzZUludChtb2xmaWxlVjJLLnN1YnN0cmluZyhiZWdpbiwgZW5kKS5yZXBsYWNlKC9eQVxccysvLCAnJykpO1xuXG4gICAgLy8gcGFyc2UgdGhlIGNhcEdyb3VwIGluZGV4XG4gICAgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoJ1InLCBlbmQpO1xuICAgIGVuZCA9IG1vbGZpbGVWMksuaW5kZXhPZignXFxuJywgYmVnaW4pO1xuICAgIGNvbnN0IGNhcEdyb3VwID0gcGFyc2VJbnQobW9sZmlsZVYySy5zdWJzdHJpbmcoYmVnaW4sIGVuZCkucmVwbGFjZSgvXlIvLCAnJykpO1xuICAgIGNhcEdyb3VwSWR4TWFwLnNldChyTm9kZSwgY2FwR3JvdXApO1xuXG4gICAgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoVjJLX0FfTElORSwgZW5kKTtcbiAgfVxuXG4gIC8vIHBhcnNlIFJHUCBsaW5lcyAobWF5IGJlIG1vcmUgdGhhbiBvbmUgaW4gUk5BIG1vbm9tZXJzKVxuICBiZWdpbiA9IG1vbGZpbGVWMksuaW5kZXhPZihWMktfUkdQX0xJTkUsIDApO1xuICBlbmQgPSBtb2xmaWxlVjJLLmluZGV4T2YoJ1xcbicsIGJlZ2luKTtcbiAgd2hpbGUgKGJlZ2luICE9PSAtMSkge1xuICAgIGJlZ2luICs9IFYyS19SR1BfU0hJRlQ7XG4gICAgZW5kID0gbW9sZmlsZVYySy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gICAgY29uc3QgcmdwU3RyaW5nUGFyc2VkID0gbW9sZmlsZVYySy5zdWJzdHJpbmcoYmVnaW4sIGVuZClcbiAgICAgIC5yZXBsYWNlQWxsKC9cXHMrL2csICcgJylcbiAgICAgIC5zcGxpdCgnICcpO1xuICAgIGNvbnN0IHJncEluZGljZXNBcnJheSA9IHJncFN0cmluZ1BhcnNlZC5tYXAoKGVsKSA9PiBwYXJzZUludChlbCkpXG4gICAgICAuc2xpY2UoMSk7IC8vIHNsaWNlIGZyb20gMSBiZWNhdXNlIHRoZSAxc3QgdmFsdWUgaXMgdGhlIG51bWJlciBvZiBwYWlycyBpbiB0aGUgbGluZVxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgcmdwSW5kaWNlc0FycmF5Lmxlbmd0aDsgaSArPSAyKSB7XG4gICAgICAvLyBub3RpY2U6IHRoZXJlIG1heSBiZSBjb25mbGljdGluZyBjYXAgZ3JvdXAgZGVmaW5pdGlvbnMsIGxpa2UgMy1PLU1ldGh5bHJpYm9zZSAoMiw1IGNvbm5lY3Rpdml0eSlcbiAgICAgIC8vICh0aGUgbGFzdCBtb25vbWVyIGluIEhFTE1Db3JlTGlicmFyeSlcbiAgICAgIC8vIHRoZXJlIHRoZSBpbmRpY2VzIG9mIGNhcCBncm91cHMgYXJlIHNlbGYtY29udHJhZGljdG9yeVxuICAgICAgLy8gdG9kbzogY2xhcmlmeSB3aHkgc3VjaCBzaXR1YXRpb25zIG9jY3VyIGluIHByaW5jaXBsZVxuICAgICAgaWYgKGNhcEdyb3VwSWR4TWFwLmhhcyhyZ3BJbmRpY2VzQXJyYXlbaV0pICYmIGNhcEdyb3VwSWR4TWFwLmdldChyZ3BJbmRpY2VzQXJyYXlbaV0pICE9PSByZ3BJbmRpY2VzQXJyYXlbaSArIDFdKVxuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYHItZ3JvdXAgaW5kZXggJHtyZ3BJbmRpY2VzQXJyYXlbaV19IGhhcyBhbHJlYWR5IGJlZW4gYWRkZWQgd2l0aCBhIGRpZmZlcmVudCB2YWx1ZWApO1xuICAgICAgZWxzZVxuICAgICAgICBjYXBHcm91cElkeE1hcC5zZXQocmdwSW5kaWNlc0FycmF5W2ldLCByZ3BJbmRpY2VzQXJyYXlbaSsxXSk7XG4gICAgfVxuXG4gICAgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoVjJLX1JHUF9MSU5FLCBlbmQpO1xuICB9XG5cbiAgcmV0dXJuIGNhcEdyb3VwSWR4TWFwO1xufVxuXG5mdW5jdGlvbiBwYXJzZUF0b21BbmRCb25kQ291bnRzKG1vbGZpbGVWM0s6IHN0cmluZyk6IHthdG9tQ291bnQ6IG51bWJlciwgYm9uZENvdW50OiBudW1iZXJ9IHtcbiAgbW9sZmlsZVYzSyA9IG1vbGZpbGVWM0sucmVwbGFjZUFsbCgnXFxyJywgJycpOyAvLyB0byBoYW5kbGUgb2xkIGFuZCBuZXcgc2RmIHN0YW5kYXJkc1xuXG4gIC8vIHBhcnNlIGF0b20gY291bnRcbiAgbGV0IGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKFYzS19CRUdJTl9DT1VOVFNfTElORSkgKyBWM0tfQ09VTlRTX1NISUZUO1xuICBsZXQgZW5kID0gbW9sZmlsZVYzSy5pbmRleE9mKCcgJywgYmVnaW4pO1xuICBjb25zdCBudW1PZkF0b21zID0gcGFyc2VJbnQobW9sZmlsZVYzSy5zdWJzdHJpbmcoYmVnaW4sIGVuZCkpO1xuXG4gIC8vIHBhcnNlIGJvbmQgY291bnRcbiAgYmVnaW4gPSBlbmQgKyAxO1xuICBlbmQgPSBtb2xmaWxlVjNLLmluZGV4T2YoJyAnLCBiZWdpbik7XG4gIGNvbnN0IG51bU9mQm9uZHMgPSBwYXJzZUludChtb2xmaWxlVjNLLnN1YnN0cmluZyhiZWdpbiwgZW5kKSk7XG5cbiAgcmV0dXJuIHthdG9tQ291bnQ6IG51bU9mQXRvbXMsIGJvbmRDb3VudDogbnVtT2ZCb25kc307XG59XG5cbi8qIFBhcnNlIFYzMDAwIGF0b20gYmxvY2sgYW5kIHJldHVybiBBdG9tcyBvYmplY3QuIE5PVElDRTogb25seSBhdG9tVHlwZXMsIHgsIHlcbiAqIGFuZCBrd2FyZ3MgZmllbGRzIGFyZSBzZXQgaW4gdGhlIHJldHVybiB2YWx1ZSwgd2l0aCBvdGhlciBmaWVsZHMgZHVtbXkgKi9cbmZ1bmN0aW9uIHBhcnNlQXRvbUJsb2NrKG1vbGZpbGVWM0s6IHN0cmluZywgYXRvbUNvdW50OiBudW1iZXIpOiBBdG9tcyB7XG4gIGNvbnN0IGF0b21UeXBlczogc3RyaW5nW10gPSBuZXcgQXJyYXkoYXRvbUNvdW50KTtcbiAgY29uc3QgeDogbnVtYmVyW10gPSBuZXcgQXJyYXkoYXRvbUNvdW50KTtcbiAgY29uc3QgeTogbnVtYmVyW10gPSBuZXcgQXJyYXkoYXRvbUNvdW50KTtcbiAgY29uc3Qga3dhcmdzOiBzdHJpbmdbXSA9IG5ldyBBcnJheShhdG9tQ291bnQpO1xuXG4gIGxldCBiZWdpbiA9IG1vbGZpbGVWM0suaW5kZXhPZihWM0tfQkVHSU5fQVRPTV9CTE9DSyk7IC8vIFYzMDAwIGF0b21zIGJsb2NrXG4gIGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gIGxldCBlbmQgPSBiZWdpbjtcblxuICBmb3IgKGxldCBpID0gMDsgaSA8IGF0b21Db3VudDsgaSsrKSB7XG4gICAgYmVnaW4gPSBtb2xmaWxlVjNLLmluZGV4T2YoVjNLX0JFR0lOX0RBVEFfTElORSwgYmVnaW4pICsgVjNLX0lEWF9TSElGVDtcbiAgICBlbmQgPSBtb2xmaWxlVjNLLmluZGV4T2YoJyAnLCBiZWdpbik7IC8vIHNraXAgdGhlIGlkeCByb3dcblxuICAgIC8vIHBhcnNlIGF0b20gdHlwZVxuICAgIGJlZ2luID0gZW5kICsgMTtcbiAgICBlbmQgPSBtb2xmaWxlVjNLLmluZGV4T2YoJyAnLCBiZWdpbik7XG4gICAgYXRvbVR5cGVzW2ldID0gbW9sZmlsZVYzSy5zdWJzdHJpbmcoYmVnaW4sIGVuZCk7XG5cbiAgICAvLyBwYXJzZSBYIGFuZCBZIGNvb3JkaW5hdGVzIG9mIHRoZSBhdG9tXG4gICAgY29uc3QgY29vcmRpbmF0ZTogbnVtYmVyW10gPSBuZXcgQXJyYXkoMik7XG4gICAgZm9yIChsZXQgayA9IDA7IGsgPCAyOyArK2spIHtcbiAgICAgIGJlZ2luID0gZW5kICsgMTtcbiAgICAgIGVuZCA9IG1vbGZpbGVWM0suaW5kZXhPZignICcsIGJlZ2luKTtcbiAgICAgIGNvb3JkaW5hdGVba10gPSBwYXJzZUZsb2F0KG1vbGZpbGVWM0suc3Vic3RyaW5nKGJlZ2luLCBlbmQpKTtcbiAgICB9XG4gICAgeFtpXSA9IGNvb3JkaW5hdGVbMF07XG4gICAgeVtpXSA9IGNvb3JkaW5hdGVbMV07XG5cbiAgICAvLyBwYXJzZSB0aGUgcmVtYWluaW5nIHBvc3NpYmxlIGtleXdvcmQgYXJndW1lbnRzXG4gICAgYmVnaW4gPSBlbmQ7XG4gICAgZW5kID0gbW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbikgKyAxO1xuICAgIGt3YXJnc1tpXSA9IG1vbGZpbGVWM0suc2xpY2UoYmVnaW4sIGVuZCk7XG5cbiAgICBiZWdpbiA9IGVuZDtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgYXRvbVR5cGVzOiBhdG9tVHlwZXMsXG4gICAgeDogeCxcbiAgICB5OiB5LFxuICAgIGt3YXJnczoga3dhcmdzLFxuICB9O1xufVxuXG4vKiBSZW1vdmUgaHlkcm9nZW4gbm9kZXMgKi9cbmZ1bmN0aW9uIHJlbW92ZUh5ZHJvZ2VuKG1vbm9tZXJHcmFwaDogTW9sR3JhcGgpOiB2b2lkIHtcbiAgbGV0IGkgPSAwO1xuICB3aGlsZSAoaSA8IG1vbm9tZXJHcmFwaC5hdG9tcy5hdG9tVHlwZXMubGVuZ3RoKSB7XG4gICAgaWYgKCBtb25vbWVyR3JhcGguYXRvbXMuYXRvbVR5cGVzW2ldID09PSBIWURST0dFTikge1xuICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgaSArIDEpOyAvLyBpICsgMSBiZWNhdXNlIG1vbGZpbGUgbm9kZSBpbmRleGluZyBzdGFydHMgZnJvbSAxXG4gICAgICAtLWk7XG4gICAgICAvLyBtb25vbWVyR3JhcGguYXRvbXMuYXRvbVR5cGVzW2ldID0gJ0xpJztcbiAgICB9XG4gICAgKytpO1xuICB9XG59XG5cbi8qIFJlbW92ZSBub2RlICdyZW1vdmVkTm9kZScgYW5kIHRoZSBhc3NvY2lhdGVkIGJvbmRzLiBOb3RpY2UsIG51bWVyYXRpb24gb2ZcbiAqIG5vZGVzIGluIG1vbGZpbGVzIHN0YXJ0cyBmcm9tIDEsIG5vdCAwICovXG5mdW5jdGlvbiByZW1vdmVOb2RlQW5kQm9uZHMobW9ub21lckdyYXBoOiBNb2xHcmFwaCwgcmVtb3ZlZE5vZGU/OiBudW1iZXIpOiB2b2lkIHtcbiAgaWYgKHR5cGVvZiByZW1vdmVkTm9kZSAhPT0gJ3VuZGVmaW5lZCcpIHtcbiAgICBjb25zdCByZW1vdmVkTm9kZUlkeCA9IHJlbW92ZWROb2RlIC0gMTtcbiAgICBjb25zdCBhdG9tcyA9IG1vbm9tZXJHcmFwaC5hdG9tcztcbiAgICBjb25zdCBib25kcyA9IG1vbm9tZXJHcmFwaC5ib25kcztcbiAgICBjb25zdCBtZXRhID0gbW9ub21lckdyYXBoLm1ldGE7XG5cbiAgICAvLyByZW1vdmUgdGhlIG5vZGUgZnJvbSBhdG9tc1xuICAgIGF0b21zLmF0b21UeXBlcy5zcGxpY2UocmVtb3ZlZE5vZGVJZHgsIDEpO1xuICAgIGF0b21zLnguc3BsaWNlKHJlbW92ZWROb2RlSWR4LCAxKTtcbiAgICBhdG9tcy55LnNwbGljZShyZW1vdmVkTm9kZUlkeCwgMSk7XG4gICAgYXRvbXMua3dhcmdzLnNwbGljZShyZW1vdmVkTm9kZUlkeCwgMSk7XG5cbiAgICAvLyB1cGRhdGUgdGhlIHZhbHVlcyBvZiB0ZXJtaW5hbCBhbmQgci1ncm91cCBub2RlcyBpZiBuZWNlc3NhcnlcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IG1ldGEudGVybWluYWxOb2Rlcy5sZW5ndGg7ICsraSkge1xuICAgICAgaWYgKG1ldGEudGVybWluYWxOb2Rlc1tpXSA+IHJlbW92ZWROb2RlKVxuICAgICAgICAtLW1ldGEudGVybWluYWxOb2Rlc1tpXTtcbiAgICAgIGVsc2UgaWYgKG1ldGEudGVybWluYWxOb2Rlc1tpXSA9PT0gcmVtb3ZlZE5vZGUpXG4gICAgICAgIG1ldGEudGVybWluYWxOb2Rlc1tpXSA9IC0xOyAvLyBzZW50aW5lbCB0byBtYXJrIHRoZSB2YWx1ZSBhcyByZW1vdmVkXG4gICAgfVxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgbWV0YS5yTm9kZXMubGVuZ3RoOyArK2kpIHtcbiAgICAgIGlmIChtZXRhLnJOb2Rlc1tpXSA+IHJlbW92ZWROb2RlKVxuICAgICAgICAtLW1ldGEuck5vZGVzW2ldO1xuICAgICAgZWxzZSBpZiAobWV0YS5yTm9kZXNbaV0gPT09IHJlbW92ZWROb2RlKVxuICAgICAgICBtZXRhLnJOb2Rlc1tpXSA9IC0xOyAvLyBzZW50aW5lbCB0byBtYXJrIHRoZSB2YWx1ZSBhcyByZW1vdmVkXG4gICAgfVxuXG4gICAgLy8gdXBkYXRlIGluZGljZXMgb2YgYXRvbXMgaW4gYm9uZHNcbiAgICBsZXQgaSA9IDA7XG4gICAgd2hpbGUgKGkgPCBib25kcy5hdG9tUGFpcnMubGVuZ3RoKSB7XG4gICAgICBjb25zdCBmaXJzdEF0b20gPSBib25kcy5hdG9tUGFpcnNbaV1bMF07XG4gICAgICBjb25zdCBzZWNvbmRBdG9tID0gYm9uZHMuYXRvbVBhaXJzW2ldWzFdO1xuICAgICAgaWYgKGZpcnN0QXRvbSA9PT0gcmVtb3ZlZE5vZGUgfHwgc2Vjb25kQXRvbSA9PT0gcmVtb3ZlZE5vZGUpIHtcbiAgICAgICAgYm9uZHMuYXRvbVBhaXJzLnNwbGljZShpLCAxKTtcbiAgICAgICAgYm9uZHMuYm9uZFR5cGVzLnNwbGljZShpLCAxKTtcbiAgICAgICAgaWYgKGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmhhcyhpKSlcbiAgICAgICAgICBib25kcy5ib25kQ29uZmlndXJhdGlvbi5kZWxldGUoaSk7XG4gICAgICAgIGlmIChib25kcy5rd2FyZ3MuaGFzKGkpKVxuICAgICAgICAgIGJvbmRzLmt3YXJncy5kZWxldGUoaSk7XG4gICAgICAgIC0taTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGJvbmRzLmF0b21QYWlyc1tpXVswXSA9IChmaXJzdEF0b20gPiByZW1vdmVkTm9kZSkgPyBmaXJzdEF0b20gLSAxIDogZmlyc3RBdG9tO1xuICAgICAgICBib25kcy5hdG9tUGFpcnNbaV1bMV0gPSAoc2Vjb25kQXRvbSA+IHJlbW92ZWROb2RlKSA/IHNlY29uZEF0b20gLSAxIDogc2Vjb25kQXRvbTtcbiAgICAgIH1cbiAgICAgICsraTtcbiAgICB9XG5cbiAgICAvLyB1cGRhdGUgYm9uZENvbmZpZ3VyYXRpb24gYW5kIGt3YXJncyBrZXlzXG4gICAgbGV0IGtleXMgPSBBcnJheS5mcm9tKGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmtleXMoKSk7XG4gICAga2V5cy5mb3JFYWNoKChrZXkpID0+IHtcbiAgICAgIGlmIChib25kcy5ib25kQ29uZmlndXJhdGlvbi5oYXMoa2V5KSAmJiBrZXkgPiByZW1vdmVkTm9kZUlkeCkge1xuICAgICAgICBjb25zdCB2YWx1ZSA9IGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmdldChrZXkpITtcbiAgICAgICAgYm9uZHMuYm9uZENvbmZpZ3VyYXRpb24uZGVsZXRlKGtleSk7XG4gICAgICAgIGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLnNldChrZXkgLSAxLCB2YWx1ZSk7XG4gICAgICB9XG4gICAgfSk7XG4gICAga2V5cyA9IEFycmF5LmZyb20oYm9uZHMua3dhcmdzLmtleXMoKSk7XG4gICAga2V5cy5mb3JFYWNoKChrZXkpID0+IHtcbiAgICAgIGlmIChib25kcy5rd2FyZ3MuaGFzKGtleSkgJiYga2V5ID4gcmVtb3ZlZE5vZGVJZHgpIHtcbiAgICAgICAgY29uc3QgdmFsdWUgPSBib25kcy5rd2FyZ3MuZ2V0KGtleSkhO1xuICAgICAgICBib25kcy5rd2FyZ3MuZGVsZXRlKGtleSk7XG4gICAgICAgIGJvbmRzLmt3YXJncy5zZXQoa2V5IC0gMSwgdmFsdWUpO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG59XG5cbi8vIHRvZG86IHJld3JpdGUgZGVzY3JpcHRpb25cbi8qIEFkanVzdCB0aGUgKHBlcHRpZGUpIG1vbm9tZXIgZ3JhcGggc28gdGhhdCBpdCBoYXMgc3RhbmRhcmQgZm9ybSAgKi9cbmZ1bmN0aW9uIGFkanVzdFBlcHRpZGVNb25vbWVyR3JhcGgobW9ub21lcjogTW9sR3JhcGgpOiB2b2lkIHtcbiAgY29uc3Qgbm9kZU9uZUlkeCA9IG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMTsgLy8gbm9kZSBpbmRleGluZyBpbiBtb2xmaWxlcyBzdGFydHMgZnJvbSAxXG4gIGNvbnN0IG5vZGVUd29JZHggPSBtb25vbWVyLm1ldGEuck5vZGVzWzBdIC0gMTtcbiAgY29uc3QgeCA9IG1vbm9tZXIuYXRvbXMueDtcbiAgY29uc3QgeSA9IG1vbm9tZXIuYXRvbXMueTtcblxuICAvLyBwbGFjZSBub2RlT25lIGF0IG9yaWdpblxuICBzaGlmdENvb3JkaW5hdGVzKG1vbm9tZXIsIC14W25vZGVPbmVJZHhdLCAteVtub2RlT25lSWR4XSk7XG5cbiAgLy8gYW5nbGUgaXMgbWVhc3VyZWQgYmV0d2VlbiBPWSBhbmQgdGhlIHJvdGF0ZWQgbm9kZVxuICBjb25zdCBhbmdsZSA9IGZpbmRBbmdsZVdpdGhPWSh4W25vZGVUd29JZHhdLCB5W25vZGVUd29JZHhdKTtcblxuICAvLyByb3RhdGUgdGhlIGNlbnRlcmVkIGdyYXBoLCBzbyB0aGF0ICdub2RlVHdvJyBlbmRzIHVwIG9uIHRoZSBwb3NpdGl2ZSByYXkgb2YgT1lcbiAgcm90YXRlQ2VudGVyZWRHcmFwaChtb25vbWVyLmF0b21zLCAtYW5nbGUpO1xuXG4gIGlmICh4W21vbm9tZXIubWV0YS5yTm9kZXNbMV0gLSAxXSA8IDApXG4gICAgZmxpcE1vbm9tZXJBcm91bmRPWShtb25vbWVyKTtcblxuICBjb25zdCBkb3VibGVCb25kZWRPeHlnZW4gPSBmaW5kRG91YmxlQm9uZGVkQ2FyYm9ueWxPeHlnZW4obW9ub21lcik7XG5cbiAgLy8gZmxpcCBjYXJib3h5bCBhbmQgUiBpZiBuZWNlc3NhcnlcbiAgZmxpcENhcmJveHlsQW5kUmFkaWNhbChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xuXG4gIC8vIGZsaXAgaHlkcm94eWwgZ3JvdXAgd2l0aCBkb3VibGUtYm91bmQgTyBpbnNpZGUgY2FyYm94eWwgZ3JvdXAgaWYgbmVjZXNzYXJ5XG4gIGZsaXBIeWRyb3hpbEdyb3VwKG1vbm9tZXIsIGRvdWJsZUJvbmRlZE94eWdlbik7XG59XG5cbmZ1bmN0aW9uIGFkanVzdFBob3NwaGF0ZU1vbm9tZXJHcmFwaChtb25vbWVyOiBNb2xHcmFwaCk6dm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBtb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDE7IC8vIG5vZGUgaW5kZXhpbmcgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMVxuICBjb25zdCBub2RlVHdvSWR4ID0gbW9ub21lci5tZXRhLnJOb2Rlc1swXSAtIDE7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIGNvbnN0IHkgPSBtb25vbWVyLmF0b21zLnk7XG5cbiAgLy8gcGxhY2Ugbm9kZU9uZSBhdCBvcmlnaW5cbiAgc2hpZnRDb29yZGluYXRlcyhtb25vbWVyLCAteFtub2RlT25lSWR4XSwgLXlbbm9kZU9uZUlkeF0pO1xuXG4gIC8vIC8vIGFuZ2xlIGlzIG1lYXN1cmVkIGJldHdlZW4gT1kgYW5kIHRoZSByb3RhdGVkIG5vZGVcbiAgLy8gY29uc3QgYW5nbGUgPSBmaW5kQW5nbGVXaXRoT1koeFtub2RlVHdvSWR4XSwgeVtub2RlVHdvSWR4XSk7XG5cbiAgLy8gLy8gcm90YXRlIHRoZSBjZW50ZXJlZCBncmFwaCwgc28gdGhhdCAnbm9kZVR3bycgZW5kcyB1cCBvbiB0aGUgcG9zaXRpdmUgcmF5IG9mIE9ZXG4gIC8vIHJvdGF0ZUNlbnRlcmVkR3JhcGgobW9ub21lci5hdG9tcywgLWFuZ2xlKTtcblxuICAvLyBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwKVxuICAvLyAgIGZsaXBNb25vbWVyQXJvdW5kT1kobW9ub21lcik7XG5cbiAgLy8gY29uc3QgZG91YmxlQm9uZGVkT3h5Z2VuID0gZmluZERvdWJsZUJvbmRlZENhcmJvbnlsT3h5Z2VuKG1vbm9tZXIpO1xuXG4gIC8vIC8vIGZsaXAgY2FyYm94eWwgYW5kIFIgaWYgbmVjZXNzYXJ5XG4gIC8vIGZsaXBDYXJib3h5bEFuZFJhZGljYWwobW9ub21lciwgZG91YmxlQm9uZGVkT3h5Z2VuKTtcblxuICAvLyAvLyBmbGlwIGh5ZHJveHlsIGdyb3VwIHdpdGggZG91YmxlLWJvdW5kIE8gaW5zaWRlIGNhcmJveHlsIGdyb3VwIGlmIG5lY2Vzc2FyeVxuICAvLyBmbGlwSHlkcm94aWxHcm91cChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xufVxuXG5mdW5jdGlvbiBhZGp1c3RTdWdhck1vbm9tZXJHcmFwaChtb25vbWVyOiBNb2xHcmFwaCk6dm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBtb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDE7IC8vIG5vZGUgaW5kZXhpbmcgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMVxuICBjb25zdCBub2RlVHdvSWR4ID0gbW9ub21lci5tZXRhLnJOb2Rlc1swXSAtIDE7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIGNvbnN0IHkgPSBtb25vbWVyLmF0b21zLnk7XG5cbiAgLy8gcGxhY2Ugbm9kZU9uZSBhdCBvcmlnaW5cbiAgc2hpZnRDb29yZGluYXRlcyhtb25vbWVyLCAteFtub2RlT25lSWR4XSwgLXlbbm9kZU9uZUlkeF0pO1xuXG4gIC8vIC8vIGFuZ2xlIGlzIG1lYXN1cmVkIGJldHdlZW4gT1kgYW5kIHRoZSByb3RhdGVkIG5vZGVcbiAgLy8gY29uc3QgYW5nbGUgPSBmaW5kQW5nbGVXaXRoT1koeFtub2RlVHdvSWR4XSwgeVtub2RlVHdvSWR4XSk7XG5cbiAgLy8gLy8gcm90YXRlIHRoZSBjZW50ZXJlZCBncmFwaCwgc28gdGhhdCAnbm9kZVR3bycgZW5kcyB1cCBvbiB0aGUgcG9zaXRpdmUgcmF5IG9mIE9ZXG4gIC8vIHJvdGF0ZUNlbnRlcmVkR3JhcGgobW9ub21lci5hdG9tcywgLWFuZ2xlKTtcblxuICAvLyBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwKVxuICAvLyAgIGZsaXBNb25vbWVyQXJvdW5kT1kobW9ub21lcik7XG5cbiAgLy8gY29uc3QgZG91YmxlQm9uZGVkT3h5Z2VuID0gZmluZERvdWJsZUJvbmRlZENhcmJvbnlsT3h5Z2VuKG1vbm9tZXIpO1xuXG4gIC8vIC8vIGZsaXAgY2FyYm94eWwgYW5kIFIgaWYgbmVjZXNzYXJ5XG4gIC8vIGZsaXBDYXJib3h5bEFuZFJhZGljYWwobW9ub21lciwgZG91YmxlQm9uZGVkT3h5Z2VuKTtcblxuICAvLyAvLyBmbGlwIGh5ZHJveHlsIGdyb3VwIHdpdGggZG91YmxlLWJvdW5kIE8gaW5zaWRlIGNhcmJveHlsIGdyb3VwIGlmIG5lY2Vzc2FyeVxuICAvLyBmbGlwSHlkcm94aWxHcm91cChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xufVxuZnVuY3Rpb24gYWRqdXN0QmFzZU1vbm9tZXJHcmFwaChtb25vbWVyOiBNb2xHcmFwaCk6dm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBtb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDE7IC8vIG5vZGUgaW5kZXhpbmcgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMVxuICBjb25zdCBub2RlVHdvSWR4ID0gbW9ub21lci5tZXRhLnJOb2Rlc1swXSAtIDE7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIGNvbnN0IHkgPSBtb25vbWVyLmF0b21zLnk7XG5cbiAgLy8gcGxhY2Ugbm9kZU9uZSBhdCBvcmlnaW5cbiAgc2hpZnRDb29yZGluYXRlcyhtb25vbWVyLCAteFtub2RlT25lSWR4XSwgLXlbbm9kZU9uZUlkeF0pO1xuXG4gIC8vIC8vIGFuZ2xlIGlzIG1lYXN1cmVkIGJldHdlZW4gT1kgYW5kIHRoZSByb3RhdGVkIG5vZGVcbiAgLy8gY29uc3QgYW5nbGUgPSBmaW5kQW5nbGVXaXRoT1koeFtub2RlVHdvSWR4XSwgeVtub2RlVHdvSWR4XSk7XG5cbiAgLy8gLy8gcm90YXRlIHRoZSBjZW50ZXJlZCBncmFwaCwgc28gdGhhdCAnbm9kZVR3bycgZW5kcyB1cCBvbiB0aGUgcG9zaXRpdmUgcmF5IG9mIE9ZXG4gIC8vIHJvdGF0ZUNlbnRlcmVkR3JhcGgobW9ub21lci5hdG9tcywgLWFuZ2xlKTtcblxuICAvLyBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwKVxuICAvLyAgIGZsaXBNb25vbWVyQXJvdW5kT1kobW9ub21lcik7XG5cbiAgLy8gY29uc3QgZG91YmxlQm9uZGVkT3h5Z2VuID0gZmluZERvdWJsZUJvbmRlZENhcmJvbnlsT3h5Z2VuKG1vbm9tZXIpO1xuXG4gIC8vIC8vIGZsaXAgY2FyYm94eWwgYW5kIFIgaWYgbmVjZXNzYXJ5XG4gIC8vIGZsaXBDYXJib3h5bEFuZFJhZGljYWwobW9ub21lciwgZG91YmxlQm9uZGVkT3h5Z2VuKTtcblxuICAvLyAvLyBmbGlwIGh5ZHJveHlsIGdyb3VwIHdpdGggZG91YmxlLWJvdW5kIE8gaW5zaWRlIGNhcmJveHlsIGdyb3VwIGlmIG5lY2Vzc2FyeVxuICAvLyBmbGlwSHlkcm94aWxHcm91cChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xufVxuXG4vKiBGbGlwIGNhcmJveHlsIGdyb3VwIHdpdGggdGhlIHJhZGljYWwgaW4gYSBwZXB0aWRlIG1vbm9tZXIgaW4gY2FzZSB0aGVcbiAqIGNhcmJveHlsIGdyb3VwIGlzIGluIHRoZSBsb3dlciBoYWxmLXBsYW5lICovXG5mdW5jdGlvbiBmbGlwQ2FyYm94eWxBbmRSYWRpY2FsKG1vbm9tZXI6IE1vbEdyYXBoLCBkb3VibGVCb25kZWRPeHlnZW46IG51bWJlcik6IHZvaWQge1xuICAvLyB2ZXJpZnkgdGhhdCB0aGUgY2FyYm94eWwgZ3JvdXAgaXMgaW4gdGhlIGxvd2VyIGhhbGYtcGxhbmVcbiAgaWYgKG1vbm9tZXIuYXRvbXMueVttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwICYmXG4gICAgbW9ub21lci5hdG9tcy55W2RvdWJsZUJvbmRlZE94eWdlbiAtIDFdIDwgMCkge1xuICAgIGZsaXBNb25vbWVyQXJvdW5kT1gobW9ub21lcik7XG5cbiAgICByb3RhdGVDZW50ZXJlZEdyYXBoKG1vbm9tZXIuYXRvbXMsXG4gICAgICAtZmluZEFuZ2xlV2l0aE9YKFxuICAgICAgICBtb25vbWVyLmF0b21zLnhbbW9ub21lci5tZXRhLnRlcm1pbmFsTm9kZXNbMV0gLSAxXSxcbiAgICAgICAgbW9ub21lci5hdG9tcy55W21vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzFdIC0gMV1cbiAgICAgIClcbiAgICApO1xuICB9XG59XG5cbi8qIEZpbmRzIGFuZ2xlIGJldHdlZW4gT1kgYW5kIHRoZSByYXkgam9pbmluZyBvcmlnaW4gd2l0aCAoeCwgeSkgKi9cbmZ1bmN0aW9uIGZpbmRBbmdsZVdpdGhPWSh4OiBudW1iZXIsIHk6IG51bWJlcik6IG51bWJlciB7XG4gIGxldCBhbmdsZTtcbiAgaWYgKHggPT09IDApIHtcbiAgICBhbmdsZSA9IHkgPiAwID8gMCA6IE1hdGguUEk7XG4gIH0gZWxzZSBpZiAoeSA9PT0gMCkge1xuICAgIGFuZ2xlID0geCA+IDAgPyAtTWF0aC5QSS8yIDogTWF0aC5QSS8yO1xuICB9IGVsc2Uge1xuICAgIGNvbnN0IHRhbiA9IHkgLyB4O1xuICAgIGNvbnN0IGF0YW4gPSBNYXRoLmF0YW4odGFuKTtcbiAgICBhbmdsZSA9ICh4IDwgMCkgPyBNYXRoLlBJLzIgKyBhdGFuIDogLU1hdGguUEkvMiArIGF0YW47XG4gIH1cbiAgcmV0dXJuIGFuZ2xlO1xufVxuXG4vKiBGaW5kcyBhbmdsZSBiZXR3ZWVuIE9YIGFuZCB0aGUgcmF5IGpvaW5pbmcgb3JpZ2luIHdpdGggKHgsIHkpICovXG5mdW5jdGlvbiBmaW5kQW5nbGVXaXRoT1goeDogbnVtYmVyLCB5OiBudW1iZXIpOiBudW1iZXIge1xuICByZXR1cm4gZmluZEFuZ2xlV2l0aE9ZKHgsIHkpICsgTWF0aC5QSS8yO1xufVxuXG4vKiAgUm90YXRlIHRoZSBncmFwaCBhcm91bmQgdGhlIG9yaWdpbiBieSAnYW5nbGUnICovXG5mdW5jdGlvbiByb3RhdGVDZW50ZXJlZEdyYXBoKGF0b21zOiBBdG9tcywgYW5nbGU6IG51bWJlcik6IHZvaWQge1xuICBpZiAoYW5nbGUgIT09IDApIHtcbiAgICBjb25zdCB4ID0gYXRvbXMueDtcbiAgICBjb25zdCB5ID0gYXRvbXMueTtcblxuICAgIGNvbnN0IGNvcyA9IE1hdGguY29zKGFuZ2xlKTtcbiAgICBjb25zdCBzaW4gPSBNYXRoLnNpbihhbmdsZSk7XG5cbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IHgubGVuZ3RoOyArK2kpIHtcbiAgICAgIGNvbnN0IHRtcCA9IHhbaV07XG4gICAgICB4W2ldID0ga2VlcFByZWNpc2lvbih0bXAqY29zIC0geVtpXSpzaW4pO1xuICAgICAgeVtpXSA9IGtlZXBQcmVjaXNpb24odG1wKnNpbiArIHlbaV0qY29zKTtcbiAgICB9XG4gIH1cbn1cblxuLyogRmxpcCBtb25vbWVyIGdyYXBoIGFyb3VuZCBPWCBheGlzIHByZXNlcnZpbmcgc3RlcmVvbWV0cnkgKi9cbmZ1bmN0aW9uIGZsaXBNb25vbWVyQXJvdW5kT1gobW9ub21lcjogTW9sR3JhcGgpOiB2b2lkIHtcbiAgZmxpcE1vbEdyYXBoKG1vbm9tZXIsIHRydWUpO1xufVxuXG4vKiBGbGlwIG1vbm9tZXIgZ3JhcGggYXJvdW5kIE9ZIGF4aXMgcHJlc2VydmluZyBzdGVyZW9tZXRyeSAqL1xuZnVuY3Rpb24gZmxpcE1vbm9tZXJBcm91bmRPWShtb25vbWVyOiBNb2xHcmFwaCk6IHZvaWQge1xuICBmbGlwTW9sR3JhcGgobW9ub21lciwgZmFsc2UpO1xufVxuXG4vKiBGbGlwIGdyYXBoIGFyb3VuZCBhIHNwZWNpZmllZCBheGlzOiAndHJ1ZScgY29ycmVzcG9uZHMgdG8gT1gsICdmYWxzZScgdG8gT1kgKi9cbmZ1bmN0aW9uIGZsaXBNb2xHcmFwaChtb2xHcmFwaDogTW9sR3JhcGgsIGF4aXM6IGJvb2xlYW4pOiB2b2lkIHtcbiAgaWYgKGF4aXMpIHsgLy8gZmxpcHBpbmcgYXJvdW5kIE9YXG4gICAgY29uc3QgeSA9IG1vbEdyYXBoLmF0b21zLnk7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCB5Lmxlbmd0aDsgaSsrKVxuICAgICAgeVtpXSA9IC15W2ldO1xuICB9IGVsc2UgeyAvLyBmbGlwcGluZyBhcm91bmQgT1lcbiAgICBjb25zdCB4ID0gbW9sR3JhcGguYXRvbXMueDtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IHgubGVuZ3RoOyBpKyspXG4gICAgICB4W2ldID0gLXhbaV07XG4gIH1cblxuICAvLyBwcmVzZXJ2ZSB0aGUgc3RlcmVvbWV0cnlcbiAgY29uc3Qgb3JpZW50YXRpb24gPSBtb2xHcmFwaC5ib25kcy5ib25kQ29uZmlndXJhdGlvbjtcbiAgZm9yIChjb25zdCBba2V5LCB2YWx1ZV0gb2Ygb3JpZW50YXRpb24pIHtcbiAgICBjb25zdCBuZXdWYWx1ZSA9IHZhbHVlID09PSAxID8gMyA6IDE7XG4gICAgb3JpZW50YXRpb24uc2V0KGtleSwgbmV3VmFsdWUpO1xuICB9XG59XG5cbi8qIEZsaXBzIGRvdWJsZS1ib25kZWQgJ08nIGluIGNhcmJvbnlsIGdyb3VwIHdpdGggJ09IJyBpbiBvcmRlciBmb3IgdGhlIG1vbm9tZXJzXG4gKiB0byBoYXZlIHN0YW5kYXJkIHJlcHJlc2VudGF0aW9uIHNpbXBsaWZ5aW5nIHRoZWlyIGNvbmNhdGVuYXRpb24uIFRoZVxuICogbW9ub21lciBtdXN0IGFscmVhZHkgYmUgYWRqdXN0ZWQgd2l0aCBhZGp1c3RQZXB0aWRlTW9ub21lckdyYXBoIGluIG9yZGVyIGZvciB0aGlzIGZ1bmN0aW9uIHRvIGJlIGltcGxlbWVudGVkICAqL1xuZnVuY3Rpb24gZmxpcEh5ZHJveGlsR3JvdXAobW9ub21lcjogTW9sR3JhcGgsIGRvdWJsZUJvbmRlZE94eWdlbjogbnVtYmVyKTogdm9pZCB7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIC8vIC0xIGJlbG93IGJlY2F1c2UgaW5kZXhpbmcgb2Ygbm9kZXMgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMSwgdW5saWtlIGFycmF5c1xuICBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPiB4W2RvdWJsZUJvbmRlZE94eWdlbiAtIDFdKVxuICAgIHN3YXBOb2Rlcyhtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4sIG1vbm9tZXIubWV0YS5yTm9kZXNbMV0pO1xufVxuXG4vKiBEZXRlcm1pbmUgdGhlIG51bWJlciBvZiBub2RlIChzdGFydGluZyBmcm9tIDEpIGNvcnJlc3BvbmRpbmcgdG8gdGhlXG4gKiBkb3VibGUtYm9uZGVkIG94eWdlbiBvZiB0aGUgY2FyYm9ueWwgZ3JvdXAgICovXG5mdW5jdGlvbiBmaW5kRG91YmxlQm9uZGVkQ2FyYm9ueWxPeHlnZW4obW9ub21lcjogTW9sR3JhcGgpOiBudW1iZXIge1xuICBjb25zdCBib25kc01hcCA9IGNvbnN0cnVjdEJvbmRzTWFwKG1vbm9tZXIpO1xuICBsZXQgZG91YmxlQm9uZGVkT3h5Z2VuID0gMDtcbiAgbGV0IGkgPSAwO1xuICAvLyBpdGVyYXRlIG92ZXIgdGhlIG5vZGVzIGJvbmRlZCB0byB0aGUgY2FyYm9uIGFuZCBmaW5kIHRoZSBkb3VibGUgb25lXG4gIHdoaWxlIChkb3VibGVCb25kZWRPeHlnZW4gPT09IDApIHtcbiAgICBjb25zdCBub2RlID0gYm9uZHNNYXAuZ2V0KG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzFdKSFbaV07XG4gICAgaWYgKG1vbm9tZXIuYXRvbXMuYXRvbVR5cGVzW25vZGUgLSAxXSA9PT0gT1hZR0VOICYmIG5vZGUgIT09IG1vbm9tZXIubWV0YS5yTm9kZXNbMV0pXG4gICAgICBkb3VibGVCb25kZWRPeHlnZW4gPSBub2RlO1xuICAgIGkrKztcbiAgfVxuICByZXR1cm4gZG91YmxlQm9uZGVkT3h5Z2VuO1xufVxuXG4vKiBTd2FwIHRoZSBDYXJ0ZXNpYW4gY29vcmRpbmF0ZXMgb2YgdGhlIHR3byBzcGVjaWZpZWQgbm9kZXMgaW4gTW9sR3JhcGggICovXG5mdW5jdGlvbiBzd2FwTm9kZXMobW9ub21lcjogTW9sR3JhcGgsIG5vZGVPbmU6IG51bWJlciwgbm9kZVR3bzogbnVtYmVyKTogdm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBub2RlT25lIC0gMTtcbiAgY29uc3Qgbm9kZVR3b0lkeCA9IG5vZGVUd28gLSAxO1xuICBjb25zdCB4ID0gbW9ub21lci5hdG9tcy54O1xuICBjb25zdCB5ID0gbW9ub21lci5hdG9tcy55O1xuICBjb25zdCB0bXBYID0geFtub2RlT25lSWR4XTtcbiAgY29uc3QgdG1wWSA9IHlbbm9kZU9uZUlkeF07XG4gIHhbbm9kZU9uZUlkeF0gPSB4W25vZGVUd29JZHhdO1xuICB5W25vZGVPbmVJZHhdID0geVtub2RlVHdvSWR4XTtcbiAgeFtub2RlVHdvSWR4XSA9IHRtcFg7XG4gIHlbbm9kZVR3b0lkeF0gPSB0bXBZO1xufVxuXG4vLyB0b2RvOiBkb2NcbmZ1bmN0aW9uIGNvbnN0cnVjdEJvbmRzTWFwKG1vbm9tZXI6IE1vbEdyYXBoKTogTWFwPG51bWJlciwgQXJyYXk8bnVtYmVyPj4ge1xuICBjb25zdCBtYXAgPSBuZXcgTWFwPG51bWJlciwgQXJyYXk8bnVtYmVyPj4oKTtcbiAgZm9yIChjb25zdCBhdG9tUGFpcnMgb2YgbW9ub21lci5ib25kcy5hdG9tUGFpcnMpIHtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IDI7IGkrKykge1xuICAgICAgY29uc3Qga2V5ID0gYXRvbVBhaXJzW2ldO1xuICAgICAgY29uc3QgdmFsdWUgPSBhdG9tUGFpcnNbKGkgKyAxKSUyXTtcbiAgICAgIGlmIChtYXAuaGFzKGtleSkpXG4gICAgICAgIG1hcC5nZXQoa2V5KT8ucHVzaCh2YWx1ZSk7XG4gICAgICBlbHNlXG4gICAgICAgIG1hcC5zZXQoa2V5LCBuZXcgQXJyYXk8bnVtYmVyPigxKS5maWxsKHZhbHVlKSk7XG4gICAgfVxuICB9XG4gIHJldHVybiBtYXA7XG59XG5cbi8qIFNoaWZ0IG1vbEdyYXBoIGluIHRoZSBYT1kgcGxhbmUgICovXG5mdW5jdGlvbiBzaGlmdENvb3JkaW5hdGVzKG1vbEdyYXBoOiBNb2xHcmFwaCwgeFNoaWZ0OiBudW1iZXIsIHlTaGlmdD86IG51bWJlcik6IHZvaWQge1xuICBjb25zdCB4ID0gbW9sR3JhcGguYXRvbXMueDtcbiAgY29uc3QgeSA9IG1vbEdyYXBoLmF0b21zLnk7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgeC5sZW5ndGg7ICsraSkge1xuICAgIHhbaV0gPSBrZWVwUHJlY2lzaW9uKHhbaV0gKyB4U2hpZnQpO1xuICAgIGlmICh0eXBlb2YgeVNoaWZ0ICE9PSAndW5kZWZpbmVkJylcbiAgICAgIHlbaV0gPSBrZWVwUHJlY2lzaW9uKHlbaV0gKyB5U2hpZnQpO1xuICB9XG59XG5cbi8qIFRyYW5zbGF0ZSBhIHNlcXVlbmNlIG9mIG1vbm9tZXIgc3ltYm9scyBpbnRvIE1vbGZpbGUgVjMwMDAgKi9cbmZ1bmN0aW9uIG1vbm9tZXJTZXFUb01vbGZpbGUoXG4gIG1vbm9tZXJTZXE6IHN0cmluZ1tdLCBtb25vbWVyc0RpY3Q6IE1hcDxzdHJpbmcsIE1vbEdyYXBoPiwgYWxwaGFiZXQ6IEFMUEhBQkVULCBwb2x5bWVyVHlwZTogSEVMTV9QT0xZTUVSX1RZUEVcbik6IHN0cmluZyB7XG4gIC8vIHRvZG86IGhhbmRsZSB0aGUgY2FzZSB3aGVuIHRoZSBwb2x5bWVyIGlzIGVtcHR5XG4gIGlmIChtb25vbWVyU2VxLmxlbmd0aCA9PT0gMClcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ21vbm9tZXJTZXEgaXMgZW1wdHknKTtcblxuICAvLyBkZWZpbmUgYXRvbSBhbmQgYm9uZCBjb3VudHMsIHRha2luZyBpbnRvIGFjY291bnQgdGhlIGJvbmQgdHlwZVxuICBjb25zdCB7YXRvbUNvdW50LCBib25kQ291bnR9ID0gZ2V0UmVzdWx0aW5nQXRvbUJvbmRDb3VudHMobW9ub21lclNlcSwgbW9ub21lcnNEaWN0LCBhbHBoYWJldCwgcG9seW1lclR5cGUpO1xuXG4gIC8vIGNyZWF0ZSBhcnJheXMgdG8gc3RvcmUgbGluZXMgb2YgdGhlIHJlc3VsdGluZyBtb2xmaWxlXG4gIGNvbnN0IG1vbGZpbGVBdG9tQmxvY2sgPSBuZXcgQXJyYXk8c3RyaW5nPihhdG9tQ291bnQpO1xuICBjb25zdCBtb2xmaWxlQm9uZEJsb2NrID0gbmV3IEFycmF5PHN0cmluZz4oYm9uZENvdW50KTtcblxuICBsZXQgYWRkTW9ub21lclRvTW9sYmxvY2s7XG4gIGxldCBjYXBNb2xibG9jaztcbiAgbGV0IG5vZGVTaGlmdEluaXRWYWx1ZTtcbiAgbGV0IGJvbmRTaGlmdEluaXRWYWx1ZTtcbiAgbGV0IHN1Z2FyID0gbnVsbDtcbiAgbGV0IHBob3NwaGF0ZSA9IG51bGw7XG5cbiAgaWYgKHBvbHltZXJUeXBlID09PSBIRUxNX1BPTFlNRVJfVFlQRS5QRVBUSURFKSB7XG4gICAgYWRkTW9ub21lclRvTW9sYmxvY2sgPSBhZGRBbWlub0FjaWRUb01vbGJsb2NrO1xuICAgIGNhcE1vbGJsb2NrID0gY2FwUGVwdGlkZU1vbGJsb2NrO1xuICAgIG5vZGVTaGlmdEluaXRWYWx1ZSA9IGJvbmRTaGlmdEluaXRWYWx1ZSA9IDA7XG4gIH0gZWxzZSB7IC8vIG51Y2xlb3RpZGVzXG4gICAgYWRkTW9ub21lclRvTW9sYmxvY2sgPSBhZGROdWNsZW90aWRlVG9Nb2xibG9jaztcbiAgICBjYXBNb2xibG9jayA9IGNhcFBlcHRpZGVNb2xibG9jazsgLy8gdG9kbzogY2xlYW51cCAmIHJlZmFjdG9yXG4gICAgbm9kZVNoaWZ0SW5pdFZhbHVlID0gMDtcbiAgICBib25kU2hpZnRJbml0VmFsdWUgPSAwO1xuICAgIHN1Z2FyID0gKGFscGhhYmV0ID09PSBBTFBIQUJFVC5ETkEpID8gbW9ub21lcnNEaWN0LmdldChERU9YWVJJQk9TRSkgOiBtb25vbWVyc0RpY3QuZ2V0KFJJQk9TRSk7XG4gICAgcGhvc3BoYXRlID0gbW9ub21lcnNEaWN0LmdldChQSE9TUEhBVEUpO1xuICB9XG5cbiAgY29uc3QgdjogTG9vcFZhcmlhYmxlcyA9IHtcbiAgICBpOiAwLFxuICAgIG5vZGVTaGlmdDogbm9kZVNoaWZ0SW5pdFZhbHVlLFxuICAgIGJvbmRTaGlmdDogYm9uZFNoaWZ0SW5pdFZhbHVlLFxuICAgIGJhY2tib25lUG9zaXRpb25TaGlmdDogbmV3IEFycmF5PG51bWJlcj4oMikuZmlsbCgwKSxcbiAgICBicmFuY2hQb3NpdGlvblNoaWZ0OiBuZXcgQXJyYXk8bnVtYmVyPigyKS5maWxsKDApLFxuICAgIGJhY2tib25lQXR0YWNoTm9kZTogMCxcbiAgICBicmFuY2hBdHRhY2hOb2RlOiAwLFxuICAgIGZsaXBGYWN0b3I6IDEsXG4gIH07XG5cbiAgY29uc3QgQzogTG9vcENvbnN0YW50cyA9IHtcbiAgICBzdWdhcjogc3VnYXIhLFxuICAgIHBob3NwaGF0ZTogcGhvc3BoYXRlISxcbiAgICBzZXFMZW5ndGg6IG1vbm9tZXJTZXEubGVuZ3RoLFxuICAgIGF0b21Db3VudDogYXRvbUNvdW50LFxuICAgIGJvbmRDb3VudDogYm9uZENvdW50LFxuICB9O1xuXG4gIGZvciAodi5pID0gMDsgdi5pIDwgQy5zZXFMZW5ndGg7ICsrdi5pKSB7XG4gICAgY29uc3QgbW9ub21lciA9IG1vbm9tZXJzRGljdC5nZXQobW9ub21lclNlcVt2LmldKSE7XG4gICAgYWRkTW9ub21lclRvTW9sYmxvY2sobW9ub21lciwgbW9sZmlsZUF0b21CbG9jaywgbW9sZmlsZUJvbmRCbG9jaywgdiwgQyk7XG4gIH1cblxuICBjYXBNb2xibG9jayhtb2xmaWxlQXRvbUJsb2NrLCBtb2xmaWxlQm9uZEJsb2NrLCB2LCBDKTtcblxuICBjb25zdCBtb2xmaWxlQ291bnRzTGluZSA9IFYzS19CRUdJTl9DT1VOVFNfTElORSArIGF0b21Db3VudCArICcgJyArIGJvbmRDb3VudCArIFYzS19DT1VOVFNfTElORV9FTkRJTkc7XG5cbiAgLy8gdG9kbzogb3B0aW1pemUgY29uY2F0ZW5hdGlvbiB1c2luZyBBbGV4YW5kZXIncyBoaW50XG4gIGNvbnN0IG1vbGZpbGVQYXJ0cyA9IFtcbiAgICBWM0tfSEVBREVSX0ZJUlNUX0xJTkUsXG4gICAgVjNLX0hFQURFUl9TRUNPTkRfTElORSxcbiAgICBWM0tfQkVHSU5fQ1RBQl9CTE9DSyxcbiAgICBtb2xmaWxlQ291bnRzTGluZSxcbiAgICBWM0tfQkVHSU5fQVRPTV9CTE9DSyxcbiAgICBtb2xmaWxlQXRvbUJsb2NrLmpvaW4oJycpLFxuICAgIFYzS19FTkRfQVRPTV9CTE9DSyxcbiAgICBWM0tfQkVHSU5fQk9ORF9CTE9DSyxcbiAgICBtb2xmaWxlQm9uZEJsb2NrLmpvaW4oJycpLFxuICAgIFYzS19FTkRfQk9ORF9CTE9DSyxcbiAgICBWM0tfRU5EX0NUQUJfQkxPQ0ssXG4gICAgVjNLX0VORCxcbiAgXTtcblxuICByZXR1cm4gbW9sZmlsZVBhcnRzLmpvaW4oJycpO1xufVxuXG4vLyB0b2RvOiBkb2NcbmZ1bmN0aW9uIGNhcFBlcHRpZGVNb2xibG9jayhcbiAgbW9sZmlsZUF0b21CbG9jazogc3RyaW5nW10sIG1vbGZpbGVCb25kQmxvY2s6IHN0cmluZ1tdLFxuICB2OiBMb29wVmFyaWFibGVzLCBDOiBMb29wQ29uc3RhbnRzXG4pOiB2b2lkIHtcbiAgLy8gYWRkIHRlcm1pbmFsIG94eWdlblxuICBjb25zdCBhdG9tSWR4ID0gdi5ub2RlU2hpZnQgKyAxO1xuICBtb2xmaWxlQXRvbUJsb2NrW0MuYXRvbUNvdW50XSA9IFYzS19CRUdJTl9EQVRBX0xJTkUgKyBhdG9tSWR4ICsgJyAnICtcbiAgICBPWFlHRU4gKyAnICcgKyBrZWVwUHJlY2lzaW9uKHYuYmFja2JvbmVQb3NpdGlvblNoaWZ0WzBdKSArICcgJyArXG4gICAgdi5mbGlwRmFjdG9yICoga2VlcFByZWNpc2lvbih2LmJhY2tib25lUG9zaXRpb25TaGlmdFsxXSkgKyAnICcgKyAnMC4wMDAwMDAgMCcgKyAnXFxuJztcblxuICAvLyBhZGQgdGVybWluYWwgYm9uZFxuICBjb25zdCBmaXJzdEF0b20gPSB2LmJhY2tib25lQXR0YWNoTm9kZTtcbiAgY29uc3Qgc2Vjb25kQXRvbSA9IGF0b21JZHg7XG4gIG1vbGZpbGVCb25kQmxvY2tbQy5ib25kQ291bnRdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIHYuYm9uZFNoaWZ0ICsgJyAnICtcbiAgICAxICsgJyAnICsgZmlyc3RBdG9tICsgJyAnICsgc2Vjb25kQXRvbSArICdcXG4nO1xufVxuXG4vLyB0b2RvOiBkb2NcbmZ1bmN0aW9uIGFkZEFtaW5vQWNpZFRvTW9sYmxvY2sobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLFxuICBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcywgQzogTG9vcENvbnN0YW50c1xuKTogdm9pZCB7XG4gIHYuZmxpcEZhY3RvciA9ICgtMSkqKih2LmkgJSAyKTsgLy8gdG8gZmxpcCBldmVyeSBldmVuIG1vbm9tZXIgb3ZlciBPWFxuICBhZGRCYWNrYm9uZU1vbm9tZXJUb01vbGJsb2NrKG1vbm9tZXIsIG1vbGZpbGVBdG9tQmxvY2ssIG1vbGZpbGVCb25kQmxvY2ssIHYsIEMpO1xufVxuXG5mdW5jdGlvbiBhZGRCYWNrYm9uZU1vbm9tZXJUb01vbGJsb2NrKFxuICBtb25vbWVyOiBNb2xHcmFwaCwgbW9sZmlsZUF0b21CbG9jazogc3RyaW5nW10sIG1vbGZpbGVCb25kQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzLCBDOiBMb29wQ29uc3RhbnRzXG4pOiB2b2lkIHtcbiAgLy8gdG9kbzogcmVtb3ZlIHRoZXNlIGNvbW1lbnRzIHRvIHRoZSBkb2NzdHJpbmdzIG9mIHRoZSBjb3JyLiBmdW5jdGlvbnNcbiAgLy8gY29uc3RydW5jdCB0aGUgbGluZXMgb2YgVjNLIG1vbGZpbGUgYXRvbSBibG9ja1xuICBmaWxsQXRvbUxpbmVzKG1vbm9tZXIsIG1vbGZpbGVBdG9tQmxvY2ssIHYpO1xuXG4gIC8vIGNvbnN0cnVjdCB0aGUgbGluZXMgb2YgVjNLIG1vbGZpbGUgYm9uZCBibG9ja1xuICBmaWxsQm9uZExpbmVzKG1vbm9tZXIsIG1vbGZpbGVCb25kQmxvY2ssIHYpO1xuXG4gIC8vIHBlcHRpZGUgYm9uZFxuICBmaWxsQ2hhaW5FeHRlbmRpbmdCb25kKG1vbm9tZXIsIG1vbGZpbGVCb25kQmxvY2ssIHYpO1xuXG4gIC8vIHVwZGF0ZSBicmFuY2ggdmFyaWFibGVzIGlmIG5lY2Vzc2FyeVxuICBpZiAobW9ub21lci5tZXRhLmJyYW5jaFNoaWZ0ICE9PSBudWxsICYmIG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzLmxlbmd0aCA+IDIpXG4gICAgdXBkYXRlQnJhbmNoVmFyaWFibGVzKG1vbm9tZXIsIHYpO1xuXG4gIC8vIHVwZGF0ZSBsb29wIHZhcmlhYmxlc1xuICB1cGRhdGVDaGFpbkV4dGVuZGluZ1ZhcmlhYmxlcyhtb25vbWVyLCB2LCBDKTtcbn1cblxuLy8gdG9kbzogZG9jXG5mdW5jdGlvbiBhZGROdWNsZW90aWRlVG9Nb2xibG9jayhcbiAgbnVjbGVvYmFzZTogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcywgQzogTG9vcENvbnN0YW50c1xuKTogdm9pZCB7XG4gIC8vIGNvbnN0cnVuY3QgdGhlIGxpbmVzIG9mIFYzSyBtb2xmaWxlIGF0b20gYmxvY2sgY29ycmVzcG9uZGluZyB0byBwaG9zcGhhdGVcbiAgLy8gYW5kIHN1Z2FyXG4gIGZvciAoY29uc3QgbW9ub21lciBvZiBbQy5waG9zcGhhdGUsIEMuc3VnYXJdKVxuICAgIGFkZEJhY2tib25lTW9ub21lclRvTW9sYmxvY2sobW9ub21lciEsIG1vbGZpbGVBdG9tQmxvY2ssIG1vbGZpbGVCb25kQmxvY2ssIHYsIEMpO1xuXG4gIGFkZEJyYW5jaE1vbm9tZXJUb01vbGJsb2NrKG51Y2xlb2Jhc2UsIG1vbGZpbGVBdG9tQmxvY2ssIG1vbGZpbGVCb25kQmxvY2ssIHYsIEMpO1xufVxuXG5mdW5jdGlvbiBhZGRCcmFuY2hNb25vbWVyVG9Nb2xibG9jayhcbiAgbW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcywgQzogTG9vcENvbnN0YW50c1xuKTogdm9pZCB7XG4gIGZpbGxCcmFuY2hBdG9tTGluZXMobW9ub21lciwgbW9sZmlsZUF0b21CbG9jaywgdik7XG4gIGZpbGxCb25kTGluZXMobW9ub21lciwgbW9sZmlsZUJvbmRCbG9jaywgdik7XG4gIGZpbGxCYWNrYm9uZVRvQnJhbmNoQm9uZChtb25vbWVyLCBtb2xmaWxlQm9uZEJsb2NrLCB2KTtcblxuICAvLyBDLU4gYm9uZFxuICBjb25zdCBib25kSWR4ID0gdi5ib25kU2hpZnQ7XG4gIGNvbnN0IGZpcnN0QXRvbSA9IHYuYnJhbmNoQXR0YWNoTm9kZTtcbiAgY29uc3Qgc2Vjb25kQXRvbSA9IG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzBdICsgdi5ub2RlU2hpZnQ7XG4gIG1vbGZpbGVCb25kQmxvY2tbYm9uZElkeCAtIDFdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIGJvbmRJZHggKyAnICcgK1xuICAgIDEgKyAnICcgKyBmaXJzdEF0b20gKyAnICcgKyBzZWNvbmRBdG9tICsgJ1xcbic7XG5cbiAgLy8gdXBkYXRlIGxvb3AgdmFyaWFibGVzXG4gIHYuYm9uZFNoaWZ0ICs9IG1vbm9tZXIuYm9uZHMuYXRvbVBhaXJzLmxlbmd0aCArIDE7XG4gIHYubm9kZVNoaWZ0ICs9IG1vbm9tZXIuYXRvbXMuYXRvbVR5cGVzLmxlbmd0aDtcbn1cblxuZnVuY3Rpb24gdXBkYXRlQ2hhaW5FeHRlbmRpbmdWYXJpYWJsZXMobW9ub21lcjogTW9sR3JhcGgsIHY6IExvb3BWYXJpYWJsZXMsIEM6IExvb3BDb25zdGFudHMpOiB2b2lkIHtcbiAgdi5iYWNrYm9uZUF0dGFjaE5vZGUgPSB2Lm5vZGVTaGlmdCArIG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzFdO1xuICB2LmJvbmRTaGlmdCArPSBtb25vbWVyLmJvbmRzLmF0b21QYWlycy5sZW5ndGggKyAxO1xuXG4gIHYubm9kZVNoaWZ0ICs9IG1vbm9tZXIuYXRvbXMuYXRvbVR5cGVzLmxlbmd0aDtcbiAgdi5iYWNrYm9uZVBvc2l0aW9uU2hpZnRbMF0gKz0gbW9ub21lci5tZXRhLmJhY2tib25lU2hpZnQhWzBdOyAvLyB0b2RvOiBub24tbnVsbCBjaGVja1xuICB2LmJhY2tib25lUG9zaXRpb25TaGlmdFsxXSArPSB2LmZsaXBGYWN0b3IgKiBtb25vbWVyLm1ldGEuYmFja2JvbmVTaGlmdCFbMV07XG59XG5cbmZ1bmN0aW9uIHVwZGF0ZUJyYW5jaFZhcmlhYmxlcyhtb25vbWVyOiBNb2xHcmFwaCwgdjogTG9vcFZhcmlhYmxlcykge1xuICB2LmJyYW5jaEF0dGFjaE5vZGUgPSB2Lm5vZGVTaGlmdCArIG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzJdO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IDI7ICsraSlcbiAgICB2LmJyYW5jaFBvc2l0aW9uU2hpZnRbaV0gPSB2LmJhY2tib25lUG9zaXRpb25TaGlmdFtpXSArIG1vbm9tZXIubWV0YS5icmFuY2hTaGlmdCFbaV07XG59XG5cbmZ1bmN0aW9uIGZpbGxBdG9tTGluZXMobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzKTogdm9pZCB7XG4gIGZvciAobGV0IGogPSAwOyBqIDwgbW9ub21lci5hdG9tcy5hdG9tVHlwZXMubGVuZ3RoOyArK2opIHtcbiAgICBjb25zdCBhdG9tSWR4ID0gdi5ub2RlU2hpZnQgKyBqICsgMTtcbiAgICBtb2xmaWxlQXRvbUJsb2NrW3Yubm9kZVNoaWZ0ICsgal0gPSBWM0tfQkVHSU5fREFUQV9MSU5FICsgYXRvbUlkeCArICcgJyArXG4gICAgICBtb25vbWVyLmF0b21zLmF0b21UeXBlc1tqXSArICcgJyArXG4gICAgICBrZWVwUHJlY2lzaW9uKHYuYmFja2JvbmVQb3NpdGlvblNoaWZ0WzBdICsgbW9ub21lci5hdG9tcy54W2pdKSArICcgJyArXG4gICAgICBrZWVwUHJlY2lzaW9uKHYuYmFja2JvbmVQb3NpdGlvblNoaWZ0WzFdICsgdi5mbGlwRmFjdG9yICogbW9ub21lci5hdG9tcy55W2pdKSArXG4gICAgICAnICcgKyBtb25vbWVyLmF0b21zLmt3YXJnc1tqXTtcbiAgfVxufVxuXG4vLyB0b2RvOiByZW1vdmUgYXMgcXVpY2tmaXhcbmZ1bmN0aW9uIGZpbGxCcmFuY2hBdG9tTGluZXMobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzKTogdm9pZCB7XG4gIGZvciAobGV0IGogPSAwOyBqIDwgbW9ub21lci5hdG9tcy5hdG9tVHlwZXMubGVuZ3RoOyArK2opIHtcbiAgICBjb25zdCBhdG9tSWR4ID0gdi5ub2RlU2hpZnQgKyBqICsgMTtcbiAgICBtb2xmaWxlQXRvbUJsb2NrW3Yubm9kZVNoaWZ0ICsgal0gPSBWM0tfQkVHSU5fREFUQV9MSU5FICsgYXRvbUlkeCArICcgJyArXG4gICAgICBtb25vbWVyLmF0b21zLmF0b21UeXBlc1tqXSArICcgJyArXG4gICAgICBrZWVwUHJlY2lzaW9uKHYuYnJhbmNoUG9zaXRpb25TaGlmdFswXSArIG1vbm9tZXIuYXRvbXMueFtqXSkgKyAnICcgK1xuICAgICAga2VlcFByZWNpc2lvbih2LmJyYW5jaFBvc2l0aW9uU2hpZnRbMV0gKyB2LmZsaXBGYWN0b3IgKiBtb25vbWVyLmF0b21zLnlbal0pICtcbiAgICAgICcgJyArIG1vbm9tZXIuYXRvbXMua3dhcmdzW2pdO1xuICB9XG59XG5cbmZ1bmN0aW9uIGZpbGxCb25kTGluZXMobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVCb25kQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzKTogdm9pZCB7XG4gIC8vIGNvbnN0cnVjdCB0aGUgbGluZXMgb2YgVjNLIG1vbGZpbGUgYm9uZCBibG9ja1xuICBmb3IgKGxldCBqID0gMDsgaiA8IG1vbm9tZXIuYm9uZHMuYXRvbVBhaXJzLmxlbmd0aDsgKytqKSB7XG4gICAgY29uc3QgYm9uZElkeCA9IHYuYm9uZFNoaWZ0ICsgaiArIDE7XG4gICAgY29uc3QgZmlyc3RBdG9tID0gbW9ub21lci5ib25kcy5hdG9tUGFpcnNbal1bMF0gKyB2Lm5vZGVTaGlmdDtcbiAgICBjb25zdCBzZWNvbmRBdG9tID0gbW9ub21lci5ib25kcy5hdG9tUGFpcnNbal1bMV0gKyB2Lm5vZGVTaGlmdDtcbiAgICBsZXQgYm9uZENmZyA9ICcnO1xuICAgIGlmIChtb25vbWVyLmJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmhhcyhqKSkge1xuICAgICAgLy8gZmxpcCBvcmllbnRhdGlvbiB3aGVuIG5lY2Vzc2FyeVxuICAgICAgbGV0IG9yaWVudGF0aW9uID0gbW9ub21lci5ib25kcy5ib25kQ29uZmlndXJhdGlvbi5nZXQoaik7XG4gICAgICBpZiAodi5mbGlwRmFjdG9yIDwgMClcbiAgICAgICAgb3JpZW50YXRpb24gPSAob3JpZW50YXRpb24gPT09IDEpID8gMyA6IDE7XG4gICAgICBib25kQ2ZnID0gJyBDRkc9JyArIG9yaWVudGF0aW9uO1xuICAgIH1cbiAgICBjb25zdCBrd2FyZ3MgPSBtb25vbWVyLmJvbmRzLmt3YXJncy5oYXMoaikgP1xuICAgICAgJyAnICsgbW9ub21lci5ib25kcy5rd2FyZ3MuZ2V0KGopIDogJyc7XG4gICAgbW9sZmlsZUJvbmRCbG9ja1t2LmJvbmRTaGlmdCArIGpdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIGJvbmRJZHggKyAnICcgK1xuICAgICAgbW9ub21lci5ib25kcy5ib25kVHlwZXNbal0gKyAnICcgK1xuICAgICAgZmlyc3RBdG9tICsgJyAnICsgc2Vjb25kQXRvbSArIGJvbmRDZmcgKyBrd2FyZ3MgKyAnXFxuJztcbiAgfVxufVxuXG5mdW5jdGlvbiBmaWxsQ2hhaW5FeHRlbmRpbmdCb25kKG1vbm9tZXI6IE1vbEdyYXBoLCBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcyk6IHZvaWQge1xuICBpZiAodi5iYWNrYm9uZUF0dGFjaE5vZGUgIT09IDApIHtcbiAgICBjb25zdCBib25kSWR4ID0gdi5ib25kU2hpZnQ7XG4gICAgY29uc3QgZmlyc3RBdG9tID0gdi5iYWNrYm9uZUF0dGFjaE5vZGU7XG4gICAgY29uc3Qgc2Vjb25kQXRvbSA9IG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzBdICsgdi5ub2RlU2hpZnQ7XG4gICAgbW9sZmlsZUJvbmRCbG9ja1t2LmJvbmRTaGlmdCAtIDFdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIGJvbmRJZHggKyAnICcgK1xuICAgICAgMSArICcgJyArIGZpcnN0QXRvbSArICcgJyArIHNlY29uZEF0b20gKyAnXFxuJztcbiAgfVxufVxuXG4vLyB0b2RvOiByZW1vdmVcbmZ1bmN0aW9uIGZpbGxCYWNrYm9uZVRvQnJhbmNoQm9uZChicmFuY2hNb25vbWVyOiBNb2xHcmFwaCwgbW9sZmlsZUJvbmRCbG9jazogc3RyaW5nW10sIHY6IExvb3BWYXJpYWJsZXMpOiB2b2lkIHtcbiAgY29uc3QgYm9uZElkeCA9IHYuYm9uZFNoaWZ0O1xuICBjb25zdCBmaXJzdEF0b20gPSB2LmJyYW5jaEF0dGFjaE5vZGU7XG4gIGNvbnN0IHNlY29uZEF0b20gPSBicmFuY2hNb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSArIHYubm9kZVNoaWZ0O1xuICBtb2xmaWxlQm9uZEJsb2NrW2JvbmRJZHggLSAxXSA9IFYzS19CRUdJTl9EQVRBX0xJTkUgKyBib25kSWR4ICsgJyAnICtcbiAgICAxICsgJyAnICsgZmlyc3RBdG9tICsgJyAnICsgc2Vjb25kQXRvbSArICdcXG4nO1xufVxuXG4vKiBDb21wdXRlIHRoZSBhdG9tL2JvbmQgY291bnRzIGZvciB0aGUgcmVzdWx0aW5nIG1vbGZpbGUsIGRlcGVuZGluZyBvbiB0aGVcbiAqIHR5cGUgb2YgcG9seW1lciAocGVwdGlkZS9udWNsZW90aWRlKSAqL1xuZnVuY3Rpb24gZ2V0UmVzdWx0aW5nQXRvbUJvbmRDb3VudHMoXG4gIG1vbm9tZXJTZXE6IHN0cmluZ1tdLCBtb25vbWVyc0RpY3Q6IE1hcDxzdHJpbmcsIE1vbEdyYXBoPixcbiAgYWxwaGFiZXQ6IEFMUEhBQkVULCBwb2x5bWVyVHlwZTogSEVMTV9QT0xZTUVSX1RZUEVcbik6IHthdG9tQ291bnQ6IG51bWJlciwgYm9uZENvdW50OiBudW1iZXJ9IHtcbiAgbGV0IGF0b21Db3VudCA9IDA7XG4gIGxldCBib25kQ291bnQgPSAwO1xuXG4gIC8vIHN1bSB1cCBhbGwgdGhlIGF0b21zL25vZGVzIHByb3ZpZGVkIGJ5IHRoZSBzZXF1ZW5jZVxuICBmb3IgKGNvbnN0IG1vbm9tZXJTeW1ib2wgb2YgbW9ub21lclNlcSkge1xuICAgIGNvbnN0IG1vbm9tZXIgPSBtb25vbWVyc0RpY3QuZ2V0KG1vbm9tZXJTeW1ib2wpITtcbiAgICBhdG9tQ291bnQgKz0gbW9ub21lci5hdG9tcy54Lmxlbmd0aDtcbiAgICBib25kQ291bnQgKz0gbW9ub21lci5ib25kcy5ib25kVHlwZXMubGVuZ3RoO1xuICB9XG5cbiAgLy8gYWRkIGV4dHJhIHZhbHVlcyBkZXBlbmRpbmcgb24gdGhlIHBvbHltZXIgdHlwZVxuICBpZiAocG9seW1lclR5cGUgPT09IEhFTE1fUE9MWU1FUl9UWVBFLlBFUFRJREUpIHtcbiAgICAvLyBhZGQgdGhlIHJpZ2h0bW9zdC90ZXJtaW5hdGluZyBjYXAgZ3JvdXAgJ09IJyAoaS5lLiAnTycpXG4gICAgYXRvbUNvdW50ICs9IDE7XG4gICAgLy8gYWRkIGNoYWluLWV4dGVuZGluZyBib25kcyAoQy1OSCBwZXIgZWFjaCBtb25vbWVyIHBhaXIgYW5kIHRlcm1pbmFsIEMtT0gpXG4gICAgYm9uZENvdW50ICs9IG1vbm9tZXJTZXEubGVuZ3RoO1xuICB9IGVsc2UgeyAvLyBudWNsZW90aWRlc1xuICAgIGNvbnN0IHN1Z2FyID0gKGFscGhhYmV0ID09PSBBTFBIQUJFVC5ETkEpID9cbiAgICAgIG1vbm9tZXJzRGljdC5nZXQoREVPWFlSSUJPU0UpISA6IG1vbm9tZXJzRGljdC5nZXQoUklCT1NFKSE7XG4gICAgY29uc3QgcGhvc3BoYXRlID0gbW9ub21lcnNEaWN0LmdldChQSE9TUEhBVEUpITtcblxuICAgIC8vIGFkZCBwaG9zcGhhdGUgYW5kIHN1Z2FyIHBlciBlYWNoIG51Y2xlb2Jhc2Ugc3ltYm9sXG4gICAgYXRvbUNvdW50ICs9IG1vbm9tZXJTZXEubGVuZ3RoICogKHBob3NwaGF0ZS5hdG9tcy54Lmxlbmd0aCArIHN1Z2FyLmF0b21zLngubGVuZ3RoKTtcbiAgICAvLyBhZGQgdGhlIGxlZnRtb3N0IGNhcCBncm91cCAnT0gnIChpLmUuICdPJykgdG8gdGhlIGZpcnN0IHBob3NwaGF0ZVxuICAgIGF0b21Db3VudCArPSAxO1xuXG4gICAgLy8gYWRkIGJvbmRzIGZyb20gcGhvc3BoYXRlIGFuZCBzdWdhclxuICAgIGJvbmRDb3VudCArPSBtb25vbWVyU2VxLmxlbmd0aCAqIChwaG9zcGhhdGUuYm9uZHMuYm9uZFR5cGVzLmxlbmd0aCArIHN1Z2FyLmJvbmRzLmJvbmRUeXBlcy5sZW5ndGgpO1xuXG4gICAgLy8gYWRkIGNoYWluLWV4dGVuZGluZyBhbmQgYnJhbmNoIGJvbmRzIChPLVAsIEMtTyBhbmQgQy1OIHBlciBlYWNoIG51Y2xlb3RpZGUpXG4gICAgYm9uZENvdW50ICs9IG1vbm9tZXJTZXEubGVuZ3RoICogMztcbiAgfVxuXG4gIHJldHVybiB7YXRvbUNvdW50LCBib25kQ291bnR9O1xufVxuXG4vKiBLZWVwIHByZWNpc2lvbiB1cG9uIGZsb2F0aW5nIHBvaW50IG9wZXJhdGlvbnMgb3ZlciBhdG9tIGNvb3JkaW5hdGVzICovXG5mdW5jdGlvbiBrZWVwUHJlY2lzaW9uKHg6IG51bWJlcikge1xuICByZXR1cm4gTWF0aC5yb3VuZChQUkVDSVNJT05fRkFDVE9SICogeCkvUFJFQ0lTSU9OX0ZBQ1RPUjtcbn1cblxuZnVuY3Rpb24gY29udmVydE1vbEdyYXBoVG9Nb2xmaWxlVjNLKG1vbEdyYXBoOiBNb2xHcmFwaCk6IHN0cmluZyB7XG4gIC8vIGNvdW50cyBsaW5lXG4gIGNvbnN0IGF0b21UeXBlID0gbW9sR3JhcGguYXRvbXMuYXRvbVR5cGVzO1xuICBjb25zdCB4ID0gbW9sR3JhcGguYXRvbXMueDtcbiAgY29uc3QgeSA9IG1vbEdyYXBoLmF0b21zLnk7XG4gIGNvbnN0IGF0b21Ld2FyZ3MgPSBtb2xHcmFwaC5hdG9tcy5rd2FyZ3M7XG4gIGNvbnN0IGJvbmRUeXBlID0gbW9sR3JhcGguYm9uZHMuYm9uZFR5cGVzO1xuICBjb25zdCBhdG9tUGFpciA9IG1vbEdyYXBoLmJvbmRzLmF0b21QYWlycztcbiAgY29uc3QgYm9uZEt3YXJncyA9IG1vbEdyYXBoLmJvbmRzLmt3YXJncztcbiAgY29uc3QgYm9uZENvbmZpZyA9IG1vbEdyYXBoLmJvbmRzLmJvbmRDb25maWd1cmF0aW9uO1xuICBjb25zdCBhdG9tQ291bnQgPSBhdG9tVHlwZS5sZW5ndGg7XG4gIGNvbnN0IGJvbmRDb3VudCA9IG1vbEdyYXBoLmJvbmRzLmJvbmRUeXBlcy5sZW5ndGg7XG5cbiAgLy8gdG9kbyByZXdyaXRlIHVzaW5nIGNvbnN0YW50c1xuICBjb25zdCBtb2xmaWxlQ291bnRzTGluZSA9IFYzS19CRUdJTl9DT1VOVFNfTElORSArIGF0b21Db3VudCArICcgJyArIGJvbmRDb3VudCArIFYzS19DT1VOVFNfTElORV9FTkRJTkc7XG5cbiAgLy8gYXRvbSBibG9ja1xuICBsZXQgbW9sZmlsZUF0b21CbG9jayA9ICcnO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IGF0b21Db3VudDsgKytpKSB7XG4gICAgY29uc3QgYXRvbUlkeCA9IGkgKyAxO1xuICAgIGNvbnN0IGNvb3JkaW5hdGUgPSBbeFtpXS50b1N0cmluZygpLCB5W2ldLnRvU3RyaW5nKCldO1xuXG4gICAgLy8gZm9ybWF0IGNvb3JkaW5hdGVzIHNvIHRoYXQgdGhleSBoYXZlIDYgZGlnaXRzIGFmdGVyIGRlY2ltYWwgcG9pbnRcbiAgICAvLyBmb3IgKGxldCBrID0gMDsgayA8IDI7ICsraykge1xuICAgIC8vICAgY29uc3QgZm9ybWF0dGVkID0gY29vcmRpbmF0ZVtrXS50b1N0cmluZygpLnNwbGl0KCcuJyk7XG4gICAgLy8gICBpZiAoZm9ybWF0dGVkLmxlbmd0aCA9PT0gMSlcbiAgICAvLyAgICAgZm9ybWF0dGVkLnB1c2goJzAnKTtcbiAgICAvLyAgIGZvcm1hdHRlZFsxXSA9IGZvcm1hdHRlZFsxXS5wYWRFbmQoVjNLX0FUT01fQ09PUkRJTkFURV9QUkVDSVNJT04sICcwJyk7XG4gICAgLy8gICBjb29yZGluYXRlW2tdID0gZm9ybWF0dGVkLmpvaW4oJy4nKTtcbiAgICAvLyB9XG5cbiAgICBjb25zdCBhdG9tTGluZSA9IFYzS19CRUdJTl9EQVRBX0xJTkUgKyBhdG9tSWR4ICsgJyAnICsgYXRvbVR5cGVbaV0gKyAnICcgK1xuICAgICAgY29vcmRpbmF0ZVswXSArICcgJyArIGNvb3JkaW5hdGVbMV0gKyAnICcgKyBhdG9tS3dhcmdzW2ldO1xuICAgIG1vbGZpbGVBdG9tQmxvY2sgKz0gYXRvbUxpbmU7XG4gIH1cblxuICAvLyBib25kIGJsb2NrXG4gIGxldCBtb2xmaWxlQm9uZEJsb2NrID0gJyc7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgYm9uZENvdW50OyArK2kpIHtcbiAgICBjb25zdCBib25kSWR4ID0gaSArIDE7XG4gICAgY29uc3QgZmlyc3RBdG9tID0gYXRvbVBhaXJbaV1bMF07XG4gICAgY29uc3Qgc2Vjb25kQXRvbSA9IGF0b21QYWlyW2ldWzFdO1xuICAgIGNvbnN0IGt3YXJncyA9IGJvbmRLd2FyZ3MuaGFzKGkpID8gJyAnICsgYm9uZEt3YXJncy5nZXQoaSkgOiAnJztcbiAgICBjb25zdCBib25kQ2ZnID0gYm9uZENvbmZpZy5oYXMoaSkgPyAnIENGRz0nICsgYm9uZENvbmZpZy5nZXQoaSkgOiAnJztcbiAgICBjb25zdCBib25kTGluZSA9IFYzS19CRUdJTl9EQVRBX0xJTkUgKyBib25kSWR4ICsgJyAnICsgYm9uZFR5cGVbaV0gKyAnICcgK1xuICAgICAgZmlyc3RBdG9tICsgJyAnICsgc2Vjb25kQXRvbSArIGJvbmRDZmcgKyBrd2FyZ3MgKyAnXFxuJztcbiAgICBtb2xmaWxlQm9uZEJsb2NrICs9IGJvbmRMaW5lO1xuICB9XG5cbiAgY29uc3QgbW9sZmlsZVBhcnRzID0gW1xuICAgIFYzS19IRUFERVJfRklSU1RfTElORSxcbiAgICBWM0tfSEVBREVSX1NFQ09ORF9MSU5FLFxuICAgIFYzS19CRUdJTl9DVEFCX0JMT0NLLFxuICAgIG1vbGZpbGVDb3VudHNMaW5lLFxuICAgIFYzS19CRUdJTl9BVE9NX0JMT0NLLFxuICAgIG1vbGZpbGVBdG9tQmxvY2ssXG4gICAgVjNLX0VORF9BVE9NX0JMT0NLLFxuICAgIFYzS19CRUdJTl9CT05EX0JMT0NLLFxuICAgIG1vbGZpbGVCb25kQmxvY2ssXG4gICAgVjNLX0VORF9CT05EX0JMT0NLLFxuICAgIFYzS19FTkRfQ1RBQl9CTE9DSyxcbiAgICBWM0tfRU5ELFxuICBdO1xuICBjb25zdCByZXN1bHRpbmdNb2xmaWxlID0gbW9sZmlsZVBhcnRzLmpvaW4oJycpO1xuICAvLyBjb25zb2xlLmxvZyhyZXN1bHRpbmdNb2xmaWxlKTtcblxuICByZXR1cm4gcmVzdWx0aW5nTW9sZmlsZTtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGdldFN5bWJvbFRvQ2FwcGVkTW9sZmlsZU1hcChtb25vbWVyc0xpYkxpc3Q6IGFueVtdKTogUHJvbWlzZTxNYXA8c3RyaW5nLCBzdHJpbmc+IHwgdW5kZWZpbmVkPiB7XG4gIGlmIChERy5GdW5jLmZpbmQoe3BhY2thZ2U6ICdDaGVtJywgbmFtZTogJ2dldFJkS2l0TW9kdWxlJ30pLmxlbmd0aCA9PT0gMCkge1xuICAgIGdyb2suc2hlbGwud2FybmluZygnVHJhbnNmb3JtYXRpb24gdG8gYXRvbWljIGxldmVsIHJlcXVpcmVzIHBhY2thZ2UgXCJDaGVtXCIgaW5zdGFsbGVkLicpO1xuICAgIHJldHVybjtcbiAgfVxuXG4gIGNvbnN0IHN5bWJvbFRvQ2FwcGVkTW9sZmlsZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBzdHJpbmc+O1xuICBjb25zdCBtb2R1bGVSZGtpdCA9IGF3YWl0IGdyb2suZnVuY3Rpb25zLmNhbGwoJ0NoZW06Z2V0UmRLaXRNb2R1bGUnKTtcblxuICBmb3IgKGNvbnN0IG1vbm9tZXJMaWJPYmplY3Qgb2YgbW9ub21lcnNMaWJMaXN0KSB7XG4gICAgY29uc3QgbW9ub21lclN5bWJvbCA9IG1vbm9tZXJMaWJPYmplY3RbSEVMTV9GSUVMRFMuU1lNQk9MXTtcbiAgICBjb25zdCBjYXBHcm91cHMgPSBwYXJzZUNhcEdyb3Vwcyhtb25vbWVyTGliT2JqZWN0W0hFTE1fRklFTERTLlJHUk9VUFNdKTtcbiAgICBjb25zdCBjYXBHcm91cElkeE1hcCA9IHBhcnNlQ2FwR3JvdXBJZHhNYXAobW9ub21lckxpYk9iamVjdFtIRUxNX0ZJRUxEUy5NT0xGSUxFXSk7XG5cbiAgICBjb25zdCBtb2xmaWxlVjNLID0gY29udmVydE1vbGZpbGVUb1YzSyhyZW1vdmVSR3JvdXBMaW5lcyhtb25vbWVyTGliT2JqZWN0W0hFTE1fRklFTERTLk1PTEZJTEVdKSwgbW9kdWxlUmRraXQpO1xuICAgIGNvbnN0IGNvdW50cyA9IHBhcnNlQXRvbUFuZEJvbmRDb3VudHMobW9sZmlsZVYzSyk7XG5cbiAgICBjb25zdCBhdG9tcyA9IHBhcnNlQXRvbUJsb2NrKG1vbGZpbGVWM0ssIGNvdW50cy5hdG9tQ291bnQpO1xuICAgIGNvbnN0IGJvbmRzID0gcGFyc2VCb25kQmxvY2sobW9sZmlsZVYzSywgY291bnRzLmJvbmRDb3VudCk7XG4gICAgY29uc3QgbWV0YSA9IGdldE1vbm9tZXJNZXRhZGF0YShhdG9tcywgYm9uZHMsIGNhcEdyb3VwcywgY2FwR3JvdXBJZHhNYXApO1xuXG4gICAgY29uc3QgbW9ub21lckdyYXBoOiBNb2xHcmFwaCA9IHthdG9tczogYXRvbXMsIGJvbmRzOiBib25kcywgbWV0YTogbWV0YX07XG5cbiAgICByZW1vdmVIeWRyb2dlbihtb25vbWVyR3JhcGgpO1xuXG4gICAgY29uc3QgbW9sZmlsZSA9IGNvbnZlcnRNb2xHcmFwaFRvTW9sZmlsZVYzSyhtb25vbWVyR3JhcGgpO1xuICAgIHN5bWJvbFRvQ2FwcGVkTW9sZmlsZU1hcC5zZXQobW9ub21lclN5bWJvbCwgbW9sZmlsZSk7XG4gIH1cbiAgcmV0dXJuIHN5bWJvbFRvQ2FwcGVkTW9sZmlsZU1hcDtcbn1cbiJdfQ==