source: trip-planner-front/node_modules/@angular/compiler-cli/ngcc/src/dependencies/esm_dependency_host.js@ 59329aa

Last change on this file since 59329aa was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 38.6 KB
Line 
1(function (factory) {
2 if (typeof module === "object" && typeof module.exports === "object") {
3 var v = factory(require, exports);
4 if (v !== undefined) module.exports = v;
5 }
6 else if (typeof define === "function" && define.amd) {
7 define("@angular/compiler-cli/ngcc/src/dependencies/esm_dependency_host", ["require", "exports", "tslib", "typescript", "@angular/compiler-cli/ngcc/src/dependencies/dependency_host"], factory);
8 }
9})(function (require, exports) {
10 "use strict";
11 Object.defineProperty(exports, "__esModule", { value: true });
12 exports.isStringImportOrReexport = exports.hasImportOrReexportStatements = exports.EsmDependencyHost = void 0;
13 var tslib_1 = require("tslib");
14 /**
15 * @license
16 * Copyright Google LLC All Rights Reserved.
17 *
18 * Use of this source code is governed by an MIT-style license that can be
19 * found in the LICENSE file at https://angular.io/license
20 */
21 var ts = require("typescript");
22 var dependency_host_1 = require("@angular/compiler-cli/ngcc/src/dependencies/dependency_host");
23 /**
24 * Helper functions for computing dependencies.
25 */
26 var EsmDependencyHost = /** @class */ (function (_super) {
27 tslib_1.__extends(EsmDependencyHost, _super);
28 function EsmDependencyHost(fs, moduleResolver, scanImportExpressions) {
29 if (scanImportExpressions === void 0) { scanImportExpressions = true; }
30 var _this = _super.call(this, fs, moduleResolver) || this;
31 _this.scanImportExpressions = scanImportExpressions;
32 // By skipping trivia here we don't have to account for it in the processing below
33 // It has no relevance to capturing imports.
34 _this.scanner = ts.createScanner(ts.ScriptTarget.Latest, /* skipTrivia */ true);
35 return _this;
36 }
37 EsmDependencyHost.prototype.canSkipFile = function (fileContents) {
38 return !hasImportOrReexportStatements(fileContents);
39 };
40 /**
41 * Extract any import paths from imports found in the contents of this file.
42 *
43 * This implementation uses the TypeScript scanner, which tokenizes source code,
44 * to process the string. This is halfway between working with the string directly,
45 * which is too difficult due to corner cases, and parsing the string into a full
46 * TypeScript Abstract Syntax Tree (AST), which ends up doing more processing than
47 * is needed.
48 *
49 * The scanning is not trivial because we must hold state between each token since
50 * the context of the token affects how it should be scanned, and the scanner does
51 * not manage this for us.
52 *
53 * Specifically, backticked strings are particularly challenging since it is possible
54 * to recursively nest backticks and TypeScript expressions within each other.
55 */
56 EsmDependencyHost.prototype.extractImports = function (file, fileContents) {
57 var imports = new Set();
58 var templateStack = [];
59 var lastToken = ts.SyntaxKind.Unknown;
60 var currentToken = ts.SyntaxKind.Unknown;
61 var stopAtIndex = findLastPossibleImportOrReexport(fileContents);
62 this.scanner.setText(fileContents);
63 while ((currentToken = this.scanner.scan()) !== ts.SyntaxKind.EndOfFileToken) {
64 if (this.scanner.getTokenPos() > stopAtIndex) {
65 break;
66 }
67 switch (currentToken) {
68 case ts.SyntaxKind.TemplateHead:
69 // TemplateHead indicates the beginning of a backticked string
70 // Capture this in the `templateStack` to indicate we are currently processing
71 // within the static text part of a backticked string.
72 templateStack.push(currentToken);
73 break;
74 case ts.SyntaxKind.OpenBraceToken:
75 if (templateStack.length > 0) {
76 // We are processing a backticked string. This indicates that we are either
77 // entering an interpolation expression or entering an object literal expression.
78 // We add it to the `templateStack` so we can track when we leave the interpolation or
79 // object literal.
80 templateStack.push(currentToken);
81 }
82 break;
83 case ts.SyntaxKind.CloseBraceToken:
84 if (templateStack.length > 0) {
85 // We are processing a backticked string then this indicates that we are either
86 // leaving an interpolation expression or leaving an object literal expression.
87 var templateToken = templateStack[templateStack.length - 1];
88 if (templateToken === ts.SyntaxKind.TemplateHead) {
89 // We have hit a nested backticked string so we need to rescan it in that context
90 currentToken = this.scanner.reScanTemplateToken(/* isTaggedTemplate */ false);
91 if (currentToken === ts.SyntaxKind.TemplateTail) {
92 // We got to the end of the backticked string so pop the token that started it off
93 // the stack.
94 templateStack.pop();
95 }
96 }
97 else {
98 // We hit the end of an object-literal expression so pop the open-brace that started
99 // it off the stack.
100 templateStack.pop();
101 }
102 }
103 break;
104 case ts.SyntaxKind.SlashToken:
105 case ts.SyntaxKind.SlashEqualsToken:
106 if (canPrecedeARegex(lastToken)) {
107 // We have hit a slash (`/`) in a context where it could be the start of a regular
108 // expression so rescan it in that context
109 currentToken = this.scanner.reScanSlashToken();
110 }
111 break;
112 case ts.SyntaxKind.ImportKeyword:
113 var importPath = this.extractImportPath();
114 if (importPath !== null) {
115 imports.add(importPath);
116 }
117 break;
118 case ts.SyntaxKind.ExportKeyword:
119 var reexportPath = this.extractReexportPath();
120 if (reexportPath !== null) {
121 imports.add(reexportPath);
122 }
123 break;
124 }
125 lastToken = currentToken;
126 }
127 // Clear the text from the scanner to avoid holding on to potentially large strings of source
128 // content after the scanning has completed.
129 this.scanner.setText('');
130 return imports;
131 };
132 /**
133 * We have found an `import` token so now try to identify the import path.
134 *
135 * This method will use the current state of `this.scanner` to extract a string literal module
136 * specifier. It expects that the current state of the scanner is that an `import` token has just
137 * been scanned.
138 *
139 * The following forms of import are matched:
140 *
141 * * `import "module-specifier";`
142 * * `import("module-specifier")`
143 * * `import defaultBinding from "module-specifier";`
144 * * `import defaultBinding, * as identifier from "module-specifier";`
145 * * `import defaultBinding, {...} from "module-specifier";`
146 * * `import * as identifier from "module-specifier";`
147 * * `import {...} from "module-specifier";`
148 *
149 * @returns the import path or null if there is no import or it is not a string literal.
150 */
151 EsmDependencyHost.prototype.extractImportPath = function () {
152 // Check for side-effect import
153 var sideEffectImportPath = this.tryStringLiteral();
154 if (sideEffectImportPath !== null) {
155 return sideEffectImportPath;
156 }
157 var kind = this.scanner.getToken();
158 // Check for dynamic import expression
159 if (kind === ts.SyntaxKind.OpenParenToken) {
160 return this.scanImportExpressions ? this.tryStringLiteral() : null;
161 }
162 // Check for defaultBinding
163 if (kind === ts.SyntaxKind.Identifier) {
164 // Skip default binding
165 kind = this.scanner.scan();
166 if (kind === ts.SyntaxKind.CommaToken) {
167 // Skip comma that indicates additional import bindings
168 kind = this.scanner.scan();
169 }
170 }
171 // Check for namespace import clause
172 if (kind === ts.SyntaxKind.AsteriskToken) {
173 kind = this.skipNamespacedClause();
174 if (kind === null) {
175 return null;
176 }
177 }
178 // Check for named imports clause
179 else if (kind === ts.SyntaxKind.OpenBraceToken) {
180 kind = this.skipNamedClause();
181 }
182 // Expect a `from` clause, if not bail out
183 if (kind !== ts.SyntaxKind.FromKeyword) {
184 return null;
185 }
186 return this.tryStringLiteral();
187 };
188 /**
189 * We have found an `export` token so now try to identify a re-export path.
190 *
191 * This method will use the current state of `this.scanner` to extract a string literal module
192 * specifier. It expects that the current state of the scanner is that an `export` token has
193 * just been scanned.
194 *
195 * There are three forms of re-export that are matched:
196 *
197 * * `export * from '...';
198 * * `export * as alias from '...';
199 * * `export {...} from '...';
200 */
201 EsmDependencyHost.prototype.extractReexportPath = function () {
202 // Skip the `export` keyword
203 var token = this.scanner.scan();
204 if (token === ts.SyntaxKind.AsteriskToken) {
205 token = this.skipNamespacedClause();
206 if (token === null) {
207 return null;
208 }
209 }
210 else if (token === ts.SyntaxKind.OpenBraceToken) {
211 token = this.skipNamedClause();
212 }
213 // Expect a `from` clause, if not bail out
214 if (token !== ts.SyntaxKind.FromKeyword) {
215 return null;
216 }
217 return this.tryStringLiteral();
218 };
219 EsmDependencyHost.prototype.skipNamespacedClause = function () {
220 // Skip past the `*`
221 var token = this.scanner.scan();
222 // Check for a `* as identifier` alias clause
223 if (token === ts.SyntaxKind.AsKeyword) {
224 // Skip past the `as` keyword
225 token = this.scanner.scan();
226 // Expect an identifier, if not bail out
227 if (token !== ts.SyntaxKind.Identifier) {
228 return null;
229 }
230 // Skip past the identifier
231 token = this.scanner.scan();
232 }
233 return token;
234 };
235 EsmDependencyHost.prototype.skipNamedClause = function () {
236 var braceCount = 1;
237 // Skip past the initial opening brace `{`
238 var token = this.scanner.scan();
239 // Search for the matching closing brace `}`
240 while (braceCount > 0 && token !== ts.SyntaxKind.EndOfFileToken) {
241 if (token === ts.SyntaxKind.OpenBraceToken) {
242 braceCount++;
243 }
244 else if (token === ts.SyntaxKind.CloseBraceToken) {
245 braceCount--;
246 }
247 token = this.scanner.scan();
248 }
249 return token;
250 };
251 EsmDependencyHost.prototype.tryStringLiteral = function () {
252 return this.scanner.scan() === ts.SyntaxKind.StringLiteral ? this.scanner.getTokenValue() :
253 null;
254 };
255 return EsmDependencyHost;
256 }(dependency_host_1.DependencyHostBase));
257 exports.EsmDependencyHost = EsmDependencyHost;
258 /**
259 * Check whether a source file needs to be parsed for imports.
260 * This is a performance short-circuit, which saves us from creating
261 * a TypeScript AST unnecessarily.
262 *
263 * @param source The content of the source file to check.
264 *
265 * @returns false if there are definitely no import or re-export statements
266 * in this file, true otherwise.
267 */
268 function hasImportOrReexportStatements(source) {
269 return /(?:import|export)[\s\S]+?(["'])(?:\\\1|.)+?\1/.test(source);
270 }
271 exports.hasImportOrReexportStatements = hasImportOrReexportStatements;
272 function findLastPossibleImportOrReexport(source) {
273 return Math.max(source.lastIndexOf('import'), source.lastIndexOf(' from '));
274 }
275 /**
276 * Check whether the given statement is an import with a string literal module specifier.
277 * @param stmt the statement node to check.
278 * @returns true if the statement is an import with a string literal module specifier.
279 */
280 function isStringImportOrReexport(stmt) {
281 return ts.isImportDeclaration(stmt) ||
282 ts.isExportDeclaration(stmt) && !!stmt.moduleSpecifier &&
283 ts.isStringLiteral(stmt.moduleSpecifier);
284 }
285 exports.isStringImportOrReexport = isStringImportOrReexport;
286 function canPrecedeARegex(kind) {
287 switch (kind) {
288 case ts.SyntaxKind.Identifier:
289 case ts.SyntaxKind.StringLiteral:
290 case ts.SyntaxKind.NumericLiteral:
291 case ts.SyntaxKind.BigIntLiteral:
292 case ts.SyntaxKind.RegularExpressionLiteral:
293 case ts.SyntaxKind.ThisKeyword:
294 case ts.SyntaxKind.PlusPlusToken:
295 case ts.SyntaxKind.MinusMinusToken:
296 case ts.SyntaxKind.CloseParenToken:
297 case ts.SyntaxKind.CloseBracketToken:
298 case ts.SyntaxKind.CloseBraceToken:
299 case ts.SyntaxKind.TrueKeyword:
300 case ts.SyntaxKind.FalseKeyword:
301 return false;
302 default:
303 return true;
304 }
305 }
306});
307//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXNtX2RlcGVuZGVuY3lfaG9zdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvbXBpbGVyLWNsaS9uZ2NjL3NyYy9kZXBlbmRlbmNpZXMvZXNtX2RlcGVuZGVuY3lfaG9zdC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7O0lBQUE7Ozs7OztPQU1HO0lBQ0gsK0JBQWlDO0lBRWpDLCtGQUFxRDtJQUdyRDs7T0FFRztJQUNIO1FBQXVDLDZDQUFrQjtRQUN2RCwyQkFDSSxFQUFzQixFQUFFLGNBQThCLEVBQzlDLHFCQUE0QjtZQUE1QixzQ0FBQSxFQUFBLDRCQUE0QjtZQUZ4QyxZQUdFLGtCQUFNLEVBQUUsRUFBRSxjQUFjLENBQUMsU0FDMUI7WUFGVywyQkFBcUIsR0FBckIscUJBQXFCLENBQU87WUFHeEMsa0ZBQWtGO1lBQ2xGLDRDQUE0QztZQUNwQyxhQUFPLEdBQUcsRUFBRSxDQUFDLGFBQWEsQ0FBQyxFQUFFLENBQUMsWUFBWSxDQUFDLE1BQU0sRUFBRSxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQzs7UUFIbEYsQ0FBQztRQUtrQix1Q0FBVyxHQUE5QixVQUErQixZQUFvQjtZQUNqRCxPQUFPLENBQUMsNkJBQTZCLENBQUMsWUFBWSxDQUFDLENBQUM7UUFDdEQsQ0FBQztRQUVEOzs7Ozs7Ozs7Ozs7Ozs7V0FlRztRQUNnQiwwQ0FBYyxHQUFqQyxVQUFrQyxJQUFvQixFQUFFLFlBQW9CO1lBQzFFLElBQU0sT0FBTyxHQUFHLElBQUksR0FBRyxFQUFVLENBQUM7WUFDbEMsSUFBTSxhQUFhLEdBQW9CLEVBQUUsQ0FBQztZQUMxQyxJQUFJLFNBQVMsR0FBa0IsRUFBRSxDQUFDLFVBQVUsQ0FBQyxPQUFPLENBQUM7WUFDckQsSUFBSSxZQUFZLEdBQWtCLEVBQUUsQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDO1lBQ3hELElBQU0sV0FBVyxHQUFHLGdDQUFnQyxDQUFD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O1lBQ25DLEtBQUssRUFBRSxDQUFDLFVBQVUsQ0FBQyxpQkFBaUIsQ0FBQztZQUNyQyxLQUFLLEVBQUUsQ0FBQyxVQUFVLENBQUMsZUFBZSxDQUFDO1lBQ25DLEtBQUssRUFBRSxDQUFDLFVBQVUsQ0FBQyxXQUFXLENBQUM7WUFDL0IsS0FBSyxFQUFFLENBQUMsVUFBVSxDQUFDLFlBQVk7Z0JBQzdCLE9BQU8sS0FBSyxDQUFDO1lBQ2Y7Z0JBQ0UsT0FBTyxJQUFJLENBQUM7U0FDZjtJQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cbmltcG9ydCAqIGFzIHRzIGZyb20gJ3R5cGVzY3JpcHQnO1xuaW1wb3J0IHtBYnNvbHV0ZUZzUGF0aCwgUmVhZG9ubHlGaWxlU3lzdGVtfSBmcm9tICcuLi8uLi8uLi9zcmMvbmd0c2MvZmlsZV9zeXN0ZW0nO1xuaW1wb3J0IHtEZXBlbmRlbmN5SG9zdEJhc2V9IGZyb20gJy4vZGVwZW5kZW5jeV9ob3N0JztcbmltcG9ydCB7TW9kdWxlUmVzb2x2ZXJ9IGZyb20gJy4vbW9kdWxlX3Jlc29sdmVyJztcblxuLyoqXG4gKiBIZWxwZXIgZnVuY3Rpb25zIGZvciBjb21wdXRpbmcgZGVwZW5kZW5jaWVzLlxuICovXG5leHBvcnQgY2xhc3MgRXNtRGVwZW5kZW5jeUhvc3QgZXh0ZW5kcyBEZXBlbmRlbmN5SG9zdEJhc2Uge1xuICBjb25zdHJ1Y3RvcihcbiAgICAgIGZzOiBSZWFkb25seUZpbGVTeXN0ZW0sIG1vZHVsZVJlc29sdmVyOiBNb2R1bGVSZXNvbHZlcixcbiAgICAgIHByaXZhdGUgc2NhbkltcG9ydEV4cHJlc3Npb25zID0gdHJ1ZSkge1xuICAgIHN1cGVyKGZzLCBtb2R1bGVSZXNvbHZlcik7XG4gIH1cbiAgLy8gQnkgc2tpcHBpbmcgdHJpdmlhIGhlcmUgd2UgZG9uJ3QgaGF2ZSB0byBhY2NvdW50IGZvciBpdCBpbiB0aGUgcHJvY2Vzc2luZyBiZWxvd1xuICAvLyBJdCBoYXMgbm8gcmVsZXZhbmNlIHRvIGNhcHR1cmluZyBpbXBvcnRzLlxuICBwcml2YXRlIHNjYW5uZXIgPSB0cy5jcmVhdGVTY2FubmVyKHRzLlNjcmlwdFRhcmdldC5MYXRlc3QsIC8qIHNraXBUcml2aWEgKi8gdHJ1ZSk7XG5cbiAgcHJvdGVjdGVkIG92ZXJyaWRlIGNhblNraXBGaWxlKGZpbGVDb250ZW50czogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuICFoYXNJbXBvcnRPclJlZXhwb3J0U3RhdGVtZW50cyhmaWxlQ29udGVudHMpO1xuICB9XG5cbiAgLyoqXG4gICAqIEV4dHJhY3QgYW55IGltcG9ydCBwYXRocyBmcm9tIGltcG9ydHMgZm91bmQgaW4gdGhlIGNvbnRlbnRzIG9mIHRoaXMgZmlsZS5cbiAgICpcbiAgICogVGhpcyBpbXBsZW1lbnRhdGlvbiB1c2VzIHRoZSBUeXBlU2NyaXB0IHNjYW5uZXIsIHdoaWNoIHRva2VuaXplcyBzb3VyY2UgY29kZSxcbiAgICogdG8gcHJvY2VzcyB0aGUgc3RyaW5nLiBUaGlzIGlzIGhhbGZ3YXkgYmV0d2VlbiB3b3JraW5nIHdpdGggdGhlIHN0cmluZyBkaXJlY3RseSxcbiAgICogd2hpY2ggaXMgdG9vIGRpZmZpY3VsdCBkdWUgdG8gY29ybmVyIGNhc2VzLCBhbmQgcGFyc2luZyB0aGUgc3RyaW5nIGludG8gYSBmdWxsXG4gICAqIFR5cGVTY3JpcHQgQWJzdHJhY3QgU3ludGF4IFRyZWUgKEFTVCksIHdoaWNoIGVuZHMgdXAgZG9pbmcgbW9yZSBwcm9jZXNzaW5nIHRoYW5cbiAgICogaXMgbmVlZGVkLlxuICAgKlxuICAgKiBUaGUgc2Nhbm5pbmcgaXMgbm90IHRyaXZpYWwgYmVjYXVzZSB3ZSBtdXN0IGhvbGQgc3RhdGUgYmV0d2VlbiBlYWNoIHRva2VuIHNpbmNlXG4gICAqIHRoZSBjb250ZXh0IG9mIHRoZSB0b2tlbiBhZmZlY3RzIGhvdyBpdCBzaG91bGQgYmUgc2Nhbm5lZCwgYW5kIHRoZSBzY2FubmVyIGRvZXNcbiAgICogbm90IG1hbmFnZSB0aGlzIGZvciB1cy5cbiAgICpcbiAgICogU3BlY2lmaWNhbGx5LCBiYWNrdGlja2VkIHN0cmluZ3MgYXJlIHBhcnRpY3VsYXJseSBjaGFsbGVuZ2luZyBzaW5jZSBpdCBpcyBwb3NzaWJsZVxuICAgKiB0byByZWN1cnNpdmVseSBuZXN0IGJhY2t0aWNrcyBhbmQgVHlwZVNjcmlwdCBleHByZXNzaW9ucyB3aXRoaW4gZWFjaCBvdGhlci5cbiAgICovXG4gIHByb3RlY3RlZCBvdmVycmlkZSBleHRyYWN0SW1wb3J0cyhmaWxlOiBBYnNvbHV0ZUZzUGF0aCwgZmlsZUNvbnRlbnRzOiBzdHJpbmcpOiBTZXQ8c3RyaW5nPiB7XG4gICAgY29uc3QgaW1wb3J0cyA9IG5ldyBTZXQ8c3RyaW5nPigpO1xuICAgIGNvbnN0IHRlbXBsYXRlU3RhY2s6IHRzLlN5bnRheEtpbmRbXSA9IFtdO1xuICAgIGxldCBsYXN0VG9rZW46IHRzLlN5bnRheEtpbmQgPSB0cy5TeW50YXhLaW5kLlVua25vd247XG4gICAgbGV0IGN1cnJlbnRUb2tlbjogdHMuU3ludGF4S2luZCA9IHRzLlN5bnRheEtpbmQuVW5rbm93bjtcbiAgICBjb25zdCBzdG9wQXRJbmRleCA9IGZpbmRMYXN0UG9zc2libGVJbXBvcnRPclJlZXhwb3J0KGZpbGVDb250ZW50cyk7XG5cbiAgICB0aGlzLnNjYW5uZXIuc2V0VGV4dChmaWxlQ29udGVudHMpO1xuXG4gICAgd2hpbGUgKChjdXJyZW50VG9rZW4gPSB0aGlzLnNjYW5uZXIuc2NhbigpKSAhPT0gdHMuU3ludGF4S2luZC5FbmRPZkZpbGVUb2tlbikge1xuICAgICAgaWYgKHRoaXMuc2Nhbm5lci5nZXRUb2tlblBvcygpID4gc3RvcEF0SW5kZXgpIHtcbiAgICAgICAgYnJlYWs7XG4gICAgICB9XG4gICAgICBzd2l0Y2ggKGN1cnJlbnRUb2tlbikge1xuICAgICAgICBjYXNlIHRzLlN5bnRheEtpbmQuVGVtcGxhdGVIZWFkOlxuICAgICAgICAgIC8vIFRlbXBsYXRlSGVhZCBpbmRpY2F0ZXMgdGhlIGJlZ2lubmluZyBvZiBhIGJhY2t0aWNrZWQgc3RyaW5nXG4gICAgICAgICAgLy8gQ2FwdHVyZSB0aGlzIGluIHRoZSBgdGVtcGxhdGVTdGFja2AgdG8gaW5kaWNhdGUgd2UgYXJlIGN1cnJlbnRseSBwcm9jZXNzaW5nXG4gICAgICAgICAgLy8gd2l0aGluIHRoZSBzdGF0aWMgdGV4dCBwYXJ0IG9mIGEgYmFja3RpY2tlZCBzdHJpbmcuXG4gICAgICAgICAgdGVtcGxhdGVTdGFjay5wdXNoKGN1cnJlbnRUb2tlbik7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgdHMuU3ludGF4S2luZC5PcGVuQnJhY2VUb2tlbjpcbiAgICAgICAgICBpZiAodGVtcGxhdGVTdGFjay5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICAvLyBXZSBhcmUgcHJvY2Vzc2luZyBhIGJhY2t0aWNrZWQgc3RyaW5nLiBUaGlzIGluZGljYXRlcyB0aGF0IHdlIGFyZSBlaXRoZXJcbiAgICAgICAgICAgIC8vIGVudGVyaW5nIGFuIGludGVycG9sYXRpb24gZXhwcmVzc2lvbiBvciBlbnRlcmluZyBhbiBvYmplY3QgbGl0ZXJhbCBleHByZXNzaW9uLlxuICAgICAgICAgICAgLy8gV2UgYWRkIGl0IHRvIHRoZSBgdGVtcGxhdGVTdGFja2Agc28gd2UgY2FuIHRyYWNrIHdoZW4gd2UgbGVhdmUgdGhlIGludGVycG9sYXRpb24gb3JcbiAgICAgICAgICAgIC8vIG9iamVjdCBsaXRlcmFsLlxuICAgICAgICAgICAgdGVtcGxhdGVTdGFjay5wdXNoKGN1cnJlbnRUb2tlbik7XG4gICAgICAgICAgfVxuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIHRzLlN5bnRheEtpbmQuQ2xvc2VCcmFjZVRva2VuOlxuICAgICAgICAgIGlmICh0ZW1wbGF0ZVN0YWNrLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgIC8vIFdlIGFyZSBwcm9jZXNzaW5nIGEgYmFja3RpY2tlZCBzdHJpbmcgdGhlbiB0aGlzIGluZGljYXRlcyB0aGF0IHdlIGFyZSBlaXRoZXJcbiAgICAgICAgICAgIC8vIGxlYXZpbmcgYW4gaW50ZXJwb2xhdGlvbiBleHByZXNzaW9uIG9yIGxlYXZpbmcgYW4gb2JqZWN0IGxpdGVyYWwgZXhwcmVzc2lvbi5cbiAgICAgICAgICAgIGNvbnN0IHRlbXBsYXRlVG9rZW4gPSB0ZW1wbGF0ZVN0YWNrW3RlbXBsYXRlU3RhY2subGVuZ3RoIC0gMV07XG4gICAgICAgICAgICBpZiAodGVtcGxhdGVUb2tlbiA9PT0gdHMuU3ludGF4S2luZC5UZW1wbGF0ZUhlYWQpIHtcbiAgICAgICAgICAgICAgLy8gV2UgaGF2ZSBoaXQgYSBuZXN0ZWQgYmFja3RpY2tlZCBzdHJpbmcgc28gd2UgbmVlZCB0byByZXNjYW4gaXQgaW4gdGhhdCBjb250ZXh0XG4gICAgICAgICAgICAgIGN1cnJlbnRUb2tlbiA9IHRoaXMuc2Nhbm5lci5yZVNjYW5UZW1wbGF0ZVRva2VuKC8qIGlzVGFnZ2VkVGVtcGxhdGUgKi8gZmFsc2UpO1xuICAgICAgICAgICAgICBpZiAoY3VycmVudFRva2VuID09PSB0cy5TeW50YXhLaW5kLlRlbXBsYXRlVGFpbCkge1xuICAgICAgICAgICAgICAgIC8vIFdlIGdvdCB0byB0aGUgZW5kIG9mIHRoZSBiYWNrdGlja2VkIHN0cmluZyBzbyBwb3AgdGhlIHRva2VuIHRoYXQgc3RhcnRlZCBpdCBvZmZcbiAgICAgICAgICAgICAgICAvLyB0aGUgc3RhY2suXG4gICAgICAgICAgICAgICAgdGVtcGxhdGVTdGFjay5wb3AoKTtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgLy8gV2UgaGl0IHRoZSBlbmQgb2YgYW4gb2JqZWN0LWxpdGVyYWwgZXhwcmVzc2lvbiBzbyBwb3AgdGhlIG9wZW4tYnJhY2UgdGhhdCBzdGFydGVkXG4gICAgICAgICAgICAgIC8vIGl0IG9mZiB0aGUgc3RhY2suXG4gICAgICAgICAgICAgIHRlbXBsYXRlU3RhY2sucG9wKCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIHRzLlN5bnRheEtpbmQuU2xhc2hUb2tlbjpcbiAgICAgICAgY2FzZSB0cy5TeW50YXhLaW5kLlNsYXNoRXF1YWxzVG9rZW46XG4gICAgICAgICAgaWYgKGNhblByZWNlZGVBUmVnZXgobGFzdFRva2VuKSkge1xuICAgICAgICAgICAgLy8gV2UgaGF2ZSBoaXQgYSBzbGFzaCAoYC9gKSBpbiBhIGNvbnRleHQgd2hlcmUgaXQgY291bGQgYmUgdGhlIHN0YXJ0IG9mIGEgcmVndWxhclxuICAgICAgICAgICAgLy8gZXhwcmVzc2lvbiBzbyByZXNjYW4gaXQgaW4gdGhhdCBjb250ZXh0XG4gICAgICAgICAgICBjdXJyZW50VG9rZW4gPSB0aGlzLnNjYW5uZXIucmVTY2FuU2xhc2hUb2tlbigpO1xuICAgICAgICAgIH1cbiAgICAgICAgICBicmVhaztcbiAgICAgICAgY2FzZSB0cy5TeW50YXhLaW5kLkltcG9ydEtleXdvcmQ6XG4gICAgICAgICAgY29uc3QgaW1wb3J0UGF0aCA9IHRoaXMuZXh0cmFjdEltcG9ydFBhdGgoKTtcbiAgICAgICAgICBpZiAoaW1wb3J0UGF0aCAhPT0gbnVsbCkge1xuICAgICAgICAgICAgaW1wb3J0cy5hZGQoaW1wb3J0UGF0aCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIGJyZWFrO1xuICAgICAgICBjYXNlIHRzLlN5bnRheEtpbmQuRXhwb3J0S2V5d29yZDpcbiAgICAgICAgICBjb25zdCByZWV4cG9ydFBhdGggPSB0aGlzLmV4dHJhY3RSZWV4cG9ydFBhdGgoKTtcbiAgICAgICAgICBpZiAocmVleHBvcnRQYXRoICE9PSBudWxsKSB7XG4gICAgICAgICAgICBpbXBvcnRzLmFkZChyZWV4cG9ydFBhdGgpO1xuICAgICAgICAgIH1cbiAgICAgICAgICBicmVhaztcbiAgICAgIH1cbiAgICAgIGxhc3RUb2tlbiA9IGN1cnJlbnRUb2tlbjtcbiAgICB9XG5cbiAgICAvLyBDbGVhciB0aGUgdGV4dCBmcm9tIHRoZSBzY2FubmVyIHRvIGF2b2lkIGhvbGRpbmcgb24gdG8gcG90ZW50aWFsbHkgbGFyZ2Ugc3RyaW5ncyBvZiBzb3VyY2VcbiAgICAvLyBjb250ZW50IGFmdGVyIHRoZSBzY2FubmluZyBoYXMgY29tcGxldGVkLlxuICAgIHRoaXMuc2Nhbm5lci5zZXRUZXh0KCcnKTtcblxuICAgIHJldHVybiBpbXBvcnRzO1xuICB9XG5cblxuICAvKipcbiAgICogV2UgaGF2ZSBmb3VuZCBhbiBgaW1wb3J0YCB0b2tlbiBzbyBub3cgdHJ5IHRvIGlkZW50aWZ5IHRoZSBpbXBvcnQgcGF0aC5cbiAgICpcbiAgICogVGhpcyBtZXRob2Qgd2lsbCB1c2UgdGhlIGN1cnJlbnQgc3RhdGUgb2YgYHRoaXMuc2Nhbm5lcmAgdG8gZXh0cmFjdCBhIHN0cmluZyBsaXRlcmFsIG1vZHVsZVxuICAgKiBzcGVjaWZpZXIuIEl0IGV4cGVjdHMgdGhhdCB0aGUgY3VycmVudCBzdGF0ZSBvZiB0aGUgc2Nhbm5lciBpcyB0aGF0IGFuIGBpbXBvcnRgIHRva2VuIGhhcyBqdXN0XG4gICAqIGJlZW4gc2Nhbm5lZC5cbiAgICpcbiAgICogVGhlIGZvbGxvd2luZyBmb3JtcyBvZiBpbXBvcnQgYXJlIG1hdGNoZWQ6XG4gICAqXG4gICAqICogYGltcG9ydCBcIm1vZHVsZS1zcGVjaWZpZXJcIjtgXG4gICAqICogYGltcG9ydChcIm1vZHVsZS1zcGVjaWZpZXJcIilgXG4gICAqICogYGltcG9ydCBkZWZhdWx0QmluZGluZyBmcm9tIFwibW9kdWxlLXNwZWNpZmllclwiO2BcbiAgICogKiBgaW1wb3J0IGRlZmF1bHRCaW5kaW5nLCAqIGFzIGlkZW50aWZpZXIgZnJvbSBcIm1vZHVsZS1zcGVjaWZpZXJcIjtgXG4gICAqICogYGltcG9ydCBkZWZhdWx0QmluZGluZywgey4uLn0gZnJvbSBcIm1vZHVsZS1zcGVjaWZpZXJcIjtgXG4gICAqICogYGltcG9ydCAqIGFzIGlkZW50aWZpZXIgZnJvbSBcIm1vZHVsZS1zcGVjaWZpZXJcIjtgXG4gICAqICogYGltcG9ydCB7Li4ufSBmcm9tIFwibW9kdWxlLXNwZWNpZmllclwiO2BcbiAgICpcbiAgICogQHJldHVybnMgdGhlIGltcG9ydCBwYXRoIG9yIG51bGwgaWYgdGhlcmUgaXMgbm8gaW1wb3J0IG9yIGl0IGlzIG5vdCBhIHN0cmluZyBsaXRlcmFsLlxuICAgKi9cbiAgcHJvdGVjdGVkIGV4dHJhY3RJbXBvcnRQYXRoKCk6IHN0cmluZ3xudWxsIHtcbiAgICAvLyBDaGVjayBmb3Igc2lkZS1lZmZlY3QgaW1wb3J0XG4gICAgbGV0IHNpZGVFZmZlY3RJbXBvcnRQYXRoID0gdGhpcy50cnlTdHJpbmdMaXRlcmFsKCk7XG4gICAgaWYgKHNpZGVFZmZlY3RJbXBvcnRQYXRoICE9PSBudWxsKSB7XG4gICAgICByZXR1cm4gc2lkZUVmZmVjdEltcG9ydFBhdGg7XG4gICAgfVxuXG4gICAgbGV0IGtpbmQ6IHRzLlN5bnRheEtpbmR8bnVsbCA9IHRoaXMuc2Nhbm5lci5nZXRUb2tlbigpO1xuXG4gICAgLy8gQ2hlY2sgZm9yIGR5bmFtaWMgaW1wb3J0IGV4cHJlc3Npb25cbiAgICBpZiAoa2luZCA9PT0gdHMuU3ludGF4S2luZC5PcGVuUGFyZW5Ub2tlbikge1xuICAgICAgcmV0dXJuIHRoaXMuc2NhbkltcG9ydEV4cHJlc3Npb25zID8gdGhpcy50cnlTdHJpbmdMaXRlcmFsKCkgOiBudWxsO1xuICAgIH1cblxuICAgIC8vIENoZWNrIGZvciBkZWZhdWx0QmluZGluZ1xuICAgIGlmIChraW5kID09PSB0cy5TeW50YXhLaW5kLklkZW50aWZpZXIpIHtcbiAgICAgIC8vIFNraXAgZGVmYXVsdCBiaW5kaW5nXG4gICAgICBraW5kID0gdGhpcy5zY2FubmVyLnNjYW4oKTtcbiAgICAgIGlmIChraW5kID09PSB0cy5TeW50YXhLaW5kLkNvbW1hVG9rZW4pIHtcbiAgICAgICAgLy8gU2tpcCBjb21tYSB0aGF0IGluZGljYXRlcyBhZGRpdGlvbmFsIGltcG9ydCBiaW5kaW5nc1xuICAgICAgICBraW5kID0gdGhpcy5zY2FubmVyLnNjYW4oKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBDaGVjayBmb3IgbmFtZXNwYWNlIGltcG9ydCBjbGF1c2VcbiAgICBpZiAoa2luZCA9PT0gdHMuU3ludGF4S2luZC5Bc3Rlcmlza1Rva2VuKSB7XG4gICAgICBraW5kID0gdGhpcy5za2lwTmFtZXNwYWNlZENsYXVzZSgpO1xuICAgICAgaWYgKGtpbmQgPT09IG51bGwpIHtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICB9XG4gICAgfVxuICAgIC8vIENoZWNrIGZvciBuYW1lZCBpbXBvcnRzIGNsYXVzZVxuICAgIGVsc2UgaWYgKGtpbmQgPT09IHRzLlN5bnRheEtpbmQuT3BlbkJyYWNlVG9rZW4pIHtcbiAgICAgIGtpbmQgPSB0aGlzLnNraXBOYW1lZENsYXVzZSgpO1xuICAgIH1cblxuICAgIC8vIEV4cGVjdCBhIGBmcm9tYCBjbGF1c2UsIGlmIG5vdCBiYWlsIG91dFxuICAgIGlmIChraW5kICE9PSB0cy5TeW50YXhLaW5kLkZyb21LZXl3b3JkKSB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG5cbiAgICByZXR1cm4gdGhpcy50cnlTdHJpbmdMaXRlcmFsKCk7XG4gIH1cblxuICAvKipcbiAgICogV2UgaGF2ZSBmb3VuZCBhbiBgZXhwb3J0YCB0b2tlbiBzbyBub3cgdHJ5IHRvIGlkZW50aWZ5IGEgcmUtZXhwb3J0IHBhdGguXG4gICAqXG4gICAqIFRoaXMgbWV0aG9kIHdpbGwgdXNlIHRoZSBjdXJyZW50IHN0YXRlIG9mIGB0aGlzLnNjYW5uZXJgIHRvIGV4dHJhY3QgYSBzdHJpbmcgbGl0ZXJhbCBtb2R1bGVcbiAgICogc3BlY2lmaWVyLiBJdCBleHBlY3RzIHRoYXQgdGhlIGN1cnJlbnQgc3RhdGUgb2YgdGhlIHNjYW5uZXIgaXMgdGhhdCBhbiBgZXhwb3J0YCB0b2tlbiBoYXNcbiAgICoganVzdCBiZWVuIHNjYW5uZWQuXG4gICAqXG4gICAqIFRoZXJlIGFyZSB0aHJlZSBmb3JtcyBvZiByZS1leHBvcnQgdGhhdCBhcmUgbWF0Y2hlZDpcbiAgICpcbiAgICogKiBgZXhwb3J0ICogZnJvbSAnLi4uJztcbiAgICogKiBgZXhwb3J0ICogYXMgYWxpYXMgZnJvbSAnLi4uJztcbiAgICogKiBgZXhwb3J0IHsuLi59IGZyb20gJy4uLic7XG4gICAqL1xuICBwcm90ZWN0ZWQgZXh0cmFjdFJlZXhwb3J0UGF0aCgpOiBzdHJpbmd8bnVsbCB7XG4gICAgLy8gU2tpcCB0aGUgYGV4cG9ydGAga2V5d29yZFxuICAgIGxldCB0b2tlbjogdHMuU3ludGF4S2luZHxudWxsID0gdGhpcy5zY2FubmVyLnNjYW4oKTtcbiAgICBpZiAodG9rZW4gPT09IHRzLlN5bnRheEtpbmQuQXN0ZXJpc2tUb2tlbikge1xuICAgICAgdG9rZW4gPSB0aGlzLnNraXBOYW1lc3BhY2VkQ2xhdXNlKCk7XG4gICAgICBpZiAodG9rZW4gPT09IG51bGwpIHtcbiAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICB9XG4gICAgfSBlbHNlIGlmICh0b2tlbiA9PT0gdHMuU3ludGF4S2luZC5PcGVuQnJhY2VUb2tlbikge1xuICAgICAgdG9rZW4gPSB0aGlzLnNraXBOYW1lZENsYXVzZSgpO1xuICAgIH1cbiAgICAvLyBFeHBlY3QgYSBgZnJvbWAgY2xhdXNlLCBpZiBub3QgYmFpbCBvdXRcbiAgICBpZiAodG9rZW4gIT09IHRzLlN5bnRheEtpbmQuRnJvbUtleXdvcmQpIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy50cnlTdHJpbmdMaXRlcmFsKCk7XG4gIH1cblxuICBwcm90ZWN0ZWQgc2tpcE5hbWVzcGFjZWRDbGF1c2UoKTogdHMuU3ludGF4S2luZHxudWxsIHtcbiAgICAvLyBTa2lwIHBhc3QgdGhlIGAqYFxuICAgIGxldCB0b2tlbiA9IHRoaXMuc2Nhbm5lci5zY2FuKCk7XG4gICAgLy8gQ2hlY2sgZm9yIGEgYCogYXMgaWRlbnRpZmllcmAgYWxpYXMgY2xhdXNlXG4gICAgaWYgKHRva2VuID09PSB0cy5TeW50YXhLaW5kLkFzS2V5d29yZCkge1xuICAgICAgLy8gU2tpcCBwYXN0IHRoZSBgYXNgIGtleXdvcmRcbiAgICAgIHRva2VuID0gdGhpcy5zY2FubmVyLnNjYW4oKTtcbiAgICAgIC8vIEV4cGVjdCBhbiBpZGVudGlmaWVyLCBpZiBub3QgYmFpbCBvdXRcbiAgICAgIGlmICh0b2tlbiAhPT0gdHMuU3ludGF4S2luZC5JZGVudGlmaWVyKSB7XG4gICAgICAgIHJldHVybiBudWxsO1xuICAgICAgfVxuICAgICAgLy8gU2tpcCBwYXN0IHRoZSBpZGVudGlmaWVyXG4gICAgICB0b2tlbiA9IHRoaXMuc2Nhbm5lci5zY2FuKCk7XG4gICAgfVxuICAgIHJldHVybiB0b2tlbjtcbiAgfVxuXG4gIHByb3RlY3RlZCBza2lwTmFtZWRDbGF1c2UoKTogdHMuU3ludGF4S2luZCB7XG4gICAgbGV0IGJyYWNlQ291bnQgPSAxO1xuICAgIC8vIFNraXAgcGFzdCB0aGUgaW5pdGlhbCBvcGVuaW5nIGJyYWNlIGB7YFxuICAgIGxldCB0b2tlbiA9IHRoaXMuc2Nhbm5lci5zY2FuKCk7XG4gICAgLy8gU2VhcmNoIGZvciB0aGUgbWF0Y2hpbmcgY2xvc2luZyBicmFjZSBgfWBcbiAgICB3aGlsZSAoYnJhY2VDb3VudCA+IDAgJiYgdG9rZW4gIT09IHRzLlN5bnRheEtpbmQuRW5kT2ZGaWxlVG9rZW4pIHtcbiAgICAgIGlmICh0b2tlbiA9PT0gdHMuU3ludGF4S2luZC5PcGVuQnJhY2VUb2tlbikge1xuICAgICAgICBicmFjZUNvdW50Kys7XG4gICAgICB9IGVsc2UgaWYgKHRva2VuID09PSB0cy5TeW50YXhLaW5kLkNsb3NlQnJhY2VUb2tlbikge1xuICAgICAgICBicmFjZUNvdW50LS07XG4gICAgICB9XG4gICAgICB0b2tlbiA9IHRoaXMuc2Nhbm5lci5zY2FuKCk7XG4gICAgfVxuICAgIHJldHVybiB0b2tlbjtcbiAgfVxuXG4gIHByb3RlY3RlZCB0cnlTdHJpbmdMaXRlcmFsKCk6IHN0cmluZ3xudWxsIHtcbiAgICByZXR1cm4gdGhpcy5zY2FubmVyLnNjYW4oKSA9PT0gdHMuU3ludGF4S2luZC5TdHJpbmdMaXRlcmFsID8gdGhpcy5zY2FubmVyLmdldFRva2VuVmFsdWUoKSA6XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG51bGw7XG4gIH1cbn1cblxuLyoqXG4gKiBDaGVjayB3aGV0aGVyIGEgc291cmNlIGZpbGUgbmVlZHMgdG8gYmUgcGFyc2VkIGZvciBpbXBvcnRzLlxuICogVGhpcyBpcyBhIHBlcmZvcm1hbmNlIHNob3J0LWNpcmN1aXQsIHdoaWNoIHNhdmVzIHVzIGZyb20gY3JlYXRpbmdcbiAqIGEgVHlwZVNjcmlwdCBBU1QgdW5uZWNlc3NhcmlseS5cbiAqXG4gKiBAcGFyYW0gc291cmNlIFRoZSBjb250ZW50IG9mIHRoZSBzb3VyY2UgZmlsZSB0byBjaGVjay5cbiAqXG4gKiBAcmV0dXJucyBmYWxzZSBpZiB0aGVyZSBhcmUgZGVmaW5pdGVseSBubyBpbXBvcnQgb3IgcmUtZXhwb3J0IHN0YXRlbWVudHNcbiAqIGluIHRoaXMgZmlsZSwgdHJ1ZSBvdGhlcndpc2UuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBoYXNJbXBvcnRPclJlZXhwb3J0U3RhdGVtZW50cyhzb3VyY2U6IHN0cmluZyk6IGJvb2xlYW4ge1xuICByZXR1cm4gLyg/OmltcG9ydHxleHBvcnQpW1xcc1xcU10rPyhbXCInXSkoPzpcXFxcXFwxfC4pKz9cXDEvLnRlc3Qoc291cmNlKTtcbn1cblxuZnVuY3Rpb24gZmluZExhc3RQb3NzaWJsZUltcG9ydE9yUmVleHBvcnQoc291cmNlOiBzdHJpbmcpOiBudW1iZXIge1xuICByZXR1cm4gTWF0aC5tYXgoc291cmNlLmxhc3RJbmRleE9mKCdpbXBvcnQnKSwgc291cmNlLmxhc3RJbmRleE9mKCcgZnJvbSAnKSk7XG59XG5cbi8qKlxuICogQ2hlY2sgd2hldGhlciB0aGUgZ2l2ZW4gc3RhdGVtZW50IGlzIGFuIGltcG9ydCB3aXRoIGEgc3RyaW5nIGxpdGVyYWwgbW9kdWxlIHNwZWNpZmllci5cbiAqIEBwYXJhbSBzdG10IHRoZSBzdGF0ZW1lbnQgbm9kZSB0byBjaGVjay5cbiAqIEByZXR1cm5zIHRydWUgaWYgdGhlIHN0YXRlbWVudCBpcyBhbiBpbXBvcnQgd2l0aCBhIHN0cmluZyBsaXRlcmFsIG1vZHVsZSBzcGVjaWZpZXIuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpc1N0cmluZ0ltcG9ydE9yUmVleHBvcnQoc3RtdDogdHMuU3RhdGVtZW50KTogc3RtdCBpcyB0cy5JbXBvcnREZWNsYXJhdGlvbiZcbiAgICB7bW9kdWxlU3BlY2lmaWVyOiB0cy5TdHJpbmdMaXRlcmFsfSB7XG4gIHJldHVybiB0cy5pc0ltcG9ydERlY2xhcmF0aW9uKHN0bXQpIHx8XG4gICAgICB0cy5pc0V4cG9ydERlY2xhcmF0aW9uKHN0bXQpICYmICEhc3RtdC5tb2R1bGVTcGVjaWZpZXIgJiZcbiAgICAgIHRzLmlzU3RyaW5nTGl0ZXJhbChzdG10Lm1vZHVsZVNwZWNpZmllcik7XG59XG5cblxuZnVuY3Rpb24gY2FuUHJlY2VkZUFSZWdleChraW5kOiB0cy5TeW50YXhLaW5kKTogYm9vbGVhbiB7XG4gIHN3aXRjaCAoa2luZCkge1xuICAgIGNhc2UgdHMuU3ludGF4S2luZC5JZGVudGlmaWVyOlxuICAgIGNhc2UgdHMuU3ludGF4S2luZC5TdHJpbmdMaXRlcmFsOlxuICAgIGNhc2UgdHMuU3ludGF4S2luZC5OdW1lcmljTGl0ZXJhbDpcbiAgICBjYXNlIHRzLlN5bnRheEtpbmQuQmlnSW50TGl0ZXJhbDpcbiAgICBjYXNlIHRzLlN5bnRheEtpbmQuUmVndWxhckV4cHJlc3Npb25MaXRlcmFsOlxuICAgIGNhc2UgdHMuU3ludGF4S2luZC5UaGlzS2V5d29yZDpcbiAgICBjYXNlIHRzLlN5bnRheEtpbmQuUGx1c1BsdXNUb2tlbjpcbiAgICBjYXNlIHRzLlN5bnRheEtpbmQuTWludXNNaW51c1Rva2VuOlxuICAgIGNhc2UgdHMuU3ludGF4S2luZC5DbG9zZVBhcmVuVG9rZW46XG4gICAgY2FzZSB0cy5TeW50YXhLaW5kLkNsb3NlQnJhY2tldFRva2VuOlxuICAgIGNhc2UgdHMuU3ludGF4S2luZC5DbG9zZUJyYWNlVG9rZW46XG4gICAgY2FzZSB0cy5TeW50YXhLaW5kLlRydWVLZXl3b3JkOlxuICAgIGNhc2UgdHMuU3ludGF4S2luZC5GYWxzZUtleXdvcmQ6XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgZGVmYXVsdDpcbiAgICAgIHJldHVybiB0cnVlO1xuICB9XG59XG4iXX0=
Note: See TracBrowser for help on using the repository browser.