@datagrok-libraries/bio 5.2.1 → 5.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +1 -1
- package/src/types/index.d.ts +1 -0
- package/src/types/index.d.ts.map +1 -1
- package/src/types/index.js +1 -1
- package/src/utils/atomic-works.d.ts +2 -0
- package/src/utils/atomic-works.d.ts.map +1 -0
- package/src/utils/atomic-works.js +354 -0
- package/src/utils/cell-renderer.d.ts.map +1 -1
- package/src/utils/cell-renderer.js +11 -6
- package/src/utils/const.d.ts +48 -0
- package/src/utils/const.d.ts.map +1 -0
- package/src/utils/const.js +29 -0
- package/src/utils/monomer-library.d.ts +43 -0
- package/src/utils/monomer-library.d.ts.map +1 -0
- package/src/utils/monomer-library.js +154 -0
- package/src/utils/monomer-utils.d.ts +10 -0
- package/src/utils/monomer-utils.d.ts.map +1 -0
- package/src/utils/monomer-utils.js +125 -0
- package/src/utils/notation-converter.d.ts.map +1 -1
- package/src/utils/notation-converter.js +5 -1
- package/src/utils/to-atomic-level.d.ts +3 -0
- package/src/utils/to-atomic-level.d.ts.map +1 -0
- package/src/utils/to-atomic-level.js +1009 -0
- package/src/utils/units-handler.d.ts +4 -0
- package/src/utils/units-handler.d.ts.map +1 -1
- package/src/utils/units-handler.js +6 -8
- package/tsconfig.json +1 -1
|
@@ -0,0 +1,1009 @@
|
|
|
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
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG8tYXRvbWljLWxldmVsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsidG8tYXRvbWljLWxldmVsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUFBLHlGQUF5RjtBQUN6RixPQUFPLEtBQUssSUFBSSxNQUFNLG1CQUFtQixDQUFDO0FBQzFDLE9BQU8sS0FBSyxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDdEMsT0FBTyxFQUFxQixXQUFXLEVBQUMsTUFBTSxhQUFhLENBQUM7QUFDNUQsT0FBTyxFQUFjLGdCQUFnQixFQUFzRCxNQUFNLFNBQVMsQ0FBQztBQUczRyxPQUFPLEVBQUMsaUJBQWlCLEVBQUMsTUFBTSxzQkFBc0IsQ0FBQztBQUV2RCxzQ0FBc0M7QUFDdEMsTUFBTSxhQUFhLEdBQUcsQ0FBQyxDQUFDO0FBQ3hCLE1BQU0sWUFBWSxHQUFHLFFBQVEsQ0FBQztBQUM5QixNQUFNLFVBQVUsR0FBRyxLQUFLLENBQUM7QUFFekIsd0RBQXdEO0FBQ3hELE1BQU0sZ0JBQWdCLEdBQUcsRUFBRSxDQUFDO0FBQzVCLE1BQU0sYUFBYSxHQUFHLENBQUMsQ0FBQztBQUN4QixNQUFNLHFCQUFxQixHQUFHLHNDQUFzQyxDQUFDO0FBQ3JFLE1BQU0sc0JBQXNCLEdBQUcsMkNBQTJDLENBQUM7QUFDM0UsTUFBTSxvQkFBb0IsR0FBRyxxQkFBcUIsQ0FBQztBQUNuRCxNQUFNLGtCQUFrQixHQUFHLG1CQUFtQixDQUFDO0FBQy9DLE1BQU0scUJBQXFCLEdBQUcsZ0JBQWdCLENBQUM7QUFDL0MsTUFBTSxzQkFBc0IsR0FBRyxVQUFVLENBQUM7QUFDMUMsTUFBTSxvQkFBb0IsR0FBRyxxQkFBcUIsQ0FBQztBQUNuRCxNQUFNLGtCQUFrQixHQUFHLG1CQUFtQixDQUFDO0FBQy9DLE1BQU0sb0JBQW9CLEdBQUcscUJBQXFCLENBQUM7QUFDbkQsTUFBTSxrQkFBa0IsR0FBRyxtQkFBbUIsQ0FBQztBQUMvQyxNQUFNLGVBQWUsR0FBRyxPQUFPLENB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RTtRQUNsQixLQUFLLEdBQUcsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsRUFBRSxHQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEVBQUUsR0FBQyxDQUFDLENBQUM7S0FDeEM7U0FBTTtRQUNMLE1BQU0sR0FBRyxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDbEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM1QixLQUFLLEdBQUcsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxFQUFFLEdBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsRUFBRSxHQUFDLENBQUMsR0FBRyxJQUFJLENBQUM7S0FDeEQ7SUFDRCxPQUFPLEtBQUssQ0FBQztBQUNmLENBQUM7QUFFRCxtRUFBbUU7QUFDbkUsU0FBUyxlQUFlLENBQUMsQ0FBUyxFQUFFLENBQVM7SUFDM0MsT0FBTyxlQUFlLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxHQUFHLElBQUksQ0FBQyxFQUFFLEdBQUMsQ0FBQyxDQUFDO0FBQzNDLENBQUM7QUFFRCxvREFBb0Q7QUFDcEQsU0FBUyxtQkFBbUIsQ0FBQyxLQUFZLEVBQUUsS0FBYTtJQUN0RCxJQUFJLEtBQUssS0FBSyxDQUFDLEVBQUU7UUFDZixNQUFNLENBQUMsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQ2xCLE1BQU0sQ0FBQyxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUM7UUFFbEIsTUFBTSxHQUFHLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM1QixNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBRTVCLEtBQUssSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxDQUFDLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FB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LENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztLQUNwQztJQUVELE9BQU8sRUFBQyxTQUFTLEVBQUUsU0FBUyxFQUFDLENBQUM7QUFDaEMsQ0FBQztBQUVELHlFQUF5RTtBQUN6RSxTQUFTLGFBQWEsQ0FBQyxDQUFTO0lBQzlCLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxnQkFBZ0IsR0FBRyxDQUFDLENBQUMsR0FBQyxnQkFBZ0IsQ0FBQztBQUMzRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyogRG8gbm90IGNoYW5nZSB0aGVzZSBpbXBvcnQgbGluZXMgdG8gbWF0Y2ggZXh0ZXJuYWwgbW9kdWxlcyBpbiB3ZWJwYWNrIGNvbmZpZ3VyYXRpb24gKi9cbmltcG9ydCAqIGFzIGdyb2sgZnJvbSAnZGF0YWdyb2stYXBpL2dyb2snO1xuaW1wb3J0ICogYXMgREcgZnJvbSAnZGF0YWdyb2stYXBpL2RnJztcbmltcG9ydCB7U3BsaXR0ZXJGdW5jLCBUQUdTLCBnZXRTcGxpdHRlcn0gZnJvbSAnLi4vLi4vaW5kZXgnO1xuaW1wb3J0IHtIRUxNX0ZJRUxEUywgSEVMTV9DT1JFX0ZJRUxEUywgSEVMTV9QT0xZTUVSX1RZUEUsIEhFTE1fTU9OT01FUl9UWVBFLCBSR1JPVVBfRklFTERTfSBmcm9tICcuL2NvbnN0JztcbmltcG9ydCB7QUxQSEFCRVQsIE5PVEFUSU9OfSBmcm9tICcuL21hY3JvbW9sZWN1bGUnO1xuaW1wb3J0IHtVbml0c0hhbmRsZXJ9IGZyb20gJy4vdW5pdHMtaGFuZGxlcic7XG5pbXBvcnQge05vdGF0aW9uQ29udmVydGVyfSBmcm9tICcuL25vdGF0aW9uLWNvbnZlcnRlcic7XG5cbi8vIGNvbnN0YW50cyBmb3IgcGFyc2luZyBtb2xmaWxlIFYyMDAwXG5jb25zdCBWMktfUkdQX1NISUZUID0gODtcbmNvbnN0IFYyS19SR1BfTElORSA9ICdNICBSR1AnO1xuY29uc3QgVjJLX0FfTElORSA9ICdBICAnO1xuXG4vLyBjb25zdGFudHMgZm9yIHBhcnNpbmcvcmVjb25zdHJ1Y3Rpb24gb2YgbW9sZmlsZSBWMzAwMFxuY29uc3QgVjNLX0NPVU5UU19TSElGVCA9IDE0O1xuY29uc3QgVjNLX0lEWF9TSElGVCA9IDc7XG5jb25zdCBWM0tfSEVBREVSX0ZJUlNUX0xJTkUgPSAnXFxuRGF0YWdyb2sgbWFjcm9tb2xlY3VsZSBoYW5kbGVyXFxuXFxuJztcbmNvbnN0IFYzS19IRUFERVJfU0VDT05EX0xJTkUgPSAnICAwICAwICAwICAwICAwICAwICAgICAgICAgICAgOTk5IFYzMDAwXFxuJztcbmNvbnN0IFYzS19CRUdJTl9DVEFCX0JMT0NLID0gJ00gIFYzMCBCRUdJTiBDVEFCXFxuJztcbmNvbnN0IFYzS19FTkRfQ1RBQl9CTE9DSyA9ICdNICBWMzAgRU5EIENUQUJcXG4nO1xuY29uc3QgVjNLX0JFR0lOX0NPVU5UU19MSU5FID0gJ00gIFYzMCBDT1VOVFMgJztcbmNvbnN0IFYzS19DT1VOVFNfTElORV9FTkRJTkcgPSAnIDAgMCAwXFxuJztcbmNvbnN0IFYzS19CRUdJTl9BVE9NX0JMT0NLID0gJ00gIFYzMCBCRUdJTiBBVE9NXFxuJztcbmNvbnN0IFYzS19FTkRfQVRPTV9CTE9DSyA9ICdNICBWMzAgRU5EIEFUT01cXG4nO1xuY29uc3QgVjNLX0JFR0lOX0JPTkRfQkxPQ0sgPSAnTSAgVjMwIEJFR0lOIEJPTkRcXG4nO1xuY29uc3QgVjNLX0VORF9CT05EX0JMT0NLID0gJ00gIFYzMCBFTkQgQk9ORFxcbic7XG5jb25zdCBWM0tfQk9ORF9DT05GSUcgPSAnIENGRz0nO1xuY29uc3QgVjNLX0JFR0lOX0RBVEFfTElORSA9ICdNICBWMzAgJztcbmNvbnN0IFYzS19FTkQgPSAnTSAgRU5EXFxuJztcblxuY29uc3QgUFJFQ0lTSU9OX0ZBQ1RPUiA9IDEwXzAwMDsgLy8gSEVMTUNvcmVMaWJyYXJ5IGhhcyA0IHNpZ25pZmljYW50IGRpZ2l0cyBhZnRlciBkZWNpbWFsIHBvaW50IGluIGF0b20gY29vcmRpbmF0ZXNcblxuLy8gc3ltYm9scyBmb3IgdGhlIGNvcnJlc3BvbmRpbmcgbW9ub21lcnMgaW4gSEVMTSBsaWJyYXJ5XG5jb25zdCBERU9YWVJJQk9TRSA9ICdkJztcbmNvbnN0IFJJQk9TRSA9ICdyJztcbmNvbnN0IFBIT1NQSEFURSA9ICdwJztcblxuY29uc3QgT1hZR0VOID0gJ08nO1xuY29uc3QgSFlEUk9HRU4gPSAnSCc7XG5cbi8qIFN0b3JlcyBuZWNlc3NhcnkgZGF0YSBhYm91dCBhdG9tcyBvZiBhIG1vbm9tZXIgcGFyc2VkIGZyb20gTW9sZmlsZSAqL1xudHlwZSBBdG9tcyA9IHtcbiAgLyogZWxlbWVudCBzeW1ib2xzIGZvciBtb25vbWVyJ3MgYXRvbXMgKi9cbiAgYXRvbVR5cGVzOiBzdHJpbmdbXSxcbiAgLyogQ2FydGVzaWFuIGNvb3JkaWFudGVzIG9mIG1vbm9tZXIncyBhdG9tcyAqL1xuICB4OiBudW1iZXJbXSwgLy8gdG9kbzogY29udmVydCB0byBGbG9hdDMyXG4gIHk6IG51bWJlcltdLFxuICAvKiBWM0sgYXRvbSBsaW5lIG1heSBjb250YWluIGtleXdvcmQgYXJncyAqL1xuICBrd2FyZ3M6IHN0cmluZ1tdLFxufVxuXG4vKiBTdG9yZXMgbmVjZXNzYXJ5IGRhdGEgYWJvdXQgYm9uZHMgb2YgYSBtb25vbWVyIHBhcnNlZCBmcm9tIE1vbGZpbGUgKi9cbnR5cGUgQm9uZHMgPSB7XG4gIC8qIGJvbmQgdHlwZXMgZm9yIGFsbCBsaW5lcyBvZiBNb2xmaWxlIGJvbmQgYmxvY2sgKi9cbiAgYm9uZFR5cGVzOiBudW1iZXJbXSwgLy8gdG9kbzogY29udmVydCB0byBJbmQzMlxuICAvKiBJbmRpY2VzIG9mIGFsbCBhdG9tIHBhaXJzLCBpbmRleGluZyBzdGFydGluZyBmcm9tIDEgICovXG4gIGF0b21QYWlyczogbnVtYmVyW11bXSxcbiAgLyogSWYgYSBib25kIGhhcyBDRkc9Li4uIGtleXdvcmQgYXJndW1lbnQsIGl0IGlzIHBhcnNlZCBhbmQgc290cmVkIGFzIGFcbiAgICogdmFsdWUgb2YgdGhlIG1hcCwgd2l0aCB0aGUga2V5IGJlaW5nIHRoZSBib25kJ3MgaW5kZXggICovXG4gIGJvbmRDb25maWd1cmF0aW9uOiBNYXA8bnVtYmVyLCBudW1iZXI+LFxuICAvKiBWM0sgYm9uZCBsaW5lIG1heSBjb250YWluIGtleXdvcmQgYXJncyAqL1xuICBrd2FyZ3M6IE1hcDxudW1iZXIsIHN0cmluZz4sXG59XG5cbi8qIE1ldGFkYXRhIGFzc29jaWF0ZWQgd2l0aCB0aGUgbW9ub21lciBuZWNlc3NhcnkgdG8gcmVzdG9yZSB0aGUgcmVzdWx0aW5nXG4gKiBtb2xmaWxlICAqL1xudHlwZSBNb25vbWVyTWV0YWRhdGEgPSB7XG4gIC8qIHRlcm1pbmFsIG5vZGVzOiAwLXRoIGNvcnJlc3BvbmRzIHRvIHRoZSBcImxlZnRtb3N0XCIgb25lLCAxc3QsIHRvIHRoZSBcInJpZ2h0bW9zdFwiLFxuICAgKiBlLmcuIE4tdGVybWludXMgYW5kIEMtdGVybWludXMgaW4gcGVwdGlkZXMgKi9cbiAgdGVybWluYWxOb2RlczogbnVtYmVyW10sXG4gIC8qIHItZ3JvdXAgbm9kZXM6IDAtdGggY29ycmVzcG9uZHMgdG8gdGhlIFwibGVmdG1vc3RcIiBvbmUsIDFzdCwgdG8gdGhlIFwicmlnaHRtb3N0XCIgKi9cbiAgck5vZGVzOiBudW1iZXJbXSxcbiAgLyogc2hpZnQgZnJvbSB0aGUgb3JpZ2luIHRvIHRoZSBuZXh0IGJhY2tib25lLCBudWxsIGZvciBicmFuY2ggbW9ub21lcnMgKi9cbiAgYmFja2JvbmVTaGlmdDogbnVtYmVyW10gfCBudWxsLFxuICAvKiBzaGlmdCBmcm9tIHRoZSBvcmlnaW4gdG8gdGhlIG5leHQgYnJhbmNoLCBudWxsIGZvciBicmFuY2ggbW9ub21lcnMgKi9cbiAgYnJhbmNoU2hpZnQ6IG51bWJlcltdIHwgbnVsbFxufVxuXG50eXBlIE1vbEdyYXBoID0ge1xuICBhdG9tczogQXRvbXMsXG4gIGJvbmRzOiBCb25kcyxcbiAgbWV0YTogTW9ub21lck1ldGFkYXRhLFxufVxuXG4vKiBIZWxwZXIgc3RydWN0dXJlIHdyYXBwaW5nIGNvbW1vbiBhcmd1bWVudHMgdG8gc2V2ZXJhbCBmdW5jdGlvbnMgKi9cbnR5cGUgTG9vcFZhcmlhYmxlcyA9IHtcbiAgaTogbnVtYmVyLFxuICBub2RlU2hpZnQ6IG51bWJlcixcbiAgYm9uZFNoaWZ0OiBudW1iZXIsXG4gIGJhY2tib25lUG9zaXRpb25TaGlmdDogbnVtYmVyW10sXG4gIGJhY2tib25lQXR0YWNoTm9kZTogbnVtYmVyOyAvLyBub2RlIHRvIHdoaWNoIHRoZSBuZXh0IGJhY2tib25lIGlzIGF0dGFjaGVkXG4gIGJyYW5jaFBvc2l0aW9uU2hpZnQ6IG51bWJlcltdLFxuICBicmFuY2hBdHRhY2hOb2RlOiBudW1iZXIsXG4gIGZsaXBGYWN0b3I6IG51bWJlcixcbiAgLy8gdG9kbzogc2hvdWxkIHdlIGNvbnNpZGVyIHJlcHJlc2VudGF0aW9ucyBvdGhlciB0aGFuIHBsYW5hcj9cbn1cblxuLyogSGVscGVyIHN0cnVjdHVyZSB3cmFwcGluZyBjb21tb24gYXJndW1lbnRzIHRvIHNldmVyYWwgZnVuY3Rpb25zICovXG50eXBlIExvb3BDb25zdGFudHMgPSB7XG4gIHN1Z2FyOiBNb2xHcmFwaCB8IG51bGwsXG4gIHBob3NwaGF0ZTogTW9sR3JhcGggfCBudWxsLFxuICBzZXFMZW5ndGg6IG51bWJlcixcbiAgYXRvbUNvdW50OiBudW1iZXIsXG4gIGJvbmRDb3VudDogbnVtYmVyLFxufVxuXG4vLyB0b2RvOiB2ZXJpZnkgdGhhdCBhbGwgZnVuY3Rpb25zIGhhdmUgcmV0dXJuIHR5cGVzXG5cbi8qIENvbnZlcnQgTWFjcm9tb2xlY3VsZSBjb2x1bW4gaW50byBNb2xlY3VsZSBjb2x1bW4gc3RvcmluZyBtb2xmaWxlIFYzMDAwIHdpdGggdGhlIGhlbHAgb2YgYSBtb25vbWVyIGxpYnJhcnkgICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gX3RvQXRvbWljTGV2ZWwoXG4gIGRmOiBERy5EYXRhRnJhbWUsIG1hY3JvTW9sQ29sOiBERy5Db2x1bW48c3RyaW5nPiwgbW9ub21lcnNMaWJMaXN0OiBhbnlbXVxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIGlmIChERy5GdW5jLmZpbmQoe3BhY2thZ2U6ICdDaGVtJywgbmFtZTogJ2dldFJkS2l0TW9kdWxlJ30pLmxlbmd0aCA9PT0gMCkge1xuICAgIGdyb2suc2hlbGwud2FybmluZygnVHJhbnNmb3JtYXRpb24gdG8gYXRvbWljIGxldmVsIHJlcXVpcmVzIHBhY2thZ2UgXCJDaGVtXCIgaW5zdGFsbGVkLicpO1xuICAgIHJldHVybjtcbiAgfVxuXG4gIGlmIChtYWNyb01vbENvbC5zZW1UeXBlICE9PSBERy5TRU1UWVBFLk1BQ1JPTU9MRUNVTEUpIHtcbiAgICBncm9rLnNoZWxsLndhcm5pbmcoXG4gICAgICBgT25seSB0aGUgJHtERy5TRU1UWVBFLk1BQ1JPTU9MRUNVTEV9IGNvbHVtbnMgY2FuIGJlIGNvbnZlcnRlZCB0byBhdG9taWNcbiAgICAgIGxldmVsLCB0aGUgY2hvc2VuIGNvbHVtbiBoYXMgc2VtVHlwZSAke21hY3JvTW9sQ29sLnNlbVR5cGV9YFxuICAgICk7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgLy8gY29udmVydCAnaGVsbScgdG8gJ3NlcGFyYXRvcicgdW5pdHNcbiAgaWYgKG1hY3JvTW9sQ29sLmdldFRhZyhERy5UQUdTLlVOSVRTKSA9PT0gTk9UQVRJT04uSEVMTSkge1xuICAgIGNvbnN0IGNvbnZlcnRlciA9IG5ldyBOb3RhdGlvbkNvbnZlcnRlcihtYWNyb01vbENvbCk7XG4gICAgY29uc3Qgc2VwYXJhdG9yID0gJy8nO1xuICAgIG1hY3JvTW9sQ29sID0gY29udmVydGVyLmNvbnZlcnQoTk9UQVRJT04uU0VQQVJBVE9SLCBzZXBhcmF0b3IpO1xuICB9XG5cbiAgY29uc3QgYWxwaGFiZXQgPSBtYWNyb01vbENvbC5nZXRUYWcoVEFHUy5hbHBoYWJldCk7XG5cbiAgLy8gZGV0ZXJtaW5lIHRoZSBwb2x5bWVyIHR5cGUgYWNjb3JkaW5nIHRvIEhFTE0gc3BlY2lmaWNhdGlvbnNcbiAgbGV0IHBvbHltZXJUeXBlO1xuICAvLyB0b2RvOiBhbiBleGNlcHRpb24gZnJvbSBkYXJ0IGNvbWVzIGJlZm9yZSB0aGlzIGNoZWNrIGlmIHRoZSBhbHBoYWJldCBpcyBVTlxuICBpZiAoYWxwaGFiZXQgPT09IEFMUEhBQkVULlBUKSB7XG4gICAgcG9seW1lclR5cGUgPSBIRUxNX1BPTFlNRVJfVFlQRS5QRVBUSURFO1xuICB9IGVsc2UgaWYgKGFscGhhYmV0ID09PSBBTFBIQUJFVC5STkEgfHwgYWxwaGFiZXQgPT09IEFMUEhBQkVULkROQSkge1xuICAgIHBvbHltZXJUeXBlID0gSEVMTV9QT0xZTUVSX1RZUEUuUk5BO1xuICB9IGVsc2Uge1xuICAgIGdyb2suc2hlbGwud2FybmluZyhcbiAgICAgIGBPbmx5IFBULCBETkEgYW5kIFJOQSBhbHBoYWJldHMgYXJlIHN1cHBvcnRlZCwgd2hpbGUgdGhlIHNlbGVjdGVkIGNvbHVtbiBoYXMgJHtwb2x5bWVyVHlwZX0gYWxwaGFiZXRgXG4gICAgKTtcbiAgICByZXR1cm47XG4gIH1cblxuICBjb25zdCBtb25vbWVyU2VxdWVuY2VzQXJyYXk6IHN0cmluZ1tdW10gPSBnZXRNb25vbWVyU2VxdWVuY2VzQXJyYXkobWFjcm9Nb2xDb2wpO1xuICAvLyB0b2RvOiBjb25zaWRlciBzZXBhcmF0ZWx5IGJhY2tib25lLCB0ZXJtaW5hbCwgYnJhbmNoIG1vbm9tZXIgdHlwZXNcbiAgY29uc3QgbW9ub21lcnNEaWN0ID0gYXdhaXQgZ2V0TW9ub21lcnNEaWN0KG1vbm9tZXJTZXF1ZW5jZXNBcnJheSwgbW9ub21lcnNMaWJMaXN0LCBwb2x5bWVyVHlwZSwgYWxwaGFiZXQpO1xuICBjb25zdCBjb2x1bW5MZW5ndGggPSBtYWNyb01vbENvbC5sZW5ndGg7XG4gIGNvbnN0IHJlY29uc3RydWN0ZWQ6IHN0cmluZ1tdID0gbmV3IEFycmF5KGNvbHVtbkxlbmd0aCk7XG4gIGZvciAobGV0IHJvdyA9IDA7IHJvdyA8IGNvbHVtbkxlbmd0aDsgKytyb3cpIHtcbiAgICBjb25zdCBtb25vbWVyU2VxID0gbW9ub21lclNlcXVlbmNlc0FycmF5W3Jvd107XG4gICAgcmVjb25zdHJ1Y3RlZFtyb3ddID0gbW9ub21lclNlcVRvTW9sZmlsZShtb25vbWVyU2VxLCBtb25vbWVyc0RpY3QsIGFscGhhYmV0LCBwb2x5bWVyVHlwZSk7XG4gICAgY29uc29sZS5sb2cocmVjb25zdHJ1Y3RlZFtyb3ddKTtcbiAgfVxuXG4gIC8vIGV4Y2x1ZGUgbmFtZSBjb2xsaXNpb25zXG4gIGNvbnN0IG5hbWUgPSAnbW9sZmlsZSgnICsgbWFjcm9Nb2xDb2wubmFtZSArICcpJztcbiAgY29uc3QgbmV3Q29sTmFtZSA9IGRmLmNvbHVtbnMuZ2V0VW51c2VkTmFtZShuYW1lKTtcbiAgY29uc3QgbmV3Q29sID0gREcuQ29sdW1uLmZyb21TdHJpbmdzKG5ld0NvbE5hbWUsIHJlY29uc3RydWN0ZWQpO1xuXG4gIG5ld0NvbC5zZW1UeXBlID0gREcuU0VNVFlQRS5NT0xFQ1VMRTtcbiAgbmV3Q29sLnNldFRhZyhERy5UQUdTLlVOSVRTLCBERy5VTklUUy5Nb2xlY3VsZS5NT0xCTE9DSyk7XG4gIGRmLmNvbHVtbnMuYWRkKG5ld0NvbCwgdHJ1ZSk7XG4gIGF3YWl0IGdyb2suZGF0YS5kZXRlY3RTZW1hbnRpY1R5cGVzKGRmKTtcbn1cblxuLyogR2V0IGEgbWFwcGluZyBvZiBwZXB0aWRlIHN5bWJvbHMgdG8gSEVMTSBtb25vbWVyIGxpYnJhcnkgb2JqZWN0cyB3aXRoXG4gKiBzZWxlY3R0ZWQgZmllbGRzICAqL1xuZnVuY3Rpb24gZ2V0Rm9ybWF0dGVkTW9ub21lckxpYihcbiAgbW9ub21lcnNMaWJMaXN0OiBhbnlbXSwgcG9seW1lclR5cGU6IEhFTE1fUE9MWU1FUl9UWVBFLCBhbHBoYWJldDogQUxQSEFCRVRcbik6IE1hcDxzdHJpbmcsIGFueT4ge1xuICBjb25zdCBtYXAgPSBuZXcgTWFwPHN0cmluZywgYW55PigpO1xuICBtb25vbWVyc0xpYkxpc3QuZm9yRWFjaChcbiAgICAoaXQpID0+IHtcbiAgICAgIGlmIChpdFtIRUxNX0ZJRUxEUy5QT0xZTUVSX1RZUEVdID09PSBwb2x5bWVyVHlwZSkge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgcG9seW1lclR5cGUgPT09IEhFTE1fUE9MWU1FUl9UWVBFLlJOQSAmJlxuICAgICAgICAgIChpdFtIRUxNX0ZJRUxEUy5NT05PTUVSX1RZUEVdID09PSBIRUxNX01PTk9NRVJfVFlQRS5CUkFOQ0ggfHxcbiAgICAgICAgICBhbHBoYWJldCA9PT0gQUxQSEFCRVQuRE5BICYmIGl0W0hFTE1fRklFTERTLlNZTUJPTF0gPT09IERFT1hZUklCT1NFIHx8XG4gICAgICAgICAgYWxwaGFiZXQgPT09IEFMUEhBQkVULlJOQSAmJiBpdFtIRUxNX0ZJRUxEUy5TWU1CT0xdID09PSBSSUJPU0UgfHxcbiAgICAgICAgICBpdFtIRUxNX0ZJRUxEUy5TWU1CT0xdID09PSBQSE9TUEhBVEUpIHx8XG4gICAgICAgICAgcG9seW1lclR5cGUgPT09IEhFTE1fUE9MWU1FUl9UWVBFLlBFUFRJREUgJiZcbiAgICAgICAgICBpdFtIRUxNX0ZJRUxEUy5NT05PTUVSX1RZUEVdICE9PSBIRUxNX01PTk9NRVJfVFlQRS5CUkFOQ0hcbiAgICAgICAgKSB7XG4gICAgICAgICAgY29uc3QgbW9ub21lck9iamVjdDogeyBba2V5OiBzdHJpbmddOiBhbnkgfSA9IHt9O1xuICAgICAgICAgIEhFTE1fQ09SRV9GSUVMRFMuZm9yRWFjaCgoZmllbGQpID0+IHtcbiAgICAgICAgICAgIG1vbm9tZXJPYmplY3RbZmllbGRdID0gaXRbZmllbGRdO1xuICAgICAgICAgIH0pO1xuICAgICAgICAgIG1hcC5zZXQoaXRbSEVMTV9GSUVMRFMuU1lNQk9MXSwgbW9ub21lck9iamVjdCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9KTtcbiAgcmV0dXJuIG1hcDtcbn1cblxuLyogR2V0IGphZ2dlZCBhcnJheSBvZiBtb25vbWVyIHN5bWJvbHMgZm9yIHRoZSBkYXRhZnJhbWUgICovXG5mdW5jdGlvbiBnZXRNb25vbWVyU2VxdWVuY2VzQXJyYXkobWFjcm9Nb2xDb2w6IERHLkNvbHVtbjxzdHJpbmc+KTogc3RyaW5nW11bXSB7XG4gIGNvbnN0IGNvbHVtbkxlbmd0aCA9IG1hY3JvTW9sQ29sLmxlbmd0aDtcbiAgY29uc3QgcmVzdWx0OiBzdHJpbmdbXVtdID0gbmV3IEFycmF5KGNvbHVtbkxlbmd0aCk7XG5cbiAgLy8gc3BsaXQgdGhlIHN0cmluZyBpbnRvIG1vbm9tZXJzXG4gIGNvbnN0IGNvbFVuaXRzID0gbWFjcm9Nb2xDb2wuZ2V0VGFnKERHLlRBR1MuVU5JVFMpO1xuICBjb25zdCBzZXBhcmF0b3IgPSBtYWNyb01vbENvbC5nZXRUYWcoVEFHUy5zZXBhcmF0b3IpO1xuICBjb25zdCBzcGxpdHRlckZ1bmM6IFNwbGl0dGVyRnVuYyA9IGdldFNwbGl0dGVyKGNvbFVuaXRzLCBzZXBhcmF0b3IpO1xuXG4gIGZvciAobGV0IHJvdyA9IDA7IHJvdyA8IGNvbHVtbkxlbmd0aDsgKytyb3cpIHtcbiAgICBjb25zdCBtYWNyb01vbGVjdWxlID0gbWFjcm9Nb2xDb2wuZ2V0KHJvdyk7XG4gICAgLy8gdG9kbzogaGFuZGxlIHRoZSBleGNlcHRpb24gY2FzZSB3aGVuIG1hY3JvTW9sZWN1bGUgaXMgbnVsbFxuICAgIHJlc3VsdFtyb3ddID0gbWFjcm9Nb2xlY3VsZSA/IHNwbGl0dGVyRnVuYyhtYWNyb01vbGVjdWxlKSA6IFtdO1xuICB9XG4gIHJldHVybiByZXN1bHQ7XG59XG5cbi8qIEdldCBhIG1hcHBpbmcgb2YgbW9ub21lciBzeW1ib2xzIHRvIE1vbEdyYXBoIG9iamVjdHMgKi9cbmFzeW5jIGZ1bmN0aW9uIGdldE1vbm9tZXJzRGljdChcbiAgbW9ub21lclNlcXVlbmNlc0FycmF5OiBzdHJpbmdbXVtdLCBtb25vbWVyc0xpYkxpc3Q6IGFueVtdLCBwb2x5bWVyVHlwZTogSEVMTV9QT0xZTUVSX1RZUEUsIGFscGhhYmV0OiBBTFBIQUJFVFxuKTogUHJvbWlzZTxNYXA8c3RyaW5nLCBNb2xHcmFwaD4+IHtcbiAgLy8gdG9kbzogZXhjZXB0aW9uIC0gbm8gZ2Fwcywgbm8gZW1wdHkgc3RyaW5nIG1vbm9tZXJzXG4gIGNvbnN0IGZvcm1hdHRlZE1vbm9tZXJMaWIgPSBnZXRGb3JtYXR0ZWRNb25vbWVyTGliKG1vbm9tZXJzTGliTGlzdCwgcG9seW1lclR5cGUsIGFscGhhYmV0KTtcbiAgY29uc3QgbW9ub21lcnNEaWN0ID0gbmV3IE1hcDxzdHJpbmcsIE1vbEdyYXBoPigpO1xuXG4gIGNvbnN0IG1vZHVsZVJka2l0ID0gYXdhaXQgZ3Jvay5mdW5jdGlvbnMuY2FsbCgnQ2hlbTpnZXRSZEtpdE1vZHVsZScpO1xuXG4gIC8vIGFkZCBkZW94eXJpYm9zZS9yaWJvc2UgYW5kIHBob3NwaGF0ZSBmb3IgbnVjbGVvdGlkZSBzZXF1ZW5jZXNcbiAgaWYgKHBvbHltZXJUeXBlID09PSBIRUxNX1BPTFlNRVJfVFlQRS5STkEpIHtcbiAgICBjb25zdCBzeW1ib2xzID0gKGFscGhhYmV0ID09PSBBTFBIQUJFVC5STkEpID9cbiAgICAgIFtSSUJPU0UsIFBIT1NQSEFURV0gOiBbREVPWFlSSUJPU0UsIFBIT1NQSEFURV07XG4gICAgZm9yIChjb25zdCBzeW0gb2Ygc3ltYm9scylcbiAgICAgIHVwZGF0ZU1vbm9tZXJzRGljdChtb25vbWVyc0RpY3QsIHN5bSwgZm9ybWF0dGVkTW9ub21lckxpYiwgbW9kdWxlUmRraXQsIHBvbHltZXJUeXBlKTtcbiAgfVxuXG4gIGZvciAobGV0IHJvdyA9IDA7IHJvdyA8IG1vbm9tZXJTZXF1ZW5jZXNBcnJheS5sZW5ndGg7ICsrcm93KSB7XG4gICAgY29uc3QgbW9ub21lclNlcTogc3RyaW5nW10gPSBtb25vbWVyU2VxdWVuY2VzQXJyYXlbcm93XTtcbiAgICBmb3IgKGNvbnN0IHN5bSBvZiBtb25vbWVyU2VxKVxuICAgICAgdXBkYXRlTW9ub21lcnNEaWN0KG1vbm9tZXJzRGljdCwgc3ltLCBmb3JtYXR0ZWRNb25vbWVyTGliLCBtb2R1bGVSZGtpdCwgcG9seW1lclR5cGUpO1xuICB9XG4gIC8vIGNvbnNvbGUubG9nKG1vbm9tZXJzRGljdCk7XG5cbiAgcmV0dXJuIG1vbm9tZXJzRGljdDtcbn1cblxuLyogQWRkcyBNb2xHcmFwaCBvYmplY3QgZm9yICdzeW0nIHRvIHRoZSBtb25vbWVycyBkaWN0IHdoZW4gbmVjZXNzYXJ5ICAqL1xuZnVuY3Rpb24gdXBkYXRlTW9ub21lcnNEaWN0KFxuICBtb25vbWVyc0RpY3Q6IE1hcDxzdHJpbmcsIE1vbEdyYXBoPiwgc3ltOiBzdHJpbmcsXG4gIGZvcm1hdHRlZE1vbm9tZXJMaWI6IE1hcDxzdHJpbmcsIGFueT4sIG1vZHVsZVJka2l0OiBhbnksIHBvbHltZXJUeXBlOiBIRUxNX1BPTFlNRVJfVFlQRVxuKTogdm9pZCB7XG4gIGlmICghbW9ub21lcnNEaWN0LmhhcyhzeW0pKSB7XG4gICAgY29uc3QgbW9ub21lckRhdGE6IE1vbEdyYXBoIHwgbnVsbCA9IGdldE1vbEdyYXBoKHN5bSwgZm9ybWF0dGVkTW9ub21lckxpYiwgbW9kdWxlUmRraXQsIHBvbHltZXJUeXBlKTtcbiAgICBpZiAobW9ub21lckRhdGEpXG4gICAgICBtb25vbWVyc0RpY3Quc2V0KHN5bSwgbW9ub21lckRhdGEpO1xuICAgIGVsc2VcbiAgICAgIHRocm93IG5ldyBFcnJvcihgTW9ub21lciB3aXRoIHN5bWJvbCAnJHtzeW19JyBpcyBhYnNlbnQgdGhlIG1vbm9tZXIgbGlicmFyeWApO1xuICAgIC8vIHRvZG86IGhhbmRsZSBleGNlcHRpb25cbiAgfVxufVxuXG4vKiBDb25zdHJ1Y3QgdGhlIE1vbEdyYXBoIG9iamVjdCBmb3Igc3BlY2lmaWVkIG1vbm9tZXJTeW1ib2w6IHRoZSBhc3NvY2lhdGVkXG4gKiBncmFwaCBpcyBhZGp1c3RlZCBpbiBYWSBwbGFuZSBhbmQgZmlsbGVkIHdpdGggZGVmYXVsdCBSLWdyb3VwcyAqL1xuZnVuY3Rpb24gZ2V0TW9sR3JhcGgoXG4gIG1vbm9tZXJTeW1ib2w6IHN0cmluZywgZm9ybWF0dGVkTW9ub21lckxpYjogTWFwPHN0cmluZywgYW55PixcbiAgbW9kdWxlUmRraXQ6IGFueSwgcG9seW1lclR5cGU6IEhFTE1fUE9MWU1FUl9UWVBFIC8vIHRvZG86IHNwZWNpZnkgdHlwZSBmb3IgbW9kdWxlUmRraXRcbik6IE1vbEdyYXBoIHwgbnVsbCB7XG4gIGlmICghZm9ybWF0dGVkTW9ub21lckxpYi5oYXMobW9ub21lclN5bWJvbCkpIHtcbiAgICByZXR1cm4gbnVsbDtcbiAgfSBlbHNlIHtcbiAgICBjb25zdCBsaWJPYmplY3QgPSBmb3JtYXR0ZWRNb25vbWVyTGliLmdldChtb25vbWVyU3ltYm9sKTtcbiAgICBjb25zdCBjYXBHcm91cHMgPSBwYXJzZUNhcEdyb3VwcyhsaWJPYmplY3RbSEVMTV9GSUVMRFMuUkdST1VQU10pO1xuICAgIGNvbnN0IGNhcEdyb3VwSWR4TWFwID0gcGFyc2VDYXBHcm91cElkeE1hcChsaWJPYmplY3RbSEVMTV9GSUVMRFMuTU9MRklMRV0pO1xuICAgIGNvbnN0IG1vbGZpbGVWM0sgPSBjb252ZXJ0TW9sZmlsZVRvVjNLKHJlbW92ZVJHcm91cExpbmVzKGxpYk9iamVjdFtIRUxNX0ZJRUxEUy5NT0xGSUxFXSksIG1vZHVsZVJka2l0KTtcbiAgICBjb25zdCBjb3VudHMgPSBwYXJzZUF0b21BbmRCb25kQ291bnRzKG1vbGZpbGVWM0spO1xuXG4gICAgY29uc3QgYXRvbXMgPSBwYXJzZUF0b21CbG9jayhtb2xmaWxlVjNLLCBjb3VudHMuYXRvbUNvdW50KTtcbiAgICBjb25zdCBib25kcyA9IHBhcnNlQm9uZEJsb2NrKG1vbGZpbGVWM0ssIGNvdW50cy5ib25kQ291bnQpO1xuICAgIGNvbnN0IG1ldGEgPSBnZXRNb25vbWVyTWV0YWRhdGEoYXRvbXMsIGJvbmRzLCBjYXBHcm91cHMsIGNhcEdyb3VwSWR4TWFwKTtcblxuICAgIGNvbnN0IG1vbm9tZXJHcmFwaDogTW9sR3JhcGggPSB7YXRvbXM6IGF0b21zLCBib25kczogYm9uZHMsIG1ldGE6IG1ldGF9O1xuXG4gICAgaWYgKHBvbHltZXJUeXBlID09PSBIRUxNX1BPTFlNRVJfVFlQRS5QRVBUSURFKSB7XG4gICAgICBhZGp1c3RQZXB0aWRlTW9ub21lckdyYXBoKG1vbm9tZXJHcmFwaCk7XG4gICAgfSBlbHNlIHsgLy8gbnVjbGVvdGlkZXNcbiAgICAgIGlmIChtb25vbWVyU3ltYm9sID09PSBSSUJPU0UgfHwgbW9ub21lclN5bWJvbCA9PT0gREVPWFlSSUJPU0UpXG4gICAgICAgIGFkanVzdFN1Z2FyTW9ub21lckdyYXBoKG1vbm9tZXJHcmFwaCk7XG4gICAgICBlbHNlIGlmIChtb25vbWVyU3ltYm9sID09PSBQSE9TUEhBVEUpXG4gICAgICAgIGFkanVzdFBob3NwaGF0ZU1vbm9tZXJHcmFwaChtb25vbWVyR3JhcGgpO1xuICAgICAgZWxzZVxuICAgICAgICBhZGp1c3RCYXNlTW9ub21lckdyYXBoKG1vbm9tZXJHcmFwaCk7XG4gICAgfVxuXG4gICAgLy8gcmVtb3ZlIHRoZSAncmlnaHRtb3N0JyBjaGFpbi1leHRlbmRpbmcgci1ncm91cCBub2RlIGluIHRoZSBiYWNrYm9uZVxuICAgIGlmIChwb2x5bWVyVHlwZSA9PT0gSEVMTV9QT0xZTUVSX1RZUEUuUEVQVElERSkge1xuICAgICAgc2V0U2hpZnRzKG1vbm9tZXJHcmFwaCwgcG9seW1lclR5cGUpO1xuICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzFdKTtcbiAgICB9IGVsc2UgeyAvLyBudWNsZW90aWRlc1xuICAgICAgaWYgKG1vbm9tZXJTeW1ib2wgPT09IFJJQk9TRSB8fCBtb25vbWVyU3ltYm9sID09PSBERU9YWVJJQk9TRSkge1xuICAgICAgICAvLyByZW1vdmUgUjJcbiAgICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzFdKTtcbiAgICAgICAgLy8gc2V0IHRlcm1pbmFsTm9kZTIgKG94eWdlbikgYXMgbmV3IFIyXG4gICAgICAgIG1vbm9tZXJHcmFwaC5tZXRhLnJOb2Rlc1sxXSA9IG1vbm9tZXJHcmFwaC5tZXRhLnRlcm1pbmFsTm9kZXNbMV07XG4gICAgICAgIHNldFRlcm1pbmFsTm9kZXMobW9ub21lckdyYXBoLmJvbmRzLCBtb25vbWVyR3JhcGgubWV0YSk7IC8vIHNldCB0ZXJtaW5hbCBub2RlcyBhbmV3XG4gICAgICAgIHNldFNoaWZ0cyhtb25vbWVyR3JhcGgsIHBvbHltZXJUeXBlKTtcbiAgICAgICAgLy8gcmVtb3ZlICduZXcnIFIyIChveHlnZW4pXG4gICAgICAgIHJlbW92ZU5vZGVBbmRCb25kcyhtb25vbWVyR3JhcGgsIG1vbm9tZXJHcmFwaC5tZXRhLnJOb2Rlc1sxXSk7XG4gICAgICAgIC8vIHJlbW92ZSBSMVxuICAgICAgICByZW1vdmVOb2RlQW5kQm9uZHMobW9ub21lckdyYXBoLCBtb25vbWVyR3JhcGgubWV0YS5yTm9kZXNbMF0pO1xuICAgICAgICAvLyByZW1vdmUgdGhlIGJyYW5jaGluZyByLWdyb3VwXG4gICAgICAgIHJlbW92ZU5vZGVBbmRCb25kcyhtb25vbWVyR3JhcGgsIG1vbm9tZXJHcmFwaC5tZXRhLnJOb2Rlc1syXSk7XG4gICAgICB9IGVsc2UgaWYgKG1vbm9tZXJTeW1ib2wgPT09IFBIT1NQSEFURSkge1xuICAgICAgICBtb25vbWVyR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdID0gbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzBdO1xuICAgICAgICBzaGlmdENvb3JkaW5hdGVzKFxuICAgICAgICAgIG1vbm9tZXJHcmFwaCxcbiAgICAgICAgICAtbW9ub21lckdyYXBoLmF0b21zLnhbbW9ub21lckdyYXBoLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDFdLFxuICAgICAgICAgIC1tb25vbWVyR3JhcGguYXRvbXMueVttb25vbWVyR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMV1cbiAgICAgICAgKTtcbiAgICAgICAgc2V0U2hpZnRzKG1vbm9tZXJHcmFwaCwgcG9seW1lclR5cGUpO1xuICAgICAgICByZW1vdmVOb2RlQW5kQm9uZHMobW9ub21lckdyYXBoLCBtb25vbWVyR3JhcGgubWV0YS5yTm9kZXNbMV0pO1xuICAgICAgfSBlbHNlIHsgLy8gbnVjbGVvYmFzZXNcbiAgICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgbW9ub21lckdyYXBoLm1ldGEuck5vZGVzWzBdKTtcbiAgICAgIH1cbiAgICB9XG4gICAgcmVtb3ZlSHlkcm9nZW4obW9ub21lckdyYXBoKTtcblxuICAgIHJldHVybiBtb25vbWVyR3JhcGg7XG4gIH1cbn1cblxuLy8gdG9kbzogZG9jXG5mdW5jdGlvbiBnZXRNb25vbWVyTWV0YWRhdGEoXG4gIGF0b21zOiBBdG9tcywgYm9uZHM6IEJvbmRzLCBjYXBHcm91cHM6IHN0cmluZ1tdLCBjYXBHcm91cElkeE1hcDogTWFwPG51bWJlciwgbnVtYmVyPlxuKTogTW9ub21lck1ldGFkYXRhIHtcbiAgY29uc3QgbWV0YTogTW9ub21lck1ldGFkYXRhID0ge1xuICAgIGJhY2tib25lU2hpZnQ6IG51bGwsXG4gICAgYnJhbmNoU2hpZnQ6IG51bGwsXG4gICAgdGVybWluYWxOb2RlczogW10sXG4gICAgck5vZGVzOiBbXSxcbiAgfTtcblxuICBzdWJzdGl0dXRlQ2FwR3JvdXBzKGF0b21zLCBjYXBHcm91cHMsIGNhcEdyb3VwSWR4TWFwKTtcbiAgc2V0Uk5vZGVzKGNhcEdyb3VwSWR4TWFwLCBtZXRhKTtcbiAgc2V0VGVybWluYWxOb2Rlcyhib25kcywgbWV0YSk7XG4gIHJldHVybiBtZXRhO1xufVxuXG4vKiBQYXJzZSBlbGVtZW50IHN5bWJvbHMgZm9yIFItZ3JvdXBzIGZyb20gdGhlIEhFTE0gbW9ub21lciBsaWJyYXJ5IFItZ3JvdXBzXG4gKiBmaWVsZCAgKi9cbmZ1bmN0aW9uIHBhcnNlQ2FwR3JvdXBzKHJHcm91cE9iakxpc3Q6IGFueVtdKTogc3RyaW5nW10ge1xuICAvLyBzcGVjaWZpY2FsbHkgZm9yIEhFTE1Db3JlTGlicmFyeVxuICAvLyBjb25zaWRlcmVkIG9ubHkgbW9ub2F0b21pYyByZ3JvdXBzXG4gIC8vIHN1cHBvc2luZyB0aGF0IGVsZW1lbnRzIGluIHJHcm91cE9iakxpc3QgYXJlIHNvcnRlZCB3LnIudC4gdGhlIHJncm91cHMgaWR4XG4gIC8vIHRvZG86IHBvc3NpYmxlIGdlbmVyYWxpemF0aW9uc1xuICBjb25zdCBjYXBHcm91cHNBcnJheTogc3RyaW5nW10gPSBbXTtcbiAgZm9yIChjb25zdCBvYmogb2Ygckdyb3VwT2JqTGlzdCkge1xuICAgIGxldCBjYXBHcm91cDogc3RyaW5nID0gb2JqW1JHUk9VUF9GSUVMRFMuQ0FQX0dST1VQX1NNSUxFU107XG5cbiAgICAvLyBpbiBzb21lIGNhc2VzIHRoZSBzbWlsZXMgZmllbGQgaXMgd3JpdHRlbiB3aXRoIHVwcGVyY2FzZVxuICAgIGlmICghY2FwR3JvdXApXG4gICAgICBjYXBHcm91cCA9IG9ialtSR1JPVVBfRklFTERTLkNBUF9HUk9VUF9TTUlMRVNfVVBQRVJDQVNFXTtcbiAgICAvLyB0b2RvOiB2ZXJpZnkgdGhhdCB0aGVyZSBhcmUgbm8gbXVsdGktZWxlbWVudCBjYXAgZ3JvdXBzLCBvciBjb25zaWRlciBob3cgdG9cbiAgICAvLyB0cmFuc2Zvcm0gdGhlbVxuICAgIGNhcEdyb3VwID0gY2FwR3JvdXAucmVwbGFjZSgvKFxcW3xcXF18XFwqfDp8XFxkKS9nLCAnJyk7XG4gICAgaWYgKGNhcEdyb3VwLmxlbmd0aCA+IDEpIC8vIHRvZG86IGNoZWNrIGlmIHN1Y2ggY2FzZXMgYXJlIHBvc3NpYmxlLCByZW1vdmUgaWYgbm90XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0RlZmF1bHQgY2FwIGdyb3VwIGhhcyBsZW5ndGggbW9yZSB0aGFuIG9uZScpO1xuICAgIGNhcEdyb3Vwc0FycmF5LnB1c2goY2FwR3JvdXApO1xuICB9XG4gIHJldHVybiBjYXBHcm91cHNBcnJheTtcbn1cblxuLyogU3Vic3RpdHV0ZSB0aGUgY2FwIGdyb3VwIGVsZW1lbnRzIGluc3RlYWQgb2YgUiMgKi9cbmZ1bmN0aW9uIHN1YnN0aXR1dGVDYXBHcm91cHMoXG4gIGF0b21zOiBBdG9tcywgY2FwR3JvdXBzOiBzdHJpbmdbXSwgY2FwR3JvdXBJZHhNYXA6IE1hcDxudW1iZXIsIG51bWJlcj5cbik6IHZvaWQge1xuICBmb3IgKGNvbnN0IFtub2RlLCBjYXBJZHhdIG9mIGNhcEdyb3VwSWR4TWFwKVxuICAgIGF0b21zLmF0b21UeXBlc1tub2RlIC0gMV0gPSBjYXBHcm91cHNbY2FwSWR4IC0gMV07IC8vIC0xIGJlY2F1c2UgbW9sZmlsZSBpbmRleGluZyBzdGFydHMgZnJvbSAxXG59XG5cbi8vdG9kbzogZG9jXG5mdW5jdGlvbiBzZXRSTm9kZXMoY2FwR3JvdXBJZHhNYXA6IE1hcDxudW1iZXIsIG51bWJlcj4sIG1ldGE6IE1vbm9tZXJNZXRhZGF0YSk6IHZvaWQge1xuICBtZXRhLnJOb2RlcyA9IEFycmF5LmZyb20oY2FwR3JvdXBJZHhNYXAua2V5cygpKTtcbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBtZXRhLnJOb2Rlcy5sZW5ndGg7IGkrKykge1xuICAgIGZvciAoY29uc3QgaiBvZiBbMSwgMl0pIHsgLy8gMSBhbmQgMiBieSBkZWYuIGNvcnJlc3BvbmQgdG8gJ2xlZnQvcmlnaHRtb3N0JyByLW5vZGVzXG4gICAgICAvLyBzd2FwIHRoZSB2YWx1ZXMgaWYgbmVjZXNzYXJ5LCBzbyB0aGF0IHRoZSBcImxlZnRtb3N0XCIgci1ub2RlIGlzIGF0IDAsXG4gICAgICAvLyBhbmQgdGhlICdyaWdodG1vc3QnLCBhdCAxXG4gICAgICBpZiAoY2FwR3JvdXBJZHhNYXAuZ2V0KG1ldGEuck5vZGVzW2ldKSA9PT0gaikge1xuICAgICAgICBjb25zdCB0bXAgPSBtZXRhLnJOb2Rlc1tqIC0gMV07XG4gICAgICAgIG1ldGEuck5vZGVzW2ogLSAxXSA9IG1ldGEuck5vZGVzW2ldO1xuICAgICAgICBtZXRhLnJOb2Rlc1tpXSA9IHRtcDtcbiAgICAgIH1cbiAgICB9XG4gIH1cbn1cblxuLy90b2RvOiBkb2NcbmZ1bmN0aW9uIHNldFRlcm1pbmFsTm9kZXMoYm9uZHM6IEJvbmRzLCBtZXRhOiBNb25vbWVyTWV0YWRhdGEpOiB2b2lkIHtcbiAgY29uc3Qgck5vZGVzID0gbWV0YS5yTm9kZXM7XG4gIG1ldGEudGVybWluYWxOb2RlcyA9IG5ldyBBcnJheTxudW1iZXI+KHJOb2Rlcy5sZW5ndGgpLmZpbGwoMCk7XG4gIGNvbnN0IHRlcm1pbmFsTm9kZXMgPSBtZXRhLnRlcm1pbmFsTm9kZXM7XG4gIGNvbnN0IGF0b21QYWlycyA9IGJvbmRzLmF0b21QYWlycztcbiAgbGV0IGkgPSAwO1xuICBsZXQgaiA9IDA7XG4gIHdoaWxlICgoaSA8IGF0b21QYWlycy5sZW5ndGgpICYmIGogPCB0ZXJtaW5hbE5vZGVzLmxlbmd0aCkge1xuICAgIC8vIHJOb2RlcyBhcnJheSBpcyBzb3J0ZWQgc28gdGhhdCBpdHMgMHRoIGFuZCAxc3QgZWxlbWVudHMgKGlmIGJvdGhcbiAgICAvLyBwcmVzZW50KSBjb3JyZXNwb25kIHRvIHRoZSBjaGFpbiBleHRlbmRpbmcgKGkuZS4gbm90IGJyYW5jaGluZykgci1ncm91cHNcbiAgICBmb3IgKGxldCBrID0gMDsgayA8IHRlcm1pbmFsTm9kZXMubGVuZ3RoOyArK2spIHtcbiAgICAgIGZvciAobGV0IGwgPSAwOyBsIDwgMjsgKytsKSB7XG4gICAgICAgIGlmIChhdG9tUGFpcnNbaV1bbF0gPT09IHJOb2Rlc1trXSkge1xuICAgICAgICAgIHRlcm1pbmFsTm9kZXNba10gPSBhdG9tUGFpcnNbaV1bKGwgKyAxKSUyXTtcbiAgICAgICAgICBpZiAock5vZGVzLmxlbmd0aCA+IDIpIHtcbiAgICAgICAgICB9XG4gICAgICAgICAgKytqO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICAgICsraTtcbiAgfVxufVxuXG4vL3RvZG86IGRvY1xuZnVuY3Rpb24gc2V0U2hpZnRzKG1vbEdyYXBoOiBNb2xHcmFwaCwgcG9seW1lclR5cGU6IEhFTE1fUE9MWU1FUl9UWVBFKTogdm9pZCB7XG4gIGlmIChtb2xHcmFwaC5tZXRhLnJOb2Rlcy5sZW5ndGggPiAxKSB7XG4gICAgbW9sR3JhcGgubWV0YS5iYWNrYm9uZVNoaWZ0ID0gW1xuICAgICAga2VlcFByZWNpc2lvbihcbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnJOb2Rlc1sxXSAtIDFdIC1cbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnRlcm1pbmFsTm9kZXNbMF0gLSAxXVxuICAgICAgKSxcbiAgICAgIGtlZXBQcmVjaXNpb24oXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS5yTm9kZXNbMV0gLSAxXSAtXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMV1cbiAgICAgICksXG4gICAgXTtcbiAgfVxuXG4gIGlmIChwb2x5bWVyVHlwZSA9PT0gSEVMTV9QT0xZTUVSX1RZUEUuUk5BICYmIG1vbEdyYXBoLm1ldGEuck5vZGVzLmxlbmd0aCA+IDIpIHtcbiAgICBtb2xHcmFwaC5tZXRhLmJyYW5jaFNoaWZ0ID0gW1xuICAgICAga2VlcFByZWNpc2lvbihcbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnJOb2Rlc1syXSAtIDFdIC1cbiAgICAgICAgbW9sR3JhcGguYXRvbXMueFttb2xHcmFwaC5tZXRhLnRlcm1pbmFsTm9kZXNbMF0gLSAxXVxuICAgICAgKSxcbiAgICAgIGtlZXBQcmVjaXNpb24oXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS5yTm9kZXNbMl0gLSAxXSAtXG4gICAgICAgIG1vbEdyYXBoLmF0b21zLnlbbW9sR3JhcGgubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMV1cbiAgICAgICksXG4gICAgXTtcbiAgfVxufVxuXG4vKiBIZWxwZXIgZnVuY3Rpb24gbmVjZXNzYXJ5IHRvIGJ1aWxkIGEgY29ycmVjdCBWMzAwMCBtb2xmaWxlIG91dCBvZiBWMjAwMCB3aXRoXG4gKiBzcGVjaWZpZWQgci1ncm91cHMqL1xuZnVuY3Rpb24gcmVtb3ZlUkdyb3VwTGluZXMobW9sZmlsZVYySzogc3RyaW5nKTogc3RyaW5nIHtcbiAgbGV0IGJlZ2luID0gbW9sZmlsZVYySy5pbmRleE9mKFYyS19BX0xJTkUsIDApO1xuICBpZiAoYmVnaW4gPT09IC0xKVxuICAgIGJlZ2luID0gbW9sZmlsZVYySy5pbmRleE9mKFYyS19SR1BfTElORSk7XG4gIGNvbnN0IGVuZCA9IG1vbGZpbGVWMksuaW5kZXhPZihWM0tfRU5ELCBiZWdpbik7XG4gIHJldHVybiBtb2xmaWxlVjJLLnN1YnN0cmluZygwLCBiZWdpbikgKyBtb2xmaWxlVjJLLnN1YnN0cmluZyhlbmQpO1xufVxuXG4vKiBWMjAwMCB0byBWMzAwMCBjb252ZXJ0ZXIgICovXG5mdW5jdGlvbiBjb252ZXJ0TW9sZmlsZVRvVjNLKG1vbGZpbGVWMks6IHN0cmluZywgbW9kdWxlUmRraXQ6IGFueSk6IHN0cmluZyB7XG4gIC8vIHRvZG86IHR5cGUgb2YgbW9kdWxlUmRraXRcbiAgLy8gdG9kbzogY29uc2lkZXIgdGhlIHVzZSBvZiBzdGFuZGFyZCBDaGVtIGNvbnZlcnRlciAocmVsaWVzIG9uIGNyZWF0aW9uIG9mIG1vZHVsZVJka2l0IG9uIGVhY2ggaXRlcmF0aW9uLCB0aG91Z2gpXG4gIGNvbnN0IG1vbE9iaiA9IG1vZHVsZVJka2l0LmdldF9tb2wobW9sZmlsZVYySyk7XG4gIGNvbnN0IG1vbGZpbGVWM0sgPSBtb2xPYmouZ2V0X3YzS21vbGJsb2NrKCk7XG4gIG1vbE9iai5kZWxldGUoKTtcbiAgcmV0dXJuIG1vbGZpbGVWM0s7XG59XG5cbi8qIFBhcnNlIFYzMDAwIGJvbmQgYmxvY2sgYW5kIGNvbnN0cnVjdCB0aGUgQm9uZHMgb2JqZWN0ICovXG5mdW5jdGlvbiBwYXJzZUJvbmRCbG9jayhtb2xmaWxlVjNLOiBzdHJpbmcsIGJvbmRDb3VudDogbnVtYmVyKTogQm9uZHMge1xuICAvLyB0b2RvOiBjb25zaWRlciB0aGUgY2FzZSB3aGVuIHRoZXJlIGlzIG5vIHNpbXBsZSBsZWZ0bW9zdC9yaWdodG1vc3QgY2hvaWNlXG4gIC8vIHRvZG86IGNvbnNpZGVyIHRoZSBjYXNlIHdoZW4gdGhlcmUgYXJlIG11bHRpcGxlIGNvbnNlcXVlbnQgTSAgUkdQIGxpbmVzLFxuICAvLyBsaWtlIGluIEhFTE1Db3JlTGlicmFyeSBudWNsZW90aWRlc1xuXG4gIGNvbnN0IGJvbmRUeXBlczogbnVtYmVyW10gPSBuZXcgQXJyYXkoYm9uZENvdW50KTtcbiAgY29uc3QgYXRvbVBhaXJzOiBudW1iZXJbXVtdID0gbmV3IEFycmF5KGJvbmRDb3VudCk7XG4gIGNvbnN0IGJvbmRDb25maWd1cmF0aW9uID0gbmV3IE1hcDxudW1iZXIsIG51bWJlcj4oKTtcbiAgY29uc3Qga3dhcmdzID0gbmV3IE1hcDxudW1iZXIsIHN0cmluZz47XG5cbiAgbGV0IGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKFYzS19CRUdJTl9CT05EX0JMT0NLKTtcbiAgYmVnaW4gPSBtb2xmaWxlVjNLLmluZGV4T2YoJ1xcbicsIGJlZ2luKTtcbiAgbGV0IGVuZCA9IGJlZ2luO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IGJvbmRDb3VudDsgKytpKSB7XG4gICAgLy8gcGFyc2UgYm9uZCB0eXBlIGFuZCBhdG9tIHBhaXJcbiAgICBjb25zdCBwYXJzZWRWYWx1ZXM6IG51bWJlcltdID0gbmV3IEFycmF5KDMpO1xuICAgIGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKFYzS19CRUdJTl9EQVRBX0xJTkUsIGVuZCkgKyBWM0tfSURYX1NISUZUO1xuICAgIGVuZCA9IG1vbGZpbGVWM0suaW5kZXhPZignICcsIGJlZ2luKTtcbiAgICBmb3IgKGxldCBrID0gMDsgayA8IDM7ICsraykge1xuICAgICAgYmVnaW4gPSBlbmQgKyAxO1xuICAgICAgZW5kID0gTWF0aC5taW4obW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbiksIG1vbGZpbGVWM0suaW5kZXhPZignICcsIGJlZ2luKSk7XG4gICAgICBwYXJzZWRWYWx1ZXNba10gPSBwYXJzZUludChtb2xmaWxlVjNLLnNsaWNlKGJlZ2luLCBlbmQpKTtcbiAgICB9XG4gICAgYm9uZFR5cGVzW2ldID0gcGFyc2VkVmFsdWVzWzBdO1xuICAgIGF0b21QYWlyc1tpXSA9IHBhcnNlZFZhbHVlcy5zbGljZSgxKTtcblxuICAgIC8vIHBhcnNlIGtleXdvcmQgYXJndW1lbnRzXG4gICAgY29uc3QgZW5kT2ZMaW5lID0gbW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gICAgbGV0IGxpbmVSZW1haW5kZXIgPSBtb2xmaWxlVjNLLnNsaWNlKGVuZCwgZW5kT2ZMaW5lKTtcbiAgICBsZXQgYmVnaW5DZmcgPSBsaW5lUmVtYWluZGVyLmluZGV4T2YoVjNLX0JPTkRfQ09ORklHKTtcbiAgICBpZiAoYmVnaW5DZmcgIT09IC0xKSB7XG4gICAgICBiZWdpbkNmZyA9IGxpbmVSZW1haW5kZXIuaW5kZXhPZignPScsIGJlZ2luQ2ZnKSArIDE7XG4gICAgICBsZXQgZW5kQ2ZnID0gbGluZVJlbWFpbmRlci5pbmRleE9mKCcgJywgYmVnaW5DZmcpO1xuICAgICAgaWYgKGVuZENmZyA9PT0gLTEpXG4gICAgICAgIGVuZENmZyA9IGxpbmVSZW1haW5kZXIubGVuZ3RoO1xuICAgICAgY29uc3QgYm9uZENvbmZpZyA9IHBhcnNlSW50KGxpbmVSZW1haW5kZXIuc2xpY2UoYmVnaW5DZmcsIGVuZENmZykpO1xuICAgICAgYm9uZENvbmZpZ3VyYXRpb24uc2V0KGksIGJvbmRDb25maWcpO1xuICAgICAgY29uc3QgcmVtb3ZlZFN1YnN0cmluZyA9IFYzS19CT05EX0NPTkZJRyArIGJvbmRDb25maWcudG9TdHJpbmcoKTtcbiAgICAgIGxpbmVSZW1haW5kZXIgPSBsaW5lUmVtYWluZGVyLnJlcGxhY2UocmVtb3ZlZFN1YnN0cmluZywgJycpO1xuICAgIH1cbiAgICBpZiAoIWxpbmVSZW1haW5kZXIpXG4gICAgICBrd2FyZ3Muc2V0KGksIGxpbmVSZW1haW5kZXIpO1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBib25kVHlwZXM6IGJvbmRUeXBlcyxcbiAgICBhdG9tUGFpcnM6IGF0b21QYWlycyxcbiAgICBib25kQ29uZmlndXJhdGlvbjogYm9uZENvbmZpZ3VyYXRpb24sXG4gICAga3dhcmdzOiBrd2FyZ3MsXG4gIH07XG59XG5cbi8qIENvbnN0cnVjdHMgbWFwcGluZyBvZiByLWdyb3VwIG5vZGVzIHRvIGRlZmF1bHQgY2FwR3JvdXBzLCBhbGwgbnVtZXJhdGlvbiBzdGFydGluZyBmcm9tIDEuXG4gKiBBY2NvcmRpbmcgdG8gaHR0cHM6Ly9wdWJzLmFjcy5vcmcvZG9pLzEwLjEwMjEvY2kzMDAxOTI1LCBSMSBhbmQgUjIgYXJlIHRoZSBjaGFpbiBleHRlbmRpbmcgYXR0YWNobWVudCBwb2ludHMsXG4gKiB3aGlsZSBSMyBpcyB0aGUgYnJhbmNoaW5nIGF0dGFjaG1lbnQgcG9pbnQuICovXG5mdW5jdGlvbiBwYXJzZUNhcEdyb3VwSWR4TWFwKG1vbGZpbGVWMks6IHN0cmluZyk6IE1hcDxudW1iZXIsIG51bWJlcj4ge1xuICBjb25zdCBjYXBHcm91cElkeE1hcCA9IG5ldyBNYXA8bnVtYmVyLCBudW1iZXI+KCk7XG5cbiAgLy8gcGFyc2UgQS1saW5lcyAoUk5BKVxuICBsZXQgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoVjJLX0FfTElORSwgMCk7XG4gIGxldCBlbmQgPSBiZWdpbjtcbiAgd2hpbGUgKGJlZ2luICE9PSAtMSkge1xuICAgIC8vIHBhcnNlIHRoZSByTm9kZSB0byB3aGljaCB0aGUgY2FwIGdyb3VwIGlzIGF0dGFjaGVkXG4gICAgZW5kID0gbW9sZmlsZVYySy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gICAgY29uc3Qgck5vZGUgPSBwYXJzZUludChtb2xmaWxlVjJLLnN1YnN0cmluZyhiZWdpbiwgZW5kKS5yZXBsYWNlKC9eQVxccysvLCAnJykpO1xuXG4gICAgLy8gcGFyc2UgdGhlIGNhcEdyb3VwIGluZGV4XG4gICAgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoJ1InLCBlbmQpO1xuICAgIGVuZCA9IG1vbGZpbGVWMksuaW5kZXhPZignXFxuJywgYmVnaW4pO1xuICAgIGNvbnN0IGNhcEdyb3VwID0gcGFyc2VJbnQobW9sZmlsZVYySy5zdWJzdHJpbmcoYmVnaW4sIGVuZCkucmVwbGFjZSgvXlIvLCAnJykpO1xuICAgIGNhcEdyb3VwSWR4TWFwLnNldChyTm9kZSwgY2FwR3JvdXApO1xuXG4gICAgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoVjJLX0FfTElORSwgZW5kKTtcbiAgfVxuXG4gIC8vIHBhcnNlIFJHUCBsaW5lcyAobWF5IGJlIG1vcmUgdGhhbiBvbmUgaW4gUk5BIG1vbm9tZXJzKVxuICBiZWdpbiA9IG1vbGZpbGVWMksuaW5kZXhPZihWMktfUkdQX0xJTkUsIDApO1xuICBlbmQgPSBtb2xmaWxlVjJLLmluZGV4T2YoJ1xcbicsIGJlZ2luKTtcbiAgd2hpbGUgKGJlZ2luICE9PSAtMSkge1xuICAgIGJlZ2luICs9IFYyS19SR1BfU0hJRlQ7XG4gICAgZW5kID0gbW9sZmlsZVYySy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gICAgY29uc3QgcmdwU3RyaW5nUGFyc2VkID0gbW9sZmlsZVYySy5zdWJzdHJpbmcoYmVnaW4sIGVuZClcbiAgICAgIC5yZXBsYWNlQWxsKC9cXHMrL2csICcgJylcbiAgICAgIC5zcGxpdCgnICcpO1xuICAgIGNvbnN0IHJncEluZGljZXNBcnJheSA9IHJncFN0cmluZ1BhcnNlZC5tYXAoKGVsKSA9PiBwYXJzZUludChlbCkpXG4gICAgICAuc2xpY2UoMSk7IC8vIHNsaWNlIGZyb20gMSBiZWNhdXNlIHRoZSAxc3QgdmFsdWUgaXMgdGhlIG51bWJlciBvZiBwYWlycyBpbiB0aGUgbGluZVxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgcmdwSW5kaWNlc0FycmF5Lmxlbmd0aDsgaSArPSAyKSB7XG4gICAgICAvLyBub3RpY2U6IHRoZXJlIG1heSBiZSBjb25mbGljdGluZyBjYXAgZ3JvdXAgZGVmaW5pdGlvbnMsIGxpa2UgMy1PLU1ldGh5bHJpYm9zZSAoMiw1IGNvbm5lY3Rpdml0eSlcbiAgICAgIC8vICh0aGUgbGFzdCBtb25vbWVyIGluIEhFTE1Db3JlTGlicmFyeSlcbiAgICAgIC8vIHRoZXJlIHRoZSBpbmRpY2VzIG9mIGNhcCBncm91cHMgYXJlIHNlbGYtY29udHJhZGljdG9yeVxuICAgICAgLy8gdG9kbzogY2xhcmlmeSB3aHkgc3VjaCBzaXR1YXRpb25zIG9jY3VyIGluIHByaW5jaXBsZVxuICAgICAgaWYgKGNhcEdyb3VwSWR4TWFwLmhhcyhyZ3BJbmRpY2VzQXJyYXlbaV0pICYmIGNhcEdyb3VwSWR4TWFwLmdldChyZ3BJbmRpY2VzQXJyYXlbaV0pICE9PSByZ3BJbmRpY2VzQXJyYXlbaSArIDFdKVxuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYHItZ3JvdXAgaW5kZXggJHtyZ3BJbmRpY2VzQXJyYXlbaV19IGhhcyBhbHJlYWR5IGJlZW4gYWRkZWQgd2l0aCBhIGRpZmZlcmVudCB2YWx1ZWApO1xuICAgICAgZWxzZVxuICAgICAgICBjYXBHcm91cElkeE1hcC5zZXQocmdwSW5kaWNlc0FycmF5W2ldLCByZ3BJbmRpY2VzQXJyYXlbaSsxXSk7XG4gICAgfVxuXG4gICAgYmVnaW4gPSBtb2xmaWxlVjJLLmluZGV4T2YoVjJLX1JHUF9MSU5FLCBlbmQpO1xuICB9XG5cbiAgcmV0dXJuIGNhcEdyb3VwSWR4TWFwO1xufVxuXG5mdW5jdGlvbiBwYXJzZUF0b21BbmRCb25kQ291bnRzKG1vbGZpbGVWM0s6IHN0cmluZyk6IHthdG9tQ291bnQ6IG51bWJlciwgYm9uZENvdW50OiBudW1iZXJ9IHtcbiAgbW9sZmlsZVYzSyA9IG1vbGZpbGVWM0sucmVwbGFjZUFsbCgnXFxyJywgJycpOyAvLyB0byBoYW5kbGUgb2xkIGFuZCBuZXcgc2RmIHN0YW5kYXJkc1xuXG4gIC8vIHBhcnNlIGF0b20gY291bnRcbiAgbGV0IGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKFYzS19CRUdJTl9DT1VOVFNfTElORSkgKyBWM0tfQ09VTlRTX1NISUZUO1xuICBsZXQgZW5kID0gbW9sZmlsZVYzSy5pbmRleE9mKCcgJywgYmVnaW4pO1xuICBjb25zdCBudW1PZkF0b21zID0gcGFyc2VJbnQobW9sZmlsZVYzSy5zdWJzdHJpbmcoYmVnaW4sIGVuZCkpO1xuXG4gIC8vIHBhcnNlIGJvbmQgY291bnRcbiAgYmVnaW4gPSBlbmQgKyAxO1xuICBlbmQgPSBtb2xmaWxlVjNLLmluZGV4T2YoJyAnLCBiZWdpbik7XG4gIGNvbnN0IG51bU9mQm9uZHMgPSBwYXJzZUludChtb2xmaWxlVjNLLnN1YnN0cmluZyhiZWdpbiwgZW5kKSk7XG5cbiAgcmV0dXJuIHthdG9tQ291bnQ6IG51bU9mQXRvbXMsIGJvbmRDb3VudDogbnVtT2ZCb25kc307XG59XG5cbi8qIFBhcnNlIFYzMDAwIGF0b20gYmxvY2sgYW5kIHJldHVybiBBdG9tcyBvYmplY3QuIE5PVElDRTogb25seSBhdG9tVHlwZXMsIHgsIHlcbiAqIGFuZCBrd2FyZ3MgZmllbGRzIGFyZSBzZXQgaW4gdGhlIHJldHVybiB2YWx1ZSwgd2l0aCBvdGhlciBmaWVsZHMgZHVtbXkgKi9cbmZ1bmN0aW9uIHBhcnNlQXRvbUJsb2NrKG1vbGZpbGVWM0s6IHN0cmluZywgYXRvbUNvdW50OiBudW1iZXIpOiBBdG9tcyB7XG4gIGNvbnN0IGF0b21UeXBlczogc3RyaW5nW10gPSBuZXcgQXJyYXkoYXRvbUNvdW50KTtcbiAgY29uc3QgeDogbnVtYmVyW10gPSBuZXcgQXJyYXkoYXRvbUNvdW50KTtcbiAgY29uc3QgeTogbnVtYmVyW10gPSBuZXcgQXJyYXkoYXRvbUNvdW50KTtcbiAgY29uc3Qga3dhcmdzOiBzdHJpbmdbXSA9IG5ldyBBcnJheShhdG9tQ291bnQpO1xuXG4gIGxldCBiZWdpbiA9IG1vbGZpbGVWM0suaW5kZXhPZihWM0tfQkVHSU5fQVRPTV9CTE9DSyk7IC8vIFYzMDAwIGF0b21zIGJsb2NrXG4gIGJlZ2luID0gbW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbik7XG4gIGxldCBlbmQgPSBiZWdpbjtcblxuICBmb3IgKGxldCBpID0gMDsgaSA8IGF0b21Db3VudDsgaSsrKSB7XG4gICAgYmVnaW4gPSBtb2xmaWxlVjNLLmluZGV4T2YoVjNLX0JFR0lOX0RBVEFfTElORSwgYmVnaW4pICsgVjNLX0lEWF9TSElGVDtcbiAgICBlbmQgPSBtb2xmaWxlVjNLLmluZGV4T2YoJyAnLCBiZWdpbik7IC8vIHNraXAgdGhlIGlkeCByb3dcblxuICAgIC8vIHBhcnNlIGF0b20gdHlwZVxuICAgIGJlZ2luID0gZW5kICsgMTtcbiAgICBlbmQgPSBtb2xmaWxlVjNLLmluZGV4T2YoJyAnLCBiZWdpbik7XG4gICAgYXRvbVR5cGVzW2ldID0gbW9sZmlsZVYzSy5zdWJzdHJpbmcoYmVnaW4sIGVuZCk7XG5cbiAgICAvLyBwYXJzZSBYIGFuZCBZIGNvb3JkaW5hdGVzIG9mIHRoZSBhdG9tXG4gICAgY29uc3QgY29vcmRpbmF0ZTogbnVtYmVyW10gPSBuZXcgQXJyYXkoMik7XG4gICAgZm9yIChsZXQgayA9IDA7IGsgPCAyOyArK2spIHtcbiAgICAgIGJlZ2luID0gZW5kICsgMTtcbiAgICAgIGVuZCA9IG1vbGZpbGVWM0suaW5kZXhPZignICcsIGJlZ2luKTtcbiAgICAgIGNvb3JkaW5hdGVba10gPSBwYXJzZUZsb2F0KG1vbGZpbGVWM0suc3Vic3RyaW5nKGJlZ2luLCBlbmQpKTtcbiAgICB9XG4gICAgeFtpXSA9IGNvb3JkaW5hdGVbMF07XG4gICAgeVtpXSA9IGNvb3JkaW5hdGVbMV07XG5cbiAgICAvLyBwYXJzZSB0aGUgcmVtYWluaW5nIHBvc3NpYmxlIGtleXdvcmQgYXJndW1lbnRzXG4gICAgYmVnaW4gPSBlbmQ7XG4gICAgZW5kID0gbW9sZmlsZVYzSy5pbmRleE9mKCdcXG4nLCBiZWdpbikgKyAxO1xuICAgIGt3YXJnc1tpXSA9IG1vbGZpbGVWM0suc2xpY2UoYmVnaW4sIGVuZCk7XG5cbiAgICBiZWdpbiA9IGVuZDtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgYXRvbVR5cGVzOiBhdG9tVHlwZXMsXG4gICAgeDogeCxcbiAgICB5OiB5LFxuICAgIGt3YXJnczoga3dhcmdzLFxuICB9O1xufVxuXG4vKiBSZW1vdmUgaHlkcm9nZW4gbm9kZXMgKi9cbmZ1bmN0aW9uIHJlbW92ZUh5ZHJvZ2VuKG1vbm9tZXJHcmFwaDogTW9sR3JhcGgpOiB2b2lkIHtcbiAgbGV0IGkgPSAwO1xuICB3aGlsZSAoaSA8IG1vbm9tZXJHcmFwaC5hdG9tcy5hdG9tVHlwZXMubGVuZ3RoKSB7XG4gICAgaWYgKCBtb25vbWVyR3JhcGguYXRvbXMuYXRvbVR5cGVzW2ldID09PSBIWURST0dFTikge1xuICAgICAgcmVtb3ZlTm9kZUFuZEJvbmRzKG1vbm9tZXJHcmFwaCwgaSArIDEpOyAvLyBpICsgMSBiZWNhdXNlIG1vbGZpbGUgbm9kZSBpbmRleGluZyBzdGFydHMgZnJvbSAxXG4gICAgICAtLWk7XG4gICAgICAvLyBtb25vbWVyR3JhcGguYXRvbXMuYXRvbVR5cGVzW2ldID0gJ0xpJztcbiAgICB9XG4gICAgKytpO1xuICB9XG59XG5cbi8qIFJlbW92ZSBub2RlICdyZW1vdmVkTm9kZScgYW5kIHRoZSBhc3NvY2lhdGVkIGJvbmRzLiBOb3RpY2UsIG51bWVyYXRpb24gb2ZcbiAqIG5vZGVzIGluIG1vbGZpbGVzIHN0YXJ0cyBmcm9tIDEsIG5vdCAwICovXG5mdW5jdGlvbiByZW1vdmVOb2RlQW5kQm9uZHMobW9ub21lckdyYXBoOiBNb2xHcmFwaCwgcmVtb3ZlZE5vZGU/OiBudW1iZXIpOiB2b2lkIHtcbiAgaWYgKHR5cGVvZiByZW1vdmVkTm9kZSAhPT0gJ3VuZGVmaW5lZCcpIHtcbiAgICBjb25zdCByZW1vdmVkTm9kZUlkeCA9IHJlbW92ZWROb2RlIC0gMTtcbiAgICBjb25zdCBhdG9tcyA9IG1vbm9tZXJHcmFwaC5hdG9tcztcbiAgICBjb25zdCBib25kcyA9IG1vbm9tZXJHcmFwaC5ib25kcztcbiAgICBjb25zdCBtZXRhID0gbW9ub21lckdyYXBoLm1ldGE7XG5cbiAgICAvLyByZW1vdmUgdGhlIG5vZGUgZnJvbSBhdG9tc1xuICAgIGF0b21zLmF0b21UeXBlcy5zcGxpY2UocmVtb3ZlZE5vZGVJZHgsIDEpO1xuICAgIGF0b21zLnguc3BsaWNlKHJlbW92ZWROb2RlSWR4LCAxKTtcbiAgICBhdG9tcy55LnNwbGljZShyZW1vdmVkTm9kZUlkeCwgMSk7XG4gICAgYXRvbXMua3dhcmdzLnNwbGljZShyZW1vdmVkTm9kZUlkeCwgMSk7XG5cbiAgICAvLyB1cGRhdGUgdGhlIHZhbHVlcyBvZiB0ZXJtaW5hbCBhbmQgci1ncm91cCBub2RlcyBpZiBuZWNlc3NhcnlcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IG1ldGEudGVybWluYWxOb2Rlcy5sZW5ndGg7ICsraSkge1xuICAgICAgaWYgKG1ldGEudGVybWluYWxOb2Rlc1tpXSA+IHJlbW92ZWROb2RlKVxuICAgICAgICAtLW1ldGEudGVybWluYWxOb2Rlc1tpXTtcbiAgICAgIGVsc2UgaWYgKG1ldGEudGVybWluYWxOb2Rlc1tpXSA9PT0gcmVtb3ZlZE5vZGUpXG4gICAgICAgIG1ldGEudGVybWluYWxOb2Rlc1tpXSA9IC0xOyAvLyBzZW50aW5lbCB0byBtYXJrIHRoZSB2YWx1ZSBhcyByZW1vdmVkXG4gICAgfVxuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgbWV0YS5yTm9kZXMubGVuZ3RoOyArK2kpIHtcbiAgICAgIGlmIChtZXRhLnJOb2Rlc1tpXSA+IHJlbW92ZWROb2RlKVxuICAgICAgICAtLW1ldGEuck5vZGVzW2ldO1xuICAgICAgZWxzZSBpZiAobWV0YS5yTm9kZXNbaV0gPT09IHJlbW92ZWROb2RlKVxuICAgICAgICBtZXRhLnJOb2Rlc1tpXSA9IC0xOyAvLyBzZW50aW5lbCB0byBtYXJrIHRoZSB2YWx1ZSBhcyByZW1vdmVkXG4gICAgfVxuXG4gICAgLy8gdXBkYXRlIGluZGljZXMgb2YgYXRvbXMgaW4gYm9uZHNcbiAgICBsZXQgaSA9IDA7XG4gICAgd2hpbGUgKGkgPCBib25kcy5hdG9tUGFpcnMubGVuZ3RoKSB7XG4gICAgICBjb25zdCBmaXJzdEF0b20gPSBib25kcy5hdG9tUGFpcnNbaV1bMF07XG4gICAgICBjb25zdCBzZWNvbmRBdG9tID0gYm9uZHMuYXRvbVBhaXJzW2ldWzFdO1xuICAgICAgaWYgKGZpcnN0QXRvbSA9PT0gcmVtb3ZlZE5vZGUgfHwgc2Vjb25kQXRvbSA9PT0gcmVtb3ZlZE5vZGUpIHtcbiAgICAgICAgYm9uZHMuYXRvbVBhaXJzLnNwbGljZShpLCAxKTtcbiAgICAgICAgYm9uZHMuYm9uZFR5cGVzLnNwbGljZShpLCAxKTtcbiAgICAgICAgaWYgKGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmhhcyhpKSlcbiAgICAgICAgICBib25kcy5ib25kQ29uZmlndXJhdGlvbi5kZWxldGUoaSk7XG4gICAgICAgIGlmIChib25kcy5rd2FyZ3MuaGFzKGkpKVxuICAgICAgICAgIGJvbmRzLmt3YXJncy5kZWxldGUoaSk7XG4gICAgICAgIC0taTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGJvbmRzLmF0b21QYWlyc1tpXVswXSA9IChmaXJzdEF0b20gPiByZW1vdmVkTm9kZSkgPyBmaXJzdEF0b20gLSAxIDogZmlyc3RBdG9tO1xuICAgICAgICBib25kcy5hdG9tUGFpcnNbaV1bMV0gPSAoc2Vjb25kQXRvbSA+IHJlbW92ZWROb2RlKSA/IHNlY29uZEF0b20gLSAxIDogc2Vjb25kQXRvbTtcbiAgICAgIH1cbiAgICAgICsraTtcbiAgICB9XG5cbiAgICAvLyB1cGRhdGUgYm9uZENvbmZpZ3VyYXRpb24gYW5kIGt3YXJncyBrZXlzXG4gICAgbGV0IGtleXMgPSBBcnJheS5mcm9tKGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmtleXMoKSk7XG4gICAga2V5cy5mb3JFYWNoKChrZXkpID0+IHtcbiAgICAgIGlmIChib25kcy5ib25kQ29uZmlndXJhdGlvbi5oYXMoa2V5KSAmJiBrZXkgPiByZW1vdmVkTm9kZUlkeCkge1xuICAgICAgICBjb25zdCB2YWx1ZSA9IGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmdldChrZXkpITtcbiAgICAgICAgYm9uZHMuYm9uZENvbmZpZ3VyYXRpb24uZGVsZXRlKGtleSk7XG4gICAgICAgIGJvbmRzLmJvbmRDb25maWd1cmF0aW9uLnNldChrZXkgLSAxLCB2YWx1ZSk7XG4gICAgICB9XG4gICAgfSk7XG4gICAga2V5cyA9IEFycmF5LmZyb20oYm9uZHMua3dhcmdzLmtleXMoKSk7XG4gICAga2V5cy5mb3JFYWNoKChrZXkpID0+IHtcbiAgICAgIGlmIChib25kcy5rd2FyZ3MuaGFzKGtleSkgJiYga2V5ID4gcmVtb3ZlZE5vZGVJZHgpIHtcbiAgICAgICAgY29uc3QgdmFsdWUgPSBib25kcy5rd2FyZ3MuZ2V0KGtleSkhO1xuICAgICAgICBib25kcy5rd2FyZ3MuZGVsZXRlKGtleSk7XG4gICAgICAgIGJvbmRzLmt3YXJncy5zZXQoa2V5IC0gMSwgdmFsdWUpO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG59XG5cbi8vIHRvZG86IHJld3JpdGUgZGVzY3JpcHRpb25cbi8qIEFkanVzdCB0aGUgKHBlcHRpZGUpIG1vbm9tZXIgZ3JhcGggc28gdGhhdCBpdCBoYXMgc3RhbmRhcmQgZm9ybSAgKi9cbmZ1bmN0aW9uIGFkanVzdFBlcHRpZGVNb25vbWVyR3JhcGgobW9ub21lcjogTW9sR3JhcGgpOiB2b2lkIHtcbiAgY29uc3Qgbm9kZU9uZUlkeCA9IG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzBdIC0gMTsgLy8gbm9kZSBpbmRleGluZyBpbiBtb2xmaWxlcyBzdGFydHMgZnJvbSAxXG4gIGNvbnN0IG5vZGVUd29JZHggPSBtb25vbWVyLm1ldGEuck5vZGVzWzBdIC0gMTtcbiAgY29uc3QgeCA9IG1vbm9tZXIuYXRvbXMueDtcbiAgY29uc3QgeSA9IG1vbm9tZXIuYXRvbXMueTtcblxuICAvLyBwbGFjZSBub2RlT25lIGF0IG9yaWdpblxuICBzaGlmdENvb3JkaW5hdGVzKG1vbm9tZXIsIC14W25vZGVPbmVJZHhdLCAteVtub2RlT25lSWR4XSk7XG5cbiAgLy8gYW5nbGUgaXMgbWVhc3VyZWQgYmV0d2VlbiBPWSBhbmQgdGhlIHJvdGF0ZWQgbm9kZVxuICBjb25zdCBhbmdsZSA9IGZpbmRBbmdsZVdpdGhPWSh4W25vZGVUd29JZHhdLCB5W25vZGVUd29JZHhdKTtcblxuICAvLyByb3RhdGUgdGhlIGNlbnRlcmVkIGdyYXBoLCBzbyB0aGF0ICdub2RlVHdvJyBlbmRzIHVwIG9uIHRoZSBwb3NpdGl2ZSByYXkgb2YgT1lcbiAgcm90YXRlQ2VudGVyZWRHcmFwaChtb25vbWVyLmF0b21zLCAtYW5nbGUpO1xuXG4gIGlmICh4W21vbm9tZXIubWV0YS5yTm9kZXNbMV0gLSAxXSA8IDApXG4gICAgZmxpcE1vbm9tZXJBcm91bmRPWShtb25vbWVyKTtcblxuICBjb25zdCBkb3VibGVCb25kZWRPeHlnZW4gPSBmaW5kRG91YmxlQm9uZGVkQ2FyYm9ueWxPeHlnZW4obW9ub21lcik7XG5cbiAgLy8gZmxpcCBjYXJib3h5bCBhbmQgUiBpZiBuZWNlc3NhcnlcbiAgZmxpcENhcmJveHlsQW5kUmFkaWNhbChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xuXG4gIC8vIGZsaXAgaHlkcm94eWwgZ3JvdXAgd2l0aCBkb3VibGUtYm91bmQgTyBpbnNpZGUgY2FyYm94eWwgZ3JvdXAgaWYgbmVjZXNzYXJ5XG4gIGZsaXBIeWRyb3hpbEdyb3VwKG1vbm9tZXIsIGRvdWJsZUJvbmRlZE94eWdlbik7XG59XG5cbmZ1bmN0aW9uIGFkanVzdFBob3NwaGF0ZU1vbm9tZXJHcmFwaChtb25vbWVyOiBNb2xHcmFwaCk6dm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBtb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDE7IC8vIG5vZGUgaW5kZXhpbmcgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMVxuICBjb25zdCBub2RlVHdvSWR4ID0gbW9ub21lci5tZXRhLnJOb2Rlc1swXSAtIDE7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIGNvbnN0IHkgPSBtb25vbWVyLmF0b21zLnk7XG5cbiAgLy8gcGxhY2Ugbm9kZU9uZSBhdCBvcmlnaW5cbiAgc2hpZnRDb29yZGluYXRlcyhtb25vbWVyLCAteFtub2RlT25lSWR4XSwgLXlbbm9kZU9uZUlkeF0pO1xuXG4gIC8vIC8vIGFuZ2xlIGlzIG1lYXN1cmVkIGJldHdlZW4gT1kgYW5kIHRoZSByb3RhdGVkIG5vZGVcbiAgLy8gY29uc3QgYW5nbGUgPSBmaW5kQW5nbGVXaXRoT1koeFtub2RlVHdvSWR4XSwgeVtub2RlVHdvSWR4XSk7XG5cbiAgLy8gLy8gcm90YXRlIHRoZSBjZW50ZXJlZCBncmFwaCwgc28gdGhhdCAnbm9kZVR3bycgZW5kcyB1cCBvbiB0aGUgcG9zaXRpdmUgcmF5IG9mIE9ZXG4gIC8vIHJvdGF0ZUNlbnRlcmVkR3JhcGgobW9ub21lci5hdG9tcywgLWFuZ2xlKTtcblxuICAvLyBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwKVxuICAvLyAgIGZsaXBNb25vbWVyQXJvdW5kT1kobW9ub21lcik7XG5cbiAgLy8gY29uc3QgZG91YmxlQm9uZGVkT3h5Z2VuID0gZmluZERvdWJsZUJvbmRlZENhcmJvbnlsT3h5Z2VuKG1vbm9tZXIpO1xuXG4gIC8vIC8vIGZsaXAgY2FyYm94eWwgYW5kIFIgaWYgbmVjZXNzYXJ5XG4gIC8vIGZsaXBDYXJib3h5bEFuZFJhZGljYWwobW9ub21lciwgZG91YmxlQm9uZGVkT3h5Z2VuKTtcblxuICAvLyAvLyBmbGlwIGh5ZHJveHlsIGdyb3VwIHdpdGggZG91YmxlLWJvdW5kIE8gaW5zaWRlIGNhcmJveHlsIGdyb3VwIGlmIG5lY2Vzc2FyeVxuICAvLyBmbGlwSHlkcm94aWxHcm91cChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xufVxuXG5mdW5jdGlvbiBhZGp1c3RTdWdhck1vbm9tZXJHcmFwaChtb25vbWVyOiBNb2xHcmFwaCk6dm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBtb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDE7IC8vIG5vZGUgaW5kZXhpbmcgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMVxuICBjb25zdCBub2RlVHdvSWR4ID0gbW9ub21lci5tZXRhLnJOb2Rlc1swXSAtIDE7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIGNvbnN0IHkgPSBtb25vbWVyLmF0b21zLnk7XG5cbiAgLy8gcGxhY2Ugbm9kZU9uZSBhdCBvcmlnaW5cbiAgc2hpZnRDb29yZGluYXRlcyhtb25vbWVyLCAteFtub2RlT25lSWR4XSwgLXlbbm9kZU9uZUlkeF0pO1xuXG4gIC8vIC8vIGFuZ2xlIGlzIG1lYXN1cmVkIGJldHdlZW4gT1kgYW5kIHRoZSByb3RhdGVkIG5vZGVcbiAgLy8gY29uc3QgYW5nbGUgPSBmaW5kQW5nbGVXaXRoT1koeFtub2RlVHdvSWR4XSwgeVtub2RlVHdvSWR4XSk7XG5cbiAgLy8gLy8gcm90YXRlIHRoZSBjZW50ZXJlZCBncmFwaCwgc28gdGhhdCAnbm9kZVR3bycgZW5kcyB1cCBvbiB0aGUgcG9zaXRpdmUgcmF5IG9mIE9ZXG4gIC8vIHJvdGF0ZUNlbnRlcmVkR3JhcGgobW9ub21lci5hdG9tcywgLWFuZ2xlKTtcblxuICAvLyBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwKVxuICAvLyAgIGZsaXBNb25vbWVyQXJvdW5kT1kobW9ub21lcik7XG5cbiAgLy8gY29uc3QgZG91YmxlQm9uZGVkT3h5Z2VuID0gZmluZERvdWJsZUJvbmRlZENhcmJvbnlsT3h5Z2VuKG1vbm9tZXIpO1xuXG4gIC8vIC8vIGZsaXAgY2FyYm94eWwgYW5kIFIgaWYgbmVjZXNzYXJ5XG4gIC8vIGZsaXBDYXJib3h5bEFuZFJhZGljYWwobW9ub21lciwgZG91YmxlQm9uZGVkT3h5Z2VuKTtcblxuICAvLyAvLyBmbGlwIGh5ZHJveHlsIGdyb3VwIHdpdGggZG91YmxlLWJvdW5kIE8gaW5zaWRlIGNhcmJveHlsIGdyb3VwIGlmIG5lY2Vzc2FyeVxuICAvLyBmbGlwSHlkcm94aWxHcm91cChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xufVxuZnVuY3Rpb24gYWRqdXN0QmFzZU1vbm9tZXJHcmFwaChtb25vbWVyOiBNb2xHcmFwaCk6dm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBtb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSAtIDE7IC8vIG5vZGUgaW5kZXhpbmcgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMVxuICBjb25zdCBub2RlVHdvSWR4ID0gbW9ub21lci5tZXRhLnJOb2Rlc1swXSAtIDE7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIGNvbnN0IHkgPSBtb25vbWVyLmF0b21zLnk7XG5cbiAgLy8gcGxhY2Ugbm9kZU9uZSBhdCBvcmlnaW5cbiAgc2hpZnRDb29yZGluYXRlcyhtb25vbWVyLCAteFtub2RlT25lSWR4XSwgLXlbbm9kZU9uZUlkeF0pO1xuXG4gIC8vIC8vIGFuZ2xlIGlzIG1lYXN1cmVkIGJldHdlZW4gT1kgYW5kIHRoZSByb3RhdGVkIG5vZGVcbiAgLy8gY29uc3QgYW5nbGUgPSBmaW5kQW5nbGVXaXRoT1koeFtub2RlVHdvSWR4XSwgeVtub2RlVHdvSWR4XSk7XG5cbiAgLy8gLy8gcm90YXRlIHRoZSBjZW50ZXJlZCBncmFwaCwgc28gdGhhdCAnbm9kZVR3bycgZW5kcyB1cCBvbiB0aGUgcG9zaXRpdmUgcmF5IG9mIE9ZXG4gIC8vIHJvdGF0ZUNlbnRlcmVkR3JhcGgobW9ub21lci5hdG9tcywgLWFuZ2xlKTtcblxuICAvLyBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwKVxuICAvLyAgIGZsaXBNb25vbWVyQXJvdW5kT1kobW9ub21lcik7XG5cbiAgLy8gY29uc3QgZG91YmxlQm9uZGVkT3h5Z2VuID0gZmluZERvdWJsZUJvbmRlZENhcmJvbnlsT3h5Z2VuKG1vbm9tZXIpO1xuXG4gIC8vIC8vIGZsaXAgY2FyYm94eWwgYW5kIFIgaWYgbmVjZXNzYXJ5XG4gIC8vIGZsaXBDYXJib3h5bEFuZFJhZGljYWwobW9ub21lciwgZG91YmxlQm9uZGVkT3h5Z2VuKTtcblxuICAvLyAvLyBmbGlwIGh5ZHJveHlsIGdyb3VwIHdpdGggZG91YmxlLWJvdW5kIE8gaW5zaWRlIGNhcmJveHlsIGdyb3VwIGlmIG5lY2Vzc2FyeVxuICAvLyBmbGlwSHlkcm94aWxHcm91cChtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4pO1xufVxuXG4vKiBGbGlwIGNhcmJveHlsIGdyb3VwIHdpdGggdGhlIHJhZGljYWwgaW4gYSBwZXB0aWRlIG1vbm9tZXIgaW4gY2FzZSB0aGVcbiAqIGNhcmJveHlsIGdyb3VwIGlzIGluIHRoZSBsb3dlciBoYWxmLXBsYW5lICovXG5mdW5jdGlvbiBmbGlwQ2FyYm94eWxBbmRSYWRpY2FsKG1vbm9tZXI6IE1vbEdyYXBoLCBkb3VibGVCb25kZWRPeHlnZW46IG51bWJlcik6IHZvaWQge1xuICAvLyB2ZXJpZnkgdGhhdCB0aGUgY2FyYm94eWwgZ3JvdXAgaXMgaW4gdGhlIGxvd2VyIGhhbGYtcGxhbmVcbiAgaWYgKG1vbm9tZXIuYXRvbXMueVttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPCAwICYmXG4gICAgbW9ub21lci5hdG9tcy55W2RvdWJsZUJvbmRlZE94eWdlbiAtIDFdIDwgMCkge1xuICAgIGZsaXBNb25vbWVyQXJvdW5kT1gobW9ub21lcik7XG5cbiAgICByb3RhdGVDZW50ZXJlZEdyYXBoKG1vbm9tZXIuYXRvbXMsXG4gICAgICAtZmluZEFuZ2xlV2l0aE9YKFxuICAgICAgICBtb25vbWVyLmF0b21zLnhbbW9ub21lci5tZXRhLnRlcm1pbmFsTm9kZXNbMV0gLSAxXSxcbiAgICAgICAgbW9ub21lci5hdG9tcy55W21vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzFdIC0gMV1cbiAgICAgIClcbiAgICApO1xuICB9XG59XG5cbi8qIEZpbmRzIGFuZ2xlIGJldHdlZW4gT1kgYW5kIHRoZSByYXkgam9pbmluZyBvcmlnaW4gd2l0aCAoeCwgeSkgKi9cbmZ1bmN0aW9uIGZpbmRBbmdsZVdpdGhPWSh4OiBudW1iZXIsIHk6IG51bWJlcik6IG51bWJlciB7XG4gIGxldCBhbmdsZTtcbiAgaWYgKHggPT09IDApIHtcbiAgICBhbmdsZSA9IHkgPiAwID8gMCA6IE1hdGguUEk7XG4gIH0gZWxzZSBpZiAoeSA9PT0gMCkge1xuICAgIGFuZ2xlID0geCA+IDAgPyAtTWF0aC5QSS8yIDogTWF0aC5QSS8yO1xuICB9IGVsc2Uge1xuICAgIGNvbnN0IHRhbiA9IHkgLyB4O1xuICAgIGNvbnN0IGF0YW4gPSBNYXRoLmF0YW4odGFuKTtcbiAgICBhbmdsZSA9ICh4IDwgMCkgPyBNYXRoLlBJLzIgKyBhdGFuIDogLU1hdGguUEkvMiArIGF0YW47XG4gIH1cbiAgcmV0dXJuIGFuZ2xlO1xufVxuXG4vKiBGaW5kcyBhbmdsZSBiZXR3ZWVuIE9YIGFuZCB0aGUgcmF5IGpvaW5pbmcgb3JpZ2luIHdpdGggKHgsIHkpICovXG5mdW5jdGlvbiBmaW5kQW5nbGVXaXRoT1goeDogbnVtYmVyLCB5OiBudW1iZXIpOiBudW1iZXIge1xuICByZXR1cm4gZmluZEFuZ2xlV2l0aE9ZKHgsIHkpICsgTWF0aC5QSS8yO1xufVxuXG4vKiAgUm90YXRlIHRoZSBncmFwaCBhcm91bmQgdGhlIG9yaWdpbiBieSAnYW5nbGUnICovXG5mdW5jdGlvbiByb3RhdGVDZW50ZXJlZEdyYXBoKGF0b21zOiBBdG9tcywgYW5nbGU6IG51bWJlcik6IHZvaWQge1xuICBpZiAoYW5nbGUgIT09IDApIHtcbiAgICBjb25zdCB4ID0gYXRvbXMueDtcbiAgICBjb25zdCB5ID0gYXRvbXMueTtcblxuICAgIGNvbnN0IGNvcyA9IE1hdGguY29zKGFuZ2xlKTtcbiAgICBjb25zdCBzaW4gPSBNYXRoLnNpbihhbmdsZSk7XG5cbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IHgubGVuZ3RoOyArK2kpIHtcbiAgICAgIGNvbnN0IHRtcCA9IHhbaV07XG4gICAgICB4W2ldID0ga2VlcFByZWNpc2lvbih0bXAqY29zIC0geVtpXSpzaW4pO1xuICAgICAgeVtpXSA9IGtlZXBQcmVjaXNpb24odG1wKnNpbiArIHlbaV0qY29zKTtcbiAgICB9XG4gIH1cbn1cblxuLyogRmxpcCBtb25vbWVyIGdyYXBoIGFyb3VuZCBPWCBheGlzIHByZXNlcnZpbmcgc3RlcmVvbWV0cnkgKi9cbmZ1bmN0aW9uIGZsaXBNb25vbWVyQXJvdW5kT1gobW9ub21lcjogTW9sR3JhcGgpOiB2b2lkIHtcbiAgZmxpcE1vbEdyYXBoKG1vbm9tZXIsIHRydWUpO1xufVxuXG4vKiBGbGlwIG1vbm9tZXIgZ3JhcGggYXJvdW5kIE9ZIGF4aXMgcHJlc2VydmluZyBzdGVyZW9tZXRyeSAqL1xuZnVuY3Rpb24gZmxpcE1vbm9tZXJBcm91bmRPWShtb25vbWVyOiBNb2xHcmFwaCk6IHZvaWQge1xuICBmbGlwTW9sR3JhcGgobW9ub21lciwgZmFsc2UpO1xufVxuXG4vKiBGbGlwIGdyYXBoIGFyb3VuZCBhIHNwZWNpZmllZCBheGlzOiAndHJ1ZScgY29ycmVzcG9uZHMgdG8gT1gsICdmYWxzZScgdG8gT1kgKi9cbmZ1bmN0aW9uIGZsaXBNb2xHcmFwaChtb2xHcmFwaDogTW9sR3JhcGgsIGF4aXM6IGJvb2xlYW4pOiB2b2lkIHtcbiAgaWYgKGF4aXMpIHsgLy8gZmxpcHBpbmcgYXJvdW5kIE9YXG4gICAgY29uc3QgeSA9IG1vbEdyYXBoLmF0b21zLnk7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCB5Lmxlbmd0aDsgaSsrKVxuICAgICAgeVtpXSA9IC15W2ldO1xuICB9IGVsc2UgeyAvLyBmbGlwcGluZyBhcm91bmQgT1lcbiAgICBjb25zdCB4ID0gbW9sR3JhcGguYXRvbXMueDtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IHgubGVuZ3RoOyBpKyspXG4gICAgICB4W2ldID0gLXhbaV07XG4gIH1cblxuICAvLyBwcmVzZXJ2ZSB0aGUgc3RlcmVvbWV0cnlcbiAgY29uc3Qgb3JpZW50YXRpb24gPSBtb2xHcmFwaC5ib25kcy5ib25kQ29uZmlndXJhdGlvbjtcbiAgZm9yIChjb25zdCBba2V5LCB2YWx1ZV0gb2Ygb3JpZW50YXRpb24pIHtcbiAgICBjb25zdCBuZXdWYWx1ZSA9IHZhbHVlID09PSAxID8gMyA6IDE7XG4gICAgb3JpZW50YXRpb24uc2V0KGtleSwgbmV3VmFsdWUpO1xuICB9XG59XG5cbi8qIEZsaXBzIGRvdWJsZS1ib25kZWQgJ08nIGluIGNhcmJvbnlsIGdyb3VwIHdpdGggJ09IJyBpbiBvcmRlciBmb3IgdGhlIG1vbm9tZXJzXG4gKiB0byBoYXZlIHN0YW5kYXJkIHJlcHJlc2VudGF0aW9uIHNpbXBsaWZ5aW5nIHRoZWlyIGNvbmNhdGVuYXRpb24uIFRoZVxuICogbW9ub21lciBtdXN0IGFscmVhZHkgYmUgYWRqdXN0ZWQgd2l0aCBhZGp1c3RQZXB0aWRlTW9ub21lckdyYXBoIGluIG9yZGVyIGZvciB0aGlzIGZ1bmN0aW9uIHRvIGJlIGltcGxlbWVudGVkICAqL1xuZnVuY3Rpb24gZmxpcEh5ZHJveGlsR3JvdXAobW9ub21lcjogTW9sR3JhcGgsIGRvdWJsZUJvbmRlZE94eWdlbjogbnVtYmVyKTogdm9pZCB7XG4gIGNvbnN0IHggPSBtb25vbWVyLmF0b21zLng7XG4gIC8vIC0xIGJlbG93IGJlY2F1c2UgaW5kZXhpbmcgb2Ygbm9kZXMgaW4gbW9sZmlsZXMgc3RhcnRzIGZyb20gMSwgdW5saWtlIGFycmF5c1xuICBpZiAoeFttb25vbWVyLm1ldGEuck5vZGVzWzFdIC0gMV0gPiB4W2RvdWJsZUJvbmRlZE94eWdlbiAtIDFdKVxuICAgIHN3YXBOb2Rlcyhtb25vbWVyLCBkb3VibGVCb25kZWRPeHlnZW4sIG1vbm9tZXIubWV0YS5yTm9kZXNbMV0pO1xufVxuXG4vKiBEZXRlcm1pbmUgdGhlIG51bWJlciBvZiBub2RlIChzdGFydGluZyBmcm9tIDEpIGNvcnJlc3BvbmRpbmcgdG8gdGhlXG4gKiBkb3VibGUtYm9uZGVkIG94eWdlbiBvZiB0aGUgY2FyYm9ueWwgZ3JvdXAgICovXG5mdW5jdGlvbiBmaW5kRG91YmxlQm9uZGVkQ2FyYm9ueWxPeHlnZW4obW9ub21lcjogTW9sR3JhcGgpOiBudW1iZXIge1xuICBjb25zdCBib25kc01hcCA9IGNvbnN0cnVjdEJvbmRzTWFwKG1vbm9tZXIpO1xuICBsZXQgZG91YmxlQm9uZGVkT3h5Z2VuID0gMDtcbiAgbGV0IGkgPSAwO1xuICAvLyBpdGVyYXRlIG92ZXIgdGhlIG5vZGVzIGJvbmRlZCB0byB0aGUgY2FyYm9uIGFuZCBmaW5kIHRoZSBkb3VibGUgb25lXG4gIHdoaWxlIChkb3VibGVCb25kZWRPeHlnZW4gPT09IDApIHtcbiAgICBjb25zdCBub2RlID0gYm9uZHNNYXAuZ2V0KG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzFdKSFbaV07XG4gICAgaWYgKG1vbm9tZXIuYXRvbXMuYXRvbVR5cGVzW25vZGUgLSAxXSA9PT0gT1hZR0VOICYmIG5vZGUgIT09IG1vbm9tZXIubWV0YS5yTm9kZXNbMV0pXG4gICAgICBkb3VibGVCb25kZWRPeHlnZW4gPSBub2RlO1xuICAgIGkrKztcbiAgfVxuICByZXR1cm4gZG91YmxlQm9uZGVkT3h5Z2VuO1xufVxuXG4vKiBTd2FwIHRoZSBDYXJ0ZXNpYW4gY29vcmRpbmF0ZXMgb2YgdGhlIHR3byBzcGVjaWZpZWQgbm9kZXMgaW4gTW9sR3JhcGggICovXG5mdW5jdGlvbiBzd2FwTm9kZXMobW9ub21lcjogTW9sR3JhcGgsIG5vZGVPbmU6IG51bWJlciwgbm9kZVR3bzogbnVtYmVyKTogdm9pZCB7XG4gIGNvbnN0IG5vZGVPbmVJZHggPSBub2RlT25lIC0gMTtcbiAgY29uc3Qgbm9kZVR3b0lkeCA9IG5vZGVUd28gLSAxO1xuICBjb25zdCB4ID0gbW9ub21lci5hdG9tcy54O1xuICBjb25zdCB5ID0gbW9ub21lci5hdG9tcy55O1xuICBjb25zdCB0bXBYID0geFtub2RlT25lSWR4XTtcbiAgY29uc3QgdG1wWSA9IHlbbm9kZU9uZUlkeF07XG4gIHhbbm9kZU9uZUlkeF0gPSB4W25vZGVUd29JZHhdO1xuICB5W25vZGVPbmVJZHhdID0geVtub2RlVHdvSWR4XTtcbiAgeFtub2RlVHdvSWR4XSA9IHRtcFg7XG4gIHlbbm9kZVR3b0lkeF0gPSB0bXBZO1xufVxuXG4vLyB0b2RvOiBkb2NcbmZ1bmN0aW9uIGNvbnN0cnVjdEJvbmRzTWFwKG1vbm9tZXI6IE1vbEdyYXBoKTogTWFwPG51bWJlciwgQXJyYXk8bnVtYmVyPj4ge1xuICBjb25zdCBtYXAgPSBuZXcgTWFwPG51bWJlciwgQXJyYXk8bnVtYmVyPj4oKTtcbiAgZm9yIChjb25zdCBhdG9tUGFpcnMgb2YgbW9ub21lci5ib25kcy5hdG9tUGFpcnMpIHtcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IDI7IGkrKykge1xuICAgICAgY29uc3Qga2V5ID0gYXRvbVBhaXJzW2ldO1xuICAgICAgY29uc3QgdmFsdWUgPSBhdG9tUGFpcnNbKGkgKyAxKSUyXTtcbiAgICAgIGlmIChtYXAuaGFzKGtleSkpXG4gICAgICAgIG1hcC5nZXQoa2V5KT8ucHVzaCh2YWx1ZSk7XG4gICAgICBlbHNlXG4gICAgICAgIG1hcC5zZXQoa2V5LCBuZXcgQXJyYXk8bnVtYmVyPigxKS5maWxsKHZhbHVlKSk7XG4gICAgfVxuICB9XG4gIHJldHVybiBtYXA7XG59XG5cbi8qIFNoaWZ0IG1vbEdyYXBoIGluIHRoZSBYT1kgcGxhbmUgICovXG5mdW5jdGlvbiBzaGlmdENvb3JkaW5hdGVzKG1vbEdyYXBoOiBNb2xHcmFwaCwgeFNoaWZ0OiBudW1iZXIsIHlTaGlmdD86IG51bWJlcik6IHZvaWQge1xuICBjb25zdCB4ID0gbW9sR3JhcGguYXRvbXMueDtcbiAgY29uc3QgeSA9IG1vbEdyYXBoLmF0b21zLnk7XG4gIGZvciAobGV0IGkgPSAwOyBpIDwgeC5sZW5ndGg7ICsraSkge1xuICAgIHhbaV0gPSBrZWVwUHJlY2lzaW9uKHhbaV0gKyB4U2hpZnQpO1xuICAgIGlmICh0eXBlb2YgeVNoaWZ0ICE9PSAndW5kZWZpbmVkJylcbiAgICAgIHlbaV0gPSBrZWVwUHJlY2lzaW9uKHlbaV0gKyB5U2hpZnQpO1xuICB9XG59XG5cbi8qIFRyYW5zbGF0ZSBhIHNlcXVlbmNlIG9mIG1vbm9tZXIgc3ltYm9scyBpbnRvIE1vbGZpbGUgVjMwMDAgKi9cbmZ1bmN0aW9uIG1vbm9tZXJTZXFUb01vbGZpbGUoXG4gIG1vbm9tZXJTZXE6IHN0cmluZ1tdLCBtb25vbWVyc0RpY3Q6IE1hcDxzdHJpbmcsIE1vbEdyYXBoPiwgYWxwaGFiZXQ6IEFMUEhBQkVULCBwb2x5bWVyVHlwZTogSEVMTV9QT0xZTUVSX1RZUEVcbik6IHN0cmluZyB7XG4gIC8vIHRvZG86IGhhbmRsZSB0aGUgY2FzZSB3aGVuIHRoZSBwb2x5bWVyIGlzIGVtcHR5XG4gIGlmIChtb25vbWVyU2VxLmxlbmd0aCA9PT0gMClcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ21vbm9tZXJTZXEgaXMgZW1wdHknKTtcblxuICAvLyBkZWZpbmUgYXRvbSBhbmQgYm9uZCBjb3VudHMsIHRha2luZyBpbnRvIGFjY291bnQgdGhlIGJvbmQgdHlwZVxuICBjb25zdCB7YXRvbUNvdW50LCBib25kQ291bnR9ID0gZ2V0UmVzdWx0aW5nQXRvbUJvbmRDb3VudHMobW9ub21lclNlcSwgbW9ub21lcnNEaWN0LCBhbHBoYWJldCwgcG9seW1lclR5cGUpO1xuXG4gIC8vIGNyZWF0ZSBhcnJheXMgdG8gc3RvcmUgbGluZXMgb2YgdGhlIHJlc3VsdGluZyBtb2xmaWxlXG4gIGNvbnN0IG1vbGZpbGVBdG9tQmxvY2sgPSBuZXcgQXJyYXk8c3RyaW5nPihhdG9tQ291bnQpO1xuICBjb25zdCBtb2xmaWxlQm9uZEJsb2NrID0gbmV3IEFycmF5PHN0cmluZz4oYm9uZENvdW50KTtcblxuICBsZXQgYWRkTW9ub21lclRvTW9sYmxvY2s7XG4gIGxldCBjYXBNb2xibG9jaztcbiAgbGV0IG5vZGVTaGlmdEluaXRWYWx1ZTtcbiAgbGV0IGJvbmRTaGlmdEluaXRWYWx1ZTtcbiAgbGV0IHN1Z2FyID0gbnVsbDtcbiAgbGV0IHBob3NwaGF0ZSA9IG51bGw7XG5cbiAgaWYgKHBvbHltZXJUeXBlID09PSBIRUxNX1BPTFlNRVJfVFlQRS5QRVBUSURFKSB7XG4gICAgYWRkTW9ub21lclRvTW9sYmxvY2sgPSBhZGRBbWlub0FjaWRUb01vbGJsb2NrO1xuICAgIGNhcE1vbGJsb2NrID0gY2FwUGVwdGlkZU1vbGJsb2NrO1xuICAgIG5vZGVTaGlmdEluaXRWYWx1ZSA9IGJvbmRTaGlmdEluaXRWYWx1ZSA9IDA7XG4gIH0gZWxzZSB7IC8vIG51Y2xlb3RpZGVzXG4gICAgYWRkTW9ub21lclRvTW9sYmxvY2sgPSBhZGROdWNsZW90aWRlVG9Nb2xibG9jaztcbiAgICBjYXBNb2xibG9jayA9IGNhcFBlcHRpZGVNb2xibG9jazsgLy8gdG9kbzogY2xlYW51cCAmIHJlZmFjdG9yXG4gICAgbm9kZVNoaWZ0SW5pdFZhbHVlID0gMDtcbiAgICBib25kU2hpZnRJbml0VmFsdWUgPSAwO1xuICAgIHN1Z2FyID0gKGFscGhhYmV0ID09PSBBTFBIQUJFVC5ETkEpID8gbW9ub21lcnNEaWN0LmdldChERU9YWVJJQk9TRSkgOiBtb25vbWVyc0RpY3QuZ2V0KFJJQk9TRSk7XG4gICAgcGhvc3BoYXRlID0gbW9ub21lcnNEaWN0LmdldChQSE9TUEhBVEUpO1xuICB9XG5cbiAgY29uc3QgdjogTG9vcFZhcmlhYmxlcyA9IHtcbiAgICBpOiAwLFxuICAgIG5vZGVTaGlmdDogbm9kZVNoaWZ0SW5pdFZhbHVlLFxuICAgIGJvbmRTaGlmdDogYm9uZFNoaWZ0SW5pdFZhbHVlLFxuICAgIGJhY2tib25lUG9zaXRpb25TaGlmdDogbmV3IEFycmF5PG51bWJlcj4oMikuZmlsbCgwKSxcbiAgICBicmFuY2hQb3NpdGlvblNoaWZ0OiBuZXcgQXJyYXk8bnVtYmVyPigyKS5maWxsKDApLFxuICAgIGJhY2tib25lQXR0YWNoTm9kZTogMCxcbiAgICBicmFuY2hBdHRhY2hOb2RlOiAwLFxuICAgIGZsaXBGYWN0b3I6IDEsXG4gIH07XG5cbiAgY29uc3QgQzogTG9vcENvbnN0YW50cyA9IHtcbiAgICBzdWdhcjogc3VnYXIhLFxuICAgIHBob3NwaGF0ZTogcGhvc3BoYXRlISxcbiAgICBzZXFMZW5ndGg6IG1vbm9tZXJTZXEubGVuZ3RoLFxuICAgIGF0b21Db3VudDogYXRvbUNvdW50LFxuICAgIGJvbmRDb3VudDogYm9uZENvdW50LFxuICB9O1xuXG4gIGZvciAodi5pID0gMDsgdi5pIDwgQy5zZXFMZW5ndGg7ICsrdi5pKSB7XG4gICAgY29uc3QgbW9ub21lciA9IG1vbm9tZXJzRGljdC5nZXQobW9ub21lclNlcVt2LmldKSE7XG4gICAgYWRkTW9ub21lclRvTW9sYmxvY2sobW9ub21lciwgbW9sZmlsZUF0b21CbG9jaywgbW9sZmlsZUJvbmRCbG9jaywgdiwgQyk7XG4gIH1cblxuICBjYXBNb2xibG9jayhtb2xmaWxlQXRvbUJsb2NrLCBtb2xmaWxlQm9uZEJsb2NrLCB2LCBDKTtcblxuICBjb25zdCBtb2xmaWxlQ291bnRzTGluZSA9IFYzS19CRUdJTl9DT1VOVFNfTElORSArIGF0b21Db3VudCArICcgJyArIGJvbmRDb3VudCArIFYzS19DT1VOVFNfTElORV9FTkRJTkc7XG5cbiAgLy8gdG9kbzogb3B0aW1pemUgY29uY2F0ZW5hdGlvbiB1c2luZyBBbGV4YW5kZXIncyBoaW50XG4gIGNvbnN0IG1vbGZpbGVQYXJ0cyA9IFtcbiAgICBWM0tfSEVBREVSX0ZJUlNUX0xJTkUsXG4gICAgVjNLX0hFQURFUl9TRUNPTkRfTElORSxcbiAgICBWM0tfQkVHSU5fQ1RBQl9CTE9DSyxcbiAgICBtb2xmaWxlQ291bnRzTGluZSxcbiAgICBWM0tfQkVHSU5fQVRPTV9CTE9DSyxcbiAgICBtb2xmaWxlQXRvbUJsb2NrLmpvaW4oJycpLFxuICAgIFYzS19FTkRfQVRPTV9CTE9DSyxcbiAgICBWM0tfQkVHSU5fQk9ORF9CTE9DSyxcbiAgICBtb2xmaWxlQm9uZEJsb2NrLmpvaW4oJycpLFxuICAgIFYzS19FTkRfQk9ORF9CTE9DSyxcbiAgICBWM0tfRU5EX0NUQUJfQkxPQ0ssXG4gICAgVjNLX0VORCxcbiAgXTtcblxuICByZXR1cm4gbW9sZmlsZVBhcnRzLmpvaW4oJycpO1xufVxuXG4vLyB0b2RvOiBkb2NcbmZ1bmN0aW9uIGNhcFBlcHRpZGVNb2xibG9jayhcbiAgbW9sZmlsZUF0b21CbG9jazogc3RyaW5nW10sIG1vbGZpbGVCb25kQmxvY2s6IHN0cmluZ1tdLFxuICB2OiBMb29wVmFyaWFibGVzLCBDOiBMb29wQ29uc3RhbnRzXG4pOiB2b2lkIHtcbiAgLy8gYWRkIHRlcm1pbmFsIG94eWdlblxuICBjb25zdCBhdG9tSWR4ID0gdi5ub2RlU2hpZnQgKyAxO1xuICBtb2xmaWxlQXRvbUJsb2NrW0MuYXRvbUNvdW50XSA9IFYzS19CRUdJTl9EQVRBX0xJTkUgKyBhdG9tSWR4ICsgJyAnICtcbiAgICBPWFlHRU4gKyAnICcgKyBrZWVwUHJlY2lzaW9uKHYuYmFja2JvbmVQb3NpdGlvblNoaWZ0WzBdKSArICcgJyArXG4gICAgdi5mbGlwRmFjdG9yICoga2VlcFByZWNpc2lvbih2LmJhY2tib25lUG9zaXRpb25TaGlmdFsxXSkgKyAnICcgKyAnMC4wMDAwMDAgMCcgKyAnXFxuJztcblxuICAvLyBhZGQgdGVybWluYWwgYm9uZFxuICBjb25zdCBmaXJzdEF0b20gPSB2LmJhY2tib25lQXR0YWNoTm9kZTtcbiAgY29uc3Qgc2Vjb25kQXRvbSA9IGF0b21JZHg7XG4gIG1vbGZpbGVCb25kQmxvY2tbQy5ib25kQ291bnRdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIHYuYm9uZFNoaWZ0ICsgJyAnICtcbiAgICAxICsgJyAnICsgZmlyc3RBdG9tICsgJyAnICsgc2Vjb25kQXRvbSArICdcXG4nO1xufVxuXG4vLyB0b2RvOiBkb2NcbmZ1bmN0aW9uIGFkZEFtaW5vQWNpZFRvTW9sYmxvY2sobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLFxuICBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcywgQzogTG9vcENvbnN0YW50c1xuKTogdm9pZCB7XG4gIHYuZmxpcEZhY3RvciA9ICgtMSkqKih2LmkgJSAyKTsgLy8gdG8gZmxpcCBldmVyeSBldmVuIG1vbm9tZXIgb3ZlciBPWFxuICBhZGRCYWNrYm9uZU1vbm9tZXJUb01vbGJsb2NrKG1vbm9tZXIsIG1vbGZpbGVBdG9tQmxvY2ssIG1vbGZpbGVCb25kQmxvY2ssIHYsIEMpO1xufVxuXG5mdW5jdGlvbiBhZGRCYWNrYm9uZU1vbm9tZXJUb01vbGJsb2NrKFxuICBtb25vbWVyOiBNb2xHcmFwaCwgbW9sZmlsZUF0b21CbG9jazogc3RyaW5nW10sIG1vbGZpbGVCb25kQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzLCBDOiBMb29wQ29uc3RhbnRzXG4pOiB2b2lkIHtcbiAgLy8gdG9kbzogcmVtb3ZlIHRoZXNlIGNvbW1lbnRzIHRvIHRoZSBkb2NzdHJpbmdzIG9mIHRoZSBjb3JyLiBmdW5jdGlvbnNcbiAgLy8gY29uc3RydW5jdCB0aGUgbGluZXMgb2YgVjNLIG1vbGZpbGUgYXRvbSBibG9ja1xuICBmaWxsQXRvbUxpbmVzKG1vbm9tZXIsIG1vbGZpbGVBdG9tQmxvY2ssIHYpO1xuXG4gIC8vIGNvbnN0cnVjdCB0aGUgbGluZXMgb2YgVjNLIG1vbGZpbGUgYm9uZCBibG9ja1xuICBmaWxsQm9uZExpbmVzKG1vbm9tZXIsIG1vbGZpbGVCb25kQmxvY2ssIHYpO1xuXG4gIC8vIHBlcHRpZGUgYm9uZFxuICBmaWxsQ2hhaW5FeHRlbmRpbmdCb25kKG1vbm9tZXIsIG1vbGZpbGVCb25kQmxvY2ssIHYpO1xuXG4gIC8vIHVwZGF0ZSBicmFuY2ggdmFyaWFibGVzIGlmIG5lY2Vzc2FyeVxuICBpZiAobW9ub21lci5tZXRhLmJyYW5jaFNoaWZ0ICE9PSBudWxsICYmIG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzLmxlbmd0aCA+IDIpXG4gICAgdXBkYXRlQnJhbmNoVmFyaWFibGVzKG1vbm9tZXIsIHYpO1xuXG4gIC8vIHVwZGF0ZSBsb29wIHZhcmlhYmxlc1xuICB1cGRhdGVDaGFpbkV4dGVuZGluZ1ZhcmlhYmxlcyhtb25vbWVyLCB2LCBDKTtcbn1cblxuLy8gdG9kbzogZG9jXG5mdW5jdGlvbiBhZGROdWNsZW90aWRlVG9Nb2xibG9jayhcbiAgbnVjbGVvYmFzZTogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcywgQzogTG9vcENvbnN0YW50c1xuKTogdm9pZCB7XG4gIC8vIGNvbnN0cnVuY3QgdGhlIGxpbmVzIG9mIFYzSyBtb2xmaWxlIGF0b20gYmxvY2sgY29ycmVzcG9uZGluZyB0byBwaG9zcGhhdGVcbiAgLy8gYW5kIHN1Z2FyXG4gIGZvciAoY29uc3QgbW9ub21lciBvZiBbQy5waG9zcGhhdGUsIEMuc3VnYXJdKVxuICAgIGFkZEJhY2tib25lTW9ub21lclRvTW9sYmxvY2sobW9ub21lciEsIG1vbGZpbGVBdG9tQmxvY2ssIG1vbGZpbGVCb25kQmxvY2ssIHYsIEMpO1xuXG4gIGFkZEJyYW5jaE1vbm9tZXJUb01vbGJsb2NrKG51Y2xlb2Jhc2UsIG1vbGZpbGVBdG9tQmxvY2ssIG1vbGZpbGVCb25kQmxvY2ssIHYsIEMpO1xufVxuXG5mdW5jdGlvbiBhZGRCcmFuY2hNb25vbWVyVG9Nb2xibG9jayhcbiAgbW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcywgQzogTG9vcENvbnN0YW50c1xuKTogdm9pZCB7XG4gIGZpbGxCcmFuY2hBdG9tTGluZXMobW9ub21lciwgbW9sZmlsZUF0b21CbG9jaywgdik7XG4gIGZpbGxCb25kTGluZXMobW9ub21lciwgbW9sZmlsZUJvbmRCbG9jaywgdik7XG4gIGZpbGxCYWNrYm9uZVRvQnJhbmNoQm9uZChtb25vbWVyLCBtb2xmaWxlQm9uZEJsb2NrLCB2KTtcblxuICAvLyBDLU4gYm9uZFxuICBjb25zdCBib25kSWR4ID0gdi5ib25kU2hpZnQ7XG4gIGNvbnN0IGZpcnN0QXRvbSA9IHYuYnJhbmNoQXR0YWNoTm9kZTtcbiAgY29uc3Qgc2Vjb25kQXRvbSA9IG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzBdICsgdi5ub2RlU2hpZnQ7XG4gIG1vbGZpbGVCb25kQmxvY2tbYm9uZElkeCAtIDFdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIGJvbmRJZHggKyAnICcgK1xuICAgIDEgKyAnICcgKyBmaXJzdEF0b20gKyAnICcgKyBzZWNvbmRBdG9tICsgJ1xcbic7XG5cbiAgLy8gdXBkYXRlIGxvb3AgdmFyaWFibGVzXG4gIHYuYm9uZFNoaWZ0ICs9IG1vbm9tZXIuYm9uZHMuYXRvbVBhaXJzLmxlbmd0aCArIDE7XG4gIHYubm9kZVNoaWZ0ICs9IG1vbm9tZXIuYXRvbXMuYXRvbVR5cGVzLmxlbmd0aDtcbn1cblxuZnVuY3Rpb24gdXBkYXRlQ2hhaW5FeHRlbmRpbmdWYXJpYWJsZXMobW9ub21lcjogTW9sR3JhcGgsIHY6IExvb3BWYXJpYWJsZXMsIEM6IExvb3BDb25zdGFudHMpOiB2b2lkIHtcbiAgdi5iYWNrYm9uZUF0dGFjaE5vZGUgPSB2Lm5vZGVTaGlmdCArIG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzFdO1xuICB2LmJvbmRTaGlmdCArPSBtb25vbWVyLmJvbmRzLmF0b21QYWlycy5sZW5ndGggKyAxO1xuXG4gIHYubm9kZVNoaWZ0ICs9IG1vbm9tZXIuYXRvbXMuYXRvbVR5cGVzLmxlbmd0aDtcbiAgdi5iYWNrYm9uZVBvc2l0aW9uU2hpZnRbMF0gKz0gbW9ub21lci5tZXRhLmJhY2tib25lU2hpZnQhWzBdOyAvLyB0b2RvOiBub24tbnVsbCBjaGVja1xuICB2LmJhY2tib25lUG9zaXRpb25TaGlmdFsxXSArPSB2LmZsaXBGYWN0b3IgKiBtb25vbWVyLm1ldGEuYmFja2JvbmVTaGlmdCFbMV07XG59XG5cbmZ1bmN0aW9uIHVwZGF0ZUJyYW5jaFZhcmlhYmxlcyhtb25vbWVyOiBNb2xHcmFwaCwgdjogTG9vcFZhcmlhYmxlcykge1xuICB2LmJyYW5jaEF0dGFjaE5vZGUgPSB2Lm5vZGVTaGlmdCArIG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzJdO1xuICBmb3IgKGxldCBpID0gMDsgaSA8IDI7ICsraSlcbiAgICB2LmJyYW5jaFBvc2l0aW9uU2hpZnRbaV0gPSB2LmJhY2tib25lUG9zaXRpb25TaGlmdFtpXSArIG1vbm9tZXIubWV0YS5icmFuY2hTaGlmdCFbaV07XG59XG5cbmZ1bmN0aW9uIGZpbGxBdG9tTGluZXMobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzKTogdm9pZCB7XG4gIGZvciAobGV0IGogPSAwOyBqIDwgbW9ub21lci5hdG9tcy5hdG9tVHlwZXMubGVuZ3RoOyArK2opIHtcbiAgICBjb25zdCBhdG9tSWR4ID0gdi5ub2RlU2hpZnQgKyBqICsgMTtcbiAgICBtb2xmaWxlQXRvbUJsb2NrW3Yubm9kZVNoaWZ0ICsgal0gPSBWM0tfQkVHSU5fREFUQV9MSU5FICsgYXRvbUlkeCArICcgJyArXG4gICAgICBtb25vbWVyLmF0b21zLmF0b21UeXBlc1tqXSArICcgJyArXG4gICAgICBrZWVwUHJlY2lzaW9uKHYuYmFja2JvbmVQb3NpdGlvblNoaWZ0WzBdICsgbW9ub21lci5hdG9tcy54W2pdKSArICcgJyArXG4gICAgICBrZWVwUHJlY2lzaW9uKHYuYmFja2JvbmVQb3NpdGlvblNoaWZ0WzFdICsgdi5mbGlwRmFjdG9yICogbW9ub21lci5hdG9tcy55W2pdKSArXG4gICAgICAnICcgKyBtb25vbWVyLmF0b21zLmt3YXJnc1tqXTtcbiAgfVxufVxuXG4vLyB0b2RvOiByZW1vdmUgYXMgcXVpY2tmaXhcbmZ1bmN0aW9uIGZpbGxCcmFuY2hBdG9tTGluZXMobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVBdG9tQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzKTogdm9pZCB7XG4gIGZvciAobGV0IGogPSAwOyBqIDwgbW9ub21lci5hdG9tcy5hdG9tVHlwZXMubGVuZ3RoOyArK2opIHtcbiAgICBjb25zdCBhdG9tSWR4ID0gdi5ub2RlU2hpZnQgKyBqICsgMTtcbiAgICBtb2xmaWxlQXRvbUJsb2NrW3Yubm9kZVNoaWZ0ICsgal0gPSBWM0tfQkVHSU5fREFUQV9MSU5FICsgYXRvbUlkeCArICcgJyArXG4gICAgICBtb25vbWVyLmF0b21zLmF0b21UeXBlc1tqXSArICcgJyArXG4gICAgICBrZWVwUHJlY2lzaW9uKHYuYnJhbmNoUG9zaXRpb25TaGlmdFswXSArIG1vbm9tZXIuYXRvbXMueFtqXSkgKyAnICcgK1xuICAgICAga2VlcFByZWNpc2lvbih2LmJyYW5jaFBvc2l0aW9uU2hpZnRbMV0gKyB2LmZsaXBGYWN0b3IgKiBtb25vbWVyLmF0b21zLnlbal0pICtcbiAgICAgICcgJyArIG1vbm9tZXIuYXRvbXMua3dhcmdzW2pdO1xuICB9XG59XG5cbmZ1bmN0aW9uIGZpbGxCb25kTGluZXMobW9ub21lcjogTW9sR3JhcGgsIG1vbGZpbGVCb25kQmxvY2s6IHN0cmluZ1tdLCB2OiBMb29wVmFyaWFibGVzKTogdm9pZCB7XG4gIC8vIGNvbnN0cnVjdCB0aGUgbGluZXMgb2YgVjNLIG1vbGZpbGUgYm9uZCBibG9ja1xuICBmb3IgKGxldCBqID0gMDsgaiA8IG1vbm9tZXIuYm9uZHMuYXRvbVBhaXJzLmxlbmd0aDsgKytqKSB7XG4gICAgY29uc3QgYm9uZElkeCA9IHYuYm9uZFNoaWZ0ICsgaiArIDE7XG4gICAgY29uc3QgZmlyc3RBdG9tID0gbW9ub21lci5ib25kcy5hdG9tUGFpcnNbal1bMF0gKyB2Lm5vZGVTaGlmdDtcbiAgICBjb25zdCBzZWNvbmRBdG9tID0gbW9ub21lci5ib25kcy5hdG9tUGFpcnNbal1bMV0gKyB2Lm5vZGVTaGlmdDtcbiAgICBsZXQgYm9uZENmZyA9ICcnO1xuICAgIGlmIChtb25vbWVyLmJvbmRzLmJvbmRDb25maWd1cmF0aW9uLmhhcyhqKSkge1xuICAgICAgLy8gZmxpcCBvcmllbnRhdGlvbiB3aGVuIG5lY2Vzc2FyeVxuICAgICAgbGV0IG9yaWVudGF0aW9uID0gbW9ub21lci5ib25kcy5ib25kQ29uZmlndXJhdGlvbi5nZXQoaik7XG4gICAgICBpZiAodi5mbGlwRmFjdG9yIDwgMClcbiAgICAgICAgb3JpZW50YXRpb24gPSAob3JpZW50YXRpb24gPT09IDEpID8gMyA6IDE7XG4gICAgICBib25kQ2ZnID0gJyBDRkc9JyArIG9yaWVudGF0aW9uO1xuICAgIH1cbiAgICBjb25zdCBrd2FyZ3MgPSBtb25vbWVyLmJvbmRzLmt3YXJncy5oYXMoaikgP1xuICAgICAgJyAnICsgbW9ub21lci5ib25kcy5rd2FyZ3MuZ2V0KGopIDogJyc7XG4gICAgbW9sZmlsZUJvbmRCbG9ja1t2LmJvbmRTaGlmdCArIGpdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIGJvbmRJZHggKyAnICcgK1xuICAgICAgbW9ub21lci5ib25kcy5ib25kVHlwZXNbal0gKyAnICcgK1xuICAgICAgZmlyc3RBdG9tICsgJyAnICsgc2Vjb25kQXRvbSArIGJvbmRDZmcgKyBrd2FyZ3MgKyAnXFxuJztcbiAgfVxufVxuXG5mdW5jdGlvbiBmaWxsQ2hhaW5FeHRlbmRpbmdCb25kKG1vbm9tZXI6IE1vbEdyYXBoLCBtb2xmaWxlQm9uZEJsb2NrOiBzdHJpbmdbXSwgdjogTG9vcFZhcmlhYmxlcyk6IHZvaWQge1xuICBpZiAodi5iYWNrYm9uZUF0dGFjaE5vZGUgIT09IDApIHtcbiAgICBjb25zdCBib25kSWR4ID0gdi5ib25kU2hpZnQ7XG4gICAgY29uc3QgZmlyc3RBdG9tID0gdi5iYWNrYm9uZUF0dGFjaE5vZGU7XG4gICAgY29uc3Qgc2Vjb25kQXRvbSA9IG1vbm9tZXIubWV0YS50ZXJtaW5hbE5vZGVzWzBdICsgdi5ub2RlU2hpZnQ7XG4gICAgbW9sZmlsZUJvbmRCbG9ja1t2LmJvbmRTaGlmdCAtIDFdID0gVjNLX0JFR0lOX0RBVEFfTElORSArIGJvbmRJZHggKyAnICcgK1xuICAgICAgMSArICcgJyArIGZpcnN0QXRvbSArICcgJyArIHNlY29uZEF0b20gKyAnXFxuJztcbiAgfVxufVxuXG4vLyB0b2RvOiByZW1vdmVcbmZ1bmN0aW9uIGZpbGxCYWNrYm9uZVRvQnJhbmNoQm9uZChicmFuY2hNb25vbWVyOiBNb2xHcmFwaCwgbW9sZmlsZUJvbmRCbG9jazogc3RyaW5nW10sIHY6IExvb3BWYXJpYWJsZXMpOiB2b2lkIHtcbiAgY29uc3QgYm9uZElkeCA9IHYuYm9uZFNoaWZ0O1xuICBjb25zdCBmaXJzdEF0b20gPSB2LmJyYW5jaEF0dGFjaE5vZGU7XG4gIGNvbnN0IHNlY29uZEF0b20gPSBicmFuY2hNb25vbWVyLm1ldGEudGVybWluYWxOb2Rlc1swXSArIHYubm9kZVNoaWZ0O1xuICBtb2xmaWxlQm9uZEJsb2NrW2JvbmRJZHggLSAxXSA9IFYzS19CRUdJTl9EQVRBX0xJTkUgKyBib25kSWR4ICsgJyAnICtcbiAgICAxICsgJyAnICsgZmlyc3RBdG9tICsgJyAnICsgc2Vjb25kQXRvbSArICdcXG4nO1xufVxuXG4vKiBDb21wdXRlIHRoZSBhdG9tL2JvbmQgY291bnRzIGZvciB0aGUgcmVzdWx0aW5nIG1vbGZpbGUsIGRlcGVuZGluZyBvbiB0aGVcbiAqIHR5cGUgb2YgcG9seW1lciAocGVwdGlkZS9udWNsZW90aWRlKSAqL1xuZnVuY3Rpb24gZ2V0UmVzdWx0aW5nQXRvbUJvbmRDb3VudHMoXG4gIG1vbm9tZXJTZXE6IHN0cmluZ1tdLCBtb25vbWVyc0RpY3Q6IE1hcDxzdHJpbmcsIE1vbEdyYXBoPixcbiAgYWxwaGFiZXQ6IEFMUEhBQkVULCBwb2x5bWVyVHlwZTogSEVMTV9QT0xZTUVSX1RZUEVcbik6IHthdG9tQ291bnQ6IG51bWJlciwgYm9uZENvdW50OiBudW1iZXJ9IHtcbiAgbGV0IGF0b21Db3VudCA9IDA7XG4gIGxldCBib25kQ291bnQgPSAwO1xuXG4gIC8vIHN1bSB1cCBhbGwgdGhlIGF0b21zL25vZGVzIHByb3ZpZGVkIGJ5IHRoZSBzZXF1ZW5jZVxuICBmb3IgKGNvbnN0IG1vbm9tZXJTeW1ib2wgb2YgbW9ub21lclNlcSkge1xuICAgIGNvbnN0IG1vbm9tZXIgPSBtb25vbWVyc0RpY3QuZ2V0KG1vbm9tZXJTeW1ib2wpITtcbiAgICBhdG9tQ291bnQgKz0gbW9ub21lci5hdG9tcy54Lmxlbmd0aDtcbiAgICBib25kQ291bnQgKz0gbW9ub21lci5ib25kcy5ib25kVHlwZXMubGVuZ3RoO1xuICB9XG5cbiAgLy8gYWRkIGV4dHJhIHZhbHVlcyBkZXBlbmRpbmcgb24gdGhlIHBvbHltZXIgdHlwZVxuICBpZiAocG9seW1lclR5cGUgPT09IEhFTE1fUE9MWU1FUl9UWVBFLlBFUFRJREUpIHtcbiAgICAvLyBhZGQgdGhlIHJpZ2h0bW9zdC90ZXJtaW5hdGluZyBjYXAgZ3JvdXAgJ09IJyAoaS5lLiAnTycpXG4gICAgYXRvbUNvdW50ICs9IDE7XG4gICAgLy8gYWRkIGNoYWluLWV4dGVuZGluZyBib25kcyAoQy1OSCBwZXIgZWFjaCBtb25vbWVyIHBhaXIgYW5kIHRlcm1pbmFsIEMtT0gpXG4gICAgYm9uZENvdW50ICs9IG1vbm9tZXJTZXEubGVuZ3RoO1xuICB9IGVsc2UgeyAvLyBudWNsZW90aWRlc1xuICAgIGNvbnN0IHN1Z2FyID0gKGFscGhhYmV0ID09PSBBTFBIQUJFVC5ETkEpID9cbiAgICAgIG1vbm9tZXJzRGljdC5nZXQoREVPWFlSSUJPU0UpISA6IG1vbm9tZXJzRGljdC5nZXQoUklCT1NFKSE7XG4gICAgY29uc3QgcGhvc3BoYXRlID0gbW9ub21lcnNEaWN0LmdldChQSE9TUEhBVEUpITtcblxuICAgIC8vIGFkZCBwaG9zcGhhdGUgYW5kIHN1Z2FyIHBlciBlYWNoIG51Y2xlb2Jhc2Ugc3ltYm9sXG4gICAgYXRvbUNvdW50ICs9IG1vbm9tZXJTZXEubGVuZ3RoICogKHBob3NwaGF0ZS5hdG9tcy54Lmxlbmd0aCArIHN1Z2FyLmF0b21zLngubGVuZ3RoKTtcbiAgICAvLyBhZGQgdGhlIGxlZnRtb3N0IGNhcCBncm91cCAnT0gnIChpLmUuICdPJykgdG8gdGhlIGZpcnN0IHBob3NwaGF0ZVxuICAgIGF0b21Db3VudCArPSAxO1xuXG4gICAgLy8gYWRkIGJvbmRzIGZyb20gcGhvc3BoYXRlIGFuZCBzdWdhclxuICAgIGJvbmRDb3VudCArPSBtb25vbWVyU2VxLmxlbmd0aCAqIChwaG9zcGhhdGUuYm9uZHMuYm9uZFR5cGVzLmxlbmd0aCArIHN1Z2FyLmJvbmRzLmJvbmRUeXBlcy5sZW5ndGgpO1xuXG4gICAgLy8gYWRkIGNoYWluLWV4dGVuZGluZyBhbmQgYnJhbmNoIGJvbmRzIChPLVAsIEMtTyBhbmQgQy1OIHBlciBlYWNoIG51Y2xlb3RpZGUpXG4gICAgYm9uZENvdW50ICs9IG1vbm9tZXJTZXEubGVuZ3RoICogMztcbiAgfVxuXG4gIHJldHVybiB7YXRvbUNvdW50LCBib25kQ291bnR9O1xufVxuXG4vKiBLZWVwIHByZWNpc2lvbiB1cG9uIGZsb2F0aW5nIHBvaW50IG9wZXJhdGlvbnMgb3ZlciBhdG9tIGNvb3JkaW5hdGVzICovXG5mdW5jdGlvbiBrZWVwUHJlY2lzaW9uKHg6IG51bWJlcikge1xuICByZXR1cm4gTWF0aC5yb3VuZChQUkVDSVNJT05fRkFDVE9SICogeCkvUFJFQ0lTSU9OX0ZBQ1RPUjtcbn1cbiJdfQ==
|