ts2famix 2.0.3 → 2.1.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (190) hide show
  1. package/.eslintrc.json +24 -24
  2. package/.vscode/settings.json +4 -0
  3. package/LICENSE +24 -24
  4. package/README.md +78 -78
  5. package/TODO +1 -0
  6. package/arwea-fix.json +1 -0
  7. package/bogus.ts +3 -0
  8. package/class-diagram.puml +792 -0
  9. package/debug.txt +13332 -0
  10. package/debuglog.txt +12073 -0
  11. package/dist/analyze.js +2 -2
  12. package/dist/analyze_functions/process_functions.js +84 -64
  13. package/dist/famix2puml.js +126 -0
  14. package/dist/famix_functions/EntityDictionary.js +865 -503
  15. package/dist/famix_functions/helpers_creation.js +1 -1
  16. package/dist/fqn.js +44 -11
  17. package/dist/getClasses-arrow-body.js +43 -0
  18. package/dist/lib/famix/famix_JSON_exporter.js +1 -1
  19. package/dist/lib/famix/famix_base_element.js +1 -1
  20. package/dist/lib/famix/famix_repository.js +57 -2
  21. package/dist/lib/famix/index.js +1 -1
  22. package/dist/lib/famix/model/famix/access.js +1 -1
  23. package/dist/lib/famix/model/famix/accessor.js +1 -1
  24. package/dist/lib/famix/model/famix/alias.js +1 -1
  25. package/dist/lib/famix/model/famix/arrow_function.js +1 -1
  26. package/dist/lib/famix/model/famix/behavioral_entity.js +1 -1
  27. package/dist/lib/famix/model/famix/class.js +1 -1
  28. package/dist/lib/famix/model/famix/comment.js +1 -1
  29. package/dist/lib/famix/model/famix/concretisation.js +1 -1
  30. package/dist/lib/famix/model/famix/container_entity.js +1 -1
  31. package/dist/lib/famix/model/famix/decorator.js +1 -1
  32. package/dist/lib/famix/model/famix/entity.js +1 -1
  33. package/dist/lib/famix/model/famix/enum.js +1 -1
  34. package/dist/lib/famix/model/famix/enum_value.js +1 -1
  35. package/dist/lib/famix/model/famix/function.js +1 -1
  36. package/dist/lib/famix/model/famix/import_clause.js +1 -1
  37. package/dist/lib/famix/model/famix/index.js +1 -1
  38. package/dist/lib/famix/model/famix/indexed_file_anchor.js +1 -1
  39. package/dist/lib/famix/model/famix/inheritance.js +1 -1
  40. package/dist/lib/famix/model/famix/interface.js +1 -1
  41. package/dist/lib/famix/model/famix/invocation.js +1 -1
  42. package/dist/lib/famix/model/famix/method.js +1 -1
  43. package/dist/lib/famix/model/famix/module.js +1 -1
  44. package/dist/lib/famix/model/famix/named_entity.js +1 -1
  45. package/dist/lib/famix/model/famix/parameter.js +1 -1
  46. package/dist/lib/famix/model/famix/parameter_concretisation.js +1 -1
  47. package/dist/lib/famix/model/famix/parameter_type.js +1 -1
  48. package/dist/lib/famix/model/famix/parametric_arrow_function.js +1 -1
  49. package/dist/lib/famix/model/famix/parametric_class.js +1 -1
  50. package/dist/lib/famix/model/famix/parametric_function.js +1 -1
  51. package/dist/lib/famix/model/famix/parametric_interface.js +1 -1
  52. package/dist/lib/famix/model/famix/parametric_method.js +1 -1
  53. package/dist/lib/famix/model/famix/primitive_type.js +1 -1
  54. package/dist/lib/famix/model/famix/property.js +1 -1
  55. package/dist/lib/famix/model/famix/reference.js +1 -1
  56. package/dist/lib/famix/model/famix/scoping_entity.js +1 -1
  57. package/dist/lib/famix/model/famix/script_entity.js +1 -1
  58. package/dist/lib/famix/model/famix/source_anchor.js +1 -1
  59. package/dist/lib/famix/model/famix/source_language.js +1 -1
  60. package/dist/lib/famix/model/famix/sourced_entity.js +1 -1
  61. package/dist/lib/famix/model/famix/structural_entity.js +1 -1
  62. package/dist/lib/famix/model/famix/type.js +1 -1
  63. package/dist/lib/famix/model/famix/variable.js +1 -1
  64. package/dist/lib/famix/src/famix_JSON_exporter.js +55 -0
  65. package/dist/lib/famix/src/famix_base_element.js +18 -0
  66. package/dist/lib/famix/src/famix_repository.js +224 -0
  67. package/dist/lib/famix/src/index.js +31 -0
  68. package/dist/lib/famix/src/model/famix/access.js +40 -0
  69. package/dist/lib/famix/src/model/famix/accessor.js +17 -0
  70. package/dist/lib/famix/src/model/famix/alias.js +33 -0
  71. package/dist/lib/famix/src/model/famix/arrowFunction.js +17 -0
  72. package/dist/lib/famix/src/model/famix/arrow_function.js +17 -0
  73. package/dist/lib/famix/src/model/famix/behavioral_entity.js +79 -0
  74. package/dist/lib/famix/src/model/famix/class.js +71 -0
  75. package/dist/lib/famix/src/model/famix/comment.js +39 -0
  76. package/dist/lib/famix/src/model/famix/concretisation.js +31 -0
  77. package/dist/lib/famix/src/model/famix/container_entity.js +126 -0
  78. package/dist/lib/famix/src/model/famix/decorator.js +32 -0
  79. package/dist/lib/famix/src/model/famix/entity.js +17 -0
  80. package/dist/lib/famix/src/model/famix/enum.js +31 -0
  81. package/dist/lib/famix/src/model/famix/enum_value.js +25 -0
  82. package/dist/lib/famix/src/model/famix/function.js +17 -0
  83. package/dist/lib/famix/src/model/famix/implicit_variable.js +17 -0
  84. package/dist/lib/famix/src/model/famix/import_clause.js +41 -0
  85. package/dist/lib/famix/src/model/famix/index.js +86 -0
  86. package/dist/lib/famix/src/model/famix/indexed_file_anchor.js +38 -0
  87. package/dist/lib/famix/src/model/famix/inheritance.js +33 -0
  88. package/dist/lib/famix/src/model/famix/interface.js +64 -0
  89. package/dist/lib/famix/src/model/famix/invocation.js +54 -0
  90. package/dist/lib/famix/src/model/famix/method.js +67 -0
  91. package/dist/lib/famix/src/model/famix/module.js +60 -0
  92. package/dist/lib/famix/src/model/famix/named_entity.js +78 -0
  93. package/dist/lib/famix/src/model/famix/parameter.js +25 -0
  94. package/dist/lib/famix/src/model/famix/parameterConcretisation.js +44 -0
  95. package/dist/lib/famix/src/model/famix/parameter_concretisation.js +44 -0
  96. package/dist/lib/famix/src/model/famix/parameter_type.js +45 -0
  97. package/dist/lib/famix/src/model/famix/parametricArrowFunction.js +29 -0
  98. package/dist/lib/famix/src/model/famix/parametric_arrow_function.js +31 -0
  99. package/dist/lib/famix/src/model/famix/parametric_class.js +44 -0
  100. package/dist/lib/famix/src/model/famix/parametric_function.js +31 -0
  101. package/dist/lib/famix/src/model/famix/parametric_interface.js +44 -0
  102. package/dist/lib/famix/src/model/famix/parametric_method.js +31 -0
  103. package/dist/lib/famix/src/model/famix/primitive_type.js +17 -0
  104. package/dist/lib/famix/src/model/famix/property.js +73 -0
  105. package/dist/lib/famix/src/model/famix/reference.js +33 -0
  106. package/dist/lib/famix/src/model/famix/scoping_entity.js +36 -0
  107. package/dist/lib/famix/src/model/famix/script_entity.js +29 -0
  108. package/dist/lib/famix/src/model/famix/source_anchor.js +27 -0
  109. package/dist/lib/famix/src/model/famix/source_language.js +35 -0
  110. package/dist/lib/famix/src/model/famix/sourced_entity.js +60 -0
  111. package/dist/lib/famix/src/model/famix/structural_entity.js +39 -0
  112. package/dist/lib/famix/src/model/famix/text_anchor.js +38 -0
  113. package/dist/lib/famix/src/model/famix/type.js +73 -0
  114. package/dist/lib/famix/src/model/famix/variable.js +24 -0
  115. package/dist/lib/ts-complex/cyclomatic-service.js +1 -1
  116. package/dist/refactorer/refactor-getter-setter.js +1 -1
  117. package/dist/ts2famix-cli-wrapper.js +1 -1
  118. package/dist/ts2famix-cli.js +1 -1
  119. package/doc-uml/famix-typescript-model.puml +607 -607
  120. package/eslint.config.mjs +28 -0
  121. package/fqn-model.json +1 -0
  122. package/iterateGenericTypes.ts +69 -0
  123. package/out/class-diagram/class-diagram.svg +1 -0
  124. package/package.json +70 -66
  125. package/sample.json +1 -0
  126. package/sample.ts +1 -0
  127. package/src/analyze.ts +120 -120
  128. package/src/analyze_functions/process_functions.ts +1040 -1019
  129. package/src/famix_functions/EntityDictionary.ts +2016 -1593
  130. package/src/famix_functions/helpers_creation.ts +135 -135
  131. package/src/fqn.ts +50 -16
  132. package/src/generate_uml.sh +20 -20
  133. package/src/lib/famix/License.md +22 -22
  134. package/src/lib/famix/famix_JSON_exporter.ts +56 -56
  135. package/src/lib/famix/famix_base_element.ts +22 -22
  136. package/src/lib/famix/famix_repository.ts +278 -243
  137. package/src/lib/famix/index.ts +8 -8
  138. package/src/lib/famix/model/famix/access.ts +50 -50
  139. package/src/lib/famix/model/famix/accessor.ts +15 -15
  140. package/src/lib/famix/model/famix/alias.ts +39 -39
  141. package/src/lib/famix/model/famix/arrow_function.ts +15 -15
  142. package/src/lib/famix/model/famix/behavioral_entity.ts +97 -97
  143. package/src/lib/famix/model/famix/class.ts +85 -85
  144. package/src/lib/famix/model/famix/comment.ts +47 -47
  145. package/src/lib/famix/model/famix/concretisation.ts +40 -40
  146. package/src/lib/famix/model/famix/container_entity.ts +160 -160
  147. package/src/lib/famix/model/famix/decorator.ts +37 -37
  148. package/src/lib/famix/model/famix/entity.ts +15 -15
  149. package/src/lib/famix/model/famix/enum.ts +30 -30
  150. package/src/lib/famix/model/famix/enum_value.ts +28 -28
  151. package/src/lib/famix/model/famix/function.ts +15 -15
  152. package/src/lib/famix/model/famix/import_clause.ts +51 -51
  153. package/src/lib/famix/model/famix/index.ts +41 -41
  154. package/src/lib/famix/model/famix/indexed_file_anchor.ts +46 -46
  155. package/src/lib/famix/model/famix/inheritance.ts +40 -40
  156. package/src/lib/famix/model/famix/interface.ts +75 -75
  157. package/src/lib/famix/model/famix/invocation.ts +65 -65
  158. package/src/lib/famix/model/famix/method.ts +89 -89
  159. package/src/lib/famix/model/famix/module.ts +71 -71
  160. package/src/lib/famix/model/famix/named_entity.ts +95 -95
  161. package/src/lib/famix/model/famix/parameter.ts +28 -28
  162. package/src/lib/famix/model/famix/parameter_concretisation.ts +51 -51
  163. package/src/lib/famix/model/famix/parameter_type.ts +58 -58
  164. package/src/lib/famix/model/famix/parametric_arrow_function.ts +32 -32
  165. package/src/lib/famix/model/famix/parametric_class.ts +49 -49
  166. package/src/lib/famix/model/famix/parametric_function.ts +32 -32
  167. package/src/lib/famix/model/famix/parametric_interface.ts +49 -49
  168. package/src/lib/famix/model/famix/parametric_method.ts +32 -32
  169. package/src/lib/famix/model/famix/primitive_type.ts +15 -15
  170. package/src/lib/famix/model/famix/property.ts +94 -94
  171. package/src/lib/famix/model/famix/reference.ts +40 -40
  172. package/src/lib/famix/model/famix/scoping_entity.ts +35 -35
  173. package/src/lib/famix/model/famix/script_entity.ts +34 -34
  174. package/src/lib/famix/model/famix/source_anchor.ts +30 -30
  175. package/src/lib/famix/model/famix/source_language.ts +35 -35
  176. package/src/lib/famix/model/famix/sourced_entity.ts +70 -70
  177. package/src/lib/famix/model/famix/structural_entity.ts +43 -43
  178. package/src/lib/famix/model/famix/type.ts +87 -87
  179. package/src/lib/famix/model/famix/variable.ts +27 -27
  180. package/src/lib/famix/package.json +28 -28
  181. package/src/lib/ts-complex/cyclomatic-service.ts +83 -83
  182. package/src/refactorer/refactor-getter-setter.ts +140 -140
  183. package/src/ts2famix-cli-wrapper.ts +21 -21
  184. package/src/ts2famix-cli.ts +60 -60
  185. package/stats.txt +3091 -0
  186. package/tabby-debug-output.txt +19433 -0
  187. package/ts2famix.log +22656 -0
  188. package/tsconfig.check-tests.json +14 -14
  189. package/tsconfig.json +72 -72
  190. package/validate-references.js +103 -0
@@ -1,4 +1,9 @@
1
1
  "use strict";
2
+ /**
3
+ * a function getOrCreateXType takes arguments name: string and element: ts-morph-type and returns a Famix.Type
4
+ * The goal is to keep track of the types (e.g., a method's definedType), for the model.
5
+ * The name doesn't need to be fully qualified (it's the name used in the source code, or the Famix model).
6
+ */
2
7
  var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
8
  if (k2 === undefined) k2 = k;
4
9
  var desc = Object.getOwnPropertyDescriptor(m, k);
@@ -27,16 +32,18 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
27
32
  };
28
33
  Object.defineProperty(exports, "__esModule", { value: true });
29
34
  exports.EntityDictionary = void 0;
35
+ exports.isPrimitiveType = isPrimitiveType;
30
36
  const ts_morph_1 = require("ts-morph");
31
37
  const process_functions_1 = require("../analyze_functions/process_functions");
32
38
  const Famix = __importStar(require("../lib/famix/model/famix"));
33
39
  const famix_repository_1 = require("../lib/famix/famix_repository");
34
40
  const analyze_1 = require("../analyze");
35
- const grapheme_splitter_1 = __importDefault(require("grapheme-splitter"));
41
+ // import GraphemeSplitter from "grapheme-splitter";
42
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
43
+ const GraphemeSplitter = require("grapheme-splitter");
36
44
  const Helpers = __importStar(require("./helpers_creation"));
37
45
  const FQNFunctions = __importStar(require("../fqn"));
38
46
  const path_1 = __importDefault(require("path"));
39
- const lodash_1 = __importDefault(require("lodash"));
40
47
  class EntityDictionary {
41
48
  constructor() {
42
49
  this.famixRep = new famix_repository_1.FamixRepository();
@@ -45,8 +52,15 @@ class EntityDictionary {
45
52
  this.fmxInterfaceMap = new Map(); // Maps the interface names to their Famix model
46
53
  this.fmxModuleMap = new Map(); // Maps the namespace names to their Famix model
47
54
  this.fmxFileMap = new Map(); // Maps the source file names to their Famix model
48
- this.fmxTypeMap = new Map(); // Maps the type names to their Famix model
55
+ this.fmxTypeMap = new Map(); // Maps the types declarations to their Famix model
56
+ this.fmxPrimitiveTypeMap = new Map(); // Maps the primitive type names to their Famix model
49
57
  this.fmxFunctionAndMethodMap = new Map; // Maps the function names to their Famix model
58
+ this.fmxArrowFunctionMap = new Map; // Maps the function names to their Famix model
59
+ this.fmxParameterMap = new Map(); // Maps the parameters to their Famix model
60
+ this.fmxVariableMap = new Map(); // Maps the variables to their Famix model
61
+ this.fmxImportClauseMap = new Map(); // Maps the import clauses to their Famix model
62
+ this.fmxEnumMap = new Map(); // Maps the enum names to their Famix model
63
+ this.fmxInheritanceMap = new Map(); // Maps the inheritance names to their Famix model
50
64
  this.UNKNOWN_VALUE = '(unknown due to parsing error)'; // The value to use when a name is not usable
51
65
  this.fmxElementObjectMap = new Map();
52
66
  this.tsMorphElementObjectMap = new Map();
@@ -76,7 +90,7 @@ class EntityDictionary {
76
90
  * Famix index file anchor.
77
91
  * It depends on code in the 'grapheme-splitter' package in npm.
78
92
  */
79
- const splitter = new grapheme_splitter_1.default();
93
+ const splitter = new GraphemeSplitter();
80
94
  const sourceFileText = node.getSourceFile().getFullText();
81
95
  const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;
82
96
  if (hasGraphemeClusters) {
@@ -94,7 +108,10 @@ class EntityDictionary {
94
108
  // The +1 is because the source anchor (Pharo) is 1-based, but ts-morph is 0-based
95
109
  sourceAnchor.startPos = sourceStart + 1;
96
110
  sourceAnchor.endPos = sourceEnd + 1;
97
- const fileName = node.getSourceFile().getFilePath();
111
+ let fileName = node.getSourceFile().getFilePath();
112
+ if (fileName.startsWith("/")) {
113
+ fileName = fileName.substring(1);
114
+ }
98
115
  sourceAnchor.element = fmx;
99
116
  sourceAnchor.fileName = fileName;
100
117
  fmx.sourceAnchor = sourceAnchor;
@@ -108,9 +125,9 @@ class EntityDictionary {
108
125
  * @param famixElement The Famix model of the source element
109
126
  */
110
127
  makeFamixIndexFileAnchor(sourceElement, famixElement) {
111
- // check if famixElement doesn't have a valid fullyQualifiedName
112
- if (typeof famixElement.getFullyQualifiedName === 'function') {
113
- // The method exists
128
+ // Famix.Comment is not a named entity (does not have a fullyQualifiedName)
129
+ if (!(famixElement instanceof Famix.Comment)) { // must be a named entity
130
+ // insanity check: named entities should have fullyQualifiedName
114
131
  const fullyQualifiedName = famixElement.fullyQualifiedName;
115
132
  if (!fullyQualifiedName || fullyQualifiedName === this.UNKNOWN_VALUE) {
116
133
  throw new Error(`Famix element ${famixElement.constructor.name} has no valid fullyQualifiedName.`);
@@ -134,13 +151,16 @@ class EntityDictionary {
134
151
  }
135
152
  // revert any backslashes to forward slashes (path.normalize on windows introduces them)
136
153
  pathInProject = pathInProject.replace(/\\/g, "/");
154
+ if (pathInProject.startsWith("/")) {
155
+ pathInProject = pathInProject.substring(1);
156
+ }
137
157
  fmxIndexFileAnchor.fileName = pathInProject;
138
- let sourceStart, sourceEnd, sourceLineStart, sourceLineEnd;
158
+ let sourceStart, sourceEnd;
139
159
  if (!(sourceElement instanceof ts_morph_1.CommentRange)) {
140
160
  sourceStart = sourceElement.getStart();
141
161
  sourceEnd = sourceElement.getEnd();
142
- sourceLineStart = sourceElement.getStartLineNumber();
143
- sourceLineEnd = sourceElement.getEndLineNumber();
162
+ // sourceLineStart = sourceElement.getStartLineNumber();
163
+ // sourceLineEnd = sourceElement.getEndLineNumber();
144
164
  }
145
165
  else {
146
166
  sourceStart = sourceElement.getPos();
@@ -157,7 +177,7 @@ class EntityDictionary {
157
177
  * Famix index file anchor.
158
178
  * It depends on code in the 'grapheme-splitter' package in npm.
159
179
  */
160
- const splitter = new grapheme_splitter_1.default();
180
+ const splitter = new GraphemeSplitter();
161
181
  const sourceFileText = sourceElement.getSourceFile().getFullText();
162
182
  const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;
163
183
  if (hasGraphemeClusters) {
@@ -222,28 +242,30 @@ class EntityDictionary {
222
242
  }
223
243
  /**
224
244
  * Creates or gets a Famix Module
225
- * @param m A module
245
+ * @param moduleDeclaration A module
226
246
  * @returns The Famix model of the module
227
247
  */
228
- createOrGetFamixModule(m) {
229
- let fmxModule;
230
- const moduleName = m.getName();
231
- const foundModuleName = this.fmxModuleMap.get(moduleName);
232
- if (!foundModuleName) {
233
- fmxModule = new Famix.Module();
234
- fmxModule.name = moduleName;
235
- fmxModule.isAmbient = (0, process_functions_1.isAmbient)(m);
236
- fmxModule.isNamespace = (0, process_functions_1.isNamespace)(m);
237
- fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;
238
- initFQN(m, fmxModule);
239
- this.makeFamixIndexFileAnchor(m, fmxModule);
240
- this.fmxModuleMap.set(moduleName, fmxModule);
241
- this.famixRep.addElement(fmxModule);
242
- }
243
- else {
244
- fmxModule = foundModuleName;
245
- }
246
- this.fmxElementObjectMap.set(fmxModule, m);
248
+ createOrGetFamixModule(moduleDeclaration) {
249
+ if (this.fmxModuleMap.has(moduleDeclaration)) {
250
+ const rModule = this.fmxModuleMap.get(moduleDeclaration);
251
+ if (rModule) {
252
+ return rModule;
253
+ }
254
+ else {
255
+ throw new Error(`Famix module ${moduleDeclaration.getName()} is not found in the module map.`);
256
+ }
257
+ }
258
+ const fmxModule = new Famix.Module();
259
+ const moduleName = moduleDeclaration.getName();
260
+ fmxModule.name = moduleName;
261
+ fmxModule.isAmbient = (0, process_functions_1.isAmbient)(moduleDeclaration);
262
+ fmxModule.isNamespace = (0, process_functions_1.isNamespace)(moduleDeclaration);
263
+ fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;
264
+ initFQN(moduleDeclaration, fmxModule);
265
+ this.makeFamixIndexFileAnchor(moduleDeclaration, fmxModule);
266
+ this.fmxModuleMap.set(moduleDeclaration, fmxModule);
267
+ this.famixRep.addElement(fmxModule);
268
+ this.fmxElementObjectMap.set(fmxModule, moduleDeclaration);
247
269
  return fmxModule;
248
270
  }
249
271
  /**
@@ -254,7 +276,8 @@ class EntityDictionary {
254
276
  createFamixAlias(a) {
255
277
  let fmxAlias;
256
278
  const aliasName = a.getName();
257
- const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);
279
+ //const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);
280
+ const aliasFullyQualifiedName = FQNFunctions.getFQN(a);
258
281
  const foundAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);
259
282
  if (!foundAlias) {
260
283
  fmxAlias = new Famix.Alias();
@@ -294,7 +317,8 @@ class EntityDictionary {
294
317
  fmxClass = new Famix.Class();
295
318
  }
296
319
  fmxClass.name = clsName;
297
- fmxClass.fullyQualifiedName = classFullyQualifiedName;
320
+ initFQN(cls, fmxClass);
321
+ // fmxClass.fullyQualifiedName = classFullyQualifiedName;
298
322
  fmxClass.isAbstract = isAbstract;
299
323
  this.makeFamixIndexFileAnchor(cls, fmxClass);
300
324
  this.fmxClassMap.set(classFullyQualifiedName, fmxClass);
@@ -343,56 +367,51 @@ class EntityDictionary {
343
367
  * @param concreteArguments concrete arguments
344
368
  * @returns A parametric Element
345
369
  */
346
- createOrGetFamixConcreteElement(concreteElement, concreteElementDeclaration, concreteArguments) {
347
- let fullyQualifiedFilename = concreteElement.fullyQualifiedName;
348
- let params = "";
349
- concreteArguments.map((param) => {
350
- params = params + param.getText() + ',';
351
- });
352
- params = params.substring(0, params.length - 1);
353
- fullyQualifiedFilename = Helpers.replaceLastBetweenTags(fullyQualifiedFilename, params);
354
- let concElement;
355
- if (!this.fmxInterfaceMap.has(fullyQualifiedFilename) &&
356
- !this.fmxClassMap.has(fullyQualifiedFilename) &&
357
- !this.fmxFunctionAndMethodMap.has(fullyQualifiedFilename)) {
358
- concElement = lodash_1.default.cloneDeep(concreteElement);
359
- concElement.fullyQualifiedName = fullyQualifiedFilename;
360
- concElement.clearGenericParameters();
361
- concreteArguments.map((param) => {
362
- const parameter = this.createOrGetFamixConcreteType(param);
363
- concElement.addConcreteParameter(parameter);
364
- });
365
- if (concreteElement instanceof Famix.ParametricClass) {
366
- this.fmxClassMap.set(fullyQualifiedFilename, concElement);
367
- }
368
- else if (concreteElement instanceof Famix.ParametricInterface) {
369
- this.fmxInterfaceMap.set(fullyQualifiedFilename, concElement);
370
- }
371
- else if (concreteElement instanceof Famix.ParametricFunction) {
372
- this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement);
373
- }
374
- else { // if (concreteElement instanceof Famix.ParametricMethod) {
375
- this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement);
376
- }
377
- this.famixRep.addElement(concElement);
378
- this.fmxElementObjectMap.set(concElement, concreteElementDeclaration);
379
- }
380
- else {
381
- if (concreteElement instanceof Famix.ParametricClass) {
382
- concElement = this.fmxClassMap.get(fullyQualifiedFilename);
383
- }
384
- else if (concreteElement instanceof Famix.ParametricInterface) {
385
- concElement = this.fmxInterfaceMap.get(fullyQualifiedFilename);
386
- }
387
- else if (concreteElement instanceof Famix.ParametricFunction) {
388
- concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename);
389
- }
390
- else { // if (concreteElement instanceof Famix.ParametricMethod) {
391
- concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename);
392
- }
393
- }
394
- return concElement;
395
- }
370
+ // public createOrGetFamixConcreteElement(concreteElement : ParametricVariantType,
371
+ // concreteElementDeclaration : ConcreteElementTSMorphType,
372
+ // concreteArguments: TypeNode[]): ParametricVariantType {
373
+ // let fullyQualifiedFilename = concreteElement.fullyQualifiedName;
374
+ // let params = "";
375
+ // concreteArguments.map((param) => {
376
+ // params = params+param.getText()+',';
377
+ // });
378
+ // params = params.substring(0, params.length - 1)
379
+ // fullyQualifiedFilename = Helpers.replaceLastBetweenTags(fullyQualifiedFilename,params);
380
+ // let concElement: ParametricVariantType;
381
+ // if (!this.fmxInterfaceMap.has(fullyQualifiedFilename) &&
382
+ // !this.fmxClassMap.has(fullyQualifiedFilename) &&
383
+ // !this.fmxFunctionAndMethodMap.has(fullyQualifiedFilename)){
384
+ // concElement = _.cloneDeep(concreteElement);
385
+ // concElement.fullyQualifiedName = fullyQualifiedFilename;
386
+ // concElement.clearGenericParameters();
387
+ // concreteArguments.map((param) => {
388
+ // const parameter = this.createOrGetFamixConcreteType(param);
389
+ // concElement.addConcreteParameter(parameter);
390
+ // })
391
+ // if (concreteElement instanceof Famix.ParametricClass) {
392
+ // this.fmxClassMap.set(fullyQualifiedFilename, concElement as Famix.ParametricClass);
393
+ // } else if (concreteElement instanceof Famix.ParametricInterface) {
394
+ // this.fmxInterfaceMap.set(fullyQualifiedFilename, concElement as Famix.ParametricInterface);
395
+ // } else if (concreteElement instanceof Famix.ParametricFunction) {
396
+ // this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricFunction);
397
+ // } else { // if (concreteElement instanceof Famix.ParametricMethod) {
398
+ // this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricMethod);
399
+ // }
400
+ // this.famixRep.addElement(concElement);
401
+ // this.fmxElementObjectMap.set(concElement,concreteElementDeclaration);
402
+ // } else {
403
+ // if (concreteElement instanceof Famix.ParametricClass) {
404
+ // concElement = this.fmxClassMap.get(fullyQualifiedFilename) as Famix.ParametricClass;
405
+ // } else if (concreteElement instanceof Famix.ParametricInterface) {
406
+ // concElement = this.fmxInterfaceMap.get(fullyQualifiedFilename) as Famix.ParametricInterface;
407
+ // } else if (concreteElement instanceof Famix.ParametricFunction) {
408
+ // concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricFunction;
409
+ // } else { // if (concreteElement instanceof Famix.ParametricMethod) {
410
+ // concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricMethod;
411
+ // }
412
+ // }
413
+ // return concElement;
414
+ // }
396
415
  /**
397
416
  * Creates a Famix property
398
417
  * @param property A property
@@ -479,7 +498,6 @@ class EntityDictionary {
479
498
  else {
480
499
  fmxMethod = new Famix.Method();
481
500
  }
482
- this.famixRep.addElement(fmxMethod);
483
501
  }
484
502
  const isConstructor = method instanceof ts_morph_1.ConstructorDeclaration;
485
503
  const isSignature = method instanceof ts_morph_1.MethodSignature;
@@ -541,6 +559,7 @@ class EntityDictionary {
541
559
  fmxMethod.numberOfStatements = 0;
542
560
  }
543
561
  initFQN(method, fmxMethod);
562
+ this.famixRep.addElement(fmxMethod);
544
563
  this.makeFamixIndexFileAnchor(method, fmxMethod);
545
564
  this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxMethod);
546
565
  }
@@ -576,7 +595,8 @@ class EntityDictionary {
576
595
  }
577
596
  fmxFunction.signature = Helpers.computeSignature(func.getText());
578
597
  fmxFunction.cyclomaticComplexity = currentCC[fmxFunction.name];
579
- fmxFunction.fullyQualifiedName = functionFullyQualifiedName;
598
+ initFQN(func, fmxFunction);
599
+ // fmxFunction.fullyQualifiedName = functionFullyQualifiedName;
580
600
  let functionTypeName = this.UNKNOWN_VALUE;
581
601
  try {
582
602
  functionTypeName = func.getReturnType().getText().trim();
@@ -605,7 +625,16 @@ class EntityDictionary {
605
625
  * @param param A parameter
606
626
  * @returns The Famix model of the parameter
607
627
  */
608
- createFamixParameter(param) {
628
+ createOrGetFamixParameter(param) {
629
+ if (this.fmxParameterMap.has(param)) {
630
+ const rParameter = this.fmxParameterMap.get(param);
631
+ if (rParameter) {
632
+ return rParameter;
633
+ }
634
+ else {
635
+ throw new Error(`Famix parameter ${param.getName()} is not found in the parameter map.`);
636
+ }
637
+ }
609
638
  const fmxParam = new Famix.Parameter();
610
639
  let paramTypeName = this.UNKNOWN_VALUE;
611
640
  try {
@@ -621,6 +650,7 @@ class EntityDictionary {
621
650
  this.makeFamixIndexFileAnchor(param, fmxParam);
622
651
  this.famixRep.addElement(fmxParam);
623
652
  this.fmxElementObjectMap.set(fmxParam, param);
653
+ this.fmxParameterMap.set(param, fmxParam);
624
654
  return fmxParam;
625
655
  }
626
656
  /**
@@ -638,25 +668,31 @@ class EntityDictionary {
638
668
  return fmxParameterType;
639
669
  }
640
670
  /**
641
- * Creates a Famix type parameter
642
- * @param tp A type parameter
643
- * @returns The Famix model of the type parameter
671
+ * Creates a Famix type in the context of concretizations
672
+ * @param typeName A type name
673
+ * @param element An element
674
+ * @returns The Famix model of the type
644
675
  */
645
- createOrGetFamixConcreteType(param) {
646
- const typeParameterDeclaration = param.getSymbol()?.getDeclarations()[0];
647
- const parameterTypeName = param.getText();
676
+ createOrGetFamixConcreteType(element) {
677
+ // TODO - refactor to stop using names as a key in the maps, use ts-morph element instead
678
+ const typeParameterDeclaration = element.getSymbol()?.getDeclarations()[0];
679
+ const parameterTypeName = element.getText();
648
680
  let fmxParameterType = undefined;
681
+ // get a TypeReference from a TypeNode
682
+ const typeReference = element.getType();
683
+ // get a TypeDeclaration from a TypeReference
684
+ const typeDeclaration = typeReference.getSymbol()?.getDeclarations()[0];
649
685
  let isClassOrInterface = false;
650
686
  if (this.fmxClassMap.has(parameterTypeName)) {
651
687
  this.fmxClassMap.forEach((obj, name) => {
652
688
  if (obj instanceof Famix.ParametricClass) {
653
- if (name === param.getText() && obj.genericParameters.size > 0) {
689
+ if (name === element.getText() && obj.genericParameters.size > 0) {
654
690
  fmxParameterType = obj;
655
691
  isClassOrInterface = true;
656
692
  }
657
693
  }
658
694
  else {
659
- if (name === param.getText()) {
695
+ if (name === element.getText()) {
660
696
  fmxParameterType = obj;
661
697
  isClassOrInterface = true;
662
698
  }
@@ -666,13 +702,13 @@ class EntityDictionary {
666
702
  if (this.fmxInterfaceMap.has(parameterTypeName)) {
667
703
  this.fmxInterfaceMap.forEach((obj, name) => {
668
704
  if (obj instanceof Famix.ParametricInterface) {
669
- if (name === param.getText() && obj.genericParameters.size > 0) {
705
+ if (name === element.getText() && obj.genericParameters.size > 0) {
670
706
  fmxParameterType = obj;
671
707
  isClassOrInterface = true;
672
708
  }
673
709
  }
674
710
  else {
675
- if (name === param.getText()) {
711
+ if (name === element.getText()) {
676
712
  fmxParameterType = obj;
677
713
  isClassOrInterface = true;
678
714
  }
@@ -680,8 +716,9 @@ class EntityDictionary {
680
716
  });
681
717
  }
682
718
  if (!isClassOrInterface) {
683
- if (!this.fmxTypeMap.has(parameterTypeName)) {
684
- if (parameterTypeName === "number" || parameterTypeName === "string" || parameterTypeName === "boolean" || parameterTypeName === "bigint" || parameterTypeName === "symbol" || parameterTypeName === "undefined" || parameterTypeName === "null" || parameterTypeName === "any" || parameterTypeName === "unknown" || parameterTypeName === "never" || parameterTypeName === "void") {
719
+ if (!this.fmxTypeMap.has(typeDeclaration)) {
720
+ // TODO refactor
721
+ if (isPrimitiveType(parameterTypeName)) {
685
722
  fmxParameterType = new Famix.PrimitiveType();
686
723
  fmxParameterType.isStub = true;
687
724
  }
@@ -690,16 +727,16 @@ class EntityDictionary {
690
727
  }
691
728
  fmxParameterType.name = parameterTypeName;
692
729
  this.famixRep.addElement(fmxParameterType);
693
- this.fmxTypeMap.set(parameterTypeName, fmxParameterType);
730
+ this.fmxTypeMap.set(typeDeclaration, fmxParameterType);
694
731
  this.fmxElementObjectMap.set(fmxParameterType, typeParameterDeclaration);
695
732
  }
696
733
  else {
697
- const result = this.fmxTypeMap.get(parameterTypeName);
734
+ const result = this.fmxTypeMap.get(typeDeclaration);
698
735
  if (result) {
699
736
  fmxParameterType = result;
700
737
  }
701
738
  else {
702
- throw new Error(`Famix type ${parameterTypeName} is not found in the Type map.`);
739
+ throw new Error(`Famix type ${typeDeclaration} is not found in the Type map.`);
703
740
  }
704
741
  }
705
742
  }
@@ -713,7 +750,16 @@ class EntityDictionary {
713
750
  * @param variable A variable
714
751
  * @returns The Famix model of the variable
715
752
  */
716
- createFamixVariable(variable) {
753
+ createOrGetFamixVariable(variable) {
754
+ if (this.fmxVariableMap.has(variable)) {
755
+ const rVariable = this.fmxVariableMap.get(variable);
756
+ if (rVariable) {
757
+ return rVariable;
758
+ }
759
+ else {
760
+ throw new Error(`Famix parameter ${variable.getName()} is not found in the variable map.`);
761
+ }
762
+ }
717
763
  const fmxVariable = new Famix.Variable();
718
764
  let variableTypeName = this.UNKNOWN_VALUE;
719
765
  try {
@@ -729,6 +775,7 @@ class EntityDictionary {
729
775
  this.makeFamixIndexFileAnchor(variable, fmxVariable);
730
776
  this.famixRep.addElement(fmxVariable);
731
777
  this.fmxElementObjectMap.set(fmxVariable, variable);
778
+ this.fmxVariableMap.set(variable, fmxVariable);
732
779
  return fmxVariable;
733
780
  }
734
781
  /**
@@ -736,13 +783,23 @@ class EntityDictionary {
736
783
  * @param enumEntity An enum
737
784
  * @returns The Famix model of the enum
738
785
  */
739
- createFamixEnum(enumEntity) {
786
+ createOrGetFamixEnum(enumEntity) {
787
+ if (this.fmxEnumMap.has(enumEntity)) {
788
+ const rEnum = this.fmxEnumMap.get(enumEntity);
789
+ if (rEnum) {
790
+ return rEnum;
791
+ }
792
+ else {
793
+ throw new Error(`Famix enum ${enumEntity.getName()} is not found in the enum map.`);
794
+ }
795
+ }
740
796
  const fmxEnum = new Famix.Enum();
741
797
  fmxEnum.name = enumEntity.getName();
742
798
  initFQN(enumEntity, fmxEnum);
743
799
  this.makeFamixIndexFileAnchor(enumEntity, fmxEnum);
744
800
  this.famixRep.addElement(fmxEnum);
745
801
  this.fmxElementObjectMap.set(fmxEnum, enumEntity);
802
+ this.fmxEnumMap.set(enumEntity, fmxEnum);
746
803
  return fmxEnum;
747
804
  }
748
805
  /**
@@ -814,47 +871,38 @@ class EntityDictionary {
814
871
  */
815
872
  createOrGetFamixType(typeName, element) {
816
873
  let fmxType;
817
- let isPrimitiveType = false;
818
- let isParameterType = false;
874
+ const isPrimitive = isPrimitiveType(typeName);
875
+ const isParametricType = element instanceof ts_morph_1.ClassDeclaration && element.getTypeParameters().length > 0 ||
876
+ element instanceof ts_morph_1.InterfaceDeclaration && element.getTypeParameters().length > 0;
877
+ // Functions and methods aren't types!
878
+ // ||
879
+ // element instanceof FunctionDeclaration && element.getTypeParameters().length > 0 ||
880
+ // element instanceof MethodDeclaration && element.getTypeParameters().length > 0 ||
881
+ // element instanceof ArrowFunction && element.getTypeParameters().length > 0;
819
882
  analyze_1.logger.debug("Creating (or getting) type: '" + typeName + "' of element: " + element?.getText() + " of kind: " + element?.getKindName());
820
- let ancestor = undefined;
821
- if (element !== undefined) {
822
- const typeAncestor = Helpers.findTypeAncestor(element);
823
- if (!typeAncestor) {
824
- throw new Error(`Ancestor not found for element ${element.getText()}.`);
825
- }
826
- const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);
827
- ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
828
- if (!ancestor) {
829
- analyze_1.logger.debug(`Ancestor ${FQNFunctions.getFQN(typeAncestor)} not found. Adding the new type.`);
830
- ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor);
831
- }
832
- }
833
- if (typeName === "number" || typeName === "string" || typeName === "boolean" || typeName === "bigint" || typeName === "symbol" || typeName === "undefined" || typeName === "null" || typeName === "any" || typeName === "unknown" || typeName === "never" || typeName === "void") {
834
- isPrimitiveType = true;
835
- }
836
- if (!isPrimitiveType && typeName.includes("<") && typeName.includes(">") && !(typeName.includes("=>"))) {
837
- isParameterType = true;
838
- }
839
- if (!this.fmxTypeMap.has(typeName)) {
840
- if (isPrimitiveType) {
841
- fmxType = new Famix.PrimitiveType();
842
- fmxType.isStub = true;
843
- }
844
- else if (isParameterType) {
845
- fmxType = new Famix.ParameterType();
846
- const parameterTypeNames = typeName.substring(typeName.indexOf("<") + 1, typeName.indexOf(">")).split(",").map(s => s.trim());
847
- const baseTypeName = typeName.substring(0, typeName.indexOf("<")).trim();
848
- parameterTypeNames.forEach(parameterTypeName => {
849
- const fmxParameterType = this.createOrGetFamixType(parameterTypeName, element);
850
- fmxType.addArgument(fmxParameterType);
851
- });
852
- const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);
853
- fmxType.baseType = fmxBaseType;
854
- }
855
- else {
856
- fmxType = new Famix.Type();
883
+ if (isPrimitive) {
884
+ return this.createOrGetFamixPrimitiveType(typeName);
885
+ }
886
+ if (isParametricType) {
887
+ // narrow the type
888
+ const parametricElement = element;
889
+ return this.createOrGetFamixParametricType(typeName, parametricElement);
890
+ }
891
+ if (!this.fmxTypeMap.has(element)) {
892
+ let ancestor = undefined;
893
+ if (element !== undefined) {
894
+ const typeAncestor = Helpers.findTypeAncestor(element);
895
+ if (!typeAncestor) {
896
+ throw new Error(`Ancestor not found for element ${element.getText()}.`);
897
+ }
898
+ const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);
899
+ ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
900
+ if (!ancestor) {
901
+ analyze_1.logger.debug(`Ancestor ${FQNFunctions.getFQN(typeAncestor)} not found. Adding the new type.`);
902
+ ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor);
903
+ }
857
904
  }
905
+ fmxType = new Famix.Type();
858
906
  fmxType.name = typeName;
859
907
  if (!ancestor) {
860
908
  throw new Error(`Ancestor not found for type ${typeName}.`);
@@ -863,10 +911,10 @@ class EntityDictionary {
863
911
  initFQN(element, fmxType);
864
912
  this.makeFamixIndexFileAnchor(element, fmxType);
865
913
  this.famixRep.addElement(fmxType);
866
- this.fmxTypeMap.set(typeName, fmxType);
914
+ this.fmxTypeMap.set(element, fmxType);
867
915
  }
868
916
  else {
869
- const result = this.fmxTypeMap.get(typeName);
917
+ const result = this.fmxTypeMap.get(element);
870
918
  if (result) {
871
919
  fmxType = result;
872
920
  }
@@ -877,6 +925,109 @@ class EntityDictionary {
877
925
  this.fmxElementObjectMap.set(fmxType, element);
878
926
  return fmxType;
879
927
  }
928
+ /**
929
+ * Creates or gets a Famix type that is parametric
930
+ * @param typeName A type name
931
+ * @param element A ts-morph element
932
+ * @returns The Famix model of the parameter type
933
+ */
934
+ createOrGetFamixParametricType(typeName, element) {
935
+ if (this.fmxTypeMap.has(element) === true) {
936
+ const result = this.fmxTypeMap.get(element);
937
+ if (result) {
938
+ return result;
939
+ }
940
+ else {
941
+ throw new Error(`Famix type ${typeName} is not found (undefined) in the Type map.`);
942
+ }
943
+ }
944
+ // A parametric type is a type that has type parameters, e.g., List<T>
945
+ // In TS it can be a class, an interface, a function, an arrow function, or a method
946
+ // create the Famix Parametric Type (maybe it's just an Interface, etc.)
947
+ let fmxType;
948
+ if (element instanceof ts_morph_1.ClassDeclaration) {
949
+ fmxType = new Famix.ParametricClass();
950
+ }
951
+ else if (element instanceof ts_morph_1.InterfaceDeclaration) {
952
+ fmxType = new Famix.ParametricInterface();
953
+ }
954
+ // functions and methods are not types
955
+ // else if (element instanceof FunctionDeclaration) {
956
+ // fmxType = new Famix.ParametricFunction();
957
+ // } else if (element instanceof ArrowFunction) {
958
+ // fmxType = new Famix.ParametricArrowFunction();
959
+ // } else if (element instanceof MethodDeclaration) {
960
+ // fmxType = new Famix.ParametricMethod();
961
+ // }
962
+ else {
963
+ throw new Error(`Element is not a class, interface, function, arrow function, or method.`);
964
+ }
965
+ // const parameters = element.getTypeParameters();
966
+ // // for each parameter, getOrCreate the FamixParameterType
967
+ // for (const parameter of parameters) {
968
+ // this.createOrGetFamixParameterType(parameter.getName(), parameter);
969
+ // }
970
+ // // TODO: the following code is not correct, it is just a placeholder
971
+ // const parameterTypeNames = typeName.substring(typeName.indexOf("<") + 1, typeName.indexOf(">"))
972
+ // .split(",").map(s => s.trim());
973
+ // const baseTypeName = typeName.substring(0, typeName.indexOf("<")).trim();
974
+ // parameterTypeNames.forEach(parameterTypeName => {
975
+ // const fmxParameterType = this.createOrGetFamixParameterType(parameterTypeName, element);
976
+ // (fmxType as Famix.ParameterType).addArgument(fmxParameterType);
977
+ // });
978
+ // const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);
979
+ // (fmxType as Famix.ParameterType).baseType = fmxBaseType;
980
+ fmxType.name = typeName;
981
+ initFQN(element, fmxType);
982
+ this.famixRep.addElement(fmxType);
983
+ this.fmxTypeMap.set(element, fmxType);
984
+ return fmxType;
985
+ }
986
+ /**
987
+ * Creates a type for a parameter in a parametric type, e.g., T in List<T>
988
+ * @param parameterTypeName
989
+ * @param element the TypeScript element (TSMorphParametricType) that the type is associated with
990
+ * @returns
991
+ */
992
+ // createOrGetFamixParameterType(parameterTypeName: string, element: ParameterDeclaration) {
993
+ // if (this.fmxTypeMap.has(element)) {
994
+ // return this.fmxTypeMap.get(element) as Famix.ParameterType;
995
+ // }
996
+ // // determine if element is a
997
+ // const fmxType = new Famix.ParameterType();
998
+ // // const parameterTypeNames = typeName.substring(typeName.indexOf("<") + 1, typeName.indexOf(">"))
999
+ // // .split(",").map(s => s.trim());
1000
+ // // const baseTypeName = typeName.substring(0, typeName.indexOf("<")).trim();
1001
+ // // parameterTypeNames.forEach(parameterTypeName => {
1002
+ // // const fmxParameterType = this.createOrGetFamixParameterType(parameterTypeName, element);
1003
+ // // (fmxType as Famix.ParameterType).addArgument(fmxParameterType);
1004
+ // // });
1005
+ // const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);
1006
+ // (fmxType as Famix.ParameterType).baseType = fmxBaseType;
1007
+ // initFQN(element, fmxType);
1008
+ // this.famixRep.addElement(fmxType);
1009
+ // this.fmxTypeMap.set(element, fmxType);
1010
+ // return fmxType;
1011
+ // }
1012
+ /**
1013
+ * Creates or gets a Famix primitive type
1014
+ * @param typeName A type name
1015
+ * @returns The Famix model of the primitive type
1016
+ */
1017
+ createOrGetFamixPrimitiveType(typeName) {
1018
+ let fmxType = new Famix.PrimitiveType();
1019
+ if (!this.fmxPrimitiveTypeMap.has(typeName)) {
1020
+ fmxType = new Famix.PrimitiveType();
1021
+ fmxType.isStub = true;
1022
+ fmxType.name = typeName;
1023
+ this.fmxPrimitiveTypeMap.set(typeName, fmxType);
1024
+ this.famixRep.addElement(fmxType);
1025
+ }
1026
+ else {
1027
+ fmxType = this.fmxPrimitiveTypeMap.get(typeName);
1028
+ }
1029
+ return fmxType;
1030
+ }
880
1031
  /**
881
1032
  * Creates a Famix access
882
1033
  * @param node A node
@@ -890,10 +1041,25 @@ class EntityDictionary {
890
1041
  analyze_1.logger.debug(`Creating FamixAccess. Node: [${node.getKindName()}] '${node.getText()}' at line ${node.getStartLineNumber()} in ${node.getSourceFile().getBaseName()}, id: ${id} refers to fmxVar '${fmxVar.fullyQualifiedName}'.`);
891
1042
  const nodeReferenceAncestor = Helpers.findAncestor(node);
892
1043
  const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);
893
- let accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
1044
+ const accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
894
1045
  if (!accessor) {
895
1046
  analyze_1.logger.error(`Ancestor ${ancestorFullyQualifiedName} of kind ${nodeReferenceAncestor.getKindName()} not found.`);
896
1047
  // accessor = this.createOrGetFamixType(ancestorFullyQualifiedName, nodeReferenceAncestor as TypeDeclaration);
1048
+ return; // bail out TODO: this is probably wrong
1049
+ }
1050
+ else {
1051
+ analyze_1.logger.debug(`Found accessor to be ${accessor.fullyQualifiedName}.`);
1052
+ }
1053
+ // make sure accessor is a method, function, script or module
1054
+ if (!(accessor instanceof Famix.Method) && !(accessor instanceof Famix.ArrowFunction) && !(accessor instanceof Famix.Function) && !(accessor instanceof Famix.ScriptEntity) && !(accessor instanceof Famix.Module)) {
1055
+ analyze_1.logger.error(`Accessor ${accessor.fullyQualifiedName} is not a method, function, etc.`);
1056
+ return;
1057
+ }
1058
+ // don't create any duplicates (e.g. if the same variable is accessed multiple times by same accessor)
1059
+ const foundAccess = this.famixRep.getFamixAccessByAccessorAndVariable(accessor, fmxVar);
1060
+ if (foundAccess) {
1061
+ analyze_1.logger.debug(`FamixAccess already exists for accessor ${accessor.fullyQualifiedName} and variable ${fmxVar.fullyQualifiedName}.`);
1062
+ return;
897
1063
  }
898
1064
  const fmxAccess = new Famix.Access();
899
1065
  fmxAccess.accessor = accessor;
@@ -903,89 +1069,136 @@ class EntityDictionary {
903
1069
  }
904
1070
  /**
905
1071
  * Creates a Famix invocation
906
- * @param node A node
907
- * @param m A method or a function
1072
+ * @param nodeReferringToInvocable A node
1073
+ * @param invocable A method or a function
908
1074
  * @param id The id of the method or the function
909
1075
  */
910
- createFamixInvocation(node, m, id) {
911
- const fmxMethodOrFunction = this.famixRep.getFamixEntityById(id);
912
- const nodeReferenceAncestor = Helpers.findAncestor(node);
913
- const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);
914
- const sender = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
915
- const receiverFullyQualifiedName = FQNFunctions.getFQN(m.getParent());
1076
+ createFamixInvocation(nodeReferringToInvocable, invocable, id) {
1077
+ const fmxInvocable = this.famixRep.getFamixEntityById(id);
1078
+ // since the node is in the AST, we need to find the ancestor that is in the Famix model
1079
+ const containerOfNode = Helpers.findAncestor(nodeReferringToInvocable);
1080
+ analyze_1.logger.debug(`Found container (ancestor) ${containerOfNode.getKindName()} for AST node ${nodeReferringToInvocable.getText()}.`);
1081
+ const containerFQN = FQNFunctions.getFQN(containerOfNode);
1082
+ analyze_1.logger.debug(`Found containerFQN ${containerFQN}.`);
1083
+ let sender = this.famixRep.getFamixEntityByFullyQualifiedName(containerFQN);
1084
+ analyze_1.logger.debug(`Found a sender that matches ${sender.fullyQualifiedName}.`);
1085
+ if (sender instanceof Famix.Type) {
1086
+ // TODO this might be an error in getFamixEntityByFullyQualifiedName
1087
+ analyze_1.logger.debug(`Oops! Sender is a type, which is not valid for an Invocation. Trying to find a container for ${sender.fullyQualifiedName}.`);
1088
+ const senderContainer = sender.container;
1089
+ if (senderContainer) {
1090
+ sender = senderContainer;
1091
+ }
1092
+ }
1093
+ const receiverFullyQualifiedName = FQNFunctions.getFQN(invocable.getParent());
916
1094
  const receiver = this.famixRep.getFamixEntityByFullyQualifiedName(receiverFullyQualifiedName);
917
1095
  const fmxInvocation = new Famix.Invocation();
918
1096
  fmxInvocation.sender = sender;
919
1097
  fmxInvocation.receiver = receiver;
920
- fmxInvocation.addCandidate(fmxMethodOrFunction);
921
- fmxInvocation.signature = fmxMethodOrFunction.signature;
1098
+ fmxInvocation.addCandidate(fmxInvocable);
1099
+ fmxInvocation.signature = fmxInvocable.signature;
922
1100
  this.famixRep.addElement(fmxInvocation);
923
- this.fmxElementObjectMap.set(fmxInvocation, node);
1101
+ this.fmxElementObjectMap.set(fmxInvocation, nodeReferringToInvocable);
924
1102
  }
925
1103
  /**
926
1104
  * Creates a Famix inheritance
927
1105
  * @param cls A class or an interface (subclass)
928
1106
  * @param inhClass The inherited class or interface (superclass)
929
1107
  */
930
- createFamixInheritance(cls, inhClass) {
1108
+ createOrGetFamixInheritance(cls, inhClass) {
1109
+ // // need a key to see if the inheritance already exists
1110
+ // const classFullyQualifiedName = FQNFunctions.getFQN(cls);
1111
+ // let inKeyword: string;
1112
+ // let inhClassFullyQualifiedName: string;
1113
+ // let inhClassName: string | undefined;
1114
+ // // if inhClass is an ExpressionWithTypeArguments, it is an interface
1115
+ // if (inhClass instanceof ExpressionWithTypeArguments) {
1116
+ // inhClassName = inhClass.getExpression().getText();
1117
+ // // what is inhClassFullyQualifiedName? TODO
1118
+ // inhClassFullyQualifiedName = 'Undefined_Scope_from_importer.' + inhClassName;
1119
+ // } else {
1120
+ // inhClassName = inhClass.getName();
1121
+ // if (!inhClassName) {
1122
+ // throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);
1123
+ // }
1124
+ // inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);
1125
+ // }
1126
+ // // build the unique key of the inheritance
1127
+ // if ((cls instanceof ClassDeclaration && inhClass instanceof ClassDeclaration)
1128
+ // || (cls instanceof InterfaceDeclaration && inhClass instanceof InterfaceDeclaration)) {
1129
+ // inKeyword = " extends ";
1130
+ // } else if (cls instanceof ClassDeclaration && (inhClass instanceof InterfaceDeclaration || inhClass instanceof ExpressionWithTypeArguments)) {
1131
+ // inKeyword = " implements ";
1132
+ // } else {
1133
+ // throw new Error(`Inheritance ${cls.getText()} and ${inhClass.getText()} is not valid.`);
1134
+ // }
1135
+ // const inheritanceFullyQualifiedName = FQNFunctions.getFQN(cls) + inKeyword + inhClassFullyQualifiedName;
1136
+ // if (this.fmxInheritanceMap.has(inheritanceFullyQualifiedName)) {
1137
+ // const rInheritance = this.fmxInheritanceMap.get(inheritanceFullyQualifiedName);
1138
+ // if (rInheritance) {
1139
+ // return; // don't do anything
1140
+ // } else {
1141
+ // throw new Error(`Inheritance ${cls.getText()} is not found in the inheritance map.`);
1142
+ // }
1143
+ // }
931
1144
  const fmxInheritance = new Famix.Inheritance();
932
- // const clsName = cls.getName();
933
- const classFullyQualifiedName = FQNFunctions.getFQN(cls);
934
- analyze_1.logger.debug(`createFamixInheritance: classFullyQualifiedName: class fqn = ${classFullyQualifiedName}`);
1145
+ // logger.debug(`createFamixInheritance: classFullyQualifiedName: class fqn = ${classFullyQualifiedName}`);
935
1146
  let subClass;
936
1147
  if (cls instanceof ts_morph_1.ClassDeclaration) {
937
- subClass = this.fmxClassMap.get(classFullyQualifiedName);
1148
+ subClass = this.createOrGetFamixClass(cls);
938
1149
  }
939
1150
  else {
940
- subClass = this.fmxInterfaceMap.get(classFullyQualifiedName);
1151
+ subClass = this.createOrGetFamixInterface(cls);
941
1152
  }
942
1153
  if (!subClass) {
943
- throw new Error(`Subclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
1154
+ throw new Error(`Subclass ${cls} not found in Class or Interface maps.`);
944
1155
  }
945
- let inhClassName;
946
- let inhClassFullyQualifiedName;
947
1156
  let superClass;
948
- if (inhClass instanceof ts_morph_1.ClassDeclaration || inhClass instanceof ts_morph_1.InterfaceDeclaration) {
949
- inhClassName = inhClass.getName();
950
- if (!inhClassName) {
951
- throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);
952
- }
953
- inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);
954
- if (inhClass instanceof ts_morph_1.ClassDeclaration) {
955
- superClass = this.fmxClassMap.get(inhClassFullyQualifiedName);
956
- }
957
- else {
958
- superClass = this.fmxInterfaceMap.get(inhClassFullyQualifiedName);
959
- }
960
- if (!superClass) {
961
- throw new Error(`Superclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
962
- }
1157
+ // if (inhClass instanceof ClassDeclaration || inhClass instanceof InterfaceDeclaration) {
1158
+ // inhClassName = inhClass.getName();
1159
+ // if (!inhClassName) {
1160
+ // throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);
1161
+ // }
1162
+ // inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);
1163
+ // if (inhClass instanceof ClassDeclaration) {
1164
+ // superClass = this.fmxClassMap.get(inhClassFullyQualifiedName);
1165
+ // }
1166
+ // else {
1167
+ // superClass = this.fmxInterfaceMap.get(inhClassFullyQualifiedName);
1168
+ // }
1169
+ // if (!superClass) {
1170
+ // throw new Error(`Superclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
1171
+ // }
1172
+ // }
1173
+ // // it shouldn't add the class/interface again to the Map, it should use createOrGet (?)
1174
+ // if (superClass === undefined) {
1175
+ if (inhClass instanceof ts_morph_1.ClassDeclaration) {
1176
+ // superClass = new Famix.Class();
1177
+ superClass = this.createOrGetFamixClass(inhClass);
1178
+ // this.fmxClassMap.set(inhClassFullyQualifiedName, superClass);
1179
+ }
1180
+ else if (inhClass instanceof ts_morph_1.InterfaceDeclaration) {
1181
+ // superClass = new Famix.Interface();
1182
+ superClass = this.createOrGetFamixInterface(inhClass);
1183
+ // this.fmxInterfaceMap.set(inhClassFullyQualifiedName, superClass);
963
1184
  }
964
1185
  else {
965
- // inhClass is an ExpressionWithTypeArguments
966
- inhClassName = inhClass.getExpression().getText();
967
- // what is inhClassFullyQualifiedName? TODO
968
- inhClassFullyQualifiedName = 'Undefined_Scope_from_importer.' + inhClassName;
969
- }
970
- if (superClass === undefined) {
971
- if (inhClass instanceof ts_morph_1.ClassDeclaration) {
972
- superClass = new Famix.Class();
973
- this.fmxClassMap.set(inhClassFullyQualifiedName, superClass);
1186
+ // inhClass instanceof ExpressionWithTypeArguments
1187
+ const interfaceDeclaration = getInterfaceDeclarationFromExpression(inhClass);
1188
+ if (interfaceDeclaration !== undefined) {
1189
+ superClass = this.createOrGetFamixInterface(interfaceDeclaration);
974
1190
  }
975
1191
  else {
976
- superClass = new Famix.Interface();
977
- this.fmxInterfaceMap.set(inhClassFullyQualifiedName, superClass);
1192
+ throw new Error(`Interface declaration not found for ${inhClass.getText()}.`);
978
1193
  }
979
- this.fmxElementObjectMap.set(superClass, inhClass);
980
- superClass.name = inhClassName;
981
- superClass.fullyQualifiedName = inhClassFullyQualifiedName;
982
- superClass.isStub = true;
983
- this.makeFamixIndexFileAnchor(inhClass, superClass);
984
- this.famixRep.addElement(superClass);
985
1194
  }
1195
+ this.fmxElementObjectMap.set(superClass, inhClass);
1196
+ this.makeFamixIndexFileAnchor(inhClass, superClass);
1197
+ this.famixRep.addElement(superClass);
986
1198
  fmxInheritance.subclass = subClass;
987
1199
  fmxInheritance.superclass = superClass;
988
1200
  this.famixRep.addElement(fmxInheritance);
1201
+ // no FQN for inheritance
989
1202
  // We don't map inheritance to the source code element because there are two elements (super, sub)
990
1203
  // this.fmxElementObjectMap.set(fmxInheritance, null);
991
1204
  }
@@ -1006,17 +1219,30 @@ class EntityDictionary {
1006
1219
  * @param isInExports A boolean indicating if the imported entity is in the exports
1007
1220
  * @param isDefaultExport A boolean indicating if the imported entity is a default export
1008
1221
  */
1009
- oldCreateFamixImportClause(importClauseInfo) {
1222
+ oldCreateOrGetFamixImportClause(importClauseInfo) {
1010
1223
  const { importDeclaration, importerSourceFile: importer, moduleSpecifierFilePath, importElement, isInExports, isDefaultExport } = importClauseInfo;
1011
- analyze_1.logger.debug(`createFamixImportClause: Creating import clause:`);
1224
+ if (importDeclaration && this.fmxImportClauseMap.has(importDeclaration)) {
1225
+ const rImportClause = this.fmxImportClauseMap.get(importDeclaration);
1226
+ if (rImportClause) {
1227
+ return; // don't do anything
1228
+ }
1229
+ else {
1230
+ throw new Error(`Import clause ${importElement.getText()} is not found in the import clause map.`);
1231
+ }
1232
+ }
1233
+ analyze_1.logger.info(`creating a new FamixImportClause for ${importDeclaration?.getText()} in ${importer.getBaseName()}.`);
1012
1234
  const fmxImportClause = new Famix.ImportClause();
1013
1235
  let importedEntity = undefined;
1014
1236
  let importedEntityName;
1015
1237
  const absolutePathProject = this.famixRep.getAbsolutePath();
1016
1238
  const absolutePath = path_1.default.normalize(moduleSpecifierFilePath);
1017
1239
  // convert the path and remove any windows backslashes introduced by path.normalize
1240
+ analyze_1.logger.debug(`createFamixImportClause: absolutePath: ${absolutePath}`);
1241
+ analyze_1.logger.debug(`createFamixImportClause: convertToRelativePath: ${this.convertToRelativePath(absolutePath, absolutePathProject)}`);
1018
1242
  const pathInProject = this.convertToRelativePath(absolutePath, absolutePathProject).replace(/\\/g, "/");
1243
+ analyze_1.logger.debug(`createFamixImportClause: pathInProject: ${pathInProject}`);
1019
1244
  let pathName = "{" + pathInProject + "}.";
1245
+ analyze_1.logger.debug(`createFamixImportClause: pathName: ${pathName}`);
1020
1246
  // Named imports, e.g. import { ClassW } from "./complexExportModule";
1021
1247
  // Start with simple import clause (without referring to the actual variable)
1022
1248
  if (importDeclaration instanceof ts_morph_1.ImportDeclaration
@@ -1032,7 +1258,9 @@ class EntityDictionary {
1032
1258
  if (!isInExports) {
1033
1259
  importedEntity.isStub = true;
1034
1260
  }
1035
- importedEntity.fullyQualifiedName = pathName;
1261
+ // logger.debug(`createFamixImportClause: Creating named entity ${importedEntityName} with FQN ${pathName}`);
1262
+ // importedEntity.fullyQualifiedName = pathName;
1263
+ initFQN(importElement, importedEntity);
1036
1264
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1037
1265
  // must add entity to repository
1038
1266
  this.famixRep.addElement(importedEntity);
@@ -1047,7 +1275,7 @@ class EntityDictionary {
1047
1275
  importedEntity.name = importedEntityName;
1048
1276
  initFQN(importDeclaration, importedEntity);
1049
1277
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1050
- importedEntity.fullyQualifiedName = pathName;
1278
+ // importedEntity.fullyQualifiedName = pathName;
1051
1279
  const anyType = this.createOrGetFamixType('any', importDeclaration);
1052
1280
  importedEntity.declaredType = anyType;
1053
1281
  }
@@ -1056,7 +1284,8 @@ class EntityDictionary {
1056
1284
  pathName = pathName + (isDefaultExport ? "defaultExport" : "namespaceExport");
1057
1285
  importedEntity = new Famix.NamedEntity();
1058
1286
  importedEntity.name = importedEntityName;
1059
- importedEntity.fullyQualifiedName = pathName;
1287
+ // importedEntity.fullyQualifiedName = pathName;
1288
+ initFQN(importElement, importedEntity);
1060
1289
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1061
1290
  }
1062
1291
  // I don't think it should be added to the repository if it exists already
@@ -1075,59 +1304,68 @@ class EntityDictionary {
1075
1304
  analyze_1.logger.debug(`createFamixImportClause: ${fmxImportClause.importedEntity?.name} (of type ${Helpers.getSubTypeName(fmxImportClause.importedEntity)}) is imported by ${fmxImportClause.importingEntity?.name}`);
1076
1305
  fmxImporter.addOutgoingImport(fmxImportClause);
1077
1306
  this.famixRep.addElement(fmxImportClause);
1078
- if (importDeclaration)
1307
+ if (importDeclaration) {
1079
1308
  this.fmxElementObjectMap.set(fmxImportClause, importDeclaration);
1309
+ this.fmxImportClauseMap.set(importDeclaration, fmxImportClause);
1310
+ }
1080
1311
  }
1081
1312
  /**
1082
1313
  * Creates a Famix Arrow Function
1083
1314
  * @param arrowExpression An Expression
1084
1315
  * @returns The Famix model of the variable
1085
1316
  */
1086
- createFamixArrowFunction(arrowExpression, currentCC) {
1317
+ createOrGetFamixArrowFunction(arrowExpression, currentCC) {
1087
1318
  let fmxArrowFunction;
1088
- const arrowFunction = arrowExpression.asKindOrThrow(ts_morph_1.SyntaxKind.ArrowFunction);
1089
- const isGeneric = arrowFunction.getTypeParameters().length > 0;
1090
- if (isGeneric) {
1091
- fmxArrowFunction = new Famix.ParametricArrowFunction();
1092
- }
1093
- else {
1094
- fmxArrowFunction = new Famix.ArrowFunction();
1095
- }
1096
- // Get the parent of the arrow function (the variable declaration)
1097
- const parent = arrowFunction.getParentIfKind(ts_morph_1.SyntaxKind.VariableDeclaration);
1098
- let functionName = '(NO_NAME)';
1099
- if (parent && parent instanceof ts_morph_1.VariableDeclaration) {
1100
- // Get the name of the variable
1101
- functionName = parent.getName();
1102
- }
1103
- if (functionName) {
1104
- fmxArrowFunction.name = functionName;
1319
+ const functionFullyQualifiedName = FQNFunctions.getFQN(arrowExpression);
1320
+ if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {
1321
+ const arrowFunction = arrowExpression.asKindOrThrow(ts_morph_1.SyntaxKind.ArrowFunction);
1322
+ const isGeneric = arrowFunction.getTypeParameters().length > 0;
1323
+ if (isGeneric) {
1324
+ fmxArrowFunction = new Famix.ParametricArrowFunction();
1325
+ }
1326
+ else {
1327
+ fmxArrowFunction = new Famix.ArrowFunction();
1328
+ }
1329
+ // Get the parent of the arrow function (the variable declaration)
1330
+ const parent = arrowFunction.getParentIfKind(ts_morph_1.SyntaxKind.VariableDeclaration);
1331
+ let functionName = '(NO_NAME)';
1332
+ if (parent && parent instanceof ts_morph_1.VariableDeclaration) {
1333
+ // Get the name of the variable
1334
+ functionName = parent.getName();
1335
+ }
1336
+ if (functionName) {
1337
+ fmxArrowFunction.name = functionName;
1338
+ }
1339
+ else {
1340
+ fmxArrowFunction.name = "anonymous";
1341
+ }
1342
+ // Signature of an arrow function is (parameters) => return_type
1343
+ const parametersSignature = arrowFunction.getParameters().map(p => p.getText()).join(", ");
1344
+ const returnTypeSignature = arrowFunction.getReturnType().getText();
1345
+ fmxArrowFunction.signature = `(${parametersSignature}) => ${returnTypeSignature}`;
1346
+ fmxArrowFunction.cyclomaticComplexity = currentCC[fmxArrowFunction.name];
1347
+ let functionTypeName = this.UNKNOWN_VALUE;
1348
+ try {
1349
+ functionTypeName = arrowFunction.getReturnType().getText().trim();
1350
+ }
1351
+ catch (error) {
1352
+ analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${functionName}. Continuing...`);
1353
+ }
1354
+ const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction);
1355
+ fmxArrowFunction.declaredType = fmxType;
1356
+ fmxArrowFunction.numberOfLinesOfCode = arrowFunction.getEndLineNumber() - arrowFunction.getStartLineNumber();
1357
+ const parameters = arrowFunction.getParameters();
1358
+ fmxArrowFunction.numberOfParameters = parameters.length;
1359
+ fmxArrowFunction.numberOfStatements = arrowFunction.getStatements().length;
1360
+ initFQN(arrowExpression, fmxArrowFunction);
1361
+ this.makeFamixIndexFileAnchor(arrowExpression, fmxArrowFunction);
1362
+ this.famixRep.addElement(fmxArrowFunction);
1363
+ this.fmxElementObjectMap.set(fmxArrowFunction, arrowFunction);
1364
+ this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxArrowFunction);
1105
1365
  }
1106
1366
  else {
1107
- fmxArrowFunction.name = "anonymous";
1108
- }
1109
- // Signature of an arrow function is (parameters) => return_type
1110
- const parametersSignature = arrowFunction.getParameters().map(p => p.getText()).join(", ");
1111
- const returnTypeSignature = arrowFunction.getReturnType().getText();
1112
- fmxArrowFunction.signature = `(${parametersSignature}) => ${returnTypeSignature}`;
1113
- fmxArrowFunction.cyclomaticComplexity = currentCC[fmxArrowFunction.name];
1114
- let functionTypeName = this.UNKNOWN_VALUE;
1115
- try {
1116
- functionTypeName = arrowFunction.getReturnType().getText().trim();
1367
+ fmxArrowFunction = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName);
1117
1368
  }
1118
- catch (error) {
1119
- analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${functionName}. Continuing...`);
1120
- }
1121
- const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction);
1122
- fmxArrowFunction.declaredType = fmxType;
1123
- fmxArrowFunction.numberOfLinesOfCode = arrowFunction.getEndLineNumber() - arrowFunction.getStartLineNumber();
1124
- const parameters = arrowFunction.getParameters();
1125
- fmxArrowFunction.numberOfParameters = parameters.length;
1126
- fmxArrowFunction.numberOfStatements = arrowFunction.getStatements().length;
1127
- initFQN(arrowExpression, fmxArrowFunction);
1128
- this.makeFamixIndexFileAnchor(arrowExpression, fmxArrowFunction);
1129
- this.famixRep.addElement(fmxArrowFunction);
1130
- this.fmxElementObjectMap.set(fmxArrowFunction, arrowFunction);
1131
1369
  return fmxArrowFunction;
1132
1370
  }
1133
1371
  /**
@@ -1135,297 +1373,421 @@ class EntityDictionary {
1135
1373
  * @param cls A class
1136
1374
  * @returns The Famix model of the concretisation
1137
1375
  */
1138
- createFamixConcretisation(conEntity, genEntity) {
1139
- const fmxConcretisation = new Famix.Concretisation();
1140
- fmxConcretisation.concreteEntity = conEntity;
1141
- fmxConcretisation.genericEntity = genEntity;
1142
- // this.fmxElementObjectMap.set(fmxConcretisation,null);
1143
- this.famixRep.addElement(fmxConcretisation);
1144
- const parameterConcretisation = this.createFamixParameterConcretisation(fmxConcretisation);
1145
- return fmxConcretisation;
1146
- }
1376
+ // public createFamixConcretisation(conEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod ,genEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod): Famix.Concretisation {
1377
+ // const fmxConcretisation : Famix.Concretisation = new Famix.Concretisation();
1378
+ // fmxConcretisation.concreteEntity = conEntity;
1379
+ // fmxConcretisation.genericEntity = genEntity;
1380
+ // // this.fmxElementObjectMap.set(fmxConcretisation,null);
1381
+ // this.famixRep.addElement(fmxConcretisation);
1382
+ // const parameterConcretisation = this.createFamixParameterConcretisation(fmxConcretisation);
1383
+ // return fmxConcretisation;
1384
+ // }
1147
1385
  /**
1148
1386
  * Creates a Famix concretisation
1149
1387
  * @param concretisation A FamixConcretisation
1150
1388
  * @returns The Famix model of the ParameterConcrestisation
1151
1389
  */
1152
- createFamixParameterConcretisation(concretisation) {
1153
- const conClass = concretisation.concreteEntity;
1154
- const genClass = concretisation.genericEntity;
1155
- analyze_1.logger.debug(`Creating parameter concretisation between ${conClass.fullyQualifiedName} and ${genClass.fullyQualifiedName}`);
1156
- const parameterConcretisations = this.famixRep._getAllEntitiesWithType("ParameterConcretisation");
1157
- const concreteParameters = conClass.concreteParameters;
1158
- const genericParameters = genClass.genericParameters;
1159
- let conClassTypeParametersIterator = concreteParameters.values();
1160
- let genClassTypeParametersIterator = genericParameters.values();
1161
- let fmxParameterConcretisation = undefined;
1162
- for (let i = 0; i < genericParameters.size; i++) {
1163
- const conClassTypeParameter = conClassTypeParametersIterator.next().value;
1164
- const genClassTypeParameter = genClassTypeParametersIterator.next().value;
1165
- let createParameterConcretisation = true;
1166
- if (conClassTypeParameter && genClassTypeParameter && conClassTypeParameter.name != genClassTypeParameter.name) {
1167
- parameterConcretisations.forEach((param) => {
1168
- if (conClassTypeParameter.name == param.concreteParameter.name && genClassTypeParameter.name == param.genericParameter.name) {
1169
- createParameterConcretisation = false;
1170
- fmxParameterConcretisation = param;
1171
- }
1172
- });
1173
- if (createParameterConcretisation) {
1174
- fmxParameterConcretisation = new Famix.ParameterConcretisation();
1175
- fmxParameterConcretisation.genericParameter = genClassTypeParameter;
1176
- fmxParameterConcretisation.concreteParameter = conClassTypeParameter;
1177
- fmxParameterConcretisation.addConcretisation(concretisation);
1178
- // this.fmxElementObjectMap.set(fmxParameterConcretisation,null);
1179
- }
1180
- else {
1181
- if (!fmxParameterConcretisation) {
1182
- throw new Error(`fmxParameterConcretisation was undefined for concretisation with generic parameter ${genClassTypeParameter.name} and concrete parameter ${conClassTypeParameter.name}`);
1183
- }
1184
- fmxParameterConcretisation.addConcretisation(concretisation);
1185
- }
1186
- this.famixRep.addElement(fmxParameterConcretisation);
1187
- }
1188
- }
1189
- if (!fmxParameterConcretisation) {
1190
- analyze_1.logger.error(`fmxParameterConcretisation was undefined for concretisation with concrete entity ${conClass.fullyQualifiedName} and generic entity ${genClass.fullyQualifiedName}`);
1191
- }
1192
- return fmxParameterConcretisation;
1193
- }
1390
+ // public createFamixParameterConcretisation(concretisation: Famix.Concretisation): Famix.ParameterConcretisation | undefined{
1391
+ // const conClass = concretisation.concreteEntity;
1392
+ // const genClass = concretisation.genericEntity;
1393
+ // logger.debug(`Creating parameter concretisation between ${conClass.fullyQualifiedName} and ${genClass.fullyQualifiedName}`);
1394
+ // const parameterConcretisations = this.famixRep._getAllEntitiesWithType("ParameterConcretisation") as Set<Famix.ParameterConcretisation>;
1395
+ // const concreteParameters = conClass.concreteParameters;
1396
+ // const genericParameters = genClass.genericParameters;
1397
+ // let conClassTypeParametersIterator = concreteParameters.values();
1398
+ // let genClassTypeParametersIterator = genericParameters.values();
1399
+ // let fmxParameterConcretisation : Famix.ParameterConcretisation | undefined = undefined;
1400
+ // for (let i = 0; i < genericParameters.size; i++) {
1401
+ // const conClassTypeParameter = conClassTypeParametersIterator.next().value as Famix.ParameterType;
1402
+ // const genClassTypeParameter = genClassTypeParametersIterator.next().value as Famix.ParameterType;
1403
+ // let createParameterConcretisation : boolean = true;
1404
+ // if(conClassTypeParameter && genClassTypeParameter && conClassTypeParameter.name != genClassTypeParameter.name){
1405
+ // parameterConcretisations.forEach((param : Famix.ParameterConcretisation) => {
1406
+ // if (conClassTypeParameter.name == param.concreteParameter.name && genClassTypeParameter.name == param.genericParameter.name) {
1407
+ // createParameterConcretisation = false;
1408
+ // fmxParameterConcretisation = param;
1409
+ // }
1410
+ // })
1411
+ // if (createParameterConcretisation) {
1412
+ // fmxParameterConcretisation = new Famix.ParameterConcretisation();
1413
+ // fmxParameterConcretisation.genericParameter = genClassTypeParameter;
1414
+ // fmxParameterConcretisation.concreteParameter = conClassTypeParameter;
1415
+ // fmxParameterConcretisation.addConcretisation(concretisation);
1416
+ // // this.fmxElementObjectMap.set(fmxParameterConcretisation,null);
1417
+ // } else {
1418
+ // if (!fmxParameterConcretisation) {
1419
+ // throw new Error(`fmxParameterConcretisation was undefined for concretisation with generic parameter ${genClassTypeParameter.name} and concrete parameter ${conClassTypeParameter.name}`);
1420
+ // }
1421
+ // fmxParameterConcretisation.addConcretisation(concretisation);
1422
+ // }
1423
+ // this.famixRep.addElement(fmxParameterConcretisation);
1424
+ // }
1425
+ // }
1426
+ // if (!fmxParameterConcretisation) {
1427
+ // logger.error(`fmxParameterConcretisation was undefined for concretisation with concrete entity ${conClass.fullyQualifiedName} and generic entity ${genClass.fullyQualifiedName}`);
1428
+ // }
1429
+ // return fmxParameterConcretisation;
1430
+ // }
1194
1431
  /**
1195
1432
  * Creates a Famix concretisation between two classes or two interfaces
1196
1433
  * @param element A class or an Interface
1197
1434
  */
1198
- createFamixConcretisationClassOrInterfaceSpecialisation(element) {
1199
- const superEntity = element.getExtends();
1200
- let superEntityArray;
1201
- if (superEntity) {
1202
- superEntityArray = Array.isArray(superEntity) ? superEntity : [superEntity];
1203
- }
1204
- if (superEntityArray && superEntityArray.length > 0) {
1205
- superEntityArray.forEach(entity => {
1206
- let entityIsGeneric;
1207
- const superEntitySymbol = entity.getExpression().getSymbolOrThrow();
1208
- let superEntityDeclaration;
1209
- if (superEntity instanceof ts_morph_1.ExpressionWithTypeArguments) {
1210
- superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts_morph_1.ts.SyntaxKind.ClassDeclaration);
1211
- }
1212
- else {
1213
- superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts_morph_1.ts.SyntaxKind.InterfaceDeclaration);
1214
- }
1215
- if (superEntityDeclaration) {
1216
- entityIsGeneric = superEntityDeclaration.getTypeParameters().length > 0;
1217
- }
1218
- if (entityIsGeneric) {
1219
- let EntityDeclaration;
1220
- let genEntity;
1221
- if (superEntity instanceof ts_morph_1.ExpressionWithTypeArguments) {
1222
- EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0];
1223
- genEntity = this.createOrGetFamixClass(EntityDeclaration);
1224
- }
1225
- else {
1226
- EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0];
1227
- genEntity = this.createOrGetFamixInterface(EntityDeclaration);
1228
- }
1229
- const genParams = EntityDeclaration.getTypeParameters().map((param) => param.getText());
1230
- const args = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1231
- const conParams = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1232
- if (!Helpers.arraysAreEqual(conParams, genParams)) {
1233
- let conEntity;
1234
- conEntity = this.createOrGetFamixConcreteElement(genEntity, EntityDeclaration, args);
1235
- const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1236
- let createConcretisation = true;
1237
- concretisations.forEach((conc) => {
1238
- if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName) {
1239
- createConcretisation = false;
1240
- }
1241
- });
1242
- if (createConcretisation) {
1243
- const fmxConcretisation = this.createFamixConcretisation(conEntity, genEntity);
1244
- }
1245
- }
1246
- }
1247
- });
1248
- }
1249
- // TODO: This function seems unfinished
1250
- }
1435
+ // public createFamixConcretisationClassOrInterfaceSpecialisation(element: ClassDeclaration | InterfaceDeclaration){
1436
+ // const superEntity = element.getExtends();
1437
+ // let superEntityArray;
1438
+ // if (superEntity){
1439
+ // superEntityArray = Array.isArray(superEntity) ? superEntity : [superEntity];
1440
+ // }
1441
+ // if (superEntityArray && superEntityArray.length > 0) {
1442
+ // superEntityArray.forEach(entity => {
1443
+ // let entityIsGeneric;
1444
+ // const superEntitySymbol = entity.getExpression().getSymbolOrThrow();
1445
+ // let superEntityDeclaration;
1446
+ // if (superEntity instanceof ExpressionWithTypeArguments) {
1447
+ // superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.ClassDeclaration);
1448
+ // } else {
1449
+ // superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.InterfaceDeclaration);
1450
+ // }
1451
+ // if (superEntityDeclaration) {
1452
+ // entityIsGeneric = superEntityDeclaration.getTypeParameters().length > 0;
1453
+ // }
1454
+ // if (entityIsGeneric) {
1455
+ // let EntityDeclaration;
1456
+ // let genEntity;
1457
+ // if (superEntity instanceof ExpressionWithTypeArguments) {
1458
+ // EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as ClassDeclaration;
1459
+ // genEntity = this.createOrGetFamixClass(EntityDeclaration) as Famix.ParametricClass;
1460
+ // } else {
1461
+ // EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;
1462
+ // genEntity = this.createOrGetFamixInterface(EntityDeclaration) as Famix.ParametricInterface;
1463
+ // }
1464
+ // const genParams = EntityDeclaration.getTypeParameters().map((param) => param.getText());
1465
+ // const args = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments()
1466
+ // const conParams = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1467
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1468
+ // let conEntity;
1469
+ // conEntity = this.createOrGetFamixConcreteElement(genEntity,EntityDeclaration,args);
1470
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1471
+ // let createConcretisation : boolean = true;
1472
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1473
+ // if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){
1474
+ // createConcretisation = false;
1475
+ // }
1476
+ // });
1477
+ // if (createConcretisation) {
1478
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);
1479
+ // }
1480
+ // }
1481
+ // }
1482
+ // });
1483
+ // }
1484
+ // // TODO: This function seems unfinished
1485
+ // }
1251
1486
  /**
1252
1487
  * Creates a Famix concretisation between a class and its instanciations
1253
1488
  * @param cls A class
1254
1489
  */
1255
- createFamixConcretisationGenericInstantiation(cls) {
1256
- const isGeneric = cls.getTypeParameters().length > 0;
1257
- if (isGeneric) {
1258
- const instances = cls.getSourceFile().getDescendantsOfKind(ts_morph_1.ts.SyntaxKind.NewExpression)
1259
- .filter(newExpr => {
1260
- const expression = newExpr.getExpression();
1261
- return expression.getText() === cls.getName();
1262
- });
1263
- instances.forEach(instance => {
1264
- const instanceIsGeneric = instance.getTypeArguments().length > 0;
1265
- if (instanceIsGeneric) {
1266
- const conParams = instance.getTypeArguments().map((param) => param.getText());
1267
- const genEntity = this.createOrGetFamixClass(cls);
1268
- const genParams = cls.getTypeParameters().map((param) => param.getText());
1269
- if (!Helpers.arraysAreEqual(conParams, genParams)) {
1270
- let conEntity;
1271
- conEntity = this.createOrGetFamixConcreteElement(genEntity, cls, instance.getTypeArguments());
1272
- const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1273
- let createConcretisation = true;
1274
- concretisations.forEach((conc) => {
1275
- if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName) {
1276
- createConcretisation = false;
1277
- }
1278
- });
1279
- if (createConcretisation) {
1280
- const fmxConcretisation = this.createFamixConcretisation(conEntity, genEntity);
1281
- }
1282
- }
1283
- }
1284
- });
1285
- }
1286
- // TODO: This function seems unfinished
1287
- }
1490
+ // public createFamixConcretisationGenericInstantiation(cls: ClassDeclaration){
1491
+ // const isGeneric = cls.getTypeParameters().length > 0;
1492
+ // if (isGeneric) {
1493
+ // const instances = cls.getSourceFile().getDescendantsOfKind(ts.SyntaxKind.NewExpression)
1494
+ // .filter(newExpr => {
1495
+ // const expression = newExpr.getExpression();
1496
+ // return expression.getText() === cls.getName();
1497
+ // });
1498
+ // instances.forEach(instance => {
1499
+ // const instanceIsGeneric = instance.getTypeArguments().length > 0;
1500
+ // if (instanceIsGeneric) {
1501
+ // const conParams = instance.getTypeArguments().map((param) => param.getText());
1502
+ // const genEntity = this.createOrGetFamixClass(cls) as Famix.ParametricClass;
1503
+ // const genParams = cls.getTypeParameters().map((param) => param.getText());
1504
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1505
+ // let conEntity;
1506
+ // conEntity = this.createOrGetFamixConcreteElement(genEntity,cls,instance.getTypeArguments());
1507
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1508
+ // let createConcretisation : boolean = true;
1509
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1510
+ // if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){
1511
+ // createConcretisation = false;
1512
+ // }
1513
+ // });
1514
+ // if (createConcretisation) {
1515
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);
1516
+ // }
1517
+ // }
1518
+ // }
1519
+ // })
1520
+ // }
1521
+ // // TODO: This function seems unfinished
1522
+ // }
1288
1523
  /**
1289
1524
  * Creates a Famix concretisation between a class and its instanciations
1290
1525
  * @param func A function
1291
1526
  */
1292
- createFamixConcretisationFunctionInstantiation(element) {
1293
- const isGeneric = element.getTypeParameters().length > 0;
1294
- if (isGeneric) {
1295
- const genParams = element.getTypeParameters().map(param => param.getText());
1296
- const uses = element.findReferencesAsNodes();
1297
- uses.forEach(usage => {
1298
- let currentNode = usage;
1299
- while (currentNode) {
1300
- if (currentNode.getKind() === ts_morph_1.SyntaxKind.CallExpression) {
1301
- const callExpression = currentNode.asKind(ts_morph_1.SyntaxKind.CallExpression);
1302
- if (!callExpression) {
1303
- throw new Error(`CallExpression not found for ${currentNode.getText()}`);
1304
- }
1305
- const instanceIsGeneric = callExpression.getTypeArguments().length > 0;
1306
- if (instanceIsGeneric) {
1307
- const args = callExpression.getTypeArguments();
1308
- const conParams = callExpression.getTypeArguments().map(param => param.getText());
1309
- if (!Helpers.arraysAreEqual(conParams, genParams)) {
1310
- let genElement;
1311
- if (element instanceof ts_morph_1.FunctionDeclaration) {
1312
- genElement = this.createOrGetFamixFunction(element, {});
1313
- }
1314
- else {
1315
- genElement = this.createOrGetFamixMethod(element, {});
1316
- }
1317
- let concElement;
1318
- concElement = this.createOrGetFamixConcreteElement(genElement, element, args);
1319
- const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1320
- let createConcretisation = true;
1321
- concretisations.forEach((conc) => {
1322
- if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName) {
1323
- createConcretisation = false;
1324
- }
1325
- });
1326
- if (createConcretisation) {
1327
- const fmxConcretisation = this.createFamixConcretisation(concElement, genElement);
1328
- }
1329
- }
1330
- }
1331
- break;
1332
- }
1333
- // Remonter à l'élément parent (utile si le nœud de référence est un enfant)
1334
- currentNode = currentNode.getParent();
1335
- }
1336
- });
1337
- }
1338
- }
1527
+ // public createFamixConcretisationFunctionInstantiation(element: FunctionDeclaration | MethodDeclaration){
1528
+ // const isGeneric = element.getTypeParameters().length > 0;
1529
+ // if (isGeneric) {
1530
+ // const genParams = element.getTypeParameters().map(param => param.getText());
1531
+ // const uses = element.findReferencesAsNodes();
1532
+ // uses.forEach(usage => {
1533
+ // let currentNode: Node | undefined = usage;
1534
+ // while (currentNode) {
1535
+ // if (currentNode.getKind() === SyntaxKind.CallExpression) {
1536
+ // const callExpression = currentNode.asKind(SyntaxKind.CallExpression);
1537
+ // if (!callExpression) {
1538
+ // throw new Error(`CallExpression not found for ${currentNode.getText()}`);
1539
+ // }
1540
+ // const instanceIsGeneric = callExpression.getTypeArguments().length > 0;
1541
+ // if (instanceIsGeneric) {
1542
+ // const args = callExpression.getTypeArguments();
1543
+ // const conParams = callExpression.getTypeArguments().map(param => param.getText());
1544
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1545
+ // let genElement;
1546
+ // if(element instanceof FunctionDeclaration){
1547
+ // genElement = this.createOrGetFamixFunction(element, {}) as Famix.ParametricFunction;
1548
+ // } else {
1549
+ // genElement = this.createOrGetFamixMethod(element, {}) as Famix.ParametricMethod;
1550
+ // }
1551
+ // let concElement;
1552
+ // concElement = this.createOrGetFamixConcreteElement(genElement,element,args);
1553
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1554
+ // let createConcretisation : boolean = true;
1555
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1556
+ // if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){
1557
+ // createConcretisation = false;
1558
+ // }
1559
+ // });
1560
+ // if (createConcretisation) {
1561
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);
1562
+ // }
1563
+ // }
1564
+ // }
1565
+ // break;
1566
+ // }
1567
+ // // Remonter à l'élément parent (utile si le nœud de référence est un enfant)
1568
+ // currentNode = currentNode.getParent();
1569
+ // }
1570
+ // });
1571
+ // }
1572
+ // }
1339
1573
  /**
1340
1574
  * Creates a Famix concretisation between a class and an interface
1341
1575
  * @param cls A class
1342
1576
  */
1343
- createFamixConcretisationInterfaceClass(cls) {
1344
- const superInterfaces = cls.getImplements();
1345
- superInterfaces.forEach(interfaceType => {
1346
- const interfaceIsGeneric = interfaceType.getTypeArguments().length > 0;
1347
- if (interfaceIsGeneric) {
1348
- const interfaceDeclaration = interfaceType.getExpression().getSymbol()?.getDeclarations()[0];
1349
- const genParams = interfaceDeclaration.getTypeParameters().map((param) => param.getText());
1350
- const conParams = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1351
- const args = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1352
- if (!Helpers.arraysAreEqual(conParams, genParams)) {
1353
- const genInterface = this.createOrGetFamixInterface(interfaceDeclaration);
1354
- const conInterface = this.createOrGetFamixConcreteElement(genInterface, interfaceDeclaration, args);
1355
- const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1356
- let createConcretisation = true;
1357
- concretisations.forEach((conc) => {
1358
- if (genInterface.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conInterface.fullyQualifiedName) {
1359
- createConcretisation = false;
1360
- }
1361
- });
1362
- if (createConcretisation) {
1363
- const fmxConcretisation = this.createFamixConcretisation(conInterface, genInterface);
1364
- }
1365
- }
1366
- }
1367
- });
1368
- }
1577
+ // public createFamixConcretisationInterfaceClass(cls: ClassDeclaration){
1578
+ // const superInterfaces = cls.getImplements();
1579
+ // superInterfaces.forEach(interfaceType => {
1580
+ // const interfaceIsGeneric = interfaceType.getTypeArguments().length>0;
1581
+ // if (interfaceIsGeneric) {
1582
+ // const interfaceDeclaration = interfaceType.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;
1583
+ // const genParams = interfaceDeclaration.getTypeParameters().map((param) => param.getText());
1584
+ // const conParams = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1585
+ // const args = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1586
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1587
+ // const genInterface = this.createOrGetFamixInterface(interfaceDeclaration) as Famix.ParametricInterface;
1588
+ // const conInterface = this.createOrGetFamixConcreteElement(genInterface,interfaceDeclaration,args);
1589
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1590
+ // let createConcretisation : boolean = true;
1591
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1592
+ // if (genInterface.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conInterface.fullyQualifiedName){
1593
+ // createConcretisation = false;
1594
+ // }
1595
+ // });
1596
+ // if (createConcretisation) {
1597
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conInterface,genInterface);
1598
+ // }
1599
+ // }
1600
+ // }
1601
+ // });
1602
+ // }
1369
1603
  /**
1370
1604
  * Creates a Famix concretisation between an interface and a Type
1371
1605
  * @param element A variable or a function
1372
1606
  * @param inter An interface
1373
1607
  */
1374
- createFamixConcretisationTypeInstanciation(element) {
1375
- const isGeneric = element.getTypeParameters().length > 0;
1376
- if (isGeneric) {
1377
- const genParams = element.getTypeParameters().map(param => param.getText());
1378
- const uses = element.findReferencesAsNodes();
1379
- uses.forEach(use => {
1380
- let parentNode = use.getParent();
1381
- while (parentNode) {
1382
- if (parentNode.getKind() === ts_morph_1.SyntaxKind.TypeReference) {
1383
- const typeReferenceNode = parentNode.asKind(ts_morph_1.SyntaxKind.TypeReference);
1384
- if (!typeReferenceNode) {
1385
- throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);
1386
- }
1387
- const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;
1388
- if (typeReferenceNodeIsGeneric) { }
1389
- const args = typeReferenceNode.getTypeArguments();
1390
- const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());
1391
- if (!Helpers.arraysAreEqual(conParams, genParams)) {
1392
- let genElement;
1393
- if (element instanceof ts_morph_1.ClassDeclaration) {
1394
- genElement = this.createOrGetFamixClass(element);
1395
- }
1396
- else {
1397
- genElement = this.createOrGetFamixInterface(element);
1398
- }
1399
- let concElement;
1400
- concElement = this.createOrGetFamixConcreteElement(genElement, element, args);
1401
- const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1402
- let createConcretisation = true;
1403
- concretisations.forEach((conc) => {
1404
- if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName) {
1405
- createConcretisation = false;
1406
- }
1407
- });
1408
- if (createConcretisation) {
1409
- const fmxConcretisation = this.createFamixConcretisation(concElement, genElement);
1410
- }
1411
- }
1412
- break;
1413
- }
1414
- parentNode = parentNode.getParent();
1415
- }
1416
- });
1417
- }
1418
- }
1608
+ // public createFamixConcretisationTypeInstanciation(element: InterfaceDeclaration | ClassDeclaration){
1609
+ // const isGeneric = element.getTypeParameters().length > 0;
1610
+ // if (isGeneric) {
1611
+ // const genParams = element.getTypeParameters().map(param => param.getText());
1612
+ // const uses = element.findReferencesAsNodes();
1613
+ // uses.forEach(use => {
1614
+ // let parentNode = use.getParent();
1615
+ // while (parentNode) {
1616
+ // if (parentNode.getKind() === SyntaxKind.TypeReference) {
1617
+ // const typeReferenceNode = parentNode.asKind(SyntaxKind.TypeReference);
1618
+ // if (!typeReferenceNode) {
1619
+ // throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);
1620
+ // }
1621
+ // const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;
1622
+ // if (typeReferenceNodeIsGeneric) {}
1623
+ // const args = typeReferenceNode.getTypeArguments();
1624
+ // const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());
1625
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1626
+ // let genElement;
1627
+ // if(element instanceof ClassDeclaration){
1628
+ // genElement = this.createOrGetFamixClass(element) as Famix.ParametricClass;
1629
+ // } else {
1630
+ // genElement = this.createOrGetFamixInterface(element) as Famix.ParametricInterface;
1631
+ // }
1632
+ // let concElement;
1633
+ // concElement = this.createOrGetFamixConcreteElement(genElement,element,args);
1634
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1635
+ // let createConcretisation : boolean = true;
1636
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1637
+ // if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){
1638
+ // createConcretisation = false;
1639
+ // }
1640
+ // });
1641
+ // if (createConcretisation) {
1642
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);
1643
+ // }
1644
+ // }
1645
+ // break;
1646
+ // }
1647
+ // parentNode = parentNode.getParent();
1648
+ // }
1649
+ // });
1650
+ // }
1651
+ // }
1419
1652
  convertToRelativePath(absolutePath, absolutePathProject) {
1420
- return absolutePath.replace(absolutePathProject, "").slice(1);
1653
+ analyze_1.logger.debug(`convertToRelativePath: absolutePath: '${absolutePath}', absolutePathProject: '${absolutePathProject}'`);
1654
+ if (absolutePath.startsWith(absolutePathProject)) {
1655
+ return absolutePath.replace(absolutePathProject, "").slice(1);
1656
+ }
1657
+ else if (absolutePath.startsWith("/")) {
1658
+ return absolutePath.slice(1);
1659
+ }
1660
+ else {
1661
+ return absolutePath;
1662
+ }
1421
1663
  }
1422
1664
  }
1423
1665
  exports.EntityDictionary = EntityDictionary;
1666
+ function isPrimitiveType(typeName) {
1667
+ return typeName === "number" ||
1668
+ typeName === "string" ||
1669
+ typeName === "boolean" ||
1670
+ typeName === "bigint" ||
1671
+ typeName === "symbol" ||
1672
+ typeName === "undefined" ||
1673
+ typeName === "null" ||
1674
+ typeName === "any" ||
1675
+ typeName === "unknown" ||
1676
+ typeName === "never" ||
1677
+ typeName === "void";
1678
+ }
1424
1679
  function initFQN(sourceElement, famixElement) {
1680
+ // handle special cases where an element is a Type -- need to change its name
1681
+ if (famixElement instanceof Famix.Type && !(sourceElement instanceof ts_morph_1.CommentRange) && isTypeContext(sourceElement)) {
1682
+ let fqn = FQNFunctions.getFQN(sourceElement);
1683
+ // using regex, replace [blah] with [blahType]
1684
+ fqn = fqn.replace(/\[([^\]]+)\]/g, "[$1Type]");
1685
+ analyze_1.logger.info("Setting fully qualified name for " + famixElement.getJSON() + " to " + fqn);
1686
+ famixElement.fullyQualifiedName = fqn;
1687
+ return;
1688
+ }
1689
+ // catch all (except comments)
1425
1690
  if (!(sourceElement instanceof ts_morph_1.CommentRange)) {
1426
1691
  const fqn = FQNFunctions.getFQN(sourceElement);
1427
- analyze_1.logger.debug("Setting fully qualified name for " + famixElement.getJSON() + " to " + fqn);
1692
+ analyze_1.logger.info("Setting fully qualified name for " + famixElement.getJSON() + " to " + fqn);
1428
1693
  famixElement.fullyQualifiedName = fqn;
1429
1694
  }
1430
1695
  }
1431
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRW50aXR5RGljdGlvbmFyeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9mYW1peF9mdW5jdGlvbnMvRW50aXR5RGljdGlvbmFyeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHVDQUE2bEI7QUFDN2xCLDhFQUFnRjtBQUNoRixnRUFBa0Q7QUFDbEQsb0VBQWdFO0FBQ2hFLHdDQUE0QztBQUM1QywwRUFBaUQ7QUFDakQsNERBQThDO0FBQzlDLHFEQUF1QztBQUN2QyxnREFBd0I7QUFDeEIsb0RBQXVCO0FBVXZCLE1BQWEsZ0JBQWdCO0lBY3pCO1FBWk8sYUFBUSxHQUFHLElBQUksa0NBQWUsRUFBRSxDQUFDO1FBQ2hDLGdCQUFXLEdBQUcsSUFBSSxHQUFHLEVBQXVCLENBQUMsQ0FBQyw0Q0FBNEM7UUFDMUYsZ0JBQVcsR0FBRyxJQUFJLEdBQUcsRUFBK0MsQ0FBQyxDQUFDLDREQUE0RDtRQUNsSSxvQkFBZSxHQUFHLElBQUksR0FBRyxFQUF1RCxDQUFDLENBQUMsZ0RBQWdEO1FBQ2xJLGlCQUFZLEdBQUcsSUFBSSxHQUFHLEVBQXdCLENBQUMsQ0FBQyxnREFBZ0Q7UUFDaEcsZUFBVSxHQUFHLElBQUksR0FBRyxFQUE2QyxDQUFDLENBQUMsa0RBQWtEO1FBQ3JILGVBQVUsR0FBRyxJQUFJLEdBQUcsRUFBa0UsQ0FBQyxDQUFDLDJDQUEyQztRQUNuSSw0QkFBdUIsR0FBRyxJQUFJLEdBQThGLENBQUEsQ0FBQywrQ0FBK0M7UUFDNUssa0JBQWEsR0FBRyxnQ0FBZ0MsQ0FBQyxDQUFDLDZDQUE2QztRQUNoRyx3QkFBbUIsR0FBRyxJQUFJLEdBQUcsRUFBa0MsQ0FBQztRQUNoRSw0QkFBdUIsR0FBRyxJQUFJLEdBQUcsRUFBa0MsQ0FBQztRQUd2RSxJQUFJLENBQUMsUUFBUSxDQUFDLHNCQUFzQixDQUFDLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO0lBQ25FLENBQUM7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RyxLQUFLLENBQUM7Z0NBQ2pDLENBQUM7NEJBQ0wsQ0FBQyxDQUFDLENBQUM7NEJBRUgsSUFBSSxvQkFBb0IsRUFBRSxDQUFDO2dDQUN2QixNQUFNLGlCQUFpQixHQUEwQixJQUFJLENBQUMseUJBQXlCLENBQUMsV0FBVyxFQUFDLFVBQVUsQ0FBQyxDQUFDOzRCQUM1RyxDQUFDO3dCQUNMLENBQUM7d0JBQ0wsTUFBTTtvQkFDVixDQUFDO29CQUNELFVBQVUsR0FBRyxVQUFVLENBQUMsU0FBUyxFQUFFLENBQUM7Z0JBQ3hDLENBQUM7WUFDTCxDQUFDLENBQUMsQ0FBQztRQUNQLENBQUM7SUFDTCxDQUFDO0lBRU0scUJBQXFCLENBQUMsWUFBb0IsRUFBRSxtQkFBMkI7UUFDMUUsT0FBTyxZQUFZLENBQUMsT0FBTyxDQUFDLG1CQUFtQixFQUFFLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsRSxDQUFDO0NBQ0o7QUE3aERELDRDQTZoREM7QUFDRCxTQUFTLE9BQU8sQ0FBQyxhQUFnQyxFQUFFLFlBQWlDO0lBQ2hGLElBQUksQ0FBQyxDQUFDLGFBQWEsWUFBWSx1QkFBWSxDQUFDLEVBQUUsQ0FBQztRQUMzQyxNQUFNLEdBQUcsR0FBRyxZQUFZLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQy9DLGdCQUFNLENBQUMsS0FBSyxDQUFDLG1DQUFtQyxHQUFHLFlBQVksQ0FBQyxPQUFPLEVBQUUsR0FBRyxNQUFNLEdBQUcsR0FBRyxDQUFDLENBQUM7UUFDekYsWUFBa0MsQ0FBQyxrQkFBa0IsR0FBRyxHQUFHLENBQUM7SUFDakUsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDbGFzc0RlY2xhcmF0aW9uLCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uLCBGdW5jdGlvbkRlY2xhcmF0aW9uLCBJZGVudGlmaWVyLCBJbnRlcmZhY2VEZWNsYXJhdGlvbiwgTWV0aG9kRGVjbGFyYXRpb24sIE1ldGhvZFNpZ25hdHVyZSwgTW9kdWxlRGVjbGFyYXRpb24sIFByb3BlcnR5RGVjbGFyYXRpb24sIFByb3BlcnR5U2lnbmF0dXJlLCBTb3VyY2VGaWxlLCBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb24sIFZhcmlhYmxlRGVjbGFyYXRpb24sIFBhcmFtZXRlckRlY2xhcmF0aW9uLCBEZWNvcmF0b3IsIEdldEFjY2Vzc29yRGVjbGFyYXRpb24sIFNldEFjY2Vzc29yRGVjbGFyYXRpb24sIEltcG9ydFNwZWNpZmllciwgQ29tbWVudFJhbmdlLCBFbnVtRGVjbGFyYXRpb24sIEVudW1NZW1iZXIsIFR5cGVBbGlhc0RlY2xhcmF0aW9uLCBGdW5jdGlvbkV4cHJlc3Npb24sIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cywgSW1wb3J0RGVjbGFyYXRpb24sIEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uLCBTeW50YXhLaW5kLCBFeHByZXNzaW9uLCBUeXBlTm9kZSwgTm9kZSwgdHMsIFNjb3BlLCBUeXBlLCBBcnJvd0Z1bmN0aW9uIH0gZnJvbSBcInRzLW1vcnBoXCI7XG5pbXBvcnQgeyBpc0FtYmllbnQsIGlzTmFtZXNwYWNlIH0gZnJvbSBcIi4uL2FuYWx5emVfZnVuY3Rpb25zL3Byb2Nlc3NfZnVuY3Rpb25zXCI7XG5pbXBvcnQgKiBhcyBGYW1peCBmcm9tIFwiLi4vbGliL2ZhbWl4L21vZGVsL2ZhbWl4XCI7XG5pbXBvcnQgeyBGYW1peFJlcG9zaXRvcnkgfSBmcm9tIFwiLi4vbGliL2ZhbWl4L2ZhbWl4X3JlcG9zaXRvcnlcIjtcbmltcG9ydCB7IGxvZ2dlciwgY29uZmlnIH0gZnJvbSBcIi4uL2FuYWx5emVcIjtcbmltcG9ydCBHcmFwaGVtZVNwbGl0dGVyIGZyb20gXCJncmFwaGVtZS1zcGxpdHRlclwiO1xuaW1wb3J0ICogYXMgSGVscGVycyBmcm9tIFwiLi9oZWxwZXJzX2NyZWF0aW9uXCI7XG5pbXBvcnQgKiBhcyBGUU5GdW5jdGlvbnMgZnJvbSBcIi4uL2ZxblwiO1xuaW1wb3J0IHBhdGggZnJvbSBcInBhdGhcIjtcbmltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5cbmV4cG9ydCB0eXBlIFRTTW9ycGhPYmplY3RUeXBlID0gSW1wb3J0RGVjbGFyYXRpb24gfCBJbXBvcnRFcXVhbHNEZWNsYXJhdGlvbiB8IFNvdXJjZUZpbGUgfCBNb2R1bGVEZWNsYXJhdGlvbiB8IENsYXNzRGVjbGFyYXRpb24gfCBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uIHwgQ29uc3RydWN0b3JEZWNsYXJhdGlvbiB8IE1ldGhvZFNpZ25hdHVyZSB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24gfCBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFZhcmlhYmxlRGVjbGFyYXRpb24gfCBQcm9wZXJ0eURlY2xhcmF0aW9uIHwgUHJvcGVydHlTaWduYXR1cmUgfCBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb24gfCBJZGVudGlmaWVyIHwgRGVjb3JhdG9yIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBJbXBvcnRTcGVjaWZpZXIgfCBDb21tZW50UmFuZ2UgfCBFbnVtRGVjbGFyYXRpb24gfCBFbnVtTWVtYmVyIHwgVHlwZUFsaWFzRGVjbGFyYXRpb24gfCBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHM7XG5cbmV4cG9ydCB0eXBlIFR5cGVEZWNsYXJhdGlvbiA9IFR5cGVBbGlhc0RlY2xhcmF0aW9uIHwgUHJvcGVydHlEZWNsYXJhdGlvbiB8IFByb3BlcnR5U2lnbmF0dXJlIHwgTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uIHwgUGFyYW1ldGVyRGVjbGFyYXRpb24gfCBWYXJpYWJsZURlY2xhcmF0aW9uIHwgRW51bU1lbWJlciB8IEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uO1xuXG50eXBlIFBhcmFtZXRyaWNWYXJpYW50VHlwZSA9IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kO1xuXG50eXBlIENvbmNyZXRlRWxlbWVudFRTTW9ycGhUeXBlID0gQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uIHwgRnVuY3Rpb25EZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uO1xuXG5leHBvcnQgY2xhc3MgRW50aXR5RGljdGlvbmFyeSB7XG4gICAgXG4gICAgcHVibGljIGZhbWl4UmVwID0gbmV3IEZhbWl4UmVwb3NpdG9yeSgpO1xuICAgIHByaXZhdGUgZm14QWxpYXNNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguQWxpYXM+KCk7IC8vIE1hcHMgdGhlIGFsaWFzIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXG4gICAgcHJpdmF0ZSBmbXhDbGFzc01hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5DbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNDbGFzcz4oKTsgLy8gTWFwcyB0aGUgZnVsbHkgcXVhbGlmaWVkIGNsYXNzIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXG4gICAgcHJpdmF0ZSBmbXhJbnRlcmZhY2VNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguSW50ZXJmYWNlIHwgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZT4oKTsgLy8gTWFwcyB0aGUgaW50ZXJmYWNlIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXG4gICAgcHJpdmF0ZSBmbXhNb2R1bGVNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguTW9kdWxlPigpOyAvLyBNYXBzIHRoZSBuYW1lc3BhY2UgbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcbiAgICBwcml2YXRlIGZteEZpbGVNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguU2NyaXB0RW50aXR5IHwgRmFtaXguTW9kdWxlPigpOyAvLyBNYXBzIHRoZSBzb3VyY2UgZmlsZSBuYW1lcyB0byB0aGVpciBGYW1peCBtb2RlbFxuICAgIHByaXZhdGUgZm14VHlwZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5UeXBlIHwgRmFtaXguUHJpbWl0aXZlVHlwZSB8IEZhbWl4LlBhcmFtZXRlclR5cGU+KCk7IC8vIE1hcHMgdGhlIHR5cGUgbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcbiAgICBwcml2YXRlIGZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwID0gbmV3IE1hcDxzdHJpbmcsIEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uIHwgRmFtaXguTWV0aG9kIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZD4gLy8gTWFwcyB0aGUgZnVuY3Rpb24gbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcbiAgICBwcml2YXRlIFVOS05PV05fVkFMVUUgPSAnKHVua25vd24gZHVlIHRvIHBhcnNpbmcgZXJyb3IpJzsgLy8gVGhlIHZhbHVlIHRvIHVzZSB3aGVuIGEgbmFtZSBpcyBub3QgdXNhYmxlXG4gICAgcHVibGljIGZteEVsZW1lbnRPYmplY3RNYXAgPSBuZXcgTWFwPEZhbWl4LkVudGl0eSxUU01vcnBoT2JqZWN0VHlwZT4oKTtcbiAgICBwdWJsaWMgdHNNb3JwaEVsZW1lbnRPYmplY3RNYXAgPSBuZXcgTWFwPFRTTW9ycGhPYmplY3RUeXBlLEZhbWl4LkVudGl0eT4oKTtcbiAgICAgICAgICAgIFxuICAgIGNvbnN0cnVjdG9yKCkge1xuICAgICAgICB0aGlzLmZhbWl4UmVwLnNldEZteEVsZW1lbnRPYmplY3RNYXAodGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwKTsgICAgICBcbiAgICB9XG5cbiAgICBwdWJsaWMgYWRkU291cmNlQW5jaG9yKGZteDogRmFtaXguU291cmNlZEVudGl0eSwgbm9kZTogVFNNb3JwaE9iamVjdFR5cGUpOiBGYW1peC5JbmRleGVkRmlsZUFuY2hvciB7XG4gICAgICAgIGNvbnN0IHNvdXJjZUFuY2hvcjogRmFtaXguSW5kZXhlZEZpbGVBbmNob3IgPSBuZXcgRmFtaXguSW5kZXhlZEZpbGVBbmNob3IoKTtcbiAgICAgICAgbGV0IHNvdXJjZVN0YXJ0LCBzb3VyY2VFbmQ6IG51bWJlcjtcbiAgICAgICAgaWYgKGZteCAmJiBub2RlKSB7XG4gICAgICAgICAgICAvLyBmaW5kIHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGUgc291cmNlIGVsZW1lbnRcbiAgICAgICAgICAgIGlmICghKG5vZGUgaW5zdGFuY2VvZiBDb21tZW50UmFuZ2UpKSB7XG4gICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBub2RlLmdldFN0YXJ0KCk7XG4gICAgICAgICAgICAgICAgc291cmNlRW5kID0gbm9kZS5nZXRFbmQoKTtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBub2RlLmdldFBvcygpO1xuICAgICAgICAgICAgICAgIHNvdXJjZUVuZCA9IG5vZGUuZ2V0RW5kKCk7XG4gICAgICAgICAgICB9XG4gICAgXG4gICAgICAgICAgICBpZiAoY29uZmlnLmV4cGVjdEdyYXBoZW1lcykge1xuICAgICAgICAgICAgICAgIC8qKlxuICAgICAgICAgICAgICAgICAqIFRoZSBmb2xsb3dpbmcgbG9naWMgaGFuZGxlcyB0aGUgY2FzZSBvZiBtdWx0aS1jb2RlIHBvaW50IGNoYXJhY3RlcnMgKGUuZy4gZW1vamkpIGluIHRoZSBzb3VyY2UgdGV4dC5cbiAgICAgICAgICAgICAgICAgKiBUaGlzIGlzIG5lZWRlZCBiZWNhdXNlIFBoYXJvL1NtYWxsdGFsayB0cmVhdHMgbXVsdGktY29kZSBwb2ludCBjaGFyYWN0ZXJzIGFzIGEgc2luZ2xlIGNoYXJhY3RlciwgXG4gICAgICAgICAgICAgICAgICogYnV0IEphdmFTY3JpcHQgdHJlYXRzIHRoZW0gYXMgbXVsdGlwbGUgY2hhcmFjdGVycy4gVGhpcyBtZWFucyB0aGF0IHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9uc1xuICAgICAgICAgICAgICAgICAqIG9mIGEgc291cmNlIGVsZW1lbnQgaW4gUGhhcm8vU21hbGx0YWxrIHdpbGwgYmUgZGlmZmVyZW50IHRoYW4gdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZVxuICAgICAgICAgICAgICAgICAqIHNhbWUgc291cmNlIGVsZW1lbnQgaW4gSmF2YVNjcmlwdC4gVGhpcyBsb2dpYyBmaW5kcyB0aGUgc3RhcnQgYW5kIGVuZCBwb3NpdGlvbnMgb2YgdGhlIHNvdXJjZVxuICAgICAgICAgICAgICAgICAqIGVsZW1lbnQgaW4gSmF2YVNjcmlwdCBhbmQgdGhlbiB1c2VzIHRob3NlIHBvc2l0aW9ucyB0byBzZXQgdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZVxuICAgICAgICAgICAgICAgICAqIEZhbWl4IGluZGV4IGZpbGUgYW5jaG9yLlxuICAgICAgICAgICAgICAgICAqIEl0IGRlcGVuZHMgb24gY29kZSBpbiB0aGUgJ2dyYXBoZW1lLXNwbGl0dGVyJyBwYWNrYWdlIGluIG5wbS5cbiAgICAgICAgICAgICAgICAgKi9cbiAgICAgICAgICAgICAgICBjb25zdCBzcGxpdHRlciA9IG5ldyBHcmFwaGVtZVNwbGl0dGVyKCk7XG4gICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRmlsZVRleHQgPSBub2RlLmdldFNvdXJjZUZpbGUoKS5nZXRGdWxsVGV4dCgpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGhhc0dyYXBoZW1lQ2x1c3RlcnMgPSBzcGxpdHRlci5jb3VudEdyYXBoZW1lcyhzb3VyY2VGaWxlVGV4dCkgPiAxO1xuICAgICAgICAgICAgICAgIGlmIChoYXNHcmFwaGVtZUNsdXN0ZXJzKSB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUVsZW1lbnRUZXh0ID0gc291cmNlRmlsZVRleHQuc3Vic3RyaW5nKHNvdXJjZVN0YXJ0LCBzb3VyY2VFbmQpO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcyA9IHNwbGl0dGVyLnNwbGl0R3JhcGhlbWVzKHNvdXJjZUVsZW1lbnRUZXh0KTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRmlsZVRleHRHcmFwaGVtZXMgPSBzcGxpdHRlci5zcGxpdEdyYXBoZW1lcyhzb3VyY2VGaWxlVGV4dCk7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IG51bWJlck9mR3JhcGhlbWVDbHVzdGVyc0JlZm9yZVN0YXJ0ID0gc3BsaXR0ZXIuY291bnRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQuc3Vic3RyaW5nKDAsIHNvdXJjZVN0YXJ0KSk7XG4gICAgICAgIFxuICAgICAgICAgICAgICAgICAgICAvLyBmaW5kIHRoZSBzdGFydCBvZiB0aGUgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMgYXJyYXkgaW4gdGhlIHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzIGFycmF5XG4gICAgICAgICAgICAgICAgICAgIHNvdXJjZVN0YXJ0ID0gSGVscGVycy5pbmRleE9mU3BsaXRBcnJheSh7c2VhcmNoQXJyYXk6IHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGFyZ2V0QXJyYXk6IHNvdXJjZUVsZW1lbnRUZXh0R3JhcGhlbWVzLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3RhcnQ6IHNvdXJjZVN0YXJ0IC0gbnVtYmVyT2ZHcmFwaGVtZUNsdXN0ZXJzQmVmb3JlU3RhcnR9KTtcbiAgICAgICAgICAgICAgICAgICAgc291cmNlRW5kID0gc291cmNlU3RhcnQgKyBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcy5sZW5ndGg7XG4gICAgICAgICAgICAgICAgfSBcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gVGhlICsxIGlzIGJlY2F1c2UgdGhlIHNvdXJjZSBhbmNob3IgKFBoYXJvKSBpcyAxLWJhc2VkLCBidXQgdHMtbW9ycGggaXMgMC1iYXNlZFxuICAgICAgICAgICAgc291cmNlQW5jaG9yLnN0YXJ0UG9zID0gc291cmNlU3RhcnQgKyAxO1xuICAgICAgICAgICAgc291cmNlQW5jaG9yLmVuZFBvcyA9IHNvdXJjZUVuZCArIDE7XG5cbiAgICAgICAgICAgIGNvbnN0IGZpbGVOYW1lID0gbm9kZS5nZXRTb3VyY2VGaWxlKCkuZ2V0RmlsZVBhdGgoKTtcblxuICAgICAgICAgICAgc291cmNlQW5jaG9yLmVsZW1lbnQgPSBmbXg7XG4gICAgICAgICAgICBzb3VyY2VBbmNob3IuZmlsZU5hbWUgPSBmaWxlTmFtZTtcbiAgICAgICAgICAgIGZteC5zb3VyY2VBbmNob3IgPSBzb3VyY2VBbmNob3I7XG4gICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoc291cmNlQW5jaG9yKTtcblxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBzb3VyY2VBbmNob3I7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogTWFrZXMgYSBGYW1peCBpbmRleCBmaWxlIGFuY2hvclxuICAgICAqIEBwYXJhbSBzb3VyY2VFbGVtZW50IEEgc291cmNlIGVsZW1lbnRcbiAgICAgKiBAcGFyYW0gZmFtaXhFbGVtZW50IFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgc291cmNlIGVsZW1lbnRcbiAgICAgKi9cbiAgICBwdWJsaWMgbWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHNvdXJjZUVsZW1lbnQ6IFRTTW9ycGhPYmplY3RUeXBlLCBmYW1peEVsZW1lbnQ6IEZhbWl4LlNvdXJjZWRFbnRpdHkpOiB2b2lkIHtcbiAgICAgICAgLy8gY2hlY2sgaWYgZmFtaXhFbGVtZW50IGRvZXNuJ3QgaGF2ZSBhIHZhbGlkIGZ1bGx5UXVhbGlmaWVkTmFtZVxuICAgICAgICBpZiAodHlwZW9mIChmYW1peEVsZW1lbnQgYXMgYW55KS5nZXRGdWxseVF1YWxpZmllZE5hbWUgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgICAgIC8vIFRoZSBtZXRob2QgZXhpc3RzXG4gICAgICAgICAgICBjb25zdCBmdWxseVF1YWxpZmllZE5hbWUgPSAoZmFtaXhFbGVtZW50IGFzIGFueSkuZnVsbHlRdWFsaWZpZWROYW1lO1xuICAgICAgICAgICAgaWYgKCFmdWxseVF1YWxpZmllZE5hbWUgfHwgZnVsbHlRdWFsaWZpZWROYW1lID09PSB0aGlzLlVOS05PV05fVkFMVUUpIHtcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhbWl4IGVsZW1lbnQgJHtmYW1peEVsZW1lbnQuY29uc3RydWN0b3IubmFtZX0gaGFzIG5vIHZhbGlkIGZ1bGx5UXVhbGlmaWVkTmFtZS5gKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhcIm1ha2luZyBpbmRleCBmaWxlIGFuY2hvciBmb3IgJ1wiICsgc291cmNlRWxlbWVudD8uZ2V0VGV4dCgpICsgXCInIHdpdGggZmFtaXhFbGVtZW50IFwiICsgZmFtaXhFbGVtZW50LmdldEpTT04oKSk7XG4gICAgICAgIGNvbnN0IGZteEluZGV4RmlsZUFuY2hvciA9IG5ldyBGYW1peC5JbmRleGVkRmlsZUFuY2hvcigpO1xuICAgICAgICBmbXhJbmRleEZpbGVBbmNob3IuZWxlbWVudCA9IGZhbWl4RWxlbWVudDtcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmYW1peEVsZW1lbnQsIHNvdXJjZUVsZW1lbnQpO1xuXG4gICAgICAgIGlmIChzb3VyY2VFbGVtZW50ICE9PSBudWxsKSB7XG4gICAgICAgICAgICBjb25zdCBhYnNvbHV0ZVBhdGhQcm9qZWN0ID0gdGhpcy5mYW1peFJlcC5nZXRBYnNvbHV0ZVBhdGgoKTtcbiAgICAgICAgXG4gICAgICAgICAgICBjb25zdCBhYnNvbHV0ZVBhdGggPSBwYXRoLm5vcm1hbGl6ZShzb3VyY2VFbGVtZW50LmdldFNvdXJjZUZpbGUoKS5nZXRGaWxlUGF0aCgpKTtcblxuICAgICAgICAgICAgY29uc3QgcG9zaXRpb25Ob2RlTW9kdWxlcyA9IGFic29sdXRlUGF0aC5pbmRleE9mKCdub2RlX21vZHVsZXMnKTtcblxuICAgICAgICAgICAgbGV0IHBhdGhJblByb2plY3Q6IHN0cmluZyA9IFwiXCI7XG5cbiAgICAgICAgICAgIGlmIChwb3NpdGlvbk5vZGVNb2R1bGVzICE9PSAtMSkge1xuICAgICAgICAgICAgICAgIGNvbnN0IHBhdGhGcm9tTm9kZU1vZHVsZXMgPSBhYnNvbHV0ZVBhdGguc3Vic3RyaW5nKHBvc2l0aW9uTm9kZU1vZHVsZXMpO1xuICAgICAgICAgICAgICAgIHBhdGhJblByb2plY3QgPSBwYXRoRnJvbU5vZGVNb2R1bGVzO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICBwYXRoSW5Qcm9qZWN0ID0gdGhpcy5jb252ZXJ0VG9SZWxhdGl2ZVBhdGgoYWJzb2x1dGVQYXRoLCBhYnNvbHV0ZVBhdGhQcm9qZWN0KTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gcmV2ZXJ0IGFueSBiYWNrc2xhc2hlcyB0byBmb3J3YXJkIHNsYXNoZXMgKHBhdGgubm9ybWFsaXplIG9uIHdpbmRvd3MgaW50cm9kdWNlcyB0aGVtKVxuICAgICAgICAgICAgcGF0aEluUHJvamVjdCA9IHBhdGhJblByb2plY3QucmVwbGFjZSgvXFxcXC9nLCBcIi9cIik7XG5cbiAgICAgICAgICAgIGZteEluZGV4RmlsZUFuY2hvci5maWxlTmFtZSA9IHBhdGhJblByb2plY3Q7XG4gICAgICAgICAgICBsZXQgc291cmNlU3RhcnQsIHNvdXJjZUVuZCwgc291cmNlTGluZVN0YXJ0LCBzb3VyY2VMaW5lRW5kOiBudW1iZXI7XG4gICAgICAgICAgICBpZiAoIShzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgQ29tbWVudFJhbmdlKSkge1xuICAgICAgICAgICAgICAgIHNvdXJjZVN0YXJ0ID0gc291cmNlRWxlbWVudC5nZXRTdGFydCgpO1xuICAgICAgICAgICAgICAgIHNvdXJjZUVuZCA9IHNvdXJjZUVsZW1lbnQuZ2V0RW5kKCk7XG4gICAgICAgICAgICAgICAgc291cmNlTGluZVN0YXJ0ID0gc291cmNlRWxlbWVudC5nZXRTdGFydExpbmVOdW1iZXIoKTtcbiAgICAgICAgICAgICAgICBzb3VyY2VMaW5lRW5kID0gc291cmNlRWxlbWVudC5nZXRFbmRMaW5lTnVtYmVyKCk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIHNvdXJjZVN0YXJ0ID0gc291cmNlRWxlbWVudC5nZXRQb3MoKTtcbiAgICAgICAgICAgICAgICBzb3VyY2VFbmQgPSBzb3VyY2VFbGVtZW50LmdldEVuZCgpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaWYgKGNvbmZpZy5leHBlY3RHcmFwaGVtZXMpIHtcbiAgICAgICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAgICAgKiBUaGUgZm9sbG93aW5nIGxvZ2ljIGhhbmRsZXMgdGhlIGNhc2Ugb2YgbXVsdGktY29kZSBwb2ludCBjaGFyYWN0ZXJzIChlLmcuIGVtb2ppKSBpbiB0aGUgc291cmNlIHRleHQuXG4gICAgICAgICAgICAgICAgICogVGhpcyBpcyBuZWVkZWQgYmVjYXVzZSBQaGFyby9TbWFsbHRhbGsgdHJlYXRzIG11bHRpLWNvZGUgcG9pbnQgY2hhcmFjdGVycyBhcyBhIHNpbmdsZSBjaGFyYWN0ZXIsIFxuICAgICAgICAgICAgICAgICAqIGJ1dCBKYXZhU2NyaXB0IHRyZWF0cyB0aGVtIGFzIG11bHRpcGxlIGNoYXJhY3RlcnMuIFRoaXMgbWVhbnMgdGhhdCB0aGUgc3RhcnQgYW5kIGVuZCBwb3NpdGlvbnNcbiAgICAgICAgICAgICAgICAgKiBvZiBhIHNvdXJjZSBlbGVtZW50IGluIFBoYXJvL1NtYWxsdGFsayB3aWxsIGJlIGRpZmZlcmVudCB0aGFuIHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGVcbiAgICAgICAgICAgICAgICAgKiBzYW1lIHNvdXJjZSBlbGVtZW50IGluIEphdmFTY3JpcHQuIFRoaXMgbG9naWMgZmluZHMgdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZSBzb3VyY2VcbiAgICAgICAgICAgICAgICAgKiBlbGVtZW50IGluIEphdmFTY3JpcHQgYW5kIHRoZW4gdXNlcyB0aG9zZSBwb3NpdGlvbnMgdG8gc2V0IHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGVcbiAgICAgICAgICAgICAgICAgKiBGYW1peCBpbmRleCBmaWxlIGFuY2hvci5cbiAgICAgICAgICAgICAgICAgKiBJdCBkZXBlbmRzIG9uIGNvZGUgaW4gdGhlICdncmFwaGVtZS1zcGxpdHRlcicgcGFja2FnZSBpbiBucG0uXG4gICAgICAgICAgICAgICAgICovXG4gICAgICAgICAgICAgICAgY29uc3Qgc3BsaXR0ZXIgPSBuZXcgR3JhcGhlbWVTcGxpdHRlcigpO1xuICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUZpbGVUZXh0ID0gc291cmNlRWxlbWVudC5nZXRTb3VyY2VGaWxlKCkuZ2V0RnVsbFRleHQoKTtcbiAgICAgICAgICAgICAgICBjb25zdCBoYXNHcmFwaGVtZUNsdXN0ZXJzID0gc3BsaXR0ZXIuY291bnRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQpID4gMTtcbiAgICAgICAgICAgICAgICBpZiAoaGFzR3JhcGhlbWVDbHVzdGVycykge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBzb3VyY2VFbGVtZW50VGV4dCA9IHNvdXJjZUZpbGVUZXh0LnN1YnN0cmluZyhzb3VyY2VTdGFydCwgc291cmNlRW5kKTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMgPSBzcGxpdHRlci5zcGxpdEdyYXBoZW1lcyhzb3VyY2VFbGVtZW50VGV4dCk7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzID0gc3BsaXR0ZXIuc3BsaXRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQpO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBudW1iZXJPZkdyYXBoZW1lQ2x1c3RlcnNCZWZvcmVTdGFydCA9IHNwbGl0dGVyLmNvdW50R3JhcGhlbWVzKHNvdXJjZUZpbGVUZXh0LnN1YnN0cmluZygwLCBzb3VyY2VTdGFydCkpO1xuICAgICAgICBcbiAgICAgICAgICAgICAgICAgICAgLy8gZmluZCB0aGUgc3RhcnQgb2YgdGhlIHNvdXJjZUVsZW1lbnRUZXh0R3JhcGhlbWVzIGFycmF5IGluIHRoZSBzb3VyY2VGaWxlVGV4dEdyYXBoZW1lcyBhcnJheVxuICAgICAgICAgICAgICAgICAgICBzb3VyY2VTdGFydCA9IEhlbHBlcnMuaW5kZXhPZlNwbGl0QXJyYXkoe3NlYXJjaEFycmF5OiBzb3VyY2VGaWxlVGV4dEdyYXBoZW1lcywgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRhcmdldEFycmF5OiBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcywgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXJ0OiBzb3VyY2VTdGFydCAtIG51bWJlck9mR3JhcGhlbWVDbHVzdGVyc0JlZm9yZVN0YXJ0fSk7XG4gICAgICAgICAgICAgICAgICAgIHNvdXJjZUVuZCA9IHNvdXJjZVN0YXJ0ICsgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMubGVuZ3RoO1xuICAgICAgICAgICAgICAgIH0gXG4gICAgICAgICAgICB9XG4gICAgICAgICAgICAvLyBub3RlOiB0aGUgKzEgaXMgYmVjYXVzZSB0aGUgc291cmNlIGFuY2hvciBpcyAxLWJhc2VkLCBidXQgdHMtbW9ycGggaXMgMC1iYXNlZFxuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLnN0YXJ0UG9zID0gc291cmNlU3RhcnQgKyAxO1xuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmVuZFBvcyA9IHNvdXJjZUVuZCArIDE7XG5cbiAgICAgICAgICAgIC8vIGlmICghKGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LkltcG9ydENsYXVzZSB8fCBmYW1peEVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5BY2Nlc3MgfHwgZmFtaXhFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguUmVmZXJlbmNlIHx8IGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4Lkludm9jYXRpb24gfHwgZmFtaXhFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguSW5oZXJpdGFuY2UpICYmICEoZmFtaXhFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguQ29tbWVudCkgJiYgIShzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgQ29tbWVudFJhbmdlKSAmJiAhKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBJZGVudGlmaWVyKSAmJiAhKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBJbXBvcnRTcGVjaWZpZXIpICYmICEoc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cykpIHtcbiAgICAgICAgICAgIC8vICAgIGluaXRGUU4oc291cmNlRWxlbWVudCwgZmFtaXhFbGVtZW50KTtcbiAgICAgICAgICAgIC8vIH1cbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIC8vIHNvdXJjZUVsZW1lbnQgaXMgbnVsbFxuICAgICAgICAgICAgbG9nZ2VyLndhcm4oXCJzb3VyY2VFbGVtZW50IGlzIG51bGwgZm9yIGZhbWl4RWxlbWVudCBcIiArIGZhbWl4RWxlbWVudC5nZXRKU09OKCkpO1xuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmZpbGVOYW1lID0gXCJ1bmtub3duXCI7XG4gICAgICAgICAgICBmbXhJbmRleEZpbGVBbmNob3Iuc3RhcnRQb3MgPSAwO1xuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmVuZFBvcyA9IDA7XG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW5kZXhGaWxlQW5jaG9yKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBzY3JpcHQgZW50aXR5IG9yIG1vZHVsZVxuICAgICAqIEBwYXJhbSBmIEEgc291cmNlIGZpbGVcbiAgICAgKiBAcGFyYW0gaXNNb2R1bGUgQSBib29sZWFuIGluZGljYXRpbmcgaWYgdGhlIHNvdXJjZSBmaWxlIGlzIGEgbW9kdWxlXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBzb3VyY2UgZmlsZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4RmlsZShmOiBTb3VyY2VGaWxlLCBpc01vZHVsZTogYm9vbGVhbik6IEZhbWl4LlNjcmlwdEVudGl0eSB8IEZhbWl4Lk1vZHVsZSB7XG4gICAgICAgIGxldCBmbXhGaWxlOiBGYW1peC5TY3JpcHRFbnRpdHk7IC8vIHwgRmFtaXguTW9kdWxlO1xuXG4gICAgICAgIGNvbnN0IGZpbGVOYW1lID0gZi5nZXRCYXNlTmFtZSgpO1xuICAgICAgICBjb25zdCBmdWxseVF1YWxpZmllZEZpbGVuYW1lID0gZi5nZXRGaWxlUGF0aCgpO1xuICAgICAgICBjb25zdCBmb3VuZEZpbGVOYW1lID0gdGhpcy5mbXhGaWxlTWFwLmdldChmdWxseVF1YWxpZmllZEZpbGVuYW1lKTtcbiAgICAgICAgaWYgKCFmb3VuZEZpbGVOYW1lKSB7XG4gICAgICAgICAgICBpZiAoaXNNb2R1bGUpIHtcbiAgICAgICAgICAgICAgICBmbXhGaWxlID0gbmV3IEZhbWl4Lk1vZHVsZSgpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14RmlsZSA9IG5ldyBGYW1peC5TY3JpcHRFbnRpdHkoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGZteEZpbGUubmFtZSA9IGZpbGVOYW1lO1xuICAgICAgICAgICAgZm14RmlsZS5udW1iZXJPZkxpbmVzT2ZUZXh0ID0gZi5nZXRFbmRMaW5lTnVtYmVyKCkgLSBmLmdldFN0YXJ0TGluZU51bWJlcigpO1xuICAgICAgICAgICAgZm14RmlsZS5udW1iZXJPZkNoYXJhY3RlcnMgPSBmLmdldEZ1bGxUZXh0KCkubGVuZ3RoO1xuXG4gICAgICAgICAgICBpbml0RlFOKGYsIGZteEZpbGUpO1xuXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihmLCBmbXhGaWxlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhGaWxlTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBmbXhGaWxlKTtcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhGaWxlKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteEZpbGUgPSBmb3VuZEZpbGVOYW1lO1xuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhGaWxlLGYpO1xuICAgICAgICByZXR1cm4gZm14RmlsZTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBNb2R1bGVcbiAgICAgKiBAcGFyYW0gbSBBIG1vZHVsZVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgbW9kdWxlXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhNb2R1bGUobTogTW9kdWxlRGVjbGFyYXRpb24pOiBGYW1peC5Nb2R1bGUge1xuICAgICAgICBsZXQgZm14TW9kdWxlOiBGYW1peC5Nb2R1bGU7XG4gICAgICAgIGNvbnN0IG1vZHVsZU5hbWUgPSBtLmdldE5hbWUoKTtcbiAgICAgICAgY29uc3QgZm91bmRNb2R1bGVOYW1lID0gdGhpcy5mbXhNb2R1bGVNYXAuZ2V0KG1vZHVsZU5hbWUpO1xuICAgICAgICBpZiAoIWZvdW5kTW9kdWxlTmFtZSkge1xuICAgICAgICAgICAgZm14TW9kdWxlID0gbmV3IEZhbWl4Lk1vZHVsZSgpO1xuICAgICAgICAgICAgZm14TW9kdWxlLm5hbWUgPSBtb2R1bGVOYW1lO1xuICAgICAgICAgICAgZm14TW9kdWxlLmlzQW1iaWVudCA9IGlzQW1iaWVudChtKTtcbiAgICAgICAgICAgIGZteE1vZHVsZS5pc05hbWVzcGFjZSA9IGlzTmFtZXNwYWNlKG0pO1xuICAgICAgICAgICAgZm14TW9kdWxlLmlzTW9kdWxlID0gIWZteE1vZHVsZS5pc05hbWVzcGFjZSAmJiAhZm14TW9kdWxlLmlzQW1iaWVudDtcblxuICAgICAgICAgICAgaW5pdEZRTihtLCBmbXhNb2R1bGUpO1xuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IobSwgZm14TW9kdWxlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhNb2R1bGVNYXAuc2V0KG1vZHVsZU5hbWUsIGZteE1vZHVsZSk7XG5cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhNb2R1bGUpO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14TW9kdWxlID0gZm91bmRNb2R1bGVOYW1lO1xuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhNb2R1bGUsbSk7XG4gICAgICAgIHJldHVybiBmbXhNb2R1bGU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGFsaWFzXG4gICAgICogQHBhcmFtIGEgQW4gYWxpYXNcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGFsaWFzXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4QWxpYXMoYTogVHlwZUFsaWFzRGVjbGFyYXRpb24pOiBGYW1peC5BbGlhcyB7XG4gICAgICAgIGxldCBmbXhBbGlhczogRmFtaXguQWxpYXM7XG4gICAgICAgIGNvbnN0IGFsaWFzTmFtZSA9IGEuZ2V0TmFtZSgpO1xuICAgICAgICBjb25zdCBhbGlhc0Z1bGx5UXVhbGlmaWVkTmFtZSA9IGEuZ2V0VHlwZSgpLmdldFRleHQoKTsgLy8gRlFORnVuY3Rpb25zLmdldEZRTihhKTtcbiAgICAgICAgY29uc3QgZm91bmRBbGlhcyA9IHRoaXMuZm14QWxpYXNNYXAuZ2V0KGFsaWFzRnVsbHlRdWFsaWZpZWROYW1lKTtcbiAgICAgICAgaWYgKCFmb3VuZEFsaWFzKSB7XG4gICAgICAgICAgICBmbXhBbGlhcyA9IG5ldyBGYW1peC5BbGlhcygpO1xuICAgICAgICAgICAgZm14QWxpYXMubmFtZSA9IGEuZ2V0TmFtZSgpO1xuICAgICAgICAgICAgY29uc3QgYWxpYXNOYW1lV2l0aEdlbmVyaWNzID0gYWxpYXNOYW1lICsgKGEuZ2V0VHlwZVBhcmFtZXRlcnMoKS5sZW5ndGggPyAoXCI8XCIgKyBhLmdldFR5cGVQYXJhbWV0ZXJzKCkubWFwKHRwID0+IHRwLmdldE5hbWUoKSkuam9pbihcIiwgXCIpICsgXCI+XCIpIDogXCJcIik7XG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoYD4gTk9URTogYWxpYXMgJHthbGlhc05hbWV9IGhhcyBmdWxseSBxdWFsaWZpZWQgbmFtZSAke2FsaWFzRnVsbHlRdWFsaWZpZWROYW1lfSBhbmQgbmFtZSB3aXRoIGdlbmVyaWNzICR7YWxpYXNOYW1lV2l0aEdlbmVyaWNzfS5gKTtcblxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoYWxpYXNOYW1lV2l0aEdlbmVyaWNzLCBhKTtcbiAgICAgICAgICAgIGZteEFsaWFzLmFsaWFzZWRFbnRpdHkgPSBmbXhUeXBlO1xuICAgICAgICAgICAgaW5pdEZRTihhLCBmbXhBbGlhcyk7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihhLCBmbXhBbGlhcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZm14QWxpYXNNYXAuc2V0KGFsaWFzRnVsbHlRdWFsaWZpZWROYW1lLCBmbXhBbGlhcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhBbGlhcyk7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICBmbXhBbGlhcyA9IGZvdW5kQWxpYXM7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhBbGlhcyxhKTtcblxuICAgICAgICByZXR1cm4gZm14QWxpYXM7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBvciBnZXRzIGEgRmFtaXggY2xhc3Mgb3IgcGFyYW1ldGVyaXphYmxlIGNsYXNzXG4gICAgICogQHBhcmFtIGNscyBBIGNsYXNzXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjbGFzc1xuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4Q2xhc3MoY2xzOiBDbGFzc0RlY2xhcmF0aW9uKTogRmFtaXguQ2xhc3MgfCBGYW1peC5QYXJhbWV0cmljQ2xhc3Mge1xuICAgICAgICBsZXQgZm14Q2xhc3M6IEZhbWl4LkNsYXNzIHwgRmFtaXguUGFyYW1ldHJpY0NsYXNzO1xuICAgICAgICBjb25zdCBpc0Fic3RyYWN0ID0gY2xzLmlzQWJzdHJhY3QoKTtcbiAgICAgICAgY29uc3QgY2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGNscyk7XG4gICAgICAgIGNvbnN0IGNsc05hbWUgPSBjbHMuZ2V0TmFtZSgpIHx8IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gY2xzLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoO1xuICAgICAgICBjb25zdCBmb3VuZENsYXNzID0gdGhpcy5mbXhDbGFzc01hcC5nZXQoY2xhc3NGdWxseVF1YWxpZmllZE5hbWUpO1xuICAgICAgICBpZiAoIWZvdW5kQ2xhc3MpIHtcbiAgICAgICAgICAgIGlmIChpc0dlbmVyaWMpIHtcbiAgICAgICAgICAgICAgICBmbXhDbGFzcyA9IG5ldyBGYW1peC5QYXJhbWV0cmljQ2xhc3MoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgIGZteENsYXNzID0gbmV3IEZhbWl4LkNsYXNzKCk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGZteENsYXNzLm5hbWUgPSBjbHNOYW1lO1xuICAgICAgICAgICAgZm14Q2xhc3MuZnVsbHlRdWFsaWZpZWROYW1lID0gY2xhc3NGdWxseVF1YWxpZmllZE5hbWU7XG4gICAgICAgICAgICBmbXhDbGFzcy5pc0Fic3RyYWN0ID0gaXNBYnN0cmFjdDtcblxuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IoY2xzLCBmbXhDbGFzcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZm14Q2xhc3NNYXAuc2V0KGNsYXNzRnVsbHlRdWFsaWZpZWROYW1lLCBmbXhDbGFzcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhDbGFzcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14Q2xhc3MsY2xzKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteENsYXNzID0gZm91bmRDbGFzcztcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBmbXhDbGFzcztcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBpbnRlcmZhY2Ugb3IgcGFyYW1ldGVyaXphYmxlIGludGVyZmFjZVxuICAgICAqIEBwYXJhbSBpbnRlciBBbiBpbnRlcmZhY2VcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGludGVyZmFjZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4SW50ZXJmYWNlKGludGVyOiBJbnRlcmZhY2VEZWNsYXJhdGlvbik6IEZhbWl4LkludGVyZmFjZSB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2Uge1xuXG4gICAgICAgIGxldCBmbXhJbnRlcmZhY2U6IEZhbWl4LkludGVyZmFjZSB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgIGNvbnN0IGludGVyTmFtZSA9IGludGVyLmdldE5hbWUoKTtcbiAgICAgICAgY29uc3QgaW50ZXJGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGludGVyKTtcbiAgICAgICAgY29uc3QgZm91bmRJbnRlcmZhY2UgPSB0aGlzLmZteEludGVyZmFjZU1hcC5nZXQoaW50ZXJGdWxseVF1YWxpZmllZE5hbWUpO1xuICAgICAgICBpZiAoIWZvdW5kSW50ZXJmYWNlKSB7XG4gICAgICAgICAgICBjb25zdCBpc0dlbmVyaWMgPSBpbnRlci5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aDtcbiAgICAgICAgICAgIGlmIChpc0dlbmVyaWMpIHtcbiAgICAgICAgICAgICAgICBmbXhJbnRlcmZhY2UgPSBuZXcgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSgpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14SW50ZXJmYWNlID0gbmV3IEZhbWl4LkludGVyZmFjZSgpO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBmbXhJbnRlcmZhY2UubmFtZSA9IGludGVyTmFtZTtcbiAgICAgICAgICAgIGluaXRGUU4oaW50ZXIsIGZteEludGVyZmFjZSk7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbnRlciwgZm14SW50ZXJmYWNlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhJbnRlcmZhY2VNYXAuc2V0KGludGVyRnVsbHlRdWFsaWZpZWROYW1lLCBmbXhJbnRlcmZhY2UpO1xuXG4gICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW50ZXJmYWNlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhJbnRlcmZhY2UsaW50ZXIpO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14SW50ZXJmYWNlID0gZm91bmRJbnRlcmZhY2U7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIGZteEludGVyZmFjZTtcbiAgICB9XG5cbiAgICBcbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBjb25jcmV0ZSBlbGVtZW50XG4gICAgICogQHBhcmFtIGNvbmNyZXRlRWxlbWVudCBBIHBhcmFtZXRyaWMgRWxlbWVudCAgIFxuICAgICAqIEBwYXJhbSBjb25jcmV0ZUVsZW1lbnREZWNsYXJhdGlvbiB0aGUgZWxlbWVudCBkZWNsYXJhdGlvblxuICAgICAqIEBwYXJhbSBjb25jcmV0ZUFyZ3VtZW50cyBjb25jcmV0ZSBhcmd1bWVudHNcbiAgICAgKiBAcmV0dXJucyBBIHBhcmFtZXRyaWMgRWxlbWVudCAgXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZUVsZW1lbnQoY29uY3JldGVFbGVtZW50IDogUGFyYW1ldHJpY1ZhcmlhbnRUeXBlLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0ZUVsZW1lbnREZWNsYXJhdGlvbiA6IENvbmNyZXRlRWxlbWVudFRTTW9ycGhUeXBlLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0ZUFyZ3VtZW50czogVHlwZU5vZGVbXSk6IFBhcmFtZXRyaWNWYXJpYW50VHlwZSB7XG4gICAgICAgIFxuICAgICAgICBsZXQgZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSA9IGNvbmNyZXRlRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWU7XG4gICAgICAgIGxldCBwYXJhbXMgPSBcIlwiO1xuICAgICAgICBcbiAgICAgICAgY29uY3JldGVBcmd1bWVudHMubWFwKChwYXJhbSkgPT4ge1xuICAgICAgICAgICAgcGFyYW1zID0gcGFyYW1zK3BhcmFtLmdldFRleHQoKSsnLCdcbiAgICAgICAgfSlcbiAgICAgICAgXG4gICAgICAgIHBhcmFtcyA9IHBhcmFtcy5zdWJzdHJpbmcoMCwgcGFyYW1zLmxlbmd0aCAtIDEpXG4gICAgICAgICAgICAgICAgXG4gICAgICAgIGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUgPSBIZWxwZXJzLnJlcGxhY2VMYXN0QmV0d2VlblRhZ3MoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSxwYXJhbXMpO1xuXG4gICAgICAgIGxldCBjb25jRWxlbWVudDogUGFyYW1ldHJpY1ZhcmlhbnRUeXBlO1xuXG4gICAgICAgIGlmICghdGhpcy5mbXhJbnRlcmZhY2VNYXAuaGFzKGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpICYmIFxuICAgICAgICAgICAgIXRoaXMuZm14Q2xhc3NNYXAuaGFzKGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpICYmIFxuICAgICAgICAgICAgIXRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuaGFzKGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpKXtcbiAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gXy5jbG9uZURlZXAoY29uY3JldGVFbGVtZW50KTsgXG4gICAgICAgICAgICBjb25jRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUgPSBmdWxseVF1YWxpZmllZEZpbGVuYW1lO1xuICAgICAgICAgICAgY29uY0VsZW1lbnQuY2xlYXJHZW5lcmljUGFyYW1ldGVycygpO1xuICAgICAgICAgICAgY29uY3JldGVBcmd1bWVudHMubWFwKChwYXJhbSkgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlciA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlVHlwZShwYXJhbSk7XG4gICAgICAgICAgICAgICAgY29uY0VsZW1lbnQuYWRkQ29uY3JldGVQYXJhbWV0ZXIocGFyYW1ldGVyKTtcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgICBcbiAgICAgICAgICAgIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljQ2xhc3MpIHtcbiAgICAgICAgICAgICAgICB0aGlzLmZteENsYXNzTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBjb25jRWxlbWVudCBhcyBGYW1peC5QYXJhbWV0cmljQ2xhc3MpO1xuICAgICAgICAgICAgfSBlbHNlIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlKSB7XG4gICAgICAgICAgICAgICAgdGhpcy5mbXhJbnRlcmZhY2VNYXAuc2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUsIGNvbmNFbGVtZW50IGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UpO1xuICAgICAgICAgICAgfSBlbHNlIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24pIHtcbiAgICAgICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBjb25jRWxlbWVudCBhcyBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24pO1xuICAgICAgICAgICAgfSBlbHNlIHsgLy8gaWYgKGNvbmNyZXRlRWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNNZXRob2QpIHtcbiAgICAgICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBjb25jRWxlbWVudCBhcyBGYW1peC5QYXJhbWV0cmljTWV0aG9kKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChjb25jRWxlbWVudCk7XG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGNvbmNFbGVtZW50LGNvbmNyZXRlRWxlbWVudERlY2xhcmF0aW9uKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljQ2xhc3MpIHtcbiAgICAgICAgICAgICAgICBjb25jRWxlbWVudCA9IHRoaXMuZm14Q2xhc3NNYXAuZ2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcbiAgICAgICAgICAgIH0gZWxzZSBpZiAoY29uY3JldGVFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSkge1xuICAgICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5mbXhJbnRlcmZhY2VNYXAuZ2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgICAgICB9IGVsc2UgaWYgKGNvbmNyZXRlRWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbikge1xuICAgICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5mbXhGdW5jdGlvbkFuZE1ldGhvZE1hcC5nZXQoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSkgYXMgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uO1xuICAgICAgICAgICAgfSBlbHNlIHsgIC8vIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljTWV0aG9kKSB7XG4gICAgICAgICAgICAgICAgY29uY0VsZW1lbnQgPSB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmdldChmdWxseVF1YWxpZmllZEZpbGVuYW1lKSBhcyBGYW1peC5QYXJhbWV0cmljTWV0aG9kO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBjb25jRWxlbWVudDtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggcHJvcGVydHlcbiAgICAgKiBAcGFyYW0gcHJvcGVydHkgQSBwcm9wZXJ0eVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgcHJvcGVydHlcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhQcm9wZXJ0eShwcm9wZXJ0eTogUHJvcGVydHlEZWNsYXJhdGlvbiB8IFByb3BlcnR5U2lnbmF0dXJlKTogRmFtaXguUHJvcGVydHkge1xuICAgICAgICBjb25zdCBmbXhQcm9wZXJ0eSA9IG5ldyBGYW1peC5Qcm9wZXJ0eSgpO1xuICAgICAgICBjb25zdCBpc1NpZ25hdHVyZSA9IHByb3BlcnR5IGluc3RhbmNlb2YgUHJvcGVydHlTaWduYXR1cmU7XG4gICAgICAgIGZteFByb3BlcnR5Lm5hbWUgPSBwcm9wZXJ0eS5nZXROYW1lKCk7XG5cbiAgICAgICAgbGV0IHByb3BUeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIHByb3BUeXBlTmFtZSA9IHByb3BlcnR5LmdldFR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHByb3BlcnR5OiAke3Byb3BlcnR5LmdldE5hbWUoKX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKHByb3BUeXBlTmFtZSwgcHJvcGVydHkpO1xuICAgICAgICBmbXhQcm9wZXJ0eS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuXG4gICAgICAgIC8vIGFkZCB0aGUgdmlzaWJpbGl0eSAocHVibGljLCBwcml2YXRlLCBldGMuKSB0byB0aGUgZm14UHJvcGVydHlcbiAgICAgICAgZm14UHJvcGVydHkudmlzaWJpbGl0eSA9IFwiXCI7XG5cbiAgICAgICAgcHJvcGVydHkuZ2V0TW9kaWZpZXJzKCkuZm9yRWFjaChtID0+IHtcbiAgICAgICAgICAgIHN3aXRjaCAobS5nZXRUZXh0KCkpIHtcbiAgICAgICAgICAgICAgICBjYXNlIFNjb3BlLlB1YmxpYzpcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkudmlzaWJpbGl0eSA9IFwicHVibGljXCI7XG4gICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIGNhc2UgU2NvcGUuUHJvdGVjdGVkOlxuICAgICAgICAgICAgICAgICAgICBmbXhQcm9wZXJ0eS52aXNpYmlsaXR5ID0gXCJwcm90ZWN0ZWRcIjtcbiAgICAgICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICAgICAgY2FzZSBTY29wZS5Qcml2YXRlOlxuICAgICAgICAgICAgICAgICAgICBmbXhQcm9wZXJ0eS52aXNpYmlsaXR5ID0gXCJwcml2YXRlXCI7XG4gICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIGNhc2UgXCJzdGF0aWNcIjpcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkuaXNDbGFzc1NpZGUgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgICAgICBjYXNlIFwicmVhZG9ubHlcIjpcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkucmVhZE9ubHkgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYgKCFpc1NpZ25hdHVyZSAmJiBwcm9wZXJ0eS5nZXRFeGNsYW1hdGlvblRva2VuTm9kZSgpKSB7XG4gICAgICAgICAgICBmbXhQcm9wZXJ0eS5pc0RlZmluaXRlbHlBc3NpZ25lZCA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHByb3BlcnR5LmdldFF1ZXN0aW9uVG9rZW5Ob2RlKCkpIHtcbiAgICAgICAgICAgIGZteFByb3BlcnR5LmlzT3B0aW9uYWwgPSB0cnVlO1xuICAgICAgICB9XG4gICAgICAgIGlmIChwcm9wZXJ0eS5nZXROYW1lKCkuc3Vic3RyaW5nKDAsIDEpID09PSBcIiNcIikge1xuICAgICAgICAgICAgZm14UHJvcGVydHkuaXNKYXZhU2NyaXB0UHJpdmF0ZSA9IHRydWU7XG4gICAgICAgIH1cblxuICAgICAgICBpbml0RlFOKHByb3BlcnR5LCBmbXhQcm9wZXJ0eSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHByb3BlcnR5LCBmbXhQcm9wZXJ0eSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFByb3BlcnR5KTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteFByb3BlcnR5LHByb3BlcnR5KTtcblxuICAgICAgICByZXR1cm4gZm14UHJvcGVydHk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IG1ldGhvZCBvciBhY2Nlc3NvclxuICAgICAqIEBwYXJhbSBtZXRob2QgQSBtZXRob2Qgb3IgYW4gYWNjZXNzb3JcbiAgICAgKiBAcGFyYW0gY3VycmVudENDIFRoZSBjeWNsb21hdGljIGNvbXBsZXhpdHkgbWV0cmljcyBvZiB0aGUgY3VycmVudCBzb3VyY2UgZmlsZVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgbWV0aG9kIG9yIHRoZSBhY2Nlc3NvclxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4TWV0aG9kKG1ldGhvZDogTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24sIGN1cnJlbnRDQzogeyBba2V5OiBzdHJpbmddOiBudW1iZXIgfSk6IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LkFjY2Vzc29yIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZCB7XG4gICAgICAgIGxldCBmbXhNZXRob2Q6IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LkFjY2Vzc29yIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZDtcbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gbWV0aG9kLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgY29uc3QgZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKG1ldGhvZCk7XG4gICAgICAgIGlmICghdGhpcy5mbXhGdW5jdGlvbkFuZE1ldGhvZE1hcC5oYXMoZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUpKSB7XG5cbiAgICAgICAgICAgIGlmIChtZXRob2QgaW5zdGFuY2VvZiBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHx8IG1ldGhvZCBpbnN0YW5jZW9mIFNldEFjY2Vzc29yRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QgPSBuZXcgRmFtaXguQWNjZXNzb3IoKTtcbiAgICAgICAgICAgICAgICBjb25zdCBpc0dldHRlciA9IG1ldGhvZCBpbnN0YW5jZW9mIEdldEFjY2Vzc29yRGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgY29uc3QgaXNTZXR0ZXIgPSBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uO1xuICAgICAgICAgICAgICAgIGlmIChpc0dldHRlcikgeyhmbXhNZXRob2QgYXMgRmFtaXguQWNjZXNzb3IpLmtpbmQgPSBcImdldHRlclwiO31cbiAgICAgICAgICAgICAgICBpZiAoaXNTZXR0ZXIpIHsoZm14TWV0aG9kIGFzIEZhbWl4LkFjY2Vzc29yKS5raW5kID0gXCJzZXR0ZXJcIjt9XG4gICAgICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteE1ldGhvZCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBpZiAoaXNHZW5lcmljKSB7XG4gICAgICAgICAgICAgICAgICAgIGZteE1ldGhvZCA9IG5ldyBGYW1peC5QYXJhbWV0cmljTWV0aG9kKCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBmbXhNZXRob2QgPSBuZXcgRmFtaXguTWV0aG9kKCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhNZXRob2QpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29uc3QgaXNDb25zdHJ1Y3RvciA9IG1ldGhvZCBpbnN0YW5jZW9mIENvbnN0cnVjdG9yRGVjbGFyYXRpb247XG4gICAgICAgICAgICBjb25zdCBpc1NpZ25hdHVyZSA9IG1ldGhvZCBpbnN0YW5jZW9mIE1ldGhvZFNpZ25hdHVyZTtcblxuICAgICAgICAgICAgbGV0IGlzQWJzdHJhY3QgPSBmYWxzZTtcbiAgICAgICAgICAgIGxldCBpc1N0YXRpYyA9IGZhbHNlO1xuICAgICAgICAgICAgaWYgKG1ldGhvZCBpbnN0YW5jZW9mIE1ldGhvZERlY2xhcmF0aW9uIHx8IG1ldGhvZCBpbnN0YW5jZW9mIEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgICAgIGlzQWJzdHJhY3QgPSBtZXRob2QuaXNBYnN0cmFjdCgpO1xuICAgICAgICAgICAgICAgIGlzU3RhdGljID0gbWV0aG9kLmlzU3RhdGljKCk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmIChpc0NvbnN0cnVjdG9yKSB7KGZteE1ldGhvZCBhcyBGYW1peC5BY2Nlc3Nvcikua2luZCA9IFwiY29uc3RydWN0b3JcIjt9XG4gICAgICAgICAgICBmbXhNZXRob2QuaXNBYnN0cmFjdCA9IGlzQWJzdHJhY3Q7XG4gICAgICAgICAgICBmbXhNZXRob2QuaXNDbGFzc1NpZGUgPSBpc1N0YXRpYztcbiAgICAgICAgICAgIGZteE1ldGhvZC5pc1ByaXZhdGUgPSAobWV0aG9kIGluc3RhbmNlb2YgTWV0aG9kRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fCBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKSA/IChtZXRob2QuZ2V0TW9kaWZpZXJzKCkuZmluZCh4ID0+IHguZ2V0VGV4dCgpID09PSAncHJpdmF0ZScpKSAhPT0gdW5kZWZpbmVkIDogZmFsc2U7XG4gICAgICAgICAgICBmbXhNZXRob2QuaXNQcm90ZWN0ZWQgPSAobWV0aG9kIGluc3RhbmNlb2YgTWV0aG9kRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fCBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKSA/IChtZXRob2QuZ2V0TW9kaWZpZXJzKCkuZmluZCh4ID0+IHguZ2V0VGV4dCgpID09PSAncHJvdGVjdGVkJykpICE9PSB1bmRlZmluZWQgOiBmYWxzZTtcbiAgICAgICAgICAgIGZteE1ldGhvZC5zaWduYXR1cmUgPSBIZWxwZXJzLmNvbXB1dGVTaWduYXR1cmUobWV0aG9kLmdldFRleHQoKSk7XG5cbiAgICAgICAgICAgIGxldCBtZXRob2ROYW1lOiBzdHJpbmc7XG4gICAgICAgICAgICBpZiAoaXNDb25zdHJ1Y3Rvcikge1xuICAgICAgICAgICAgICAgIG1ldGhvZE5hbWUgPSBcImNvbnN0cnVjdG9yXCI7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBtZXRob2ROYW1lID0gKG1ldGhvZCBhcyBNZXRob2REZWNsYXJhdGlvbiB8IE1ldGhvZFNpZ25hdHVyZSB8IEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKS5nZXROYW1lKCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBmbXhNZXRob2QubmFtZSA9IG1ldGhvZE5hbWU7XG5cbiAgICAgICAgICAgIGlmICghaXNDb25zdHJ1Y3Rvcikge1xuICAgICAgICAgICAgICAgIGlmIChtZXRob2QuZ2V0TmFtZSgpLnN1YnN0cmluZygwLCAxKSA9PT0gXCIjXCIpIHtcbiAgICAgICAgICAgICAgICAgICAgZm14TWV0aG9kLmlzUHJpdmF0ZSA9IHRydWU7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZiAoIWZteE1ldGhvZC5pc1ByaXZhdGUgJiYgIWZteE1ldGhvZC5pc1Byb3RlY3RlZCkge1xuICAgICAgICAgICAgICAgIGZteE1ldGhvZC5pc1B1YmxpYyA9IHRydWU7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QuaXNQdWJsaWMgPSBmYWxzZTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgaWYgKCFpc1NpZ25hdHVyZSkge1xuICAgICAgICAgICAgICAgIGZteE1ldGhvZC5jeWNsb21hdGljQ29tcGxleGl0eSA9IGN1cnJlbnRDQ1tmbXhNZXRob2QubmFtZV07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QuY3ljbG9tYXRpY0NvbXBsZXhpdHkgPSAwO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBsZXQgbWV0aG9kVHlwZU5hbWUgPSB0aGlzLlVOS05PV05fVkFMVUU7IFxuICAgICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgICAgICBtZXRob2RUeXBlTmFtZSA9IG1ldGhvZC5nZXRSZXR1cm5UeXBlKCkuZ2V0VGV4dCgpLnRyaW0oKTsgICAgICAgICAgICBcbiAgICAgICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHJldHVybiB0eXBlIG9mIG1ldGhvZDogJHtmbXhNZXRob2QubmFtZX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUobWV0aG9kVHlwZU5hbWUsIG1ldGhvZCk7XG4gICAgICAgICAgICBmbXhNZXRob2QuZGVjbGFyZWRUeXBlID0gZm14VHlwZTtcbiAgICAgICAgICAgIGZteE1ldGhvZC5udW1iZXJPZkxpbmVzT2ZDb2RlID0gbWV0aG9kLmdldEVuZExpbmVOdW1iZXIoKSAtIG1ldGhvZC5nZXRTdGFydExpbmVOdW1iZXIoKTtcbiAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPSBtZXRob2QuZ2V0UGFyYW1ldGVycygpO1xuICAgICAgICAgICAgZm14TWV0aG9kLm51bWJlck9mUGFyYW1ldGVycyA9IHBhcmFtZXRlcnMubGVuZ3RoO1xuXG4gICAgICAgICAgICBpZiAoIWlzU2lnbmF0dXJlKSB7XG4gICAgICAgICAgICAgICAgZm14TWV0aG9kLm51bWJlck9mU3RhdGVtZW50cyA9IG1ldGhvZC5nZXRTdGF0ZW1lbnRzKCkubGVuZ3RoO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14TWV0aG9kLm51bWJlck9mU3RhdGVtZW50cyA9IDA7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBcbiAgICAgICAgICAgIGluaXRGUU4obWV0aG9kLCBmbXhNZXRob2QpO1xuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IobWV0aG9kLCBmbXhNZXRob2QpO1xuXG4gICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSwgZm14TWV0aG9kKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteE1ldGhvZCA9IHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuZ2V0KGZ1bmN0aW9uRnVsbHlRdWFsaWZpZWROYW1lKSBhcyAoRmFtaXguTWV0aG9kIHwgRmFtaXguQWNjZXNzb3IgfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14TWV0aG9kLG1ldGhvZCk7XG4gICAgICAgIFxuICAgICAgICByZXR1cm4gZm14TWV0aG9kO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBmdW5jdGlvblxuICAgICAqIEBwYXJhbSBmdW5jIEEgZnVuY3Rpb25cbiAgICAgKiBAcGFyYW0gY3VycmVudENDIFRoZSBjeWNsb21hdGljIGNvbXBsZXhpdHkgbWV0cmljcyBvZiB0aGUgY3VycmVudCBzb3VyY2UgZmlsZVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgZnVuY3Rpb25cbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peEZ1bmN0aW9uKGZ1bmM6IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24sIGN1cnJlbnRDQzogeyBba2V5OiBzdHJpbmddOiBudW1iZXIgfSk6IEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uIHtcbiAgICAgICAgbGV0IGZteEZ1bmN0aW9uOiBGYW1peC5GdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbjtcbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gZnVuYy5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7ICAgICAgICBcbiAgICAgICAgY29uc3QgZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGZ1bmMpO1xuICAgICAgICBpZiAoIXRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuaGFzKGZ1bmN0aW9uRnVsbHlRdWFsaWZpZWROYW1lKSkge1xuICAgICAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgICAgIGZteEZ1bmN0aW9uID0gbmV3IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbigpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14RnVuY3Rpb24gPSBuZXcgRmFtaXguRnVuY3Rpb24oKTtcbiAgICAgICAgICAgIH1cbiAgICBcbiAgICAgICAgICAgIGNvbnN0IG5hbWUgPSBmdW5jLmdldE5hbWUoKTtcbiAgICAgICAgICAgIGlmIChuYW1lKSB7XG4gICAgICAgICAgICAgICAgZm14RnVuY3Rpb24ubmFtZSA9IG5hbWU7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBmbXhGdW5jdGlvbi5uYW1lID0gXCJhbm9ueW1vdXNcIjtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgZm14RnVuY3Rpb24uc2lnbmF0dXJlID0gSGVscGVycy5jb21wdXRlU2lnbmF0dXJlKGZ1bmMuZ2V0VGV4dCgpKTtcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLmN5Y2xvbWF0aWNDb21wbGV4aXR5ID0gY3VycmVudENDW2ZteEZ1bmN0aW9uLm5hbWVdO1xuICAgICAgICAgICAgZm14RnVuY3Rpb24uZnVsbHlRdWFsaWZpZWROYW1lID0gZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWU7XG4gICAgXG4gICAgICAgICAgICBsZXQgZnVuY3Rpb25UeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgICAgZnVuY3Rpb25UeXBlTmFtZSA9IGZ1bmMuZ2V0UmV0dXJuVHlwZSgpLmdldFRleHQoKS50cmltKCk7XG4gICAgICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgICAgIGxvZ2dlci5lcnJvcihgPiBXQVJOSU5HOiBnb3QgZXhjZXB0aW9uICR7ZXJyb3J9LiBGYWlsZWQgdG8gZ2V0IHVzYWJsZSBuYW1lIGZvciByZXR1cm4gdHlwZSBvZiBmdW5jdGlvbjogJHtmdW5jLmdldE5hbWUoKX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgICAgIH1cbiAgICBcbiAgICAgICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKGZ1bmN0aW9uVHlwZU5hbWUsIGZ1bmMpO1xuICAgICAgICAgICAgZm14RnVuY3Rpb24uZGVjbGFyZWRUeXBlID0gZm14VHlwZTtcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLm51bWJlck9mTGluZXNPZkNvZGUgPSBmdW5jLmdldEVuZExpbmVOdW1iZXIoKSAtIGZ1bmMuZ2V0U3RhcnRMaW5lTnVtYmVyKCk7XG4gICAgICAgICAgICBjb25zdCBwYXJhbWV0ZXJzID0gZnVuYy5nZXRQYXJhbWV0ZXJzKCk7XG4gICAgICAgICAgICBmbXhGdW5jdGlvbi5udW1iZXJPZlBhcmFtZXRlcnMgPSBwYXJhbWV0ZXJzLmxlbmd0aDtcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLm51bWJlck9mU3RhdGVtZW50cyA9IGZ1bmMuZ2V0U3RhdGVtZW50cygpLmxlbmd0aDtcbiAgICAgICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGZ1bmMsIGZteEZ1bmN0aW9uKTtcbiAgICBcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhGdW5jdGlvbik7XG4gICAgXG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEZ1bmN0aW9uLGZ1bmMpO1xuXG4gICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSwgZm14RnVuY3Rpb24pO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14RnVuY3Rpb24gPSB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmdldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgKEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBmbXhGdW5jdGlvbjtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggcGFyYW1ldGVyXG4gICAgICogQHBhcmFtIHBhcmFtIEEgcGFyYW1ldGVyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBwYXJhbWV0ZXJcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhQYXJhbWV0ZXIocGFyYW06IFBhcmFtZXRlckRlY2xhcmF0aW9uKTogRmFtaXguUGFyYW1ldGVyIHtcbiAgICAgICAgY29uc3QgZm14UGFyYW0gPSBuZXcgRmFtaXguUGFyYW1ldGVyKCk7XG5cbiAgICAgICAgbGV0IHBhcmFtVHlwZU5hbWUgPSB0aGlzLlVOS05PV05fVkFMVUU7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgICBwYXJhbVR5cGVOYW1lID0gcGFyYW0uZ2V0VHlwZSgpLmdldFRleHQoKS50cmltKCk7XG4gICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgICAgICBsb2dnZXIuZXJyb3IoYD4gV0FSTklORzogZ290IGV4Y2VwdGlvbiAke2Vycm9yfS4gRmFpbGVkIHRvIGdldCB1c2FibGUgbmFtZSBmb3IgcGFyYW1ldGVyOiAke3BhcmFtLmdldE5hbWUoKX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKHBhcmFtVHlwZU5hbWUsIHBhcmFtKTtcbiAgICAgICAgZm14UGFyYW0uZGVjbGFyZWRUeXBlID0gZm14VHlwZTtcbiAgICAgICAgZm14UGFyYW0ubmFtZSA9IHBhcmFtLmdldE5hbWUoKTtcblxuICAgICAgICBpbml0RlFOKHBhcmFtLCBmbXhQYXJhbSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHBhcmFtLCBmbXhQYXJhbSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtKTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteFBhcmFtLHBhcmFtKTtcblxuICAgICAgICByZXR1cm4gZm14UGFyYW07XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IHR5cGUgcGFyYW1ldGVyXG4gICAgICogQHBhcmFtIHRwIEEgdHlwZSBwYXJhbWV0ZXJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHR5cGUgcGFyYW1ldGVyXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4UGFyYW1ldGVyVHlwZSh0cDogVHlwZVBhcmFtZXRlckRlY2xhcmF0aW9uKTogRmFtaXguUGFyYW1ldGVyVHlwZSB7XG4gICAgICAgIFxuICAgICAgICBjb25zdCBmbXhQYXJhbWV0ZXJUeXBlID0gbmV3IEZhbWl4LlBhcmFtZXRlclR5cGUoKTtcbiAgIFxuICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlLm5hbWUgPSB0cC5nZXROYW1lKCk7ICAgICAgXG4gICAgICAgIGluaXRGUU4odHAsIGZteFBhcmFtZXRlclR5cGUpO1xuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcih0cCwgZm14UGFyYW1ldGVyVHlwZSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtZXRlclR5cGUpO1xuXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UGFyYW1ldGVyVHlwZSx0cCk7XG5cbiAgICAgICAgcmV0dXJuIGZteFBhcmFtZXRlclR5cGU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IHR5cGUgcGFyYW1ldGVyXG4gICAgICogQHBhcmFtIHRwIEEgdHlwZSBwYXJhbWV0ZXJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHR5cGUgcGFyYW1ldGVyXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZVR5cGUocGFyYW06IFR5cGVOb2RlKTogRmFtaXguUGFyYW1ldGVyVHlwZSB8IEZhbWl4LlByaW1pdGl2ZVR5cGUgfCBGYW1peC5DbGFzcyB8IEZhbWl4LkludGVyZmFjZSB7XG4gICAgICAgIGNvbnN0IHR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbiA9IHBhcmFtLmdldFN5bWJvbCgpPy5nZXREZWNsYXJhdGlvbnMoKVswXSBhcyBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb247XG4gICAgICAgIGNvbnN0IHBhcmFtZXRlclR5cGVOYW1lIDogc3RyaW5nID0gcGFyYW0uZ2V0VGV4dCgpO1xuICAgICAgICBsZXQgZm14UGFyYW1ldGVyVHlwZTogRmFtaXguVHlwZSB8IEZhbWl4LkNsYXNzIHwgRmFtaXguSW50ZXJmYWNlIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkO1xuXG4gICAgICAgIGxldCBpc0NsYXNzT3JJbnRlcmZhY2UgPSBmYWxzZTtcbiAgICAgICAgaWYgKHRoaXMuZm14Q2xhc3NNYXAuaGFzKHBhcmFtZXRlclR5cGVOYW1lKSl7XG4gICAgICAgICAgICB0aGlzLmZteENsYXNzTWFwLmZvckVhY2goKG9iaiwgbmFtZSkgPT4ge1xuICAgICAgICAgICAgICAgIGlmKG9iaiBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNDbGFzcyl7XG4gICAgICAgICAgICAgICAgICAgIGlmIChuYW1lID09PSBwYXJhbS5nZXRUZXh0KCkgJiYgb2JqLmdlbmVyaWNQYXJhbWV0ZXJzLnNpemU+MCkge1xuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IG9iajtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlzQ2xhc3NPckludGVyZmFjZSA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgIH0gXG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgaWYgKG5hbWUgPT09IHBhcmFtLmdldFRleHQoKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IG9iajtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlzQ2xhc3NPckludGVyZmFjZSA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgIH0gXG4gICAgICAgICAgICAgICAgfSAgIFxuICAgICAgICAgICAgfSlcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh0aGlzLmZteEludGVyZmFjZU1hcC5oYXMocGFyYW1ldGVyVHlwZU5hbWUpKXtcbiAgICAgICAgICAgIHRoaXMuZm14SW50ZXJmYWNlTWFwLmZvckVhY2goKG9iaiwgbmFtZSkgPT4ge1xuICAgICAgICAgICAgICAgIGlmKG9iaiBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2Upe1xuICAgICAgICAgICAgICAgICAgICBpZiAobmFtZSA9PT0gcGFyYW0uZ2V0VGV4dCgpICYmIG9iai5nZW5lcmljUGFyYW1ldGVycy5zaXplPjApIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlclR5cGUgPSBvYmo7XG4gICAgICAgICAgICAgICAgICAgICAgICBpc0NsYXNzT3JJbnRlcmZhY2UgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICB9IFxuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIGlmIChuYW1lID09PSBwYXJhbS5nZXRUZXh0KCkpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlclR5cGUgPSBvYmo7XG4gICAgICAgICAgICAgICAgICAgICAgICBpc0NsYXNzT3JJbnRlcmZhY2UgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICB9IFxuICAgICAgICAgICAgICAgIH0gICBcbiAgICAgICAgICAgIH0pXG4gICAgICAgIH1cblxuICAgICAgICBpZighaXNDbGFzc09ySW50ZXJmYWNlKXtcbiAgICAgICAgICAgIGlmICghdGhpcy5mbXhUeXBlTWFwLmhhcyhwYXJhbWV0ZXJUeXBlTmFtZSkpIHsgICAgICAgICAgIFxuICAgICAgICAgICAgICAgIGlmIChwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJudW1iZXJcIiB8fCBwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJzdHJpbmdcIiB8fCBwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJib29sZWFuXCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwiYmlnaW50XCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwic3ltYm9sXCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwidW5kZWZpbmVkXCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwibnVsbFwiIHx8IHBhcmFtZXRlclR5cGVOYW1lID09PSBcImFueVwiIHx8IHBhcmFtZXRlclR5cGVOYW1lID09PSBcInVua25vd25cIiB8fCBwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJuZXZlclwiIHx8IHBhcmFtZXRlclR5cGVOYW1lID09PSBcInZvaWRcIikge1xuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlID0gbmV3IEZhbWl4LlByaW1pdGl2ZVR5cGUoKTtcbiAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZS5pc1N0dWIgPSB0cnVlO1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlclR5cGUgPSBuZXcgRmFtaXguUGFyYW1ldGVyVHlwZSgpO1xuICAgICAgICAgICAgICAgIH0gXG4gICAgXG4gICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZS5uYW1lID0gcGFyYW1ldGVyVHlwZU5hbWU7XG4gICAgICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtZXRlclR5cGUpO1xuICAgICAgICAgICAgICAgIHRoaXMuZm14VHlwZU1hcC5zZXQocGFyYW1ldGVyVHlwZU5hbWUsIGZteFBhcmFtZXRlclR5cGUpO1xuICAgICAgICAgICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UGFyYW1ldGVyVHlwZSx0eXBlUGFyYW1ldGVyRGVjbGFyYXRpb24pO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gdGhpcy5mbXhUeXBlTWFwLmdldChwYXJhbWV0ZXJUeXBlTmFtZSk7XG4gICAgICAgICAgICAgICAgaWYgKHJlc3VsdCkge1xuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlID0gcmVzdWx0O1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgRmFtaXggdHlwZSAke3BhcmFtZXRlclR5cGVOYW1lfSBpcyBub3QgZm91bmQgaW4gdGhlIFR5cGUgbWFwLmApO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICBpZiAoIWZteFBhcmFtZXRlclR5cGUpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgZm14UGFyYW1ldGVyVHlwZSB3YXMgdW5kZWZpbmVkIGZvciBwYXJhbWV0ZXJUeXBlTmFtZSAke3BhcmFtZXRlclR5cGVOYW1lfWApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBmbXhQYXJhbWV0ZXJUeXBlO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCB2YXJpYWJsZVxuICAgICAqIEBwYXJhbSB2YXJpYWJsZSBBIHZhcmlhYmxlXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSB2YXJpYWJsZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peFZhcmlhYmxlKHZhcmlhYmxlOiBWYXJpYWJsZURlY2xhcmF0aW9uKTogRmFtaXguVmFyaWFibGUge1xuICAgICAgICBjb25zdCBmbXhWYXJpYWJsZSA9IG5ldyBGYW1peC5WYXJpYWJsZSgpO1xuICAgIFxuICAgICAgICBsZXQgdmFyaWFibGVUeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIHZhcmlhYmxlVHlwZU5hbWUgPSB2YXJpYWJsZS5nZXRUeXBlKCkuZ2V0VGV4dCgpLnRyaW0oKTtcbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAgIGxvZ2dlci5lcnJvcihgPiBXQVJOSU5HOiBnb3QgZXhjZXB0aW9uICR7ZXJyb3J9LiBGYWlsZWQgdG8gZ2V0IHVzYWJsZSBuYW1lIGZvciB2YXJpYWJsZTogJHt2YXJpYWJsZS5nZXROYW1lKCl9LiBDb250aW51aW5nLi4uYCk7XG4gICAgICAgIH1cbiAgICBcbiAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUodmFyaWFibGVUeXBlTmFtZSwgdmFyaWFibGUpO1xuICAgICAgICBmbXhWYXJpYWJsZS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuICAgICAgICBmbXhWYXJpYWJsZS5uYW1lID0gdmFyaWFibGUuZ2V0TmFtZSgpO1xuICAgICAgICBpbml0RlFOKHZhcmlhYmxlLCBmbXhWYXJpYWJsZSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHZhcmlhYmxlLCBmbXhWYXJpYWJsZSk7XG4gICAgXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhWYXJpYWJsZSk7XG4gICAgXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14VmFyaWFibGUsdmFyaWFibGUpO1xuICAgIFxuICAgICAgICByZXR1cm4gZm14VmFyaWFibGU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGVudW1cbiAgICAgKiBAcGFyYW0gZW51bUVudGl0eSBBbiBlbnVtXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBlbnVtXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4RW51bShlbnVtRW50aXR5OiBFbnVtRGVjbGFyYXRpb24pOiBGYW1peC5FbnVtIHtcbiAgICAgICAgY29uc3QgZm14RW51bSA9IG5ldyBGYW1peC5FbnVtKCk7XG4gICAgICAgIGZteEVudW0ubmFtZSA9IGVudW1FbnRpdHkuZ2V0TmFtZSgpO1xuICAgICAgICBpbml0RlFOKGVudW1FbnRpdHksIGZteEVudW0pO1xuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihlbnVtRW50aXR5LCBmbXhFbnVtKTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14RW51bSk7XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhFbnVtLGVudW1FbnRpdHkpO1xuXG4gICAgICAgIHJldHVybiBmbXhFbnVtO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBlbnVtIHZhbHVlXG4gICAgICogQHBhcmFtIGVudW1NZW1iZXIgQW4gZW51bSBtZW1iZXJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGVudW0gbWVtYmVyXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4RW51bVZhbHVlKGVudW1NZW1iZXI6IEVudW1NZW1iZXIpOiBGYW1peC5FbnVtVmFsdWUge1xuICAgICAgICBjb25zdCBmbXhFbnVtVmFsdWUgPSBuZXcgRmFtaXguRW51bVZhbHVlKCk7XG5cbiAgICAgICAgbGV0IGVudW1WYWx1ZVR5cGVOYW1lID0gdGhpcy5VTktOT1dOX1ZBTFVFO1xuICAgICAgICB0cnkge1xuICAgICAgICAgICAgZW51bVZhbHVlVHlwZU5hbWUgPSBlbnVtTWVtYmVyLmdldFR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIGVudW0gdmFsdWU6ICR7ZW51bU1lbWJlci5nZXROYW1lKCl9LiBDb250aW51aW5nLi4uYCk7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBmbXhUeXBlID0gdGhpcy5jcmVhdGVPckdldEZhbWl4VHlwZShlbnVtVmFsdWVUeXBlTmFtZSwgZW51bU1lbWJlcik7XG4gICAgICAgIGZteEVudW1WYWx1ZS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuICAgICAgICBmbXhFbnVtVmFsdWUubmFtZSA9IGVudW1NZW1iZXIuZ2V0TmFtZSgpO1xuICAgICAgICBpbml0RlFOKGVudW1NZW1iZXIsIGZteEVudW1WYWx1ZSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGVudW1NZW1iZXIsIGZteEVudW1WYWx1ZSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEVudW1WYWx1ZSk7XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhFbnVtVmFsdWUsZW51bU1lbWJlcik7XG5cbiAgICAgICAgcmV0dXJuIGZteEVudW1WYWx1ZTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBkZWNvcmF0b3JcbiAgICAgKiBAcGFyYW0gZGVjb3JhdG9yIEEgZGVjb3JhdG9yXG4gICAgICogQHBhcmFtIGRlY29yYXRlZEVudGl0eSBBIGNsYXNzLCBhIG1ldGhvZCwgYSBwYXJhbWV0ZXIgb3IgYSBwcm9wZXJ0eVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgZGVjb3JhdG9yXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhEZWNvcmF0b3IoZGVjb3JhdG9yOiBEZWNvcmF0b3IsIGRlY29yYXRlZEVudGl0eTogQ2xhc3NEZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFByb3BlcnR5RGVjbGFyYXRpb24pOiBGYW1peC5EZWNvcmF0b3Ige1xuICAgICAgICBjb25zdCBmbXhEZWNvcmF0b3IgPSBuZXcgRmFtaXguRGVjb3JhdG9yKCk7XG4gICAgICAgIGNvbnN0IGRlY29yYXRvck5hbWUgPSBcIkBcIiArIGRlY29yYXRvci5nZXROYW1lKCk7XG4gICAgICAgIGNvbnN0IGRlY29yYXRvckV4cHJlc3Npb24gPSBkZWNvcmF0b3IuZ2V0VGV4dCgpLnN1YnN0cmluZygxKTtcblxuICAgICAgICBmbXhEZWNvcmF0b3IubmFtZSA9IGRlY29yYXRvck5hbWU7XG4gICAgICAgIGZteERlY29yYXRvci5kZWNvcmF0b3JFeHByZXNzaW9uID0gZGVjb3JhdG9yRXhwcmVzc2lvbjtcbiAgICAgICAgY29uc3QgZGVjb3JhdGVkRW50aXR5RnVsbHlRdWFsaWZpZWROYW1lID0gRlFORnVuY3Rpb25zLmdldEZRTihkZWNvcmF0ZWRFbnRpdHkpO1xuICAgICAgICBjb25zdCBmbXhEZWNvcmF0ZWRFbnRpdHkgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUoZGVjb3JhdGVkRW50aXR5RnVsbHlRdWFsaWZpZWROYW1lKSBhcyBGYW1peC5OYW1lZEVudGl0eTtcbiAgICAgICAgZm14RGVjb3JhdG9yLmRlY29yYXRlZEVudGl0eSA9IGZteERlY29yYXRlZEVudGl0eTtcbiAgICAgICAgaW5pdEZRTihkZWNvcmF0b3IsIGZteERlY29yYXRvcik7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGRlY29yYXRvciwgZm14RGVjb3JhdG9yKTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14RGVjb3JhdG9yKTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteERlY29yYXRvcixkZWNvcmF0b3IpO1xuXG4gICAgICAgIHJldHVybiBmbXhEZWNvcmF0b3I7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGNvbW1lbnRcbiAgICAgKiBAcGFyYW0gY29tbWVudCBBIGNvbW1lbnRcbiAgICAgKiBAcGFyYW0gZm14U2NvcGUgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjb21tZW50J3MgY29udGFpbmVyXG4gICAgICogQHBhcmFtIGlzSlNEb2MgQSBib29sZWFuIGluZGljYXRpbmcgaWYgdGhlIGNvbW1lbnQgaXMgYSBKU0RvY1xuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgY29tbWVudFxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbW1lbnQoY29tbWVudDogQ29tbWVudFJhbmdlLCBmbXhTY29wZTogRmFtaXguTmFtZWRFbnRpdHksIGlzSlNEb2M6IGJvb2xlYW4pOiBGYW1peC5Db21tZW50IHtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGA+IE5PVEU6IGNyZWF0aW5nIGNvbW1lbnQgJHtjb21tZW50LmdldFRleHQoKX0gaW4gc2NvcGUgJHtmbXhTY29wZS5uYW1lfS5gKTtcbiAgICAgICAgY29uc3QgZm14Q29tbWVudCA9IG5ldyBGYW1peC5Db21tZW50KCk7XG4gICAgICAgIGZteENvbW1lbnQuY29udGFpbmVyID0gZm14U2NvcGU7ICAvLyBhZGRzIGNvbW1lbnQgdG8gdGhlIGNvbnRhaW5lcidzIGNvbW1lbnRzIGNvbGxlY3Rpb25cbiAgICAgICAgZm14Q29tbWVudC5pc0pTRG9jID0gaXNKU0RvYztcblxuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihjb21tZW50LCBmbXhDb21tZW50KTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14Q29tbWVudCk7XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhDb21tZW50LGNvbW1lbnQpO1xuXG4gICAgICAgIHJldHVybiBmbXhDb21tZW50O1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgb3IgZ2V0cyBhIEZhbWl4IHR5cGVcbiAgICAgKiBAcGFyYW0gdHlwZU5hbWUgQSB0eXBlIG5hbWVcbiAgICAgKiBAcGFyYW0gZWxlbWVudCBBIHRzLW1vcnBoIGVsZW1lbnRcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHR5cGVcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peFR5cGUodHlwZU5hbWU6IHN0cmluZywgZWxlbWVudDogVHlwZURlY2xhcmF0aW9uKTogRmFtaXguVHlwZSB8IEZhbWl4LlByaW1pdGl2ZVR5cGUgfCBGYW1peC5QYXJhbWV0ZXJUeXBlIHtcbiAgICAgICAgbGV0IGZteFR5cGU6IEZhbWl4LlR5cGUgfCBGYW1peC5QcmltaXRpdmVUeXBlIHwgRmFtaXguUGFyYW1ldGVyVHlwZTtcbiAgICAgICAgbGV0IGlzUHJpbWl0aXZlVHlwZSA9IGZhbHNlO1xuICAgICAgICBsZXQgaXNQYXJhbWV0ZXJUeXBlID0gZmFsc2U7XG5cbiAgICAgICAgbG9nZ2VyLmRlYnVnKFwiQ3JlYXRpbmcgKG9yIGdldHRpbmcpIHR5cGU6ICdcIiArIHR5cGVOYW1lICsgXCInIG9mIGVsZW1lbnQ6IFwiICsgZWxlbWVudD8uZ2V0VGV4dCgpICsgXCIgb2Yga2luZDogXCIgKyBlbGVtZW50Py5nZXRLaW5kTmFtZSgpKTtcbiAgICAgICAgbGV0IGFuY2VzdG9yOiBGYW1peC5Db250YWluZXJFbnRpdHkgfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ7XG4gICAgICAgIGlmIChlbGVtZW50ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHR5cGVBbmNlc3RvciA9IEhlbHBlcnMuZmluZFR5cGVBbmNlc3RvcihlbGVtZW50KTtcbiAgICAgICAgICAgIGlmICghdHlwZUFuY2VzdG9yKSB7XG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBBbmNlc3RvciBub3QgZm91bmQgZm9yIGVsZW1lbnQgJHtlbGVtZW50LmdldFRleHQoKX0uYCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjb25zdCBhbmNlc3RvckZ1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4odHlwZUFuY2VzdG9yKTtcbiAgICAgICAgICAgIGFuY2VzdG9yID0gdGhpcy5mYW1peFJlcC5nZXRGYW1peEVudGl0eUJ5RnVsbHlRdWFsaWZpZWROYW1lKGFuY2VzdG9yRnVsbHlRdWFsaWZpZWROYW1lKSBhcyBGYW1peC5Db250YWluZXJFbnRpdHk7XG4gICAgICAgICAgICBpZiAoIWFuY2VzdG9yKSB7XG4gICAgICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBBbmNlc3RvciAke0ZRTkZ1bmN0aW9ucy5nZXRGUU4odHlwZUFuY2VzdG9yKX0gbm90IGZvdW5kLiBBZGRpbmcgdGhlIG5ldyB0eXBlLmApO1xuICAgICAgICAgICAgICAgIGFuY2VzdG9yID0gdGhpcy5jcmVhdGVPckdldEZhbWl4VHlwZSh0eXBlQW5jZXN0b3IuZ2V0VGV4dCgpLCB0eXBlQW5jZXN0b3IgYXMgVHlwZURlY2xhcmF0aW9uKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh0eXBlTmFtZSA9PT0gXCJudW1iZXJcIiB8fCB0eXBlTmFtZSA9PT0gXCJzdHJpbmdcIiB8fCB0eXBlTmFtZSA9PT0gXCJib29sZWFuXCIgfHwgdHlwZU5hbWUgPT09IFwiYmlnaW50XCIgfHwgdHlwZU5hbWUgPT09IFwic3ltYm9sXCIgfHwgdHlwZU5hbWUgPT09IFwidW5kZWZpbmVkXCIgfHwgdHlwZU5hbWUgPT09IFwibnVsbFwiIHx8IHR5cGVOYW1lID09PSBcImFueVwiIHx8IHR5cGVOYW1lID09PSBcInVua25vd25cIiB8fCB0eXBlTmFtZSA9PT0gXCJuZXZlclwiIHx8IHR5cGVOYW1lID09PSBcInZvaWRcIikge1xuICAgICAgICAgICAgaXNQcmltaXRpdmVUeXBlID0gdHJ1ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmKCFpc1ByaW1pdGl2ZVR5cGUgJiYgdHlwZU5hbWUuaW5jbHVkZXMoXCI8XCIpICYmIHR5cGVOYW1lLmluY2x1ZGVzKFwiPlwiKSAmJiAhKHR5cGVOYW1lLmluY2x1ZGVzKFwiPT5cIikpKSB7XG4gICAgICAgICAgICBpc1BhcmFtZXRlclR5cGUgPSB0cnVlO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKCF0aGlzLmZteFR5cGVNYXAuaGFzKHR5cGVOYW1lKSkge1xuICAgICAgICAgICAgaWYgKGlzUHJpbWl0aXZlVHlwZSkge1xuICAgICAgICAgICAgICAgIGZteFR5cGUgPSBuZXcgRmFtaXguUHJpbWl0aXZlVHlwZSgpO1xuICAgICAgICAgICAgICAgIGZteFR5cGUuaXNTdHViID0gdHJ1ZTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2UgaWYgKGlzUGFyYW1ldGVyVHlwZSkge1xuICAgICAgICAgICAgICAgIGZteFR5cGUgPSBuZXcgRmFtaXguUGFyYW1ldGVyVHlwZSgpO1xuICAgICAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlclR5cGVOYW1lcyA9IHR5cGVOYW1lLnN1YnN0cmluZyh0eXBlTmFtZS5pbmRleE9mKFwiPFwiKSArIDEsIHR5cGVOYW1lLmluZGV4T2YoXCI+XCIpKS5zcGxpdChcIixcIikubWFwKHMgPT4gcy50cmltKCkpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGJhc2VUeXBlTmFtZSA9IHR5cGVOYW1lLnN1YnN0cmluZygwLCB0eXBlTmFtZS5pbmRleE9mKFwiPFwiKSkudHJpbSgpO1xuICAgICAgICAgICAgICAgIHBhcmFtZXRlclR5cGVOYW1lcy5mb3JFYWNoKHBhcmFtZXRlclR5cGVOYW1lID0+IHtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZm14UGFyYW1ldGVyVHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUocGFyYW1ldGVyVHlwZU5hbWUsIGVsZW1lbnQpO1xuICAgICAgICAgICAgICAgICAgICAoZm14VHlwZSBhcyBGYW1peC5QYXJhbWV0ZXJUeXBlKS5hZGRBcmd1bWVudChmbXhQYXJhbWV0ZXJUeXBlKTtcbiAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICBjb25zdCBmbXhCYXNlVHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoYmFzZVR5cGVOYW1lLCBlbGVtZW50KTtcbiAgICAgICAgICAgICAgICAoZm14VHlwZSBhcyBGYW1peC5QYXJhbWV0ZXJUeXBlKS5iYXNlVHlwZSA9IGZteEJhc2VUeXBlO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14VHlwZSA9IG5ldyBGYW1peC5UeXBlKCk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGZteFR5cGUubmFtZSA9IHR5cGVOYW1lO1xuICAgICAgICAgICAgaWYgKCFhbmNlc3Rvcikge1xuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgQW5jZXN0b3Igbm90IGZvdW5kIGZvciB0eXBlICR7dHlwZU5hbWV9LmApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZm14VHlwZS5jb250YWluZXIgPSBhbmNlc3RvcjtcbiAgICAgICAgICAgIGluaXRGUU4oZWxlbWVudCwgZm14VHlwZSk7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihlbGVtZW50LCBmbXhUeXBlKTtcblxuICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFR5cGUpO1xuXG4gICAgICAgICAgICB0aGlzLmZteFR5cGVNYXAuc2V0KHR5cGVOYW1lLCBmbXhUeXBlKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGNvbnN0IHJlc3VsdCA9IHRoaXMuZm14VHlwZU1hcC5nZXQodHlwZU5hbWUpO1xuICAgICAgICAgICAgaWYgKHJlc3VsdCkge1xuICAgICAgICAgICAgICAgIGZteFR5cGUgPSByZXN1bHQ7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgRmFtaXggdHlwZSAke3R5cGVOYW1lfSBpcyBub3QgZm91bmQgaW4gdGhlIFR5cGUgbWFwLmApO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhUeXBlLGVsZW1lbnQpO1xuXG4gICAgICAgIHJldHVybiBmbXhUeXBlO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBhY2Nlc3NcbiAgICAgKiBAcGFyYW0gbm9kZSBBIG5vZGVcbiAgICAgKiBAcGFyYW0gaWQgQW4gaWQgb2YgYSBwYXJhbWV0ZXIsIGEgdmFyaWFibGUsIGEgcHJvcGVydHkgb3IgYW4gZW51bSBtZW1iZXJcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhBY2Nlc3Mobm9kZTogSWRlbnRpZmllciwgaWQ6IG51bWJlcik6IHZvaWQge1xuICAgICAgICBjb25zdCBmbXhWYXIgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlJZChpZCkgYXMgRmFtaXguU3RydWN0dXJhbEVudGl0eTtcbiAgICAgICAgaWYgKCFmbXhWYXIpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgRmFtaXggZW50aXR5IHdpdGggaWQgJHtpZH0gbm90IGZvdW5kLCBmb3Igbm9kZSAke25vZGUuZ2V0VGV4dCgpfSBpbiAke25vZGUuZ2V0U291cmNlRmlsZSgpLmdldEJhc2VOYW1lKCl9IGF0IGxpbmUgJHtub2RlLmdldFN0YXJ0TGluZU51bWJlcigpfS5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgQ3JlYXRpbmcgRmFtaXhBY2Nlc3MuIE5vZGU6IFske25vZGUuZ2V0S2luZE5hbWUoKX1dICcke25vZGUuZ2V0VGV4dCgpfScgYXQgbGluZSAke25vZGUuZ2V0U3RhcnRMaW5lTnVtYmVyKCl9IGluICR7bm9kZS5nZXRTb3VyY2VGaWxlKCkuZ2V0QmFzZU5hbWUoKX0sIGlkOiAke2lkfSByZWZlcnMgdG8gZm14VmFyICcke2ZteFZhci5mdWxseVF1YWxpZmllZE5hbWV9Jy5gKTtcblxuICAgICAgICBjb25zdCBub2RlUmVmZXJlbmNlQW5jZXN0b3IgPSBIZWxwZXJzLmZpbmRBbmNlc3Rvcihub2RlKTtcbiAgICAgICAgY29uc3QgYW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKG5vZGVSZWZlcmVuY2VBbmNlc3Rvcik7XG4gICAgICAgIGxldCBhY2Nlc3NvciA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUZ1bGx5UXVhbGlmaWVkTmFtZShhbmNlc3RvckZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgRmFtaXguQ29udGFpbmVyRW50aXR5O1xuICAgICAgICBpZiAoIWFjY2Vzc29yKSB7XG4gICAgICAgICAgICBsb2dnZXIuZXJyb3IoYEFuY2VzdG9yICR7YW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWV9IG9mIGtpbmQgJHtub2RlUmVmZXJlbmNlQW5jZXN0b3IuZ2V0S2luZE5hbWUoKX0gbm90IGZvdW5kLmApO1xuICAgICAgICAgICAgLy8gYWNjZXNzb3IgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKGFuY2VzdG9yRnVsbHlRdWFsaWZpZWROYW1lLCBub2RlUmVmZXJlbmNlQW5jZXN0b3IgYXMgVHlwZURlY2xhcmF0aW9uKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteEFjY2VzcyA9IG5ldyBGYW1peC5BY2Nlc3MoKTtcbiAgICAgICAgZm14QWNjZXNzLmFjY2Vzc29yID0gYWNjZXNzb3I7XG4gICAgICAgIGZteEFjY2Vzcy52YXJpYWJsZSA9IGZteFZhcjtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14QWNjZXNzKTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEFjY2Vzcyxub2RlKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW52b2NhdGlvblxuICAgICAqIEBwYXJhbSBub2RlIEEgbm9kZVxuICAgICAqIEBwYXJhbSBtIEEgbWV0aG9kIG9yIGEgZnVuY3Rpb25cbiAgICAgKiBAcGFyYW0gaWQgVGhlIGlkIG9mIHRoZSBtZXRob2Qgb3IgdGhlIGZ1bmN0aW9uXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4SW52b2NhdGlvbihub2RlOiBJZGVudGlmaWVyLCBtOiBNZXRob2REZWNsYXJhdGlvbiB8IENvbnN0cnVjdG9yRGVjbGFyYXRpb24gfCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24sIGlkOiBudW1iZXIpOiB2b2lkIHtcbiAgICAgICAgY29uc3QgZm14TWV0aG9kT3JGdW5jdGlvbiA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUlkKGlkKSBhcyBGYW1peC5CZWhhdmlvcmFsRW50aXR5O1xuICAgICAgICBjb25zdCBub2RlUmVmZXJlbmNlQW5jZXN0b3IgPSBIZWxwZXJzLmZpbmRBbmNlc3Rvcihub2RlKTtcbiAgICAgICAgY29uc3QgYW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKG5vZGVSZWZlcmVuY2VBbmNlc3Rvcik7XG4gICAgICAgIGNvbnN0IHNlbmRlciA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUZ1bGx5UXVhbGlmaWVkTmFtZShhbmNlc3RvckZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgRmFtaXguQ29udGFpbmVyRW50aXR5O1xuICAgICAgICBjb25zdCByZWNlaXZlckZ1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4obS5nZXRQYXJlbnQoKSk7XG4gICAgICAgIGNvbnN0IHJlY2VpdmVyID0gdGhpcy5mYW1peFJlcC5nZXRGYW1peEVudGl0eUJ5RnVsbHlRdWFsaWZpZWROYW1lKHJlY2VpdmVyRnVsbHlRdWFsaWZpZWROYW1lKSBhcyBGYW1peC5OYW1lZEVudGl0eTtcblxuICAgICAgICBjb25zdCBmbXhJbnZvY2F0aW9uID0gbmV3IEZhbWl4Lkludm9jYXRpb24oKTtcbiAgICAgICAgZm14SW52b2NhdGlvbi5zZW5kZXIgPSBzZW5kZXI7XG4gICAgICAgIGZteEludm9jYXRpb24ucmVjZWl2ZXIgPSByZWNlaXZlcjtcbiAgICAgICAgZm14SW52b2NhdGlvbi5hZGRDYW5kaWRhdGUoZm14TWV0aG9kT3JGdW5jdGlvbik7XG4gICAgICAgIGZteEludm9jYXRpb24uc2lnbmF0dXJlID0gZm14TWV0aG9kT3JGdW5jdGlvbi5zaWduYXR1cmU7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEludm9jYXRpb24pO1xuXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14SW52b2NhdGlvbixub2RlKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW5oZXJpdGFuY2VcbiAgICAgKiBAcGFyYW0gY2xzIEEgY2xhc3Mgb3IgYW4gaW50ZXJmYWNlIChzdWJjbGFzcylcbiAgICAgKiBAcGFyYW0gaW5oQ2xhc3MgVGhlIGluaGVyaXRlZCBjbGFzcyBvciBpbnRlcmZhY2UgKHN1cGVyY2xhc3MpXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4SW5oZXJpdGFuY2UoY2xzOiBDbGFzc0RlY2xhcmF0aW9uIHwgSW50ZXJmYWNlRGVjbGFyYXRpb24sIGluaENsYXNzOiBDbGFzc0RlY2xhcmF0aW9uIHwgSW50ZXJmYWNlRGVjbGFyYXRpb24gfCBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHMpOiB2b2lkIHtcbiAgICAgICAgY29uc3QgZm14SW5oZXJpdGFuY2UgPSBuZXcgRmFtaXguSW5oZXJpdGFuY2UoKTtcbiAgICAgICAgLy8gY29uc3QgY2xzTmFtZSA9IGNscy5nZXROYW1lKCk7XG4gICAgICAgIGNvbnN0IGNsYXNzRnVsbHlRdWFsaWZpZWROYW1lID0gRlFORnVuY3Rpb25zLmdldEZRTihjbHMpO1xuICAgICAgICBsb2dnZXIuZGVidWcoYGNyZWF0ZUZhbWl4SW5oZXJpdGFuY2U6IGNsYXNzRnVsbHlRdWFsaWZpZWROYW1lOiBjbGFzcyBmcW4gPSAke2NsYXNzRnVsbHlRdWFsaWZpZWROYW1lfWApO1xuICAgICAgICBsZXQgc3ViQ2xhc3M6IEZhbWl4LkNsYXNzIHwgRmFtaXguSW50ZXJmYWNlIHwgdW5kZWZpbmVkO1xuICAgICAgICBpZiAoY2xzIGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgc3ViQ2xhc3MgPSB0aGlzLmZteENsYXNzTWFwLmdldChjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICBzdWJDbGFzcyA9IHRoaXMuZm14SW50ZXJmYWNlTWFwLmdldChjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKCFzdWJDbGFzcykge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBTdWJjbGFzcyAke2NsYXNzRnVsbHlRdWFsaWZpZWROYW1lfSBub3QgZm91bmQgaW4gQ2xhc3Mgb3IgSW50ZXJmYWNlIG1hcHMuYCk7XG4gICAgICAgIH1cbiAgICAgICAgXG4gICAgICAgIGxldCBpbmhDbGFzc05hbWU6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICAgICAgbGV0IGluaENsYXNzRnVsbHlRdWFsaWZpZWROYW1lOiBzdHJpbmc7XG4gICAgICAgIGxldCBzdXBlckNsYXNzOiBGYW1peC5DbGFzcyB8IEZhbWl4LkludGVyZmFjZSB8IHVuZGVmaW5lZDtcbiAgICAgICAgaWYgKGluaENsYXNzIGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbiB8fCBpbmhDbGFzcyBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uKSB7XG4gICAgICAgICAgICBpbmhDbGFzc05hbWUgPSBpbmhDbGFzcy5nZXROYW1lKCk7XG4gICAgICAgICAgICBpZiAoIWluaENsYXNzTmFtZSkge1xuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgSW5oZXJpdGVkIGNsYXNzIG9yIGludGVyZmFjZSBuYW1lIG5vdCBmb3VuZCBmb3IgJHtpbmhDbGFzcy5nZXRUZXh0KCl9LmApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGluaENsYXNzKTtcbiAgICAgICAgICAgIGlmIChpbmhDbGFzcyBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgICAgICBzdXBlckNsYXNzID0gdGhpcy5mbXhDbGFzc01hcC5nZXQoaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgc3VwZXJDbGFzcyA9IHRoaXMuZm14SW50ZXJmYWNlTWFwLmdldChpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBpZiAoIXN1cGVyQ2xhc3MpIHtcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFN1cGVyY2xhc3MgJHtjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZX0gbm90IGZvdW5kIGluIENsYXNzIG9yIEludGVyZmFjZSBtYXBzLmApO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgLy8gaW5oQ2xhc3MgaXMgYW4gRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzXG4gICAgICAgICAgICBpbmhDbGFzc05hbWUgPSBpbmhDbGFzcy5nZXRFeHByZXNzaW9uKCkuZ2V0VGV4dCgpO1xuICAgICAgICAgICAgLy8gd2hhdCBpcyBpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZT8gVE9ET1xuICAgICAgICAgICAgaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSAnVW5kZWZpbmVkX1Njb3BlX2Zyb21faW1wb3J0ZXIuJyArIGluaENsYXNzTmFtZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChzdXBlckNsYXNzID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGlmIChpbmhDbGFzcyBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgICAgICBzdXBlckNsYXNzID0gbmV3IEZhbWl4LkNsYXNzKCk7XG4gICAgICAgICAgICAgICAgdGhpcy5mbXhDbGFzc01hcC5zZXQoaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUsIHN1cGVyQ2xhc3MpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgc3VwZXJDbGFzcyA9IG5ldyBGYW1peC5JbnRlcmZhY2UoKTtcbiAgICAgICAgICAgICAgICB0aGlzLmZteEludGVyZmFjZU1hcC5zZXQoaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUsIHN1cGVyQ2xhc3MpO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KHN1cGVyQ2xhc3MsaW5oQ2xhc3MpO1xuXG4gICAgICAgICAgICBzdXBlckNsYXNzLm5hbWUgPSBpbmhDbGFzc05hbWU7XG4gICAgICAgICAgICBzdXBlckNsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZSA9IGluaENsYXNzRnVsbHlRdWFsaWZpZWROYW1lO1xuICAgICAgICAgICAgc3VwZXJDbGFzcy5pc1N0dWIgPSB0cnVlO1xuXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbmhDbGFzcywgc3VwZXJDbGFzcyk7XG4gICAgICAgIFxuICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KHN1cGVyQ2xhc3MpO1xuICAgICAgICB9XG5cbiAgICAgICAgZm14SW5oZXJpdGFuY2Uuc3ViY2xhc3MgPSBzdWJDbGFzcztcbiAgICAgICAgZm14SW5oZXJpdGFuY2Uuc3VwZXJjbGFzcyA9IHN1cGVyQ2xhc3M7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEluaGVyaXRhbmNlKTtcblxuICAgICAgICAvLyBXZSBkb24ndCBtYXAgaW5oZXJpdGFuY2UgdG8gdGhlIHNvdXJjZSBjb2RlIGVsZW1lbnQgYmVjYXVzZSB0aGVyZSBhcmUgdHdvIGVsZW1lbnRzIChzdXBlciwgc3ViKVxuICAgICAgICAvLyB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEluaGVyaXRhbmNlLCBudWxsKTtcblxuICAgIH1cblxuICAgIHB1YmxpYyBjcmVhdGVGYW1peEltcG9ydENsYXVzZShpbXBvcnRlZEVudGl0eTogRmFtaXguTmFtZWRFbnRpdHksIGltcG9ydGluZ0VudGl0eTogRmFtaXguTW9kdWxlKSB7XG4gICAgICAgIGNvbnN0IGZteEltcG9ydENsYXVzZSA9IG5ldyBGYW1peC5JbXBvcnRDbGF1c2UoKTtcbiAgICAgICAgZm14SW1wb3J0Q2xhdXNlLmltcG9ydGVkRW50aXR5ID0gaW1wb3J0ZWRFbnRpdHk7XG4gICAgICAgIGZteEltcG9ydENsYXVzZS5pbXBvcnRpbmdFbnRpdHkgPSBpbXBvcnRpbmdFbnRpdHk7XG4gICAgICAgIGltcG9ydGluZ0VudGl0eS5hZGRPdXRnb2luZ0ltcG9ydChmbXhJbXBvcnRDbGF1c2UpO1xuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW1wb3J0Q2xhdXNlKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW1wb3J0IGNsYXVzZVxuICAgICAqIEBwYXJhbSBpbXBvcnRDbGF1c2VJbmZvIFRoZSBpbmZvcm1hdGlvbiBuZWVkZWQgdG8gY3JlYXRlIGEgRmFtaXggaW1wb3J0IGNsYXVzZVxuICAgICAqIEBwYXJhbSBpbXBvcnREZWNsYXJhdGlvbiBUaGUgaW1wb3J0IGRlY2xhcmF0aW9uXG4gICAgICogQHBhcmFtIGltcG9ydGVyIEEgc291cmNlIGZpbGUgd2hpY2ggaXMgYSBtb2R1bGVcbiAgICAgKiBAcGFyYW0gbW9kdWxlU3BlY2lmaWVyRmlsZVBhdGggVGhlIHBhdGggb2YgdGhlIG1vZHVsZSB3aGVyZSB0aGUgZXhwb3J0IGRlY2xhcmF0aW9uIGlzXG4gICAgICogQHBhcmFtIGltcG9ydEVsZW1lbnQgVGhlIGltcG9ydGVkIGVudGl0eVxuICAgICAqIEBwYXJhbSBpc0luRXhwb3J0cyBBIGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgaW1wb3J0ZWQgZW50aXR5IGlzIGluIHRoZSBleHBvcnRzXG4gICAgICogQHBhcmFtIGlzRGVmYXVsdEV4cG9ydCBBIGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgaW1wb3J0ZWQgZW50aXR5IGlzIGEgZGVmYXVsdCBleHBvcnRcbiAgICAgKi9cbiAgICBwdWJsaWMgb2xkQ3JlYXRlRmFtaXhJbXBvcnRDbGF1c2UoaW1wb3J0Q2xhdXNlSW5mbzoge2ltcG9ydERlY2xhcmF0aW9uPzogSW1wb3J0RGVjbGFyYXRpb24gfCBJbXBvcnRFcXVhbHNEZWNsYXJhdGlvbiwgaW1wb3J0ZXJTb3VyY2VGaWxlOiBTb3VyY2VGaWxlLCBtb2R1bGVTcGVjaWZpZXJGaWxlUGF0aDogc3RyaW5nLCBpbXBvcnRFbGVtZW50OiBJbXBvcnRTcGVjaWZpZXIgfCBJZGVudGlmaWVyLCBpc0luRXhwb3J0czogYm9vbGVhbiwgaXNEZWZhdWx0RXhwb3J0OiBib29sZWFufSk6IHZvaWQge1xuICAgICAgICBjb25zdCB7aW1wb3J0RGVjbGFyYXRpb24sIGltcG9ydGVyU291cmNlRmlsZTogaW1wb3J0ZXIsIG1vZHVsZVNwZWNpZmllckZpbGVQYXRoLCBpbXBvcnRFbGVtZW50LCBpc0luRXhwb3J0cywgaXNEZWZhdWx0RXhwb3J0fSA9IGltcG9ydENsYXVzZUluZm87XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgY3JlYXRlRmFtaXhJbXBvcnRDbGF1c2U6IENyZWF0aW5nIGltcG9ydCBjbGF1c2U6YCk7XG4gICAgICAgIGNvbnN0IGZteEltcG9ydENsYXVzZSA9IG5ldyBGYW1peC5JbXBvcnRDbGF1c2UoKTtcblxuICAgICAgICBsZXQgaW1wb3J0ZWRFbnRpdHk6IEZhbWl4Lk5hbWVkRW50aXR5IHwgRmFtaXguU3RydWN0dXJhbEVudGl0eSB8IHVuZGVmaW5lZCA9IHVuZGVmaW5lZDtcbiAgICAgICAgbGV0IGltcG9ydGVkRW50aXR5TmFtZTogc3RyaW5nO1xuXG4gICAgICAgIGNvbnN0IGFic29sdXRlUGF0aFByb2plY3QgPSB0aGlzLmZhbWl4UmVwLmdldEFic29sdXRlUGF0aCgpO1xuICAgICAgICBcbiAgICAgICAgY29uc3QgYWJzb2x1dGVQYXRoID0gcGF0aC5ub3JtYWxpemUobW9kdWxlU3BlY2lmaWVyRmlsZVBhdGgpO1xuICAgICAgICAvLyBjb252ZXJ0IHRoZSBwYXRoIGFuZCByZW1vdmUgYW55IHdpbmRvd3MgYmFja3NsYXNoZXMgaW50cm9kdWNlZCBieSBwYXRoLm5vcm1hbGl6ZVxuICAgICAgICBjb25zdCBwYXRoSW5Qcm9qZWN0OiBzdHJpbmcgPSB0aGlzLmNvbnZlcnRUb1JlbGF0aXZlUGF0aChhYnNvbHV0ZVBhdGgsIGFic29sdXRlUGF0aFByb2plY3QpLnJlcGxhY2UoL1xcXFwvZywgXCIvXCIpO1xuICAgICAgICBsZXQgcGF0aE5hbWUgPSBcIntcIiArIHBhdGhJblByb2plY3QgKyBcIn0uXCI7XG5cbiAgICAgICAgLy8gTmFtZWQgaW1wb3J0cywgZS5nLiBpbXBvcnQgeyBDbGFzc1cgfSBmcm9tIFwiLi9jb21wbGV4RXhwb3J0TW9kdWxlXCI7XG5cbiAgICAgICAgLy8gU3RhcnQgd2l0aCBzaW1wbGUgaW1wb3J0IGNsYXVzZSAod2l0aG91dCByZWZlcnJpbmcgdG8gdGhlIGFjdHVhbCB2YXJpYWJsZSlcblxuICAgICAgICBpZiAoaW1wb3J0RGVjbGFyYXRpb24gaW5zdGFuY2VvZiBJbXBvcnREZWNsYXJhdGlvbiBcbiAgICAgICAgICAgICYmIGltcG9ydEVsZW1lbnQgaW5zdGFuY2VvZiBJbXBvcnRTcGVjaWZpZXIpIHsgXG4gICAgICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHlOYW1lID0gaW1wb3J0RWxlbWVudC5nZXROYW1lKCk7XG4gICAgICAgICAgICBwYXRoTmFtZSA9IHBhdGhOYW1lICsgaW1wb3J0ZWRFbnRpdHlOYW1lO1xuICAgICAgICAgICAgaWYgKGlzSW5FeHBvcnRzKSB7XG4gICAgICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUocGF0aE5hbWUpIGFzIEZhbWl4Lk5hbWVkRW50aXR5O1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaWYgKGltcG9ydGVkRW50aXR5ID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eSA9IG5ldyBGYW1peC5OYW1lZEVudGl0eSgpO1xuICAgICAgICAgICAgICAgIGltcG9ydGVkRW50aXR5Lm5hbWUgPSBpbXBvcnRlZEVudGl0eU5hbWU7XG4gICAgICAgICAgICAgICAgaWYgKCFpc0luRXhwb3J0cykge1xuICAgICAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5pc1N0dWIgPSB0cnVlO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPSBwYXRoTmFtZTtcbiAgICAgICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbXBvcnRFbGVtZW50LCBpbXBvcnRlZEVudGl0eSk7XG4gICAgICAgICAgICAgICAgLy8gbXVzdCBhZGQgZW50aXR5IHRvIHJlcG9zaXRvcnlcbiAgICAgICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoaW1wb3J0ZWRFbnRpdHkpO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIC8vIGhhbmRsZSBpbXBvcnQgZXF1YWxzIGRlY2xhcmF0aW9ucywgZS5nLiBpbXBvcnQgbXlNb2R1bGUgPSByZXF1aXJlKFwiLi9jb21wbGV4RXhwb3J0TW9kdWxlXCIpO1xuICAgICAgICAvLyBUeXBlU2NyaXB0IGNhbid0IGRldGVybWluZSB0aGUgdHlwZSBvZiB0aGUgaW1wb3J0ZWQgbW9kdWxlLCBzbyB3ZSBjcmVhdGUgYSBNb2R1bGUgZW50aXR5XG4gICAgICAgIGVsc2UgaWYgKGltcG9ydERlY2xhcmF0aW9uIGluc3RhbmNlb2YgSW1wb3J0RXF1YWxzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgIGltcG9ydGVkRW50aXR5TmFtZSA9IGltcG9ydERlY2xhcmF0aW9uPy5nZXROYW1lKCk7XG4gICAgICAgICAgICBwYXRoTmFtZSA9IHBhdGhOYW1lICsgaW1wb3J0ZWRFbnRpdHlOYW1lO1xuICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkgPSBuZXcgRmFtaXguU3RydWN0dXJhbEVudGl0eSgpO1xuICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkubmFtZSA9IGltcG9ydGVkRW50aXR5TmFtZTtcbiAgICAgICAgICAgIGluaXRGUU4oaW1wb3J0RGVjbGFyYXRpb24sIGltcG9ydGVkRW50aXR5KTtcbiAgICAgICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGltcG9ydEVsZW1lbnQsIGltcG9ydGVkRW50aXR5KTtcbiAgICAgICAgICAgIGltcG9ydGVkRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9IHBhdGhOYW1lO1xuICAgICAgICAgICAgY29uc3QgYW55VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoJ2FueScsIGltcG9ydERlY2xhcmF0aW9uKTtcbiAgICAgICAgICAgIChpbXBvcnRlZEVudGl0eSBhcyBGYW1peC5TdHJ1Y3R1cmFsRW50aXR5KS5kZWNsYXJlZFR5cGUgPSBhbnlUeXBlO1xuICAgICAgICB9IGVsc2UgeyAgLy8gZGVmYXVsdCBpbXBvcnRzLCBlLmcuIGltcG9ydCBDbGFzc1cgZnJvbSBcIi4vY29tcGxleEV4cG9ydE1vZHVsZVwiOyAgXG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eU5hbWUgPSBpbXBvcnRFbGVtZW50LmdldFRleHQoKTtcbiAgICAgICAgICAgIHBhdGhOYW1lID0gcGF0aE5hbWUgKyAoaXNEZWZhdWx0RXhwb3J0ID8gXCJkZWZhdWx0RXhwb3J0XCIgOiBcIm5hbWVzcGFjZUV4cG9ydFwiKTtcbiAgICAgICAgICAgIGltcG9ydGVkRW50aXR5ID0gbmV3IEZhbWl4Lk5hbWVkRW50aXR5KCk7XG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5uYW1lID0gaW1wb3J0ZWRFbnRpdHlOYW1lO1xuICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lID0gcGF0aE5hbWU7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbXBvcnRFbGVtZW50LCBpbXBvcnRlZEVudGl0eSk7XG4gICAgICAgIH1cbiAgICAgICAgLy8gSSBkb24ndCB0aGluayBpdCBzaG91bGQgYmUgYWRkZWQgdG8gdGhlIHJlcG9zaXRvcnkgaWYgaXQgZXhpc3RzIGFscmVhZHlcbiAgICAgICAgaWYgKCFpc0luRXhwb3J0cykgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGltcG9ydGVkRW50aXR5KTtcbiAgICAgICAgY29uc3QgaW1wb3J0ZXJGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGltcG9ydGVyKTtcbiAgICAgICAgY29uc3QgZm14SW1wb3J0ZXIgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUoaW1wb3J0ZXJGdWxseVF1YWxpZmllZE5hbWUpIGFzIEZhbWl4Lk1vZHVsZTtcbiAgICAgICAgZm14SW1wb3J0Q2xhdXNlLmltcG9ydGluZ0VudGl0eSA9IGZteEltcG9ydGVyO1xuICAgICAgICBmbXhJbXBvcnRDbGF1c2UuaW1wb3J0ZWRFbnRpdHkgPSBpbXBvcnRlZEVudGl0eTtcbiAgICAgICAgaWYgKGltcG9ydERlY2xhcmF0aW9uIGluc3RhbmNlb2YgSW1wb3J0RXF1YWxzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgIGZteEltcG9ydENsYXVzZS5tb2R1bGVTcGVjaWZpZXIgPSBpbXBvcnREZWNsYXJhdGlvbj8uZ2V0TW9kdWxlUmVmZXJlbmNlKCkuZ2V0VGV4dCgpIGFzIHN0cmluZztcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGZteEltcG9ydENsYXVzZS5tb2R1bGVTcGVjaWZpZXIgPSBpbXBvcnREZWNsYXJhdGlvbj8uZ2V0TW9kdWxlU3BlY2lmaWVyVmFsdWUoKSBhcyBzdHJpbmc7XG4gICAgICAgIH1cbiAgICBcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBjcmVhdGVGYW1peEltcG9ydENsYXVzZTogJHtmbXhJbXBvcnRDbGF1c2UuaW1wb3J0ZWRFbnRpdHk/Lm5hbWV9IChvZiB0eXBlICR7XG4gICAgICAgICAgICBIZWxwZXJzLmdldFN1YlR5cGVOYW1lKGZteEltcG9ydENsYXVzZS5pbXBvcnRlZEVudGl0eSl9KSBpcyBpbXBvcnRlZCBieSAke2ZteEltcG9ydENsYXVzZS5pbXBvcnRpbmdFbnRpdHk/Lm5hbWV9YCk7XG5cbiAgICAgICAgZm14SW1wb3J0ZXIuYWRkT3V0Z29pbmdJbXBvcnQoZm14SW1wb3J0Q2xhdXNlKTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW1wb3J0Q2xhdXNlKTtcblxuICAgICAgICBpZiAoaW1wb3J0RGVjbGFyYXRpb24pIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14SW1wb3J0Q2xhdXNlLCBpbXBvcnREZWNsYXJhdGlvbik7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IEFycm93IEZ1bmN0aW9uXG4gICAgICogQHBhcmFtIGFycm93RXhwcmVzc2lvbiBBbiBFeHByZXNzaW9uXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSB2YXJpYWJsZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peEFycm93RnVuY3Rpb24oYXJyb3dFeHByZXNzaW9uOiBFeHByZXNzaW9uLCBjdXJyZW50Q0M6IHsgW2tleTogc3RyaW5nXTogbnVtYmVyIH0gKTogRmFtaXguQXJyb3dGdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNBcnJvd0Z1bmN0aW9uIHtcbiAgICAgICAgXG4gICAgICAgIGxldCBmbXhBcnJvd0Z1bmN0aW9uOiBGYW1peC5BcnJvd0Z1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Fycm93RnVuY3Rpb247XG5cbiAgICAgICAgY29uc3QgYXJyb3dGdW5jdGlvbiA9IGFycm93RXhwcmVzc2lvbi5hc0tpbmRPclRocm93KFN5bnRheEtpbmQuQXJyb3dGdW5jdGlvbik7XG5cbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gYXJyb3dGdW5jdGlvbi5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XG5cbiAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgZm14QXJyb3dGdW5jdGlvbiA9IG5ldyBGYW1peC5QYXJhbWV0cmljQXJyb3dGdW5jdGlvbigpO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14QXJyb3dGdW5jdGlvbiA9IG5ldyBGYW1peC5BcnJvd0Z1bmN0aW9uKCk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBHZXQgdGhlIHBhcmVudCBvZiB0aGUgYXJyb3cgZnVuY3Rpb24gKHRoZSB2YXJpYWJsZSBkZWNsYXJhdGlvbilcbiAgICAgICAgY29uc3QgcGFyZW50ID0gYXJyb3dGdW5jdGlvbi5nZXRQYXJlbnRJZktpbmQoU3ludGF4S2luZC5WYXJpYWJsZURlY2xhcmF0aW9uKTtcbiAgICAgICAgbGV0IGZ1bmN0aW9uTmFtZSA9ICcoTk9fTkFNRSknO1xuXG4gICAgICAgIGlmIChwYXJlbnQgJiYgcGFyZW50IGluc3RhbmNlb2YgVmFyaWFibGVEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgLy8gR2V0IHRoZSBuYW1lIG9mIHRoZSB2YXJpYWJsZVxuICAgICAgICAgICAgZnVuY3Rpb25OYW1lID0gcGFyZW50LmdldE5hbWUoKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChmdW5jdGlvbk5hbWUpIHtcbiAgICAgICAgICAgIGZteEFycm93RnVuY3Rpb24ubmFtZSA9IGZ1bmN0aW9uTmFtZTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteEFycm93RnVuY3Rpb24ubmFtZSA9IFwiYW5vbnltb3VzXCI7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBTaWduYXR1cmUgb2YgYW4gYXJyb3cgZnVuY3Rpb24gaXMgKHBhcmFtZXRlcnMpID0+IHJldHVybl90eXBlXG4gICAgICAgIGNvbnN0IHBhcmFtZXRlcnNTaWduYXR1cmUgPSBhcnJvd0Z1bmN0aW9uLmdldFBhcmFtZXRlcnMoKS5tYXAocCA9PiBwLmdldFRleHQoKSkuam9pbihcIiwgXCIpO1xuICAgICAgICBjb25zdCByZXR1cm5UeXBlU2lnbmF0dXJlID0gYXJyb3dGdW5jdGlvbi5nZXRSZXR1cm5UeXBlKCkuZ2V0VGV4dCgpO1xuICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLnNpZ25hdHVyZSA9IGAoJHtwYXJhbWV0ZXJzU2lnbmF0dXJlfSkgPT4gJHtyZXR1cm5UeXBlU2lnbmF0dXJlfWA7XG4gICAgICAgIGZteEFycm93RnVuY3Rpb24uY3ljbG9tYXRpY0NvbXBsZXhpdHkgPSBjdXJyZW50Q0NbZm14QXJyb3dGdW5jdGlvbi5uYW1lXTtcblxuICAgICAgICBsZXQgZnVuY3Rpb25UeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGZ1bmN0aW9uVHlwZU5hbWUgPSBhcnJvd0Z1bmN0aW9uLmdldFJldHVyblR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHJldHVybiB0eXBlIG9mIGZ1bmN0aW9uOiAke2Z1bmN0aW9uTmFtZX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKGZ1bmN0aW9uVHlwZU5hbWUsIGFycm93RnVuY3Rpb24gYXMgdW5rbm93biBhcyBGdW5jdGlvbkRlY2xhcmF0aW9uKTtcbiAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLm51bWJlck9mTGluZXNPZkNvZGUgPSBhcnJvd0Z1bmN0aW9uLmdldEVuZExpbmVOdW1iZXIoKSAtIGFycm93RnVuY3Rpb24uZ2V0U3RhcnRMaW5lTnVtYmVyKCk7XG4gICAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPSBhcnJvd0Z1bmN0aW9uLmdldFBhcmFtZXRlcnMoKTtcbiAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5udW1iZXJPZlBhcmFtZXRlcnMgPSBwYXJhbWV0ZXJzLmxlbmd0aDtcbiAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5udW1iZXJPZlN0YXRlbWVudHMgPSBhcnJvd0Z1bmN0aW9uLmdldFN0YXRlbWVudHMoKS5sZW5ndGg7XG4gICAgICAgIGluaXRGUU4oYXJyb3dFeHByZXNzaW9uIGFzIHVua25vd24gYXMgVFNNb3JwaE9iamVjdFR5cGUsIGZteEFycm93RnVuY3Rpb24pO1xuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihhcnJvd0V4cHJlc3Npb24gYXMgdW5rbm93biBhcyBUU01vcnBoT2JqZWN0VHlwZSwgZm14QXJyb3dGdW5jdGlvbik7XG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhBcnJvd0Z1bmN0aW9uKTtcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhBcnJvd0Z1bmN0aW9uLGFycm93RnVuY3Rpb24gYXMgdW5rbm93biBhcyBUU01vcnBoT2JqZWN0VHlwZSk7XG5cbiAgICAgICAgcmV0dXJuIGZteEFycm93RnVuY3Rpb247XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGNvbmNyZXRpc2F0aW9uXG4gICAgICogQHBhcmFtIGNscyBBIGNsYXNzXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjb25jcmV0aXNhdGlvblxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uKGNvbkVudGl0eSA6IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kICxnZW5FbnRpdHkgOiBGYW1peC5QYXJhbWV0cmljQ2xhc3MgfCBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZCk6IEZhbWl4LkNvbmNyZXRpc2F0aW9uIHtcbiAgICAgICAgXG4gICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSBuZXcgRmFtaXguQ29uY3JldGlzYXRpb24oKTsgICAgICAgICAgICAgIFxuICAgICAgICBcbiAgICAgICAgZm14Q29uY3JldGlzYXRpb24uY29uY3JldGVFbnRpdHkgPSBjb25FbnRpdHk7XG4gICAgICAgIGZteENvbmNyZXRpc2F0aW9uLmdlbmVyaWNFbnRpdHkgPSBnZW5FbnRpdHk7XG4gICAgICAgIC8vIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14Q29uY3JldGlzYXRpb24sbnVsbCk7XG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhDb25jcmV0aXNhdGlvbik7ICAgIFxuICAgICAgICBjb25zdCBwYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbihmbXhDb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICBcbiAgICAgICAgcmV0dXJuIGZteENvbmNyZXRpc2F0aW9uO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvblxuICAgICAqIEBwYXJhbSBjb25jcmV0aXNhdGlvbiBBIEZhbWl4Q29uY3JldGlzYXRpb25cbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIFBhcmFtZXRlckNvbmNyZXN0aXNhdGlvblxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKGNvbmNyZXRpc2F0aW9uOiBGYW1peC5Db25jcmV0aXNhdGlvbik6IEZhbWl4LlBhcmFtZXRlckNvbmNyZXRpc2F0aW9uIHwgdW5kZWZpbmVke1xuICAgICAgICBjb25zdCBjb25DbGFzcyA9IGNvbmNyZXRpc2F0aW9uLmNvbmNyZXRlRW50aXR5O1xuICAgICAgICBjb25zdCBnZW5DbGFzcyA9IGNvbmNyZXRpc2F0aW9uLmdlbmVyaWNFbnRpdHk7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgQ3JlYXRpbmcgcGFyYW1ldGVyIGNvbmNyZXRpc2F0aW9uIGJldHdlZW4gJHtjb25DbGFzcy5mdWxseVF1YWxpZmllZE5hbWV9IGFuZCAke2dlbkNsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcbiAgICAgICAgY29uc3QgcGFyYW1ldGVyQ29uY3JldGlzYXRpb25zID0gdGhpcy5mYW1peFJlcC5fZ2V0QWxsRW50aXRpZXNXaXRoVHlwZShcIlBhcmFtZXRlckNvbmNyZXRpc2F0aW9uXCIpIGFzIFNldDxGYW1peC5QYXJhbWV0ZXJDb25jcmV0aXNhdGlvbj47XG4gICAgICAgIGNvbnN0IGNvbmNyZXRlUGFyYW1ldGVycyA9IGNvbkNsYXNzLmNvbmNyZXRlUGFyYW1ldGVycztcbiAgICAgICAgY29uc3QgZ2VuZXJpY1BhcmFtZXRlcnMgPSBnZW5DbGFzcy5nZW5lcmljUGFyYW1ldGVycztcbiAgICAgICAgXG4gICAgICAgIGxldCBjb25DbGFzc1R5cGVQYXJhbWV0ZXJzSXRlcmF0b3IgPSBjb25jcmV0ZVBhcmFtZXRlcnMudmFsdWVzKCk7XG4gICAgICAgIGxldCBnZW5DbGFzc1R5cGVQYXJhbWV0ZXJzSXRlcmF0b3IgPSBnZW5lcmljUGFyYW1ldGVycy52YWx1ZXMoKTtcbiAgICAgICAgbGV0IGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uIDogRmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24gfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ7XG5cbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCBnZW5lcmljUGFyYW1ldGVycy5zaXplOyBpKyspIHtcbiAgICAgICAgICAgIGNvbnN0IGNvbkNsYXNzVHlwZVBhcmFtZXRlciA9IGNvbkNsYXNzVHlwZVBhcmFtZXRlcnNJdGVyYXRvci5uZXh0KCkudmFsdWUgYXMgRmFtaXguUGFyYW1ldGVyVHlwZTtcbiAgICAgICAgICAgIGNvbnN0IGdlbkNsYXNzVHlwZVBhcmFtZXRlciA9IGdlbkNsYXNzVHlwZVBhcmFtZXRlcnNJdGVyYXRvci5uZXh0KCkudmFsdWUgYXMgRmFtaXguUGFyYW1ldGVyVHlwZTtcbiAgICAgICAgICAgIGxldCBjcmVhdGVQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xuICAgICAgICAgICAgaWYoY29uQ2xhc3NUeXBlUGFyYW1ldGVyICYmIGdlbkNsYXNzVHlwZVBhcmFtZXRlciAmJiBjb25DbGFzc1R5cGVQYXJhbWV0ZXIubmFtZSAhPSBnZW5DbGFzc1R5cGVQYXJhbWV0ZXIubmFtZSl7XG4gICAgICAgICAgICAgICAgcGFyYW1ldGVyQ29uY3JldGlzYXRpb25zLmZvckVhY2goKHBhcmFtIDogRmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24pID0+IHtcbiAgICAgICAgICAgICAgICAgICAgaWYgKGNvbkNsYXNzVHlwZVBhcmFtZXRlci5uYW1lID09IHBhcmFtLmNvbmNyZXRlUGFyYW1ldGVyLm5hbWUgJiYgZ2VuQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWUgPT0gcGFyYW0uZ2VuZXJpY1BhcmFtZXRlci5uYW1lKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjcmVhdGVQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyQ29uY3JldGlzYXRpb24gPSBwYXJhbTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgaWYgKGNyZWF0ZVBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKSB7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uID0gbmV3IEZhbWl4LlBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKCk7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLmdlbmVyaWNQYXJhbWV0ZXIgPSBnZW5DbGFzc1R5cGVQYXJhbWV0ZXI7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLmNvbmNyZXRlUGFyYW1ldGVyID0gY29uQ2xhc3NUeXBlUGFyYW1ldGVyO1xuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbi5hZGRDb25jcmV0aXNhdGlvbihjb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICAgICAgICAgIC8vIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UGFyYW1ldGVyQ29uY3JldGlzYXRpb24sbnVsbCk7XG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgaWYgKCFmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiB3YXMgdW5kZWZpbmVkIGZvciBjb25jcmV0aXNhdGlvbiB3aXRoIGdlbmVyaWMgcGFyYW1ldGVyICR7Z2VuQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWV9IGFuZCBjb25jcmV0ZSBwYXJhbWV0ZXIgJHtjb25DbGFzc1R5cGVQYXJhbWV0ZXIubmFtZX1gKTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbi5hZGRDb25jcmV0aXNhdGlvbihjb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgaWYgKCFmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiB3YXMgdW5kZWZpbmVkIGZvciBjb25jcmV0aXNhdGlvbiB3aXRoIGNvbmNyZXRlIGVudGl0eSAke2NvbkNsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZX0gYW5kIGdlbmVyaWMgZW50aXR5ICR7Z2VuQ2xhc3MuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbjtcblxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIHR3byBjbGFzc2VzIG9yIHR3byBpbnRlcmZhY2VzXG4gICAgICogQHBhcmFtIGVsZW1lbnQgQSBjbGFzcyBvciBhbiBJbnRlcmZhY2VcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkNsYXNzT3JJbnRlcmZhY2VTcGVjaWFsaXNhdGlvbihlbGVtZW50OiBDbGFzc0RlY2xhcmF0aW9uIHwgSW50ZXJmYWNlRGVjbGFyYXRpb24pe1xuICAgICAgICBcbiAgICAgICAgY29uc3Qgc3VwZXJFbnRpdHkgPSBlbGVtZW50LmdldEV4dGVuZHMoKTtcbiAgICAgICAgbGV0IHN1cGVyRW50aXR5QXJyYXk7XG4gICAgICAgIGlmIChzdXBlckVudGl0eSl7XG4gICAgICAgICAgICBzdXBlckVudGl0eUFycmF5ID0gQXJyYXkuaXNBcnJheShzdXBlckVudGl0eSkgPyBzdXBlckVudGl0eSA6IFtzdXBlckVudGl0eV07XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHN1cGVyRW50aXR5QXJyYXkgJiYgc3VwZXJFbnRpdHlBcnJheS5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICBzdXBlckVudGl0eUFycmF5LmZvckVhY2goZW50aXR5ID0+IHtcbiAgICAgICAgICAgICAgICBsZXQgZW50aXR5SXNHZW5lcmljO1xuICAgICAgICAgICAgICAgIGNvbnN0IHN1cGVyRW50aXR5U3ltYm9sID0gZW50aXR5LmdldEV4cHJlc3Npb24oKS5nZXRTeW1ib2xPclRocm93KCk7XG4gICAgICAgICAgICAgICAgbGV0IHN1cGVyRW50aXR5RGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgaWYgKHN1cGVyRW50aXR5IGluc3RhbmNlb2YgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzKSB7XG4gICAgICAgICAgICAgICAgICAgIHN1cGVyRW50aXR5RGVjbGFyYXRpb24gPSBzdXBlckVudGl0eVN5bWJvbC5nZXREZWNsYXJhdGlvbnMoKVswXS5hc0tpbmQodHMuU3ludGF4S2luZC5DbGFzc0RlY2xhcmF0aW9uKTtcbiAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBzdXBlckVudGl0eURlY2xhcmF0aW9uID0gc3VwZXJFbnRpdHlTeW1ib2wuZ2V0RGVjbGFyYXRpb25zKClbMF0uYXNLaW5kKHRzLlN5bnRheEtpbmQuSW50ZXJmYWNlRGVjbGFyYXRpb24pO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBpZiAoc3VwZXJFbnRpdHlEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICBlbnRpdHlJc0dlbmVyaWMgPSBzdXBlckVudGl0eURlY2xhcmF0aW9uLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgaWYgKGVudGl0eUlzR2VuZXJpYykge1xuICAgICAgICAgICAgICAgICAgICBsZXQgRW50aXR5RGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgICAgIGxldCBnZW5FbnRpdHk7XG4gICAgICAgICAgICAgICAgICAgIGlmIChzdXBlckVudGl0eSBpbnN0YW5jZW9mIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cykge1xuICAgICAgICAgICAgICAgICAgICAgICAgRW50aXR5RGVjbGFyYXRpb24gPSBlbnRpdHkuZ2V0RXhwcmVzc2lvbigpLmdldFN5bWJvbCgpPy5nZXREZWNsYXJhdGlvbnMoKVswXSBhcyBDbGFzc0RlY2xhcmF0aW9uO1xuICAgICAgICAgICAgICAgICAgICAgICAgZ2VuRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q2xhc3MoRW50aXR5RGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcbiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIEVudGl0eURlY2xhcmF0aW9uID0gZW50aXR5LmdldEV4cHJlc3Npb24oKS5nZXRTeW1ib2woKT8uZ2V0RGVjbGFyYXRpb25zKClbMF0gYXMgSW50ZXJmYWNlRGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgICAgICAgICBnZW5FbnRpdHkgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoRW50aXR5RGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gRW50aXR5RGVjbGFyYXRpb24uZ2V0VHlwZVBhcmFtZXRlcnMoKS5tYXAoKHBhcmFtKSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBhcmdzID0gZWxlbWVudC5nZXRIZXJpdGFnZUNsYXVzZXMoKVswXS5nZXRUeXBlTm9kZXMoKVswXS5nZXRUeXBlQXJndW1lbnRzKClcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gZWxlbWVudC5nZXRIZXJpdGFnZUNsYXVzZXMoKVswXS5nZXRUeXBlTm9kZXMoKVswXS5nZXRUeXBlQXJndW1lbnRzKCkubWFwKChwYXJhbSkgPT4gcGFyYW0uZ2V0VGV4dCgpKTtcbiAgICAgICAgICAgICAgICAgICAgaWYgKCFIZWxwZXJzLmFycmF5c0FyZUVxdWFsKGNvblBhcmFtcyxnZW5QYXJhbXMpKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBsZXQgY29uRW50aXR5O1xuICAgICAgICAgICAgICAgICAgICAgICAgY29uRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q29uY3JldGVFbGVtZW50KGdlbkVudGl0eSxFbnRpdHlEZWNsYXJhdGlvbixhcmdzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xuICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNyZWF0ZUNvbmNyZXRpc2F0aW9uIDogYm9vbGVhbiA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0aXNhdGlvbnMuZm9yRWFjaCgoY29uYyA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25FbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lKXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY3JlYXRlQ29uY3JldGlzYXRpb24gPSBmYWxzZTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIFxuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGNyZWF0ZUNvbmNyZXRpc2F0aW9uKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZm14Q29uY3JldGlzYXRpb24gOiBGYW1peC5Db25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbihjb25FbnRpdHksZ2VuRW50aXR5KTtcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICAgIC8vIFRPRE86IFRoaXMgZnVuY3Rpb24gc2VlbXMgdW5maW5pc2hlZFxuICAgIH0gICAgXG4gICAgXG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggY29uY3JldGlzYXRpb24gYmV0d2VlbiBhIGNsYXNzIGFuZCBpdHMgaW5zdGFuY2lhdGlvbnNcbiAgICAgKiBAcGFyYW0gY2xzIEEgY2xhc3NcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkdlbmVyaWNJbnN0YW50aWF0aW9uKGNsczogQ2xhc3NEZWNsYXJhdGlvbil7XG4gICAgICAgXG4gICAgICAgIGNvbnN0IGlzR2VuZXJpYyA9IGNscy5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XG4gICAgICAgIGlmIChpc0dlbmVyaWMpIHtcbiAgICAgICAgICAgIGNvbnN0IGluc3RhbmNlcyA9IGNscy5nZXRTb3VyY2VGaWxlKCkuZ2V0RGVzY2VuZGFudHNPZktpbmQodHMuU3ludGF4S2luZC5OZXdFeHByZXNzaW9uKVxuICAgICAgICAgICAgICAgIC5maWx0ZXIobmV3RXhwciA9PiB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGV4cHJlc3Npb24gPSBuZXdFeHByLmdldEV4cHJlc3Npb24oKTtcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGV4cHJlc3Npb24uZ2V0VGV4dCgpID09PSBjbHMuZ2V0TmFtZSgpO1xuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIGluc3RhbmNlcy5mb3JFYWNoKGluc3RhbmNlID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCBpbnN0YW5jZUlzR2VuZXJpYyA9IGluc3RhbmNlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGggPiAwO1xuICAgICAgICAgICAgICAgIGlmIChpbnN0YW5jZUlzR2VuZXJpYykge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBjb25QYXJhbXMgPSBpbnN0YW5jZS5nZXRUeXBlQXJndW1lbnRzKCkubWFwKChwYXJhbSkgPT4gcGFyYW0uZ2V0VGV4dCgpKTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZ2VuRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q2xhc3MoY2xzKSBhcyBGYW1peC5QYXJhbWV0cmljQ2xhc3M7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGdlblBhcmFtcyA9IGNscy5nZXRUeXBlUGFyYW1ldGVycygpLm1hcCgocGFyYW0pID0+IHBhcmFtLmdldFRleHQoKSk7XG4gICAgICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNvbkVudGl0eTtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbkVudGl0eSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChnZW5FbnRpdHksY2xzLGluc3RhbmNlLmdldFR5cGVBcmd1bWVudHMoKSk7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjb25jcmV0aXNhdGlvbnMgPSB0aGlzLmZhbWl4UmVwLl9nZXRBbGxFbnRpdGllc1dpdGhUeXBlKFwiQ29uY3JldGlzYXRpb25cIikgYXMgU2V0PEZhbWl4LkNvbmNyZXRpc2F0aW9uPjtcbiAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjcmVhdGVDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICAgICAgY29uY3JldGlzYXRpb25zLmZvckVhY2goKGNvbmMgOiBGYW1peC5Db25jcmV0aXNhdGlvbikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChnZW5FbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lID09IGNvbmMuZ2VuZXJpY0VudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgJiYgY29uYy5jb25jcmV0ZUVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSl7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbmNyZXRpc2F0aW9uID0gZmFsc2U7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjcmVhdGVDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSB0aGlzLmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb24oY29uRW50aXR5LGdlbkVudGl0eSk7XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KVxuICAgICAgICB9XG4gICAgICAgIC8vIFRPRE86IFRoaXMgZnVuY3Rpb24gc2VlbXMgdW5maW5pc2hlZFxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGEgY2xhc3MgYW5kIGl0cyBpbnN0YW5jaWF0aW9uc1xuICAgICAqIEBwYXJhbSBmdW5jIEEgZnVuY3Rpb25cbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkZ1bmN0aW9uSW5zdGFudGlhdGlvbihlbGVtZW50OiBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgTWV0aG9kRGVjbGFyYXRpb24pe1xuICAgICAgICBjb25zdCBpc0dlbmVyaWMgPSBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gZWxlbWVudC5nZXRUeXBlUGFyYW1ldGVycygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgY29uc3QgdXNlcyA9IGVsZW1lbnQuZmluZFJlZmVyZW5jZXNBc05vZGVzKCk7ICAgIFxuICAgICAgICAgICAgdXNlcy5mb3JFYWNoKHVzYWdlID0+IHtcbiAgICAgICAgICAgICAgICBsZXQgY3VycmVudE5vZGU6IE5vZGUgfCB1bmRlZmluZWQgPSB1c2FnZTtcblxuICAgICAgICAgICAgICAgIHdoaWxlIChjdXJyZW50Tm9kZSkge1xuICAgICAgICAgICAgICAgICAgICBpZiAoY3VycmVudE5vZGUuZ2V0S2luZCgpID09PSBTeW50YXhLaW5kLkNhbGxFeHByZXNzaW9uKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjYWxsRXhwcmVzc2lvbiA9IGN1cnJlbnROb2RlLmFzS2luZChTeW50YXhLaW5kLkNhbGxFeHByZXNzaW9uKTtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmICghY2FsbEV4cHJlc3Npb24pIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYENhbGxFeHByZXNzaW9uIG5vdCBmb3VuZCBmb3IgJHtjdXJyZW50Tm9kZS5nZXRUZXh0KCl9YCk7XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBpbnN0YW5jZUlzR2VuZXJpYyA9IGNhbGxFeHByZXNzaW9uLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGggPiAwO1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGluc3RhbmNlSXNHZW5lcmljKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgYXJncyA9IGNhbGxFeHByZXNzaW9uLmdldFR5cGVBcmd1bWVudHMoKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjb25QYXJhbXMgPSBjYWxsRXhwcmVzc2lvbi5nZXRUeXBlQXJndW1lbnRzKCkubWFwKHBhcmFtID0+IHBhcmFtLmdldFRleHQoKSk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCFIZWxwZXJzLmFycmF5c0FyZUVxdWFsKGNvblBhcmFtcyxnZW5QYXJhbXMpKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBnZW5FbGVtZW50O1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZihlbGVtZW50IGluc3RhbmNlb2YgRnVuY3Rpb25EZWNsYXJhdGlvbil7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZW5FbGVtZW50ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4RnVuY3Rpb24oZWxlbWVudCwge30pIGFzIEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbjtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdlbkVsZW1lbnQgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhNZXRob2QoZWxlbWVudCwge30pIGFzIEZhbWl4LlBhcmFtZXRyaWNNZXRob2Q7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNvbmNFbGVtZW50O1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jRWxlbWVudCA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChnZW5FbGVtZW50LGVsZW1lbnQsYXJncyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgY3JlYXRlQ29uY3JldGlzYXRpb24gOiBib29sZWFuID0gdHJ1ZTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uY3JldGlzYXRpb25zLmZvckVhY2goKGNvbmMgOiBGYW1peC5Db25jcmV0aXNhdGlvbikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkVsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lID09IGNvbmMuZ2VuZXJpY0VudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgJiYgY29uYy5jb25jcmV0ZUVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uY0VsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lKXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjcmVhdGVDb25jcmV0aXNhdGlvbiA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjcmVhdGVDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZm14Q29uY3JldGlzYXRpb24gOiBGYW1peC5Db25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbihjb25jRWxlbWVudCxnZW5FbGVtZW50KTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIC8vIFJlbW9udGVyIMOgIGwnw6lsw6ltZW50IHBhcmVudCAodXRpbGUgc2kgbGUgbsWTdWQgZGUgcsOpZsOpcmVuY2UgZXN0IHVuIGVuZmFudClcbiAgICAgICAgICAgICAgICAgICAgY3VycmVudE5vZGUgPSBjdXJyZW50Tm9kZS5nZXRQYXJlbnQoKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGEgY2xhc3MgYW5kIGFuIGludGVyZmFjZVxuICAgICAqIEBwYXJhbSBjbHMgQSBjbGFzc1xuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uSW50ZXJmYWNlQ2xhc3MoY2xzOiBDbGFzc0RlY2xhcmF0aW9uKXtcbiAgICBcbiAgICAgICAgY29uc3Qgc3VwZXJJbnRlcmZhY2VzID0gY2xzLmdldEltcGxlbWVudHMoKTtcbiAgICAgICAgc3VwZXJJbnRlcmZhY2VzLmZvckVhY2goaW50ZXJmYWNlVHlwZSA9PiB7XG4gICAgICAgICAgICBjb25zdCBpbnRlcmZhY2VJc0dlbmVyaWMgPSBpbnRlcmZhY2VUeXBlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGg+MDtcbiAgICAgICAgICAgIGlmIChpbnRlcmZhY2VJc0dlbmVyaWMpIHtcbiAgICAgICAgICAgICAgICBjb25zdCBpbnRlcmZhY2VEZWNsYXJhdGlvbiA9IGludGVyZmFjZVR5cGUuZ2V0RXhwcmVzc2lvbigpLmdldFN5bWJvbCgpPy5nZXREZWNsYXJhdGlvbnMoKVswXSBhcyBJbnRlcmZhY2VEZWNsYXJhdGlvbjtcbiAgICAgICAgICAgICAgICBjb25zdCBnZW5QYXJhbXMgPSBpbnRlcmZhY2VEZWNsYXJhdGlvbi5nZXRUeXBlUGFyYW1ldGVycygpLm1hcCgocGFyYW0pID0+IHBhcmFtLmdldFRleHQoKSk7XG4gICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gY2xzLmdldEhlcml0YWdlQ2xhdXNlcygpWzBdLmdldFR5cGVOb2RlcygpWzBdLmdldFR5cGVBcmd1bWVudHMoKS5tYXAoKHBhcmFtKSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGFyZ3MgPSBjbHMuZ2V0SGVyaXRhZ2VDbGF1c2VzKClbMF0uZ2V0VHlwZU5vZGVzKClbMF0uZ2V0VHlwZUFyZ3VtZW50cygpO1xuICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBnZW5JbnRlcmZhY2UgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoaW50ZXJmYWNlRGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbkludGVyZmFjZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChnZW5JbnRlcmZhY2UsaW50ZXJmYWNlRGVjbGFyYXRpb24sYXJncyk7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xuICAgICAgICAgICAgICAgICAgICBsZXQgY3JlYXRlQ29uY3JldGlzYXRpb24gOiBib29sZWFuID0gdHJ1ZTtcbiAgICAgICAgICAgICAgICAgICAgY29uY3JldGlzYXRpb25zLmZvckVhY2goKGNvbmMgOiBGYW1peC5Db25jcmV0aXNhdGlvbikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkludGVyZmFjZS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25JbnRlcmZhY2UuZnVsbHlRdWFsaWZpZWROYW1lKXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjcmVhdGVDb25jcmV0aXNhdGlvbiA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIFxuICAgICAgICAgICAgICAgICAgICBpZiAoY3JlYXRlQ29uY3JldGlzYXRpb24pIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSB0aGlzLmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb24oY29uSW50ZXJmYWNlLGdlbkludGVyZmFjZSk7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGFuIGludGVyZmFjZSBhbmQgYSBUeXBlXG4gICAgICogQHBhcmFtIGVsZW1lbnQgQSB2YXJpYWJsZSBvciBhIGZ1bmN0aW9uXG4gICAgICogQHBhcmFtIGludGVyIEFuIGludGVyZmFjZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uVHlwZUluc3RhbmNpYXRpb24oZWxlbWVudDogSW50ZXJmYWNlRGVjbGFyYXRpb24gfCBDbGFzc0RlY2xhcmF0aW9uKXtcblxuICAgICAgICBjb25zdCBpc0dlbmVyaWMgPSBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gZWxlbWVudC5nZXRUeXBlUGFyYW1ldGVycygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgY29uc3QgdXNlcyA9IGVsZW1lbnQuZmluZFJlZmVyZW5jZXNBc05vZGVzKCk7XG4gICAgICAgICAgICB1c2VzLmZvckVhY2godXNlID0+IHsgICAgICAgIFxuICAgICAgICAgICAgICAgIGxldCBwYXJlbnROb2RlID0gdXNlLmdldFBhcmVudCgpO1xuICAgICAgICAgICAgICAgIHdoaWxlIChwYXJlbnROb2RlKSB7XG4gICAgICAgICAgICAgICAgICAgIGlmIChwYXJlbnROb2RlLmdldEtpbmQoKSA9PT0gU3ludGF4S2luZC5UeXBlUmVmZXJlbmNlKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB0eXBlUmVmZXJlbmNlTm9kZSA9IHBhcmVudE5vZGUuYXNLaW5kKFN5bnRheEtpbmQuVHlwZVJlZmVyZW5jZSk7XG4gICAgICAgICAgICAgICAgICAgICAgICBpZiAoIXR5cGVSZWZlcmVuY2VOb2RlKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBUeXBlUmVmZXJlbmNlTm9kZSBub3QgZm91bmQgZm9yICR7cGFyZW50Tm9kZS5nZXRUZXh0KCl9YCk7XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB0eXBlUmVmZXJlbmNlTm9kZUlzR2VuZXJpYyA9IHR5cGVSZWZlcmVuY2VOb2RlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGggPiAwO1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHR5cGVSZWZlcmVuY2VOb2RlSXNHZW5lcmljKSB7fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGFyZ3MgPSB0eXBlUmVmZXJlbmNlTm9kZS5nZXRUeXBlQXJndW1lbnRzKCk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gdHlwZVJlZmVyZW5jZU5vZGUuZ2V0VHlwZUFyZ3VtZW50cygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgZ2VuRWxlbWVudDtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYoZWxlbWVudCBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2VuRWxlbWVudCA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENsYXNzKGVsZW1lbnQpIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdlbkVsZW1lbnQgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoZWxlbWVudCkgYXMgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgY29uY0VsZW1lbnQ7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q29uY3JldGVFbGVtZW50KGdlbkVsZW1lbnQsZWxlbWVudCxhcmdzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uY3JldGlzYXRpb25zID0gdGhpcy5mYW1peFJlcC5fZ2V0QWxsRW50aXRpZXNXaXRoVHlwZShcIkNvbmNyZXRpc2F0aW9uXCIpIGFzIFNldDxGYW1peC5Db25jcmV0aXNhdGlvbj47XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjcmVhdGVDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0aXNhdGlvbnMuZm9yRWFjaCgoY29uYyA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZ2VuRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25jRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUpe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbmNyZXRpc2F0aW9uID0gZmFsc2U7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGNyZWF0ZUNvbmNyZXRpc2F0aW9uKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBmbXhDb25jcmV0aXNhdGlvbiA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uID0gdGhpcy5jcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uKGNvbmNFbGVtZW50LGdlbkVsZW1lbnQpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgcGFyZW50Tm9kZSA9IHBhcmVudE5vZGUuZ2V0UGFyZW50KCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwdWJsaWMgY29udmVydFRvUmVsYXRpdmVQYXRoKGFic29sdXRlUGF0aDogc3RyaW5nLCBhYnNvbHV0ZVBhdGhQcm9qZWN0OiBzdHJpbmcpIHtcbiAgICAgICAgcmV0dXJuIGFic29sdXRlUGF0aC5yZXBsYWNlKGFic29sdXRlUGF0aFByb2plY3QsIFwiXCIpLnNsaWNlKDEpO1xuICAgIH1cbn1cbmZ1bmN0aW9uIGluaXRGUU4oc291cmNlRWxlbWVudDogVFNNb3JwaE9iamVjdFR5cGUsIGZhbWl4RWxlbWVudDogRmFtaXguU291cmNlZEVudGl0eSkge1xuICAgIGlmICghKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBDb21tZW50UmFuZ2UpKSB7XG4gICAgICAgIGNvbnN0IGZxbiA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oc291cmNlRWxlbWVudCk7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhcIlNldHRpbmcgZnVsbHkgcXVhbGlmaWVkIG5hbWUgZm9yIFwiICsgZmFtaXhFbGVtZW50LmdldEpTT04oKSArIFwiIHRvIFwiICsgZnFuKTtcbiAgICAgICAgKGZhbWl4RWxlbWVudCBhcyBGYW1peC5OYW1lZEVudGl0eSkuZnVsbHlRdWFsaWZpZWROYW1lID0gZnFuO1xuICAgIH1cbn1cblxuIl19
1696
+ function isTypeContext(sourceElement) {
1697
+ return sourceElement instanceof ts_morph_1.ConstructorDeclaration
1698
+ || sourceElement instanceof ts_morph_1.MethodDeclaration
1699
+ || sourceElement instanceof ts_morph_1.FunctionDeclaration
1700
+ || sourceElement instanceof ts_morph_1.FunctionExpression
1701
+ || sourceElement instanceof ts_morph_1.ArrowFunction
1702
+ ||
1703
+ sourceElement instanceof ts_morph_1.ParameterDeclaration ||
1704
+ sourceElement instanceof ts_morph_1.VariableDeclaration ||
1705
+ sourceElement instanceof ts_morph_1.PropertyDeclaration ||
1706
+ sourceElement instanceof ts_morph_1.PropertySignature ||
1707
+ sourceElement instanceof ts_morph_1.TypeParameterDeclaration ||
1708
+ sourceElement instanceof ts_morph_1.Identifier ||
1709
+ sourceElement instanceof ts_morph_1.Decorator ||
1710
+ sourceElement instanceof ts_morph_1.GetAccessorDeclaration ||
1711
+ sourceElement instanceof ts_morph_1.SetAccessorDeclaration ||
1712
+ sourceElement instanceof ts_morph_1.ImportSpecifier ||
1713
+ sourceElement instanceof ts_morph_1.EnumDeclaration ||
1714
+ sourceElement instanceof ts_morph_1.EnumMember ||
1715
+ sourceElement instanceof ts_morph_1.TypeAliasDeclaration
1716
+ || sourceElement instanceof ts_morph_1.ImportDeclaration
1717
+ || sourceElement instanceof ts_morph_1.ExpressionWithTypeArguments;
1718
+ }
1719
+ function getInterfaceDeclarationFromExpression(expression) {
1720
+ // Step 1: Get the type of the expression
1721
+ const type = expression.getType();
1722
+ // Step 2: Get the symbol associated with the type
1723
+ let symbol = type.getSymbol();
1724
+ if (!symbol) {
1725
+ // If symbol is not found, try to get the symbol from the identifier
1726
+ const identifier = expression.getFirstDescendantByKind(ts_morph_1.SyntaxKind.Identifier);
1727
+ if (!identifier) {
1728
+ throw new Error(`Identifier not found for ${expression.getText()}.`);
1729
+ }
1730
+ symbol = identifier.getSymbol();
1731
+ if (!symbol) {
1732
+ throw new Error(`Symbol not found for ${identifier.getText()}.`);
1733
+ }
1734
+ }
1735
+ // Step 3: Resolve the symbol to find the actual declaration
1736
+ const interfaceDeclaration = resolveSymbolToInterfaceDeclaration(symbol);
1737
+ if (!interfaceDeclaration) {
1738
+ analyze_1.logger.error(`Interface declaration not found for ${expression.getText()}.`);
1739
+ }
1740
+ return interfaceDeclaration;
1741
+ }
1742
+ function resolveSymbolToInterfaceDeclaration(symbol) {
1743
+ // Get the declarations associated with the symbol
1744
+ const declarations = symbol.getDeclarations();
1745
+ // Filter for InterfaceDeclaration
1746
+ const interfaceDeclaration = declarations.find(declaration => declaration instanceof ts_morph_1.InterfaceDeclaration);
1747
+ if (interfaceDeclaration) {
1748
+ return interfaceDeclaration;
1749
+ }
1750
+ // Handle imports: If the symbol is imported, resolve the import to find the actual declaration
1751
+ for (const declaration of declarations) {
1752
+ if (declaration.getKind() === ts_morph_1.SyntaxKind.ImportSpecifier) {
1753
+ const importSpecifier = declaration;
1754
+ const importDeclaration = importSpecifier.getImportDeclaration();
1755
+ const moduleSpecifier = importDeclaration.getModuleSpecifierSourceFile();
1756
+ if (moduleSpecifier) {
1757
+ const exportedSymbols = moduleSpecifier.getExportSymbols();
1758
+ const exportedSymbol = exportedSymbols.find(symbol => symbol.getName() === importSpecifier.getName());
1759
+ if (exportedSymbol) {
1760
+ return resolveSymbolToInterfaceDeclaration(exportedSymbol);
1761
+ }
1762
+ }
1763
+ }
1764
+ }
1765
+ return undefined;
1766
+ }
1767
+ // function oldGetInterfaceDeclarationFromExpression(expression: ExpressionWithTypeArguments): InterfaceDeclaration | undefined {
1768
+ // // Two cases:
1769
+ // // class A implements ImportedInterface, DeclaredInterface {}
1770
+ // const type = expression.getType();
1771
+ // // ImportedInterface: type will a symbol
1772
+ // let symbol = type.getAliasSymbol(); // will be defined for imported interfaces
1773
+ // if (!symbol) {
1774
+ // // DeclaredInterface: type will be an InterfaceDeclaration on Identifier node that is the child of the ExpressionWithTypeArguments
1775
+ // const identifier = expression.getFirstDescendantByKind(SyntaxKind.Identifier);
1776
+ // if (!identifier) {
1777
+ // throw new Error(`Identifier not found for ${expression.getText()}.`);
1778
+ // }
1779
+ // symbol = identifier.getSymbol();
1780
+ // if (!symbol) {
1781
+ // throw new Error(`Symbol not found for ${identifier.getText()}.`);
1782
+ // }
1783
+ // }
1784
+ // // Step 3: Get the declarations associated with the symbol
1785
+ // const declarations = symbol.getDeclarations();
1786
+ // // Step 4: Filter for InterfaceDeclaration
1787
+ // const interfaceDeclaration = declarations.find(declaration => declaration instanceof InterfaceDeclaration) as InterfaceDeclaration | undefined;
1788
+ // if (!interfaceDeclaration) {
1789
+ // throw new Error(`Interface declaration not found for ${expression.getText()}.`);
1790
+ // }
1791
+ // return interfaceDeclaration;
1792
+ // }
1793
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRW50aXR5RGljdGlvbmFyeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9mYW1peF9mdW5jdGlvbnMvRW50aXR5RGljdGlvbmFyeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFvMERILDBDQVlDO0FBNzBERCx1Q0FBNmtCO0FBQzdrQiw4RUFBZ0Y7QUFDaEYsZ0VBQWtEO0FBQ2xELG9FQUFnRTtBQUNoRSx3Q0FBNEM7QUFDNUMsb0RBQW9EO0FBQ3BELGlFQUFpRTtBQUNqRSxzREFBdUQ7QUFDdkQsNERBQThDO0FBQzlDLHFEQUF1QztBQUN2QyxnREFBd0I7QUFleEIsTUFBYSxnQkFBZ0I7SUFxQnpCO1FBbkJPLGFBQVEsR0FBRyxJQUFJLGtDQUFlLEVBQUUsQ0FBQztRQUNoQyxnQkFBVyxHQUFHLElBQUksR0FBRyxFQUF1QixDQUFDLENBQUMsNENBQTRDO1FBQzFGLGdCQUFXLEdBQUcsSUFBSSxHQUFHLEVBQStDLENBQUMsQ0FBQyw0REFBNEQ7UUFDbEksb0JBQWUsR0FBRyxJQUFJLEdBQUcsRUFBdUQsQ0FBQyxDQUFDLGdEQUFnRDtRQUNsSSxpQkFBWSxHQUFHLElBQUksR0FBRyxFQUFtQyxDQUFDLENBQUMsZ0RBQWdEO1FBQzNHLGVBQVUsR0FBRyxJQUFJLEdBQUcsRUFBNkMsQ0FBQyxDQUFDLGtEQUFrRDtRQUNySCxlQUFVLEdBQUcsSUFBSSxHQUFHLEVBQTRELENBQUMsQ0FBQyxtREFBbUQ7UUFDckksd0JBQW1CLEdBQUcsSUFBSSxHQUFHLEVBQStCLENBQUMsQ0FBQyxxREFBcUQ7UUFDbkgsNEJBQXVCLEdBQUcsSUFBSSxHQUE4RixDQUFDLENBQUMsK0NBQStDO1FBQzdLLHdCQUFtQixHQUFHLElBQUksR0FBZ0MsQ0FBQyxDQUFDLCtDQUErQztRQUMzRyxvQkFBZSxHQUFHLElBQUksR0FBRyxFQUF5QyxDQUFDLENBQUMsMkNBQTJDO1FBQy9H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bUMsQ0FBQyxjQUFjLENBQUMsQ0FBQztnQkFDL0QsQ0FBQztZQUNMLENBQUM7UUFDTCxDQUFDO0lBQ0wsQ0FBQztJQUVELE9BQU8sU0FBUyxDQUFDO0FBQ3JCLENBQUM7QUFFRCxpSUFBaUk7QUFDakksb0JBQW9CO0FBQ3BCLG9FQUFvRTtBQUNwRSx5Q0FBeUM7QUFFekMsZ0RBQWdEO0FBQ2hELHNGQUFzRjtBQUV0RixxQkFBcUI7QUFDckIsNklBQTZJO0FBQzdJLHlGQUF5RjtBQUN6Riw2QkFBNkI7QUFDN0Isb0ZBQW9GO0FBQ3BGLFlBQVk7QUFDWiwyQ0FBMkM7QUFDM0MseUJBQXlCO0FBQ3pCLGdGQUFnRjtBQUNoRixZQUFZO0FBQ1osUUFBUTtBQUVSLGlFQUFpRTtBQUNqRSxxREFBcUQ7QUFFckQsaURBQWlEO0FBQ2pELHNKQUFzSjtBQUV0SixtQ0FBbUM7QUFDbkMsMkZBQTJGO0FBQzNGLFFBQVE7QUFFUixtQ0FBbUM7QUFDbkMsSUFBSSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxyXG4gKiBhIGZ1bmN0aW9uIGdldE9yQ3JlYXRlWFR5cGUgdGFrZXMgYXJndW1lbnRzIG5hbWU6IHN0cmluZyBhbmQgZWxlbWVudDogdHMtbW9ycGgtdHlwZSBhbmQgcmV0dXJucyBhIEZhbWl4LlR5cGVcclxuICogVGhlIGdvYWwgaXMgdG8ga2VlcCB0cmFjayBvZiB0aGUgdHlwZXMgKGUuZy4sIGEgbWV0aG9kJ3MgZGVmaW5lZFR5cGUpLCBmb3IgdGhlIG1vZGVsLlxyXG4gKiBUaGUgbmFtZSBkb2Vzbid0IG5lZWQgdG8gYmUgZnVsbHkgcXVhbGlmaWVkIChpdCdzIHRoZSBuYW1lIHVzZWQgaW4gdGhlIHNvdXJjZSBjb2RlLCBvciB0aGUgRmFtaXggbW9kZWwpLlxyXG4gKi9cclxuXHJcblxyXG5pbXBvcnQgeyBDbGFzc0RlY2xhcmF0aW9uLCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uLCBGdW5jdGlvbkRlY2xhcmF0aW9uLCBJZGVudGlmaWVyLCBJbnRlcmZhY2VEZWNsYXJhdGlvbiwgTWV0aG9kRGVjbGFyYXRpb24sIE1ldGhvZFNpZ25hdHVyZSwgTW9kdWxlRGVjbGFyYXRpb24sIFByb3BlcnR5RGVjbGFyYXRpb24sIFByb3BlcnR5U2lnbmF0dXJlLCBTb3VyY2VGaWxlLCBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb24sIFZhcmlhYmxlRGVjbGFyYXRpb24sIFBhcmFtZXRlckRlY2xhcmF0aW9uLCBEZWNvcmF0b3IsIEdldEFjY2Vzc29yRGVjbGFyYXRpb24sIFNldEFjY2Vzc29yRGVjbGFyYXRpb24sIEltcG9ydFNwZWNpZmllciwgQ29tbWVudFJhbmdlLCBFbnVtRGVjbGFyYXRpb24sIEVudW1NZW1iZXIsIFR5cGVBbGlhc0RlY2xhcmF0aW9uLCBGdW5jdGlvbkV4cHJlc3Npb24sIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cywgSW1wb3J0RGVjbGFyYXRpb24sIEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uLCBTeW50YXhLaW5kLCBFeHByZXNzaW9uLCBUeXBlTm9kZSwgU2NvcGUsIEFycm93RnVuY3Rpb24gfSBmcm9tIFwidHMtbW9ycGhcIjtcclxuaW1wb3J0IHsgaXNBbWJpZW50LCBpc05hbWVzcGFjZSB9IGZyb20gXCIuLi9hbmFseXplX2Z1bmN0aW9ucy9wcm9jZXNzX2Z1bmN0aW9uc1wiO1xyXG5pbXBvcnQgKiBhcyBGYW1peCBmcm9tIFwiLi4vbGliL2ZhbWl4L21vZGVsL2ZhbWl4XCI7XHJcbmltcG9ydCB7IEZhbWl4UmVwb3NpdG9yeSB9IGZyb20gXCIuLi9saWIvZmFtaXgvZmFtaXhfcmVwb3NpdG9yeVwiO1xyXG5pbXBvcnQgeyBsb2dnZXIsIGNvbmZpZyB9IGZyb20gXCIuLi9hbmFseXplXCI7XHJcbi8vIGltcG9ydCBHcmFwaGVtZVNwbGl0dGVyIGZyb20gXCJncmFwaGVtZS1zcGxpdHRlclwiO1xyXG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0c1xyXG5pbXBvcnQgR3JhcGhlbWVTcGxpdHRlciA9IHJlcXVpcmUoJ2dyYXBoZW1lLXNwbGl0dGVyJyk7XHJcbmltcG9ydCAqIGFzIEhlbHBlcnMgZnJvbSBcIi4vaGVscGVyc19jcmVhdGlvblwiO1xyXG5pbXBvcnQgKiBhcyBGUU5GdW5jdGlvbnMgZnJvbSBcIi4uL2ZxblwiO1xyXG5pbXBvcnQgcGF0aCBmcm9tIFwicGF0aFwiO1xyXG4vLyBpbXBvcnQgXyBmcm9tICdsb2Rhc2gnO1xyXG5cclxuZXhwb3J0IHR5cGUgVFNNb3JwaE9iamVjdFR5cGUgPSBJbXBvcnREZWNsYXJhdGlvbiB8IEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uIHwgU291cmNlRmlsZSB8IE1vZHVsZURlY2xhcmF0aW9uIHwgQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uIHwgTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgRnVuY3Rpb25EZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRXhwcmVzc2lvbiB8IFBhcmFtZXRlckRlY2xhcmF0aW9uIHwgVmFyaWFibGVEZWNsYXJhdGlvbiB8IFByb3BlcnR5RGVjbGFyYXRpb24gfCBQcm9wZXJ0eVNpZ25hdHVyZSB8IFR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IElkZW50aWZpZXIgfCBEZWNvcmF0b3IgfCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IEltcG9ydFNwZWNpZmllciB8IENvbW1lbnRSYW5nZSB8IEVudW1EZWNsYXJhdGlvbiB8IEVudW1NZW1iZXIgfCBUeXBlQWxpYXNEZWNsYXJhdGlvbiB8IEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cyB8IFRTTW9ycGhQYXJhbWV0cmljVHlwZTtcclxuXHJcbmV4cG9ydCB0eXBlIFRTTW9ycGhUeXBlRGVjbGFyYXRpb24gPSBUeXBlQWxpYXNEZWNsYXJhdGlvbiB8IFByb3BlcnR5RGVjbGFyYXRpb24gfCBQcm9wZXJ0eVNpZ25hdHVyZSB8IENvbnN0cnVjdG9yRGVjbGFyYXRpb24gfCBNZXRob2RTaWduYXR1cmUgfCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRXhwcmVzc2lvbiB8IFBhcmFtZXRlckRlY2xhcmF0aW9uIHwgVmFyaWFibGVEZWNsYXJhdGlvbiB8IEVudW1NZW1iZXIgfCBJbXBvcnRFcXVhbHNEZWNsYXJhdGlvbiB8IFRTTW9ycGhQYXJhbWV0cmljVHlwZSB8IFR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbjtcclxuXHJcbmV4cG9ydCB0eXBlIFRTTW9ycGhQYXJhbWV0cmljVHlwZSA9IENsYXNzRGVjbGFyYXRpb24gfCBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBNZXRob2REZWNsYXJhdGlvbiB8IEFycm93RnVuY3Rpb247XHJcblxyXG4vLyB0eXBlIFBhcmFtZXRyaWNWYXJpYW50VHlwZSA9IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kO1xyXG5cclxuLy8gdHlwZSBDb25jcmV0ZUVsZW1lbnRUU01vcnBoVHlwZSA9IENsYXNzRGVjbGFyYXRpb24gfCBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBNZXRob2REZWNsYXJhdGlvbjtcclxuXHJcbmV4cG9ydCB0eXBlIEludm9jYWJsZVR5cGUgPSBNZXRob2REZWNsYXJhdGlvbiB8IENvbnN0cnVjdG9yRGVjbGFyYXRpb24gfCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24gfCBBcnJvd0Z1bmN0aW9uO1xyXG5cclxuZXhwb3J0IGNsYXNzIEVudGl0eURpY3Rpb25hcnkge1xyXG4gICAgXHJcbiAgICBwdWJsaWMgZmFtaXhSZXAgPSBuZXcgRmFtaXhSZXBvc2l0b3J5KCk7XHJcbiAgICBwcml2YXRlIGZteEFsaWFzTWFwID0gbmV3IE1hcDxzdHJpbmcsIEZhbWl4LkFsaWFzPigpOyAvLyBNYXBzIHRoZSBhbGlhcyBuYW1lcyB0byB0aGVpciBGYW1peCBtb2RlbFxyXG4gICAgcHJpdmF0ZSBmbXhDbGFzc01hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5DbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNDbGFzcz4oKTsgLy8gTWFwcyB0aGUgZnVsbHkgcXVhbGlmaWVkIGNsYXNzIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXHJcbiAgICBwcml2YXRlIGZteEludGVyZmFjZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5JbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlPigpOyAvLyBNYXBzIHRoZSBpbnRlcmZhY2UgbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcclxuICAgIHByaXZhdGUgZm14TW9kdWxlTWFwID0gbmV3IE1hcDxNb2R1bGVEZWNsYXJhdGlvbiwgRmFtaXguTW9kdWxlPigpOyAvLyBNYXBzIHRoZSBuYW1lc3BhY2UgbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcclxuICAgIHByaXZhdGUgZm14RmlsZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5TY3JpcHRFbnRpdHkgfCBGYW1peC5Nb2R1bGU+KCk7IC8vIE1hcHMgdGhlIHNvdXJjZSBmaWxlIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXHJcbiAgICBwcml2YXRlIGZteFR5cGVNYXAgPSBuZXcgTWFwPFRTTW9ycGhUeXBlRGVjbGFyYXRpb24sIEZhbWl4LlR5cGUgfCBGYW1peC5QYXJhbWV0ZXJUeXBlPigpOyAvLyBNYXBzIHRoZSB0eXBlcyBkZWNsYXJhdGlvbnMgdG8gdGhlaXIgRmFtaXggbW9kZWxcclxuICAgIHByaXZhdGUgZm14UHJpbWl0aXZlVHlwZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5QcmltaXRpdmVUeXBlPigpOyAvLyBNYXBzIHRoZSBwcmltaXRpdmUgdHlwZSBuYW1lcyB0byB0aGVpciBGYW1peCBtb2RlbFxyXG4gICAgcHJpdmF0ZSBmbXhGdW5jdGlvbkFuZE1ldGhvZE1hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5GdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbiB8IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LlBhcmFtZXRyaWNNZXRob2Q+OyAvLyBNYXBzIHRoZSBmdW5jdGlvbiBuYW1lcyB0byB0aGVpciBGYW1peCBtb2RlbFxyXG4gICAgcHJpdmF0ZSBmbXhBcnJvd0Z1bmN0aW9uTWFwID0gbmV3IE1hcDxzdHJpbmcsIEZhbWl4LkFycm93RnVuY3Rpb24+OyAvLyBNYXBzIHRoZSBmdW5jdGlvbiBuYW1lcyB0byB0aGVpciBGYW1peCBtb2RlbFxyXG4gICAgcHJpdmF0ZSBmbXhQYXJhbWV0ZXJNYXAgPSBuZXcgTWFwPFBhcmFtZXRlckRlY2xhcmF0aW9uLCBGYW1peC5QYXJhbWV0ZXI+KCk7IC8vIE1hcHMgdGhlIHBhcmFtZXRlcnMgdG8gdGhlaXIgRmFtaXggbW9kZWxcclxuICAgIHByaXZhdGUgZm14VmFyaWFibGVNYXAgPSBuZXcgTWFwPFZhcmlhYmxlRGVjbGFyYXRpb24sIEZhbWl4LlZhcmlhYmxlPigpOyAvLyBNYXBzIHRoZSB2YXJpYWJsZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcclxuICAgIHByaXZhdGUgZm14SW1wb3J0Q2xhdXNlTWFwID0gbmV3IE1hcDxJbXBvcnREZWNsYXJhdGlvbiB8IEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uLCBGYW1peC5JbXBvcnRDbGF1c2U+KCk7IC8vIE1hcHMgdGhlIGltcG9ydCBjbGF1c2VzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXHJcbiAgICBwcml2YXRlIGZteEVudW1NYXAgPSBuZXcgTWFwPEVudW1EZWNsYXJhdGlvbiwgRmFtaXguRW51bT4oKTsgLy8gTWFwcyB0aGUgZW51bSBuYW1lcyB0byB0aGVpciBGYW1peCBtb2RlbFxyXG4gICAgcHJpdmF0ZSBmbXhJbmhlcml0YW5jZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5Jbmhlcml0YW5jZT4oKTsgLy8gTWFwcyB0aGUgaW5oZXJpdGFuY2UgbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcclxuICAgIHByaXZhdGUgVU5LTk9XTl9WQUxVRSA9ICcodW5rbm93biBkdWUgdG8gcGFyc2luZyBlcnJvciknOyAvLyBUaGUgdmFsdWUgdG8gdXNlIHdoZW4gYSBuYW1lIGlzIG5vdCB1c2FibGVcclxuICAgIHB1YmxpYyBmbXhFbGVtZW50T2JqZWN0TWFwID0gbmV3IE1hcDxGYW1peC5FbnRpdHksVFNNb3JwaE9iamVjdFR5cGU+KCk7XHJcbiAgICBwdWJsaWMgdHNNb3JwaEVsZW1lbnRPYmplY3RNYXAgPSBuZXcgTWFwPFRTTW9ycGhPYmplY3RUeXBlLEZhbWl4LkVudGl0eT4oKTtcclxuICAgICAgICAgICAgXHJcbiAgICBjb25zdHJ1Y3RvcigpIHtcclxuICAgICAgICB0aGlzLmZhbWl4UmVwLnNldEZteEVsZW1lbnRPYmplY3RNYXAodGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwKTsgICAgICBcclxuICAgIH1cclxuXHJcbiAgICBwdWJsaWMgYWRkU291cmNlQW5jaG9yKGZteDogRmFtaXguU291cmNlZEVudGl0eSwgbm9kZTogVFNNb3JwaE9iamVjdFR5cGUpOiBGYW1peC5JbmRleGVkRmlsZUFuY2hvciB7XHJcbiAgICAgICAgY29uc3Qgc291cmNlQW5jaG9yOiBGYW1peC5JbmRleGVkRmlsZUFuY2hvciA9IG5ldyBGYW1peC5JbmRleGVkRmlsZUFuY2hvcigpO1xyXG4gICAgICAgIGxldCBzb3VyY2VTdGFydCwgc291cmNlRW5kOiBudW1iZXI7XHJcbiAgICAgICAgaWYgKGZteCAmJiBub2RlKSB7XHJcbiAgICAgICAgICAgIC8vIGZpbmQgdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZSBzb3VyY2UgZWxlbWVudFxyXG4gICAgICAgICAgICBpZiAoIShub2RlIGluc3RhbmNlb2YgQ29tbWVudFJhbmdlKSkge1xyXG4gICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBub2RlLmdldFN0YXJ0KCk7XHJcbiAgICAgICAgICAgICAgICBzb3VyY2VFbmQgPSBub2RlLmdldEVuZCgpO1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBub2RlLmdldFBvcygpO1xyXG4gICAgICAgICAgICAgICAgc291cmNlRW5kID0gbm9kZS5nZXRFbmQoKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgXHJcbiAgICAgICAgICAgIGlmIChjb25maWcuZXhwZWN0R3JhcGhlbWVzKSB7XHJcbiAgICAgICAgICAgICAgICAvKipcclxuICAgICAgICAgICAgICAgICAqIFRoZSBmb2xsb3dpbmcgbG9naWMgaGFuZGxlcyB0aGUgY2FzZSBvZiBtdWx0aS1jb2RlIHBvaW50IGNoYXJhY3RlcnMgKGUuZy4gZW1vamkpIGluIHRoZSBzb3VyY2UgdGV4dC5cclxuICAgICAgICAgICAgICAgICAqIFRoaXMgaXMgbmVlZGVkIGJlY2F1c2UgUGhhcm8vU21hbGx0YWxrIHRyZWF0cyBtdWx0aS1jb2RlIHBvaW50IGNoYXJhY3RlcnMgYXMgYSBzaW5nbGUgY2hhcmFjdGVyLCBcclxuICAgICAgICAgICAgICAgICAqIGJ1dCBKYXZhU2NyaXB0IHRyZWF0cyB0aGVtIGFzIG11bHRpcGxlIGNoYXJhY3RlcnMuIFRoaXMgbWVhbnMgdGhhdCB0aGUgc3RhcnQgYW5kIGVuZCBwb3NpdGlvbnNcclxuICAgICAgICAgICAgICAgICAqIG9mIGEgc291cmNlIGVsZW1lbnQgaW4gUGhhcm8vU21hbGx0YWxrIHdpbGwgYmUgZGlmZmVyZW50IHRoYW4gdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZVxyXG4gICAgICAgICAgICAgICAgICogc2FtZSBzb3VyY2UgZWxlbWVudCBpbiBKYXZhU2NyaXB0LiBUaGlzIGxvZ2ljIGZpbmRzIHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGUgc291cmNlXHJcbiAgICAgICAgICAgICAgICAgKiBlbGVtZW50IGluIEphdmFTY3JpcHQgYW5kIHRoZW4gdXNlcyB0aG9zZSBwb3NpdGlvbnMgdG8gc2V0IHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGVcclxuICAgICAgICAgICAgICAgICAqIEZhbWl4IGluZGV4IGZpbGUgYW5jaG9yLlxyXG4gICAgICAgICAgICAgICAgICogSXQgZGVwZW5kcyBvbiBjb2RlIGluIHRoZSAnZ3JhcGhlbWUtc3BsaXR0ZXInIHBhY2thZ2UgaW4gbnBtLlxyXG4gICAgICAgICAgICAgICAgICovXHJcbiAgICAgICAgICAgICAgICBjb25zdCBzcGxpdHRlciA9IG5ldyBHcmFwaGVtZVNwbGl0dGVyKCk7XHJcbiAgICAgICAgICAgICAgICBjb25zdCBzb3VyY2VGaWxlVGV4dCA9IG5vZGUuZ2V0U291cmNlRmlsZSgpLmdldEZ1bGxUZXh0KCk7XHJcbiAgICAgICAgICAgICAgICBjb25zdCBoYXNHcmFwaGVtZUNsdXN0ZXJzID0gc3BsaXR0ZXIuY291bnRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQpID4gMTtcclxuICAgICAgICAgICAgICAgIGlmIChoYXNHcmFwaGVtZUNsdXN0ZXJzKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRWxlbWVudFRleHQgPSBzb3VyY2VGaWxlVGV4dC5zdWJzdHJpbmcoc291cmNlU3RhcnQsIHNvdXJjZUVuZCk7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMgPSBzcGxpdHRlci5zcGxpdEdyYXBoZW1lcyhzb3VyY2VFbGVtZW50VGV4dCk7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRmlsZVRleHRHcmFwaGVtZXMgPSBzcGxpdHRlci5zcGxpdEdyYXBoZW1lcyhzb3VyY2VGaWxlVGV4dCk7XHJcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgbnVtYmVyT2ZHcmFwaGVtZUNsdXN0ZXJzQmVmb3JlU3RhcnQgPSBzcGxpdHRlci5jb3VudEdyYXBoZW1lcyhzb3VyY2VGaWxlVGV4dC5zdWJzdHJpbmcoMCwgc291cmNlU3RhcnQpKTtcclxuICAgICAgICBcclxuICAgICAgICAgICAgICAgICAgICAvLyBmaW5kIHRoZSBzdGFydCBvZiB0aGUgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMgYXJyYXkgaW4gdGhlIHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzIGFycmF5XHJcbiAgICAgICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBIZWxwZXJzLmluZGV4T2ZTcGxpdEFycmF5KHtzZWFyY2hBcnJheTogc291cmNlRmlsZVRleHRHcmFwaGVtZXMsIFxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRhcmdldEFycmF5OiBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcywgXHJcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3RhcnQ6IHNvdXJjZVN0YXJ0IC0gbnVtYmVyT2ZHcmFwaGVtZUNsdXN0ZXJzQmVmb3JlU3RhcnR9KTtcclxuICAgICAgICAgICAgICAgICAgICBzb3VyY2VFbmQgPSBzb3VyY2VTdGFydCArIHNvdXJjZUVsZW1lbnRUZXh0R3JhcGhlbWVzLmxlbmd0aDtcclxuICAgICAgICAgICAgICAgIH0gXHJcbiAgICAgICAgICAgIH1cclxuXHJcbiAgICAgICAgICAgIC8vIFRoZSArMSBpcyBiZWNhdXNlIHRoZSBzb3VyY2UgYW5jaG9yIChQaGFybykgaXMgMS1iYXNlZCwgYnV0IHRzLW1vcnBoIGlzIDAtYmFzZWRcclxuICAgICAgICAgICAgc291cmNlQW5jaG9yLnN0YXJ0UG9zID0gc291cmNlU3RhcnQgKyAxO1xyXG4gICAgICAgICAgICBzb3VyY2VBbmNob3IuZW5kUG9zID0gc291cmNlRW5kICsgMTtcclxuXHJcbiAgICAgICAgICAgIGxldCBmaWxlTmFtZSA9IG5vZGUuZ2V0U291cmNlRmlsZSgpLmdldEZpbGVQYXRoKCkgYXMgc3RyaW5nO1xyXG4gICAgICAgICAgICBpZiAoZmlsZU5hbWUuc3RhcnRzV2l0aChcIi9cIikpIHtcclxuICAgICAgICAgICAgICAgIGZpbGVOYW1lID0gZmlsZU5hbWUuc3Vic3RyaW5nKDEpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIHNvdXJjZUFuY2hvci5lbGVtZW50ID0gZm14O1xyXG4gICAgICAgICAgICBzb3VyY2VBbmNob3IuZmlsZU5hbWUgPSBmaWxlTmFtZTtcclxuICAgICAgICAgICAgZm14LnNvdXJjZUFuY2hvciA9IHNvdXJjZUFuY2hvcjtcclxuICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KHNvdXJjZUFuY2hvcik7XHJcblxyXG4gICAgICAgIH1cclxuICAgICAgICByZXR1cm4gc291cmNlQW5jaG9yO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogTWFrZXMgYSBGYW1peCBpbmRleCBmaWxlIGFuY2hvclxyXG4gICAgICogQHBhcmFtIHNvdXJjZUVsZW1lbnQgQSBzb3VyY2UgZWxlbWVudFxyXG4gICAgICogQHBhcmFtIGZhbWl4RWxlbWVudCBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHNvdXJjZSBlbGVtZW50XHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBtYWtlRmFtaXhJbmRleEZpbGVBbmNob3Ioc291cmNlRWxlbWVudDogVFNNb3JwaE9iamVjdFR5cGUsIGZhbWl4RWxlbWVudDogRmFtaXguU291cmNlZEVudGl0eSk6IHZvaWQge1xyXG4gICAgICAgIC8vIEZhbWl4LkNvbW1lbnQgaXMgbm90IGEgbmFtZWQgZW50aXR5IChkb2VzIG5vdCBoYXZlIGEgZnVsbHlRdWFsaWZpZWROYW1lKVxyXG4gICAgICAgIGlmICghKGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LkNvbW1lbnQpKSB7ICAvLyBtdXN0IGJlIGEgbmFtZWQgZW50aXR5XHJcbiAgICAgICAgICAgIC8vIGluc2FuaXR5IGNoZWNrOiBuYW1lZCBlbnRpdGllcyBzaG91bGQgaGF2ZSBmdWxseVF1YWxpZmllZE5hbWVcclxuICAgICAgICAgICAgY29uc3QgZnVsbHlRdWFsaWZpZWROYW1lID0gKGZhbWl4RWxlbWVudCBhcyBGYW1peC5OYW1lZEVudGl0eSkuZnVsbHlRdWFsaWZpZWROYW1lO1xyXG4gICAgICAgICAgICBpZiAoIWZ1bGx5UXVhbGlmaWVkTmFtZSB8fCBmdWxseVF1YWxpZmllZE5hbWUgPT09IHRoaXMuVU5LTk9XTl9WQUxVRSkge1xyXG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBGYW1peCBlbGVtZW50ICR7ZmFtaXhFbGVtZW50LmNvbnN0cnVjdG9yLm5hbWV9IGhhcyBubyB2YWxpZCBmdWxseVF1YWxpZmllZE5hbWUuYCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhcIm1ha2luZyBpbmRleCBmaWxlIGFuY2hvciBmb3IgJ1wiICsgc291cmNlRWxlbWVudD8uZ2V0VGV4dCgpICsgXCInIHdpdGggZmFtaXhFbGVtZW50IFwiICsgZmFtaXhFbGVtZW50LmdldEpTT04oKSk7XHJcbiAgICAgICAgY29uc3QgZm14SW5kZXhGaWxlQW5jaG9yID0gbmV3IEZhbWl4LkluZGV4ZWRGaWxlQW5jaG9yKCk7XHJcbiAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmVsZW1lbnQgPSBmYW1peEVsZW1lbnQ7XHJcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmYW1peEVsZW1lbnQsIHNvdXJjZUVsZW1lbnQpO1xyXG5cclxuICAgICAgICBpZiAoc291cmNlRWxlbWVudCAhPT0gbnVsbCkge1xyXG4gICAgICAgICAgICBjb25zdCBhYnNvbHV0ZVBhdGhQcm9qZWN0ID0gdGhpcy5mYW1peFJlcC5nZXRBYnNvbHV0ZVBhdGgoKTtcclxuICAgICAgICBcclxuICAgICAgICAgICAgY29uc3QgYWJzb2x1dGVQYXRoID0gcGF0aC5ub3JtYWxpemUoc291cmNlRWxlbWVudC5nZXRTb3VyY2VGaWxlKCkuZ2V0RmlsZVBhdGgoKSk7XHJcblxyXG4gICAgICAgICAgICBjb25zdCBwb3NpdGlvbk5vZGVNb2R1bGVzID0gYWJzb2x1dGVQYXRoLmluZGV4T2YoJ25vZGVfbW9kdWxlcycpO1xyXG5cclxuICAgICAgICAgICAgbGV0IHBhdGhJblByb2plY3Q6IHN0cmluZyA9IFwiXCI7XHJcblxyXG4gICAgICAgICAgICBpZiAocG9zaXRpb25Ob2RlTW9kdWxlcyAhPT0gLTEpIHtcclxuICAgICAgICAgICAgICAgIGNvbnN0IHBhdGhGcm9tTm9kZU1vZHVsZXMgPSBhYnNvbHV0ZVBhdGguc3Vic3RyaW5nKHBvc2l0aW9uTm9kZU1vZHVsZXMpO1xyXG4gICAgICAgICAgICAgICAgcGF0aEluUHJvamVjdCA9IHBhdGhGcm9tTm9kZU1vZHVsZXM7XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBwYXRoSW5Qcm9qZWN0ID0gdGhpcy5jb252ZXJ0VG9SZWxhdGl2ZVBhdGgoYWJzb2x1dGVQYXRoLCBhYnNvbHV0ZVBhdGhQcm9qZWN0KTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgLy8gcmV2ZXJ0IGFueSBiYWNrc2xhc2hlcyB0byBmb3J3YXJkIHNsYXNoZXMgKHBhdGgubm9ybWFsaXplIG9uIHdpbmRvd3MgaW50cm9kdWNlcyB0aGVtKVxyXG4gICAgICAgICAgICBwYXRoSW5Qcm9qZWN0ID0gcGF0aEluUHJvamVjdC5yZXBsYWNlKC9cXFxcL2csIFwiL1wiKTtcclxuXHJcbiAgICAgICAgICAgIGlmIChwYXRoSW5Qcm9qZWN0LnN0YXJ0c1dpdGgoXCIvXCIpKSB7XHJcbiAgICAgICAgICAgICAgICBwYXRoSW5Qcm9qZWN0ID0gcGF0aEluUHJvamVjdC5zdWJzdHJpbmcoMSk7XHJcbiAgICAgICAgICAgIH1cclxuXHJcbiAgICAgICAgICAgIGZteEluZGV4RmlsZUFuY2hvci5maWxlTmFtZSA9IHBhdGhJblByb2plY3Q7XHJcbiAgICAgICAgICAgIGxldCBzb3VyY2VTdGFydCwgc291cmNlRW5kXHJcbiAgICAgICAgICAgIC8vICxzb3VyY2VMaW5lU3RhcnQsIHNvdXJjZUxpbmVFbmRcclxuICAgICAgICAgICAgOiBudW1iZXI7XHJcbiAgICAgICAgICAgIGlmICghKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBDb21tZW50UmFuZ2UpKSB7XHJcbiAgICAgICAgICAgICAgICBzb3VyY2VTdGFydCA9IHNvdXJjZUVsZW1lbnQuZ2V0U3RhcnQoKTtcclxuICAgICAgICAgICAgICAgIHNvdXJjZUVuZCA9IHNvdXJjZUVsZW1lbnQuZ2V0RW5kKCk7XHJcbiAgICAgICAgICAgICAgICAvLyBzb3VyY2VMaW5lU3RhcnQgPSBzb3VyY2VFbGVtZW50LmdldFN0YXJ0TGluZU51bWJlcigpO1xyXG4gICAgICAgICAgICAgICAgLy8gc291cmNlTGluZUVuZCA9IHNvdXJjZUVsZW1lbnQuZ2V0RW5kTGluZU51bWJlcigpO1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBzb3VyY2VFbGVtZW50LmdldFBvcygpO1xyXG4gICAgICAgICAgICAgICAgc291cmNlRW5kID0gc291cmNlRWxlbWVudC5nZXRFbmQoKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBpZiAoY29uZmlnLmV4cGVjdEdyYXBoZW1lcykge1xyXG4gICAgICAgICAgICAgICAgLyoqXHJcbiAgICAgICAgICAgICAgICAgKiBUaGUgZm9sbG93aW5nIGxvZ2ljIGhhbmRsZXMgdGhlIGNhc2Ugb2YgbXVsdGktY29kZSBwb2ludCBjaGFyYWN0ZXJzIChlLmcuIGVtb2ppKSBpbiB0aGUgc291cmNlIHRleHQuXHJcbiAgICAgICAgICAgICAgICAgKiBUaGlzIGlzIG5lZWRlZCBiZWNhdXNlIFBoYXJvL1NtYWxsdGFsayB0cmVhdHMgbXVsdGktY29kZSBwb2ludCBjaGFyYWN0ZXJzIGFzIGEgc2luZ2xlIGNoYXJhY3RlciwgXHJcbiAgICAgICAgICAgICAgICAgKiBidXQgSmF2YVNjcmlwdCB0cmVhdHMgdGhlbSBhcyBtdWx0aXBsZSBjaGFyYWN0ZXJzLiBUaGlzIG1lYW5zIHRoYXQgdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zXHJcbiAgICAgICAgICAgICAgICAgKiBvZiBhIHNvdXJjZSBlbGVtZW50IGluIFBoYXJvL1NtYWxsdGFsayB3aWxsIGJlIGRpZmZlcmVudCB0aGFuIHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGVcclxuICAgICAgICAgICAgICAgICAqIHNhbWUgc291cmNlIGVsZW1lbnQgaW4gSmF2YVNjcmlwdC4gVGhpcyBsb2dpYyBmaW5kcyB0aGUgc3RhcnQgYW5kIGVuZCBwb3NpdGlvbnMgb2YgdGhlIHNvdXJjZVxyXG4gICAgICAgICAgICAgICAgICogZWxlbWVudCBpbiBKYXZhU2NyaXB0IGFuZCB0aGVuIHVzZXMgdGhvc2UgcG9zaXRpb25zIHRvIHNldCB0aGUgc3RhcnQgYW5kIGVuZCBwb3NpdGlvbnMgb2YgdGhlXHJcbiAgICAgICAgICAgICAgICAgKiBGYW1peCBpbmRleCBmaWxlIGFuY2hvci5cclxuICAgICAgICAgICAgICAgICAqIEl0IGRlcGVuZHMgb24gY29kZSBpbiB0aGUgJ2dyYXBoZW1lLXNwbGl0dGVyJyBwYWNrYWdlIGluIG5wbS5cclxuICAgICAgICAgICAgICAgICAqL1xyXG4gICAgICAgICAgICAgICAgY29uc3Qgc3BsaXR0ZXIgPSBuZXcgR3JhcGhlbWVTcGxpdHRlcigpO1xyXG4gICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRmlsZVRleHQgPSBzb3VyY2VFbGVtZW50LmdldFNvdXJjZUZpbGUoKS5nZXRGdWxsVGV4dCgpO1xyXG4gICAgICAgICAgICAgICAgY29uc3QgaGFzR3JhcGhlbWVDbHVzdGVycyA9IHNwbGl0dGVyLmNvdW50R3JhcGhlbWVzKHNvdXJjZUZpbGVUZXh0KSA+IDE7XHJcbiAgICAgICAgICAgICAgICBpZiAoaGFzR3JhcGhlbWVDbHVzdGVycykge1xyXG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUVsZW1lbnRUZXh0ID0gc291cmNlRmlsZVRleHQuc3Vic3RyaW5nKHNvdXJjZVN0YXJ0LCBzb3VyY2VFbmQpO1xyXG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUVsZW1lbnRUZXh0R3JhcGhlbWVzID0gc3BsaXR0ZXIuc3BsaXRHcmFwaGVtZXMoc291cmNlRWxlbWVudFRleHQpO1xyXG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzID0gc3BsaXR0ZXIuc3BsaXRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQpO1xyXG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IG51bWJlck9mR3JhcGhlbWVDbHVzdGVyc0JlZm9yZVN0YXJ0ID0gc3BsaXR0ZXIuY291bnRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQuc3Vic3RyaW5nKDAsIHNvdXJjZVN0YXJ0KSk7XHJcbiAgICAgICAgXHJcbiAgICAgICAgICAgICAgICAgICAgLy8gZmluZCB0aGUgc3RhcnQgb2YgdGhlIHNvdXJjZUVsZW1lbnRUZXh0R3JhcGhlbWVzIGFycmF5IGluIHRoZSBzb3VyY2VGaWxlVGV4dEdyYXBoZW1lcyBhcnJheVxyXG4gICAgICAgICAgICAgICAgICAgIHNvdXJjZVN0YXJ0ID0gSGVscGVycy5pbmRleE9mU3BsaXRBcnJheSh7c2VhcmNoQXJyYXk6IHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzLCBcclxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0YXJnZXRBcnJheTogc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMsIFxyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXJ0OiBzb3VyY2VTdGFydCAtIG51bWJlck9mR3JhcGhlbWVDbHVzdGVyc0JlZm9yZVN0YXJ0fSk7XHJcbiAgICAgICAgICAgICAgICAgICAgc291cmNlRW5kID0gc291cmNlU3RhcnQgKyBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcy5sZW5ndGg7XHJcbiAgICAgICAgICAgICAgICB9IFxyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIC8vIG5vdGU6IHRoZSArMSBpcyBiZWNhdXNlIHRoZSBzb3VyY2UgYW5jaG9yIGlzIDEtYmFzZWQsIGJ1dCB0cy1tb3JwaCBpcyAwLWJhc2VkXHJcbiAgICAgICAgICAgIGZteEluZGV4RmlsZUFuY2hvci5zdGFydFBvcyA9IHNvdXJjZVN0YXJ0ICsgMTtcclxuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmVuZFBvcyA9IHNvdXJjZUVuZCArIDE7XHJcblxyXG4gICAgICAgICAgICAvLyBpZiAoIShmYW1peEVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5JbXBvcnRDbGF1c2UgfHwgZmFtaXhFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguQWNjZXNzIHx8IGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlJlZmVyZW5jZSB8fCBmYW1peEVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5JbnZvY2F0aW9uIHx8IGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LkluaGVyaXRhbmNlKSAmJiAhKGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LkNvbW1lbnQpICYmICEoc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIENvbW1lbnRSYW5nZSkgJiYgIShzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgSWRlbnRpZmllcikgJiYgIShzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgSW1wb3J0U3BlY2lmaWVyKSAmJiAhKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHMpKSB7XHJcbiAgICAgICAgICAgIC8vICAgIGluaXRGUU4oc291cmNlRWxlbWVudCwgZmFtaXhFbGVtZW50KTtcclxuICAgICAgICAgICAgLy8gfVxyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIC8vIHNvdXJjZUVsZW1lbnQgaXMgbnVsbFxyXG4gICAgICAgICAgICBsb2dnZXIud2FybihcInNvdXJjZUVsZW1lbnQgaXMgbnVsbCBmb3IgZmFtaXhFbGVtZW50IFwiICsgZmFtaXhFbGVtZW50LmdldEpTT04oKSk7XHJcbiAgICAgICAgICAgIGZteEluZGV4RmlsZUFuY2hvci5maWxlTmFtZSA9IFwidW5rbm93blwiO1xyXG4gICAgICAgICAgICBmbXhJbmRleEZpbGVBbmNob3Iuc3RhcnRQb3MgPSAwO1xyXG4gICAgICAgICAgICBmbXhJbmRleEZpbGVBbmNob3IuZW5kUG9zID0gMDtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhJbmRleEZpbGVBbmNob3IpO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBvciBnZXRzIGEgRmFtaXggc2NyaXB0IGVudGl0eSBvciBtb2R1bGVcclxuICAgICAqIEBwYXJhbSBmIEEgc291cmNlIGZpbGVcclxuICAgICAqIEBwYXJhbSBpc01vZHVsZSBBIGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgc291cmNlIGZpbGUgaXMgYSBtb2R1bGVcclxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgc291cmNlIGZpbGVcclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhGaWxlKGY6IFNvdXJjZUZpbGUsIGlzTW9kdWxlOiBib29sZWFuKTogRmFtaXguU2NyaXB0RW50aXR5IHwgRmFtaXguTW9kdWxlIHtcclxuICAgICAgICBsZXQgZm14RmlsZTogRmFtaXguU2NyaXB0RW50aXR5OyAvLyB8IEZhbWl4Lk1vZHVsZTtcclxuXHJcbiAgICAgICAgY29uc3QgZmlsZU5hbWUgPSBmLmdldEJhc2VOYW1lKCk7XHJcbiAgICAgICAgY29uc3QgZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSA9IGYuZ2V0RmlsZVBhdGgoKTtcclxuICAgICAgICBjb25zdCBmb3VuZEZpbGVOYW1lID0gdGhpcy5mbXhGaWxlTWFwLmdldChmdWxseVF1YWxpZmllZEZpbGVuYW1lKTtcclxuICAgICAgICBpZiAoIWZvdW5kRmlsZU5hbWUpIHtcclxuICAgICAgICAgICAgaWYgKGlzTW9kdWxlKSB7XHJcbiAgICAgICAgICAgICAgICBmbXhGaWxlID0gbmV3IEZhbWl4Lk1vZHVsZSgpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgZm14RmlsZSA9IG5ldyBGYW1peC5TY3JpcHRFbnRpdHkoKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBmbXhGaWxlLm5hbWUgPSBmaWxlTmFtZTtcclxuICAgICAgICAgICAgZm14RmlsZS5udW1iZXJPZkxpbmVzT2ZUZXh0ID0gZi5nZXRFbmRMaW5lTnVtYmVyKCkgLSBmLmdldFN0YXJ0TGluZU51bWJlcigpO1xyXG4gICAgICAgICAgICBmbXhGaWxlLm51bWJlck9mQ2hhcmFjdGVycyA9IGYuZ2V0RnVsbFRleHQoKS5sZW5ndGg7XHJcblxyXG4gICAgICAgICAgICBpbml0RlFOKGYsIGZteEZpbGUpO1xyXG5cclxuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IoZiwgZm14RmlsZSk7XHJcblxyXG4gICAgICAgICAgICB0aGlzLmZteEZpbGVNYXAuc2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUsIGZteEZpbGUpO1xyXG4gICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14RmlsZSk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICBmbXhGaWxlID0gZm91bmRGaWxlTmFtZTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14RmlsZSxmKTtcclxuICAgICAgICByZXR1cm4gZm14RmlsZTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgb3IgZ2V0cyBhIEZhbWl4IE1vZHVsZVxyXG4gICAgICogQHBhcmFtIG1vZHVsZURlY2xhcmF0aW9uIEEgbW9kdWxlXHJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIG1vZHVsZVxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peE1vZHVsZShtb2R1bGVEZWNsYXJhdGlvbjogTW9kdWxlRGVjbGFyYXRpb24pOiBGYW1peC5Nb2R1bGUge1xyXG4gICAgICAgIGlmICh0aGlzLmZteE1vZHVsZU1hcC5oYXMobW9kdWxlRGVjbGFyYXRpb24pKSB7XHJcbiAgICAgICAgICAgIGNvbnN0IHJNb2R1bGUgPSB0aGlzLmZteE1vZHVsZU1hcC5nZXQobW9kdWxlRGVjbGFyYXRpb24pO1xyXG4gICAgICAgICAgICBpZiAock1vZHVsZSkgeyBcclxuICAgICAgICAgICAgICAgcmV0dXJuIHJNb2R1bGU7XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhbWl4IG1vZHVsZSAke21vZHVsZURlY2xhcmF0aW9uLmdldE5hbWUoKX0gaXMgbm90IGZvdW5kIGluIHRoZSBtb2R1bGUgbWFwLmApO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBmbXhNb2R1bGUgPSBuZXcgRmFtaXguTW9kdWxlKCk7XHJcbiAgICAgICAgY29uc3QgbW9kdWxlTmFtZSA9IG1vZHVsZURlY2xhcmF0aW9uLmdldE5hbWUoKTtcclxuICAgICAgICBmbXhNb2R1bGUubmFtZSA9IG1vZHVsZU5hbWU7XHJcbiAgICAgICAgZm14TW9kdWxlLmlzQW1iaWVudCA9IGlzQW1iaWVudChtb2R1bGVEZWNsYXJhdGlvbik7XHJcbiAgICAgICAgZm14TW9kdWxlLmlzTmFtZXNwYWNlID0gaXNOYW1lc3BhY2UobW9kdWxlRGVjbGFyYXRpb24pO1xyXG4gICAgICAgIGZteE1vZHVsZS5pc01vZHVsZSA9ICFmbXhNb2R1bGUuaXNOYW1lc3BhY2UgJiYgIWZteE1vZHVsZS5pc0FtYmllbnQ7XHJcblxyXG4gICAgICAgIGluaXRGUU4obW9kdWxlRGVjbGFyYXRpb24sIGZteE1vZHVsZSk7XHJcbiAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IobW9kdWxlRGVjbGFyYXRpb24sIGZteE1vZHVsZSk7XHJcblxyXG4gICAgICAgIHRoaXMuZm14TW9kdWxlTWFwLnNldChtb2R1bGVEZWNsYXJhdGlvbiwgZm14TW9kdWxlKTtcclxuXHJcbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteE1vZHVsZSk7XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14TW9kdWxlLG1vZHVsZURlY2xhcmF0aW9uKTtcclxuICAgICAgICByZXR1cm4gZm14TW9kdWxlO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGFsaWFzXHJcbiAgICAgKiBAcGFyYW0gYSBBbiBhbGlhc1xyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBhbGlhc1xyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhBbGlhcyhhOiBUeXBlQWxpYXNEZWNsYXJhdGlvbik6IEZhbWl4LkFsaWFzIHtcclxuICAgICAgICBsZXQgZm14QWxpYXM6IEZhbWl4LkFsaWFzO1xyXG4gICAgICAgIGNvbnN0IGFsaWFzTmFtZSA9IGEuZ2V0TmFtZSgpO1xyXG4gICAgICAgIC8vY29uc3QgYWxpYXNGdWxseVF1YWxpZmllZE5hbWUgPSBhLmdldFR5cGUoKS5nZXRUZXh0KCk7IC8vIEZRTkZ1bmN0aW9ucy5nZXRGUU4oYSk7XHJcbiAgICAgICAgY29uc3QgYWxpYXNGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGEpO1xyXG4gICAgICAgIGNvbnN0IGZvdW5kQWxpYXMgPSB0aGlzLmZteEFsaWFzTWFwLmdldChhbGlhc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XHJcbiAgICAgICAgaWYgKCFmb3VuZEFsaWFzKSB7XHJcbiAgICAgICAgICAgIGZteEFsaWFzID0gbmV3IEZhbWl4LkFsaWFzKCk7XHJcbiAgICAgICAgICAgIGZteEFsaWFzLm5hbWUgPSBhLmdldE5hbWUoKTtcclxuICAgICAgICAgICAgY29uc3QgYWxpYXNOYW1lV2l0aEdlbmVyaWNzID0gYWxpYXNOYW1lICsgKGEuZ2V0VHlwZVBhcmFtZXRlcnMoKS5sZW5ndGggPyAoXCI8XCIgKyBhLmdldFR5cGVQYXJhbWV0ZXJzKCkubWFwKHRwID0+IHRwLmdldE5hbWUoKSkuam9pbihcIiwgXCIpICsgXCI+XCIpIDogXCJcIik7XHJcbiAgICAgICAgICAgIGxvZ2dlci5kZWJ1ZyhgPiBOT1RFOiBhbGlhcyAke2FsaWFzTmFtZX0gaGFzIGZ1bGx5IHF1YWxpZmllZCBuYW1lICR7YWxpYXNGdWxseVF1YWxpZmllZE5hbWV9IGFuZCBuYW1lIHdpdGggZ2VuZXJpY3MgJHthbGlhc05hbWVXaXRoR2VuZXJpY3N9LmApO1xyXG5cclxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoYWxpYXNOYW1lV2l0aEdlbmVyaWNzLCBhKTtcclxuICAgICAgICAgICAgZm14QWxpYXMuYWxpYXNlZEVudGl0eSA9IGZteFR5cGU7XHJcbiAgICAgICAgICAgIGluaXRGUU4oYSwgZm14QWxpYXMpO1xyXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihhLCBmbXhBbGlhcyk7XHJcblxyXG4gICAgICAgICAgICB0aGlzLmZteEFsaWFzTWFwLnNldChhbGlhc0Z1bGx5UXVhbGlmaWVkTmFtZSwgZm14QWxpYXMpO1xyXG5cclxuICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEFsaWFzKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgIGZteEFsaWFzID0gZm91bmRBbGlhcztcclxuICAgICAgICB9XHJcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhBbGlhcyxhKTtcclxuXHJcbiAgICAgICAgcmV0dXJuIGZteEFsaWFzO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBvciBnZXRzIGEgRmFtaXggY2xhc3Mgb3IgcGFyYW1ldGVyaXphYmxlIGNsYXNzXHJcbiAgICAgKiBAcGFyYW0gY2xzIEEgY2xhc3NcclxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgY2xhc3NcclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhDbGFzcyhjbHM6IENsYXNzRGVjbGFyYXRpb24pOiBGYW1peC5DbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB7XHJcbiAgICAgICAgbGV0IGZteENsYXNzOiBGYW1peC5DbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcclxuICAgICAgICBjb25zdCBpc0Fic3RyYWN0ID0gY2xzLmlzQWJzdHJhY3QoKTtcclxuICAgICAgICBjb25zdCBjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oY2xzKTtcclxuICAgICAgICBjb25zdCBjbHNOYW1lID0gY2xzLmdldE5hbWUoKSB8fCB0aGlzLlVOS05PV05fVkFMVUU7XHJcbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gY2xzLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoO1xyXG4gICAgICAgIGNvbnN0IGZvdW5kQ2xhc3MgPSB0aGlzLmZteENsYXNzTWFwLmdldChjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XHJcbiAgICAgICAgaWYgKCFmb3VuZENsYXNzKSB7XHJcbiAgICAgICAgICAgIGlmIChpc0dlbmVyaWMpIHtcclxuICAgICAgICAgICAgICAgIGZteENsYXNzID0gbmV3IEZhbWl4LlBhcmFtZXRyaWNDbGFzcygpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgZm14Q2xhc3MgPSBuZXcgRmFtaXguQ2xhc3MoKTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgZm14Q2xhc3MubmFtZSA9IGNsc05hbWU7XHJcbiAgICAgICAgICAgIGluaXRGUU4oY2xzLCBmbXhDbGFzcyk7XHJcbiAgICAgICAgICAgIC8vIGZteENsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZSA9IGNsYXNzRnVsbHlRdWFsaWZpZWROYW1lO1xyXG4gICAgICAgICAgICBmbXhDbGFzcy5pc0Fic3RyYWN0ID0gaXNBYnN0cmFjdDtcclxuXHJcbiAgICAgICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGNscywgZm14Q2xhc3MpO1xyXG5cclxuICAgICAgICAgICAgdGhpcy5mbXhDbGFzc01hcC5zZXQoY2xhc3NGdWxseVF1YWxpZmllZE5hbWUsIGZteENsYXNzKTtcclxuXHJcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhDbGFzcyk7XHJcblxyXG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteENsYXNzLGNscyk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICBmbXhDbGFzcyA9IGZvdW5kQ2xhc3M7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICByZXR1cm4gZm14Q2xhc3M7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBpbnRlcmZhY2Ugb3IgcGFyYW1ldGVyaXphYmxlIGludGVyZmFjZVxyXG4gICAgICogQHBhcmFtIGludGVyIEFuIGludGVyZmFjZVxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBpbnRlcmZhY2VcclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoaW50ZXI6IEludGVyZmFjZURlY2xhcmF0aW9uKTogRmFtaXguSW50ZXJmYWNlIHwgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSB7XHJcblxyXG4gICAgICAgIGxldCBmbXhJbnRlcmZhY2U6IEZhbWl4LkludGVyZmFjZSB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XHJcbiAgICAgICAgY29uc3QgaW50ZXJOYW1lID0gaW50ZXIuZ2V0TmFtZSgpO1xyXG4gICAgICAgIGNvbnN0IGludGVyRnVsbHlRdWFsaWZpZWROYW1lID0gRlFORnVuY3Rpb25zLmdldEZRTihpbnRlcik7XHJcbiAgICAgICAgY29uc3QgZm91bmRJbnRlcmZhY2UgPSB0aGlzLmZteEludGVyZmFjZU1hcC5nZXQoaW50ZXJGdWxseVF1YWxpZmllZE5hbWUpO1xyXG4gICAgICAgIGlmICghZm91bmRJbnRlcmZhY2UpIHtcclxuICAgICAgICAgICAgY29uc3QgaXNHZW5lcmljID0gaW50ZXIuZ2V0VHlwZVBhcmFtZXRlcnMoKS5sZW5ndGg7XHJcbiAgICAgICAgICAgIGlmIChpc0dlbmVyaWMpIHtcclxuICAgICAgICAgICAgICAgIGZteEludGVyZmFjZSA9IG5ldyBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlKCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBmbXhJbnRlcmZhY2UgPSBuZXcgRmFtaXguSW50ZXJmYWNlKCk7XHJcbiAgICAgICAgICAgIH1cclxuXHJcbiAgICAgICAgICAgIGZteEludGVyZmFjZS5uYW1lID0gaW50ZXJOYW1lO1xyXG4gICAgICAgICAgICBpbml0RlFOKGludGVyLCBmbXhJbnRlcmZhY2UpO1xyXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbnRlciwgZm14SW50ZXJmYWNlKTtcclxuXHJcbiAgICAgICAgICAgIHRoaXMuZm14SW50ZXJmYWNlTWFwLnNldChpbnRlckZ1bGx5UXVhbGlmaWVkTmFtZSwgZm14SW50ZXJmYWNlKTtcclxuXHJcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhJbnRlcmZhY2UpO1xyXG5cclxuICAgICAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhJbnRlcmZhY2UsaW50ZXIpO1xyXG4gICAgICAgIH1cclxuICAgICAgICBlbHNlIHtcclxuICAgICAgICAgICAgZm14SW50ZXJmYWNlID0gZm91bmRJbnRlcmZhY2U7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHJldHVybiBmbXhJbnRlcmZhY2U7XHJcbiAgICB9XHJcblxyXG4gICAgXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgb3IgZ2V0cyBhIEZhbWl4IGNvbmNyZXRlIGVsZW1lbnRcclxuICAgICAqIEBwYXJhbSBjb25jcmV0ZUVsZW1lbnQgQSBwYXJhbWV0cmljIEVsZW1lbnQgICBcclxuICAgICAqIEBwYXJhbSBjb25jcmV0ZUVsZW1lbnREZWNsYXJhdGlvbiB0aGUgZWxlbWVudCBkZWNsYXJhdGlvblxyXG4gICAgICogQHBhcmFtIGNvbmNyZXRlQXJndW1lbnRzIGNvbmNyZXRlIGFyZ3VtZW50c1xyXG4gICAgICogQHJldHVybnMgQSBwYXJhbWV0cmljIEVsZW1lbnQgIFxyXG4gICAgICovXHJcbiAgICAvLyBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChjb25jcmV0ZUVsZW1lbnQgOiBQYXJhbWV0cmljVmFyaWFudFR5cGUsIFxyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uY3JldGVFbGVtZW50RGVjbGFyYXRpb24gOiBDb25jcmV0ZUVsZW1lbnRUU01vcnBoVHlwZSwgXHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0ZUFyZ3VtZW50czogVHlwZU5vZGVbXSk6IFBhcmFtZXRyaWNWYXJpYW50VHlwZSB7XHJcbiAgICAgICAgXHJcbiAgICAvLyAgICAgbGV0IGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUgPSBjb25jcmV0ZUVsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lO1xyXG4gICAgLy8gICAgIGxldCBwYXJhbXMgPSBcIlwiO1xyXG4gICAgICAgIFxyXG4gICAgLy8gICAgIGNvbmNyZXRlQXJndW1lbnRzLm1hcCgocGFyYW0pID0+IHtcclxuICAgIC8vICAgICAgICAgcGFyYW1zID0gcGFyYW1zK3BhcmFtLmdldFRleHQoKSsnLCc7XHJcbiAgICAvLyAgICAgfSk7XHJcbiAgICAgICAgXHJcbiAgICAvLyAgICAgcGFyYW1zID0gcGFyYW1zLnN1YnN0cmluZygwLCBwYXJhbXMubGVuZ3RoIC0gMSlcclxuICAgICAgICAgICAgICAgIFxyXG4gICAgLy8gICAgIGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUgPSBIZWxwZXJzLnJlcGxhY2VMYXN0QmV0d2VlblRhZ3MoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSxwYXJhbXMpO1xyXG5cclxuICAgIC8vICAgICBsZXQgY29uY0VsZW1lbnQ6IFBhcmFtZXRyaWNWYXJpYW50VHlwZTtcclxuXHJcbiAgICAvLyAgICAgaWYgKCF0aGlzLmZteEludGVyZmFjZU1hcC5oYXMoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSkgJiYgXHJcbiAgICAvLyAgICAgICAgICF0aGlzLmZteENsYXNzTWFwLmhhcyhmdWxseVF1YWxpZmllZEZpbGVuYW1lKSAmJiBcclxuICAgIC8vICAgICAgICAgIXRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuaGFzKGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpKXtcclxuICAgIC8vICAgICAgICAgY29uY0VsZW1lbnQgPSBfLmNsb25lRGVlcChjb25jcmV0ZUVsZW1lbnQpOyBcclxuICAgIC8vICAgICAgICAgY29uY0VsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lID0gZnVsbHlRdWFsaWZpZWRGaWxlbmFtZTtcclxuICAgIC8vICAgICAgICAgY29uY0VsZW1lbnQuY2xlYXJHZW5lcmljUGFyYW1ldGVycygpO1xyXG4gICAgLy8gICAgICAgICBjb25jcmV0ZUFyZ3VtZW50cy5tYXAoKHBhcmFtKSA9PiB7XHJcbiAgICAvLyAgICAgICAgICAgICBjb25zdCBwYXJhbWV0ZXIgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZVR5cGUocGFyYW0pO1xyXG4gICAgLy8gICAgICAgICAgICAgY29uY0VsZW1lbnQuYWRkQ29uY3JldGVQYXJhbWV0ZXIocGFyYW1ldGVyKTtcclxuICAgIC8vICAgICAgICAgfSlcclxuICAgICAgICAgICAgXHJcbiAgICAvLyAgICAgICAgIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljQ2xhc3MpIHtcclxuICAgIC8vICAgICAgICAgICAgIHRoaXMuZm14Q2xhc3NNYXAuc2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUsIGNvbmNFbGVtZW50IGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcyk7XHJcbiAgICAvLyAgICAgICAgIH0gZWxzZSBpZiAoY29uY3JldGVFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSkge1xyXG4gICAgLy8gICAgICAgICAgICAgdGhpcy5mbXhJbnRlcmZhY2VNYXAuc2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUsIGNvbmNFbGVtZW50IGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UpO1xyXG4gICAgLy8gICAgICAgICB9IGVsc2UgaWYgKGNvbmNyZXRlRWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbikge1xyXG4gICAgLy8gICAgICAgICAgICAgdGhpcy5mbXhGdW5jdGlvbkFuZE1ldGhvZE1hcC5zZXQoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSwgY29uY0VsZW1lbnQgYXMgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uKTtcclxuICAgIC8vICAgICAgICAgfSBlbHNlIHsgLy8gaWYgKGNvbmNyZXRlRWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNNZXRob2QpIHtcclxuICAgIC8vICAgICAgICAgICAgIHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuc2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUsIGNvbmNFbGVtZW50IGFzIEZhbWl4LlBhcmFtZXRyaWNNZXRob2QpO1xyXG4gICAgLy8gICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChjb25jRWxlbWVudCk7XHJcbiAgICAvLyAgICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoY29uY0VsZW1lbnQsY29uY3JldGVFbGVtZW50RGVjbGFyYXRpb24pO1xyXG4gICAgLy8gICAgIH0gZWxzZSB7XHJcbiAgICAvLyAgICAgICAgIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljQ2xhc3MpIHtcclxuICAgIC8vICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5mbXhDbGFzc01hcC5nZXQoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSkgYXMgRmFtaXguUGFyYW1ldHJpY0NsYXNzO1xyXG4gICAgLy8gICAgICAgICB9IGVsc2UgaWYgKGNvbmNyZXRlRWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UpIHtcclxuICAgIC8vICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5mbXhJbnRlcmZhY2VNYXAuZ2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XHJcbiAgICAvLyAgICAgICAgIH0gZWxzZSBpZiAoY29uY3JldGVFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICBjb25jRWxlbWVudCA9IHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuZ2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpIGFzIEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbjtcclxuICAgIC8vICAgICAgICAgfSBlbHNlIHsgIC8vIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljTWV0aG9kKSB7XHJcbiAgICAvLyAgICAgICAgICAgICBjb25jRWxlbWVudCA9IHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuZ2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpIGFzIEZhbWl4LlBhcmFtZXRyaWNNZXRob2Q7XHJcbiAgICAvLyAgICAgICAgIH1cclxuICAgIC8vICAgICB9XHJcbiAgICAvLyAgICAgcmV0dXJuIGNvbmNFbGVtZW50O1xyXG4gICAgLy8gfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IHByb3BlcnR5XHJcbiAgICAgKiBAcGFyYW0gcHJvcGVydHkgQSBwcm9wZXJ0eVxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBwcm9wZXJ0eVxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhQcm9wZXJ0eShwcm9wZXJ0eTogUHJvcGVydHlEZWNsYXJhdGlvbiB8IFByb3BlcnR5U2lnbmF0dXJlKTogRmFtaXguUHJvcGVydHkge1xyXG4gICAgICAgIGNvbnN0IGZteFByb3BlcnR5ID0gbmV3IEZhbWl4LlByb3BlcnR5KCk7XHJcbiAgICAgICAgY29uc3QgaXNTaWduYXR1cmUgPSBwcm9wZXJ0eSBpbnN0YW5jZW9mIFByb3BlcnR5U2lnbmF0dXJlO1xyXG4gICAgICAgIGZteFByb3BlcnR5Lm5hbWUgPSBwcm9wZXJ0eS5nZXROYW1lKCk7XHJcblxyXG4gICAgICAgIGxldCBwcm9wVHlwZU5hbWUgPSB0aGlzLlVOS05PV05fVkFMVUU7XHJcbiAgICAgICAgdHJ5IHtcclxuICAgICAgICAgICAgcHJvcFR5cGVOYW1lID0gcHJvcGVydHkuZ2V0VHlwZSgpLmdldFRleHQoKS50cmltKCk7XHJcbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHByb3BlcnR5OiAke3Byb3BlcnR5LmdldE5hbWUoKX0uIENvbnRpbnVpbmcuLi5gKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKHByb3BUeXBlTmFtZSwgcHJvcGVydHkpO1xyXG4gICAgICAgIGZteFByb3BlcnR5LmRlY2xhcmVkVHlwZSA9IGZteFR5cGU7XHJcblxyXG4gICAgICAgIC8vIGFkZCB0aGUgdmlzaWJpbGl0eSAocHVibGljLCBwcml2YXRlLCBldGMuKSB0byB0aGUgZm14UHJvcGVydHlcclxuICAgICAgICBmbXhQcm9wZXJ0eS52aXNpYmlsaXR5ID0gXCJcIjtcclxuXHJcbiAgICAgICAgcHJvcGVydHkuZ2V0TW9kaWZpZXJzKCkuZm9yRWFjaChtID0+IHtcclxuICAgICAgICAgICAgc3dpdGNoIChtLmdldFRleHQoKSkge1xyXG4gICAgICAgICAgICAgICAgY2FzZSBTY29wZS5QdWJsaWM6XHJcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkudmlzaWJpbGl0eSA9IFwicHVibGljXCI7XHJcbiAgICAgICAgICAgICAgICAgICAgYnJlYWs7XHJcbiAgICAgICAgICAgICAgICBjYXNlIFNjb3BlLlByb3RlY3RlZDpcclxuICAgICAgICAgICAgICAgICAgICBmbXhQcm9wZXJ0eS52aXNpYmlsaXR5ID0gXCJwcm90ZWN0ZWRcIjtcclxuICAgICAgICAgICAgICAgICAgICBicmVhaztcclxuICAgICAgICAgICAgICAgIGNhc2UgU2NvcGUuUHJpdmF0ZTpcclxuICAgICAgICAgICAgICAgICAgICBmbXhQcm9wZXJ0eS52aXNpYmlsaXR5ID0gXCJwcml2YXRlXCI7XHJcbiAgICAgICAgICAgICAgICAgICAgYnJlYWs7XHJcbiAgICAgICAgICAgICAgICBjYXNlIFwic3RhdGljXCI6XHJcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkuaXNDbGFzc1NpZGUgPSB0cnVlO1xyXG4gICAgICAgICAgICAgICAgICAgIGJyZWFrO1xyXG4gICAgICAgICAgICAgICAgY2FzZSBcInJlYWRvbmx5XCI6XHJcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkucmVhZE9ubHkgPSB0cnVlO1xyXG4gICAgICAgICAgICAgICAgICAgIGJyZWFrO1xyXG4gICAgICAgICAgICAgICAgZGVmYXVsdDpcclxuICAgICAgICAgICAgICAgICAgICBicmVhaztcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgIH0pO1xyXG5cclxuICAgICAgICBpZiAoIWlzU2lnbmF0dXJlICYmIHByb3BlcnR5LmdldEV4Y2xhbWF0aW9uVG9rZW5Ob2RlKCkpIHtcclxuICAgICAgICAgICAgZm14UHJvcGVydHkuaXNEZWZpbml0ZWx5QXNzaWduZWQgPSB0cnVlO1xyXG4gICAgICAgIH1cclxuICAgICAgICBpZiAocHJvcGVydHkuZ2V0UXVlc3Rpb25Ub2tlbk5vZGUoKSkge1xyXG4gICAgICAgICAgICBmbXhQcm9wZXJ0eS5pc09wdGlvbmFsID0gdHJ1ZTtcclxuICAgICAgICB9XHJcbiAgICAgICAgaWYgKHByb3BlcnR5LmdldE5hbWUoKS5zdWJzdHJpbmcoMCwgMSkgPT09IFwiI1wiKSB7XHJcbiAgICAgICAgICAgIGZteFByb3BlcnR5LmlzSmF2YVNjcmlwdFByaXZhdGUgPSB0cnVlO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgaW5pdEZRTihwcm9wZXJ0eSwgZm14UHJvcGVydHkpO1xyXG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHByb3BlcnR5LCBmbXhQcm9wZXJ0eSk7XHJcblxyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhQcm9wZXJ0eSk7XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UHJvcGVydHkscHJvcGVydHkpO1xyXG5cclxuICAgICAgICByZXR1cm4gZm14UHJvcGVydHk7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggbWV0aG9kIG9yIGFjY2Vzc29yXHJcbiAgICAgKiBAcGFyYW0gbWV0aG9kIEEgbWV0aG9kIG9yIGFuIGFjY2Vzc29yXHJcbiAgICAgKiBAcGFyYW0gY3VycmVudENDIFRoZSBjeWNsb21hdGljIGNvbXBsZXhpdHkgbWV0cmljcyBvZiB0aGUgY3VycmVudCBzb3VyY2UgZmlsZVxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBtZXRob2Qgb3IgdGhlIGFjY2Vzc29yXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4TWV0aG9kKG1ldGhvZDogTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24sIGN1cnJlbnRDQzogeyBba2V5OiBzdHJpbmddOiBudW1iZXIgfSk6IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LkFjY2Vzc29yIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZCB7XHJcbiAgICAgICAgbGV0IGZteE1ldGhvZDogRmFtaXguTWV0aG9kIHwgRmFtaXguQWNjZXNzb3IgfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kO1xyXG4gICAgICAgIGNvbnN0IGlzR2VuZXJpYyA9IG1ldGhvZC5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XHJcbiAgICAgICAgY29uc3QgZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKG1ldGhvZCk7XHJcbiAgICAgICAgaWYgKCF0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmhhcyhmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSkpIHtcclxuXHJcbiAgICAgICAgICAgIGlmIChtZXRob2QgaW5zdGFuY2VvZiBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHx8IG1ldGhvZCBpbnN0YW5jZW9mIFNldEFjY2Vzc29yRGVjbGFyYXRpb24pIHtcclxuICAgICAgICAgICAgICAgIGZteE1ldGhvZCA9IG5ldyBGYW1peC5BY2Nlc3NvcigpO1xyXG4gICAgICAgICAgICAgICAgY29uc3QgaXNHZXR0ZXIgPSBtZXRob2QgaW5zdGFuY2VvZiBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uO1xyXG4gICAgICAgICAgICAgICAgY29uc3QgaXNTZXR0ZXIgPSBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uO1xyXG4gICAgICAgICAgICAgICAgaWYgKGlzR2V0dGVyKSB7KGZteE1ldGhvZCBhcyBGYW1peC5BY2Nlc3Nvcikua2luZCA9IFwiZ2V0dGVyXCI7fVxyXG4gICAgICAgICAgICAgICAgaWYgKGlzU2V0dGVyKSB7KGZteE1ldGhvZCBhcyBGYW1peC5BY2Nlc3Nvcikua2luZCA9IFwic2V0dGVyXCI7fVxyXG4gICAgICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteE1ldGhvZCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBpZiAoaXNHZW5lcmljKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgZm14TWV0aG9kID0gbmV3IEZhbWl4LlBhcmFtZXRyaWNNZXRob2QoKTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgICAgIGZteE1ldGhvZCA9IG5ldyBGYW1peC5NZXRob2QoKTtcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBjb25zdCBpc0NvbnN0cnVjdG9yID0gbWV0aG9kIGluc3RhbmNlb2YgQ29uc3RydWN0b3JEZWNsYXJhdGlvbjtcclxuICAgICAgICAgICAgY29uc3QgaXNTaWduYXR1cmUgPSBtZXRob2QgaW5zdGFuY2VvZiBNZXRob2RTaWduYXR1cmU7XHJcblxyXG4gICAgICAgICAgICBsZXQgaXNBYnN0cmFjdCA9IGZhbHNlO1xyXG4gICAgICAgICAgICBsZXQgaXNTdGF0aWMgPSBmYWxzZTtcclxuICAgICAgICAgICAgaWYgKG1ldGhvZCBpbnN0YW5jZW9mIE1ldGhvZERlY2xhcmF0aW9uIHx8IG1ldGhvZCBpbnN0YW5jZW9mIEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbikge1xyXG4gICAgICAgICAgICAgICAgaXNBYnN0cmFjdCA9IG1ldGhvZC5pc0Fic3RyYWN0KCk7XHJcbiAgICAgICAgICAgICAgICBpc1N0YXRpYyA9IG1ldGhvZC5pc1N0YXRpYygpO1xyXG4gICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICBpZiAoaXNDb25zdHJ1Y3RvcikgeyhmbXhNZXRob2QgYXMgRmFtaXguQWNjZXNzb3IpLmtpbmQgPSBcImNvbnN0cnVjdG9yXCI7fVxyXG4gICAgICAgICAgICBmbXhNZXRob2QuaXNBYnN0cmFjdCA9IGlzQWJzdHJhY3Q7XHJcbiAgICAgICAgICAgIGZteE1ldGhvZC5pc0NsYXNzU2lkZSA9IGlzU3RhdGljO1xyXG4gICAgICAgICAgICBmbXhNZXRob2QuaXNQcml2YXRlID0gKG1ldGhvZCBpbnN0YW5jZW9mIE1ldGhvZERlY2xhcmF0aW9uIHx8IG1ldGhvZCBpbnN0YW5jZW9mIEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbikgPyAobWV0aG9kLmdldE1vZGlmaWVycygpLmZpbmQoeCA9PiB4LmdldFRleHQoKSA9PT0gJ3ByaXZhdGUnKSkgIT09IHVuZGVmaW5lZCA6IGZhbHNlO1xyXG4gICAgICAgICAgICBmbXhNZXRob2QuaXNQcm90ZWN0ZWQgPSAobWV0aG9kIGluc3RhbmNlb2YgTWV0aG9kRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fCBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKSA/IChtZXRob2QuZ2V0TW9kaWZpZXJzKCkuZmluZCh4ID0+IHguZ2V0VGV4dCgpID09PSAncHJvdGVjdGVkJykpICE9PSB1bmRlZmluZWQgOiBmYWxzZTtcclxuICAgICAgICAgICAgZm14TWV0aG9kLnNpZ25hdHVyZSA9IEhlbHBlcnMuY29tcHV0ZVNpZ25hdHVyZShtZXRob2QuZ2V0VGV4dCgpKTtcclxuXHJcbiAgICAgICAgICAgIGxldCBtZXRob2ROYW1lOiBzdHJpbmc7XHJcbiAgICAgICAgICAgIGlmIChpc0NvbnN0cnVjdG9yKSB7XHJcbiAgICAgICAgICAgICAgICBtZXRob2ROYW1lID0gXCJjb25zdHJ1Y3RvclwiO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgbWV0aG9kTmFtZSA9IChtZXRob2QgYXMgTWV0aG9kRGVjbGFyYXRpb24gfCBNZXRob2RTaWduYXR1cmUgfCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbikuZ2V0TmFtZSgpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGZteE1ldGhvZC5uYW1lID0gbWV0aG9kTmFtZTtcclxuXHJcbiAgICAgICAgICAgIGlmICghaXNDb25zdHJ1Y3Rvcikge1xyXG4gICAgICAgICAgICAgICAgaWYgKG1ldGhvZC5nZXROYW1lKCkuc3Vic3RyaW5nKDAsIDEpID09PSBcIiNcIikge1xyXG4gICAgICAgICAgICAgICAgICAgIGZteE1ldGhvZC5pc1ByaXZhdGUgPSB0cnVlO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICBpZiAoIWZteE1ldGhvZC5pc1ByaXZhdGUgJiYgIWZteE1ldGhvZC5pc1Byb3RlY3RlZCkge1xyXG4gICAgICAgICAgICAgICAgZm14TWV0aG9kLmlzUHVibGljID0gdHJ1ZTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBlbHNlIHtcclxuICAgICAgICAgICAgICAgIGZteE1ldGhvZC5pc1B1YmxpYyA9IGZhbHNlO1xyXG4gICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICBpZiAoIWlzU2lnbmF0dXJlKSB7XHJcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QuY3ljbG9tYXRpY0NvbXBsZXhpdHkgPSBjdXJyZW50Q0NbZm14TWV0aG9kLm5hbWVdO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgZm14TWV0aG9kLmN5Y2xvbWF0aWNDb21wbGV4aXR5ID0gMDtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgbGV0IG1ldGhvZFR5cGVOYW1lID0gdGhpcy5VTktOT1dOX1ZBTFVFOyBcclxuICAgICAgICAgICAgdHJ5IHtcclxuICAgICAgICAgICAgICAgIG1ldGhvZFR5cGVOYW1lID0gbWV0aG9kLmdldFJldHVyblR5cGUoKS5nZXRUZXh0KCkudHJpbSgpOyAgICAgICAgICAgIFxyXG4gICAgICAgICAgICB9IGNhdGNoIChlcnJvcikge1xyXG4gICAgICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHJldHVybiB0eXBlIG9mIG1ldGhvZDogJHtmbXhNZXRob2QubmFtZX0uIENvbnRpbnVpbmcuLi5gKTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUobWV0aG9kVHlwZU5hbWUsIG1ldGhvZCk7XHJcbiAgICAgICAgICAgIGZteE1ldGhvZC5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xyXG4gICAgICAgICAgICBmbXhNZXRob2QubnVtYmVyT2ZMaW5lc09mQ29kZSA9IG1ldGhvZC5nZXRFbmRMaW5lTnVtYmVyKCkgLSBtZXRob2QuZ2V0U3RhcnRMaW5lTnVtYmVyKCk7XHJcbiAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPSBtZXRob2QuZ2V0UGFyYW1ldGVycygpO1xyXG4gICAgICAgICAgICBmbXhNZXRob2QubnVtYmVyT2ZQYXJhbWV0ZXJzID0gcGFyYW1ldGVycy5sZW5ndGg7XHJcblxyXG4gICAgICAgICAgICBpZiAoIWlzU2lnbmF0dXJlKSB7XHJcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QubnVtYmVyT2ZTdGF0ZW1lbnRzID0gbWV0aG9kLmdldFN0YXRlbWVudHMoKS5sZW5ndGg7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QubnVtYmVyT2ZTdGF0ZW1lbnRzID0gMDtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBcclxuICAgICAgICAgICAgaW5pdEZRTihtZXRob2QsIGZteE1ldGhvZCk7XHJcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhNZXRob2QpO1xyXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihtZXRob2QsIGZteE1ldGhvZCk7XHJcblxyXG4gICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSwgZm14TWV0aG9kKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgIGZteE1ldGhvZCA9IHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuZ2V0KGZ1bmN0aW9uRnVsbHlRdWFsaWZpZWROYW1lKSBhcyAoRmFtaXguTWV0aG9kIHwgRmFtaXguQWNjZXNzb3IgfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14TWV0aG9kLG1ldGhvZCk7XHJcbiAgICAgICAgXHJcbiAgICAgICAgcmV0dXJuIGZteE1ldGhvZDtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBmdW5jdGlvblxyXG4gICAgICogQHBhcmFtIGZ1bmMgQSBmdW5jdGlvblxyXG4gICAgICogQHBhcmFtIGN1cnJlbnRDQyBUaGUgY3ljbG9tYXRpYyBjb21wbGV4aXR5IG1ldHJpY3Mgb2YgdGhlIGN1cnJlbnQgc291cmNlIGZpbGVcclxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgZnVuY3Rpb25cclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhGdW5jdGlvbihmdW5jOiBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uLCBjdXJyZW50Q0M6IHsgW2tleTogc3RyaW5nXTogbnVtYmVyIH0pOiBGYW1peC5GdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbiB7XHJcbiAgICAgICAgbGV0IGZteEZ1bmN0aW9uOiBGYW1peC5GdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbjtcclxuICAgICAgICBjb25zdCBpc0dlbmVyaWMgPSBmdW5jLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcclxuICAgICAgICBjb25zdCBmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oZnVuYyk7XHJcbiAgICAgICAgaWYgKCF0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmhhcyhmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSkpIHtcclxuICAgICAgICAgICAgaWYgKGlzR2VuZXJpYykge1xyXG4gICAgICAgICAgICAgICAgZm14RnVuY3Rpb24gPSBuZXcgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uKCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBmbXhGdW5jdGlvbiA9IG5ldyBGYW1peC5GdW5jdGlvbigpO1xyXG4gICAgICAgICAgICB9XHJcbiAgICBcclxuICAgICAgICAgICAgY29uc3QgbmFtZSA9IGZ1bmMuZ2V0TmFtZSgpO1xyXG4gICAgICAgICAgICBpZiAobmFtZSkge1xyXG4gICAgICAgICAgICAgICAgZm14RnVuY3Rpb24ubmFtZSA9IG5hbWU7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBmbXhGdW5jdGlvbi5uYW1lID0gXCJhbm9ueW1vdXNcIjtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgZm14RnVuY3Rpb24uc2lnbmF0dXJlID0gSGVscGVycy5jb21wdXRlU2lnbmF0dXJlKGZ1bmMuZ2V0VGV4dCgpKTtcclxuICAgICAgICAgICAgZm14RnVuY3Rpb24uY3ljbG9tYXRpY0NvbXBsZXhpdHkgPSBjdXJyZW50Q0NbZm14RnVuY3Rpb24ubmFtZV07XHJcbiAgICAgICAgICAgIGluaXRGUU4oZnVuYywgZm14RnVuY3Rpb24pO1xyXG4gICAgICAgICAgICAvLyBmbXhGdW5jdGlvbi5mdWxseVF1YWxpZmllZE5hbWUgPSBmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZTtcclxuICAgIFxyXG4gICAgICAgICAgICBsZXQgZnVuY3Rpb25UeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcclxuICAgICAgICAgICAgdHJ5IHtcclxuICAgICAgICAgICAgICAgIGZ1bmN0aW9uVHlwZU5hbWUgPSBmdW5jLmdldFJldHVyblR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xyXG4gICAgICAgICAgICB9IGNhdGNoIChlcnJvcikge1xyXG4gICAgICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHJldHVybiB0eXBlIG9mIGZ1bmN0aW9uOiAke2Z1bmMuZ2V0TmFtZSgpfS4gQ29udGludWluZy4uLmApO1xyXG4gICAgICAgICAgICB9XHJcbiAgICBcclxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoZnVuY3Rpb25UeXBlTmFtZSwgZnVuYyk7XHJcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLmRlY2xhcmVkVHlwZSA9IGZteFR5cGU7XHJcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLm51bWJlck9mTGluZXNPZkNvZGUgPSBmdW5jLmdldEVuZExpbmVOdW1iZXIoKSAtIGZ1bmMuZ2V0U3RhcnRMaW5lTnVtYmVyKCk7XHJcbiAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPSBmdW5jLmdldFBhcmFtZXRlcnMoKTtcclxuICAgICAgICAgICAgZm14RnVuY3Rpb24ubnVtYmVyT2ZQYXJhbWV0ZXJzID0gcGFyYW1ldGVycy5sZW5ndGg7XHJcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLm51bWJlck9mU3RhdGVtZW50cyA9IGZ1bmMuZ2V0U3RhdGVtZW50cygpLmxlbmd0aDtcclxuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IoZnVuYywgZm14RnVuY3Rpb24pO1xyXG4gICAgXHJcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhGdW5jdGlvbik7XHJcbiAgICBcclxuICAgICAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhGdW5jdGlvbixmdW5jKTtcclxuXHJcbiAgICAgICAgICAgIHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuc2V0KGZ1bmN0aW9uRnVsbHlRdWFsaWZpZWROYW1lLCBmbXhGdW5jdGlvbik7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICBmbXhGdW5jdGlvbiA9IHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuZ2V0KGZ1bmN0aW9uRnVsbHlRdWFsaWZpZWROYW1lKSBhcyAoRmFtaXguRnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24pO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgcmV0dXJuIGZteEZ1bmN0aW9uO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IHBhcmFtZXRlclxyXG4gICAgICogQHBhcmFtIHBhcmFtIEEgcGFyYW1ldGVyXHJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHBhcmFtZXRlclxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peFBhcmFtZXRlcihwYXJhbTogUGFyYW1ldGVyRGVjbGFyYXRpb24pOiBGYW1peC5QYXJhbWV0ZXIge1xyXG4gICAgICAgIGlmICh0aGlzLmZteFBhcmFtZXRlck1hcC5oYXMocGFyYW0pKSB7XHJcbiAgICAgICAgICAgIGNvbnN0IHJQYXJhbWV0ZXIgPSB0aGlzLmZteFBhcmFtZXRlck1hcC5nZXQocGFyYW0pO1xyXG4gICAgICAgICAgICBpZiAoclBhcmFtZXRlcikgeyBcclxuICAgICAgICAgICAgICAgcmV0dXJuIHJQYXJhbWV0ZXI7XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhbWl4IHBhcmFtZXRlciAke3BhcmFtLmdldE5hbWUoKX0gaXMgbm90IGZvdW5kIGluIHRoZSBwYXJhbWV0ZXIgbWFwLmApO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBmbXhQYXJhbSA9IG5ldyBGYW1peC5QYXJhbWV0ZXIoKTtcclxuXHJcbiAgICAgICAgbGV0IHBhcmFtVHlwZU5hbWUgPSB0aGlzLlVOS05PV05fVkFMVUU7XHJcbiAgICAgICAgdHJ5IHtcclxuICAgICAgICAgICAgcGFyYW1UeXBlTmFtZSA9IHBhcmFtLmdldFR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xyXG4gICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgICAgICAgIGxvZ2dlci5lcnJvcihgPiBXQVJOSU5HOiBnb3QgZXhjZXB0aW9uICR7ZXJyb3J9LiBGYWlsZWQgdG8gZ2V0IHVzYWJsZSBuYW1lIGZvciBwYXJhbWV0ZXI6ICR7cGFyYW0uZ2V0TmFtZSgpfS4gQ29udGludWluZy4uLmApO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUocGFyYW1UeXBlTmFtZSwgcGFyYW0pO1xyXG4gICAgICAgIGZteFBhcmFtLmRlY2xhcmVkVHlwZSA9IGZteFR5cGU7XHJcbiAgICAgICAgZm14UGFyYW0ubmFtZSA9IHBhcmFtLmdldE5hbWUoKTtcclxuXHJcbiAgICAgICAgaW5pdEZRTihwYXJhbSwgZm14UGFyYW0pO1xyXG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHBhcmFtLCBmbXhQYXJhbSk7XHJcblxyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhQYXJhbSk7XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UGFyYW0sIHBhcmFtKTtcclxuICAgICAgICB0aGlzLmZteFBhcmFtZXRlck1hcC5zZXQocGFyYW0sIGZteFBhcmFtKTtcclxuXHJcbiAgICAgICAgcmV0dXJuIGZteFBhcmFtO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IHR5cGUgcGFyYW1ldGVyXHJcbiAgICAgKiBAcGFyYW0gdHAgQSB0eXBlIHBhcmFtZXRlclxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSB0eXBlIHBhcmFtZXRlclxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhQYXJhbWV0ZXJUeXBlKHRwOiBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb24pOiBGYW1peC5QYXJhbWV0ZXJUeXBlIHtcclxuICAgICAgICBcclxuICAgICAgICBjb25zdCBmbXhQYXJhbWV0ZXJUeXBlID0gbmV3IEZhbWl4LlBhcmFtZXRlclR5cGUoKTtcclxuICAgXHJcbiAgICAgICAgZm14UGFyYW1ldGVyVHlwZS5uYW1lID0gdHAuZ2V0TmFtZSgpOyAgICAgIFxyXG4gICAgICAgIGluaXRGUU4odHAsIGZteFBhcmFtZXRlclR5cGUpO1xyXG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHRwLCBmbXhQYXJhbWV0ZXJUeXBlKTtcclxuXHJcbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtZXRlclR5cGUpO1xyXG5cclxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteFBhcmFtZXRlclR5cGUsdHApO1xyXG5cclxuICAgICAgICByZXR1cm4gZm14UGFyYW1ldGVyVHlwZTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgYSBGYW1peCB0eXBlIGluIHRoZSBjb250ZXh0IG9mIGNvbmNyZXRpemF0aW9uc1xyXG4gICAgICogQHBhcmFtIHR5cGVOYW1lIEEgdHlwZSBuYW1lXHJcbiAgICAgKiBAcGFyYW0gZWxlbWVudCBBbiBlbGVtZW50XHJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHR5cGVcclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZVR5cGUoZWxlbWVudDogVHlwZU5vZGUpOiBcclxuICAgICAgICBGYW1peC5QYXJhbWV0ZXJUeXBlIHwgRmFtaXguUHJpbWl0aXZlVHlwZSB8IEZhbWl4LkNsYXNzIHwgRmFtaXguSW50ZXJmYWNlIHtcclxuICAgICAgICAvLyBUT0RPIC0gcmVmYWN0b3IgdG8gc3RvcCB1c2luZyBuYW1lcyBhcyBhIGtleSBpbiB0aGUgbWFwcywgdXNlIHRzLW1vcnBoIGVsZW1lbnQgaW5zdGVhZFxyXG4gICAgICAgIGNvbnN0IHR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbiA9IGVsZW1lbnQuZ2V0U3ltYm9sKCk/LmdldERlY2xhcmF0aW9ucygpWzBdIGFzIFR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbjtcclxuICAgICAgICBjb25zdCBwYXJhbWV0ZXJUeXBlTmFtZSA6IHN0cmluZyA9IGVsZW1lbnQuZ2V0VGV4dCgpO1xyXG4gICAgICAgIGxldCBmbXhQYXJhbWV0ZXJUeXBlOiBGYW1peC5UeXBlIHwgRmFtaXguQ2xhc3MgfCBGYW1peC5JbnRlcmZhY2UgfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ7XHJcblxyXG4gICAgICAgIC8vIGdldCBhIFR5cGVSZWZlcmVuY2UgZnJvbSBhIFR5cGVOb2RlXHJcbiAgICAgICAgY29uc3QgdHlwZVJlZmVyZW5jZSA9IGVsZW1lbnQuZ2V0VHlwZSgpO1xyXG4gICAgICAgIC8vIGdldCBhIFR5cGVEZWNsYXJhdGlvbiBmcm9tIGEgVHlwZVJlZmVyZW5jZVxyXG4gICAgICAgIGNvbnN0IHR5cGVEZWNsYXJhdGlvbiA9IHR5cGVSZWZlcmVuY2UuZ2V0U3ltYm9sKCk/LmdldERlY2xhcmF0aW9ucygpWzBdIGFzIFRTTW9ycGhUeXBlRGVjbGFyYXRpb247XHJcblxyXG4gICAgICAgIGxldCBpc0NsYXNzT3JJbnRlcmZhY2UgPSBmYWxzZTtcclxuICAgICAgICBpZiAodGhpcy5mbXhDbGFzc01hcC5oYXMocGFyYW1ldGVyVHlwZU5hbWUpKXtcclxuICAgICAgICAgICAgdGhpcy5mbXhDbGFzc01hcC5mb3JFYWNoKChvYmosIG5hbWUpID0+IHtcclxuICAgICAgICAgICAgICAgIGlmKG9iaiBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNDbGFzcyl7XHJcbiAgICAgICAgICAgICAgICAgICAgaWYgKG5hbWUgPT09IGVsZW1lbnQuZ2V0VGV4dCgpICYmIG9iai5nZW5lcmljUGFyYW1ldGVycy5zaXplPjApIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IG9iajtcclxuICAgICAgICAgICAgICAgICAgICAgICAgaXNDbGFzc09ySW50ZXJmYWNlID0gdHJ1ZTtcclxuICAgICAgICAgICAgICAgICAgICB9IFxyXG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgICAgICAgICBpZiAobmFtZSA9PT0gZWxlbWVudC5nZXRUZXh0KCkpIHtcclxuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IG9iajtcclxuICAgICAgICAgICAgICAgICAgICAgICAgaXNDbGFzc09ySW50ZXJmYWNlID0gdHJ1ZTtcclxuICAgICAgICAgICAgICAgICAgICB9IFxyXG4gICAgICAgICAgICAgICAgfSAgIFxyXG4gICAgICAgICAgICB9KTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmICh0aGlzLmZteEludGVyZmFjZU1hcC5oYXMocGFyYW1ldGVyVHlwZU5hbWUpKXtcclxuICAgICAgICAgICAgdGhpcy5mbXhJbnRlcmZhY2VNYXAuZm9yRWFjaCgob2JqLCBuYW1lKSA9PiB7XHJcbiAgICAgICAgICAgICAgICBpZihvYmogaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlKXtcclxuICAgICAgICAgICAgICAgICAgICBpZiAobmFtZSA9PT0gZWxlbWVudC5nZXRUZXh0KCkgJiYgb2JqLmdlbmVyaWNQYXJhbWV0ZXJzLnNpemU+MCkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlID0gb2JqO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBpc0NsYXNzT3JJbnRlcmZhY2UgPSB0cnVlO1xyXG4gICAgICAgICAgICAgICAgICAgIH0gXHJcbiAgICAgICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgICAgIGlmIChuYW1lID09PSBlbGVtZW50LmdldFRleHQoKSkge1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlID0gb2JqO1xyXG4gICAgICAgICAgICAgICAgICAgICAgICBpc0NsYXNzT3JJbnRlcmZhY2UgPSB0cnVlO1xyXG4gICAgICAgICAgICAgICAgICAgIH0gXHJcbiAgICAgICAgICAgICAgICB9ICAgXHJcbiAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgaWYoIWlzQ2xhc3NPckludGVyZmFjZSl7XHJcbiAgICAgICAgICAgIGlmICghdGhpcy5mbXhUeXBlTWFwLmhhcyh0eXBlRGVjbGFyYXRpb24pKSB7ICAgIFxyXG4gICAgICAgICAgICAgICAgLy8gVE9ETyByZWZhY3RvciBcclxuICAgICAgICAgICAgICAgIGlmIChpc1ByaW1pdGl2ZVR5cGUocGFyYW1ldGVyVHlwZU5hbWUpKSB7XHJcbiAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IG5ldyBGYW1peC5QcmltaXRpdmVUeXBlKCk7XHJcbiAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZS5pc1N0dWIgPSB0cnVlO1xyXG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlID0gbmV3IEZhbWl4LlBhcmFtZXRlclR5cGUoKTtcclxuICAgICAgICAgICAgICAgIH0gXHJcbiAgICBcclxuICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlclR5cGUubmFtZSA9IHBhcmFtZXRlclR5cGVOYW1lO1xyXG4gICAgICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtZXRlclR5cGUpO1xyXG4gICAgICAgICAgICAgICAgdGhpcy5mbXhUeXBlTWFwLnNldCh0eXBlRGVjbGFyYXRpb24sIGZteFBhcmFtZXRlclR5cGUpO1xyXG4gICAgICAgICAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhQYXJhbWV0ZXJUeXBlLHR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbik7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBjb25zdCByZXN1bHQgPSB0aGlzLmZteFR5cGVNYXAuZ2V0KHR5cGVEZWNsYXJhdGlvbik7XHJcbiAgICAgICAgICAgICAgICBpZiAocmVzdWx0KSB7XHJcbiAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IHJlc3VsdDtcclxuICAgICAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBGYW1peCB0eXBlICR7dHlwZURlY2xhcmF0aW9ufSBpcyBub3QgZm91bmQgaW4gdGhlIFR5cGUgbWFwLmApO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGlmICghZm14UGFyYW1ldGVyVHlwZSkge1xyXG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYGZteFBhcmFtZXRlclR5cGUgd2FzIHVuZGVmaW5lZCBmb3IgcGFyYW1ldGVyVHlwZU5hbWUgJHtwYXJhbWV0ZXJUeXBlTmFtZX1gKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgcmV0dXJuIGZteFBhcmFtZXRlclR5cGU7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggdmFyaWFibGVcclxuICAgICAqIEBwYXJhbSB2YXJpYWJsZSBBIHZhcmlhYmxlXHJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHZhcmlhYmxlXHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4VmFyaWFibGUodmFyaWFibGU6IFZhcmlhYmxlRGVjbGFyYXRpb24pOiBGYW1peC5WYXJpYWJsZSB7XHJcbiAgICAgICAgaWYgKHRoaXMuZm14VmFyaWFibGVNYXAuaGFzKHZhcmlhYmxlKSkge1xyXG4gICAgICAgICAgICBjb25zdCByVmFyaWFibGUgPSB0aGlzLmZteFZhcmlhYmxlTWFwLmdldCh2YXJpYWJsZSk7XHJcbiAgICAgICAgICAgIGlmIChyVmFyaWFibGUpIHsgXHJcbiAgICAgICAgICAgICAgIHJldHVybiByVmFyaWFibGU7XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhbWl4IHBhcmFtZXRlciAke3ZhcmlhYmxlLmdldE5hbWUoKX0gaXMgbm90IGZvdW5kIGluIHRoZSB2YXJpYWJsZSBtYXAuYCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICAgICAgY29uc3QgZm14VmFyaWFibGUgPSBuZXcgRmFtaXguVmFyaWFibGUoKTtcclxuICAgIFxyXG4gICAgICAgIGxldCB2YXJpYWJsZVR5cGVOYW1lID0gdGhpcy5VTktOT1dOX1ZBTFVFO1xyXG4gICAgICAgIHRyeSB7XHJcbiAgICAgICAgICAgIHZhcmlhYmxlVHlwZU5hbWUgPSB2YXJpYWJsZS5nZXRUeXBlKCkuZ2V0VGV4dCgpLnRyaW0oKTtcclxuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xyXG4gICAgICAgICAgICBsb2dnZXIuZXJyb3IoYD4gV0FSTklORzogZ290IGV4Y2VwdGlvbiAke2Vycm9yfS4gRmFpbGVkIHRvIGdldCB1c2FibGUgbmFtZSBmb3IgdmFyaWFibGU6ICR7dmFyaWFibGUuZ2V0TmFtZSgpfS4gQ29udGludWluZy4uLmApO1xyXG4gICAgICAgIH1cclxuICAgIFxyXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKHZhcmlhYmxlVHlwZU5hbWUsIHZhcmlhYmxlKTtcclxuICAgICAgICBmbXhWYXJpYWJsZS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xyXG4gICAgICAgIGZteFZhcmlhYmxlLm5hbWUgPSB2YXJpYWJsZS5nZXROYW1lKCk7XHJcbiAgICAgICAgaW5pdEZRTih2YXJpYWJsZSwgZm14VmFyaWFibGUpO1xyXG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHZhcmlhYmxlLCBmbXhWYXJpYWJsZSk7XHJcbiAgICBcclxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14VmFyaWFibGUpO1xyXG4gICAgXHJcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhWYXJpYWJsZSx2YXJpYWJsZSk7XHJcbiAgICAgICAgdGhpcy5mbXhWYXJpYWJsZU1hcC5zZXQodmFyaWFibGUsIGZteFZhcmlhYmxlKTtcclxuICAgIFxyXG4gICAgICAgIHJldHVybiBmbXhWYXJpYWJsZTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBlbnVtXHJcbiAgICAgKiBAcGFyYW0gZW51bUVudGl0eSBBbiBlbnVtXHJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGVudW1cclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhFbnVtKGVudW1FbnRpdHk6IEVudW1EZWNsYXJhdGlvbik6IEZhbWl4LkVudW0ge1xyXG4gICAgICAgIGlmICh0aGlzLmZteEVudW1NYXAuaGFzKGVudW1FbnRpdHkpKSB7XHJcbiAgICAgICAgICAgIGNvbnN0IHJFbnVtID0gdGhpcy5mbXhFbnVtTWFwLmdldChlbnVtRW50aXR5KTtcclxuICAgICAgICAgICAgaWYgKHJFbnVtKSB7IFxyXG4gICAgICAgICAgICAgICByZXR1cm4gckVudW07XHJcbiAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhbWl4IGVudW0gJHtlbnVtRW50aXR5LmdldE5hbWUoKX0gaXMgbm90IGZvdW5kIGluIHRoZSBlbnVtIG1hcC5gKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgIH1cclxuICAgICAgICBjb25zdCBmbXhFbnVtID0gbmV3IEZhbWl4LkVudW0oKTtcclxuICAgICAgICBmbXhFbnVtLm5hbWUgPSBlbnVtRW50aXR5LmdldE5hbWUoKTtcclxuICAgICAgICBpbml0RlFOKGVudW1FbnRpdHksIGZteEVudW0pO1xyXG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGVudW1FbnRpdHksIGZteEVudW0pO1xyXG5cclxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14RW51bSk7XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14RW51bSxlbnVtRW50aXR5KTtcclxuICAgICAgICB0aGlzLmZteEVudW1NYXAuc2V0KGVudW1FbnRpdHksIGZteEVudW0pO1xyXG5cclxuICAgICAgICByZXR1cm4gZm14RW51bTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBlbnVtIHZhbHVlXHJcbiAgICAgKiBAcGFyYW0gZW51bU1lbWJlciBBbiBlbnVtIG1lbWJlclxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBlbnVtIG1lbWJlclxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhFbnVtVmFsdWUoZW51bU1lbWJlcjogRW51bU1lbWJlcik6IEZhbWl4LkVudW1WYWx1ZSB7XHJcbiAgICAgICAgY29uc3QgZm14RW51bVZhbHVlID0gbmV3IEZhbWl4LkVudW1WYWx1ZSgpO1xyXG5cclxuICAgICAgICBsZXQgZW51bVZhbHVlVHlwZU5hbWUgPSB0aGlzLlVOS05PV05fVkFMVUU7XHJcbiAgICAgICAgdHJ5IHtcclxuICAgICAgICAgICAgZW51bVZhbHVlVHlwZU5hbWUgPSBlbnVtTWVtYmVyLmdldFR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xyXG4gICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XHJcbiAgICAgICAgICAgIGxvZ2dlci5lcnJvcihgPiBXQVJOSU5HOiBnb3QgZXhjZXB0aW9uICR7ZXJyb3J9LiBGYWlsZWQgdG8gZ2V0IHVzYWJsZSBuYW1lIGZvciBlbnVtIHZhbHVlOiAke2VudW1NZW1iZXIuZ2V0TmFtZSgpfS4gQ29udGludWluZy4uLmApO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoZW51bVZhbHVlVHlwZU5hbWUsIGVudW1NZW1iZXIpO1xyXG4gICAgICAgIGZteEVudW1WYWx1ZS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xyXG4gICAgICAgIGZteEVudW1WYWx1ZS5uYW1lID0gZW51bU1lbWJlci5nZXROYW1lKCk7XHJcbiAgICAgICAgaW5pdEZRTihlbnVtTWVtYmVyLCBmbXhFbnVtVmFsdWUpO1xyXG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGVudW1NZW1iZXIsIGZteEVudW1WYWx1ZSk7XHJcblxyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhFbnVtVmFsdWUpO1xyXG5cclxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEVudW1WYWx1ZSxlbnVtTWVtYmVyKTtcclxuXHJcbiAgICAgICAgcmV0dXJuIGZteEVudW1WYWx1ZTtcclxuICAgIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgb3IgZ2V0cyBhIEZhbWl4IGRlY29yYXRvclxyXG4gICAgICogQHBhcmFtIGRlY29yYXRvciBBIGRlY29yYXRvclxyXG4gICAgICogQHBhcmFtIGRlY29yYXRlZEVudGl0eSBBIGNsYXNzLCBhIG1ldGhvZCwgYSBwYXJhbWV0ZXIgb3IgYSBwcm9wZXJ0eVxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBkZWNvcmF0b3JcclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhEZWNvcmF0b3IoZGVjb3JhdG9yOiBEZWNvcmF0b3IsIGRlY29yYXRlZEVudGl0eTogQ2xhc3NEZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFByb3BlcnR5RGVjbGFyYXRpb24pOiBGYW1peC5EZWNvcmF0b3Ige1xyXG4gICAgICAgIGNvbnN0IGZteERlY29yYXRvciA9IG5ldyBGYW1peC5EZWNvcmF0b3IoKTtcclxuICAgICAgICBjb25zdCBkZWNvcmF0b3JOYW1lID0gXCJAXCIgKyBkZWNvcmF0b3IuZ2V0TmFtZSgpO1xyXG4gICAgICAgIGNvbnN0IGRlY29yYXRvckV4cHJlc3Npb24gPSBkZWNvcmF0b3IuZ2V0VGV4dCgpLnN1YnN0cmluZygxKTtcclxuXHJcbiAgICAgICAgZm14RGVjb3JhdG9yLm5hbWUgPSBkZWNvcmF0b3JOYW1lO1xyXG4gICAgICAgIGZteERlY29yYXRvci5kZWNvcmF0b3JFeHByZXNzaW9uID0gZGVjb3JhdG9yRXhwcmVzc2lvbjtcclxuICAgICAgICBjb25zdCBkZWNvcmF0ZWRFbnRpdHlGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGRlY29yYXRlZEVudGl0eSk7XHJcbiAgICAgICAgY29uc3QgZm14RGVjb3JhdGVkRW50aXR5ID0gdGhpcy5mYW1peFJlcC5nZXRGYW1peEVudGl0eUJ5RnVsbHlRdWFsaWZpZWROYW1lKGRlY29yYXRlZEVudGl0eUZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgRmFtaXguTmFtZWRFbnRpdHk7XHJcbiAgICAgICAgZm14RGVjb3JhdG9yLmRlY29yYXRlZEVudGl0eSA9IGZteERlY29yYXRlZEVudGl0eTtcclxuICAgICAgICBpbml0RlFOKGRlY29yYXRvciwgZm14RGVjb3JhdG9yKTtcclxuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihkZWNvcmF0b3IsIGZteERlY29yYXRvcik7XHJcblxyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhEZWNvcmF0b3IpO1xyXG5cclxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteERlY29yYXRvcixkZWNvcmF0b3IpO1xyXG5cclxuICAgICAgICByZXR1cm4gZm14RGVjb3JhdG9yO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGNvbW1lbnRcclxuICAgICAqIEBwYXJhbSBjb21tZW50IEEgY29tbWVudFxyXG4gICAgICogQHBhcmFtIGZteFNjb3BlIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgY29tbWVudCdzIGNvbnRhaW5lclxyXG4gICAgICogQHBhcmFtIGlzSlNEb2MgQSBib29sZWFuIGluZGljYXRpbmcgaWYgdGhlIGNvbW1lbnQgaXMgYSBKU0RvY1xyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjb21tZW50XHJcbiAgICAgKi9cclxuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbW1lbnQoY29tbWVudDogQ29tbWVudFJhbmdlLCBmbXhTY29wZTogRmFtaXguTmFtZWRFbnRpdHksIGlzSlNEb2M6IGJvb2xlYW4pOiBGYW1peC5Db21tZW50IHtcclxuICAgICAgICBsb2dnZXIuZGVidWcoYD4gTk9URTogY3JlYXRpbmcgY29tbWVudCAke2NvbW1lbnQuZ2V0VGV4dCgpfSBpbiBzY29wZSAke2ZteFNjb3BlLm5hbWV9LmApO1xyXG4gICAgICAgIGNvbnN0IGZteENvbW1lbnQgPSBuZXcgRmFtaXguQ29tbWVudCgpO1xyXG4gICAgICAgIGZteENvbW1lbnQuY29udGFpbmVyID0gZm14U2NvcGU7ICAvLyBhZGRzIGNvbW1lbnQgdG8gdGhlIGNvbnRhaW5lcidzIGNvbW1lbnRzIGNvbGxlY3Rpb25cclxuICAgICAgICBmbXhDb21tZW50LmlzSlNEb2MgPSBpc0pTRG9jO1xyXG5cclxuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihjb21tZW50LCBmbXhDb21tZW50KTtcclxuXHJcbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteENvbW1lbnQpO1xyXG5cclxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteENvbW1lbnQsY29tbWVudCk7XHJcblxyXG4gICAgICAgIHJldHVybiBmbXhDb21tZW50O1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBvciBnZXRzIGEgRmFtaXggdHlwZVxyXG4gICAgICogQHBhcmFtIHR5cGVOYW1lIEEgdHlwZSBuYW1lXHJcbiAgICAgKiBAcGFyYW0gZWxlbWVudCBBIHRzLW1vcnBoIGVsZW1lbnRcclxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgdHlwZVxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peFR5cGUodHlwZU5hbWU6IHN0cmluZywgZWxlbWVudDogVFNNb3JwaFR5cGVEZWNsYXJhdGlvbik6IEZhbWl4LlR5cGUge1xyXG4gICAgICAgIGxldCBmbXhUeXBlOiBGYW1peC5UeXBlO1xyXG4gICAgICAgIGNvbnN0IGlzUHJpbWl0aXZlID0gaXNQcmltaXRpdmVUeXBlKHR5cGVOYW1lKTtcclxuICAgICAgICBjb25zdCBpc1BhcmFtZXRyaWNUeXBlID1cclxuICAgICAgICAgICAgZWxlbWVudCBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24gJiYgZWxlbWVudC5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDAgfHxcclxuICAgICAgICAgICAgZWxlbWVudCBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uICYmIGVsZW1lbnQuZ2V0VHlwZVBhcmFtZXRlcnMoKS5sZW5ndGggPiAwOyBcclxuXHJcbiAgICAgICAgICAgIC8vIEZ1bmN0aW9ucyBhbmQgbWV0aG9kcyBhcmVuJ3QgdHlwZXMhXHJcbiAgICAgICAgICAgIC8vIHx8XHJcbiAgICAgICAgICAgIC8vIGVsZW1lbnQgaW5zdGFuY2VvZiBGdW5jdGlvbkRlY2xhcmF0aW9uICYmIGVsZW1lbnQuZ2V0VHlwZVBhcmFtZXRlcnMoKS5sZW5ndGggPiAwIHx8XHJcbiAgICAgICAgICAgIC8vIGVsZW1lbnQgaW5zdGFuY2VvZiBNZXRob2REZWNsYXJhdGlvbiAmJiBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMCB8fFxyXG4gICAgICAgICAgICAvLyBlbGVtZW50IGluc3RhbmNlb2YgQXJyb3dGdW5jdGlvbiAmJiBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcclxuXHJcbiAgICAgICAgbG9nZ2VyLmRlYnVnKFwiQ3JlYXRpbmcgKG9yIGdldHRpbmcpIHR5cGU6ICdcIiArIHR5cGVOYW1lICsgXCInIG9mIGVsZW1lbnQ6IFwiICsgZWxlbWVudD8uZ2V0VGV4dCgpICsgXCIgb2Yga2luZDogXCIgKyBlbGVtZW50Py5nZXRLaW5kTmFtZSgpKTtcclxuXHJcbiAgICAgICAgaWYgKGlzUHJpbWl0aXZlKSB7XHJcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhQcmltaXRpdmVUeXBlKHR5cGVOYW1lKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmIChpc1BhcmFtZXRyaWNUeXBlKSB7XHJcbiAgICAgICAgICAgIC8vIG5hcnJvdyB0aGUgdHlwZVxyXG4gICAgICAgICAgICBjb25zdCBwYXJhbWV0cmljRWxlbWVudCA9IGVsZW1lbnQgYXMgVFNNb3JwaFBhcmFtZXRyaWNUeXBlO1xyXG4gICAgICAgICAgICByZXR1cm4gdGhpcy5jcmVhdGVPckdldEZhbWl4UGFyYW1ldHJpY1R5cGUodHlwZU5hbWUsIHBhcmFtZXRyaWNFbGVtZW50KTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmICghdGhpcy5mbXhUeXBlTWFwLmhhcyhlbGVtZW50KSkge1xyXG4gICAgICAgICAgICBsZXQgYW5jZXN0b3I6IEZhbWl4LkNvbnRhaW5lckVudGl0eSB8IHVuZGVmaW5lZCA9IHVuZGVmaW5lZDtcclxuIFxyXG4gICAgICAgICAgICBpZiAoZWxlbWVudCAhPT0gdW5kZWZpbmVkKSB7XHJcbiAgICAgICAgICAgICAgICBjb25zdCB0eXBlQW5jZXN0b3IgPSBIZWxwZXJzLmZpbmRUeXBlQW5jZXN0b3IoZWxlbWVudCk7XHJcbiAgICAgICAgICAgICAgICBpZiAoIXR5cGVBbmNlc3Rvcikge1xyXG4gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgQW5jZXN0b3Igbm90IGZvdW5kIGZvciBlbGVtZW50ICR7ZWxlbWVudC5nZXRUZXh0KCl9LmApO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgY29uc3QgYW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKHR5cGVBbmNlc3Rvcik7XHJcbiAgICAgICAgICAgICAgICBhbmNlc3RvciA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUZ1bGx5UXVhbGlmaWVkTmFtZShhbmNlc3RvckZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgRmFtaXguQ29udGFpbmVyRW50aXR5O1xyXG4gICAgICAgICAgICAgICAgaWYgKCFhbmNlc3Rvcikge1xyXG4gICAgICAgICAgICAgICAgICAgIGxvZ2dlci5kZWJ1ZyhgQW5jZXN0b3IgJHtGUU5GdW5jdGlvbnMuZ2V0RlFOKHR5cGVBbmNlc3Rvcil9IG5vdCBmb3VuZC4gQWRkaW5nIHRoZSBuZXcgdHlwZS5gKTtcclxuICAgICAgICAgICAgICAgICAgICBhbmNlc3RvciA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUodHlwZUFuY2VzdG9yLmdldFRleHQoKSwgdHlwZUFuY2VzdG9yIGFzIFRTTW9ycGhUeXBlRGVjbGFyYXRpb24pO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICBmbXhUeXBlID0gbmV3IEZhbWl4LlR5cGUoKTtcclxuICAgICAgICAgICAgZm14VHlwZS5uYW1lID0gdHlwZU5hbWU7XHJcblxyXG4gICAgICAgICAgICBpZiAoIWFuY2VzdG9yKSB7XHJcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEFuY2VzdG9yIG5vdCBmb3VuZCBmb3IgdHlwZSAke3R5cGVOYW1lfS5gKTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICBmbXhUeXBlLmNvbnRhaW5lciA9IGFuY2VzdG9yO1xyXG4gICAgICAgICAgICBpbml0RlFOKGVsZW1lbnQsIGZteFR5cGUpO1xyXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihlbGVtZW50LCBmbXhUeXBlKTtcclxuXHJcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhUeXBlKTtcclxuXHJcbiAgICAgICAgICAgIHRoaXMuZm14VHlwZU1hcC5zZXQoZWxlbWVudCwgZm14VHlwZSk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICBjb25zdCByZXN1bHQgPSB0aGlzLmZteFR5cGVNYXAuZ2V0KGVsZW1lbnQpO1xyXG4gICAgICAgICAgICBpZiAocmVzdWx0KSB7XHJcbiAgICAgICAgICAgICAgICBmbXhUeXBlID0gcmVzdWx0O1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBGYW1peCB0eXBlICR7dHlwZU5hbWV9IGlzIG5vdCBmb3VuZCBpbiB0aGUgVHlwZSBtYXAuYCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14VHlwZSxlbGVtZW50KTtcclxuXHJcbiAgICAgICAgcmV0dXJuIGZteFR5cGU7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCB0eXBlIHRoYXQgaXMgcGFyYW1ldHJpY1xyXG4gICAgICogQHBhcmFtIHR5cGVOYW1lIEEgdHlwZSBuYW1lXHJcbiAgICAgKiBAcGFyYW0gZWxlbWVudCBBIHRzLW1vcnBoIGVsZW1lbnRcclxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgcGFyYW1ldGVyIHR5cGVcclxuICAgICAqL1xyXG4gICAgY3JlYXRlT3JHZXRGYW1peFBhcmFtZXRyaWNUeXBlKHR5cGVOYW1lOiBzdHJpbmcsIGVsZW1lbnQ6IFRTTW9ycGhQYXJhbWV0cmljVHlwZSk6IEZhbWl4LlR5cGUge1xyXG5cclxuICAgICAgICBpZiAodGhpcy5mbXhUeXBlTWFwLmhhcyhlbGVtZW50KSA9PT0gdHJ1ZSkge1xyXG4gICAgICAgICAgICBjb25zdCByZXN1bHQgPSB0aGlzLmZteFR5cGVNYXAuZ2V0KGVsZW1lbnQpO1xyXG4gICAgICAgICAgICBpZiAocmVzdWx0KSB7XHJcbiAgICAgICAgICAgICAgICByZXR1cm4gcmVzdWx0O1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBGYW1peCB0eXBlICR7dHlwZU5hbWV9IGlzIG5vdCBmb3VuZCAodW5kZWZpbmVkKSBpbiB0aGUgVHlwZSBtYXAuYCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIEEgcGFyYW1ldHJpYyB0eXBlIGlzIGEgdHlwZSB0aGF0IGhhcyB0eXBlIHBhcmFtZXRlcnMsIGUuZy4sIExpc3Q8VD5cclxuICAgICAgICAvLyBJbiBUUyBpdCBjYW4gYmUgYSBjbGFzcywgYW4gaW50ZXJmYWNlLCBhIGZ1bmN0aW9uLCBhbiBhcnJvdyBmdW5jdGlvbiwgb3IgYSBtZXRob2RcclxuXHJcbiAgICAgICAgLy8gY3JlYXRlIHRoZSBGYW1peCBQYXJhbWV0cmljIFR5cGUgKG1heWJlIGl0J3MganVzdCBhbiBJbnRlcmZhY2UsIGV0Yy4pXHJcbiAgICAgICAgbGV0IGZteFR5cGU6IEZhbWl4LlR5cGU7XHJcblxyXG4gICAgICAgIGlmIChlbGVtZW50IGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbikge1xyXG4gICAgICAgICAgICBmbXhUeXBlID0gbmV3IEZhbWl4LlBhcmFtZXRyaWNDbGFzcygpO1xyXG4gICAgICAgIH0gZWxzZSBpZiAoZWxlbWVudCBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uKSB7XHJcbiAgICAgICAgICAgIGZteFR5cGUgPSBuZXcgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSgpO1xyXG4gICAgICAgIH1cclxuICAgICAgICAvLyBmdW5jdGlvbnMgYW5kIG1ldGhvZHMgYXJlIG5vdCB0eXBlcyBcclxuICAgICAgICAvLyBlbHNlIGlmIChlbGVtZW50IGluc3RhbmNlb2YgRnVuY3Rpb25EZWNsYXJhdGlvbikge1xyXG4gICAgICAgIC8vICAgICBmbXhUeXBlID0gbmV3IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbigpO1xyXG4gICAgICAgIC8vIH0gZWxzZSBpZiAoZWxlbWVudCBpbnN0YW5jZW9mIEFycm93RnVuY3Rpb24pIHtcclxuICAgICAgICAvLyAgICAgZm14VHlwZSA9IG5ldyBGYW1peC5QYXJhbWV0cmljQXJyb3dGdW5jdGlvbigpO1xyXG4gICAgICAgIC8vIH0gZWxzZSBpZiAoZWxlbWVudCBpbnN0YW5jZW9mIE1ldGhvZERlY2xhcmF0aW9uKSB7XHJcbiAgICAgICAgLy8gICAgIGZteFR5cGUgPSBuZXcgRmFtaXguUGFyYW1ldHJpY01ldGhvZCgpO1xyXG4gICAgICAgIC8vIH0gXHJcbiAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgRWxlbWVudCBpcyBub3QgYSBjbGFzcywgaW50ZXJmYWNlLCBmdW5jdGlvbiwgYXJyb3cgZnVuY3Rpb24sIG9yIG1ldGhvZC5gKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIGNvbnN0IHBhcmFtZXRlcnMgPSBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCk7XHJcblxyXG4gICAgICAgIC8vIC8vIGZvciBlYWNoIHBhcmFtZXRlciwgZ2V0T3JDcmVhdGUgdGhlIEZhbWl4UGFyYW1ldGVyVHlwZVxyXG4gICAgICAgIC8vIGZvciAoY29uc3QgcGFyYW1ldGVyIG9mIHBhcmFtZXRlcnMpIHtcclxuICAgICAgICAvLyAgICAgdGhpcy5jcmVhdGVPckdldEZhbWl4UGFyYW1ldGVyVHlwZShwYXJhbWV0ZXIuZ2V0TmFtZSgpLCBwYXJhbWV0ZXIpO1xyXG4gICAgICAgIC8vIH1cclxuXHJcbiAgICAgICAgLy8gLy8gVE9ETzogdGhlIGZvbGxvd2luZyBjb2RlIGlzIG5vdCBjb3JyZWN0LCBpdCBpcyBqdXN0IGEgcGxhY2Vob2xkZXJcclxuICAgICAgICAvLyBjb25zdCBwYXJhbWV0ZXJUeXBlTmFtZXMgPSB0eXBlTmFtZS5zdWJzdHJpbmcodHlwZU5hbWUuaW5kZXhPZihcIjxcIikgKyAxLCB0eXBlTmFtZS5pbmRleE9mKFwiPlwiKSlcclxuICAgICAgICAvLyAgICAgLnNwbGl0KFwiLFwiKS5tYXAocyA9PiBzLnRyaW0oKSk7XHJcbiAgICAgICAgLy8gY29uc3QgYmFzZVR5cGVOYW1lID0gdHlwZU5hbWUuc3Vic3RyaW5nKDAsIHR5cGVOYW1lLmluZGV4T2YoXCI8XCIpKS50cmltKCk7XHJcbiAgICAgICAgLy8gcGFyYW1ldGVyVHlwZU5hbWVzLmZvckVhY2gocGFyYW1ldGVyVHlwZU5hbWUgPT4ge1xyXG4gICAgICAgIC8vICAgICBjb25zdCBmbXhQYXJhbWV0ZXJUeXBlID0gdGhpcy5jcmVhdGVPckdldEZhbWl4UGFyYW1ldGVyVHlwZShwYXJhbWV0ZXJUeXBlTmFtZSwgZWxlbWVudCk7XHJcbiAgICAgICAgLy8gICAgIChmbXhUeXBlIGFzIEZhbWl4LlBhcmFtZXRlclR5cGUpLmFkZEFyZ3VtZW50KGZteFBhcmFtZXRlclR5cGUpO1xyXG4gICAgICAgIC8vIH0pO1xyXG4gICAgICAgIC8vIGNvbnN0IGZteEJhc2VUeXBlID0gdGhpcy5jcmVhdGVPckdldEZhbWl4VHlwZShiYXNlVHlwZU5hbWUsIGVsZW1lbnQpO1xyXG5cclxuICAgICAgICAvLyAoZm14VHlwZSBhcyBGYW1peC5QYXJhbWV0ZXJUeXBlKS5iYXNlVHlwZSA9IGZteEJhc2VUeXBlO1xyXG5cclxuICAgICAgICBmbXhUeXBlLm5hbWUgPSB0eXBlTmFtZTtcclxuICAgICAgICBpbml0RlFOKGVsZW1lbnQsIGZteFR5cGUpO1xyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhUeXBlKTtcclxuICAgICAgICB0aGlzLmZteFR5cGVNYXAuc2V0KGVsZW1lbnQsIGZteFR5cGUpO1xyXG4gICAgICAgIHJldHVybiBmbXhUeXBlO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIHR5cGUgZm9yIGEgcGFyYW1ldGVyIGluIGEgcGFyYW1ldHJpYyB0eXBlLCBlLmcuLCBUIGluIExpc3Q8VD5cclxuICAgICAqIEBwYXJhbSBwYXJhbWV0ZXJUeXBlTmFtZSBcclxuICAgICAqIEBwYXJhbSBlbGVtZW50IHRoZSBUeXBlU2NyaXB0IGVsZW1lbnQgKFRTTW9ycGhQYXJhbWV0cmljVHlwZSkgdGhhdCB0aGUgdHlwZSBpcyBhc3NvY2lhdGVkIHdpdGhcclxuICAgICAqIEByZXR1cm5zIFxyXG4gICAgICovXHJcbiAgICAvLyBjcmVhdGVPckdldEZhbWl4UGFyYW1ldGVyVHlwZShwYXJhbWV0ZXJUeXBlTmFtZTogc3RyaW5nLCBlbGVtZW50OiBQYXJhbWV0ZXJEZWNsYXJhdGlvbikge1xyXG4gICAgLy8gICAgIGlmICh0aGlzLmZteFR5cGVNYXAuaGFzKGVsZW1lbnQpKSB7XHJcbiAgICAvLyAgICAgICAgIHJldHVybiB0aGlzLmZteFR5cGVNYXAuZ2V0KGVsZW1lbnQpIGFzIEZhbWl4LlBhcmFtZXRlclR5cGU7XHJcbiAgICAvLyAgICAgfVxyXG5cclxuICAgIC8vICAgICAvLyBkZXRlcm1pbmUgaWYgZWxlbWVudCBpcyBhIFxyXG4gICAgLy8gICAgIGNvbnN0IGZteFR5cGUgPSBuZXcgRmFtaXguUGFyYW1ldGVyVHlwZSgpO1xyXG4gICAgLy8gICAgIC8vIGNvbnN0IHBhcmFtZXRlclR5cGVOYW1lcyA9IHR5cGVOYW1lLnN1YnN0cmluZyh0eXBlTmFtZS5pbmRleE9mKFwiPFwiKSArIDEsIHR5cGVOYW1lLmluZGV4T2YoXCI+XCIpKVxyXG4gICAgLy8gICAgIC8vICAgICAuc3BsaXQoXCIsXCIpLm1hcChzID0+IHMudHJpbSgpKTtcclxuICAgIC8vICAgICAvLyBjb25zdCBiYXNlVHlwZU5hbWUgPSB0eXBlTmFtZS5zdWJzdHJpbmcoMCwgdHlwZU5hbWUuaW5kZXhPZihcIjxcIikpLnRyaW0oKTtcclxuICAgIC8vICAgICAvLyBwYXJhbWV0ZXJUeXBlTmFtZXMuZm9yRWFjaChwYXJhbWV0ZXJUeXBlTmFtZSA9PiB7XHJcbiAgICAvLyAgICAgLy8gICAgIGNvbnN0IGZteFBhcmFtZXRlclR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhQYXJhbWV0ZXJUeXBlKHBhcmFtZXRlclR5cGVOYW1lLCBlbGVtZW50KTtcclxuICAgIC8vICAgICAvLyAgICAgKGZteFR5cGUgYXMgRmFtaXguUGFyYW1ldGVyVHlwZSkuYWRkQXJndW1lbnQoZm14UGFyYW1ldGVyVHlwZSk7XHJcbiAgICAvLyAgICAgLy8gfSk7XHJcbiAgICAvLyAgICAgY29uc3QgZm14QmFzZVR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKGJhc2VUeXBlTmFtZSwgZWxlbWVudCk7XHJcbiAgICAvLyAgICAgKGZteFR5cGUgYXMgRmFtaXguUGFyYW1ldGVyVHlwZSkuYmFzZVR5cGUgPSBmbXhCYXNlVHlwZTtcclxuICAgIC8vICAgICBpbml0RlFOKGVsZW1lbnQsIGZteFR5cGUpO1xyXG4gICAgLy8gICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhUeXBlKTtcclxuICAgIC8vICAgICB0aGlzLmZteFR5cGVNYXAuc2V0KGVsZW1lbnQsIGZteFR5cGUpO1xyXG4gICAgLy8gICAgIHJldHVybiBmbXhUeXBlO1xyXG4gICAgLy8gfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBvciBnZXRzIGEgRmFtaXggcHJpbWl0aXZlIHR5cGVcclxuICAgICAqIEBwYXJhbSB0eXBlTmFtZSBBIHR5cGUgbmFtZVxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBwcmltaXRpdmUgdHlwZVxyXG4gICAgICovXHJcbiAgICBjcmVhdGVPckdldEZhbWl4UHJpbWl0aXZlVHlwZSh0eXBlTmFtZTogc3RyaW5nKTogRmFtaXguUHJpbWl0aXZlVHlwZSB7XHJcbiAgICAgICAgbGV0IGZteFR5cGU6IEZhbWl4LlByaW1pdGl2ZVR5cGUgPSBuZXcgRmFtaXguUHJpbWl0aXZlVHlwZSgpO1xyXG4gICAgICAgIGlmICghdGhpcy5mbXhQcmltaXRpdmVUeXBlTWFwLmhhcyh0eXBlTmFtZSkpIHtcclxuICAgICAgICAgICAgZm14VHlwZSA9IG5ldyBGYW1peC5QcmltaXRpdmVUeXBlKCk7XHJcbiAgICAgICAgICAgIGZteFR5cGUuaXNTdHViID0gdHJ1ZTtcclxuICAgICAgICAgICAgZm14VHlwZS5uYW1lID0gdHlwZU5hbWU7XHJcbiAgICAgICAgICAgIHRoaXMuZm14UHJpbWl0aXZlVHlwZU1hcC5zZXQodHlwZU5hbWUsIGZteFR5cGUpO1xyXG4gICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14VHlwZSk7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgZm14VHlwZSA9IHRoaXMuZm14UHJpbWl0aXZlVHlwZU1hcC5nZXQodHlwZU5hbWUpIGFzIEZhbWl4LlByaW1pdGl2ZVR5cGU7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIHJldHVybiBmbXhUeXBlO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGFjY2Vzc1xyXG4gICAgICogQHBhcmFtIG5vZGUgQSBub2RlXHJcbiAgICAgKiBAcGFyYW0gaWQgQW4gaWQgb2YgYSBwYXJhbWV0ZXIsIGEgdmFyaWFibGUsIGEgcHJvcGVydHkgb3IgYW4gZW51bSBtZW1iZXJcclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4QWNjZXNzKG5vZGU6IElkZW50aWZpZXIsIGlkOiBudW1iZXIpOiB2b2lkIHtcclxuICAgICAgICBjb25zdCBmbXhWYXIgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlJZChpZCkgYXMgRmFtaXguU3RydWN0dXJhbEVudGl0eTtcclxuICAgICAgICBpZiAoIWZteFZhcikge1xyXG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhbWl4IGVudGl0eSB3aXRoIGlkICR7aWR9IG5vdCBmb3VuZCwgZm9yIG5vZGUgJHtub2RlLmdldFRleHQoKX0gaW4gJHtub2RlLmdldFNvdXJjZUZpbGUoKS5nZXRCYXNlTmFtZSgpfSBhdCBsaW5lICR7bm9kZS5nZXRTdGFydExpbmVOdW1iZXIoKX0uYCk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBsb2dnZXIuZGVidWcoYENyZWF0aW5nIEZhbWl4QWNjZXNzLiBOb2RlOiBbJHtub2RlLmdldEtpbmROYW1lKCl9XSAnJHtub2RlLmdldFRleHQoKX0nIGF0IGxpbmUgJHtub2RlLmdldFN0YXJ0TGluZU51bWJlcigpfSBpbiAke25vZGUuZ2V0U291cmNlRmlsZSgpLmdldEJhc2VOYW1lKCl9LCBpZDogJHtpZH0gcmVmZXJzIHRvIGZteFZhciAnJHtmbXhWYXIuZnVsbHlRdWFsaWZpZWROYW1lfScuYCk7XHJcblxyXG4gICAgICAgIGNvbnN0IG5vZGVSZWZlcmVuY2VBbmNlc3RvciA9IEhlbHBlcnMuZmluZEFuY2VzdG9yKG5vZGUpO1xyXG4gICAgICAgIGNvbnN0IGFuY2VzdG9yRnVsbHlRdWFsaWZpZWROYW1lID0gRlFORnVuY3Rpb25zLmdldEZRTihub2RlUmVmZXJlbmNlQW5jZXN0b3IpO1xyXG4gICAgICAgIGNvbnN0IGFjY2Vzc29yID0gdGhpcy5mYW1peFJlcC5nZXRGYW1peEVudGl0eUJ5RnVsbHlRdWFsaWZpZWROYW1lKGFuY2VzdG9yRnVsbHlRdWFsaWZpZWROYW1lKSBhcyBGYW1peC5Db250YWluZXJFbnRpdHk7XHJcbiAgICAgICAgaWYgKCFhY2Nlc3Nvcikge1xyXG4gICAgICAgICAgICBsb2dnZXIuZXJyb3IoYEFuY2VzdG9yICR7YW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWV9IG9mIGtpbmQgJHtub2RlUmVmZXJlbmNlQW5jZXN0b3IuZ2V0S2luZE5hbWUoKX0gbm90IGZvdW5kLmApO1xyXG4gICAgICAgICAgICAvLyBhY2Nlc3NvciA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoYW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWUsIG5vZGVSZWZlcmVuY2VBbmNlc3RvciBhcyBUeXBlRGVjbGFyYXRpb24pO1xyXG4gICAgICAgICAgICByZXR1cm47IC8vIGJhaWwgb3V0IFRPRE86IHRoaXMgaXMgcHJvYmFibHkgd3JvbmdcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoYEZvdW5kIGFjY2Vzc29yIHRvIGJlICR7YWNjZXNzb3IuZnVsbHlRdWFsaWZpZWROYW1lfS5gKTtcclxuICAgICAgICB9XHJcblxyXG5cclxuICAgICAgICAvLyBtYWtlIHN1cmUgYWNjZXNzb3IgaXMgYSBtZXRob2QsIGZ1bmN0aW9uLCBzY3JpcHQgb3IgbW9kdWxlXHJcbiAgICAgICAgaWYgKCEoYWNjZXNzb3IgaW5zdGFuY2VvZiBGYW1peC5NZXRob2QpICYmICEoYWNjZXNzb3IgaW5zdGFuY2VvZiBGYW1peC5BcnJvd0Z1bmN0aW9uKSAmJiAhKGFjY2Vzc29yIGluc3RhbmNlb2YgRmFtaXguRnVuY3Rpb24pICYmICEoYWNjZXNzb3IgaW5zdGFuY2VvZiBGYW1peC5TY3JpcHRFbnRpdHkpICYmICEoYWNjZXNzb3IgaW5zdGFuY2VvZiBGYW1peC5Nb2R1bGUpKSB7XHJcbiAgICAgICAgICAgIGxvZ2dlci5lcnJvcihgQWNjZXNzb3IgJHthY2Nlc3Nvci5mdWxseVF1YWxpZmllZE5hbWV9IGlzIG5vdCBhIG1ldGhvZCwgZnVuY3Rpb24sIGV0Yy5gKTtcclxuICAgICAgICAgICAgcmV0dXJuO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgLy8gZG9uJ3QgY3JlYXRlIGFueSBkdXBsaWNhdGVzIChlLmcuIGlmIHRoZSBzYW1lIHZhcmlhYmxlIGlzIGFjY2Vzc2VkIG11bHRpcGxlIHRpbWVzIGJ5IHNhbWUgYWNjZXNzb3IpXHJcbiAgICAgICAgY29uc3QgZm91bmRBY2Nlc3MgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4QWNjZXNzQnlBY2Nlc3NvckFuZFZhcmlhYmxlKGFjY2Vzc29yLCBmbXhWYXIpO1xyXG4gICAgICAgIGlmIChmb3VuZEFjY2Vzcykge1xyXG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoYEZhbWl4QWNjZXNzIGFscmVhZHkgZXhpc3RzIGZvciBhY2Nlc3NvciAke2FjY2Vzc29yLmZ1bGx5UXVhbGlmaWVkTmFtZX0gYW5kIHZhcmlhYmxlICR7Zm14VmFyLmZ1bGx5UXVhbGlmaWVkTmFtZX0uYCk7XHJcbiAgICAgICAgICAgIHJldHVybjtcclxuICAgICAgICB9XHJcbiAgICAgICAgY29uc3QgZm14QWNjZXNzID0gbmV3IEZhbWl4LkFjY2VzcygpO1xyXG4gICAgICAgIGZteEFjY2Vzcy5hY2Nlc3NvciA9IGFjY2Vzc29yO1xyXG4gICAgICAgIGZteEFjY2Vzcy52YXJpYWJsZSA9IGZteFZhcjtcclxuXHJcbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEFjY2Vzcyk7XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14QWNjZXNzLG5vZGUpO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGludm9jYXRpb25cclxuICAgICAqIEBwYXJhbSBub2RlUmVmZXJyaW5nVG9JbnZvY2FibGUgQSBub2RlXHJcbiAgICAgKiBAcGFyYW0gaW52b2NhYmxlIEEgbWV0aG9kIG9yIGEgZnVuY3Rpb25cclxuICAgICAqIEBwYXJhbSBpZCBUaGUgaWQgb2YgdGhlIG1ldGhvZCBvciB0aGUgZnVuY3Rpb25cclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4SW52b2NhdGlvbihub2RlUmVmZXJyaW5nVG9JbnZvY2FibGU6IElkZW50aWZpZXIsIGludm9jYWJsZTogSW52b2NhYmxlVHlwZSwgaWQ6IG51bWJlcik6IHZvaWQge1xyXG4gICAgICAgIGNvbnN0IGZteEludm9jYWJsZSA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUlkKGlkKSBhcyBGYW1peC5CZWhhdmlvcmFsRW50aXR5O1xyXG4gICAgICAgIC8vIHNpbmNlIHRoZSBub2RlIGlzIGluIHRoZSBBU1QsIHdlIG5lZWQgdG8gZmluZCB0aGUgYW5jZXN0b3IgdGhhdCBpcyBpbiB0aGUgRmFtaXggbW9kZWxcclxuICAgICAgICBjb25zdCBjb250YWluZXJPZk5vZGUgPSBIZWxwZXJzLmZpbmRBbmNlc3Rvcihub2RlUmVmZXJyaW5nVG9JbnZvY2FibGUpO1xyXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgRm91bmQgY29udGFpbmVyIChhbmNlc3RvcikgJHtjb250YWluZXJPZk5vZGUuZ2V0S2luZE5hbWUoKX0gZm9yIEFTVCBub2RlICR7bm9kZVJlZmVycmluZ1RvSW52b2NhYmxlLmdldFRleHQoKX0uYCk7XHJcbiAgICAgICAgY29uc3QgY29udGFpbmVyRlFOID0gRlFORnVuY3Rpb25zLmdldEZRTihjb250YWluZXJPZk5vZGUpO1xyXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgRm91bmQgY29udGFpbmVyRlFOICR7Y29udGFpbmVyRlFOfS5gKTtcclxuICAgICAgICBsZXQgc2VuZGVyID0gdGhpcy5mYW1peFJlcC5nZXRGYW1peEVudGl0eUJ5RnVsbHlRdWFsaWZpZWROYW1lKGNvbnRhaW5lckZRTikgYXMgRmFtaXguQ29udGFpbmVyRW50aXR5O1xyXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgRm91bmQgYSBzZW5kZXIgdGhhdCBtYXRjaGVzICR7c2VuZGVyLmZ1bGx5UXVhbGlmaWVkTmFtZX0uYCk7XHJcbiAgICAgICAgaWYgKHNlbmRlciBpbnN0YW5jZW9mIEZhbWl4LlR5cGUpIHtcclxuICAgICAgICAgICAgLy8gVE9ETyB0aGlzIG1pZ2h0IGJlIGFuIGVycm9yIGluIGdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWVcclxuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBPb3BzISBTZW5kZXIgaXMgYSB0eXBlLCB3aGljaCBpcyBub3QgdmFsaWQgZm9yIGFuIEludm9jYXRpb24uIFRyeWluZyB0byBmaW5kIGEgY29udGFpbmVyIGZvciAke3NlbmRlci5mdWxseVF1YWxpZmllZE5hbWV9LmApO1xyXG4gICAgICAgICAgICBjb25zdCBzZW5kZXJDb250YWluZXIgPSBzZW5kZXIuY29udGFpbmVyO1xyXG4gICAgICAgICAgICBpZiAoc2VuZGVyQ29udGFpbmVyKSB7XHJcbiAgICAgICAgICAgICAgICBzZW5kZXIgPSBzZW5kZXJDb250YWluZXI7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICAgICAgY29uc3QgcmVjZWl2ZXJGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGludm9jYWJsZS5nZXRQYXJlbnQoKSk7XHJcbiAgICAgICAgY29uc3QgcmVjZWl2ZXIgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUocmVjZWl2ZXJGdWxseVF1YWxpZmllZE5hbWUpIGFzIEZhbWl4Lk5hbWVkRW50aXR5O1xyXG5cclxuICAgICAgICBjb25zdCBmbXhJbnZvY2F0aW9uID0gbmV3IEZhbWl4Lkludm9jYXRpb24oKTtcclxuICAgICAgICBmbXhJbnZvY2F0aW9uLnNlbmRlciA9IHNlbmRlcjtcclxuICAgICAgICBmbXhJbnZvY2F0aW9uLnJlY2VpdmVyID0gcmVjZWl2ZXI7XHJcbiAgICAgICAgZm14SW52b2NhdGlvbi5hZGRDYW5kaWRhdGUoZm14SW52b2NhYmxlKTtcclxuICAgICAgICBmbXhJbnZvY2F0aW9uLnNpZ25hdHVyZSA9IGZteEludm9jYWJsZS5zaWduYXR1cmU7XHJcblxyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhJbnZvY2F0aW9uKTtcclxuXHJcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhJbnZvY2F0aW9uLG5vZGVSZWZlcnJpbmdUb0ludm9jYWJsZSk7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW5oZXJpdGFuY2VcclxuICAgICAqIEBwYXJhbSBjbHMgQSBjbGFzcyBvciBhbiBpbnRlcmZhY2UgKHN1YmNsYXNzKVxyXG4gICAgICogQHBhcmFtIGluaENsYXNzIFRoZSBpbmhlcml0ZWQgY2xhc3Mgb3IgaW50ZXJmYWNlIChzdXBlcmNsYXNzKVxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peEluaGVyaXRhbmNlKGNsczogQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uLCBpbmhDbGFzczogQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uIHwgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzKTogdm9pZCB7XHJcbiAgICAgICAgLy8gLy8gbmVlZCBhIGtleSB0byBzZWUgaWYgdGhlIGluaGVyaXRhbmNlIGFscmVhZHkgZXhpc3RzXHJcbiAgICAgICAgLy8gY29uc3QgY2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGNscyk7XHJcbiAgICAgICAgLy8gbGV0IGluS2V5d29yZDogc3RyaW5nO1xyXG4gICAgICAgIC8vIGxldCBpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZTogc3RyaW5nO1xyXG4gICAgICAgIC8vIGxldCBpbmhDbGFzc05hbWU6IHN0cmluZyB8IHVuZGVmaW5lZDtcclxuICAgICAgICAvLyAvLyBpZiBpbmhDbGFzcyBpcyBhbiBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHMsIGl0IGlzIGFuIGludGVyZmFjZVxyXG4gICAgICAgIC8vIGlmIChpbmhDbGFzcyBpbnN0YW5jZW9mIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cykge1xyXG4gICAgICAgIC8vICAgICBpbmhDbGFzc05hbWUgPSBpbmhDbGFzcy5nZXRFeHByZXNzaW9uKCkuZ2V0VGV4dCgpO1xyXG4gICAgICAgIC8vICAgICAvLyB3aGF0IGlzIGluaENsYXNzRnVsbHlRdWFsaWZpZWROYW1lPyBUT0RPXHJcbiAgICAgICAgLy8gICAgIGluaENsYXNzRnVsbHlRdWFsaWZpZWROYW1lID0gJ1VuZGVmaW5lZF9TY29wZV9mcm9tX2ltcG9ydGVyLicgKyBpbmhDbGFzc05hbWU7XHJcbiAgICAgICAgLy8gfSBlbHNlIHtcclxuICAgICAgICAvLyAgICAgaW5oQ2xhc3NOYW1lID0gaW5oQ2xhc3MuZ2V0TmFtZSgpO1xyXG4gICAgICAgIC8vICAgICBpZiAoIWluaENsYXNzTmFtZSkge1xyXG4gICAgICAgIC8vICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBJbmhlcml0ZWQgY2xhc3Mgb3IgaW50ZXJmYWNlIG5hbWUgbm90IGZvdW5kIGZvciAke2luaENsYXNzLmdldFRleHQoKX0uYCk7XHJcbiAgICAgICAgLy8gICAgIH1cclxuICAgICAgICAvLyAgICAgaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGluaENsYXNzKTtcclxuICAgICAgICAvLyB9XHJcbiAgICAgICAgLy8gLy8gYnVpbGQgdGhlIHVuaXF1ZSBrZXkgb2YgdGhlIGluaGVyaXRhbmNlXHJcbiAgICAgICAgLy8gaWYgKChjbHMgaW5zdGFuY2VvZiBDbGFzc0RlY2xhcmF0aW9uICYmIGluaENsYXNzIGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbikgXHJcbiAgICAgICAgLy8gICAgIHx8IChjbHMgaW5zdGFuY2VvZiBJbnRlcmZhY2VEZWNsYXJhdGlvbiAmJiBpbmhDbGFzcyBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uKSkge1xyXG4gICAgICAgIC8vICAgICBpbktleXdvcmQgPSBcIiBleHRlbmRzIFwiO1xyXG4gICAgICAgIC8vIH0gZWxzZSBpZiAoY2xzIGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbiAmJiAoaW5oQ2xhc3MgaW5zdGFuY2VvZiBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8fCBpbmhDbGFzcyBpbnN0YW5jZW9mIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cykpIHtcclxuICAgICAgICAvLyAgICAgaW5LZXl3b3JkID0gXCIgaW1wbGVtZW50cyBcIjtcclxuICAgICAgICAvLyB9IGVsc2Uge1xyXG4gICAgICAgIC8vICAgICB0aHJvdyBuZXcgRXJyb3IoYEluaGVyaXRhbmNlICR7Y2xzLmdldFRleHQoKX0gYW5kICR7aW5oQ2xhc3MuZ2V0VGV4dCgpfSBpcyBub3QgdmFsaWQuYCk7XHJcbiAgICAgICAgLy8gfVxyXG5cclxuICAgICAgICAvLyBjb25zdCBpbmhlcml0YW5jZUZ1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oY2xzKSArIGluS2V5d29yZCArIGluaENsYXNzRnVsbHlRdWFsaWZpZWROYW1lO1xyXG4gICAgICAgIC8vIGlmICh0aGlzLmZteEluaGVyaXRhbmNlTWFwLmhhcyhpbmhlcml0YW5jZUZ1bGx5UXVhbGlmaWVkTmFtZSkpIHtcclxuICAgICAgICAvLyAgICAgY29uc3QgckluaGVyaXRhbmNlID0gdGhpcy5mbXhJbmhlcml0YW5jZU1hcC5nZXQoaW5oZXJpdGFuY2VGdWxseVF1YWxpZmllZE5hbWUpO1xyXG4gICAgICAgIC8vICAgICBpZiAockluaGVyaXRhbmNlKSB7IFxyXG4gICAgICAgIC8vICAgICAgICByZXR1cm47IC8vIGRvbid0IGRvIGFueXRoaW5nXHJcbiAgICAgICAgLy8gICAgIH0gZWxzZSB7XHJcbiAgICAgICAgLy8gICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEluaGVyaXRhbmNlICR7Y2xzLmdldFRleHQoKX0gaXMgbm90IGZvdW5kIGluIHRoZSBpbmhlcml0YW5jZSBtYXAuYCk7XHJcbiAgICAgICAgLy8gICAgIH1cclxuICAgICAgICAvLyB9XHJcblxyXG4gICAgICAgIGNvbnN0IGZteEluaGVyaXRhbmNlID0gbmV3IEZhbWl4LkluaGVyaXRhbmNlKCk7XHJcblxyXG4gICAgICAgIC8vIGxvZ2dlci5kZWJ1ZyhgY3JlYXRlRmFtaXhJbmhlcml0YW5jZTogY2xhc3NGdWxseVF1YWxpZmllZE5hbWU6IGNsYXNzIGZxbiA9ICR7Y2xhc3NGdWxseVF1YWxpZmllZE5hbWV9YCk7XHJcblxyXG4gICAgICAgIGxldCBzdWJDbGFzczogRmFtaXguQ2xhc3MgfCBGYW1peC5JbnRlcmZhY2UgfCB1bmRlZmluZWQ7XHJcbiAgICAgICAgaWYgKGNscyBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pIHtcclxuICAgICAgICAgICAgc3ViQ2xhc3MgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhDbGFzcyhjbHMpO1xyXG4gICAgICAgIH1cclxuICAgICAgICBlbHNlIHtcclxuICAgICAgICAgICAgc3ViQ2xhc3MgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoY2xzKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmICghc3ViQ2xhc3MpIHtcclxuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBTdWJjbGFzcyAke2Nsc30gbm90IGZvdW5kIGluIENsYXNzIG9yIEludGVyZmFjZSBtYXBzLmApO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgbGV0IHN1cGVyQ2xhc3M6IEZhbWl4LkNsYXNzIHwgRmFtaXguSW50ZXJmYWNlIHwgdW5kZWZpbmVkO1xyXG4gICAgICAgIC8vIGlmIChpbmhDbGFzcyBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24gfHwgaW5oQ2xhc3MgaW5zdGFuY2VvZiBJbnRlcmZhY2VEZWNsYXJhdGlvbikge1xyXG4gICAgICAgIC8vICAgICBpbmhDbGFzc05hbWUgPSBpbmhDbGFzcy5nZXROYW1lKCk7XHJcbiAgICAgICAgLy8gICAgIGlmICghaW5oQ2xhc3NOYW1lKSB7XHJcbiAgICAgICAgLy8gICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEluaGVyaXRlZCBjbGFzcyBvciBpbnRlcmZhY2UgbmFtZSBub3QgZm91bmQgZm9yICR7aW5oQ2xhc3MuZ2V0VGV4dCgpfS5gKTtcclxuICAgICAgICAvLyAgICAgfVxyXG4gICAgICAgIC8vICAgICBpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oaW5oQ2xhc3MpO1xyXG4gICAgICAgIC8vICAgICBpZiAoaW5oQ2xhc3MgaW5zdGFuY2VvZiBDbGFzc0RlY2xhcmF0aW9uKSB7XHJcbiAgICAgICAgLy8gICAgICAgICBzdXBlckNsYXNzID0gdGhpcy5mbXhDbGFzc01hcC5nZXQoaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUpO1xyXG4gICAgICAgIC8vICAgICB9XHJcbiAgICAgICAgLy8gICAgIGVsc2Uge1xyXG4gICAgICAgIC8vICAgICAgICAgc3VwZXJDbGFzcyA9IHRoaXMuZm14SW50ZXJmYWNlTWFwLmdldChpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XHJcbiAgICAgICAgLy8gICAgIH1cclxuICAgICAgICAvLyAgICAgaWYgKCFzdXBlckNsYXNzKSB7XHJcbiAgICAgICAgLy8gICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFN1cGVyY2xhc3MgJHtjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZX0gbm90IGZvdW5kIGluIENsYXNzIG9yIEludGVyZmFjZSBtYXBzLmApO1xyXG4gICAgICAgIC8vICAgICB9XHJcbiAgICAgICAgLy8gfVxyXG5cclxuICAgICAgICAvLyAvLyBpdCBzaG91bGRuJ3QgYWRkIHRoZSBjbGFzcy9pbnRlcmZhY2UgYWdhaW4gdG8gdGhlIE1hcCwgaXQgc2hvdWxkIHVzZSBjcmVhdGVPckdldCAoPylcclxuICAgICAgICAvLyBpZiAoc3VwZXJDbGFzcyA9PT0gdW5kZWZpbmVkKSB7XHJcblxyXG4gICAgICAgIGlmIChpbmhDbGFzcyBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pIHtcclxuICAgICAgICAgICAgLy8gc3VwZXJDbGFzcyA9IG5ldyBGYW1peC5DbGFzcygpO1xyXG4gICAgICAgICAgICBzdXBlckNsYXNzID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q2xhc3MoaW5oQ2xhc3MpO1xyXG4gICAgICAgICAgICAvLyB0aGlzLmZteENsYXNzTWFwLnNldChpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSwgc3VwZXJDbGFzcyk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGVsc2UgaWYgKGluaENsYXNzIGluc3RhbmNlb2YgSW50ZXJmYWNlRGVjbGFyYXRpb24pIHtcclxuICAgICAgICAgICAgLy8gc3VwZXJDbGFzcyA9IG5ldyBGYW1peC5JbnRlcmZhY2UoKTtcclxuICAgICAgICAgICAgc3VwZXJDbGFzcyA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peEludGVyZmFjZShpbmhDbGFzcyk7XHJcbiAgICAgICAgICAgIC8vIHRoaXMuZm14SW50ZXJmYWNlTWFwLnNldChpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSwgc3VwZXJDbGFzcyk7XHJcbiAgICAgICAgfSBlbHNlIHtcclxuICAgICAgICAgICAgLy8gaW5oQ2xhc3MgaW5zdGFuY2VvZiBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHNcclxuICAgICAgICAgICAgY29uc3QgaW50ZXJmYWNlRGVjbGFyYXRpb24gPSBnZXRJbnRlcmZhY2VEZWNsYXJhdGlvbkZyb21FeHByZXNzaW9uKGluaENsYXNzKTtcclxuICAgICAgICAgICAgaWYgKGludGVyZmFjZURlY2xhcmF0aW9uICE9PSB1bmRlZmluZWQpIHtcclxuICAgICAgICAgICAgICAgIHN1cGVyQ2xhc3MgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoaW50ZXJmYWNlRGVjbGFyYXRpb24pO1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBJbnRlcmZhY2UgZGVjbGFyYXRpb24gbm90IGZvdW5kIGZvciAke2luaENsYXNzLmdldFRleHQoKX0uYCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoc3VwZXJDbGFzcywgaW5oQ2xhc3MpO1xyXG5cclxuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbmhDbGFzcywgc3VwZXJDbGFzcyk7XHJcblxyXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChzdXBlckNsYXNzKTtcclxuXHJcbiAgICAgICAgZm14SW5oZXJpdGFuY2Uuc3ViY2xhc3MgPSBzdWJDbGFzcztcclxuICAgICAgICBmbXhJbmhlcml0YW5jZS5zdXBlcmNsYXNzID0gc3VwZXJDbGFzcztcclxuXHJcbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEluaGVyaXRhbmNlKTtcclxuICAgICAgICAvLyBubyBGUU4gZm9yIGluaGVyaXRhbmNlXHJcblxyXG4gICAgICAgIC8vIFdlIGRvbid0IG1hcCBpbmhlcml0YW5jZSB0byB0aGUgc291cmNlIGNvZGUgZWxlbWVudCBiZWNhdXNlIHRoZXJlIGFyZSB0d28gZWxlbWVudHMgKHN1cGVyLCBzdWIpXHJcbiAgICAgICAgLy8gdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhJbmhlcml0YW5jZSwgbnVsbCk7XHJcblxyXG4gICAgfVxyXG5cclxuICAgIHB1YmxpYyBjcmVhdGVGYW1peEltcG9ydENsYXVzZShpbXBvcnRlZEVudGl0eTogRmFtaXguTmFtZWRFbnRpdHksIGltcG9ydGluZ0VudGl0eTogRmFtaXguTW9kdWxlKSB7XHJcbiAgICAgICAgY29uc3QgZm14SW1wb3J0Q2xhdXNlID0gbmV3IEZhbWl4LkltcG9ydENsYXVzZSgpO1xyXG4gICAgICAgIGZteEltcG9ydENsYXVzZS5pbXBvcnRlZEVudGl0eSA9IGltcG9ydGVkRW50aXR5O1xyXG4gICAgICAgIGZteEltcG9ydENsYXVzZS5pbXBvcnRpbmdFbnRpdHkgPSBpbXBvcnRpbmdFbnRpdHk7XHJcbiAgICAgICAgaW1wb3J0aW5nRW50aXR5LmFkZE91dGdvaW5nSW1wb3J0KGZteEltcG9ydENsYXVzZSk7XHJcbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEltcG9ydENsYXVzZSk7XHJcbiAgICB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW1wb3J0IGNsYXVzZVxyXG4gICAgICogQHBhcmFtIGltcG9ydENsYXVzZUluZm8gVGhlIGluZm9ybWF0aW9uIG5lZWRlZCB0byBjcmVhdGUgYSBGYW1peCBpbXBvcnQgY2xhdXNlXHJcbiAgICAgKiBAcGFyYW0gaW1wb3J0RGVjbGFyYXRpb24gVGhlIGltcG9ydCBkZWNsYXJhdGlvblxyXG4gICAgICogQHBhcmFtIGltcG9ydGVyIEEgc291cmNlIGZpbGUgd2hpY2ggaXMgYSBtb2R1bGVcclxuICAgICAqIEBwYXJhbSBtb2R1bGVTcGVjaWZpZXJGaWxlUGF0aCBUaGUgcGF0aCBvZiB0aGUgbW9kdWxlIHdoZXJlIHRoZSBleHBvcnQgZGVjbGFyYXRpb24gaXNcclxuICAgICAqIEBwYXJhbSBpbXBvcnRFbGVtZW50IFRoZSBpbXBvcnRlZCBlbnRpdHlcclxuICAgICAqIEBwYXJhbSBpc0luRXhwb3J0cyBBIGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgaW1wb3J0ZWQgZW50aXR5IGlzIGluIHRoZSBleHBvcnRzXHJcbiAgICAgKiBAcGFyYW0gaXNEZWZhdWx0RXhwb3J0IEEgYm9vbGVhbiBpbmRpY2F0aW5nIGlmIHRoZSBpbXBvcnRlZCBlbnRpdHkgaXMgYSBkZWZhdWx0IGV4cG9ydFxyXG4gICAgICovXHJcbiAgICBwdWJsaWMgb2xkQ3JlYXRlT3JHZXRGYW1peEltcG9ydENsYXVzZShpbXBvcnRDbGF1c2VJbmZvOiB7aW1wb3J0RGVjbGFyYXRpb24/OiBJbXBvcnREZWNsYXJhdGlvbiB8IEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uLCBpbXBvcnRlclNvdXJjZUZpbGU6IFNvdXJjZUZpbGUsIG1vZHVsZVNwZWNpZmllckZpbGVQYXRoOiBzdHJpbmcsIGltcG9ydEVsZW1lbnQ6IEltcG9ydFNwZWNpZmllciB8IElkZW50aWZpZXIsIGlzSW5FeHBvcnRzOiBib29sZWFuLCBpc0RlZmF1bHRFeHBvcnQ6IGJvb2xlYW59KTogdm9pZCB7XHJcbiAgICAgICAgY29uc3Qge2ltcG9ydERlY2xhcmF0aW9uLCBpbXBvcnRlclNvdXJjZUZpbGU6IGltcG9ydGVyLCBtb2R1bGVTcGVjaWZpZXJGaWxlUGF0aCwgaW1wb3J0RWxlbWVudCwgaXNJbkV4cG9ydHMsIGlzRGVmYXVsdEV4cG9ydH0gPSBpbXBvcnRDbGF1c2VJbmZvO1xyXG4gICAgICAgIGlmIChpbXBvcnREZWNsYXJhdGlvbiAmJiB0aGlzLmZteEltcG9ydENsYXVzZU1hcC5oYXMoaW1wb3J0RGVjbGFyYXRpb24pKSB7XHJcbiAgICAgICAgICAgIGNvbnN0IHJJbXBvcnRDbGF1c2UgPSB0aGlzLmZteEltcG9ydENsYXVzZU1hcC5nZXQoaW1wb3J0RGVjbGFyYXRpb24pO1xyXG4gICAgICAgICAgICBpZiAockltcG9ydENsYXVzZSkgeyBcclxuICAgICAgICAgICAgICAgcmV0dXJuOyAvLyBkb24ndCBkbyBhbnl0aGluZ1xyXG4gICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBJbXBvcnQgY2xhdXNlICR7aW1wb3J0RWxlbWVudC5nZXRUZXh0KCl9IGlzIG5vdCBmb3VuZCBpbiB0aGUgaW1wb3J0IGNsYXVzZSBtYXAuYCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGxvZ2dlci5pbmZvKGBjcmVhdGluZyBhIG5ldyBGYW1peEltcG9ydENsYXVzZSBmb3IgJHtpbXBvcnREZWNsYXJhdGlvbj8uZ2V0VGV4dCgpfSBpbiAke2ltcG9ydGVyLmdldEJhc2VOYW1lKCl9LmApO1xyXG4gICAgICAgIGNvbnN0IGZteEltcG9ydENsYXVzZSA9IG5ldyBGYW1peC5JbXBvcnRDbGF1c2UoKTtcclxuXHJcbiAgICAgICAgbGV0IGltcG9ydGVkRW50aXR5OiBGYW1peC5OYW1lZEVudGl0eSB8IEZhbWl4LlN0cnVjdHVyYWxFbnRpdHkgfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ7XHJcbiAgICAgICAgbGV0IGltcG9ydGVkRW50aXR5TmFtZTogc3RyaW5nO1xyXG5cclxuICAgICAgICBjb25zdCBhYnNvbHV0ZVBhdGhQcm9qZWN0ID0gdGhpcy5mYW1peFJlcC5nZXRBYnNvbHV0ZVBhdGgoKTtcclxuICAgICAgICBcclxuICAgICAgICBjb25zdCBhYnNvbHV0ZVBhdGggPSBwYXRoLm5vcm1hbGl6ZShtb2R1bGVTcGVjaWZpZXJGaWxlUGF0aCk7XHJcbiAgICAgICAgLy8gY29udmVydCB0aGUgcGF0aCBhbmQgcmVtb3ZlIGFueSB3aW5kb3dzIGJhY2tzbGFzaGVzIGludHJvZHVjZWQgYnkgcGF0aC5ub3JtYWxpemVcclxuICAgICAgICBsb2dnZXIuZGVidWcoYGNyZWF0ZUZhbWl4SW1wb3J0Q2xhdXNlOiBhYnNvbHV0ZVBhdGg6ICR7YWJzb2x1dGVQYXRofWApO1xyXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgY3JlYXRlRmFtaXhJbXBvcnRDbGF1c2U6IGNvbnZlcnRUb1JlbGF0aXZlUGF0aDogJHt0aGlzLmNvbnZlcnRUb1JlbGF0aXZlUGF0aChhYnNvbHV0ZVBhdGgsIGFic29sdXRlUGF0aFByb2plY3QpfWApO1xyXG4gICAgICAgIGNvbnN0IHBhdGhJblByb2plY3Q6IHN0cmluZyA9IHRoaXMuY29udmVydFRvUmVsYXRpdmVQYXRoKGFic29sdXRlUGF0aCwgYWJzb2x1dGVQYXRoUHJvamVjdCkucmVwbGFjZSgvXFxcXC9nLCBcIi9cIik7XHJcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBjcmVhdGVGYW1peEltcG9ydENsYXVzZTogcGF0aEluUHJvamVjdDogJHtwYXRoSW5Qcm9qZWN0fWApO1xyXG4gICAgICAgIGxldCBwYXRoTmFtZSA9IFwie1wiICsgcGF0aEluUHJvamVjdCArIFwifS5cIjtcclxuICAgICAgICBsb2dnZXIuZGVidWcoYGNyZWF0ZUZhbWl4SW1wb3J0Q2xhdXNlOiBwYXRoTmFtZTogJHtwYXRoTmFtZX1gKTtcclxuXHJcbiAgICAgICAgLy8gTmFtZWQgaW1wb3J0cywgZS5nLiBpbXBvcnQgeyBDbGFzc1cgfSBmcm9tIFwiLi9jb21wbGV4RXhwb3J0TW9kdWxlXCI7XHJcblxyXG4gICAgICAgIC8vIFN0YXJ0IHdpdGggc2ltcGxlIGltcG9ydCBjbGF1c2UgKHdpdGhvdXQgcmVmZXJyaW5nIHRvIHRoZSBhY3R1YWwgdmFyaWFibGUpXHJcblxyXG4gICAgICAgIGlmIChpbXBvcnREZWNsYXJhdGlvbiBpbnN0YW5jZW9mIEltcG9ydERlY2xhcmF0aW9uIFxyXG4gICAgICAgICAgICAmJiBpbXBvcnRFbGVtZW50IGluc3RhbmNlb2YgSW1wb3J0U3BlY2lmaWVyKSB7IFxyXG4gICAgICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHlOYW1lID0gaW1wb3J0RWxlbWVudC5nZXROYW1lKCk7XHJcbiAgICAgICAgICAgIHBhdGhOYW1lID0gcGF0aE5hbWUgKyBpbXBvcnRlZEVudGl0eU5hbWU7XHJcbiAgICAgICAgICAgIGlmIChpc0luRXhwb3J0cykge1xyXG4gICAgICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUocGF0aE5hbWUpIGFzIEZhbWl4Lk5hbWVkRW50aXR5O1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGlmIChpbXBvcnRlZEVudGl0eSA9PT0gdW5kZWZpbmVkKSB7XHJcbiAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eSA9IG5ldyBGYW1peC5OYW1lZEVudGl0eSgpO1xyXG4gICAgICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkubmFtZSA9IGltcG9ydGVkRW50aXR5TmFtZTtcclxuICAgICAgICAgICAgICAgIGlmICghaXNJbkV4cG9ydHMpIHtcclxuICAgICAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5pc1N0dWIgPSB0cnVlO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICAgICAgLy8gbG9nZ2VyLmRlYnVnKGBjcmVhdGVGYW1peEltcG9ydENsYXVzZTogQ3JlYXRpbmcgbmFtZWQgZW50aXR5ICR7aW1wb3J0ZWRFbnRpdHlOYW1lfSB3aXRoIEZRTiAke3BhdGhOYW1lfWApO1xyXG4gICAgICAgICAgICAgICAgLy8gaW1wb3J0ZWRFbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lID0gcGF0aE5hbWU7XHJcbiAgICAgICAgICAgICAgICBpbml0RlFOKGltcG9ydEVsZW1lbnQsIGltcG9ydGVkRW50aXR5KTtcclxuXHJcbiAgICAgICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbXBvcnRFbGVtZW50LCBpbXBvcnRlZEVudGl0eSk7XHJcbiAgICAgICAgICAgICAgICAvLyBtdXN0IGFkZCBlbnRpdHkgdG8gcmVwb3NpdG9yeVxyXG4gICAgICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGltcG9ydGVkRW50aXR5KTtcclxuICAgICAgICAgICAgfVxyXG4gICAgICAgIH1cclxuICAgICAgICAvLyBoYW5kbGUgaW1wb3J0IGVxdWFscyBkZWNsYXJhdGlvbnMsIGUuZy4gaW1wb3J0IG15TW9kdWxlID0gcmVxdWlyZShcIi4vY29tcGxleEV4cG9ydE1vZHVsZVwiKTtcclxuICAgICAgICAvLyBUeXBlU2NyaXB0IGNhbid0IGRldGVybWluZSB0aGUgdHlwZSBvZiB0aGUgaW1wb3J0ZWQgbW9kdWxlLCBzbyB3ZSBjcmVhdGUgYSBNb2R1bGUgZW50aXR5XHJcbiAgICAgICAgZWxzZSBpZiAoaW1wb3J0RGVjbGFyYXRpb24gaW5zdGFuY2VvZiBJbXBvcnRFcXVhbHNEZWNsYXJhdGlvbikge1xyXG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eU5hbWUgPSBpbXBvcnREZWNsYXJhdGlvbj8uZ2V0TmFtZSgpO1xyXG4gICAgICAgICAgICBwYXRoTmFtZSA9IHBhdGhOYW1lICsgaW1wb3J0ZWRFbnRpdHlOYW1lO1xyXG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eSA9IG5ldyBGYW1peC5TdHJ1Y3R1cmFsRW50aXR5KCk7XHJcbiAgICAgICAgICAgIGltcG9ydGVkRW50aXR5Lm5hbWUgPSBpbXBvcnRlZEVudGl0eU5hbWU7XHJcbiAgICAgICAgICAgIGluaXRGUU4oaW1wb3J0RGVjbGFyYXRpb24sIGltcG9ydGVkRW50aXR5KTtcclxuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IoaW1wb3J0RWxlbWVudCwgaW1wb3J0ZWRFbnRpdHkpO1xyXG4gICAgICAgICAgICAvLyBpbXBvcnRlZEVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPSBwYXRoTmFtZTtcclxuICAgICAgICAgICAgY29uc3QgYW55VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoJ2FueScsIGltcG9ydERlY2xhcmF0aW9uKTtcclxuICAgICAgICAgICAgKGltcG9ydGVkRW50aXR5IGFzIEZhbWl4LlN0cnVjdHVyYWxFbnRpdHkpLmRlY2xhcmVkVHlwZSA9IGFueVR5cGU7XHJcbiAgICAgICAgfSBlbHNlIHsgIC8vIGRlZmF1bHQgaW1wb3J0cywgZS5nLiBpbXBvcnQgQ2xhc3NXIGZyb20gXCIuL2NvbXBsZXhFeHBvcnRNb2R1bGVcIjsgIFxyXG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eU5hbWUgPSBpbXBvcnRFbGVtZW50LmdldFRleHQoKTtcclxuICAgICAgICAgICAgcGF0aE5hbWUgPSBwYXRoTmFtZSArIChpc0RlZmF1bHRFeHBvcnQgPyBcImRlZmF1bHRFeHBvcnRcIiA6IFwibmFtZXNwYWNlRXhwb3J0XCIpO1xyXG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eSA9IG5ldyBGYW1peC5OYW1lZEVudGl0eSgpO1xyXG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5uYW1lID0gaW1wb3J0ZWRFbnRpdHlOYW1lO1xyXG4gICAgICAgICAgICAvLyBpbXBvcnRlZEVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPSBwYXRoTmFtZTtcclxuICAgICAgICAgICAgaW5pdEZRTihpbXBvcnRFbGVtZW50LCBpbXBvcnRlZEVudGl0eSk7XHJcbiAgICAgICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGltcG9ydEVsZW1lbnQsIGltcG9ydGVkRW50aXR5KTtcclxuICAgICAgICB9XHJcbiAgICAgICAgLy8gSSBkb24ndCB0aGluayBpdCBzaG91bGQgYmUgYWRkZWQgdG8gdGhlIHJlcG9zaXRvcnkgaWYgaXQgZXhpc3RzIGFscmVhZHlcclxuICAgICAgICBpZiAoIWlzSW5FeHBvcnRzKSB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoaW1wb3J0ZWRFbnRpdHkpO1xyXG4gICAgICAgIGNvbnN0IGltcG9ydGVyRnVsbHlRdWFsaWZpZWROYW1lID0gRlFORnVuY3Rpb25zLmdldEZRTihpbXBvcnRlcik7XHJcbiAgICAgICAgY29uc3QgZm14SW1wb3J0ZXIgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUoaW1wb3J0ZXJGdWxseVF1YWxpZmllZE5hbWUpIGFzIEZhbWl4Lk1vZHVsZTtcclxuICAgICAgICBmbXhJbXBvcnRDbGF1c2UuaW1wb3J0aW5nRW50aXR5ID0gZm14SW1wb3J0ZXI7XHJcbiAgICAgICAgZm14SW1wb3J0Q2xhdXNlLmltcG9ydGVkRW50aXR5ID0gaW1wb3J0ZWRFbnRpdHk7XHJcbiAgICAgICAgaWYgKGltcG9ydERlY2xhcmF0aW9uIGluc3RhbmNlb2YgSW1wb3J0RXF1YWxzRGVjbGFyYXRpb24pIHtcclxuICAgICAgICAgICAgZm14SW1wb3J0Q2xhdXNlLm1vZHVsZVNwZWNpZmllciA9IGltcG9ydERlY2xhcmF0aW9uPy5nZXRNb2R1bGVSZWZlcmVuY2UoKS5nZXRUZXh0KCkgYXMgc3RyaW5nO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIGZteEltcG9ydENsYXVzZS5tb2R1bGVTcGVjaWZpZXIgPSBpbXBvcnREZWNsYXJhdGlvbj8uZ2V0TW9kdWxlU3BlY2lmaWVyVmFsdWUoKSBhcyBzdHJpbmc7XHJcbiAgICAgICAgfVxyXG4gICAgXHJcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBjcmVhdGVGYW1peEltcG9ydENsYXVzZTogJHtmbXhJbXBvcnRDbGF1c2UuaW1wb3J0ZWRFbnRpdHk/Lm5hbWV9IChvZiB0eXBlICR7XHJcbiAgICAgICAgICAgIEhlbHBlcnMuZ2V0U3ViVHlwZU5hbWUoZm14SW1wb3J0Q2xhdXNlLmltcG9ydGVkRW50aXR5KX0pIGlzIGltcG9ydGVkIGJ5ICR7Zm14SW1wb3J0Q2xhdXNlLmltcG9ydGluZ0VudGl0eT8ubmFtZX1gKTtcclxuXHJcbiAgICAgICAgZm14SW1wb3J0ZXIuYWRkT3V0Z29pbmdJbXBvcnQoZm14SW1wb3J0Q2xhdXNlKTtcclxuXHJcbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEltcG9ydENsYXVzZSk7XHJcblxyXG4gICAgICAgIGlmIChpbXBvcnREZWNsYXJhdGlvbikge1xyXG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEltcG9ydENsYXVzZSwgaW1wb3J0RGVjbGFyYXRpb24pO1xyXG4gICAgICAgICAgICB0aGlzLmZteEltcG9ydENsYXVzZU1hcC5zZXQoaW1wb3J0RGVjbGFyYXRpb24sIGZteEltcG9ydENsYXVzZSk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IEFycm93IEZ1bmN0aW9uXHJcbiAgICAgKiBAcGFyYW0gYXJyb3dFeHByZXNzaW9uIEFuIEV4cHJlc3Npb25cclxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgdmFyaWFibGVcclxuICAgICAqL1xyXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhBcnJvd0Z1bmN0aW9uKGFycm93RXhwcmVzc2lvbjogRXhwcmVzc2lvbiwgY3VycmVudENDOiB7IFtrZXk6IHN0cmluZ106IG51bWJlciB9ICk6IEZhbWl4LkFycm93RnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljQXJyb3dGdW5jdGlvbiB7XHJcbiAgICAgICAgXHJcbiAgICAgICAgbGV0IGZteEFycm93RnVuY3Rpb246IEZhbWl4LkFycm93RnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljQXJyb3dGdW5jdGlvbjtcclxuICAgICAgICBjb25zdCBmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oYXJyb3dFeHByZXNzaW9uKTtcclxuXHJcbiAgICAgICAgaWYgKCF0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmhhcyhmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSkpIHtcclxuXHJcbiAgICAgICAgICAgIGNvbnN0IGFycm93RnVuY3Rpb24gPSBhcnJvd0V4cHJlc3Npb24uYXNLaW5kT3JUaHJvdyhTeW50YXhLaW5kLkFycm93RnVuY3Rpb24pO1xyXG5cclxuICAgICAgICAgICAgY29uc3QgaXNHZW5lcmljID0gYXJyb3dGdW5jdGlvbi5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XHJcblxyXG4gICAgICAgICAgICBpZiAoaXNHZW5lcmljKSB7XHJcbiAgICAgICAgICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uID0gbmV3IEZhbWl4LlBhcmFtZXRyaWNBcnJvd0Z1bmN0aW9uKCk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uID0gbmV3IEZhbWl4LkFycm93RnVuY3Rpb24oKTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgLy8gR2V0IHRoZSBwYXJlbnQgb2YgdGhlIGFycm93IGZ1bmN0aW9uICh0aGUgdmFyaWFibGUgZGVjbGFyYXRpb24pXHJcbiAgICAgICAgICAgIGNvbnN0IHBhcmVudCA9IGFycm93RnVuY3Rpb24uZ2V0UGFyZW50SWZLaW5kKFN5bnRheEtpbmQuVmFyaWFibGVEZWNsYXJhdGlvbik7XHJcbiAgICAgICAgICAgIGxldCBmdW5jdGlvbk5hbWUgPSAnKE5PX05BTUUpJztcclxuXHJcbiAgICAgICAgICAgIGlmIChwYXJlbnQgJiYgcGFyZW50IGluc3RhbmNlb2YgVmFyaWFibGVEZWNsYXJhdGlvbikge1xyXG4gICAgICAgICAgICAgICAgLy8gR2V0IHRoZSBuYW1lIG9mIHRoZSB2YXJpYWJsZVxyXG4gICAgICAgICAgICAgICAgZnVuY3Rpb25OYW1lID0gcGFyZW50LmdldE5hbWUoKTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgaWYgKGZ1bmN0aW9uTmFtZSkge1xyXG4gICAgICAgICAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5uYW1lID0gZnVuY3Rpb25OYW1lO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5uYW1lID0gXCJhbm9ueW1vdXNcIjtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgLy8gU2lnbmF0dXJlIG9mIGFuIGFycm93IGZ1bmN0aW9uIGlzIChwYXJhbWV0ZXJzKSA9PiByZXR1cm5fdHlwZVxyXG4gICAgICAgICAgICBjb25zdCBwYXJhbWV0ZXJzU2lnbmF0dXJlID0gYXJyb3dGdW5jdGlvbi5nZXRQYXJhbWV0ZXJzKCkubWFwKHAgPT4gcC5nZXRUZXh0KCkpLmpvaW4oXCIsIFwiKTtcclxuICAgICAgICAgICAgY29uc3QgcmV0dXJuVHlwZVNpZ25hdHVyZSA9IGFycm93RnVuY3Rpb24uZ2V0UmV0dXJuVHlwZSgpLmdldFRleHQoKTtcclxuICAgICAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5zaWduYXR1cmUgPSBgKCR7cGFyYW1ldGVyc1NpZ25hdHVyZX0pID0+ICR7cmV0dXJuVHlwZVNpZ25hdHVyZX1gO1xyXG4gICAgICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLmN5Y2xvbWF0aWNDb21wbGV4aXR5ID0gY3VycmVudENDW2ZteEFycm93RnVuY3Rpb24ubmFtZV07XHJcblxyXG4gICAgICAgICAgICBsZXQgZnVuY3Rpb25UeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcclxuICAgICAgICAgICAgdHJ5IHtcclxuICAgICAgICAgICAgICAgIGZ1bmN0aW9uVHlwZU5hbWUgPSBhcnJvd0Z1bmN0aW9uLmdldFJldHVyblR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xyXG4gICAgICAgICAgICB9IGNhdGNoIChlcnJvcikge1xyXG4gICAgICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHJldHVybiB0eXBlIG9mIGZ1bmN0aW9uOiAke2Z1bmN0aW9uTmFtZX0uIENvbnRpbnVpbmcuLi5gKTtcclxuICAgICAgICAgICAgfVxyXG5cclxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoZnVuY3Rpb25UeXBlTmFtZSwgYXJyb3dGdW5jdGlvbiBhcyB1bmtub3duIGFzIEZ1bmN0aW9uRGVjbGFyYXRpb24pO1xyXG4gICAgICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLmRlY2xhcmVkVHlwZSA9IGZteFR5cGU7XHJcbiAgICAgICAgICAgIGZteEFycm93RnVuY3Rpb24ubnVtYmVyT2ZMaW5lc09mQ29kZSA9IGFycm93RnVuY3Rpb24uZ2V0RW5kTGluZU51bWJlcigpIC0gYXJyb3dGdW5jdGlvbi5nZXRTdGFydExpbmVOdW1iZXIoKTtcclxuICAgICAgICAgICAgY29uc3QgcGFyYW1ldGVycyA9IGFycm93RnVuY3Rpb24uZ2V0UGFyYW1ldGVycygpO1xyXG4gICAgICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLm51bWJlck9mUGFyYW1ldGVycyA9IHBhcmFtZXRlcnMubGVuZ3RoO1xyXG4gICAgICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLm51bWJlck9mU3RhdGVtZW50cyA9IGFycm93RnVuY3Rpb24uZ2V0U3RhdGVtZW50cygpLmxlbmd0aDtcclxuICAgICAgICAgICAgaW5pdEZRTihhcnJvd0V4cHJlc3Npb24gYXMgdW5rbm93biBhcyBUU01vcnBoT2JqZWN0VHlwZSwgZm14QXJyb3dGdW5jdGlvbik7XHJcbiAgICAgICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGFycm93RXhwcmVzc2lvbiBhcyB1bmtub3duIGFzIFRTTW9ycGhPYmplY3RUeXBlLCBmbXhBcnJvd0Z1bmN0aW9uKTtcclxuICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEFycm93RnVuY3Rpb24pO1xyXG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEFycm93RnVuY3Rpb24sYXJyb3dGdW5jdGlvbiBhcyB1bmtub3duIGFzIFRTTW9ycGhPYmplY3RUeXBlKTtcclxuICAgICAgICAgICAgdGhpcy5mbXhGdW5jdGlvbkFuZE1ldGhvZE1hcC5zZXQoZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUsIGZteEFycm93RnVuY3Rpb24pO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIGZteEFycm93RnVuY3Rpb24gPSB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmdldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgRmFtaXguQXJyb3dGdW5jdGlvbjtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIHJldHVybiBmbXhBcnJvd0Z1bmN0aW9uO1xyXG4gICAgfVxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGNvbmNyZXRpc2F0aW9uXHJcbiAgICAgKiBAcGFyYW0gY2xzIEEgY2xhc3NcclxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgY29uY3JldGlzYXRpb25cclxuICAgICAqL1xyXG4gICAgLy8gcHVibGljIGNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb24oY29uRW50aXR5IDogRmFtaXguUGFyYW1ldHJpY0NsYXNzIHwgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSB8IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNNZXRob2QgLGdlbkVudGl0eSA6IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kKTogRmFtaXguQ29uY3JldGlzYXRpb24ge1xyXG4gICAgICAgIFxyXG4gICAgLy8gICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSBuZXcgRmFtaXguQ29uY3JldGlzYXRpb24oKTsgICAgICAgICAgICAgIFxyXG4gICAgICAgIFxyXG4gICAgLy8gICAgIGZteENvbmNyZXRpc2F0aW9uLmNvbmNyZXRlRW50aXR5ID0gY29uRW50aXR5O1xyXG4gICAgLy8gICAgIGZteENvbmNyZXRpc2F0aW9uLmdlbmVyaWNFbnRpdHkgPSBnZW5FbnRpdHk7XHJcbiAgICAvLyAgICAgLy8gdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhDb25jcmV0aXNhdGlvbixudWxsKTtcclxuICAgIC8vICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14Q29uY3JldGlzYXRpb24pOyAgICBcclxuICAgIC8vICAgICBjb25zdCBwYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbihmbXhDb25jcmV0aXNhdGlvbik7XHJcbiAgICAgICAgICAgIFxyXG4gICAgLy8gICAgIHJldHVybiBmbXhDb25jcmV0aXNhdGlvbjtcclxuICAgIC8vIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvblxyXG4gICAgICogQHBhcmFtIGNvbmNyZXRpc2F0aW9uIEEgRmFtaXhDb25jcmV0aXNhdGlvblxyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBQYXJhbWV0ZXJDb25jcmVzdGlzYXRpb25cclxuICAgICAqL1xyXG4gICAgLy8gcHVibGljIGNyZWF0ZUZhbWl4UGFyYW1ldGVyQ29uY3JldGlzYXRpb24oY29uY3JldGlzYXRpb246IEZhbWl4LkNvbmNyZXRpc2F0aW9uKTogRmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24gfCB1bmRlZmluZWR7XHJcbiAgICAvLyAgICAgY29uc3QgY29uQ2xhc3MgPSBjb25jcmV0aXNhdGlvbi5jb25jcmV0ZUVudGl0eTtcclxuICAgIC8vICAgICBjb25zdCBnZW5DbGFzcyA9IGNvbmNyZXRpc2F0aW9uLmdlbmVyaWNFbnRpdHk7XHJcbiAgICAvLyAgICAgbG9nZ2VyLmRlYnVnKGBDcmVhdGluZyBwYXJhbWV0ZXIgY29uY3JldGlzYXRpb24gYmV0d2VlbiAke2NvbkNsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZX0gYW5kICR7Z2VuQ2xhc3MuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xyXG4gICAgLy8gICAgIGNvbnN0IHBhcmFtZXRlckNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJQYXJhbWV0ZXJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24+O1xyXG4gICAgLy8gICAgIGNvbnN0IGNvbmNyZXRlUGFyYW1ldGVycyA9IGNvbkNsYXNzLmNvbmNyZXRlUGFyYW1ldGVycztcclxuICAgIC8vICAgICBjb25zdCBnZW5lcmljUGFyYW1ldGVycyA9IGdlbkNsYXNzLmdlbmVyaWNQYXJhbWV0ZXJzO1xyXG4gICAgICAgIFxyXG4gICAgLy8gICAgIGxldCBjb25DbGFzc1R5cGVQYXJhbWV0ZXJzSXRlcmF0b3IgPSBjb25jcmV0ZVBhcmFtZXRlcnMudmFsdWVzKCk7XHJcbiAgICAvLyAgICAgbGV0IGdlbkNsYXNzVHlwZVBhcmFtZXRlcnNJdGVyYXRvciA9IGdlbmVyaWNQYXJhbWV0ZXJzLnZhbHVlcygpO1xyXG4gICAgLy8gICAgIGxldCBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA6IEZhbWl4LlBhcmFtZXRlckNvbmNyZXRpc2F0aW9uIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkO1xyXG5cclxuICAgIC8vICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGdlbmVyaWNQYXJhbWV0ZXJzLnNpemU7IGkrKykge1xyXG4gICAgLy8gICAgICAgICBjb25zdCBjb25DbGFzc1R5cGVQYXJhbWV0ZXIgPSBjb25DbGFzc1R5cGVQYXJhbWV0ZXJzSXRlcmF0b3IubmV4dCgpLnZhbHVlIGFzIEZhbWl4LlBhcmFtZXRlclR5cGU7XHJcbiAgICAvLyAgICAgICAgIGNvbnN0IGdlbkNsYXNzVHlwZVBhcmFtZXRlciA9IGdlbkNsYXNzVHlwZVBhcmFtZXRlcnNJdGVyYXRvci5uZXh0KCkudmFsdWUgYXMgRmFtaXguUGFyYW1ldGVyVHlwZTtcclxuICAgIC8vICAgICAgICAgbGV0IGNyZWF0ZVBhcmFtZXRlckNvbmNyZXRpc2F0aW9uIDogYm9vbGVhbiA9IHRydWU7XHJcbiAgICAvLyAgICAgICAgIGlmKGNvbkNsYXNzVHlwZVBhcmFtZXRlciAmJiBnZW5DbGFzc1R5cGVQYXJhbWV0ZXIgJiYgY29uQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWUgIT0gZ2VuQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWUpe1xyXG4gICAgLy8gICAgICAgICAgICAgcGFyYW1ldGVyQ29uY3JldGlzYXRpb25zLmZvckVhY2goKHBhcmFtIDogRmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24pID0+IHtcclxuICAgIC8vICAgICAgICAgICAgICAgICBpZiAoY29uQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWUgPT0gcGFyYW0uY29uY3JldGVQYXJhbWV0ZXIubmFtZSAmJiBnZW5DbGFzc1R5cGVQYXJhbWV0ZXIubmFtZSA9PSBwYXJhbS5nZW5lcmljUGFyYW1ldGVyLm5hbWUpIHtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgY3JlYXRlUGFyYW1ldGVyQ29uY3JldGlzYXRpb24gPSBmYWxzZTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyQ29uY3JldGlzYXRpb24gPSBwYXJhbTtcclxuICAgIC8vICAgICAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgICAgICB9KVxyXG4gICAgLy8gICAgICAgICAgICAgaWYgKGNyZWF0ZVBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyQ29uY3JldGlzYXRpb24gPSBuZXcgRmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24oKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbi5nZW5lcmljUGFyYW1ldGVyID0gZ2VuQ2xhc3NUeXBlUGFyYW1ldGVyO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLmNvbmNyZXRlUGFyYW1ldGVyID0gY29uQ2xhc3NUeXBlUGFyYW1ldGVyO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLmFkZENvbmNyZXRpc2F0aW9uKGNvbmNyZXRpc2F0aW9uKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAvLyB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLG51bGwpO1xyXG4gICAgLy8gICAgICAgICAgICAgfSBlbHNlIHtcclxuICAgIC8vICAgICAgICAgICAgICAgICBpZiAoIWZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgZm14UGFyYW1ldGVyQ29uY3JldGlzYXRpb24gd2FzIHVuZGVmaW5lZCBmb3IgY29uY3JldGlzYXRpb24gd2l0aCBnZW5lcmljIHBhcmFtZXRlciAke2dlbkNsYXNzVHlwZVBhcmFtZXRlci5uYW1lfSBhbmQgY29uY3JldGUgcGFyYW1ldGVyICR7Y29uQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWV9YCk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLmFkZENvbmNyZXRpc2F0aW9uKGNvbmNyZXRpc2F0aW9uKTtcclxuICAgIC8vICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbik7XHJcbiAgICAvLyAgICAgICAgIH1cclxuICAgIC8vICAgICB9XHJcbiAgICAvLyAgICAgaWYgKCFmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbikge1xyXG4gICAgLy8gICAgICAgICBsb2dnZXIuZXJyb3IoYGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uIHdhcyB1bmRlZmluZWQgZm9yIGNvbmNyZXRpc2F0aW9uIHdpdGggY29uY3JldGUgZW50aXR5ICR7Y29uQ2xhc3MuZnVsbHlRdWFsaWZpZWROYW1lfSBhbmQgZ2VuZXJpYyBlbnRpdHkgJHtnZW5DbGFzcy5mdWxseVF1YWxpZmllZE5hbWV9YCk7XHJcbiAgICAvLyAgICAgfVxyXG4gICAgLy8gICAgIHJldHVybiBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbjtcclxuXHJcbiAgICAvLyB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggY29uY3JldGlzYXRpb24gYmV0d2VlbiB0d28gY2xhc3NlcyBvciB0d28gaW50ZXJmYWNlc1xyXG4gICAgICogQHBhcmFtIGVsZW1lbnQgQSBjbGFzcyBvciBhbiBJbnRlcmZhY2VcclxuICAgICAqL1xyXG4gICAgLy8gcHVibGljIGNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb25DbGFzc09ySW50ZXJmYWNlU3BlY2lhbGlzYXRpb24oZWxlbWVudDogQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uKXtcclxuICAgICAgICBcclxuICAgIC8vICAgICBjb25zdCBzdXBlckVudGl0eSA9IGVsZW1lbnQuZ2V0RXh0ZW5kcygpO1xyXG4gICAgLy8gICAgIGxldCBzdXBlckVudGl0eUFycmF5O1xyXG4gICAgLy8gICAgIGlmIChzdXBlckVudGl0eSl7XHJcbiAgICAvLyAgICAgICAgIHN1cGVyRW50aXR5QXJyYXkgPSBBcnJheS5pc0FycmF5KHN1cGVyRW50aXR5KSA/IHN1cGVyRW50aXR5IDogW3N1cGVyRW50aXR5XTtcclxuICAgIC8vICAgICB9XHJcbiAgICAvLyAgICAgaWYgKHN1cGVyRW50aXR5QXJyYXkgJiYgc3VwZXJFbnRpdHlBcnJheS5sZW5ndGggPiAwKSB7XHJcbiAgICAvLyAgICAgICAgIHN1cGVyRW50aXR5QXJyYXkuZm9yRWFjaChlbnRpdHkgPT4ge1xyXG4gICAgLy8gICAgICAgICAgICAgbGV0IGVudGl0eUlzR2VuZXJpYztcclxuICAgIC8vICAgICAgICAgICAgIGNvbnN0IHN1cGVyRW50aXR5U3ltYm9sID0gZW50aXR5LmdldEV4cHJlc3Npb24oKS5nZXRTeW1ib2xPclRocm93KCk7XHJcbiAgICAvLyAgICAgICAgICAgICBsZXQgc3VwZXJFbnRpdHlEZWNsYXJhdGlvbjtcclxuICAgIC8vICAgICAgICAgICAgIGlmIChzdXBlckVudGl0eSBpbnN0YW5jZW9mIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cykge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIHN1cGVyRW50aXR5RGVjbGFyYXRpb24gPSBzdXBlckVudGl0eVN5bWJvbC5nZXREZWNsYXJhdGlvbnMoKVswXS5hc0tpbmQodHMuU3ludGF4S2luZC5DbGFzc0RlY2xhcmF0aW9uKTtcclxuICAgIC8vICAgICAgICAgICAgIH0gZWxzZSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgc3VwZXJFbnRpdHlEZWNsYXJhdGlvbiA9IHN1cGVyRW50aXR5U3ltYm9sLmdldERlY2xhcmF0aW9ucygpWzBdLmFzS2luZCh0cy5TeW50YXhLaW5kLkludGVyZmFjZURlY2xhcmF0aW9uKTtcclxuICAgIC8vICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgIGlmIChzdXBlckVudGl0eURlY2xhcmF0aW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgZW50aXR5SXNHZW5lcmljID0gc3VwZXJFbnRpdHlEZWNsYXJhdGlvbi5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XHJcbiAgICAvLyAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgICAgICBpZiAoZW50aXR5SXNHZW5lcmljKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgbGV0IEVudGl0eURlY2xhcmF0aW9uO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGxldCBnZW5FbnRpdHk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgaWYgKHN1cGVyRW50aXR5IGluc3RhbmNlb2YgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIEVudGl0eURlY2xhcmF0aW9uID0gZW50aXR5LmdldEV4cHJlc3Npb24oKS5nZXRTeW1ib2woKT8uZ2V0RGVjbGFyYXRpb25zKClbMF0gYXMgQ2xhc3NEZWNsYXJhdGlvbjtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgZ2VuRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q2xhc3MoRW50aXR5RGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcclxuICAgIC8vICAgICAgICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBFbnRpdHlEZWNsYXJhdGlvbiA9IGVudGl0eS5nZXRFeHByZXNzaW9uKCkuZ2V0U3ltYm9sKCk/LmdldERlY2xhcmF0aW9ucygpWzBdIGFzIEludGVyZmFjZURlY2xhcmF0aW9uO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBnZW5FbnRpdHkgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoRW50aXR5RGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgICAgIGNvbnN0IGdlblBhcmFtcyA9IEVudGl0eURlY2xhcmF0aW9uLmdldFR5cGVQYXJhbWV0ZXJzKCkubWFwKChwYXJhbSkgPT4gcGFyYW0uZ2V0VGV4dCgpKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICBjb25zdCBhcmdzID0gZWxlbWVudC5nZXRIZXJpdGFnZUNsYXVzZXMoKVswXS5nZXRUeXBlTm9kZXMoKVswXS5nZXRUeXBlQXJndW1lbnRzKClcclxuICAgIC8vICAgICAgICAgICAgICAgICBjb25zdCBjb25QYXJhbXMgPSBlbGVtZW50LmdldEhlcml0YWdlQ2xhdXNlcygpWzBdLmdldFR5cGVOb2RlcygpWzBdLmdldFR5cGVBcmd1bWVudHMoKS5tYXAoKHBhcmFtKSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBsZXQgY29uRW50aXR5O1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBjb25FbnRpdHkgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZUVsZW1lbnQoZ2VuRW50aXR5LEVudGl0eURlY2xhcmF0aW9uLGFyZ3MpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBjb25zdCBjb25jcmV0aXNhdGlvbnMgPSB0aGlzLmZhbWl4UmVwLl9nZXRBbGxFbnRpdGllc1dpdGhUeXBlKFwiQ29uY3JldGlzYXRpb25cIikgYXMgU2V0PEZhbWl4LkNvbmNyZXRpc2F0aW9uPjtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgbGV0IGNyZWF0ZUNvbmNyZXRpc2F0aW9uIDogYm9vbGVhbiA9IHRydWU7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIGNvbmNyZXRpc2F0aW9ucy5mb3JFYWNoKChjb25jIDogRmFtaXguQ29uY3JldGlzYXRpb24pID0+IHtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChnZW5FbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lID09IGNvbmMuZ2VuZXJpY0VudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgJiYgY29uYy5jb25jcmV0ZUVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSl7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY3JlYXRlQ29uY3JldGlzYXRpb24gPSBmYWxzZTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgIFxyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBpZiAoY3JlYXRlQ29uY3JldGlzYXRpb24pIHtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSB0aGlzLmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb24oY29uRW50aXR5LGdlbkVudGl0eSk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgIH0pO1xyXG4gICAgLy8gICAgIH1cclxuICAgIC8vICAgICAvLyBUT0RPOiBUaGlzIGZ1bmN0aW9uIHNlZW1zIHVuZmluaXNoZWRcclxuICAgIC8vIH0gICAgXHJcbiAgICBcclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGEgY2xhc3MgYW5kIGl0cyBpbnN0YW5jaWF0aW9uc1xyXG4gICAgICogQHBhcmFtIGNscyBBIGNsYXNzXHJcbiAgICAgKi9cclxuICAgIC8vIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uR2VuZXJpY0luc3RhbnRpYXRpb24oY2xzOiBDbGFzc0RlY2xhcmF0aW9uKXtcclxuICAgICAgIFxyXG4gICAgLy8gICAgIGNvbnN0IGlzR2VuZXJpYyA9IGNscy5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XHJcbiAgICAvLyAgICAgaWYgKGlzR2VuZXJpYykge1xyXG4gICAgLy8gICAgICAgICBjb25zdCBpbnN0YW5jZXMgPSBjbHMuZ2V0U291cmNlRmlsZSgpLmdldERlc2NlbmRhbnRzT2ZLaW5kKHRzLlN5bnRheEtpbmQuTmV3RXhwcmVzc2lvbilcclxuICAgIC8vICAgICAgICAgICAgIC5maWx0ZXIobmV3RXhwciA9PiB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgY29uc3QgZXhwcmVzc2lvbiA9IG5ld0V4cHIuZ2V0RXhwcmVzc2lvbigpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIHJldHVybiBleHByZXNzaW9uLmdldFRleHQoKSA9PT0gY2xzLmdldE5hbWUoKTtcclxuICAgIC8vICAgICAgICAgfSk7XHJcblxyXG4gICAgLy8gICAgICAgICBpbnN0YW5jZXMuZm9yRWFjaChpbnN0YW5jZSA9PiB7XHJcbiAgICAvLyAgICAgICAgICAgICBjb25zdCBpbnN0YW5jZUlzR2VuZXJpYyA9IGluc3RhbmNlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGggPiAwO1xyXG4gICAgLy8gICAgICAgICAgICAgaWYgKGluc3RhbmNlSXNHZW5lcmljKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gaW5zdGFuY2UuZ2V0VHlwZUFyZ3VtZW50cygpLm1hcCgocGFyYW0pID0+IHBhcmFtLmdldFRleHQoKSk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgY29uc3QgZ2VuRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q2xhc3MoY2xzKSBhcyBGYW1peC5QYXJhbWV0cmljQ2xhc3M7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gY2xzLmdldFR5cGVQYXJhbWV0ZXJzKCkubWFwKChwYXJhbSkgPT4gcGFyYW0uZ2V0VGV4dCgpKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICBpZiAoIUhlbHBlcnMuYXJyYXlzQXJlRXF1YWwoY29uUGFyYW1zLGdlblBhcmFtcykpIHtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgbGV0IGNvbkVudGl0eTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgY29uRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q29uY3JldGVFbGVtZW50KGdlbkVudGl0eSxjbHMsaW5zdGFuY2UuZ2V0VHlwZUFyZ3VtZW50cygpKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uY3JldGlzYXRpb25zID0gdGhpcy5mYW1peFJlcC5fZ2V0QWxsRW50aXRpZXNXaXRoVHlwZShcIkNvbmNyZXRpc2F0aW9uXCIpIGFzIFNldDxGYW1peC5Db25jcmV0aXNhdGlvbj47XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIGxldCBjcmVhdGVDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBjb25jcmV0aXNhdGlvbnMuZm9yRWFjaCgoY29uYyA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uKSA9PiB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZ2VuRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25jLmdlbmVyaWNFbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lICYmIGNvbmMuY29uY3JldGVFbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lID09IGNvbkVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUpe1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbmNyZXRpc2F0aW9uID0gZmFsc2U7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIH0pO1xyXG4gICAgICAgICAgICBcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgaWYgKGNyZWF0ZUNvbmNyZXRpc2F0aW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBmbXhDb25jcmV0aXNhdGlvbiA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uID0gdGhpcy5jcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uKGNvbkVudGl0eSxnZW5FbnRpdHkpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICB9KVxyXG4gICAgLy8gICAgIH1cclxuICAgIC8vICAgICAvLyBUT0RPOiBUaGlzIGZ1bmN0aW9uIHNlZW1zIHVuZmluaXNoZWRcclxuICAgIC8vIH1cclxuXHJcbiAgICAvKipcclxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGEgY2xhc3MgYW5kIGl0cyBpbnN0YW5jaWF0aW9uc1xyXG4gICAgICogQHBhcmFtIGZ1bmMgQSBmdW5jdGlvblxyXG4gICAgICovXHJcbiAgICAvLyBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkZ1bmN0aW9uSW5zdGFudGlhdGlvbihlbGVtZW50OiBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgTWV0aG9kRGVjbGFyYXRpb24pe1xyXG4gICAgLy8gICAgIGNvbnN0IGlzR2VuZXJpYyA9IGVsZW1lbnQuZ2V0VHlwZVBhcmFtZXRlcnMoKS5sZW5ndGggPiAwO1xyXG4gICAgLy8gICAgIGlmIChpc0dlbmVyaWMpIHtcclxuICAgIC8vICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gZWxlbWVudC5nZXRUeXBlUGFyYW1ldGVycygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xyXG4gICAgLy8gICAgICAgICBjb25zdCB1c2VzID0gZWxlbWVudC5maW5kUmVmZXJlbmNlc0FzTm9kZXMoKTsgICAgXHJcbiAgICAvLyAgICAgICAgIHVzZXMuZm9yRWFjaCh1c2FnZSA9PiB7XHJcbiAgICAvLyAgICAgICAgICAgICBsZXQgY3VycmVudE5vZGU6IE5vZGUgfCB1bmRlZmluZWQgPSB1c2FnZTtcclxuXHJcbiAgICAvLyAgICAgICAgICAgICB3aGlsZSAoY3VycmVudE5vZGUpIHtcclxuICAgIC8vICAgICAgICAgICAgICAgICBpZiAoY3VycmVudE5vZGUuZ2V0S2luZCgpID09PSBTeW50YXhLaW5kLkNhbGxFeHByZXNzaW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNhbGxFeHByZXNzaW9uID0gY3VycmVudE5vZGUuYXNLaW5kKFN5bnRheEtpbmQuQ2FsbEV4cHJlc3Npb24pO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICBpZiAoIWNhbGxFeHByZXNzaW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYENhbGxFeHByZXNzaW9uIG5vdCBmb3VuZCBmb3IgJHtjdXJyZW50Tm9kZS5nZXRUZXh0KCl9YCk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgY29uc3QgaW5zdGFuY2VJc0dlbmVyaWMgPSBjYWxsRXhwcmVzc2lvbi5nZXRUeXBlQXJndW1lbnRzKCkubGVuZ3RoID4gMDtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgaWYgKGluc3RhbmNlSXNHZW5lcmljKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBhcmdzID0gY2FsbEV4cHJlc3Npb24uZ2V0VHlwZUFyZ3VtZW50cygpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gY2FsbEV4cHJlc3Npb24uZ2V0VHlwZUFyZ3VtZW50cygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCFIZWxwZXJzLmFycmF5c0FyZUVxdWFsKGNvblBhcmFtcyxnZW5QYXJhbXMpKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGdlbkVsZW1lbnQ7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYoZWxlbWVudCBpbnN0YW5jZW9mIEZ1bmN0aW9uRGVjbGFyYXRpb24pe1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZW5FbGVtZW50ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4RnVuY3Rpb24oZWxlbWVudCwge30pIGFzIEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbjtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZW5FbGVtZW50ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4TWV0aG9kKGVsZW1lbnQsIHt9KSBhcyBGYW1peC5QYXJhbWV0cmljTWV0aG9kO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgY29uY0VsZW1lbnQ7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uY0VsZW1lbnQgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZUVsZW1lbnQoZ2VuRWxlbWVudCxlbGVtZW50LGFyZ3MpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjcmVhdGVDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmNyZXRpc2F0aW9ucy5mb3JFYWNoKChjb25jIDogRmFtaXguQ29uY3JldGlzYXRpb24pID0+IHtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkVsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lID09IGNvbmMuZ2VuZXJpY0VudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgJiYgY29uYy5jb25jcmV0ZUVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uY0VsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lKXtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbmNyZXRpc2F0aW9uID0gZmFsc2U7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9KTtcclxuICAgICAgICBcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoY3JlYXRlQ29uY3JldGlzYXRpb24pIHtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZm14Q29uY3JldGlzYXRpb24gOiBGYW1peC5Db25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbihjb25jRWxlbWVudCxnZW5FbGVtZW50KTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICB9XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgYnJlYWs7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgICAgIC8vIFJlbW9udGVyIMOgIGwnw6lsw6ltZW50IHBhcmVudCAodXRpbGUgc2kgbGUgbsWTdWQgZGUgcsOpZsOpcmVuY2UgZXN0IHVuIGVuZmFudClcclxuICAgIC8vICAgICAgICAgICAgICAgICBjdXJyZW50Tm9kZSA9IGN1cnJlbnROb2RlLmdldFBhcmVudCgpO1xyXG4gICAgLy8gICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICB9KTtcclxuICAgIC8vICAgICB9XHJcbiAgICAvLyB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggY29uY3JldGlzYXRpb24gYmV0d2VlbiBhIGNsYXNzIGFuZCBhbiBpbnRlcmZhY2VcclxuICAgICAqIEBwYXJhbSBjbHMgQSBjbGFzc1xyXG4gICAgICovXHJcbiAgICAvLyBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkludGVyZmFjZUNsYXNzKGNsczogQ2xhc3NEZWNsYXJhdGlvbil7XHJcbiAgICBcclxuICAgIC8vICAgICBjb25zdCBzdXBlckludGVyZmFjZXMgPSBjbHMuZ2V0SW1wbGVtZW50cygpO1xyXG4gICAgLy8gICAgIHN1cGVySW50ZXJmYWNlcy5mb3JFYWNoKGludGVyZmFjZVR5cGUgPT4ge1xyXG4gICAgLy8gICAgICAgICBjb25zdCBpbnRlcmZhY2VJc0dlbmVyaWMgPSBpbnRlcmZhY2VUeXBlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGg+MDtcclxuICAgIC8vICAgICAgICAgaWYgKGludGVyZmFjZUlzR2VuZXJpYykge1xyXG4gICAgLy8gICAgICAgICAgICAgY29uc3QgaW50ZXJmYWNlRGVjbGFyYXRpb24gPSBpbnRlcmZhY2VUeXBlLmdldEV4cHJlc3Npb24oKS5nZXRTeW1ib2woKT8uZ2V0RGVjbGFyYXRpb25zKClbMF0gYXMgSW50ZXJmYWNlRGVjbGFyYXRpb247XHJcbiAgICAvLyAgICAgICAgICAgICBjb25zdCBnZW5QYXJhbXMgPSBpbnRlcmZhY2VEZWNsYXJhdGlvbi5nZXRUeXBlUGFyYW1ldGVycygpLm1hcCgocGFyYW0pID0+IHBhcmFtLmdldFRleHQoKSk7XHJcbiAgICAvLyAgICAgICAgICAgICBjb25zdCBjb25QYXJhbXMgPSBjbHMuZ2V0SGVyaXRhZ2VDbGF1c2VzKClbMF0uZ2V0VHlwZU5vZGVzKClbMF0uZ2V0VHlwZUFyZ3VtZW50cygpLm1hcCgocGFyYW0pID0+IHBhcmFtLmdldFRleHQoKSk7XHJcbiAgICAvLyAgICAgICAgICAgICBjb25zdCBhcmdzID0gY2xzLmdldEhlcml0YWdlQ2xhdXNlcygpWzBdLmdldFR5cGVOb2RlcygpWzBdLmdldFR5cGVBcmd1bWVudHMoKTtcclxuICAgIC8vICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGNvbnN0IGdlbkludGVyZmFjZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peEludGVyZmFjZShpbnRlcmZhY2VEZWNsYXJhdGlvbikgYXMgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZTtcclxuICAgIC8vICAgICAgICAgICAgICAgICBjb25zdCBjb25JbnRlcmZhY2UgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZUVsZW1lbnQoZ2VuSW50ZXJmYWNlLGludGVyZmFjZURlY2xhcmF0aW9uLGFyZ3MpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGxldCBjcmVhdGVDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGNvbmNyZXRpc2F0aW9ucy5mb3JFYWNoKChjb25jIDogRmFtaXguQ29uY3JldGlzYXRpb24pID0+IHtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkludGVyZmFjZS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25JbnRlcmZhY2UuZnVsbHlRdWFsaWZpZWROYW1lKXtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbmNyZXRpc2F0aW9uID0gZmFsc2U7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICB9KTtcclxuICAgICAgICAgICAgXHJcbiAgICAvLyAgICAgICAgICAgICAgICAgaWYgKGNyZWF0ZUNvbmNyZXRpc2F0aW9uKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSB0aGlzLmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb24oY29uSW50ZXJmYWNlLGdlbkludGVyZmFjZSk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICB9XHJcbiAgICAvLyAgICAgfSk7XHJcbiAgICAvLyB9XHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggY29uY3JldGlzYXRpb24gYmV0d2VlbiBhbiBpbnRlcmZhY2UgYW5kIGEgVHlwZVxyXG4gICAgICogQHBhcmFtIGVsZW1lbnQgQSB2YXJpYWJsZSBvciBhIGZ1bmN0aW9uXHJcbiAgICAgKiBAcGFyYW0gaW50ZXIgQW4gaW50ZXJmYWNlXHJcbiAgICAgKi9cclxuICAgIC8vIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uVHlwZUluc3RhbmNpYXRpb24oZWxlbWVudDogSW50ZXJmYWNlRGVjbGFyYXRpb24gfCBDbGFzc0RlY2xhcmF0aW9uKXtcclxuXHJcbiAgICAvLyAgICAgY29uc3QgaXNHZW5lcmljID0gZWxlbWVudC5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XHJcbiAgICAvLyAgICAgaWYgKGlzR2VuZXJpYykge1xyXG4gICAgLy8gICAgICAgICBjb25zdCBnZW5QYXJhbXMgPSBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCkubWFwKHBhcmFtID0+IHBhcmFtLmdldFRleHQoKSk7XHJcbiAgICAvLyAgICAgICAgIGNvbnN0IHVzZXMgPSBlbGVtZW50LmZpbmRSZWZlcmVuY2VzQXNOb2RlcygpO1xyXG4gICAgLy8gICAgICAgICB1c2VzLmZvckVhY2godXNlID0+IHsgICAgICAgIFxyXG4gICAgLy8gICAgICAgICAgICAgbGV0IHBhcmVudE5vZGUgPSB1c2UuZ2V0UGFyZW50KCk7XHJcbiAgICAvLyAgICAgICAgICAgICB3aGlsZSAocGFyZW50Tm9kZSkge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgIGlmIChwYXJlbnROb2RlLmdldEtpbmQoKSA9PT0gU3ludGF4S2luZC5UeXBlUmVmZXJlbmNlKSB7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHR5cGVSZWZlcmVuY2VOb2RlID0gcGFyZW50Tm9kZS5hc0tpbmQoU3ludGF4S2luZC5UeXBlUmVmZXJlbmNlKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgaWYgKCF0eXBlUmVmZXJlbmNlTm9kZSkge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBUeXBlUmVmZXJlbmNlTm9kZSBub3QgZm91bmQgZm9yICR7cGFyZW50Tm9kZS5nZXRUZXh0KCl9YCk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgY29uc3QgdHlwZVJlZmVyZW5jZU5vZGVJc0dlbmVyaWMgPSB0eXBlUmVmZXJlbmNlTm9kZS5nZXRUeXBlQXJndW1lbnRzKCkubGVuZ3RoID4gMDtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgaWYgKHR5cGVSZWZlcmVuY2VOb2RlSXNHZW5lcmljKSB7fVxyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgYXJncyA9IHR5cGVSZWZlcmVuY2VOb2RlLmdldFR5cGVBcmd1bWVudHMoKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvblBhcmFtcyA9IHR5cGVSZWZlcmVuY2VOb2RlLmdldFR5cGVBcmd1bWVudHMoKS5tYXAocGFyYW0gPT4gcGFyYW0uZ2V0VGV4dCgpKTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBnZW5FbGVtZW50O1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmKGVsZW1lbnQgaW5zdGFuY2VvZiBDbGFzc0RlY2xhcmF0aW9uKXtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2VuRWxlbWVudCA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENsYXNzKGVsZW1lbnQpIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9IGVsc2Uge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZW5FbGVtZW50ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4SW50ZXJmYWNlKGVsZW1lbnQpIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjb25jRWxlbWVudDtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jRWxlbWVudCA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChnZW5FbGVtZW50LGVsZW1lbnQsYXJncyk7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uY3JldGlzYXRpb25zID0gdGhpcy5mYW1peFJlcC5fZ2V0QWxsRW50aXRpZXNXaXRoVHlwZShcIkNvbmNyZXRpc2F0aW9uXCIpIGFzIFNldDxGYW1peC5Db25jcmV0aXNhdGlvbj47XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNyZWF0ZUNvbmNyZXRpc2F0aW9uIDogYm9vbGVhbiA9IHRydWU7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uY3JldGlzYXRpb25zLmZvckVhY2goKGNvbmMgOiBGYW1peC5Db25jcmV0aXNhdGlvbikgPT4ge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZ2VuRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25jRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUpe1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY3JlYXRlQ29uY3JldGlzYXRpb24gPSBmYWxzZTtcclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pO1xyXG4gICAgICAgIFxyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjcmVhdGVDb25jcmV0aXNhdGlvbikge1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBmbXhDb25jcmV0aXNhdGlvbiA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uID0gdGhpcy5jcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uKGNvbmNFbGVtZW50LGdlbkVsZW1lbnQpO1xyXG4gICAgLy8gICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgICAgIH1cclxuICAgIC8vICAgICAgICAgICAgICAgICAgICAgYnJlYWs7XHJcbiAgICAvLyAgICAgICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICAgICAgICAgIHBhcmVudE5vZGUgPSBwYXJlbnROb2RlLmdldFBhcmVudCgpO1xyXG4gICAgLy8gICAgICAgICAgICAgfVxyXG4gICAgLy8gICAgICAgICB9KTtcclxuICAgIC8vICAgICB9XHJcbiAgICAvLyB9XHJcblxyXG4gICAgcHVibGljIGNvbnZlcnRUb1JlbGF0aXZlUGF0aChhYnNvbHV0ZVBhdGg6IHN0cmluZywgYWJzb2x1dGVQYXRoUHJvamVjdDogc3RyaW5nKSB7XHJcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBjb252ZXJ0VG9SZWxhdGl2ZVBhdGg6IGFic29sdXRlUGF0aDogJyR7YWJzb2x1dGVQYXRofScsIGFic29sdXRlUGF0aFByb2plY3Q6ICcke2Fic29sdXRlUGF0aFByb2plY3R9J2ApO1xyXG4gICAgICAgIGlmIChhYnNvbHV0ZVBhdGguc3RhcnRzV2l0aChhYnNvbHV0ZVBhdGhQcm9qZWN0KSkge1xyXG4gICAgICAgICAgICByZXR1cm4gYWJzb2x1dGVQYXRoLnJlcGxhY2UoYWJzb2x1dGVQYXRoUHJvamVjdCwgXCJcIikuc2xpY2UoMSk7XHJcbiAgICAgICAgfSBlbHNlIGlmIChhYnNvbHV0ZVBhdGguc3RhcnRzV2l0aChcIi9cIikpIHtcclxuICAgICAgICAgICAgcmV0dXJuIGFic29sdXRlUGF0aC5zbGljZSgxKTtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICByZXR1cm4gYWJzb2x1dGVQYXRoO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxufVxyXG5cclxuZXhwb3J0IGZ1bmN0aW9uIGlzUHJpbWl0aXZlVHlwZSh0eXBlTmFtZTogc3RyaW5nKSB7XHJcbiAgICByZXR1cm4gdHlwZU5hbWUgPT09IFwibnVtYmVyXCIgfHxcclxuICAgICAgICB0eXBlTmFtZSA9PT0gXCJzdHJpbmdcIiB8fFxyXG4gICAgICAgIHR5cGVOYW1lID09PSBcImJvb2xlYW5cIiB8fFxyXG4gICAgICAgIHR5cGVOYW1lID09PSBcImJpZ2ludFwiIHx8XHJcbiAgICAgICAgdHlwZU5hbWUgPT09IFwic3ltYm9sXCIgfHxcclxuICAgICAgICB0eXBlTmFtZSA9PT0gXCJ1bmRlZmluZWRcIiB8fFxyXG4gICAgICAgIHR5cGVOYW1lID09PSBcIm51bGxcIiB8fFxyXG4gICAgICAgIHR5cGVOYW1lID09PSBcImFueVwiIHx8XHJcbiAgICAgICAgdHlwZU5hbWUgPT09IFwidW5rbm93blwiIHx8XHJcbiAgICAgICAgdHlwZU5hbWUgPT09IFwibmV2ZXJcIiB8fFxyXG4gICAgICAgIHR5cGVOYW1lID09PSBcInZvaWRcIjtcclxufVxyXG5cclxuZnVuY3Rpb24gaW5pdEZRTihzb3VyY2VFbGVtZW50OiBUU01vcnBoT2JqZWN0VHlwZSwgZmFtaXhFbGVtZW50OiBGYW1peC5Tb3VyY2VkRW50aXR5KSB7XHJcbiAgICAvLyBoYW5kbGUgc3BlY2lhbCBjYXNlcyB3aGVyZSBhbiBlbGVtZW50IGlzIGEgVHlwZSAtLSBuZWVkIHRvIGNoYW5nZSBpdHMgbmFtZVxyXG4gICAgaWYgKGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlR5cGUgJiYgIShzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgQ29tbWVudFJhbmdlKSAmJiBpc1R5cGVDb250ZXh0KHNvdXJjZUVsZW1lbnQpKSB7XHJcbiAgICAgICAgbGV0IGZxbiA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oc291cmNlRWxlbWVudCk7XHJcbiAgICAgICAgLy8gdXNpbmcgcmVnZXgsIHJlcGxhY2UgW2JsYWhdIHdpdGggW2JsYWhUeXBlXVxyXG4gICAgICAgIGZxbiA9IGZxbi5yZXBsYWNlKC9cXFsoW15cXF1dKylcXF0vZywgXCJbJDFUeXBlXVwiKTtcclxuICAgICAgICBsb2dnZXIuaW5mbyhcIlNldHRpbmcgZnVsbHkgcXVhbGlmaWVkIG5hbWUgZm9yIFwiICsgZmFtaXhFbGVtZW50LmdldEpTT04oKSArIFwiIHRvIFwiICsgZnFuKTtcclxuICAgICAgICBmYW1peEVsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lID0gZnFuO1xyXG4gICAgICAgIHJldHVybjtcclxuICAgIH1cclxuICAgIC8vIGNhdGNoIGFsbCAoZXhjZXB0IGNvbW1lbnRzKVxyXG4gICAgaWYgKCEoc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIENvbW1lbnRSYW5nZSkpIHtcclxuICAgICAgICBjb25zdCBmcW4gPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKHNvdXJjZUVsZW1lbnQpO1xyXG4gICAgICAgIGxvZ2dlci5pbmZvKFwiU2V0dGluZyBmdWxseSBxdWFsaWZpZWQgbmFtZSBmb3IgXCIgKyBmYW1peEVsZW1lbnQuZ2V0SlNPTigpICsgXCIgdG8gXCIgKyBmcW4pO1xyXG4gICAgICAgIChmYW1peEVsZW1lbnQgYXMgRmFtaXguTmFtZWRFbnRpdHkpLmZ1bGx5UXVhbGlmaWVkTmFtZSA9IGZxbjtcclxuICAgIH0gXHJcbn1cclxuXHJcblxyXG5mdW5jdGlvbiBpc1R5cGVDb250ZXh0KHNvdXJjZUVsZW1lbnQ6IEltcG9ydERlY2xhcmF0aW9uIHwgSW1wb3J0RXF1YWxzRGVjbGFyYXRpb24gfCBTb3VyY2VGaWxlIHwgTW9kdWxlRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgRnVuY3Rpb25FeHByZXNzaW9uIHwgUGFyYW1ldGVyRGVjbGFyYXRpb24gfCBWYXJpYWJsZURlY2xhcmF0aW9uIHwgUHJvcGVydHlEZWNsYXJhdGlvbiB8IFByb3BlcnR5U2lnbmF0dXJlIHwgVHlwZVBhcmFtZXRlckRlY2xhcmF0aW9uIHwgSWRlbnRpZmllciB8IERlY29yYXRvciB8IEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgSW1wb3J0U3BlY2lmaWVyIHwgRW51bURlY2xhcmF0aW9uIHwgRW51bU1lbWJlciB8IFR5cGVBbGlhc0RlY2xhcmF0aW9uIHwgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzIHwgVFNNb3JwaFBhcmFtZXRyaWNUeXBlKSB7XHJcbiAgICByZXR1cm4gc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIENvbnN0cnVjdG9yRGVjbGFyYXRpb25cclxuICAgICAgICB8fCBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgTWV0aG9kRGVjbGFyYXRpb24gIFxyXG4gICAgICAgIHx8IHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBGdW5jdGlvbkRlY2xhcmF0aW9uIFxyXG4gICAgICAgIHx8IHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBGdW5jdGlvbkV4cHJlc3Npb24gXHJcbiAgICAgICAgfHwgc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIEFycm93RnVuY3Rpb25cclxuICAgICAgICB8fFxyXG4gICAgICAgIHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8fFxyXG4gICAgICAgIHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBWYXJpYWJsZURlY2xhcmF0aW9uIHx8XHJcbiAgICAgICAgc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIFByb3BlcnR5RGVjbGFyYXRpb24gfHxcclxuICAgICAgICBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgUHJvcGVydHlTaWduYXR1cmUgfHxcclxuICAgICAgICBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgVHlwZVBhcmFtZXRlckRlY2xhcmF0aW9uIHx8XHJcbiAgICAgICAgc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIElkZW50aWZpZXIgfHxcclxuICAgICAgICBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgRGVjb3JhdG9yIHx8XHJcbiAgICAgICAgc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfHxcclxuICAgICAgICBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fFxyXG4gICAgICAgIHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBJbXBvcnRTcGVjaWZpZXIgfHxcclxuICAgICAgICBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgRW51bURlY2xhcmF0aW9uIHx8XHJcbiAgICAgICAgc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIEVudW1NZW1iZXIgfHxcclxuICAgICAgICBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgVHlwZUFsaWFzRGVjbGFyYXRpb25cclxuICAgICAgICB8fCBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgSW1wb3J0RGVjbGFyYXRpb25cclxuICAgICAgICB8fCBzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzO1xyXG59XHJcblxyXG5cclxuZnVuY3Rpb24gZ2V0SW50ZXJmYWNlRGVjbGFyYXRpb25Gcm9tRXhwcmVzc2lvbihleHByZXNzaW9uOiBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHMpOiBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8IHVuZGVmaW5lZCB7XHJcbiAgICAvLyBTdGVwIDE6IEdldCB0aGUgdHlwZSBvZiB0aGUgZXhwcmVzc2lvblxyXG4gICAgY29uc3QgdHlwZSA9IGV4cHJlc3Npb24uZ2V0VHlwZSgpO1xyXG5cclxuICAgIC8vIFN0ZXAgMjogR2V0IHRoZSBzeW1ib2wgYXNzb2NpYXRlZCB3aXRoIHRoZSB0eXBlXHJcbiAgICBsZXQgc3ltYm9sID0gdHlwZS5nZXRTeW1ib2woKTtcclxuXHJcbiAgICBpZiAoIXN5bWJvbCkge1xyXG4gICAgICAgIC8vIElmIHN5bWJvbCBpcyBub3QgZm91bmQsIHRyeSB0byBnZXQgdGhlIHN5bWJvbCBmcm9tIHRoZSBpZGVudGlmaWVyXHJcbiAgICAgICAgY29uc3QgaWRlbnRpZmllciA9IGV4cHJlc3Npb24uZ2V0Rmlyc3REZXNjZW5kYW50QnlLaW5kKFN5bnRheEtpbmQuSWRlbnRpZmllcik7XHJcbiAgICAgICAgaWYgKCFpZGVudGlmaWVyKSB7XHJcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgSWRlbnRpZmllciBub3QgZm91bmQgZm9yICR7ZXhwcmVzc2lvbi5nZXRUZXh0KCl9LmApO1xyXG4gICAgICAgIH1cclxuICAgICAgICBzeW1ib2wgPSBpZGVudGlmaWVyLmdldFN5bWJvbCgpO1xyXG4gICAgICAgIGlmICghc3ltYm9sKSB7XHJcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgU3ltYm9sIG5vdCBmb3VuZCBmb3IgJHtpZGVudGlmaWVyLmdldFRleHQoKX0uYCk7XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIC8vIFN0ZXAgMzogUmVzb2x2ZSB0aGUgc3ltYm9sIHRvIGZpbmQgdGhlIGFjdHVhbCBkZWNsYXJhdGlvblxyXG4gICAgY29uc3QgaW50ZXJmYWNlRGVjbGFyYXRpb24gPSByZXNvbHZlU3ltYm9sVG9JbnRlcmZhY2VEZWNsYXJhdGlvbihzeW1ib2wpO1xyXG5cclxuICAgIGlmICghaW50ZXJmYWNlRGVjbGFyYXRpb24pIHtcclxuICAgICAgICBsb2dnZXIuZXJyb3IoYEludGVyZmFjZSBkZWNsYXJhdGlvbiBub3QgZm91bmQgZm9yICR7ZXhwcmVzc2lvbi5nZXRUZXh0KCl9LmApO1xyXG4gICAgfVxyXG5cclxuICAgIHJldHVybiBpbnRlcmZhY2VEZWNsYXJhdGlvbjtcclxufVxyXG5cclxuaW1wb3J0IHsgU3ltYm9sIGFzIFRTTW9ycGhTeW1ib2wgfSBmcm9tIFwidHMtbW9ycGhcIjtcclxuXHJcbmZ1bmN0aW9uIHJlc29sdmVTeW1ib2xUb0ludGVyZmFjZURlY2xhcmF0aW9uKHN5bWJvbDogVFNNb3JwaFN5bWJvbCk6IEludGVyZmFjZURlY2xhcmF0aW9uIHwgdW5kZWZpbmVkIHtcclxuICAgIC8vIEdldCB0aGUgZGVjbGFyYXRpb25zIGFzc29jaWF0ZWQgd2l0aCB0aGUgc3ltYm9sXHJcbiAgICBjb25zdCBkZWNsYXJhdGlvbnMgPSBzeW1ib2wuZ2V0RGVjbGFyYXRpb25zKCk7XHJcblxyXG4gICAgLy8gRmlsdGVyIGZvciBJbnRlcmZhY2VEZWNsYXJhdGlvblxyXG4gICAgY29uc3QgaW50ZXJmYWNlRGVjbGFyYXRpb24gPSBkZWNsYXJhdGlvbnMuZmluZChkZWNsYXJhdGlvbiA9PiBkZWNsYXJhdGlvbiBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uKSBhcyBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8IHVuZGVmaW5lZDtcclxuXHJcbiAgICBpZiAoaW50ZXJmYWNlRGVjbGFyYXRpb24pIHtcclxuICAgICAgICByZXR1cm4gaW50ZXJmYWNlRGVjbGFyYXRpb247XHJcbiAgICB9XHJcblxyXG4gICAgLy8gSGFuZGxlIGltcG9ydHM6IElmIHRoZSBzeW1ib2wgaXMgaW1wb3J0ZWQsIHJlc29sdmUgdGhlIGltcG9ydCB0byBmaW5kIHRoZSBhY3R1YWwgZGVjbGFyYXRpb25cclxuICAgIGZvciAoY29uc3QgZGVjbGFyYXRpb24gb2YgZGVjbGFyYXRpb25zKSB7XHJcbiAgICAgICAgaWYgKGRlY2xhcmF0aW9uLmdldEtpbmQoKSA9PT0gU3ludGF4S2luZC5JbXBvcnRTcGVjaWZpZXIpIHtcclxuICAgICAgICAgICAgY29uc3QgaW1wb3J0U3BlY2lmaWVyID0gZGVjbGFyYXRpb24gYXMgSW1wb3J0U3BlY2lmaWVyO1xyXG4gICAgICAgICAgICBjb25zdCBpbXBvcnREZWNsYXJhdGlvbiA9IGltcG9ydFNwZWNpZmllci5nZXRJbXBvcnREZWNsYXJhdGlvbigpO1xyXG4gICAgICAgICAgICBjb25zdCBtb2R1bGVTcGVjaWZpZXIgPSBpbXBvcnREZWNsYXJhdGlvbi5nZXRNb2R1bGVTcGVjaWZpZXJTb3VyY2VGaWxlKCk7XHJcblxyXG4gICAgICAgICAgICBpZiAobW9kdWxlU3BlY2lmaWVyKSB7XHJcbiAgICAgICAgICAgICAgICBjb25zdCBleHBvcnRlZFN5bWJvbHMgPSBtb2R1bGVTcGVjaWZpZXIuZ2V0RXhwb3J0U3ltYm9scygpO1xyXG4gICAgICAgICAgICAgICAgY29uc3QgZXhwb3J0ZWRTeW1ib2wgPSBleHBvcnRlZFN5bWJvbHMuZmluZChzeW1ib2wgPT4gc3ltYm9sLmdldE5hbWUoKSA9PT0gaW1wb3J0U3BlY2lmaWVyLmdldE5hbWUoKSk7XHJcbiAgICAgICAgICAgICAgICBpZiAoZXhwb3J0ZWRTeW1ib2wpIHtcclxuICAgICAgICAgICAgICAgICAgICByZXR1cm4gcmVzb2x2ZVN5bWJvbFRvSW50ZXJmYWNlRGVjbGFyYXRpb24oZXhwb3J0ZWRTeW1ib2wpO1xyXG4gICAgICAgICAgICAgICAgfVxyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIHJldHVybiB1bmRlZmluZWQ7XHJcbn1cclxuXHJcbi8vIGZ1bmN0aW9uIG9sZEdldEludGVyZmFjZURlY2xhcmF0aW9uRnJvbUV4cHJlc3Npb24oZXhwcmVzc2lvbjogRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzKTogSW50ZXJmYWNlRGVjbGFyYXRpb24gfCB1bmRlZmluZWQge1xyXG4vLyAgICAgLy8gVHdvIGNhc2VzOlxyXG4vLyAgICAgLy8gY2xhc3MgQSBpbXBsZW1lbnRzIEltcG9ydGVkSW50ZXJmYWNlLCBEZWNsYXJlZEludGVyZmFjZSB7fVxyXG4vLyAgICAgY29uc3QgdHlwZSA9IGV4cHJlc3Npb24uZ2V0VHlwZSgpO1xyXG5cclxuLy8gICAgIC8vIEltcG9ydGVkSW50ZXJmYWNlOiB0eXBlIHdpbGwgYSBzeW1ib2wgXHJcbi8vICAgICBsZXQgc3ltYm9sID0gdHlwZS5nZXRBbGlhc1N5bWJvbCgpOyAgLy8gd2lsbCBiZSBkZWZpbmVkIGZvciBpbXBvcnRlZCBpbnRlcmZhY2VzXHJcblxyXG4vLyAgICAgaWYgKCFzeW1ib2wpIHtcclxuLy8gICAgICAgICAvLyBEZWNsYXJlZEludGVyZmFjZTogdHlwZSB3aWxsIGJlIGFuIEludGVyZmFjZURlY2xhcmF0aW9uIG9uIElkZW50aWZpZXIgbm9kZSB0aGF0IGlzIHRoZSBjaGlsZCBvZiB0aGUgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzXHJcbi8vICAgICAgICAgY29uc3QgaWRlbnRpZmllciA9IGV4cHJlc3Npb24uZ2V0Rmlyc3REZXNjZW5kYW50QnlLaW5kKFN5bnRheEtpbmQuSWRlbnRpZmllcik7XHJcbi8vICAgICAgICAgaWYgKCFpZGVudGlmaWVyKSB7XHJcbi8vICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgSWRlbnRpZmllciBub3QgZm91bmQgZm9yICR7ZXhwcmVzc2lvbi5nZXRUZXh0KCl9LmApO1xyXG4vLyAgICAgICAgIH1cclxuLy8gICAgICAgICBzeW1ib2wgPSBpZGVudGlmaWVyLmdldFN5bWJvbCgpO1xyXG4vLyAgICAgICAgIGlmICghc3ltYm9sKSB7XHJcbi8vICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgU3ltYm9sIG5vdCBmb3VuZCBmb3IgJHtpZGVudGlmaWVyLmdldFRleHQoKX0uYCk7XHJcbi8vICAgICAgICAgfVxyXG4vLyAgICAgfVxyXG5cclxuLy8gICAgIC8vIFN0ZXAgMzogR2V0IHRoZSBkZWNsYXJhdGlvbnMgYXNzb2NpYXRlZCB3aXRoIHRoZSBzeW1ib2xcclxuLy8gICAgIGNvbnN0IGRlY2xhcmF0aW9ucyA9IHN5bWJvbC5nZXREZWNsYXJhdGlvbnMoKTtcclxuXHJcbi8vICAgICAvLyBTdGVwIDQ6IEZpbHRlciBmb3IgSW50ZXJmYWNlRGVjbGFyYXRpb25cclxuLy8gICAgIGNvbnN0IGludGVyZmFjZURlY2xhcmF0aW9uID0gZGVjbGFyYXRpb25zLmZpbmQoZGVjbGFyYXRpb24gPT4gZGVjbGFyYXRpb24gaW5zdGFuY2VvZiBJbnRlcmZhY2VEZWNsYXJhdGlvbikgYXMgSW50ZXJmYWNlRGVjbGFyYXRpb24gfCB1bmRlZmluZWQ7XHJcblxyXG4vLyAgICAgaWYgKCFpbnRlcmZhY2VEZWNsYXJhdGlvbikge1xyXG4vLyAgICAgICAgIHRocm93IG5ldyBFcnJvcihgSW50ZXJmYWNlIGRlY2xhcmF0aW9uIG5vdCBmb3VuZCBmb3IgJHtleHByZXNzaW9uLmdldFRleHQoKX0uYCk7XHJcbi8vICAgICB9XHJcblxyXG4vLyAgICAgcmV0dXJuIGludGVyZmFjZURlY2xhcmF0aW9uO1xyXG4vLyB9XHJcbiJdfQ==