ts2famix 1.4.1 → 2.0.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 (210) hide show
  1. package/LICENSE +1 -0
  2. package/README.md +30 -61
  3. package/dist/analyze.js +4 -2
  4. package/dist/analyze_functions/process_functions.js +285 -131
  5. package/dist/famix_functions/EntityDictionary.js +864 -231
  6. package/dist/famix_functions/helpers_creation.js +61 -10
  7. package/dist/fqn.js +160 -111
  8. package/dist/lib/famix/famix_JSON_exporter.js +55 -0
  9. package/dist/lib/famix/famix_base_element.js +18 -0
  10. package/dist/lib/famix/famix_repository.js +224 -0
  11. package/dist/lib/famix/{src/index.js → index.js} +1 -0
  12. package/dist/lib/famix/model/famix/access.js +40 -0
  13. package/dist/lib/famix/model/famix/accessor.js +17 -0
  14. package/dist/lib/famix/model/famix/alias.js +33 -0
  15. package/dist/lib/famix/model/famix/arrow_function.js +17 -0
  16. package/dist/lib/famix/model/famix/behavioral_entity.js +79 -0
  17. package/dist/lib/famix/model/famix/class.js +71 -0
  18. package/dist/lib/famix/model/famix/comment.js +39 -0
  19. package/dist/lib/famix/model/famix/concretisation.js +31 -0
  20. package/dist/lib/famix/model/famix/container_entity.js +126 -0
  21. package/dist/lib/famix/model/famix/decorator.js +32 -0
  22. package/dist/lib/famix/model/famix/entity.js +17 -0
  23. package/dist/lib/famix/model/famix/enum.js +31 -0
  24. package/dist/lib/famix/model/famix/enum_value.js +25 -0
  25. package/dist/lib/famix/model/famix/function.js +17 -0
  26. package/dist/lib/famix/model/famix/import_clause.js +41 -0
  27. package/dist/lib/famix/model/famix/index.js +86 -0
  28. package/dist/lib/famix/model/famix/indexed_file_anchor.js +38 -0
  29. package/dist/lib/famix/model/famix/inheritance.js +33 -0
  30. package/dist/lib/famix/model/famix/interface.js +64 -0
  31. package/dist/lib/famix/model/famix/invocation.js +54 -0
  32. package/dist/lib/famix/model/famix/method.js +67 -0
  33. package/dist/lib/famix/model/famix/module.js +60 -0
  34. package/dist/lib/famix/model/famix/named_entity.js +78 -0
  35. package/dist/lib/famix/model/famix/parameter.js +25 -0
  36. package/dist/lib/famix/model/famix/parameter_concretisation.js +44 -0
  37. package/dist/lib/famix/model/famix/parameter_type.js +45 -0
  38. package/dist/lib/famix/model/famix/parametric_arrow_function.js +31 -0
  39. package/dist/lib/famix/model/famix/parametric_class.js +44 -0
  40. package/dist/lib/famix/model/famix/parametric_function.js +31 -0
  41. package/dist/lib/famix/model/famix/parametric_interface.js +44 -0
  42. package/dist/lib/famix/model/famix/parametric_method.js +31 -0
  43. package/dist/lib/famix/model/famix/primitive_type.js +17 -0
  44. package/dist/lib/famix/model/famix/property.js +73 -0
  45. package/dist/lib/famix/model/famix/reference.js +33 -0
  46. package/dist/lib/famix/model/famix/scoping_entity.js +36 -0
  47. package/dist/lib/famix/model/famix/script_entity.js +29 -0
  48. package/dist/lib/famix/model/famix/source_anchor.js +27 -0
  49. package/dist/lib/famix/model/famix/source_language.js +35 -0
  50. package/dist/lib/famix/model/famix/sourced_entity.js +60 -0
  51. package/dist/lib/famix/model/famix/structural_entity.js +39 -0
  52. package/dist/lib/famix/model/famix/type.js +73 -0
  53. package/dist/lib/famix/model/famix/variable.js +24 -0
  54. package/dist/lib/ts-complex/cyclomatic-service.js +3 -3
  55. package/dist/refactorer/refactor-getter-setter.js +142 -0
  56. package/dist/ts2famix-cli-wrapper.js +42 -0
  57. package/dist/ts2famix-cli.js +8 -1
  58. package/dist/ts2famix-tsconfig.js +1 -0
  59. package/doc-uml/famix-typescript-model.puml +608 -0
  60. package/doc-uml/famix-typescript-model.svg +1 -0
  61. package/jest.config.json +2 -1
  62. package/package.json +13 -12
  63. package/src/analyze.ts +24 -23
  64. package/src/analyze_functions/process_functions.ts +310 -129
  65. package/src/famix_functions/EntityDictionary.ts +949 -271
  66. package/src/famix_functions/helpers_creation.ts +64 -6
  67. package/src/fqn.ts +169 -96
  68. package/{dist/lib/famix/src/famix_JSON_exporter.js → src/lib/famix/famix_JSON_exporter.ts} +16 -14
  69. package/src/lib/famix/famix_base_element.ts +22 -0
  70. package/{dist/lib/famix/src/famix_repository.js → src/lib/famix/famix_repository.ts} +96 -75
  71. package/src/lib/famix/model/famix/access.ts +50 -0
  72. package/src/lib/famix/model/famix/alias.ts +39 -0
  73. package/src/lib/famix/{src/model/famix/implicit_variable.ts → model/famix/arrow_function.ts} +3 -3
  74. package/src/lib/famix/model/famix/behavioral_entity.ts +97 -0
  75. package/src/lib/famix/model/famix/class.ts +85 -0
  76. package/src/lib/famix/model/famix/comment.ts +47 -0
  77. package/src/lib/famix/model/famix/concretisation.ts +40 -0
  78. package/src/lib/famix/model/famix/container_entity.ts +160 -0
  79. package/src/lib/famix/model/famix/decorator.ts +37 -0
  80. package/src/lib/famix/model/famix/enum.ts +30 -0
  81. package/src/lib/famix/model/famix/enum_value.ts +28 -0
  82. package/src/lib/famix/model/famix/import_clause.ts +51 -0
  83. package/src/lib/famix/{src/model → model}/famix/index.ts +8 -7
  84. package/src/lib/famix/model/famix/indexed_file_anchor.ts +46 -0
  85. package/src/lib/famix/model/famix/inheritance.ts +40 -0
  86. package/src/lib/famix/model/famix/interface.ts +75 -0
  87. package/src/lib/famix/model/famix/invocation.ts +65 -0
  88. package/src/lib/famix/model/famix/method.ts +89 -0
  89. package/src/lib/famix/model/famix/module.ts +71 -0
  90. package/src/lib/famix/model/famix/named_entity.ts +95 -0
  91. package/src/lib/famix/{src/model → model}/famix/parameter.ts +11 -12
  92. package/src/lib/famix/model/famix/parameter_concretisation.ts +51 -0
  93. package/src/lib/famix/model/famix/parameter_type.ts +58 -0
  94. package/src/lib/famix/model/famix/parametric_arrow_function.ts +32 -0
  95. package/src/lib/famix/model/famix/parametric_class.ts +49 -0
  96. package/src/lib/famix/model/famix/parametric_function.ts +32 -0
  97. package/src/lib/famix/model/famix/parametric_interface.ts +49 -0
  98. package/src/lib/famix/model/famix/parametric_method.ts +32 -0
  99. package/src/lib/famix/model/famix/primitive_type.ts +15 -0
  100. package/src/lib/famix/model/famix/property.ts +94 -0
  101. package/src/lib/famix/model/famix/reference.ts +40 -0
  102. package/src/lib/famix/model/famix/scoping_entity.ts +35 -0
  103. package/src/lib/famix/model/famix/script_entity.ts +34 -0
  104. package/src/lib/famix/model/famix/source_anchor.ts +30 -0
  105. package/src/lib/famix/model/famix/source_language.ts +35 -0
  106. package/src/lib/famix/model/famix/sourced_entity.ts +70 -0
  107. package/src/lib/famix/model/famix/structural_entity.ts +43 -0
  108. package/src/lib/famix/model/famix/type.ts +87 -0
  109. package/src/lib/famix/model/famix/variable.ts +27 -0
  110. package/src/lib/famix/package.json +1 -1
  111. package/src/lib/ts-complex/cyclomatic-service.ts +10 -10
  112. package/src/refactorer/refactor-getter-setter.ts +140 -0
  113. package/src/ts2famix-cli-wrapper.ts +21 -0
  114. package/src/ts2famix-cli.ts +8 -2
  115. package/tsconfig.check-tests.json +14 -0
  116. package/tsconfig.json +71 -69
  117. package/dist/famix2puml.js +0 -125
  118. package/dist/lib/famix/src/famix_base_element.js +0 -17
  119. package/dist/lib/famix/src/model/famix/access.js +0 -39
  120. package/dist/lib/famix/src/model/famix/accessor.js +0 -16
  121. package/dist/lib/famix/src/model/famix/alias.js +0 -32
  122. package/dist/lib/famix/src/model/famix/association.js +0 -36
  123. package/dist/lib/famix/src/model/famix/behavioral_entity.js +0 -81
  124. package/dist/lib/famix/src/model/famix/class.js +0 -70
  125. package/dist/lib/famix/src/model/famix/comment.js +0 -38
  126. package/dist/lib/famix/src/model/famix/container_entity.js +0 -125
  127. package/dist/lib/famix/src/model/famix/decorator.js +0 -31
  128. package/dist/lib/famix/src/model/famix/entity.js +0 -16
  129. package/dist/lib/famix/src/model/famix/enum.js +0 -30
  130. package/dist/lib/famix/src/model/famix/enum_value.js +0 -24
  131. package/dist/lib/famix/src/model/famix/function.js +0 -16
  132. package/dist/lib/famix/src/model/famix/implicit_variable.js +0 -16
  133. package/dist/lib/famix/src/model/famix/import_clause.js +0 -39
  134. package/dist/lib/famix/src/model/famix/index.js +0 -83
  135. package/dist/lib/famix/src/model/famix/indexed_file_anchor.js +0 -51
  136. package/dist/lib/famix/src/model/famix/inheritance.js +0 -32
  137. package/dist/lib/famix/src/model/famix/interface.js +0 -63
  138. package/dist/lib/famix/src/model/famix/invocation.js +0 -53
  139. package/dist/lib/famix/src/model/famix/method.js +0 -66
  140. package/dist/lib/famix/src/model/famix/module.js +0 -31
  141. package/dist/lib/famix/src/model/famix/named_entity.js +0 -77
  142. package/dist/lib/famix/src/model/famix/namespace.js +0 -24
  143. package/dist/lib/famix/src/model/famix/parameter.js +0 -24
  144. package/dist/lib/famix/src/model/famix/parameter_type.js +0 -24
  145. package/dist/lib/famix/src/model/famix/parameterizable_class.js +0 -30
  146. package/dist/lib/famix/src/model/famix/parameterizable_interface.js +0 -30
  147. package/dist/lib/famix/src/model/famix/parameterized_type.js +0 -36
  148. package/dist/lib/famix/src/model/famix/primitive_type.js +0 -16
  149. package/dist/lib/famix/src/model/famix/property.js +0 -44
  150. package/dist/lib/famix/src/model/famix/reference.js +0 -32
  151. package/dist/lib/famix/src/model/famix/scoping_entity.js +0 -35
  152. package/dist/lib/famix/src/model/famix/script_entity.js +0 -30
  153. package/dist/lib/famix/src/model/famix/source_anchor.js +0 -26
  154. package/dist/lib/famix/src/model/famix/source_language.js +0 -35
  155. package/dist/lib/famix/src/model/famix/sourced_entity.js +0 -59
  156. package/dist/lib/famix/src/model/famix/structural_entity.js +0 -38
  157. package/dist/lib/famix/src/model/famix/text_anchor.js +0 -37
  158. package/dist/lib/famix/src/model/famix/type.js +0 -71
  159. package/dist/lib/famix/src/model/famix/variable.js +0 -23
  160. package/doc-uml/metamodel-full.svg +0 -1
  161. package/doc-uml/metamodel.svg +0 -1
  162. package/jest.config-old.ts +0 -199
  163. package/plantuml.jar +0 -0
  164. package/src/famix2puml.ts +0 -119
  165. package/src/lib/famix/package-lock.json +0 -301
  166. package/src/lib/famix/readme.md +0 -5
  167. package/src/lib/famix/src/famix_JSON_exporter.ts +0 -56
  168. package/src/lib/famix/src/famix_base_element.ts +0 -22
  169. package/src/lib/famix/src/famix_repository.ts +0 -243
  170. package/src/lib/famix/src/model/famix/access.ts +0 -53
  171. package/src/lib/famix/src/model/famix/alias.ts +0 -41
  172. package/src/lib/famix/src/model/famix/association.ts +0 -44
  173. package/src/lib/famix/src/model/famix/behavioral_entity.ts +0 -107
  174. package/src/lib/famix/src/model/famix/class.ts +0 -86
  175. package/src/lib/famix/src/model/famix/comment.ts +0 -50
  176. package/src/lib/famix/src/model/famix/container_entity.ts +0 -165
  177. package/src/lib/famix/src/model/famix/decorator.ts +0 -39
  178. package/src/lib/famix/src/model/famix/enum.ts +0 -31
  179. package/src/lib/famix/src/model/famix/enum_value.ts +0 -29
  180. package/src/lib/famix/src/model/famix/import_clause.ts +0 -53
  181. package/src/lib/famix/src/model/famix/indexed_file_anchor.ts +0 -71
  182. package/src/lib/famix/src/model/famix/inheritance.ts +0 -42
  183. package/src/lib/famix/src/model/famix/interface.ts +0 -75
  184. package/src/lib/famix/src/model/famix/invocation.ts +0 -68
  185. package/src/lib/famix/src/model/famix/method.ts +0 -96
  186. package/src/lib/famix/src/model/famix/module.ts +0 -31
  187. package/src/lib/famix/src/model/famix/named_entity.ts +0 -98
  188. package/src/lib/famix/src/model/famix/namespace.ts +0 -28
  189. package/src/lib/famix/src/model/famix/parameter_type.ts +0 -33
  190. package/src/lib/famix/src/model/famix/parameterizable_class.ts +0 -31
  191. package/src/lib/famix/src/model/famix/parameterizable_interface.ts +0 -31
  192. package/src/lib/famix/src/model/famix/parameterized_type.ts +0 -40
  193. package/src/lib/famix/src/model/famix/primitive_type.ts +0 -15
  194. package/src/lib/famix/src/model/famix/property.ts +0 -54
  195. package/src/lib/famix/src/model/famix/reference.ts +0 -42
  196. package/src/lib/famix/src/model/famix/scoping_entity.ts +0 -35
  197. package/src/lib/famix/src/model/famix/script_entity.ts +0 -38
  198. package/src/lib/famix/src/model/famix/source_anchor.ts +0 -31
  199. package/src/lib/famix/src/model/famix/source_language.ts +0 -37
  200. package/src/lib/famix/src/model/famix/sourced_entity.ts +0 -73
  201. package/src/lib/famix/src/model/famix/structural_entity.ts +0 -44
  202. package/src/lib/famix/src/model/famix/text_anchor.ts +0 -49
  203. package/src/lib/famix/src/model/famix/type.ts +0 -88
  204. package/src/lib/famix/src/model/famix/variable.ts +0 -28
  205. package/src/lib/famix/tsconfig.json +0 -27
  206. package/src/lib/famix/tslint.json +0 -15
  207. /package/src/lib/famix/{src/index.ts → index.ts} +0 -0
  208. /package/src/lib/famix/{src/model → model}/famix/accessor.ts +0 -0
  209. /package/src/lib/famix/{src/model → model}/famix/entity.ts +0 -0
  210. /package/src/lib/famix/{src/model → model}/famix/function.ts +0 -0
@@ -23,47 +23,58 @@ var __importStar = (this && this.__importStar) || function (mod) {
23
23
  return result;
24
24
  };
25
25
  Object.defineProperty(exports, "__esModule", { value: true });
26
- exports.processInvocations = exports.processInheritances = exports.processImportClausesForModules = exports.processImportClausesForImportEqualsDeclarations = exports.processAccesses = exports.processFiles = exports.getImplementedOrExtendedInterfaces = exports.getModulePath = exports.currentCC = exports.exportedMap = exports.modules = exports.interfaces = exports.classes = exports.accessMap = exports.methodsAndFunctionsWithId = void 0;
26
+ exports.currentCC = exports.listOfExportMaps = exports.modules = exports.interfaces = exports.classes = exports.accessMap = exports.methodsAndFunctionsWithId = void 0;
27
+ exports.getModulePath = getModulePath;
28
+ exports.getImplementedOrExtendedInterfaces = getImplementedOrExtendedInterfaces;
29
+ exports.processFiles = processFiles;
30
+ exports.isAmbient = isAmbient;
31
+ exports.isNamespace = isNamespace;
32
+ exports.processAccesses = processAccesses;
33
+ exports.processImportClausesForImportEqualsDeclarations = processImportClausesForImportEqualsDeclarations;
34
+ exports.processImportClausesForModules = processImportClausesForModules;
35
+ exports.processInheritances = processInheritances;
36
+ exports.processInvocations = processInvocations;
37
+ exports.processConcretisations = processConcretisations;
27
38
  const ts_morph_1 = require("ts-morph");
28
- const Famix = __importStar(require("../lib/famix/src/model/famix"));
39
+ const Famix = __importStar(require("../lib/famix/model/famix"));
29
40
  const cyclomatic_service_1 = require("../lib/ts-complex/cyclomatic-service");
30
41
  const fs = __importStar(require("fs"));
31
42
  const analyze_1 = require("../analyze");
43
+ const fqn_1 = require("../fqn");
32
44
  exports.methodsAndFunctionsWithId = new Map(); // Maps the Famix method, constructor, getter, setter and function ids to their ts-morph method, constructor, getter, setter or function object
33
45
  exports.accessMap = new Map(); // Maps the Famix parameter, variable, property and enum value ids to their ts-morph parameter, variable, property or enum member object
34
46
  exports.classes = new Array(); // Array of all the classes of the source files
35
47
  exports.interfaces = new Array(); // Array of all the interfaces of the source files
36
48
  exports.modules = new Array(); // Array of all the source files which are modules
37
- exports.exportedMap = new Array(); // Array of all the exports
49
+ exports.listOfExportMaps = new Array(); // Array of all the export maps
38
50
  /**
39
51
  * Checks if the file has any imports or exports to be considered a module
40
52
  * @param sourceFile A source file
41
53
  * @returns A boolean indicating if the file is a module
42
54
  */
43
- function ismodule(sourceFile) {
55
+ function isSourceFileAModule(sourceFile) {
44
56
  return sourceFile.getImportDeclarations().length > 0 || sourceFile.getExportedDeclarations().size > 0;
45
57
  }
46
58
  /**
47
59
  * Gets the path of a module to be imported
48
- * @param i An import declaration
60
+ * @param importDecl An import declaration
49
61
  * @returns The path of the module to be imported
50
62
  */
51
- function getModulePath(i) {
63
+ function getModulePath(importDecl) {
52
64
  let path;
53
- if (i.getModuleSpecifierSourceFile() === undefined) {
54
- if (i.getModuleSpecifierValue().substring(i.getModuleSpecifierValue().length - 3) === ".ts") {
55
- path = i.getModuleSpecifierValue();
65
+ if (importDecl.getModuleSpecifierSourceFile() === undefined) {
66
+ if (importDecl.getModuleSpecifierValue().substring(importDecl.getModuleSpecifierValue().length - 3) === ".ts") {
67
+ path = importDecl.getModuleSpecifierValue();
56
68
  }
57
69
  else {
58
- path = i.getModuleSpecifierValue() + ".ts";
70
+ path = importDecl.getModuleSpecifierValue() + ".ts";
59
71
  }
60
72
  }
61
73
  else {
62
- path = i.getModuleSpecifierSourceFile().getFilePath();
74
+ path = importDecl.getModuleSpecifierSourceFile().getFilePath();
63
75
  }
64
76
  return path;
65
77
  }
66
- exports.getModulePath = getModulePath;
67
78
  /**
68
79
  * Gets the interfaces implemented or extended by a class or an interface
69
80
  * @param interfaces An array of interfaces
@@ -90,7 +101,6 @@ function getImplementedOrExtendedInterfaces(interfaces, subClass) {
90
101
  });
91
102
  return implementedOrExtendedInterfaces;
92
103
  }
93
- exports.getImplementedOrExtendedInterfaces = getImplementedOrExtendedInterfaces;
94
104
  /**
95
105
  * Builds a Famix model for an array of source files
96
106
  * @param sourceFiles An array of source files
@@ -102,23 +112,24 @@ function processFiles(sourceFiles) {
102
112
  if (fs.existsSync(file.getFilePath()))
103
113
  exports.currentCC = (0, cyclomatic_service_1.calculate)(file.getFilePath());
104
114
  else
105
- exports.currentCC = 0;
115
+ exports.currentCC = {};
106
116
  processFile(file);
107
117
  });
108
118
  }
109
- exports.processFiles = processFiles;
110
119
  /**
111
120
  * Builds a Famix model for a source file
112
121
  * @param f A source file
113
122
  */
114
123
  function processFile(f) {
115
- const isModule = ismodule(f);
124
+ const isModule = isSourceFileAModule(f);
116
125
  if (isModule) {
117
126
  exports.modules.push(f);
118
- exports.exportedMap.push(f.getExportedDeclarations());
119
127
  }
128
+ const exportMap = f.getExportedDeclarations();
129
+ if (exportMap)
130
+ exports.listOfExportMaps.push(exportMap);
120
131
  const fmxFile = analyze_1.entityDictionary.createOrGetFamixFile(f, isModule);
121
- analyze_1.logger.debug(`processFile: file: ${f.getBaseName()}, fqn = ${fmxFile.getFullyQualifiedName()}`);
132
+ analyze_1.logger.debug(`processFile: file: ${f.getBaseName()}, fqn = ${fmxFile.fullyQualifiedName}`);
122
133
  processComments(f, fmxFile);
123
134
  processAliases(f, fmxFile);
124
135
  processClasses(f, fmxFile);
@@ -126,25 +137,34 @@ function processFile(f) {
126
137
  processVariables(f, fmxFile);
127
138
  processEnums(f, fmxFile);
128
139
  processFunctions(f, fmxFile);
129
- processNamespaces(f, fmxFile);
140
+ processModules(f, fmxFile);
141
+ }
142
+ function isAmbient(node) {
143
+ // An ambient module has the DeclareKeyword modifier.
144
+ return (node.getModifiers()?.some(modifier => modifier.getKind() === ts_morph_1.SyntaxKind.DeclareKeyword)) ?? false;
145
+ }
146
+ function isNamespace(node) {
147
+ // Check if the module declaration has a namespace keyword.
148
+ // This approach uses the getChildren() method to inspect the syntax directly.
149
+ return node.getChildrenOfKind(ts_morph_1.SyntaxKind.NamespaceKeyword).length > 0;
130
150
  }
131
151
  /**
132
- * Builds a Famix model for a namespace
152
+ * Builds a Famix model for a module (also namespace)
133
153
  * @param m A namespace
134
- * @returns A Famix.Namespace representing the namespace
135
- */
136
- function processNamespace(m) {
137
- const fmxNamespace = analyze_1.entityDictionary.createOrGetFamixNamespace(m);
138
- analyze_1.logger.debug(`processNamespace: namespace: ${m.getName()}, (${m.getType().getText()}), ${fmxNamespace.getFullyQualifiedName()}`);
139
- processComments(m, fmxNamespace);
140
- processAliases(m, fmxNamespace);
141
- processClasses(m, fmxNamespace);
142
- processInterfaces(m, fmxNamespace);
143
- processVariables(m, fmxNamespace);
144
- processEnums(m, fmxNamespace);
145
- processFunctions(m, fmxNamespace);
146
- processNamespaces(m, fmxNamespace);
147
- return fmxNamespace;
154
+ * @returns A Famix.Module representing the module
155
+ */
156
+ function processModule(m) {
157
+ const fmxModule = analyze_1.entityDictionary.createOrGetFamixModule(m);
158
+ analyze_1.logger.debug(`module: ${m.getName()}, (${m.getType().getText()}), ${fmxModule.fullyQualifiedName}`);
159
+ processComments(m, fmxModule);
160
+ processAliases(m, fmxModule);
161
+ processClasses(m, fmxModule);
162
+ processInterfaces(m, fmxModule);
163
+ processVariables(m, fmxModule);
164
+ processEnums(m, fmxModule);
165
+ processFunctions(m, fmxModule);
166
+ processModules(m, fmxModule);
167
+ return fmxModule;
148
168
  }
149
169
  /**
150
170
  * Builds a Famix model for the aliases of a container
@@ -165,11 +185,34 @@ function processAliases(m, fmxScope) {
165
185
  */
166
186
  function processClasses(m, fmxScope) {
167
187
  analyze_1.logger.debug(`processClasses: ---------- Finding Classes:`);
168
- m.getClasses().forEach(c => {
188
+ const classesInArrowFunctions = getClassesDeclaredInArrowFunctions(m);
189
+ const classes = m.getClasses().concat(classesInArrowFunctions);
190
+ classes.forEach(c => {
169
191
  const fmxClass = processClass(c);
170
192
  fmxScope.addType(fmxClass);
171
193
  });
172
194
  }
195
+ function getArrowFunctionClasses(f) {
196
+ const classes = [];
197
+ function findClasses(node) {
198
+ if (node.getKind() === ts_morph_1.SyntaxKind.ClassDeclaration) {
199
+ classes.push(node);
200
+ }
201
+ node.getChildren().forEach(findClasses);
202
+ }
203
+ findClasses(f);
204
+ return classes;
205
+ }
206
+ /**
207
+ * ts-morph doesn't find classes in arrow functions, so we need to find them manually
208
+ * @param s A source file
209
+ * @returns the ClassDeclaration objects found in arrow functions of the source file
210
+ */
211
+ function getClassesDeclaredInArrowFunctions(s) {
212
+ const arrowFunctions = s.getDescendantsOfKind(ts_morph_1.SyntaxKind.ArrowFunction);
213
+ const classesInArrowFunctions = arrowFunctions.map(f => getArrowFunctionClasses(f)).flat();
214
+ return classesInArrowFunctions;
215
+ }
173
216
  /**
174
217
  * Builds a Famix model for the interfaces of a container
175
218
  * @param m A container (a source file or a namespace)
@@ -219,17 +262,24 @@ function processFunctions(m, fmxScope) {
219
262
  const fmxFunction = processFunction(f);
220
263
  fmxScope.addFunction(fmxFunction);
221
264
  });
265
+ //find arrow functions
266
+ analyze_1.logger.debug(`Finding Functions:`);
267
+ const arrowFunctions = m.getDescendantsOfKind(ts_morph_1.SyntaxKind.ArrowFunction);
268
+ arrowFunctions.forEach(af => {
269
+ const fmxFunction = processFunction(af);
270
+ fmxScope.addFunction(fmxFunction);
271
+ });
222
272
  }
223
273
  /**
224
- * Builds a Famix model for the namespaces of a container
274
+ * Builds a Famix model for the modules of a container.
225
275
  * @param m A container (a source file or a namespace)
226
276
  * @param fmxScope The Famix model of the container
227
277
  */
228
- function processNamespaces(m, fmxScope) {
229
- analyze_1.logger.debug(`Finding Namespaces:`);
278
+ function processModules(m, fmxScope) {
279
+ analyze_1.logger.debug(`Finding Modules:`);
230
280
  m.getModules().forEach(md => {
231
- const fmxNsp = processNamespace(md);
232
- fmxScope.addNamespace(fmxNsp);
281
+ const fmxModule = processModule(md);
282
+ fmxScope.addModule(fmxModule);
233
283
  });
234
284
  }
235
285
  /**
@@ -239,19 +289,19 @@ function processNamespaces(m, fmxScope) {
239
289
  */
240
290
  function processAlias(a) {
241
291
  const fmxAlias = analyze_1.entityDictionary.createFamixAlias(a);
242
- analyze_1.logger.debug(`Alias: ${a.getName()}, (${a.getType().getText()}), fqn = ${fmxAlias.getFullyQualifiedName()}`);
292
+ analyze_1.logger.debug(`Alias: ${a.getName()}, (${a.getType().getText()}), fqn = ${fmxAlias.fullyQualifiedName}`);
243
293
  processComments(a, fmxAlias);
244
294
  return fmxAlias;
245
295
  }
246
296
  /**
247
297
  * Builds a Famix model for a class
248
298
  * @param c A class
249
- * @returns A Famix.Class or a Famix.ParameterizableClass representing the class
299
+ * @returns A Famix.Class or a Famix.ParametricClass representing the class
250
300
  */
251
301
  function processClass(c) {
252
302
  exports.classes.push(c);
253
303
  const fmxClass = analyze_1.entityDictionary.createOrGetFamixClass(c);
254
- analyze_1.logger.debug(`Class: ${c.getName()}, (${c.getType().getText()}), fqn = ${fmxClass.getFullyQualifiedName()}`);
304
+ analyze_1.logger.debug(`Class: ${c.getName()}, (${c.getType().getText()}), fqn = ${fmxClass.fullyQualifiedName}`);
255
305
  processComments(c, fmxClass);
256
306
  processDecorators(c, fmxClass);
257
307
  processStructuredType(c, fmxClass);
@@ -272,12 +322,12 @@ function processClass(c) {
272
322
  /**
273
323
  * Builds a Famix model for an interface
274
324
  * @param i An interface
275
- * @returns A Famix.Interface or a Famix.ParameterizableInterface representing the interface
325
+ * @returns A Famix.Interface or a Famix.ParametricInterface representing the interface
276
326
  */
277
327
  function processInterface(i) {
278
328
  exports.interfaces.push(i);
279
329
  const fmxInterface = analyze_1.entityDictionary.createOrGetFamixInterface(i);
280
- analyze_1.logger.debug(`Interface: ${i.getName()}, (${i.getType().getText()}), fqn = ${fmxInterface.getFullyQualifiedName()}`);
330
+ analyze_1.logger.debug(`Interface: ${i.getName()}, (${i.getType().getText()}), fqn = ${fmxInterface.fullyQualifiedName}`);
281
331
  processComments(i, fmxInterface);
282
332
  processStructuredType(i, fmxInterface);
283
333
  return fmxInterface;
@@ -289,7 +339,7 @@ function processInterface(i) {
289
339
  */
290
340
  function processStructuredType(c, fmxScope) {
291
341
  analyze_1.logger.debug(`Finding Properties and Methods:`);
292
- if (fmxScope instanceof Famix.ParameterizableClass || fmxScope instanceof Famix.ParameterizableInterface) {
342
+ if (fmxScope instanceof Famix.ParametricClass || fmxScope instanceof Famix.ParametricInterface) {
293
343
  processTypeParameters(c, fmxScope);
294
344
  }
295
345
  c.getProperties().forEach(prop => {
@@ -308,15 +358,16 @@ function processStructuredType(c, fmxScope) {
308
358
  */
309
359
  function processProperty(p) {
310
360
  const fmxProperty = analyze_1.entityDictionary.createFamixProperty(p);
311
- analyze_1.logger.debug(`property: ${p.getName()}, (${p.getType().getText()}), fqn = ${fmxProperty.getFullyQualifiedName()}`);
361
+ analyze_1.logger.debug(`property: ${p.getName()}, (${p.getType().getText()}), fqn = ${fmxProperty.fullyQualifiedName}`);
312
362
  analyze_1.logger.debug(` ---> It's a Property${(p instanceof ts_morph_1.PropertySignature) ? "Signature" : "Declaration"}!`);
313
363
  const ancestor = p.getFirstAncestorOrThrow();
314
364
  analyze_1.logger.debug(` ---> Its first ancestor is a ${ancestor.getKindName()}`);
365
+ // decorators
315
366
  if (!(p instanceof ts_morph_1.PropertySignature)) {
316
367
  processDecorators(p, fmxProperty);
317
368
  // only add access if the p's first ancestor is not a PropertyDeclaration
318
369
  if (ancestor.getKindName() !== "PropertyDeclaration") {
319
- analyze_1.logger.debug(`adding access: ${p.getName()}, (${p.getType().getText()}) Famix ${fmxProperty.getName()}`);
370
+ analyze_1.logger.debug(`adding access to map: ${p.getName()}, (${p.getType().getText()}) Famix ${fmxProperty.name} id: ${fmxProperty.id}`);
320
371
  exports.accessMap.set(fmxProperty.id, p);
321
372
  }
322
373
  }
@@ -329,8 +380,8 @@ function processProperty(p) {
329
380
  * @returns A Famix.Method or a Famix.Accessor representing the method or the accessor
330
381
  */
331
382
  function processMethod(m) {
332
- const fmxMethod = analyze_1.entityDictionary.createFamixMethod(m, exports.currentCC);
333
- analyze_1.logger.debug(`Method: ${!(m instanceof ts_morph_1.ConstructorDeclaration) ? m.getName() : "constructor"}, (${m.getType().getText()}), parent: ${m.getParent().getName()}, fqn = ${fmxMethod.getFullyQualifiedName()}`);
383
+ const fmxMethod = analyze_1.entityDictionary.createOrGetFamixMethod(m, exports.currentCC);
384
+ analyze_1.logger.debug(`Method: ${!(m instanceof ts_morph_1.ConstructorDeclaration) ? m.getName() : "constructor"}, (${m.getType().getText()}), parent: ${m.getParent().getName()}, fqn = ${fmxMethod.fullyQualifiedName}`);
334
385
  processComments(m, fmxMethod);
335
386
  processTypeParameters(m, fmxMethod);
336
387
  processParameters(m, fmxMethod);
@@ -353,8 +404,14 @@ function processMethod(m) {
353
404
  * @returns A Famix.Function representing the function
354
405
  */
355
406
  function processFunction(f) {
356
- const fmxFunction = analyze_1.entityDictionary.createFamixFunction(f, exports.currentCC);
357
- analyze_1.logger.debug(`Function: ${(f.getName()) ? f.getName() : "anonymous"}, (${f.getType().getText()}), fqn = ${fmxFunction.getFullyQualifiedName()}`);
407
+ analyze_1.logger.debug(`Function: ${(f instanceof ts_morph_1.ArrowFunction ? "anonymous" : f.getName() ? f.getName() : "anonymous")}, (${f.getType().getText()}), fqn = ${(0, fqn_1.getFQN)(f)}`);
408
+ let fmxFunction;
409
+ if (f instanceof ts_morph_1.ArrowFunction) {
410
+ fmxFunction = analyze_1.entityDictionary.createFamixArrowFunction(f, exports.currentCC);
411
+ }
412
+ else {
413
+ fmxFunction = analyze_1.entityDictionary.createOrGetFamixFunction(f, exports.currentCC);
414
+ }
358
415
  processComments(f, fmxFunction);
359
416
  processAliases(f, fmxFunction);
360
417
  processTypeParameters(f, fmxFunction);
@@ -391,22 +448,87 @@ function processParameters(m, fmxScope) {
391
448
  m.getParameters().forEach(param => {
392
449
  const fmxParam = processParameter(param);
393
450
  fmxScope.addParameter(fmxParam);
451
+ // Additional handling for Parameter Properties in constructors
452
+ if (m instanceof ts_morph_1.ConstructorDeclaration) {
453
+ // Check if the parameter has any visibility modifier
454
+ if (param.hasModifier(ts_morph_1.SyntaxKind.PrivateKeyword) || param.hasModifier(ts_morph_1.SyntaxKind.PublicKeyword) || param.hasModifier(ts_morph_1.SyntaxKind.ProtectedKeyword) || param.hasModifier(ts_morph_1.SyntaxKind.ReadonlyKeyword)) {
455
+ const classOfConstructor = m.getParent();
456
+ analyze_1.logger.info(`Parameter Property ${param.getName()} in constructor of ${classOfConstructor.getName()}.`);
457
+ // Treat the parameter as a property and add it to the class
458
+ const fmxProperty = processParameterAsProperty(param, classOfConstructor);
459
+ fmxProperty.readOnly = param.hasModifier(ts_morph_1.SyntaxKind.ReadonlyKeyword);
460
+ }
461
+ }
394
462
  });
395
463
  }
464
+ // This function should create a Famix.Property model from a ParameterDeclaration
465
+ // You'll need to implement it according to your Famix model structure
466
+ function processParameterAsProperty(param, classDecl) {
467
+ // Convert the parameter into a Property
468
+ const propertyRepresentation = convertParameterToPropertyRepresentation(param);
469
+ // Add the property to the class so we can have a PropertyDeclaration object
470
+ classDecl.addProperty(propertyRepresentation);
471
+ const property = classDecl.getProperty(propertyRepresentation.name);
472
+ if (!property) {
473
+ throw new Error(`Property ${propertyRepresentation.name} not found in class ${classDecl.getName()}`);
474
+ }
475
+ const fmxProperty = analyze_1.entityDictionary.createFamixProperty(property);
476
+ if (classDecl instanceof ts_morph_1.ClassDeclaration) {
477
+ const fmxClass = analyze_1.entityDictionary.createOrGetFamixClass(classDecl);
478
+ fmxClass.addProperty(fmxProperty);
479
+ }
480
+ else {
481
+ throw new Error("Unexpected type ClassExpression.");
482
+ }
483
+ processComments(property, fmxProperty);
484
+ // remove the property from the class
485
+ property.remove();
486
+ return fmxProperty;
487
+ }
488
+ function convertParameterToPropertyRepresentation(param) {
489
+ // Extract name
490
+ const paramName = param.getName();
491
+ // Extract type
492
+ const paramType = param.getType().getText(param);
493
+ // Determine visibility
494
+ let scope;
495
+ if (param.hasModifier(ts_morph_1.SyntaxKind.PrivateKeyword)) {
496
+ scope = ts_morph_1.Scope.Private;
497
+ }
498
+ else if (param.hasModifier(ts_morph_1.SyntaxKind.ProtectedKeyword)) {
499
+ scope = ts_morph_1.Scope.Protected;
500
+ }
501
+ else if (param.hasModifier(ts_morph_1.SyntaxKind.PublicKeyword)) {
502
+ scope = ts_morph_1.Scope.Public;
503
+ }
504
+ else {
505
+ throw new Error(`Parameter property ${paramName} in constructor does not have a visibility modifier.`);
506
+ }
507
+ // Determine if readonly
508
+ const isReadonly = param.hasModifier(ts_morph_1.SyntaxKind.ReadonlyKeyword);
509
+ // Create a representation of the property
510
+ const propertyRepresentation = {
511
+ name: paramName,
512
+ type: paramType,
513
+ scope: scope,
514
+ isReadonly: isReadonly,
515
+ };
516
+ return propertyRepresentation;
517
+ }
396
518
  /**
397
519
  * Builds a Famix model for a parameter
398
- * @param p A parameter
520
+ * @param paramDecl A parameter
399
521
  * @returns A Famix.Parameter representing the parameter
400
522
  */
401
- function processParameter(p) {
402
- const fmxParam = analyze_1.entityDictionary.createFamixParameter(p);
403
- analyze_1.logger.debug(`parameter: ${p.getName()}, (${p.getType().getText()}), fqn = ${fmxParam.getFullyQualifiedName()}`);
404
- processComments(p, fmxParam);
405
- processDecorators(p, fmxParam);
406
- const parent = p.getParent();
523
+ function processParameter(paramDecl) {
524
+ const fmxParam = analyze_1.entityDictionary.createFamixParameter(paramDecl);
525
+ analyze_1.logger.debug(`parameter: ${paramDecl.getName()}, (${paramDecl.getType().getText()}), fqn = ${fmxParam.fullyQualifiedName}`);
526
+ processComments(paramDecl, fmxParam);
527
+ processDecorators(paramDecl, fmxParam);
528
+ const parent = paramDecl.getParent();
407
529
  if (!(parent instanceof ts_morph_1.MethodSignature)) {
408
- analyze_1.logger.debug(`adding access: ${p.getName()}, (${p.getType().getText()}) Famix ${fmxParam.getName()}`);
409
- exports.accessMap.set(fmxParam.id, p);
530
+ analyze_1.logger.debug(`adding access: ${paramDecl.getName()}, (${paramDecl.getType().getText()}) Famix ${fmxParam.name}`);
531
+ exports.accessMap.set(fmxParam.id, paramDecl);
410
532
  }
411
533
  return fmxParam;
412
534
  }
@@ -419,7 +541,7 @@ function processTypeParameters(e, fmxScope) {
419
541
  analyze_1.logger.debug(`Finding Type Parameters:`);
420
542
  e.getTypeParameters().forEach(tp => {
421
543
  const fmxParam = processTypeParameter(tp);
422
- fmxScope.addParameterType(fmxParam);
544
+ fmxScope.addGenericParameter(fmxParam);
423
545
  });
424
546
  }
425
547
  /**
@@ -429,7 +551,7 @@ function processTypeParameters(e, fmxScope) {
429
551
  */
430
552
  function processTypeParameter(tp) {
431
553
  const fmxTypeParameter = analyze_1.entityDictionary.createFamixParameterType(tp);
432
- analyze_1.logger.debug(`type parameter: ${tp.getName()}, (${tp.getType().getText()}), fqn = ${fmxTypeParameter.getFullyQualifiedName()}`);
554
+ analyze_1.logger.debug(`type parameter: ${tp.getName()}, (${tp.getType().getText()}), fqn = ${fmxTypeParameter.fullyQualifiedName}`);
433
555
  processComments(tp, fmxTypeParameter);
434
556
  return fmxTypeParameter;
435
557
  }
@@ -455,9 +577,9 @@ function processVariableStatement(v) {
455
577
  */
456
578
  function processVariable(v) {
457
579
  const fmxVar = analyze_1.entityDictionary.createFamixVariable(v);
458
- analyze_1.logger.debug(`variable: ${v.getName()}, (${v.getType().getText()}), ${v.getInitializer() ? "initializer: " + v.getInitializer().getText() : "initializer: "}, fqn = ${fmxVar.getFullyQualifiedName()}`);
580
+ analyze_1.logger.debug(`variable: ${v.getName()}, (${v.getType().getText()}), ${v.getInitializer() ? "initializer: " + v.getInitializer().getText() : "initializer: "}, fqn = ${fmxVar.fullyQualifiedName}`);
459
581
  processComments(v, fmxVar);
460
- analyze_1.logger.debug(`adding access: ${v.getName()}, (${v.getType().getText()}) Famix ${fmxVar.getName()}`);
582
+ analyze_1.logger.debug(`adding access: ${v.getName()}, (${v.getType().getText()}) Famix ${fmxVar.name}`);
461
583
  exports.accessMap.set(fmxVar.id, v);
462
584
  return fmxVar;
463
585
  }
@@ -468,7 +590,7 @@ function processVariable(v) {
468
590
  */
469
591
  function processEnum(e) {
470
592
  const fmxEnum = analyze_1.entityDictionary.createFamixEnum(e);
471
- analyze_1.logger.debug(`enum: ${e.getName()}, (${e.getType().getText()}), fqn = ${fmxEnum.getFullyQualifiedName()}`);
593
+ analyze_1.logger.debug(`enum: ${e.getName()}, (${e.getType().getText()}), fqn = ${fmxEnum.fullyQualifiedName}`);
472
594
  processComments(e, fmxEnum);
473
595
  e.getMembers().forEach(m => {
474
596
  const fmxEnumValue = processEnumValue(m);
@@ -483,9 +605,9 @@ function processEnum(e) {
483
605
  */
484
606
  function processEnumValue(v) {
485
607
  const fmxEnumValue = analyze_1.entityDictionary.createFamixEnumValue(v);
486
- analyze_1.logger.debug(`enum value: ${v.getName()}, (${v.getType().getText()}), fqn = ${fmxEnumValue.getFullyQualifiedName()}`);
608
+ analyze_1.logger.debug(`enum value: ${v.getName()}, (${v.getType().getText()}), fqn = ${fmxEnumValue.fullyQualifiedName}`);
487
609
  processComments(v, fmxEnumValue);
488
- analyze_1.logger.debug(`adding access: ${v.getName()}, (${v.getType().getText()}) Famix ${fmxEnumValue.getName()}`);
610
+ analyze_1.logger.debug(`adding access: ${v.getName()}, (${v.getType().getText()}) Famix ${fmxEnumValue.name}`);
489
611
  exports.accessMap.set(fmxEnumValue.id, v);
490
612
  return fmxEnumValue;
491
613
  }
@@ -509,7 +631,7 @@ function processDecorators(e, fmxScope) {
509
631
  */
510
632
  function processDecorator(d, e) {
511
633
  const fmxDec = analyze_1.entityDictionary.createOrGetFamixDecorator(d, e);
512
- analyze_1.logger.debug(`decorator: ${d.getName()}, (${d.getType().getText()}), fqn = ${fmxDec.getFullyQualifiedName()}`);
634
+ analyze_1.logger.debug(`decorator: ${d.getName()}, (${d.getType().getText()}), fqn = ${fmxDec.fullyQualifiedName}`);
513
635
  processComments(d, fmxDec);
514
636
  return fmxDec;
515
637
  }
@@ -548,41 +670,40 @@ function processComment(c, fmxScope) {
548
670
  * @param accessMap A map of parameters, variables, properties and enum members with their id
549
671
  */
550
672
  function processAccesses(accessMap) {
551
- analyze_1.logger.debug(`processAccesses: Creating accesses:`);
673
+ analyze_1.logger.debug(`Creating accesses:`);
552
674
  accessMap.forEach((v, id) => {
553
- analyze_1.logger.debug(`processAccesses: Accesses to ${v.getName()}`);
554
- try {
555
- const temp_nodes = v.findReferencesAsNodes();
556
- temp_nodes.forEach(node => processNodeForAccesses(node, id));
557
- }
558
- catch (error) {
559
- analyze_1.logger.error(`> WARNING: got exception ${error}. Continuing...`);
560
- }
675
+ analyze_1.logger.debug(`Accesses to ${v.getName()}`);
676
+ // try {
677
+ const temp_nodes = v.findReferencesAsNodes();
678
+ temp_nodes.forEach(node => processNodeForAccesses(node, id));
679
+ // } catch (error) {
680
+ // logger.error(`> WARNING: got exception "${error}".\nContinuing...`);
681
+ // }
561
682
  });
562
683
  }
563
- exports.processAccesses = processAccesses;
564
684
  /**
565
685
  * Builds a Famix model for an access on a parameter, variable, property or enum member
566
686
  * @param n A node
567
687
  * @param id An id of a parameter, a variable, a property or an enum member
568
688
  */
569
689
  function processNodeForAccesses(n, id) {
570
- try {
571
- // sometimes node's first ancestor is a PropertyDeclaration, which is not an access
572
- // see https://github.com/fuhrmanator/FamixTypeScriptImporter/issues/9
573
- // check for a node whose first ancestor is a property declaration and bail?
574
- // This may be a bug in ts-morph?
575
- if (n.getFirstAncestorOrThrow().getKindName() === "PropertyDeclaration") {
576
- analyze_1.logger.debug(`processNodeForAccesses: node kind: ${n.getKindName()}, ${n.getText()}, (${n.getType().getText()})'s first ancestor is a PropertyDeclaration. Skipping...`);
577
- return;
578
- }
579
- analyze_1.entityDictionary.createFamixAccess(n, id);
580
- analyze_1.logger.debug(`processNodeForAccesses: node kind: ${n.getKindName()}, ${n.getText()}, (${n.getType().getText()})`);
581
- }
582
- catch (error) {
583
- analyze_1.logger.error(`> WARNING: got exception ${error}. ScopeDeclaration invalid for ${n.getSymbol().getFullyQualifiedName()}. Continuing...`);
690
+ // try {
691
+ // sometimes node's first ancestor is a PropertyDeclaration, which is not an access
692
+ // see https://github.com/fuhrmanator/FamixTypeScriptImporter/issues/9
693
+ // check for a node whose first ancestor is a property declaration and bail?
694
+ // This may be a bug in ts-morph?
695
+ if (n.getFirstAncestorOrThrow().getKindName() === "PropertyDeclaration") {
696
+ analyze_1.logger.debug(`processNodeForAccesses: node kind: ${n.getKindName()}, ${n.getText()}, (${n.getType().getText()})'s first ancestor is a PropertyDeclaration. Skipping...`);
697
+ return;
584
698
  }
585
- }
699
+ analyze_1.entityDictionary.createFamixAccess(n, id);
700
+ analyze_1.logger.debug(`processNodeForAccesses: node kind: ${n.getKindName()}, ${n.getText()}, (${n.getType().getText()})`);
701
+ // } catch (error) {
702
+ // logger.error(`> Got exception "${error}".\nScopeDeclaration invalid for "${n.getSymbol().fullyQualifiedName}".\nContinuing...`);
703
+ // }
704
+ }
705
+ // exports has name -> Declaration -- the declaration can be used to find the FamixElement
706
+ // handle `import path = require("path")` for example
586
707
  function processImportClausesForImportEqualsDeclarations(sourceFiles, exports) {
587
708
  analyze_1.logger.info(`Creating import clauses from ImportEqualsDeclarations in source files:`);
588
709
  sourceFiles.forEach(sourceFile => {
@@ -591,41 +712,39 @@ function processImportClausesForImportEqualsDeclarations(sourceFiles, exports) {
591
712
  // You've found an ImportEqualsDeclaration
592
713
  analyze_1.logger.info("Declaration Name:", node.getName());
593
714
  analyze_1.logger.info("Module Reference Text:", node.getModuleReference().getText());
715
+ // what's the name of the imported entity?
716
+ // const importedEntity = node.getName();
594
717
  // create a famix import clause
595
718
  const namedImport = node.getNameNode();
596
- analyze_1.entityDictionary.createFamixImportClause({ importDeclaration: node,
597
- importer: sourceFile,
719
+ analyze_1.entityDictionary.oldCreateFamixImportClause({ importDeclaration: node,
720
+ importerSourceFile: sourceFile,
598
721
  moduleSpecifierFilePath: node.getModuleReference().getText(),
599
722
  importElement: namedImport,
600
723
  isInExports: exports.find(e => e.has(namedImport.getText())) !== undefined,
601
724
  isDefaultExport: false });
725
+ // entityDictionary.createFamixImportClause(importedEntity, importingEntity);
602
726
  }
603
727
  });
604
728
  });
605
729
  }
606
- exports.processImportClausesForImportEqualsDeclarations = processImportClausesForImportEqualsDeclarations;
607
730
  /**
608
- * Builds a Famix model for the import clauses of the source files which are modules
609
- * @param modules An array of modules
610
- * @param exports An array of maps of exported declarations
611
- */
731
+ * Builds a Famix model for the import clauses of the source files which are modules
732
+ * @param modules An array of modules
733
+ * @param exports An array of maps of exported declarations
734
+ */
612
735
  function processImportClausesForModules(modules, exports) {
613
736
  analyze_1.logger.info(`Creating import clauses from ${modules.length} modules:`);
614
737
  modules.forEach(module => {
738
+ const modulePath = module.getFilePath() + module.getBaseName();
615
739
  module.getImportDeclarations().forEach(impDecl => {
616
- analyze_1.logger.debug(`Importing ${impDecl.getModuleSpecifierValue()}`);
740
+ analyze_1.logger.info(`Importing ${impDecl.getModuleSpecifierValue()} in ${modulePath}`);
617
741
  const path = getModulePath(impDecl);
618
742
  impDecl.getNamedImports().forEach(namedImport => {
619
- analyze_1.logger.debug(`Importing (named) ${namedImport.getName()} from ${impDecl.getModuleSpecifierValue()}`);
743
+ analyze_1.logger.info(`Importing (named) ${namedImport.getName()} from ${impDecl.getModuleSpecifierValue()} in ${modulePath}`);
620
744
  const importedEntityName = namedImport.getName();
621
- let importFoundInExports = false;
622
- exports.forEach(e => {
623
- if (e.has(importedEntityName)) {
624
- importFoundInExports = true;
625
- }
626
- });
627
- analyze_1.entityDictionary.createFamixImportClause({ importDeclaration: impDecl,
628
- importer: module,
745
+ let importFoundInExports = isInExports(exports, importedEntityName);
746
+ analyze_1.entityDictionary.oldCreateFamixImportClause({ importDeclaration: impDecl,
747
+ importerSourceFile: module,
629
748
  moduleSpecifierFilePath: path,
630
749
  importElement: namedImport,
631
750
  isInExports: importFoundInExports,
@@ -633,10 +752,10 @@ function processImportClausesForModules(modules, exports) {
633
752
  });
634
753
  const defaultImport = impDecl.getDefaultImport();
635
754
  if (defaultImport !== undefined) {
636
- analyze_1.logger.debug(`Importing (default) ${defaultImport.getText()} from ${impDecl.getModuleSpecifierValue()}`);
755
+ analyze_1.logger.info(`Importing (default) ${defaultImport.getText()} from ${impDecl.getModuleSpecifierValue()} in ${modulePath}`);
637
756
  // call with module, impDecl.getModuleSpecifierValue(), path, defaultImport, false, true
638
- analyze_1.entityDictionary.createFamixImportClause({ importDeclaration: impDecl,
639
- importer: module,
757
+ analyze_1.entityDictionary.oldCreateFamixImportClause({ importDeclaration: impDecl,
758
+ importerSourceFile: module,
640
759
  moduleSpecifierFilePath: path,
641
760
  importElement: defaultImport,
642
761
  isInExports: false,
@@ -644,9 +763,9 @@ function processImportClausesForModules(modules, exports) {
644
763
  }
645
764
  const namespaceImport = impDecl.getNamespaceImport();
646
765
  if (namespaceImport !== undefined) {
647
- analyze_1.logger.debug(`Importing (namespace) ${namespaceImport.getText()} from ${impDecl.getModuleSpecifierValue()}`);
648
- analyze_1.entityDictionary.createFamixImportClause({ importDeclaration: impDecl,
649
- importer: module,
766
+ analyze_1.logger.info(`Importing (namespace) ${namespaceImport.getText()} from ${impDecl.getModuleSpecifierValue()} in ${modulePath}`);
767
+ analyze_1.entityDictionary.oldCreateFamixImportClause({ importDeclaration: impDecl,
768
+ importerSourceFile: module,
650
769
  moduleSpecifierFilePath: path,
651
770
  importElement: namespaceImport,
652
771
  isInExports: false,
@@ -656,12 +775,20 @@ function processImportClausesForModules(modules, exports) {
656
775
  });
657
776
  });
658
777
  }
659
- exports.processImportClausesForModules = processImportClausesForModules;
778
+ function isInExports(exports, importedEntityName) {
779
+ let importFoundInExports = false;
780
+ exports.forEach(e => {
781
+ if (e.has(importedEntityName)) {
782
+ importFoundInExports = true;
783
+ }
784
+ });
785
+ return importFoundInExports;
786
+ }
660
787
  /**
661
- * Builds a Famix model for the inheritances of the classes and interfaces of the source files
662
- * @param classes An array of classes
663
- * @param interfaces An array of interfaces
664
- */
788
+ * Builds a Famix model for the inheritances of the classes and interfaces of the source files
789
+ * @param classes An array of classes
790
+ * @param interfaces An array of interfaces
791
+ */
665
792
  function processInheritances(classes, interfaces) {
666
793
  analyze_1.logger.info(`processInheritances: Creating inheritances:`);
667
794
  classes.forEach(cls => {
@@ -687,7 +814,6 @@ function processInheritances(classes, interfaces) {
687
814
  });
688
815
  });
689
816
  }
690
- exports.processInheritances = processInheritances;
691
817
  /**
692
818
  * Builds a Famix model for the invocations of the methods and functions of the source files
693
819
  * @param methodsAndFunctionsWithId A map of methods and functions with their id
@@ -695,17 +821,18 @@ exports.processInheritances = processInheritances;
695
821
  function processInvocations(methodsAndFunctionsWithId) {
696
822
  analyze_1.logger.info(`Creating invocations:`);
697
823
  methodsAndFunctionsWithId.forEach((m, id) => {
698
- analyze_1.logger.debug(`Invocations to ${(m instanceof ts_morph_1.MethodDeclaration || m instanceof ts_morph_1.GetAccessorDeclaration || m instanceof ts_morph_1.SetAccessorDeclaration || m instanceof ts_morph_1.FunctionDeclaration) ? m.getName() : ((m instanceof ts_morph_1.ConstructorDeclaration) ? 'constructor' : (m.getName() ? m.getName() : 'anonymous'))}`);
699
- try {
700
- const temp_nodes = m.findReferencesAsNodes();
701
- temp_nodes.forEach(node => processNodeForInvocations(node, m, id));
702
- }
703
- catch (error) {
704
- analyze_1.logger.error(`> WARNING: got exception ${error}. Continuing...`);
824
+ if (!(m instanceof ts_morph_1.ArrowFunction)) {
825
+ analyze_1.logger.debug(`Invocations to ${(m instanceof ts_morph_1.MethodDeclaration || m instanceof ts_morph_1.GetAccessorDeclaration || m instanceof ts_morph_1.SetAccessorDeclaration || m instanceof ts_morph_1.FunctionDeclaration) ? m.getName() : ((m instanceof ts_morph_1.ConstructorDeclaration) ? 'constructor' : (m.getName() ? m.getName() : 'anonymous'))}`);
826
+ try {
827
+ const temp_nodes = m.findReferencesAsNodes();
828
+ temp_nodes.forEach(node => processNodeForInvocations(node, m, id));
829
+ }
830
+ catch (error) {
831
+ analyze_1.logger.error(`> WARNING: got exception ${error}. Continuing...`);
832
+ }
705
833
  }
706
834
  });
707
835
  }
708
- exports.processInvocations = processInvocations;
709
836
  /**
710
837
  * Builds a Famix model for an invocation of a method or a function
711
838
  * @param n A node
@@ -721,3 +848,30 @@ function processNodeForInvocations(n, m, id) {
721
848
  analyze_1.logger.error(`> WARNING: got exception ${error}. ScopeDeclaration invalid for ${n.getSymbol().getFullyQualifiedName()}. Continuing...`);
722
849
  }
723
850
  }
851
+ /**
852
+ * Builds a Famix model for the inheritances of the classes and interfaces of the source files
853
+ * @param classes An array of classes
854
+ * @param interfaces An array of interfaces
855
+ */
856
+ function processConcretisations(classes, interfaces, functions) {
857
+ analyze_1.logger.info(`processConcretisations: Creating concretisations:`);
858
+ classes.forEach(cls => {
859
+ analyze_1.logger.debug(`processConcretisations: Checking class concretisation for ${cls.getName()}`);
860
+ analyze_1.entityDictionary.createFamixConcretisationClassOrInterfaceSpecialisation(cls);
861
+ analyze_1.entityDictionary.createFamixConcretisationGenericInstantiation(cls);
862
+ analyze_1.entityDictionary.createFamixConcretisationInterfaceClass(cls);
863
+ analyze_1.entityDictionary.createFamixConcretisationTypeInstanciation(cls);
864
+ });
865
+ interfaces.forEach(inter => {
866
+ analyze_1.logger.debug(`processConcretisations: Checking interface concretisation for ${inter.getName()}`);
867
+ analyze_1.entityDictionary.createFamixConcretisationTypeInstanciation(inter);
868
+ analyze_1.entityDictionary.createFamixConcretisationClassOrInterfaceSpecialisation(inter);
869
+ });
870
+ functions.forEach(func => {
871
+ if (func instanceof ts_morph_1.FunctionDeclaration || func instanceof ts_morph_1.MethodDeclaration) {
872
+ analyze_1.logger.debug(`processConcretisations: Checking Method concretisation`);
873
+ analyze_1.entityDictionary.createFamixConcretisationFunctionInstantiation(func);
874
+ }
875
+ });
876
+ }
877
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvY2Vzc19mdW5jdGlvbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYW5hbHl6ZV9mdW5jdGlvbnMvcHJvY2Vzc19mdW5jdGlvbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFpQ0Esc0NBY0M7QUFRRCxnRkFzQkM7QUFNRCxvQ0FZQztBQXFDRCw4QkFHQztBQUVELGtDQUlDO0FBOG9CRCwwQ0FXQztBQTRCRCwwR0F1QkM7QUFPRCx3RUE2Q0M7QUFpQkQsa0RBNkJDO0FBTUQsZ0RBYUM7QUF1QkQsd0RBcUJDO0FBMS9CRCx1Q0FBb2xCO0FBQ3BsQixnRUFBa0Q7QUFDbEQsNkVBQWlFO0FBQ2pFLHVDQUF5QjtBQUN6Qix3Q0FBdUQ7QUFDdkQsZ0NBQWdDO0FBS25CLFFBQUEseUJBQXlCLEdBQUcsSUFBSSxHQUFHLEVBQW1LLENBQUMsQ0FBQywrSUFBK0k7QUFFdlYsUUFBQSxTQUFTLEdBQUcsSUFBSSxHQUFHLEVBQXFDLENBQUMsQ0FBQyx3SUFBd0k7QUFDbE0sUUFBQSxPQUFPLEdBQUcsSUFBSSxLQUFLLEVBQW9CLENBQUMsQ0FBQywrQ0FBK0M7QUFDeEYsUUFBQSxVQUFVLEdBQUcsSUFBSSxLQUFLLEVBQXdCLENBQUMsQ0FBQyxrREFBa0Q7QUFDbEcsUUFBQSxPQUFPLEdBQUcsSUFBSSxLQUFLLEVBQWMsQ0FBQyxDQUFDLGtEQUFrRDtBQUNyRixRQUFBLGdCQUFnQixHQUFHLElBQUksS0FBSyxFQUErQyxDQUFDLENBQUMsK0JBQStCO0FBR3pIOzs7O0dBSUc7QUFDSCxTQUFTLG1CQUFtQixDQUFDLFVBQXNCO0lBQy9DLE9BQU8sVUFBVSxDQUFDLHFCQUFxQixFQUFFLENBQUMsTUFBTSxHQUFHLENBQUMsSUFBSSxVQUFVLENBQUMsdUJBQXVCLEVBQUUsQ0FBQyxJQUFJLEdBQUcsQ0FBQyxDQUFDO0FBQzFHLENBQUM7QUFFRDs7OztHQUlHO0FBQ0gsU0FBZ0IsYUFBYSxD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QkFBZ0IsQ0FBQyx1REFBdUQsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM5RSwwQkFBZ0IsQ0FBQyw2Q0FBNkMsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUNwRSwwQkFBZ0IsQ0FBQyx1Q0FBdUMsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUM5RCwwQkFBZ0IsQ0FBQywwQ0FBMEMsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUVyRSxDQUFDLENBQUMsQ0FBQztJQUNILFVBQVUsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLEVBQUU7UUFDdkIsZ0JBQU0sQ0FBQyxLQUFLLENBQUMsaUVBQWlFLEtBQUssQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFDakcsMEJBQWdCLENBQUMsMENBQTBDLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDbkUsMEJBQWdCLENBQUMsdURBQXVELENBQUMsS0FBSyxDQUFDLENBQUE7SUFDbkYsQ0FBQyxDQUFDLENBQUM7SUFDSCxTQUFTLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxFQUFFO1FBQ3JCLElBQUcsSUFBSSxZQUFZLDhCQUFtQixJQUFJLElBQUksWUFBWSw0QkFBaUIsRUFBRSxDQUFDO1lBQzFFLGdCQUFNLENBQUMsS0FBSyxDQUFDLHdEQUF3RCxDQUFDLENBQUM7WUFDdkUsMEJBQWdCLENBQUMsOENBQThDLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDMUUsQ0FBQztJQUNMLENBQUMsQ0FBQyxDQUFBO0FBQ04sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENsYXNzRGVjbGFyYXRpb24sIE1ldGhvZERlY2xhcmF0aW9uLCBWYXJpYWJsZVN0YXRlbWVudCwgRnVuY3Rpb25EZWNsYXJhdGlvbiwgVmFyaWFibGVEZWNsYXJhdGlvbiwgSW50ZXJmYWNlRGVjbGFyYXRpb24sIFBhcmFtZXRlckRlY2xhcmF0aW9uLCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uLCBNZXRob2RTaWduYXR1cmUsIFNvdXJjZUZpbGUsIE1vZHVsZURlY2xhcmF0aW9uLCBQcm9wZXJ0eURlY2xhcmF0aW9uLCBQcm9wZXJ0eVNpZ25hdHVyZSwgRGVjb3JhdG9yLCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uLCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uLCBFeHBvcnRlZERlY2xhcmF0aW9ucywgQ29tbWVudFJhbmdlLCBFbnVtRGVjbGFyYXRpb24sIEVudW1NZW1iZXIsIFR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbiwgVHlwZUFsaWFzRGVjbGFyYXRpb24sIFN5bnRheEtpbmQsIEZ1bmN0aW9uRXhwcmVzc2lvbiwgQmxvY2ssIElkZW50aWZpZXIsIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cywgSW1wb3J0RGVjbGFyYXRpb24sIE5vZGUsIEFycm93RnVuY3Rpb24sIFNjb3BlLCBDbGFzc0V4cHJlc3Npb24gfSBmcm9tIFwidHMtbW9ycGhcIjtcbmltcG9ydCAqIGFzIEZhbWl4IGZyb20gXCIuLi9saWIvZmFtaXgvbW9kZWwvZmFtaXhcIjtcbmltcG9ydCB7IGNhbGN1bGF0ZSB9IGZyb20gXCIuLi9saWIvdHMtY29tcGxleC9jeWNsb21hdGljLXNlcnZpY2VcIjtcbmltcG9ydCAqIGFzIGZzIGZyb20gJ2ZzJztcbmltcG9ydCB7IGxvZ2dlciAsIGVudGl0eURpY3Rpb25hcnkgfSBmcm9tIFwiLi4vYW5hbHl6ZVwiO1xuaW1wb3J0IHsgZ2V0RlFOIH0gZnJvbSBcIi4uL2ZxblwiO1xuXG5leHBvcnQgdHlwZSBBY2Nlc3NpYmxlVFNNb3JwaEVsZW1lbnQgPSBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFZhcmlhYmxlRGVjbGFyYXRpb24gfCBQcm9wZXJ0eURlY2xhcmF0aW9uIHwgRW51bU1lbWJlcjtcbmV4cG9ydCB0eXBlIEZhbWl4SUQgPSBudW1iZXI7XG5cbmV4cG9ydCBjb25zdCBtZXRob2RzQW5kRnVuY3Rpb25zV2l0aElkID0gbmV3IE1hcDxudW1iZXIsIE1ldGhvZERlY2xhcmF0aW9uIHwgQ29uc3RydWN0b3JEZWNsYXJhdGlvbiB8IEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgRnVuY3Rpb25EZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRXhwcmVzc2lvbiB8IEFycm93RnVuY3Rpb24+KCk7IC8vIE1hcHMgdGhlIEZhbWl4IG1ldGhvZCwgY29uc3RydWN0b3IsIGdldHRlciwgc2V0dGVyIGFuZCBmdW5jdGlvbiBpZHMgdG8gdGhlaXIgdHMtbW9ycGggbWV0aG9kLCBjb25zdHJ1Y3RvciwgZ2V0dGVyLCBzZXR0ZXIgb3IgZnVuY3Rpb24gb2JqZWN0XG5cbmV4cG9ydCBjb25zdCBhY2Nlc3NNYXAgPSBuZXcgTWFwPEZhbWl4SUQsIEFjY2Vzc2libGVUU01vcnBoRWxlbWVudD4oKTsgLy8gTWFwcyB0aGUgRmFtaXggcGFyYW1ldGVyLCB2YXJpYWJsZSwgcHJvcGVydHkgYW5kIGVudW0gdmFsdWUgaWRzIHRvIHRoZWlyIHRzLW1vcnBoIHBhcmFtZXRlciwgdmFyaWFibGUsIHByb3BlcnR5IG9yIGVudW0gbWVtYmVyIG9iamVjdFxuZXhwb3J0IGNvbnN0IGNsYXNzZXMgPSBuZXcgQXJyYXk8Q2xhc3NEZWNsYXJhdGlvbj4oKTsgLy8gQXJyYXkgb2YgYWxsIHRoZSBjbGFzc2VzIG9mIHRoZSBzb3VyY2UgZmlsZXNcbmV4cG9ydCBjb25zdCBpbnRlcmZhY2VzID0gbmV3IEFycmF5PEludGVyZmFjZURlY2xhcmF0aW9uPigpOyAvLyBBcnJheSBvZiBhbGwgdGhlIGludGVyZmFjZXMgb2YgdGhlIHNvdXJjZSBmaWxlc1xuZXhwb3J0IGNvbnN0IG1vZHVsZXMgPSBuZXcgQXJyYXk8U291cmNlRmlsZT4oKTsgLy8gQXJyYXkgb2YgYWxsIHRoZSBzb3VyY2UgZmlsZXMgd2hpY2ggYXJlIG1vZHVsZXNcbmV4cG9ydCBjb25zdCBsaXN0T2ZFeHBvcnRNYXBzID0gbmV3IEFycmF5PFJlYWRvbmx5TWFwPHN0cmluZywgRXhwb3J0ZWREZWNsYXJhdGlvbnNbXT4+KCk7IC8vIEFycmF5IG9mIGFsbCB0aGUgZXhwb3J0IG1hcHNcbmV4cG9ydCBsZXQgY3VycmVudENDOiB7IFtrZXk6IHN0cmluZ106IG51bWJlciB9OyAvLyBTdG9yZXMgdGhlIGN5Y2xvbWF0aWMgY29tcGxleGl0eSBtZXRyaWNzIGZvciB0aGUgY3VycmVudCBzb3VyY2UgZmlsZVxuXG4vKipcbiAqIENoZWNrcyBpZiB0aGUgZmlsZSBoYXMgYW55IGltcG9ydHMgb3IgZXhwb3J0cyB0byBiZSBjb25zaWRlcmVkIGEgbW9kdWxlXG4gKiBAcGFyYW0gc291cmNlRmlsZSBBIHNvdXJjZSBmaWxlXG4gKiBAcmV0dXJucyBBIGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgZmlsZSBpcyBhIG1vZHVsZVxuICovXG5mdW5jdGlvbiBpc1NvdXJjZUZpbGVBTW9kdWxlKHNvdXJjZUZpbGU6IFNvdXJjZUZpbGUpOiBib29sZWFuIHtcbiAgICByZXR1cm4gc291cmNlRmlsZS5nZXRJbXBvcnREZWNsYXJhdGlvbnMoKS5sZW5ndGggPiAwIHx8IHNvdXJjZUZpbGUuZ2V0RXhwb3J0ZWREZWNsYXJhdGlvbnMoKS5zaXplID4gMDtcbn1cblxuLyoqXG4gKiBHZXRzIHRoZSBwYXRoIG9mIGEgbW9kdWxlIHRvIGJlIGltcG9ydGVkXG4gKiBAcGFyYW0gaW1wb3J0RGVjbCBBbiBpbXBvcnQgZGVjbGFyYXRpb25cbiAqIEByZXR1cm5zIFRoZSBwYXRoIG9mIHRoZSBtb2R1bGUgdG8gYmUgaW1wb3J0ZWRcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldE1vZHVsZVBhdGgoaW1wb3J0RGVjbDogSW1wb3J0RGVjbGFyYXRpb24pOiBzdHJpbmcge1xuICAgIGxldCBwYXRoOiBzdHJpbmc7XG4gICAgaWYgKGltcG9ydERlY2wuZ2V0TW9kdWxlU3BlY2lmaWVyU291cmNlRmlsZSgpID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgaWYgKGltcG9ydERlY2wuZ2V0TW9kdWxlU3BlY2lmaWVyVmFsdWUoKS5zdWJzdHJpbmcoaW1wb3J0RGVjbC5nZXRNb2R1bGVTcGVjaWZpZXJWYWx1ZSgpLmxlbmd0aCAtIDMpID09PSBcIi50c1wiKSB7XG4gICAgICAgICAgICBwYXRoID0gaW1wb3J0RGVjbC5nZXRNb2R1bGVTcGVjaWZpZXJWYWx1ZSgpO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgcGF0aCA9IGltcG9ydERlY2wuZ2V0TW9kdWxlU3BlY2lmaWVyVmFsdWUoKSArIFwiLnRzXCI7XG4gICAgICAgIH1cbiAgICB9XG4gICAgZWxzZSB7XG4gICAgICAgIHBhdGggPSBpbXBvcnREZWNsLmdldE1vZHVsZVNwZWNpZmllclNvdXJjZUZpbGUoKSEuZ2V0RmlsZVBhdGgoKTtcbiAgICB9XG4gICAgcmV0dXJuIHBhdGg7XG59XG5cbi8qKlxuICogR2V0cyB0aGUgaW50ZXJmYWNlcyBpbXBsZW1lbnRlZCBvciBleHRlbmRlZCBieSBhIGNsYXNzIG9yIGFuIGludGVyZmFjZVxuICogQHBhcmFtIGludGVyZmFjZXMgQW4gYXJyYXkgb2YgaW50ZXJmYWNlc1xuICogQHBhcmFtIHN1YkNsYXNzIEEgY2xhc3Mgb3IgYW4gaW50ZXJmYWNlXG4gKiBAcmV0dXJucyBBbiBhcnJheSBvZiBJbnRlcmZhY2VEZWNsYXJhdGlvbiBhbmQgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzIGNvbnRhaW5pbmcgdGhlIGludGVyZmFjZXMgaW1wbGVtZW50ZWQgb3IgZXh0ZW5kZWQgYnkgdGhlIHN1YkNsYXNzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBnZXRJbXBsZW1lbnRlZE9yRXh0ZW5kZWRJbnRlcmZhY2VzKGludGVyZmFjZXM6IEFycmF5PEludGVyZmFjZURlY2xhcmF0aW9uPiwgc3ViQ2xhc3M6IENsYXNzRGVjbGFyYXRpb24gfCBJbnRlcmZhY2VEZWNsYXJhdGlvbik6IEFycmF5PEludGVyZmFjZURlY2xhcmF0aW9uIHwgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzPiB7XG4gICAgbGV0IGltcE9yRXh0SW50ZXJmYWNlczogQXJyYXk8RXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzPjtcbiAgICBpZiAoc3ViQ2xhc3MgaW5zdGFuY2VvZiBDbGFzc0RlY2xhcmF0aW9uKSB7XG4gICAgICAgIGltcE9yRXh0SW50ZXJmYWNlcyA9IHN1YkNsYXNzLmdldEltcGxlbWVudHMoKTtcbiAgICB9XG4gICAgZWxzZSB7XG4gICAgICAgIGltcE9yRXh0SW50ZXJmYWNlcyA9IHN1YkNsYXNzLmdldEV4dGVuZHMoKTtcbiAgICB9XG5cbiAgICBjb25zdCBpbnRlcmZhY2VzTmFtZXMgPSBpbnRlcmZhY2VzLm1hcChpID0+IGkuZ2V0TmFtZSgpKTtcbiAgICBjb25zdCBpbXBsZW1lbnRlZE9yRXh0ZW5kZWRJbnRlcmZhY2VzID0gbmV3IEFycmF5PEludGVyZmFjZURlY2xhcmF0aW9uIHwgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzPigpO1xuXG4gICAgaW1wT3JFeHRJbnRlcmZhY2VzLmZvckVhY2goaSA9PiB7XG4gICAgICAgIGlmIChpbnRlcmZhY2VzTmFtZXMuaW5jbHVkZXMoaS5nZXRFeHByZXNzaW9uKCkuZ2V0VGV4dCgpKSkge1xuICAgICAgICAgICAgaW1wbGVtZW50ZWRPckV4dGVuZGVkSW50ZXJmYWNlcy5wdXNoKGludGVyZmFjZXNbaW50ZXJmYWNlc05hbWVzLmluZGV4T2YoaS5nZXRFeHByZXNzaW9uKCkuZ2V0VGV4dCgpKV0pO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgaW1wbGVtZW50ZWRPckV4dGVuZGVkSW50ZXJmYWNlcy5wdXNoKGkpO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICByZXR1cm4gaW1wbGVtZW50ZWRPckV4dGVuZGVkSW50ZXJmYWNlcztcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgYW4gYXJyYXkgb2Ygc291cmNlIGZpbGVzXG4gKiBAcGFyYW0gc291cmNlRmlsZXMgQW4gYXJyYXkgb2Ygc291cmNlIGZpbGVzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcm9jZXNzRmlsZXMoc291cmNlRmlsZXM6IEFycmF5PFNvdXJjZUZpbGU+KTogdm9pZCB7XG4gICAgc291cmNlRmlsZXMuZm9yRWFjaChmaWxlID0+IHtcbiAgICAgICAgbG9nZ2VyLmluZm8oYEZpbGU6ID4+Pj4+Pj4+Pj4gJHtmaWxlLmdldEZpbGVQYXRoKCl9YCk7XG5cbiAgICAgICAgLy8gQ29tcHV0ZXMgdGhlIGN5Y2xvbWF0aWMgY29tcGxleGl0eSBtZXRyaWNzIGZvciB0aGUgY3VycmVudCBzb3VyY2UgZmlsZSBpZiBpdCBleGlzdHMgKGkuZS4gaWYgaXQgaXMgbm90IGZyb20gYSBqZXN0IHRlc3QpXG4gICAgICAgIGlmIChmcy5leGlzdHNTeW5jKGZpbGUuZ2V0RmlsZVBhdGgoKSkpXG4gICAgICAgICAgICBjdXJyZW50Q0MgPSBjYWxjdWxhdGUoZmlsZS5nZXRGaWxlUGF0aCgpKTtcbiAgICAgICAgZWxzZVxuICAgICAgICAgICAgY3VycmVudENDID0ge307XG5cbiAgICAgICAgcHJvY2Vzc0ZpbGUoZmlsZSk7XG4gICAgfSk7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIGEgc291cmNlIGZpbGVcbiAqIEBwYXJhbSBmIEEgc291cmNlIGZpbGVcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc0ZpbGUoZjogU291cmNlRmlsZSk6IHZvaWQge1xuICAgIGNvbnN0IGlzTW9kdWxlID0gaXNTb3VyY2VGaWxlQU1vZHVsZShmKTtcblxuICAgIGlmIChpc01vZHVsZSkge1xuICAgICAgICBtb2R1bGVzLnB1c2goZik7XG4gICAgfVxuXG4gICAgY29uc3QgZXhwb3J0TWFwID0gZi5nZXRFeHBvcnRlZERlY2xhcmF0aW9ucygpO1xuICAgIGlmIChleHBvcnRNYXApIGxpc3RPZkV4cG9ydE1hcHMucHVzaChleHBvcnRNYXApO1xuXG4gICAgY29uc3QgZm14RmlsZSA9IGVudGl0eURpY3Rpb25hcnkuY3JlYXRlT3JHZXRGYW1peEZpbGUoZiwgaXNNb2R1bGUpO1xuXG4gICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzRmlsZTogZmlsZTogJHtmLmdldEJhc2VOYW1lKCl9LCBmcW4gPSAke2ZteEZpbGUuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuXG4gICAgcHJvY2Vzc0NvbW1lbnRzKGYsIGZteEZpbGUpO1xuXG4gICAgcHJvY2Vzc0FsaWFzZXMoZiwgZm14RmlsZSk7XG5cbiAgICBwcm9jZXNzQ2xhc3NlcyhmLCBmbXhGaWxlKTtcblxuICAgIHByb2Nlc3NJbnRlcmZhY2VzKGYsIGZteEZpbGUpO1xuXG4gICAgcHJvY2Vzc1ZhcmlhYmxlcyhmLCBmbXhGaWxlKTtcblxuICAgIHByb2Nlc3NFbnVtcyhmLCBmbXhGaWxlKTtcblxuICAgIHByb2Nlc3NGdW5jdGlvbnMoZiwgZm14RmlsZSk7XG5cbiAgICBwcm9jZXNzTW9kdWxlcyhmLCBmbXhGaWxlKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGlzQW1iaWVudChub2RlOiBNb2R1bGVEZWNsYXJhdGlvbik6IGJvb2xlYW4ge1xuICAgIC8vIEFuIGFtYmllbnQgbW9kdWxlIGhhcyB0aGUgRGVjbGFyZUtleXdvcmQgbW9kaWZpZXIuXG4gICAgcmV0dXJuIChub2RlLmdldE1vZGlmaWVycygpPy5zb21lKG1vZGlmaWVyID0+IG1vZGlmaWVyLmdldEtpbmQoKSA9PT0gU3ludGF4S2luZC5EZWNsYXJlS2V5d29yZCkpID8/IGZhbHNlO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaXNOYW1lc3BhY2Uobm9kZTogTW9kdWxlRGVjbGFyYXRpb24pOiBib29sZWFuIHtcbiAgICAvLyBDaGVjayBpZiB0aGUgbW9kdWxlIGRlY2xhcmF0aW9uIGhhcyBhIG5hbWVzcGFjZSBrZXl3b3JkLlxuICAgIC8vIFRoaXMgYXBwcm9hY2ggdXNlcyB0aGUgZ2V0Q2hpbGRyZW4oKSBtZXRob2QgdG8gaW5zcGVjdCB0aGUgc3ludGF4IGRpcmVjdGx5LlxuICAgIHJldHVybiBub2RlLmdldENoaWxkcmVuT2ZLaW5kKFN5bnRheEtpbmQuTmFtZXNwYWNlS2V5d29yZCkubGVuZ3RoID4gMDtcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgYSBtb2R1bGUgKGFsc28gbmFtZXNwYWNlKVxuICogQHBhcmFtIG0gQSBuYW1lc3BhY2VcbiAqIEByZXR1cm5zIEEgRmFtaXguTW9kdWxlIHJlcHJlc2VudGluZyB0aGUgbW9kdWxlXG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NNb2R1bGUobTogTW9kdWxlRGVjbGFyYXRpb24pOiBGYW1peC5Nb2R1bGUge1xuICAgIGNvbnN0IGZteE1vZHVsZSA9IGVudGl0eURpY3Rpb25hcnkuY3JlYXRlT3JHZXRGYW1peE1vZHVsZShtKTtcblxuICAgIGxvZ2dlci5kZWJ1ZyhgbW9kdWxlOiAke20uZ2V0TmFtZSgpfSwgKCR7bS5nZXRUeXBlKCkuZ2V0VGV4dCgpfSksICR7Zm14TW9kdWxlLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcblxuICAgIHByb2Nlc3NDb21tZW50cyhtLCBmbXhNb2R1bGUpO1xuXG4gICAgcHJvY2Vzc0FsaWFzZXMobSwgZm14TW9kdWxlKTtcblxuICAgIHByb2Nlc3NDbGFzc2VzKG0sIGZteE1vZHVsZSk7XG5cbiAgICBwcm9jZXNzSW50ZXJmYWNlcyhtLCBmbXhNb2R1bGUpO1xuXG4gICAgcHJvY2Vzc1ZhcmlhYmxlcyhtLCBmbXhNb2R1bGUpO1xuXG4gICAgcHJvY2Vzc0VudW1zKG0sIGZteE1vZHVsZSk7XG4gICAgXG4gICAgcHJvY2Vzc0Z1bmN0aW9ucyhtLCBmbXhNb2R1bGUpO1xuXG4gICAgcHJvY2Vzc01vZHVsZXMobSwgZm14TW9kdWxlKTtcblxuICAgIHJldHVybiBmbXhNb2R1bGU7XG59XG5cbnR5cGUgQ29udGFpbmVyVHlwZXMgPSBTb3VyY2VGaWxlIHwgTW9kdWxlRGVjbGFyYXRpb24gfCBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uIHwgTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBBcnJvd0Z1bmN0aW9uO1xuXG50eXBlIFNjb3BlZFR5cGVzID0gRmFtaXguU2NyaXB0RW50aXR5IHwgRmFtaXguTW9kdWxlIHwgRmFtaXguRnVuY3Rpb24gfCBGYW1peC5NZXRob2QgfCBGYW1peC5BY2Nlc3NvcjtcblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgdGhlIGFsaWFzZXMgb2YgYSBjb250YWluZXJcbiAqIEBwYXJhbSBtIEEgY29udGFpbmVyIChhIHNvdXJjZSBmaWxlLCBhIG5hbWVzcGFjZSwgYSBmdW5jdGlvbiBvciBhIG1ldGhvZClcbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGNvbnRhaW5lclxuICovXG5mdW5jdGlvbiBwcm9jZXNzQWxpYXNlcyhtOiBDb250YWluZXJUeXBlcywgZm14U2NvcGU6IFNjb3BlZFR5cGVzKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzQWxpYXNlczogLS0tLS0tLS0tLSBGaW5kaW5nIEFsaWFzZXM6YCk7XG4gICAgbS5nZXRUeXBlQWxpYXNlcygpLmZvckVhY2goYSA9PiB7XG4gICAgICAgIGNvbnN0IGZteEFsaWFzID0gcHJvY2Vzc0FsaWFzKGEpO1xuICAgICAgICBmbXhTY29wZS5hZGRBbGlhcyhmbXhBbGlhcyk7XG4gICAgfSk7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIHRoZSBjbGFzc2VzIG9mIGEgY29udGFpbmVyXG4gKiBAcGFyYW0gbSBBIGNvbnRhaW5lciAoYSBzb3VyY2UgZmlsZSBvciBhIG5hbWVzcGFjZSlcbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGNvbnRhaW5lclxuICovXG5mdW5jdGlvbiBwcm9jZXNzQ2xhc3NlcyhtOiBTb3VyY2VGaWxlIHwgTW9kdWxlRGVjbGFyYXRpb24sIGZteFNjb3BlOiBGYW1peC5TY3JpcHRFbnRpdHkgfCBGYW1peC5Nb2R1bGUgKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzQ2xhc3NlczogLS0tLS0tLS0tLSBGaW5kaW5nIENsYXNzZXM6YCk7XG4gICAgY29uc3QgY2xhc3Nlc0luQXJyb3dGdW5jdGlvbnMgPSBnZXRDbGFzc2VzRGVjbGFyZWRJbkFycm93RnVuY3Rpb25zKG0pO1xuICAgIGNvbnN0IGNsYXNzZXMgPSBtLmdldENsYXNzZXMoKS5jb25jYXQoY2xhc3Nlc0luQXJyb3dGdW5jdGlvbnMpO1xuICAgIGNsYXNzZXMuZm9yRWFjaChjID0+IHtcbiAgICAgICAgY29uc3QgZm14Q2xhc3MgPSBwcm9jZXNzQ2xhc3MoYyk7XG4gICAgICAgIGZteFNjb3BlLmFkZFR5cGUoZm14Q2xhc3MpO1xuICAgIH0pO1xufVxuXG5mdW5jdGlvbiBnZXRBcnJvd0Z1bmN0aW9uQ2xhc3NlcyhmOiBBcnJvd0Z1bmN0aW9uKTogQ2xhc3NEZWNsYXJhdGlvbltdIHtcbiAgICBjb25zdCBjbGFzc2VzOiBDbGFzc0RlY2xhcmF0aW9uW10gPSBbXTtcblxuICAgIGZ1bmN0aW9uIGZpbmRDbGFzc2VzKG5vZGU6IGFueSkge1xuICAgICAgICBpZiAobm9kZS5nZXRLaW5kKCkgPT09IFN5bnRheEtpbmQuQ2xhc3NEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgY2xhc3Nlcy5wdXNoKG5vZGUgYXMgQ2xhc3NEZWNsYXJhdGlvbik7XG4gICAgICAgIH1cbiAgICAgICAgbm9kZS5nZXRDaGlsZHJlbigpLmZvckVhY2goZmluZENsYXNzZXMpO1xuICAgIH1cblxuICAgIGZpbmRDbGFzc2VzKGYpO1xuICAgIHJldHVybiBjbGFzc2VzO1xufVxuXG4vKipcbiAqIHRzLW1vcnBoIGRvZXNuJ3QgZmluZCBjbGFzc2VzIGluIGFycm93IGZ1bmN0aW9ucywgc28gd2UgbmVlZCB0byBmaW5kIHRoZW0gbWFudWFsbHlcbiAqIEBwYXJhbSBzIEEgc291cmNlIGZpbGUgXG4gKiBAcmV0dXJucyB0aGUgQ2xhc3NEZWNsYXJhdGlvbiBvYmplY3RzIGZvdW5kIGluIGFycm93IGZ1bmN0aW9ucyBvZiB0aGUgc291cmNlIGZpbGVcbiAqL1xuZnVuY3Rpb24gZ2V0Q2xhc3Nlc0RlY2xhcmVkSW5BcnJvd0Z1bmN0aW9ucyhzOiBTb3VyY2VGaWxlIHwgTW9kdWxlRGVjbGFyYXRpb24pOiBDbGFzc0RlY2xhcmF0aW9uW10ge1xuICAgIGNvbnN0IGFycm93RnVuY3Rpb25zID0gcy5nZXREZXNjZW5kYW50c09mS2luZChTeW50YXhLaW5kLkFycm93RnVuY3Rpb24pO1xuICAgIGNvbnN0IGNsYXNzZXNJbkFycm93RnVuY3Rpb25zID0gYXJyb3dGdW5jdGlvbnMubWFwKGYgPT4gZ2V0QXJyb3dGdW5jdGlvbkNsYXNzZXMoZikpLmZsYXQoKTtcbiAgICByZXR1cm4gY2xhc3Nlc0luQXJyb3dGdW5jdGlvbnM7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIHRoZSBpbnRlcmZhY2VzIG9mIGEgY29udGFpbmVyXG4gKiBAcGFyYW0gbSBBIGNvbnRhaW5lciAoYSBzb3VyY2UgZmlsZSBvciBhIG5hbWVzcGFjZSlcbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGNvbnRhaW5lclxuICovXG5mdW5jdGlvbiBwcm9jZXNzSW50ZXJmYWNlcyhtOiBTb3VyY2VGaWxlIHwgTW9kdWxlRGVjbGFyYXRpb24sIGZteFNjb3BlOiBGYW1peC5TY3JpcHRFbnRpdHkgfCBGYW1peC5Nb2R1bGUgKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzSW50ZXJmYWNlczogLS0tLS0tLS0tLSBGaW5kaW5nIEludGVyZmFjZXM6YCk7XG4gICAgbS5nZXRJbnRlcmZhY2VzKCkuZm9yRWFjaChpID0+IHtcbiAgICAgICAgY29uc3QgZm14SW50ZXJmYWNlID0gcHJvY2Vzc0ludGVyZmFjZShpKTtcbiAgICAgICAgZm14U2NvcGUuYWRkVHlwZShmbXhJbnRlcmZhY2UpO1xuICAgIH0pO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgdmFyaWFibGVzIG9mIGEgY29udGFpbmVyXG4gKiBAcGFyYW0gbSBBIGNvbnRhaW5lciAoYSBzb3VyY2UgZmlsZSwgYSBuYW1lc3BhY2UsIGEgZnVuY3Rpb24gb3IgYSBtZXRob2QpXG4gKiBAcGFyYW0gZm14U2NvcGUgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjb250YWluZXJcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc1ZhcmlhYmxlcyhtOiBDb250YWluZXJUeXBlcywgZm14U2NvcGU6IEZhbWl4LlNjcmlwdEVudGl0eSB8IEZhbWl4Lk1vZHVsZSB8IEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguTWV0aG9kIHwgRmFtaXguQWNjZXNzb3IpOiB2b2lkIHtcbiAgICBsb2dnZXIuZGVidWcoYHByb2Nlc3NWYXJpYWJsZXM6IC0tLS0tLS0tLS0gRmluZGluZyBWYXJpYWJsZXM6YCk7XG4gICAgbS5nZXRWYXJpYWJsZVN0YXRlbWVudHMoKS5mb3JFYWNoKHYgPT4ge1xuICAgICAgICBjb25zdCBmbXhWYXJpYWJsZXMgPSBwcm9jZXNzVmFyaWFibGVTdGF0ZW1lbnQodik7XG4gICAgICAgIGZteFZhcmlhYmxlcy5mb3JFYWNoKGZteFZhcmlhYmxlID0+IHtcbiAgICAgICAgICAgIGZteFNjb3BlLmFkZFZhcmlhYmxlKGZteFZhcmlhYmxlKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIHRoZSBlbnVtcyBvZiBhIGNvbnRhaW5lclxuICogQHBhcmFtIG0gQSBjb250YWluZXIgKGEgc291cmNlIGZpbGUsIGEgbmFtZXNwYWNlLCBhIGZ1bmN0aW9uIG9yIGEgbWV0aG9kKVxuICogQHBhcmFtIGZteFNjb3BlIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgY29udGFpbmVyXG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NFbnVtcyhtOiBDb250YWluZXJUeXBlcywgZm14U2NvcGU6IFNjb3BlZFR5cGVzKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzRW51bXM6IC0tLS0tLS0tLS0gRmluZGluZyBFbnVtczpgKTtcbiAgICBtLmdldEVudW1zKCkuZm9yRWFjaChlID0+IHtcbiAgICAgICAgY29uc3QgZm14RW51bSA9IHByb2Nlc3NFbnVtKGUpO1xuICAgICAgICBmbXhTY29wZS5hZGRUeXBlKGZteEVudW0pO1xuICAgIH0pO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgZnVuY3Rpb25zIG9mIGEgY29udGFpbmVyXG4gKiBAcGFyYW0gbSBBIGNvbnRhaW5lciAoYSBzb3VyY2UgZmlsZSwgYSBuYW1lc3BhY2UsIGEgZnVuY3Rpb24gb3IgYSBtZXRob2QpXG4gKiBAcGFyYW0gZm14U2NvcGUgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjb250YWluZXJcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc0Z1bmN0aW9ucyhtOiBDb250YWluZXJUeXBlcywgZm14U2NvcGU6IFNjb3BlZFR5cGVzKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBGaW5kaW5nIEZ1bmN0aW9uczpgKTtcbiAgICBtLmdldEZ1bmN0aW9ucygpLmZvckVhY2goZiA9PiB7XG4gICAgICAgIGNvbnN0IGZteEZ1bmN0aW9uID0gcHJvY2Vzc0Z1bmN0aW9uKGYpO1xuICAgICAgICBmbXhTY29wZS5hZGRGdW5jdGlvbihmbXhGdW5jdGlvbik7XG4gICAgfSk7XG5cbiAgICAvL2ZpbmQgYXJyb3cgZnVuY3Rpb25zXG4gICAgbG9nZ2VyLmRlYnVnKGBGaW5kaW5nIEZ1bmN0aW9uczpgKTtcbiAgICBjb25zdCBhcnJvd0Z1bmN0aW9ucyA9IG0uZ2V0RGVzY2VuZGFudHNPZktpbmQoU3ludGF4S2luZC5BcnJvd0Z1bmN0aW9uKTtcbiAgICBhcnJvd0Z1bmN0aW9ucy5mb3JFYWNoKGFmID0+IHtcbiAgICAgICAgY29uc3QgZm14RnVuY3Rpb24gPSBwcm9jZXNzRnVuY3Rpb24oYWYpO1xuICAgICAgICBmbXhTY29wZS5hZGRGdW5jdGlvbihmbXhGdW5jdGlvbik7XG4gICAgfSlcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgdGhlIG1vZHVsZXMgb2YgYSBjb250YWluZXIuXG4gKiBAcGFyYW0gbSBBIGNvbnRhaW5lciAoYSBzb3VyY2UgZmlsZSBvciBhIG5hbWVzcGFjZSlcbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGNvbnRhaW5lclxuICovXG5mdW5jdGlvbiBwcm9jZXNzTW9kdWxlcyhtOiBTb3VyY2VGaWxlIHwgTW9kdWxlRGVjbGFyYXRpb24sIGZteFNjb3BlOiBGYW1peC5TY3JpcHRFbnRpdHkgfCBGYW1peC5Nb2R1bGUgKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBGaW5kaW5nIE1vZHVsZXM6YCk7XG4gICAgbS5nZXRNb2R1bGVzKCkuZm9yRWFjaChtZCA9PiB7XG4gICAgICAgIGNvbnN0IGZteE1vZHVsZSA9IHByb2Nlc3NNb2R1bGUobWQpO1xuICAgICAgICBmbXhTY29wZS5hZGRNb2R1bGUoZm14TW9kdWxlKTtcbiAgICB9KTtcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgYW4gYWxpYXNcbiAqIEBwYXJhbSBhIEFuIGFsaWFzXG4gKiBAcmV0dXJucyBBIEZhbWl4LkFsaWFzIHJlcHJlc2VudGluZyB0aGUgYWxpYXNcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc0FsaWFzKGE6IFR5cGVBbGlhc0RlY2xhcmF0aW9uKTogRmFtaXguQWxpYXMge1xuICAgIGNvbnN0IGZteEFsaWFzID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peEFsaWFzKGEpO1xuXG4gICAgbG9nZ2VyLmRlYnVnKGBBbGlhczogJHthLmdldE5hbWUoKX0sICgke2EuZ2V0VHlwZSgpLmdldFRleHQoKX0pLCBmcW4gPSAke2ZteEFsaWFzLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcblxuICAgIHByb2Nlc3NDb21tZW50cyhhLCBmbXhBbGlhcyk7XG5cbiAgICByZXR1cm4gZm14QWxpYXM7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIGEgY2xhc3NcbiAqIEBwYXJhbSBjIEEgY2xhc3NcbiAqIEByZXR1cm5zIEEgRmFtaXguQ2xhc3Mgb3IgYSBGYW1peC5QYXJhbWV0cmljQ2xhc3MgcmVwcmVzZW50aW5nIHRoZSBjbGFzc1xuICovXG5mdW5jdGlvbiBwcm9jZXNzQ2xhc3MoYzogQ2xhc3NEZWNsYXJhdGlvbik6IEZhbWl4LkNsYXNzIHwgRmFtaXguUGFyYW1ldHJpY0NsYXNzIHtcbiAgICBjbGFzc2VzLnB1c2goYyk7XG5cbiAgICBjb25zdCBmbXhDbGFzcyA9IGVudGl0eURpY3Rpb25hcnkuY3JlYXRlT3JHZXRGYW1peENsYXNzKGMpO1xuXG4gICAgbG9nZ2VyLmRlYnVnKGBDbGFzczogJHtjLmdldE5hbWUoKX0sICgke2MuZ2V0VHlwZSgpLmdldFRleHQoKX0pLCBmcW4gPSAke2ZteENsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcblxuICAgIHByb2Nlc3NDb21tZW50cyhjLCBmbXhDbGFzcyk7XG5cbiAgICBwcm9jZXNzRGVjb3JhdG9ycyhjLCBmbXhDbGFzcyk7XG5cbiAgICBwcm9jZXNzU3RydWN0dXJlZFR5cGUoYywgZm14Q2xhc3MpO1xuXG4gICAgYy5nZXRDb25zdHJ1Y3RvcnMoKS5mb3JFYWNoKGNvbiA9PiB7XG4gICAgICAgIGNvbnN0IGZteENvbiA9IHByb2Nlc3NNZXRob2QoY29uKTtcbiAgICAgICAgZm14Q2xhc3MuYWRkTWV0aG9kKGZteENvbik7XG4gICAgfSk7XG5cbiAgICBjLmdldEdldEFjY2Vzc29ycygpLmZvckVhY2goYWNjID0+IHtcbiAgICAgICAgY29uc3QgZm14QWNjID0gcHJvY2Vzc01ldGhvZChhY2MpO1xuICAgICAgICBmbXhDbGFzcy5hZGRNZXRob2QoZm14QWNjKTtcbiAgICB9KTtcbiAgICBcbiAgICBjLmdldFNldEFjY2Vzc29ycygpLmZvckVhY2goYWNjID0+IHtcbiAgICAgICAgY29uc3QgZm14QWNjID0gcHJvY2Vzc01ldGhvZChhY2MpO1xuICAgICAgICBmbXhDbGFzcy5hZGRNZXRob2QoZm14QWNjKTtcbiAgICB9KTtcblxuICAgIHJldHVybiBmbXhDbGFzcztcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgYW4gaW50ZXJmYWNlXG4gKiBAcGFyYW0gaSBBbiBpbnRlcmZhY2VcbiAqIEByZXR1cm5zIEEgRmFtaXguSW50ZXJmYWNlIG9yIGEgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSByZXByZXNlbnRpbmcgdGhlIGludGVyZmFjZVxuICovXG5mdW5jdGlvbiBwcm9jZXNzSW50ZXJmYWNlKGk6IEludGVyZmFjZURlY2xhcmF0aW9uKTogRmFtaXguSW50ZXJmYWNlIHwgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSB7XG4gICAgaW50ZXJmYWNlcy5wdXNoKGkpO1xuXG4gICAgY29uc3QgZm14SW50ZXJmYWNlID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVPckdldEZhbWl4SW50ZXJmYWNlKGkpO1xuXG4gICAgbG9nZ2VyLmRlYnVnKGBJbnRlcmZhY2U6ICR7aS5nZXROYW1lKCl9LCAoJHtpLmdldFR5cGUoKS5nZXRUZXh0KCl9KSwgZnFuID0gJHtmbXhJbnRlcmZhY2UuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuXG4gICAgcHJvY2Vzc0NvbW1lbnRzKGksIGZteEludGVyZmFjZSk7XG5cbiAgICBwcm9jZXNzU3RydWN0dXJlZFR5cGUoaSwgZm14SW50ZXJmYWNlKTtcblxuICAgIHJldHVybiBmbXhJbnRlcmZhY2U7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIHRoZSB0eXBlIHBhcmFtZXRlcnMsIHByb3BlcnRpZXMgYW5kIG1ldGhvZHMgb2YgYSBzdHJ1Y3R1cmVkIHR5cGVcbiAqIEBwYXJhbSBjIEEgc3RydWN0dXJlZCB0eXBlIChhIGNsYXNzIG9yIGFuIGludGVyZmFjZSlcbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHN0cnVjdHVyZWQgdHlwZVxuICovXG5mdW5jdGlvbiBwcm9jZXNzU3RydWN0dXJlZFR5cGUoYzogQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uLCBmbXhTY29wZTogRmFtaXguQ2xhc3MgfCBGYW1peC5QYXJhbWV0cmljQ2xhc3MgfCBGYW1peC5JbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBGaW5kaW5nIFByb3BlcnRpZXMgYW5kIE1ldGhvZHM6YCk7XG4gICAgaWYgKGZteFNjb3BlIGluc3RhbmNlb2YgRmFtaXguUGFyYW1ldHJpY0NsYXNzIHx8IGZteFNjb3BlIGluc3RhbmNlb2YgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSkge1xuICAgICAgICBwcm9jZXNzVHlwZVBhcmFtZXRlcnMoYywgZm14U2NvcGUpO1xuICAgIH1cblxuICAgIGMuZ2V0UHJvcGVydGllcygpLmZvckVhY2gocHJvcCA9PiB7XG4gICAgICAgIGNvbnN0IGZteFByb3BlcnR5ID0gcHJvY2Vzc1Byb3BlcnR5KHByb3ApO1xuICAgICAgICBmbXhTY29wZS5hZGRQcm9wZXJ0eShmbXhQcm9wZXJ0eSk7XG4gICAgfSk7XG5cbiAgICBjLmdldE1ldGhvZHMoKS5mb3JFYWNoKG0gPT4ge1xuICAgICAgICBjb25zdCBmbXhNZXRob2QgPSBwcm9jZXNzTWV0aG9kKG0pO1xuICAgICAgICBmbXhTY29wZS5hZGRNZXRob2QoZm14TWV0aG9kKTtcbiAgICB9KTtcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgYSBwcm9wZXJ0eVxuICogQHBhcmFtIHAgQSBwcm9wZXJ0eVxuICogQHJldHVybnMgQSBGYW1peC5Qcm9wZXJ0eSByZXByZXNlbnRpbmcgdGhlIHByb3BlcnR5XG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NQcm9wZXJ0eShwOiBQcm9wZXJ0eURlY2xhcmF0aW9uIHwgUHJvcGVydHlTaWduYXR1cmUpOiBGYW1peC5Qcm9wZXJ0eSB7XG4gICAgY29uc3QgZm14UHJvcGVydHkgPSBlbnRpdHlEaWN0aW9uYXJ5LmNyZWF0ZUZhbWl4UHJvcGVydHkocCk7XG5cbiAgICBsb2dnZXIuZGVidWcoYHByb3BlcnR5OiAke3AuZ2V0TmFtZSgpfSwgKCR7cC5nZXRUeXBlKCkuZ2V0VGV4dCgpfSksIGZxbiA9ICR7Zm14UHJvcGVydHkuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuICAgIGxvZ2dlci5kZWJ1ZyhgIC0tLT4gSXQncyBhIFByb3BlcnR5JHsocCBpbnN0YW5jZW9mIFByb3BlcnR5U2lnbmF0dXJlKSA/IFwiU2lnbmF0dXJlXCIgOiBcIkRlY2xhcmF0aW9uXCJ9IWApO1xuICAgIGNvbnN0IGFuY2VzdG9yID0gcC5nZXRGaXJzdEFuY2VzdG9yT3JUaHJvdygpO1xuICAgIGxvZ2dlci5kZWJ1ZyhgIC0tLT4gSXRzIGZpcnN0IGFuY2VzdG9yIGlzIGEgJHthbmNlc3Rvci5nZXRLaW5kTmFtZSgpfWApO1xuXG4gICAgLy8gZGVjb3JhdG9yc1xuICAgIGlmICghKHAgaW5zdGFuY2VvZiBQcm9wZXJ0eVNpZ25hdHVyZSkpIHtcbiAgICAgICAgcHJvY2Vzc0RlY29yYXRvcnMocCwgZm14UHJvcGVydHkpO1xuICAgICAgICAvLyBvbmx5IGFkZCBhY2Nlc3MgaWYgdGhlIHAncyBmaXJzdCBhbmNlc3RvciBpcyBub3QgYSBQcm9wZXJ0eURlY2xhcmF0aW9uXG4gICAgICAgIGlmIChhbmNlc3Rvci5nZXRLaW5kTmFtZSgpICE9PSBcIlByb3BlcnR5RGVjbGFyYXRpb25cIikge1xuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBhZGRpbmcgYWNjZXNzIHRvIG1hcDogJHtwLmdldE5hbWUoKX0sICgke3AuZ2V0VHlwZSgpLmdldFRleHQoKX0pIEZhbWl4ICR7Zm14UHJvcGVydHkubmFtZX0gaWQ6ICR7Zm14UHJvcGVydHkuaWR9YCk7XG4gICAgICAgICAgICBhY2Nlc3NNYXAuc2V0KGZteFByb3BlcnR5LmlkLCBwKTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIHByb2Nlc3NDb21tZW50cyhwLCBmbXhQcm9wZXJ0eSk7XG5cbiAgICByZXR1cm4gZm14UHJvcGVydHk7XG59XG5cbi8qKlxuICAgICAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciBhIG1ldGhvZCBvciBhbiBhY2Nlc3NvclxuICAgICAqIEBwYXJhbSBtIEEgbWV0aG9kIG9yIGFuIGFjY2Vzc29yXG4gICAgICogQHJldHVybnMgQSBGYW1peC5NZXRob2Qgb3IgYSBGYW1peC5BY2Nlc3NvciByZXByZXNlbnRpbmcgdGhlIG1ldGhvZCBvciB0aGUgYWNjZXNzb3JcbiAgICAgKi9cbmZ1bmN0aW9uIHByb2Nlc3NNZXRob2QobTogTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24pOiBGYW1peC5NZXRob2QgfCBGYW1peC5BY2Nlc3NvciB7XG4gICAgY29uc3QgZm14TWV0aG9kID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVPckdldEZhbWl4TWV0aG9kKG0sIGN1cnJlbnRDQyk7XG5cbiAgICBsb2dnZXIuZGVidWcoYE1ldGhvZDogJHshKG0gaW5zdGFuY2VvZiBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uKSA/IG0uZ2V0TmFtZSgpIDogXCJjb25zdHJ1Y3RvclwifSwgKCR7bS5nZXRUeXBlKCkuZ2V0VGV4dCgpfSksIHBhcmVudDogJHsobS5nZXRQYXJlbnQoKSBhcyBDbGFzc0RlY2xhcmF0aW9uIHwgSW50ZXJmYWNlRGVjbGFyYXRpb24pLmdldE5hbWUoKX0sIGZxbiA9ICR7Zm14TWV0aG9kLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcblxuICAgIHByb2Nlc3NDb21tZW50cyhtLCBmbXhNZXRob2QpO1xuXG4gICAgcHJvY2Vzc1R5cGVQYXJhbWV0ZXJzKG0sIGZteE1ldGhvZCk7XG5cbiAgICBwcm9jZXNzUGFyYW1ldGVycyhtLCBmbXhNZXRob2QpO1xuXG4gICAgaWYgKCEobSBpbnN0YW5jZW9mIE1ldGhvZFNpZ25hdHVyZSkpIHtcbiAgICAgICAgcHJvY2Vzc0FsaWFzZXMobSwgZm14TWV0aG9kKTtcblxuICAgICAgICBwcm9jZXNzVmFyaWFibGVzKG0sIGZteE1ldGhvZCk7XG5cbiAgICAgICAgcHJvY2Vzc0VudW1zKG0sIGZteE1ldGhvZCk7XG5cbiAgICAgICAgcHJvY2Vzc0Z1bmN0aW9ucyhtLCBmbXhNZXRob2QpO1xuXG4gICAgICAgIHByb2Nlc3NGdW5jdGlvbkV4cHJlc3Npb25zKG0sIGZteE1ldGhvZCk7XG5cbiAgICAgICAgbWV0aG9kc0FuZEZ1bmN0aW9uc1dpdGhJZC5zZXQoZm14TWV0aG9kLmlkLCBtKTtcbiAgICB9XG5cbiAgICBpZiAobSBpbnN0YW5jZW9mIE1ldGhvZERlY2xhcmF0aW9uIHx8IG0gaW5zdGFuY2VvZiBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHx8IG0gaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKSB7XG4gICAgICAgIHByb2Nlc3NEZWNvcmF0b3JzKG0sIGZteE1ldGhvZCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGZteE1ldGhvZDtcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgYSBmdW5jdGlvblxuICogQHBhcmFtIGYgQSBmdW5jdGlvblxuICogQHJldHVybnMgQSBGYW1peC5GdW5jdGlvbiByZXByZXNlbnRpbmcgdGhlIGZ1bmN0aW9uXG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NGdW5jdGlvbihmOiBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uIHwgQXJyb3dGdW5jdGlvbik6IEZhbWl4LkZ1bmN0aW9uIHtcblxuICAgIGxvZ2dlci5kZWJ1ZyhgRnVuY3Rpb246ICR7KGYgaW5zdGFuY2VvZiBBcnJvd0Z1bmN0aW9uID8gXCJhbm9ueW1vdXNcIiA6IGYuZ2V0TmFtZSgpID8gZi5nZXROYW1lKCkgOiBcImFub255bW91c1wiKX0sICgke2YuZ2V0VHlwZSgpLmdldFRleHQoKX0pLCBmcW4gPSAke2dldEZRTihmKX1gKTtcblxuICAgIGxldCBmbXhGdW5jdGlvbjtcbiAgICBpZiggZiBpbnN0YW5jZW9mIEFycm93RnVuY3Rpb24pIHtcbiAgICAgICAgZm14RnVuY3Rpb24gPSBlbnRpdHlEaWN0aW9uYXJ5LmNyZWF0ZUZhbWl4QXJyb3dGdW5jdGlvbihmLCBjdXJyZW50Q0MpO1xuICAgIH0gZWxzZSB7XG4gICAgICAgIGZteEZ1bmN0aW9uID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVPckdldEZhbWl4RnVuY3Rpb24oZiwgY3VycmVudENDKTtcbiAgICB9XG5cbiAgICBwcm9jZXNzQ29tbWVudHMoZiwgZm14RnVuY3Rpb24pO1xuXG4gICAgcHJvY2Vzc0FsaWFzZXMoZiwgZm14RnVuY3Rpb24pO1xuXG4gICAgcHJvY2Vzc1R5cGVQYXJhbWV0ZXJzKGYsIGZteEZ1bmN0aW9uKTtcblxuICAgIHByb2Nlc3NQYXJhbWV0ZXJzKGYsIGZteEZ1bmN0aW9uKTtcblxuICAgIHByb2Nlc3NWYXJpYWJsZXMoZiwgZm14RnVuY3Rpb24pO1xuXG4gICAgcHJvY2Vzc0VudW1zKGYsIGZteEZ1bmN0aW9uKTtcblxuICAgIHByb2Nlc3NGdW5jdGlvbnMoZiwgZm14RnVuY3Rpb24pO1xuXG4gICAgaWYgKGYgaW5zdGFuY2VvZiBGdW5jdGlvbkRlY2xhcmF0aW9uICYmICEoZi5nZXRQYXJlbnQoKSBpbnN0YW5jZW9mIEJsb2NrKSkge1xuICAgICAgICBwcm9jZXNzRnVuY3Rpb25FeHByZXNzaW9ucyhmLCBmbXhGdW5jdGlvbik7XG4gICAgfVxuXG4gICAgbWV0aG9kc0FuZEZ1bmN0aW9uc1dpdGhJZC5zZXQoZm14RnVuY3Rpb24uaWQsIGYpO1xuXG4gICAgcmV0dXJuIGZteEZ1bmN0aW9uO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgZnVuY3Rpb24gZXhwcmVzc2lvbnMgb2YgYSBmdW5jdGlvbiBvciBhIG1ldGhvZFxuICogQHBhcmFtIGYgQSBmdW5jdGlvbiBvciBhIG1ldGhvZFxuICogQHBhcmFtIGZteFNjb3BlIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgZnVuY3Rpb24gb3IgdGhlIG1ldGhvZFxuICovXG5mdW5jdGlvbiBwcm9jZXNzRnVuY3Rpb25FeHByZXNzaW9ucyhmOiBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24sIGZteFNjb3BlOiBGYW1peC5GdW5jdGlvbiB8IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LkFjY2Vzc29yKTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBGaW5kaW5nIEZ1bmN0aW9uIEV4cHJlc3Npb25zOmApO1xuICAgIGNvbnN0IGZ1bmN0aW9uRXhwcmVzc2lvbnMgPSBmLmdldERlc2NlbmRhbnRzT2ZLaW5kKFN5bnRheEtpbmQuRnVuY3Rpb25FeHByZXNzaW9uKTtcbiAgICBmdW5jdGlvbkV4cHJlc3Npb25zLmZvckVhY2goKGZ1bmMpID0+IHtcbiAgICAgICAgY29uc3QgZm14RnVuYyA9IHByb2Nlc3NGdW5jdGlvbihmdW5jKTtcbiAgICAgICAgZm14U2NvcGUuYWRkRnVuY3Rpb24oZm14RnVuYyk7XG4gICAgfSk7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIHRoZSBwYXJhbWV0ZXJzIG9mIGEgbWV0aG9kIG9yIGEgZnVuY3Rpb25cbiAqIEBwYXJhbSBtIEEgbWV0aG9kIG9yIGEgZnVuY3Rpb25cbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIG1ldGhvZCBvciB0aGUgZnVuY3Rpb25cbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc1BhcmFtZXRlcnMobTogTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uIHwgQXJyb3dGdW5jdGlvbiwgZm14U2NvcGU6IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LkFjY2Vzc29yIHwgRmFtaXguRnVuY3Rpb24pOiB2b2lkIHtcbiAgICBsb2dnZXIuZGVidWcoYEZpbmRpbmcgUGFyYW1ldGVyczpgKTtcbiAgICBtLmdldFBhcmFtZXRlcnMoKS5mb3JFYWNoKHBhcmFtID0+IHtcbiAgICAgICAgY29uc3QgZm14UGFyYW0gPSBwcm9jZXNzUGFyYW1ldGVyKHBhcmFtKTtcbiAgICAgICAgZm14U2NvcGUuYWRkUGFyYW1ldGVyKGZteFBhcmFtKTtcbiAgICAgICAgLy8gQWRkaXRpb25hbCBoYW5kbGluZyBmb3IgUGFyYW1ldGVyIFByb3BlcnRpZXMgaW4gY29uc3RydWN0b3JzXG4gICAgICAgIGlmIChtIGluc3RhbmNlb2YgQ29uc3RydWN0b3JEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgLy8gQ2hlY2sgaWYgdGhlIHBhcmFtZXRlciBoYXMgYW55IHZpc2liaWxpdHkgbW9kaWZpZXJcbiAgICAgICAgICAgIGlmIChwYXJhbS5oYXNNb2RpZmllcihTeW50YXhLaW5kLlByaXZhdGVLZXl3b3JkKSB8fCBwYXJhbS5oYXNNb2RpZmllcihTeW50YXhLaW5kLlB1YmxpY0tleXdvcmQpIHx8IHBhcmFtLmhhc01vZGlmaWVyKFN5bnRheEtpbmQuUHJvdGVjdGVkS2V5d29yZCkgfHwgcGFyYW0uaGFzTW9kaWZpZXIoU3ludGF4S2luZC5SZWFkb25seUtleXdvcmQpKSB7XG4gICAgICAgICAgICAgICAgY29uc3QgY2xhc3NPZkNvbnN0cnVjdG9yID0gbS5nZXRQYXJlbnQoKTtcbiAgICAgICAgICAgICAgICBsb2dnZXIuaW5mbyhgUGFyYW1ldGVyIFByb3BlcnR5ICR7cGFyYW0uZ2V0TmFtZSgpfSBpbiBjb25zdHJ1Y3RvciBvZiAke2NsYXNzT2ZDb25zdHJ1Y3Rvci5nZXROYW1lKCl9LmApO1xuICAgICAgICAgICAgICAgIC8vIFRyZWF0IHRoZSBwYXJhbWV0ZXIgYXMgYSBwcm9wZXJ0eSBhbmQgYWRkIGl0IHRvIHRoZSBjbGFzc1xuICAgICAgICAgICAgICAgIGNvbnN0IGZteFByb3BlcnR5ID0gcHJvY2Vzc1BhcmFtZXRlckFzUHJvcGVydHkocGFyYW0sIGNsYXNzT2ZDb25zdHJ1Y3Rvcik7XG4gICAgICAgICAgICAgICAgZm14UHJvcGVydHkucmVhZE9ubHkgPSBwYXJhbS5oYXNNb2RpZmllcihTeW50YXhLaW5kLlJlYWRvbmx5S2V5d29yZCk7XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgIH0pO1xufVxuXG4vLyBUaGlzIGZ1bmN0aW9uIHNob3VsZCBjcmVhdGUgYSBGYW1peC5Qcm9wZXJ0eSBtb2RlbCBmcm9tIGEgUGFyYW1ldGVyRGVjbGFyYXRpb25cbi8vIFlvdSdsbCBuZWVkIHRvIGltcGxlbWVudCBpdCBhY2NvcmRpbmcgdG8geW91ciBGYW1peCBtb2RlbCBzdHJ1Y3R1cmVcbmZ1bmN0aW9uIHByb2Nlc3NQYXJhbWV0ZXJBc1Byb3BlcnR5KHBhcmFtOiBQYXJhbWV0ZXJEZWNsYXJhdGlvbiwgY2xhc3NEZWNsOiBDbGFzc0RlY2xhcmF0aW9uIHwgQ2xhc3NFeHByZXNzaW9uKTogRmFtaXguUHJvcGVydHkge1xuICAgIC8vIENvbnZlcnQgdGhlIHBhcmFtZXRlciBpbnRvIGEgUHJvcGVydHlcbiAgICBjb25zdCBwcm9wZXJ0eVJlcHJlc2VudGF0aW9uID0gY29udmVydFBhcmFtZXRlclRvUHJvcGVydHlSZXByZXNlbnRhdGlvbihwYXJhbSk7XG5cbiAgICAvLyBBZGQgdGhlIHByb3BlcnR5IHRvIHRoZSBjbGFzcyBzbyB3ZSBjYW4gaGF2ZSBhIFByb3BlcnR5RGVjbGFyYXRpb24gb2JqZWN0XG4gICAgY2xhc3NEZWNsLmFkZFByb3BlcnR5KHByb3BlcnR5UmVwcmVzZW50YXRpb24pO1xuXG4gICAgY29uc3QgcHJvcGVydHkgPSBjbGFzc0RlY2wuZ2V0UHJvcGVydHkocHJvcGVydHlSZXByZXNlbnRhdGlvbi5uYW1lKTtcbiAgICBpZiAoIXByb3BlcnR5KSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgUHJvcGVydHkgJHtwcm9wZXJ0eVJlcHJlc2VudGF0aW9uLm5hbWV9IG5vdCBmb3VuZCBpbiBjbGFzcyAke2NsYXNzRGVjbC5nZXROYW1lKCl9YCk7XG4gICAgfVxuICAgIGNvbnN0IGZteFByb3BlcnR5ID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peFByb3BlcnR5KHByb3BlcnR5KTtcbiAgICBpZiAoY2xhc3NEZWNsIGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbikge1xuICAgICAgICBjb25zdCBmbXhDbGFzcyA9IGVudGl0eURpY3Rpb25hcnkuY3JlYXRlT3JHZXRGYW1peENsYXNzKGNsYXNzRGVjbCk7XG4gICAgICAgIGZteENsYXNzLmFkZFByb3BlcnR5KGZteFByb3BlcnR5KTtcbiAgICB9IGVsc2UgeyBcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKFwiVW5leHBlY3RlZCB0eXBlIENsYXNzRXhwcmVzc2lvbi5cIik7XG4gICAgfVxuXG4gICAgcHJvY2Vzc0NvbW1lbnRzKHByb3BlcnR5LCBmbXhQcm9wZXJ0eSk7XG5cbiAgICAvLyByZW1vdmUgdGhlIHByb3BlcnR5IGZyb20gdGhlIGNsYXNzXG4gICAgcHJvcGVydHkucmVtb3ZlKCk7XG5cbiAgICByZXR1cm4gZm14UHJvcGVydHk7XG5cbn1cblxuZnVuY3Rpb24gY29udmVydFBhcmFtZXRlclRvUHJvcGVydHlSZXByZXNlbnRhdGlvbihwYXJhbTogUGFyYW1ldGVyRGVjbGFyYXRpb24pIHtcbiAgICAvLyBFeHRyYWN0IG5hbWVcbiAgICBjb25zdCBwYXJhbU5hbWUgPSBwYXJhbS5nZXROYW1lKCk7XG5cbiAgICAvLyBFeHRyYWN0IHR5cGVcbiAgICBjb25zdCBwYXJhbVR5cGUgPSBwYXJhbS5nZXRUeXBlKCkuZ2V0VGV4dChwYXJhbSk7XG5cbiAgICAvLyBEZXRlcm1pbmUgdmlzaWJpbGl0eVxuICAgIGxldCBzY29wZTogU2NvcGU7XG4gICAgaWYgKHBhcmFtLmhhc01vZGlmaWVyKFN5bnRheEtpbmQuUHJpdmF0ZUtleXdvcmQpKSB7XG4gICAgICAgIHNjb3BlID0gU2NvcGUuUHJpdmF0ZTtcbiAgICB9IGVsc2UgaWYgKHBhcmFtLmhhc01vZGlmaWVyKFN5bnRheEtpbmQuUHJvdGVjdGVkS2V5d29yZCkpIHtcbiAgICAgICAgc2NvcGUgPSBTY29wZS5Qcm90ZWN0ZWQ7XG4gICAgfSBlbHNlIGlmIChwYXJhbS5oYXNNb2RpZmllcihTeW50YXhLaW5kLlB1YmxpY0tleXdvcmQpKSB7XG4gICAgICAgIHNjb3BlID0gU2NvcGUuUHVibGljO1xuICAgIH0gZWxzZSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgUGFyYW1ldGVyIHByb3BlcnR5ICR7cGFyYW1OYW1lfSBpbiBjb25zdHJ1Y3RvciBkb2VzIG5vdCBoYXZlIGEgdmlzaWJpbGl0eSBtb2RpZmllci5gKTtcbiAgICB9XG5cbiAgICAvLyBEZXRlcm1pbmUgaWYgcmVhZG9ubHlcbiAgICBjb25zdCBpc1JlYWRvbmx5ID0gcGFyYW0uaGFzTW9kaWZpZXIoU3ludGF4S2luZC5SZWFkb25seUtleXdvcmQpO1xuXG4gICAgLy8gQ3JlYXRlIGEgcmVwcmVzZW50YXRpb24gb2YgdGhlIHByb3BlcnR5XG4gICAgY29uc3QgcHJvcGVydHlSZXByZXNlbnRhdGlvbiA9IHtcbiAgICAgICAgbmFtZTogcGFyYW1OYW1lLFxuICAgICAgICB0eXBlOiBwYXJhbVR5cGUsXG4gICAgICAgIHNjb3BlOiBzY29wZSxcbiAgICAgICAgaXNSZWFkb25seTogaXNSZWFkb25seSxcbiAgICB9O1xuXG4gICAgcmV0dXJuIHByb3BlcnR5UmVwcmVzZW50YXRpb247XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIGEgcGFyYW1ldGVyXG4gKiBAcGFyYW0gcGFyYW1EZWNsIEEgcGFyYW1ldGVyXG4gKiBAcmV0dXJucyBBIEZhbWl4LlBhcmFtZXRlciByZXByZXNlbnRpbmcgdGhlIHBhcmFtZXRlclxuICovXG5mdW5jdGlvbiBwcm9jZXNzUGFyYW1ldGVyKHBhcmFtRGVjbDogUGFyYW1ldGVyRGVjbGFyYXRpb24pOiBGYW1peC5QYXJhbWV0ZXIge1xuICAgIGNvbnN0IGZteFBhcmFtID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peFBhcmFtZXRlcihwYXJhbURlY2wpO1xuXG4gICAgbG9nZ2VyLmRlYnVnKGBwYXJhbWV0ZXI6ICR7cGFyYW1EZWNsLmdldE5hbWUoKX0sICgke3BhcmFtRGVjbC5nZXRUeXBlKCkuZ2V0VGV4dCgpfSksIGZxbiA9ICR7Zm14UGFyYW0uZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuXG4gICAgcHJvY2Vzc0NvbW1lbnRzKHBhcmFtRGVjbCwgZm14UGFyYW0pO1xuXG4gICAgcHJvY2Vzc0RlY29yYXRvcnMocGFyYW1EZWNsLCBmbXhQYXJhbSk7XG5cbiAgICBjb25zdCBwYXJlbnQgPSBwYXJhbURlY2wuZ2V0UGFyZW50KCk7XG5cbiAgICBpZiAoIShwYXJlbnQgaW5zdGFuY2VvZiBNZXRob2RTaWduYXR1cmUpKSB7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgYWRkaW5nIGFjY2VzczogJHtwYXJhbURlY2wuZ2V0TmFtZSgpfSwgKCR7cGFyYW1EZWNsLmdldFR5cGUoKS5nZXRUZXh0KCl9KSBGYW1peCAke2ZteFBhcmFtLm5hbWV9YCk7XG4gICAgICAgIGFjY2Vzc01hcC5zZXQoZm14UGFyYW0uaWQsIHBhcmFtRGVjbCk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGZteFBhcmFtO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgdHlwZSBwYXJhbWV0ZXJzIG9mIGEgY2xhc3MsIGFuIGludGVyZmFjZSwgYSBtZXRob2Qgb3IgYSBmdW5jdGlvblxuICogQHBhcmFtIGUgQSBjbGFzcywgYW4gaW50ZXJmYWNlLCBhIG1ldGhvZCBvciBhIGZ1bmN0aW9uXG4gKiBAcGFyYW0gZm14U2NvcGUgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjbGFzcywgdGhlIGludGVyZmFjZSwgdGhlIG1ldGhvZCBvciB0aGUgZnVuY3Rpb25cbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc1R5cGVQYXJhbWV0ZXJzKGU6IENsYXNzRGVjbGFyYXRpb24gfCBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uIHwgQ29uc3RydWN0b3JEZWNsYXJhdGlvbiB8IE1ldGhvZFNpZ25hdHVyZSB8IEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgRnVuY3Rpb25EZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRXhwcmVzc2lvbiB8QXJyb3dGdW5jdGlvbiwgZm14U2NvcGU6IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UgfCBGYW1peC5NZXRob2QgfCBGYW1peC5BY2Nlc3NvciB8IEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguQXJyb3dGdW5jdGlvbik6IHZvaWQge1xuICAgIGxvZ2dlci5kZWJ1ZyhgRmluZGluZyBUeXBlIFBhcmFtZXRlcnM6YCk7XG4gICAgZS5nZXRUeXBlUGFyYW1ldGVycygpLmZvckVhY2godHAgPT4ge1xuICAgICAgICBjb25zdCBmbXhQYXJhbSA9IHByb2Nlc3NUeXBlUGFyYW1ldGVyKHRwKTtcbiAgICAgICAgZm14U2NvcGUuYWRkR2VuZXJpY1BhcmFtZXRlcihmbXhQYXJhbSk7XG4gICAgfSk7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIGEgdHlwZSBwYXJhbWV0ZXJcbiAqIEBwYXJhbSB0cCBBIHR5cGUgcGFyYW1ldGVyXG4gKiBAcmV0dXJucyBBIEZhbWl4LlR5cGVQYXJhbWV0ZXIgcmVwcmVzZW50aW5nIHRoZSB0eXBlIHBhcmFtZXRlclxuICovXG5mdW5jdGlvbiBwcm9jZXNzVHlwZVBhcmFtZXRlcih0cDogVHlwZVBhcmFtZXRlckRlY2xhcmF0aW9uKTogRmFtaXguUGFyYW1ldGVyVHlwZSB7XG4gICAgY29uc3QgZm14VHlwZVBhcmFtZXRlciA9IGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhQYXJhbWV0ZXJUeXBlKHRwKTtcblxuICAgIGxvZ2dlci5kZWJ1ZyhgdHlwZSBwYXJhbWV0ZXI6ICR7dHAuZ2V0TmFtZSgpfSwgKCR7dHAuZ2V0VHlwZSgpLmdldFRleHQoKX0pLCBmcW4gPSAke2ZteFR5cGVQYXJhbWV0ZXIuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuXG4gICAgcHJvY2Vzc0NvbW1lbnRzKHRwLCBmbXhUeXBlUGFyYW1ldGVyKTtcblxuICAgIHJldHVybiBmbXhUeXBlUGFyYW1ldGVyO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgdmFyaWFibGVzIG9mIGEgdmFyaWFibGUgc3RhdGVtZW50XG4gKiBAcGFyYW0gdiBBIHZhcmlhYmxlIHN0YXRlbWVudFxuICogQHJldHVybnMgQW4gYXJyYXkgb2YgRmFtaXguVmFyaWFibGUgcmVwcmVzZW50aW5nIHRoZSB2YXJpYWJsZXNcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc1ZhcmlhYmxlU3RhdGVtZW50KHY6IFZhcmlhYmxlU3RhdGVtZW50KTogQXJyYXk8RmFtaXguVmFyaWFibGU+IHtcbiAgICBjb25zdCBmbXhWYXJpYWJsZXMgPSBuZXcgQXJyYXk8RmFtaXguVmFyaWFibGU+KCk7XG5cbiAgICBsb2dnZXIuZGVidWcoYFZhcmlhYmxlIHN0YXRlbWVudDogJHt2LmdldFRleHQoKX0sICgke3YuZ2V0VHlwZSgpLmdldFRleHQoKX0pLCAke3YuZ2V0RGVjbGFyYXRpb25LaW5kS2V5d29yZHMoKVswXX0sIGZxbiA9ICR7di5nZXREZWNsYXJhdGlvbnMoKVswXS5nZXROYW1lKCl9YCk7XG5cbiAgICB2LmdldERlY2xhcmF0aW9ucygpLmZvckVhY2godmFyaWFibGUgPT4ge1xuICAgICAgICBjb25zdCBmbXhWYXIgPSBwcm9jZXNzVmFyaWFibGUodmFyaWFibGUpO1xuICAgICAgICBwcm9jZXNzQ29tbWVudHModiwgZm14VmFyKTtcbiAgICAgICAgZm14VmFyaWFibGVzLnB1c2goZm14VmFyKTtcbiAgICB9KTsgXG5cbiAgICByZXR1cm4gZm14VmFyaWFibGVzO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciBhIHZhcmlhYmxlXG4gKiBAcGFyYW0gdiBBIHZhcmlhYmxlXG4gKiBAcmV0dXJucyBBIEZhbWl4LlZhcmlhYmxlIHJlcHJlc2VudGluZyB0aGUgdmFyaWFibGVcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc1ZhcmlhYmxlKHY6IFZhcmlhYmxlRGVjbGFyYXRpb24pOiBGYW1peC5WYXJpYWJsZSB7XG4gICAgY29uc3QgZm14VmFyID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peFZhcmlhYmxlKHYpO1xuXG4gICAgbG9nZ2VyLmRlYnVnKGB2YXJpYWJsZTogJHt2LmdldE5hbWUoKX0sICgke3YuZ2V0VHlwZSgpLmdldFRleHQoKX0pLCAke3YuZ2V0SW5pdGlhbGl6ZXIoKSA/IFwiaW5pdGlhbGl6ZXI6IFwiICsgdi5nZXRJbml0aWFsaXplcigpIS5nZXRUZXh0KCkgOiBcImluaXRpYWxpemVyOiBcIn0sIGZxbiA9ICR7Zm14VmFyLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcblxuICAgIHByb2Nlc3NDb21tZW50cyh2LCBmbXhWYXIpO1xuXG4gICAgbG9nZ2VyLmRlYnVnKGBhZGRpbmcgYWNjZXNzOiAke3YuZ2V0TmFtZSgpfSwgKCR7di5nZXRUeXBlKCkuZ2V0VGV4dCgpfSkgRmFtaXggJHtmbXhWYXIubmFtZX1gKTtcbiAgICBhY2Nlc3NNYXAuc2V0KGZteFZhci5pZCwgdik7XG5cbiAgICByZXR1cm4gZm14VmFyO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciBhbiBlbnVtXG4gKiBAcGFyYW0gZSBBbiBlbnVtXG4gKiBAcmV0dXJucyBBIEZhbWl4LkVudW0gcmVwcmVzZW50aW5nIHRoZSBlbnVtXG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NFbnVtKGU6IEVudW1EZWNsYXJhdGlvbik6IEZhbWl4LkVudW0ge1xuICAgIGNvbnN0IGZteEVudW0gPSBlbnRpdHlEaWN0aW9uYXJ5LmNyZWF0ZUZhbWl4RW51bShlKTtcblxuICAgIGxvZ2dlci5kZWJ1ZyhgZW51bTogJHtlLmdldE5hbWUoKX0sICgke2UuZ2V0VHlwZSgpLmdldFRleHQoKX0pLCBmcW4gPSAke2ZteEVudW0uZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuXG4gICAgcHJvY2Vzc0NvbW1lbnRzKGUsIGZteEVudW0pO1xuXG4gICAgZS5nZXRNZW1iZXJzKCkuZm9yRWFjaChtID0+IHtcbiAgICAgICAgY29uc3QgZm14RW51bVZhbHVlID0gcHJvY2Vzc0VudW1WYWx1ZShtKTtcbiAgICAgICAgZm14RW51bS5hZGRWYWx1ZShmbXhFbnVtVmFsdWUpO1xuICAgIH0pO1xuXG4gICAgcmV0dXJuIGZteEVudW07XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIGFuIGVudW0gbWVtYmVyXG4gKiBAcGFyYW0gdiBBbiBlbnVtIG1lbWJlclxuICogQHJldHVybnMgQSBGYW1peC5FbnVtVmFsdWUgcmVwcmVzZW50aW5nIHRoZSBlbnVtIG1lbWJlclxuICovXG5mdW5jdGlvbiBwcm9jZXNzRW51bVZhbHVlKHY6IEVudW1NZW1iZXIpOiBGYW1peC5FbnVtVmFsdWUge1xuICAgIGNvbnN0IGZteEVudW1WYWx1ZSA9IGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhFbnVtVmFsdWUodik7XG5cbiAgICBsb2dnZXIuZGVidWcoYGVudW0gdmFsdWU6ICR7di5nZXROYW1lKCl9LCAoJHt2LmdldFR5cGUoKS5nZXRUZXh0KCl9KSwgZnFuID0gJHtmbXhFbnVtVmFsdWUuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuXG4gICAgcHJvY2Vzc0NvbW1lbnRzKHYsIGZteEVudW1WYWx1ZSk7XG5cbiAgICBsb2dnZXIuZGVidWcoYGFkZGluZyBhY2Nlc3M6ICR7di5nZXROYW1lKCl9LCAoJHt2LmdldFR5cGUoKS5nZXRUZXh0KCl9KSBGYW1peCAke2ZteEVudW1WYWx1ZS5uYW1lfWApO1xuICAgIGFjY2Vzc01hcC5zZXQoZm14RW51bVZhbHVlLmlkLCB2KTtcblxuICAgIHJldHVybiBmbXhFbnVtVmFsdWU7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIHRoZSBkZWNvcmF0b3JzIG9mIGEgY2xhc3MsIGEgbWV0aG9kLCBhIHBhcmFtZXRlciBvciBhIHByb3BlcnR5XG4gKiBAcGFyYW0gZSBBIGNsYXNzLCBhIG1ldGhvZCwgYSBwYXJhbWV0ZXIgb3IgYSBwcm9wZXJ0eVxuICogQHBhcmFtIGZteFNjb3BlIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgY2xhc3MsIHRoZSBtZXRob2QsIHRoZSBwYXJhbWV0ZXIgb3IgdGhlIHByb3BlcnR5XG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NEZWNvcmF0b3JzKGU6IENsYXNzRGVjbGFyYXRpb24gfCBNZXRob2REZWNsYXJhdGlvbiB8IEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgUGFyYW1ldGVyRGVjbGFyYXRpb24gfCBQcm9wZXJ0eURlY2xhcmF0aW9uLCBmbXhTY29wZTogRmFtaXguQ2xhc3MgfCBGYW1peC5QYXJhbWV0cmljQ2xhc3MgfCBGYW1peC5NZXRob2QgfCBGYW1peC5BY2Nlc3NvciB8IEZhbWl4LlBhcmFtZXRlciB8IEZhbWl4LlByb3BlcnR5KTogdm9pZCB7XG4gICAgbG9nZ2VyLmRlYnVnKGBGaW5kaW5nIERlY29yYXRvcnM6YCk7XG4gICAgZS5nZXREZWNvcmF0b3JzKCkuZm9yRWFjaChkZWMgPT4ge1xuICAgICAgICBjb25zdCBmbXhEZWMgPSBwcm9jZXNzRGVjb3JhdG9yKGRlYywgZSk7XG4gICAgICAgIGZteFNjb3BlLmFkZERlY29yYXRvcihmbXhEZWMpO1xuICAgIH0pO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciBhIGRlY29yYXRvclxuICogQHBhcmFtIGQgQSBkZWNvcmF0b3JcbiAqIEBwYXJhbSBlIEEgY2xhc3MsIGEgbWV0aG9kLCBhIHBhcmFtZXRlciBvciBhIHByb3BlcnR5XG4gKiBAcmV0dXJucyBBIEZhbWl4LkRlY29yYXRvciByZXByZXNlbnRpbmcgdGhlIGRlY29yYXRvclxuICovXG5mdW5jdGlvbiBwcm9jZXNzRGVjb3JhdG9yKGQ6IERlY29yYXRvciwgZTogQ2xhc3NEZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFByb3BlcnR5RGVjbGFyYXRpb24pOiBGYW1peC5EZWNvcmF0b3Ige1xuICAgIGNvbnN0IGZteERlYyA9IGVudGl0eURpY3Rpb25hcnkuY3JlYXRlT3JHZXRGYW1peERlY29yYXRvcihkLCBlKTtcblxuICAgIGxvZ2dlci5kZWJ1ZyhgZGVjb3JhdG9yOiAke2QuZ2V0TmFtZSgpfSwgKCR7ZC5nZXRUeXBlKCkuZ2V0VGV4dCgpfSksIGZxbiA9ICR7Zm14RGVjLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcblxuICAgIHByb2Nlc3NDb21tZW50cyhkLCBmbXhEZWMpO1xuXG4gICAgcmV0dXJuIGZteERlYztcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgdGhlIGNvbW1lbnRzXG4gKiBAcGFyYW0gZSBBIHRzLW1vcnBoIGVsZW1lbnRcbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIG5hbWVkIGVudGl0eVxuICovXG5mdW5jdGlvbiBwcm9jZXNzQ29tbWVudHMoZTogU291cmNlRmlsZSB8IE1vZHVsZURlY2xhcmF0aW9uIHwgQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uIHwgTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uIHwgUGFyYW1ldGVyRGVjbGFyYXRpb24gfCBWYXJpYWJsZURlY2xhcmF0aW9uIHwgUHJvcGVydHlEZWNsYXJhdGlvbiB8IFByb3BlcnR5U2lnbmF0dXJlIHwgRGVjb3JhdG9yIHwgRW51bURlY2xhcmF0aW9uIHwgRW51bU1lbWJlciB8IFR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFZhcmlhYmxlU3RhdGVtZW50IHwgVHlwZUFsaWFzRGVjbGFyYXRpb24gfCBBcnJvd0Z1bmN0aW9uLCBmbXhTY29wZTogRmFtaXguTmFtZWRFbnRpdHkpOiB2b2lkIHtcbiAgICBsb2dnZXIuZGVidWcoYFByb2Nlc3MgY29tbWVudHM6YCk7XG4gICAgZS5nZXRMZWFkaW5nQ29tbWVudFJhbmdlcygpLmZvckVhY2goYyA9PiB7XG4gICAgICAgIGNvbnN0IGZteENvbW1lbnQgPSBwcm9jZXNzQ29tbWVudChjLCBmbXhTY29wZSk7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgbGVhZGluZyBjb21tZW50cywgYWRkQ29tbWVudDogJyR7Yy5nZXRUZXh0KCl9J2ApO1xuICAgICAgICBmbXhTY29wZS5hZGRDb21tZW50KGZteENvbW1lbnQpOyAvLyByZWR1bmRhbnQsIGJ1dCBqdXN0IGluIGNhc2VcbiAgICB9KTtcbiAgICBlLmdldFRyYWlsaW5nQ29tbWVudFJhbmdlcygpLmZvckVhY2goYyA9PiB7XG4gICAgICAgIGNvbnN0IGZteENvbW1lbnQgPSBwcm9jZXNzQ29tbWVudChjLCBmbXhTY29wZSk7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgdHJhaWxpbmcgY29tbWVudHMsIGFkZENvbW1lbnQ6ICcke2MuZ2V0VGV4dCgpfSdgKTtcbiAgICAgICAgZm14U2NvcGUuYWRkQ29tbWVudChmbXhDb21tZW50KTtcbiAgICB9KTtcbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgYSBjb21tZW50XG4gKiBAcGFyYW0gYyBBIGNvbW1lbnRcbiAqIEBwYXJhbSBmbXhTY29wZSBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGNvbW1lbnQncyBjb250YWluZXJcbiAqIEByZXR1cm5zIEEgRmFtaXguQ29tbWVudCByZXByZXNlbnRpbmcgdGhlIGNvbW1lbnRcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc0NvbW1lbnQoYzogQ29tbWVudFJhbmdlLCBmbXhTY29wZTogRmFtaXguTmFtZWRFbnRpdHkpOiBGYW1peC5Db21tZW50IHtcbiAgICBjb25zdCBpc0pTRG9jID0gYy5nZXRUZXh0KCkuc3RhcnRzV2l0aChcIi8qKlwiKTtcbiAgICBsb2dnZXIuZGVidWcoYHByb2Nlc3NDb21tZW50OiBjb21tZW50OiAke2MuZ2V0VGV4dCgpfSwgaXNKU0RvYyA9ICR7aXNKU0RvY31gKTtcbiAgICBjb25zdCBmbXhDb21tZW50ID0gZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peENvbW1lbnQoYywgZm14U2NvcGUsIGlzSlNEb2MpO1xuXG4gICAgcmV0dXJuIGZteENvbW1lbnQ7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIHRoZSBhY2Nlc3NlcyBvbiB0aGUgcGFyYW1ldGVycywgdmFyaWFibGVzLCBwcm9wZXJ0aWVzIGFuZCBlbnVtIG1lbWJlcnMgb2YgdGhlIHNvdXJjZSBmaWxlc1xuICogQHBhcmFtIGFjY2Vzc01hcCBBIG1hcCBvZiBwYXJhbWV0ZXJzLCB2YXJpYWJsZXMsIHByb3BlcnRpZXMgYW5kIGVudW0gbWVtYmVycyB3aXRoIHRoZWlyIGlkXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcm9jZXNzQWNjZXNzZXMoYWNjZXNzTWFwOiBNYXA8RmFtaXhJRCwgQWNjZXNzaWJsZVRTTW9ycGhFbGVtZW50Pik6IHZvaWQge1xuICAgIGxvZ2dlci5kZWJ1ZyhgQ3JlYXRpbmcgYWNjZXNzZXM6YCk7XG4gICAgYWNjZXNzTWFwLmZvckVhY2goKHYsIGlkKSA9PiB7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgQWNjZXNzZXMgdG8gJHt2LmdldE5hbWUoKX1gKTtcbiAgICAgICAgLy8gdHJ5IHtcbiAgICAgICAgICAgIGNvbnN0IHRlbXBfbm9kZXMgPSB2LmZpbmRSZWZlcmVuY2VzQXNOb2RlcygpIGFzIEFycmF5PElkZW50aWZpZXI+O1xuICAgICAgICAgICAgdGVtcF9ub2Rlcy5mb3JFYWNoKG5vZGUgPT4gcHJvY2Vzc05vZGVGb3JBY2Nlc3Nlcyhub2RlLCBpZCkpO1xuICAgICAgICAvLyB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAvLyAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gXCIke2Vycm9yfVwiLlxcbkNvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgLy8gfVxuICAgIH0pO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciBhbiBhY2Nlc3Mgb24gYSBwYXJhbWV0ZXIsIHZhcmlhYmxlLCBwcm9wZXJ0eSBvciBlbnVtIG1lbWJlclxuICogQHBhcmFtIG4gQSBub2RlXG4gKiBAcGFyYW0gaWQgQW4gaWQgb2YgYSBwYXJhbWV0ZXIsIGEgdmFyaWFibGUsIGEgcHJvcGVydHkgb3IgYW4gZW51bSBtZW1iZXJcbiAqL1xuZnVuY3Rpb24gcHJvY2Vzc05vZGVGb3JBY2Nlc3NlcyhuOiBJZGVudGlmaWVyLCBpZDogbnVtYmVyKTogdm9pZCB7XG4gICAgLy8gdHJ5IHtcbiAgICAgICAgLy8gc29tZXRpbWVzIG5vZGUncyBmaXJzdCBhbmNlc3RvciBpcyBhIFByb3BlcnR5RGVjbGFyYXRpb24sIHdoaWNoIGlzIG5vdCBhbiBhY2Nlc3NcbiAgICAgICAgLy8gc2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9mdWhybWFuYXRvci9GYW1peFR5cGVTY3JpcHRJbXBvcnRlci9pc3N1ZXMvOVxuICAgICAgICAvLyBjaGVjayBmb3IgYSBub2RlIHdob3NlIGZpcnN0IGFuY2VzdG9yIGlzIGEgcHJvcGVydHkgZGVjbGFyYXRpb24gYW5kIGJhaWw/XG4gICAgICAgIC8vIFRoaXMgbWF5IGJlIGEgYnVnIGluIHRzLW1vcnBoP1xuICAgICAgICBpZiAobi5nZXRGaXJzdEFuY2VzdG9yT3JUaHJvdygpLmdldEtpbmROYW1lKCkgPT09IFwiUHJvcGVydHlEZWNsYXJhdGlvblwiKSB7XG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoYHByb2Nlc3NOb2RlRm9yQWNjZXNzZXM6IG5vZGUga2luZDogJHtuLmdldEtpbmROYW1lKCl9LCAke24uZ2V0VGV4dCgpfSwgKCR7bi5nZXRUeXBlKCkuZ2V0VGV4dCgpfSkncyBmaXJzdCBhbmNlc3RvciBpcyBhIFByb3BlcnR5RGVjbGFyYXRpb24uIFNraXBwaW5nLi4uYCk7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICAgICAgZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peEFjY2VzcyhuLCBpZCk7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgcHJvY2Vzc05vZGVGb3JBY2Nlc3Nlczogbm9kZSBraW5kOiAke24uZ2V0S2luZE5hbWUoKX0sICR7bi5nZXRUZXh0KCl9LCAoJHtuLmdldFR5cGUoKS5nZXRUZXh0KCl9KWApO1xuICAgIC8vIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgLy8gICAgIGxvZ2dlci5lcnJvcihgPiBHb3QgZXhjZXB0aW9uIFwiJHtlcnJvcn1cIi5cXG5TY29wZURlY2xhcmF0aW9uIGludmFsaWQgZm9yIFwiJHtuLmdldFN5bWJvbCgpLmZ1bGx5UXVhbGlmaWVkTmFtZX1cIi5cXG5Db250aW51aW5nLi4uYCk7XG4gICAgLy8gfVxufVxuXG5cbi8vIGV4cG9ydHMgaGFzIG5hbWUgLT4gRGVjbGFyYXRpb24gLS0gdGhlIGRlY2xhcmF0aW9uIGNhbiBiZSB1c2VkIHRvIGZpbmQgdGhlIEZhbWl4RWxlbWVudFxuXG4vLyBoYW5kbGUgYGltcG9ydCBwYXRoID0gcmVxdWlyZShcInBhdGhcIilgIGZvciBleGFtcGxlXG5leHBvcnQgZnVuY3Rpb24gcHJvY2Vzc0ltcG9ydENsYXVzZXNGb3JJbXBvcnRFcXVhbHNEZWNsYXJhdGlvbnMoc291cmNlRmlsZXM6IEFycmF5PFNvdXJjZUZpbGU+LCBleHBvcnRzOiBBcnJheTxSZWFkb25seU1hcDxzdHJpbmcsIEV4cG9ydGVkRGVjbGFyYXRpb25zW10+Pik6IHZvaWQge1xuICAgIGxvZ2dlci5pbmZvKGBDcmVhdGluZyBpbXBvcnQgY2xhdXNlcyBmcm9tIEltcG9ydEVxdWFsc0RlY2xhcmF0aW9ucyBpbiBzb3VyY2UgZmlsZXM6YCk7XG4gICAgc291cmNlRmlsZXMuZm9yRWFjaChzb3VyY2VGaWxlID0+IHtcbiAgICAgICAgc291cmNlRmlsZS5mb3JFYWNoRGVzY2VuZGFudChub2RlID0+IHtcbiAgICAgICAgICAgIGlmIChOb2RlLmlzSW1wb3J0RXF1YWxzRGVjbGFyYXRpb24obm9kZSkpIHtcbiAgICAgICAgICAgICAgICAvLyBZb3UndmUgZm91bmQgYW4gSW1wb3J0RXF1YWxzRGVjbGFyYXRpb25cbiAgICAgICAgICAgICAgICBsb2dnZXIuaW5mbyhcIkRlY2xhcmF0aW9uIE5hbWU6XCIsIG5vZGUuZ2V0TmFtZSgpKTtcbiAgICAgICAgICAgICAgICBsb2dnZXIuaW5mbyhcIk1vZHVsZSBSZWZlcmVuY2UgVGV4dDpcIiwgbm9kZS5nZXRNb2R1bGVSZWZlcmVuY2UoKS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgICAgIC8vIHdoYXQncyB0aGUgbmFtZSBvZiB0aGUgaW1wb3J0ZWQgZW50aXR5P1xuICAgICAgICAgICAgICAgIC8vIGNvbnN0IGltcG9ydGVkRW50aXR5ID0gbm9kZS5nZXROYW1lKCk7XG4gICAgICAgICAgICAgICAgLy8gY3JlYXRlIGEgZmFtaXggaW1wb3J0IGNsYXVzZVxuICAgICAgICAgICAgICAgIGNvbnN0IG5hbWVkSW1wb3J0ID0gbm9kZS5nZXROYW1lTm9kZSgpO1xuICAgICAgICAgICAgICAgIGVudGl0eURpY3Rpb25hcnkub2xkQ3JlYXRlRmFtaXhJbXBvcnRDbGF1c2Uoe2ltcG9ydERlY2xhcmF0aW9uOiBub2RlLFxuICAgICAgICAgICAgICAgICAgICBpbXBvcnRlclNvdXJjZUZpbGU6IHNvdXJjZUZpbGUsIFxuICAgICAgICAgICAgICAgICAgICBtb2R1bGVTcGVjaWZpZXJGaWxlUGF0aDogbm9kZS5nZXRNb2R1bGVSZWZlcmVuY2UoKS5nZXRUZXh0KCksIFxuICAgICAgICAgICAgICAgICAgICBpbXBvcnRFbGVtZW50OiBuYW1lZEltcG9ydCwgXG4gICAgICAgICAgICAgICAgICAgIGlzSW5FeHBvcnRzOiBleHBvcnRzLmZpbmQoZSA9PiBlLmhhcyhuYW1lZEltcG9ydC5nZXRUZXh0KCkpKSAhPT0gdW5kZWZpbmVkLCBcbiAgICAgICAgICAgICAgICAgICAgaXNEZWZhdWx0RXhwb3J0OiBmYWxzZX0pO1xuICAgICAgICAgICAgICAgIC8vIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhJbXBvcnRDbGF1c2UoaW1wb3J0ZWRFbnRpdHksIGltcG9ydGluZ0VudGl0eSk7XG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgIH1cbiAgICApO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgaW1wb3J0IGNsYXVzZXMgb2YgdGhlIHNvdXJjZSBmaWxlcyB3aGljaCBhcmUgbW9kdWxlc1xuICogQHBhcmFtIG1vZHVsZXMgQW4gYXJyYXkgb2YgbW9kdWxlc1xuICogQHBhcmFtIGV4cG9ydHMgQW4gYXJyYXkgb2YgbWFwcyBvZiBleHBvcnRlZCBkZWNsYXJhdGlvbnNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByb2Nlc3NJbXBvcnRDbGF1c2VzRm9yTW9kdWxlcyhtb2R1bGVzOiBBcnJheTxTb3VyY2VGaWxlPiwgZXhwb3J0czogQXJyYXk8UmVhZG9ubHlNYXA8c3RyaW5nLCBFeHBvcnRlZERlY2xhcmF0aW9uc1tdPj4pOiB2b2lkIHtcbiAgICBsb2dnZXIuaW5mbyhgQ3JlYXRpbmcgaW1wb3J0IGNsYXVzZXMgZnJvbSAke21vZHVsZXMubGVuZ3RofSBtb2R1bGVzOmApO1xuICAgIG1vZHVsZXMuZm9yRWFjaChtb2R1bGUgPT4ge1xuICAgICAgICBjb25zdCBtb2R1bGVQYXRoID0gbW9kdWxlLmdldEZpbGVQYXRoKCkgKyBtb2R1bGUuZ2V0QmFzZU5hbWUoKTtcbiAgICAgICAgbW9kdWxlLmdldEltcG9ydERlY2xhcmF0aW9ucygpLmZvckVhY2goaW1wRGVjbCA9PiB7XG4gICAgICAgICAgICBsb2dnZXIuaW5mbyhgSW1wb3J0aW5nICR7aW1wRGVjbC5nZXRNb2R1bGVTcGVjaWZpZXJWYWx1ZSgpfSBpbiAke21vZHVsZVBhdGh9YCk7XG4gICAgICAgICAgICBjb25zdCBwYXRoID0gZ2V0TW9kdWxlUGF0aChpbXBEZWNsKTtcblxuICAgICAgICAgICAgaW1wRGVjbC5nZXROYW1lZEltcG9ydHMoKS5mb3JFYWNoKG5hbWVkSW1wb3J0ID0+IHtcbiAgICAgICAgICAgICAgICBsb2dnZXIuaW5mbyhgSW1wb3J0aW5nIChuYW1lZCkgJHtuYW1lZEltcG9ydC5nZXROYW1lKCl9IGZyb20gJHtpbXBEZWNsLmdldE1vZHVsZVNwZWNpZmllclZhbHVlKCl9IGluICR7bW9kdWxlUGF0aH1gKTtcbiAgICAgICAgICAgICAgICBjb25zdCBpbXBvcnRlZEVudGl0eU5hbWUgPSBuYW1lZEltcG9ydC5nZXROYW1lKCk7XG4gICAgICAgICAgICAgICAgbGV0IGltcG9ydEZvdW5kSW5FeHBvcnRzID0gaXNJbkV4cG9ydHMoZXhwb3J0cywgaW1wb3J0ZWRFbnRpdHlOYW1lKTtcbiAgICAgICAgICAgICAgICBlbnRpdHlEaWN0aW9uYXJ5Lm9sZENyZWF0ZUZhbWl4SW1wb3J0Q2xhdXNlKHtpbXBvcnREZWNsYXJhdGlvbjogaW1wRGVjbCxcbiAgICAgICAgICAgICAgICAgICAgaW1wb3J0ZXJTb3VyY2VGaWxlOiBtb2R1bGUsIFxuICAgICAgICAgICAgICAgICAgICBtb2R1bGVTcGVjaWZpZXJGaWxlUGF0aDogcGF0aCwgXG4gICAgICAgICAgICAgICAgICAgIGltcG9ydEVsZW1lbnQ6IG5hbWVkSW1wb3J0LCBcbiAgICAgICAgICAgICAgICAgICAgaXNJbkV4cG9ydHM6IGltcG9ydEZvdW5kSW5FeHBvcnRzLCBcbiAgICAgICAgICAgICAgICAgICAgaXNEZWZhdWx0RXhwb3J0OiBmYWxzZX0pO1xuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIGNvbnN0IGRlZmF1bHRJbXBvcnQgPSBpbXBEZWNsLmdldERlZmF1bHRJbXBvcnQoKTtcbiAgICAgICAgICAgIGlmIChkZWZhdWx0SW1wb3J0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBsb2dnZXIuaW5mbyhgSW1wb3J0aW5nIChkZWZhdWx0KSAke2RlZmF1bHRJbXBvcnQuZ2V0VGV4dCgpfSBmcm9tICR7aW1wRGVjbC5nZXRNb2R1bGVTcGVjaWZpZXJWYWx1ZSgpfSBpbiAke21vZHVsZVBhdGh9YCk7XG4gICAgICAgICAgICAgICAgLy8gY2FsbCB3aXRoIG1vZHVsZSwgaW1wRGVjbC5nZXRNb2R1bGVTcGVjaWZpZXJWYWx1ZSgpLCBwYXRoLCBkZWZhdWx0SW1wb3J0LCBmYWxzZSwgdHJ1ZVxuICAgICAgICAgICAgICAgIGVudGl0eURpY3Rpb25hcnkub2xkQ3JlYXRlRmFtaXhJbXBvcnRDbGF1c2Uoe2ltcG9ydERlY2xhcmF0aW9uOiBpbXBEZWNsLFxuICAgICAgICAgICAgICAgICAgICBpbXBvcnRlclNvdXJjZUZpbGU6IG1vZHVsZSxcbiAgICAgICAgICAgICAgICAgICAgbW9kdWxlU3BlY2lmaWVyRmlsZVBhdGg6IHBhdGgsXG4gICAgICAgICAgICAgICAgICAgIGltcG9ydEVsZW1lbnQ6IGRlZmF1bHRJbXBvcnQsXG4gICAgICAgICAgICAgICAgICAgIGlzSW5FeHBvcnRzOiBmYWxzZSxcbiAgICAgICAgICAgICAgICAgICAgaXNEZWZhdWx0RXhwb3J0OiB0cnVlfSk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGNvbnN0IG5hbWVzcGFjZUltcG9ydCA9IGltcERlY2wuZ2V0TmFtZXNwYWNlSW1wb3J0KCk7XG4gICAgICAgICAgICBpZiAobmFtZXNwYWNlSW1wb3J0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBsb2dnZXIuaW5mbyhgSW1wb3J0aW5nIChuYW1lc3BhY2UpICR7bmFtZXNwYWNlSW1wb3J0LmdldFRleHQoKX0gZnJvbSAke2ltcERlY2wuZ2V0TW9kdWxlU3BlY2lmaWVyVmFsdWUoKX0gaW4gJHttb2R1bGVQYXRofWApO1xuICAgICAgICAgICAgICAgIGVudGl0eURpY3Rpb25hcnkub2xkQ3JlYXRlRmFtaXhJbXBvcnRDbGF1c2Uoe2ltcG9ydERlY2xhcmF0aW9uOiBpbXBEZWNsLFxuICAgICAgICAgICAgICAgICAgICBpbXBvcnRlclNvdXJjZUZpbGU6IG1vZHVsZSwgXG4gICAgICAgICAgICAgICAgICAgIG1vZHVsZVNwZWNpZmllckZpbGVQYXRoOiBwYXRoLCBcbiAgICAgICAgICAgICAgICAgICAgaW1wb3J0RWxlbWVudDogbmFtZXNwYWNlSW1wb3J0LCBcbiAgICAgICAgICAgICAgICAgICAgaXNJbkV4cG9ydHM6IGZhbHNlLCBcbiAgICAgICAgICAgICAgICAgICAgaXNEZWZhdWx0RXhwb3J0OiBmYWxzZX0pO1xuICAgICAgICAgICAgICAgIC8vIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhJbXBvcnRDbGF1c2UobW9kdWxlLCBpbXBEZWNsLmdldE1vZHVsZVNwZWNpZmllclZhbHVlKCksIHBhdGgsIG5hbWVzcGFjZUltcG9ydCwgZmFsc2UsIGZhbHNlKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7IFxuICAgIH0pO1xufVxuXG5mdW5jdGlvbiBpc0luRXhwb3J0cyhleHBvcnRzOiBSZWFkb25seU1hcDxzdHJpbmcsIEV4cG9ydGVkRGVjbGFyYXRpb25zW10+W10sIGltcG9ydGVkRW50aXR5TmFtZTogc3RyaW5nKSB7XG4gICAgbGV0IGltcG9ydEZvdW5kSW5FeHBvcnRzID0gZmFsc2U7XG4gICAgZXhwb3J0cy5mb3JFYWNoKGUgPT4ge1xuICAgICAgICBpZiAoZS5oYXMoaW1wb3J0ZWRFbnRpdHlOYW1lKSkge1xuICAgICAgICAgICAgaW1wb3J0Rm91bmRJbkV4cG9ydHMgPSB0cnVlO1xuICAgICAgICB9XG4gICAgfSk7XG4gICAgcmV0dXJuIGltcG9ydEZvdW5kSW5FeHBvcnRzO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgaW5oZXJpdGFuY2VzIG9mIHRoZSBjbGFzc2VzIGFuZCBpbnRlcmZhY2VzIG9mIHRoZSBzb3VyY2UgZmlsZXNcbiAqIEBwYXJhbSBjbGFzc2VzIEFuIGFycmF5IG9mIGNsYXNzZXNcbiAqIEBwYXJhbSBpbnRlcmZhY2VzIEFuIGFycmF5IG9mIGludGVyZmFjZXNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByb2Nlc3NJbmhlcml0YW5jZXMoY2xhc3NlczogQ2xhc3NEZWNsYXJhdGlvbltdLCBpbnRlcmZhY2VzOiBJbnRlcmZhY2VEZWNsYXJhdGlvbltdKTogdm9pZCB7XG4gICAgbG9nZ2VyLmluZm8oYHByb2Nlc3NJbmhlcml0YW5jZXM6IENyZWF0aW5nIGluaGVyaXRhbmNlczpgKTtcbiAgICBjbGFzc2VzLmZvckVhY2goY2xzID0+IHtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzSW5oZXJpdGFuY2VzOiBDaGVja2luZyBjbGFzcyBpbmhlcml0YW5jZSBmb3IgJHtjbHMuZ2V0TmFtZSgpfWApO1xuICAgICAgICBjb25zdCBleHRDbGFzcyA9IGNscy5nZXRCYXNlQ2xhc3MoKTtcbiAgICAgICAgaWYgKGV4dENsYXNzICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhJbmhlcml0YW5jZShjbHMsIGV4dENsYXNzKTtcbiAgICAgICAgICAgIFxuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzSW5oZXJpdGFuY2VzOiBjbGFzczogJHtjbHMuZ2V0TmFtZSgpfSwgKCR7Y2xzLmdldFR5cGUoKS5nZXRUZXh0KCl9KSwgZXh0Q2xhc3M6ICR7ZXh0Q2xhc3MuZ2V0TmFtZSgpfSwgKCR7ZXh0Q2xhc3MuZ2V0VHlwZSgpLmdldFRleHQoKX0pYCk7XG4gICAgICAgIH1cblxuICAgICAgICBsb2dnZXIuZGVidWcoYHByb2Nlc3NJbmhlcml0YW5jZXM6IENoZWNraW5nIGludGVyZmFjZSBpbmhlcml0YW5jZSBmb3IgJHtjbHMuZ2V0TmFtZSgpfWApO1xuICAgICAgICBjb25zdCBpbXBsZW1lbnRlZEludGVyZmFjZXMgPSBnZXRJbXBsZW1lbnRlZE9yRXh0ZW5kZWRJbnRlcmZhY2VzKGludGVyZmFjZXMsIGNscyk7XG4gICAgICAgIGltcGxlbWVudGVkSW50ZXJmYWNlcy5mb3JFYWNoKGltcEludGVyID0+IHtcbiAgICAgICAgICAgIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhJbmhlcml0YW5jZShjbHMsIGltcEludGVyKTtcblxuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzSW5oZXJpdGFuY2VzOiBjbGFzczogJHtjbHMuZ2V0TmFtZSgpfSwgKCR7Y2xzLmdldFR5cGUoKS5nZXRUZXh0KCl9KSwgaW1wSW50ZXI6ICR7KGltcEludGVyIGluc3RhbmNlb2YgSW50ZXJmYWNlRGVjbGFyYXRpb24pID8gaW1wSW50ZXIuZ2V0TmFtZSgpIDogaW1wSW50ZXIuZ2V0RXhwcmVzc2lvbigpLmdldFRleHQoKX0sICgkeyhpbXBJbnRlciBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uKSA/IGltcEludGVyLmdldFR5cGUoKS5nZXRUZXh0KCkgOiBpbXBJbnRlci5nZXRFeHByZXNzaW9uKCkuZ2V0VGV4dCgpfSlgKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICBpbnRlcmZhY2VzLmZvckVhY2goaW50ZXIgPT4ge1xuICAgICAgICBsb2dnZXIuZGVidWcoYHByb2Nlc3NJbmhlcml0YW5jZXM6IENoZWNraW5nIGludGVyZmFjZSBpbmhlcml0YW5jZSBmb3IgJHtpbnRlci5nZXROYW1lKCl9YCk7XG4gICAgICAgIGNvbnN0IGV4dGVuZGVkSW50ZXJmYWNlcyA9IGdldEltcGxlbWVudGVkT3JFeHRlbmRlZEludGVyZmFjZXMoaW50ZXJmYWNlcywgaW50ZXIpO1xuICAgICAgICBleHRlbmRlZEludGVyZmFjZXMuZm9yRWFjaChleHRJbnRlciA9PiB7XG4gICAgICAgICAgICBlbnRpdHlEaWN0aW9uYXJ5LmNyZWF0ZUZhbWl4SW5oZXJpdGFuY2UoaW50ZXIsIGV4dEludGVyKTtcblxuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzSW5oZXJpdGFuY2VzOiBpbnRlcjogJHtpbnRlci5nZXROYW1lKCl9LCAoJHtpbnRlci5nZXRUeXBlKCkuZ2V0VGV4dCgpfSksIGV4dEludGVyOiAkeyhleHRJbnRlciBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uKSA/IGV4dEludGVyLmdldE5hbWUoKSA6IGV4dEludGVyLmdldEV4cHJlc3Npb24oKS5nZXRUZXh0KCl9LCAoJHsoZXh0SW50ZXIgaW5zdGFuY2VvZiBJbnRlcmZhY2VEZWNsYXJhdGlvbikgPyBleHRJbnRlci5nZXRUeXBlKCkuZ2V0VGV4dCgpIDogZXh0SW50ZXIuZ2V0RXhwcmVzc2lvbigpLmdldFRleHQoKX0pYCk7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufVxuXG4vKipcbiAqIEJ1aWxkcyBhIEZhbWl4IG1vZGVsIGZvciB0aGUgaW52b2NhdGlvbnMgb2YgdGhlIG1ldGhvZHMgYW5kIGZ1bmN0aW9ucyBvZiB0aGUgc291cmNlIGZpbGVzXG4gKiBAcGFyYW0gbWV0aG9kc0FuZEZ1bmN0aW9uc1dpdGhJZCBBIG1hcCBvZiBtZXRob2RzIGFuZCBmdW5jdGlvbnMgd2l0aCB0aGVpciBpZFxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJvY2Vzc0ludm9jYXRpb25zKG1ldGhvZHNBbmRGdW5jdGlvbnNXaXRoSWQ6IE1hcDxudW1iZXIsIE1ldGhvZERlY2xhcmF0aW9uIHwgQ29uc3RydWN0b3JEZWNsYXJhdGlvbiB8IEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgRnVuY3Rpb25EZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRXhwcmVzc2lvbiB8IEFycm93RnVuY3Rpb24+KTogdm9pZCB7XG4gICAgbG9nZ2VyLmluZm8oYENyZWF0aW5nIGludm9jYXRpb25zOmApO1xuICAgIG1ldGhvZHNBbmRGdW5jdGlvbnNXaXRoSWQuZm9yRWFjaCgobSwgaWQpID0+IHtcbiAgICAgICAgaWYgKCEobSBpbnN0YW5jZW9mIEFycm93RnVuY3Rpb24pKSB7XG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoYEludm9jYXRpb25zIHRvICR7KG0gaW5zdGFuY2VvZiBNZXRob2REZWNsYXJhdGlvbiB8fCBtIGluc3RhbmNlb2YgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fCBtIGluc3RhbmNlb2YgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fCBtIGluc3RhbmNlb2YgRnVuY3Rpb25EZWNsYXJhdGlvbikgPyBtLmdldE5hbWUoKSA6ICgobSBpbnN0YW5jZW9mIENvbnN0cnVjdG9yRGVjbGFyYXRpb24pID8gJ2NvbnN0cnVjdG9yJyA6IChtLmdldE5hbWUoKSA/IG0uZ2V0TmFtZSgpIDogJ2Fub255bW91cycpKX1gKTtcbiAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgICAgY29uc3QgdGVtcF9ub2RlcyA9IG0uZmluZFJlZmVyZW5jZXNBc05vZGVzKCkgYXMgQXJyYXk8SWRlbnRpZmllcj47XG4gICAgICAgICAgICAgICAgdGVtcF9ub2Rlcy5mb3JFYWNoKG5vZGUgPT4gcHJvY2Vzc05vZGVGb3JJbnZvY2F0aW9ucyhub2RlLCBtLCBpZCkpO1xuICAgICAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAgICAgICBsb2dnZXIuZXJyb3IoYD4gV0FSTklORzogZ290IGV4Y2VwdGlvbiAke2Vycm9yfS4gQ29udGludWluZy4uLmApO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgfSk7XG59XG5cbi8qKlxuICogQnVpbGRzIGEgRmFtaXggbW9kZWwgZm9yIGFuIGludm9jYXRpb24gb2YgYSBtZXRob2Qgb3IgYSBmdW5jdGlvblxuICogQHBhcmFtIG4gQSBub2RlXG4gKiBAcGFyYW0gbSBBIG1ldGhvZCBvciBhIGZ1bmN0aW9uXG4gKiBAcGFyYW0gaWQgVGhlIGlkIG9mIHRoZSBtZXRob2Qgb3IgdGhlIGZ1bmN0aW9uXG4gKi9cbmZ1bmN0aW9uIHByb2Nlc3NOb2RlRm9ySW52b2NhdGlvbnMobjogSWRlbnRpZmllciwgbTogTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uLCBpZDogbnVtYmVyKTogdm9pZCB7XG4gICAgdHJ5IHtcbiAgICAgICAgZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peEludm9jYXRpb24obiwgbSwgaWQpO1xuXG4gICAgICAgIGxvZ2dlci5kZWJ1Zyhgbm9kZTogbm9kZSwgKCR7bi5nZXRUeXBlKCkuZ2V0VGV4dCgpfSlgKTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICBsb2dnZXIuZXJyb3IoYD4gV0FSTklORzogZ290IGV4Y2VwdGlvbiAke2Vycm9yfS4gU2NvcGVEZWNsYXJhdGlvbiBpbnZhbGlkIGZvciAke24uZ2V0U3ltYm9sKCkhLmdldEZ1bGx5UXVhbGlmaWVkTmFtZSgpfS4gQ29udGludWluZy4uLmApO1xuICAgIH1cbn1cblxuLyoqXG4gKiBCdWlsZHMgYSBGYW1peCBtb2RlbCBmb3IgdGhlIGluaGVyaXRhbmNlcyBvZiB0aGUgY2xhc3NlcyBhbmQgaW50ZXJmYWNlcyBvZiB0aGUgc291cmNlIGZpbGVzXG4gKiBAcGFyYW0gY2xhc3NlcyBBbiBhcnJheSBvZiBjbGFzc2VzXG4gKiBAcGFyYW0gaW50ZXJmYWNlcyBBbiBhcnJheSBvZiBpbnRlcmZhY2VzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBwcm9jZXNzQ29uY3JldGlzYXRpb25zKGNsYXNzZXM6IENsYXNzRGVjbGFyYXRpb25bXSwgaW50ZXJmYWNlczogSW50ZXJmYWNlRGVjbGFyYXRpb25bXSwgZnVuY3Rpb25zOiBNYXA8bnVtYmVyLCBNZXRob2REZWNsYXJhdGlvbiB8IENvbnN0cnVjdG9yRGVjbGFyYXRpb24gfCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24gfCBBcnJvd0Z1bmN0aW9uPik6IHZvaWQge1xuICAgIGxvZ2dlci5pbmZvKGBwcm9jZXNzQ29uY3JldGlzYXRpb25zOiBDcmVhdGluZyBjb25jcmV0aXNhdGlvbnM6YCk7XG4gICAgY2xhc3Nlcy5mb3JFYWNoKGNscyA9PiB7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgcHJvY2Vzc0NvbmNyZXRpc2F0aW9uczogQ2hlY2tpbmcgY2xhc3MgY29uY3JldGlzYXRpb24gZm9yICR7Y2xzLmdldE5hbWUoKX1gKTtcbiAgICAgICAgZW50aXR5RGljdGlvbmFyeS5jcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uQ2xhc3NPckludGVyZmFjZVNwZWNpYWxpc2F0aW9uKGNscyk7XG4gICAgICAgIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkdlbmVyaWNJbnN0YW50aWF0aW9uKGNscyk7XG4gICAgICAgIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkludGVyZmFjZUNsYXNzKGNscyk7XG4gICAgICAgIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvblR5cGVJbnN0YW5jaWF0aW9uKGNscyk7IFxuXG4gICAgfSk7XG4gICAgaW50ZXJmYWNlcy5mb3JFYWNoKGludGVyID0+IHtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzQ29uY3JldGlzYXRpb25zOiBDaGVja2luZyBpbnRlcmZhY2UgY29uY3JldGlzYXRpb24gZm9yICR7aW50ZXIuZ2V0TmFtZSgpfWApO1xuICAgICAgICBlbnRpdHlEaWN0aW9uYXJ5LmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb25UeXBlSW5zdGFuY2lhdGlvbihpbnRlcik7IFxuICAgICAgICBlbnRpdHlEaWN0aW9uYXJ5LmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb25DbGFzc09ySW50ZXJmYWNlU3BlY2lhbGlzYXRpb24oaW50ZXIpXG4gICAgfSk7XG4gICAgZnVuY3Rpb25zLmZvckVhY2goZnVuYyA9PiB7XG4gICAgICAgIGlmKGZ1bmMgaW5zdGFuY2VvZiBGdW5jdGlvbkRlY2xhcmF0aW9uIHx8IGZ1bmMgaW5zdGFuY2VvZiBNZXRob2REZWNsYXJhdGlvbiApe1xuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBwcm9jZXNzQ29uY3JldGlzYXRpb25zOiBDaGVja2luZyBNZXRob2QgY29uY3JldGlzYXRpb25gKTtcbiAgICAgICAgICAgIGVudGl0eURpY3Rpb25hcnkuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkZ1bmN0aW9uSW5zdGFudGlhdGlvbihmdW5jKTtcbiAgICAgICAgfVxuICAgIH0pXG59XG4iXX0=