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
@@ -28,35 +28,97 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
28
28
  Object.defineProperty(exports, "__esModule", { value: true });
29
29
  exports.EntityDictionary = void 0;
30
30
  const ts_morph_1 = require("ts-morph");
31
- const Famix = __importStar(require("../lib/famix/src/model/famix"));
31
+ const process_functions_1 = require("../analyze_functions/process_functions");
32
+ const Famix = __importStar(require("../lib/famix/model/famix"));
33
+ const famix_repository_1 = require("../lib/famix/famix_repository");
32
34
  const analyze_1 = require("../analyze");
33
35
  const grapheme_splitter_1 = __importDefault(require("grapheme-splitter"));
34
36
  const Helpers = __importStar(require("./helpers_creation"));
35
37
  const FQNFunctions = __importStar(require("../fqn"));
36
- const famix_repository_1 = require("../lib/famix/src/famix_repository");
37
38
  const path_1 = __importDefault(require("path"));
39
+ const lodash_1 = __importDefault(require("lodash"));
38
40
  class EntityDictionary {
39
41
  constructor() {
40
42
  this.famixRep = new famix_repository_1.FamixRepository();
41
43
  this.fmxAliasMap = new Map(); // Maps the alias names to their Famix model
42
44
  this.fmxClassMap = new Map(); // Maps the fully qualified class names to their Famix model
43
45
  this.fmxInterfaceMap = new Map(); // Maps the interface names to their Famix model
44
- this.fmxNamespaceMap = new Map(); // Maps the namespace names to their Famix model
46
+ this.fmxModuleMap = new Map(); // Maps the namespace names to their Famix model
45
47
  this.fmxFileMap = new Map(); // Maps the source file names to their Famix model
46
48
  this.fmxTypeMap = new Map(); // Maps the type names to their Famix model
49
+ this.fmxFunctionAndMethodMap = new Map; // Maps the function names to their Famix model
47
50
  this.UNKNOWN_VALUE = '(unknown due to parsing error)'; // The value to use when a name is not usable
48
51
  this.fmxElementObjectMap = new Map();
52
+ this.tsMorphElementObjectMap = new Map();
49
53
  this.famixRep.setFmxElementObjectMap(this.fmxElementObjectMap);
50
54
  }
55
+ addSourceAnchor(fmx, node) {
56
+ const sourceAnchor = new Famix.IndexedFileAnchor();
57
+ let sourceStart, sourceEnd;
58
+ if (fmx && node) {
59
+ // find the start and end positions of the source element
60
+ if (!(node instanceof ts_morph_1.CommentRange)) {
61
+ sourceStart = node.getStart();
62
+ sourceEnd = node.getEnd();
63
+ }
64
+ else {
65
+ sourceStart = node.getPos();
66
+ sourceEnd = node.getEnd();
67
+ }
68
+ if (analyze_1.config.expectGraphemes) {
69
+ /**
70
+ * The following logic handles the case of multi-code point characters (e.g. emoji) in the source text.
71
+ * This is needed because Pharo/Smalltalk treats multi-code point characters as a single character,
72
+ * but JavaScript treats them as multiple characters. This means that the start and end positions
73
+ * of a source element in Pharo/Smalltalk will be different than the start and end positions of the
74
+ * same source element in JavaScript. This logic finds the start and end positions of the source
75
+ * element in JavaScript and then uses those positions to set the start and end positions of the
76
+ * Famix index file anchor.
77
+ * It depends on code in the 'grapheme-splitter' package in npm.
78
+ */
79
+ const splitter = new grapheme_splitter_1.default();
80
+ const sourceFileText = node.getSourceFile().getFullText();
81
+ const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;
82
+ if (hasGraphemeClusters) {
83
+ const sourceElementText = sourceFileText.substring(sourceStart, sourceEnd);
84
+ const sourceElementTextGraphemes = splitter.splitGraphemes(sourceElementText);
85
+ const sourceFileTextGraphemes = splitter.splitGraphemes(sourceFileText);
86
+ const numberOfGraphemeClustersBeforeStart = splitter.countGraphemes(sourceFileText.substring(0, sourceStart));
87
+ // find the start of the sourceElementTextGraphemes array in the sourceFileTextGraphemes array
88
+ sourceStart = Helpers.indexOfSplitArray({ searchArray: sourceFileTextGraphemes,
89
+ targetArray: sourceElementTextGraphemes,
90
+ start: sourceStart - numberOfGraphemeClustersBeforeStart });
91
+ sourceEnd = sourceStart + sourceElementTextGraphemes.length;
92
+ }
93
+ }
94
+ // The +1 is because the source anchor (Pharo) is 1-based, but ts-morph is 0-based
95
+ sourceAnchor.startPos = sourceStart + 1;
96
+ sourceAnchor.endPos = sourceEnd + 1;
97
+ const fileName = node.getSourceFile().getFilePath();
98
+ sourceAnchor.element = fmx;
99
+ sourceAnchor.fileName = fileName;
100
+ fmx.sourceAnchor = sourceAnchor;
101
+ this.famixRep.addElement(sourceAnchor);
102
+ }
103
+ return sourceAnchor;
104
+ }
51
105
  /**
52
106
  * Makes a Famix index file anchor
53
107
  * @param sourceElement A source element
54
108
  * @param famixElement The Famix model of the source element
55
109
  */
56
110
  makeFamixIndexFileAnchor(sourceElement, famixElement) {
57
- analyze_1.logger.debug("making index file anchor for '" + (sourceElement === null || sourceElement === void 0 ? void 0 : sourceElement.getText()) + "' with famixElement " + famixElement.getJSON());
111
+ // check if famixElement doesn't have a valid fullyQualifiedName
112
+ if (typeof famixElement.getFullyQualifiedName === 'function') {
113
+ // The method exists
114
+ const fullyQualifiedName = famixElement.fullyQualifiedName;
115
+ if (!fullyQualifiedName || fullyQualifiedName === this.UNKNOWN_VALUE) {
116
+ throw new Error(`Famix element ${famixElement.constructor.name} has no valid fullyQualifiedName.`);
117
+ }
118
+ }
119
+ analyze_1.logger.debug("making index file anchor for '" + sourceElement?.getText() + "' with famixElement " + famixElement.getJSON());
58
120
  const fmxIndexFileAnchor = new Famix.IndexedFileAnchor();
59
- fmxIndexFileAnchor.setElement(famixElement);
121
+ fmxIndexFileAnchor.element = famixElement;
60
122
  this.fmxElementObjectMap.set(famixElement, sourceElement);
61
123
  if (sourceElement !== null) {
62
124
  const absolutePathProject = this.famixRep.getAbsolutePath();
@@ -72,7 +134,7 @@ class EntityDictionary {
72
134
  }
73
135
  // revert any backslashes to forward slashes (path.normalize on windows introduces them)
74
136
  pathInProject = pathInProject.replace(/\\/g, "/");
75
- fmxIndexFileAnchor.setFileName(pathInProject);
137
+ fmxIndexFileAnchor.fileName = pathInProject;
76
138
  let sourceStart, sourceEnd, sourceLineStart, sourceLineEnd;
77
139
  if (!(sourceElement instanceof ts_morph_1.CommentRange)) {
78
140
  sourceStart = sourceElement.getStart();
@@ -111,22 +173,18 @@ class EntityDictionary {
111
173
  }
112
174
  }
113
175
  // note: the +1 is because the source anchor is 1-based, but ts-morph is 0-based
114
- fmxIndexFileAnchor.setStartPos(sourceStart + 1);
115
- fmxIndexFileAnchor.setEndPos(sourceEnd + 1);
116
- if (!(sourceElement instanceof ts_morph_1.CommentRange)) {
117
- fmxIndexFileAnchor.setStartLine(sourceLineStart);
118
- fmxIndexFileAnchor.setEndLine(sourceLineEnd);
119
- }
120
- if (!(famixElement instanceof Famix.Association) && !(famixElement instanceof Famix.Comment) && !(sourceElement instanceof ts_morph_1.CommentRange) && !(sourceElement instanceof ts_morph_1.Identifier) && !(sourceElement instanceof ts_morph_1.ImportSpecifier) && !(sourceElement instanceof ts_morph_1.ExpressionWithTypeArguments)) {
121
- famixElement.setFullyQualifiedName(FQNFunctions.getFQN(sourceElement));
122
- }
176
+ fmxIndexFileAnchor.startPos = sourceStart + 1;
177
+ fmxIndexFileAnchor.endPos = sourceEnd + 1;
178
+ // if (!(famixElement instanceof Famix.ImportClause || famixElement instanceof Famix.Access || famixElement instanceof Famix.Reference || famixElement instanceof Famix.Invocation || famixElement instanceof Famix.Inheritance) && !(famixElement instanceof Famix.Comment) && !(sourceElement instanceof CommentRange) && !(sourceElement instanceof Identifier) && !(sourceElement instanceof ImportSpecifier) && !(sourceElement instanceof ExpressionWithTypeArguments)) {
179
+ // initFQN(sourceElement, famixElement);
180
+ // }
123
181
  }
124
182
  else {
125
183
  // sourceElement is null
126
184
  analyze_1.logger.warn("sourceElement is null for famixElement " + famixElement.getJSON());
127
- fmxIndexFileAnchor.setFileName("unknown");
128
- fmxIndexFileAnchor.setStartPos(0);
129
- fmxIndexFileAnchor.setEndPos(0);
185
+ fmxIndexFileAnchor.fileName = "unknown";
186
+ fmxIndexFileAnchor.startPos = 0;
187
+ fmxIndexFileAnchor.endPos = 0;
130
188
  }
131
189
  this.famixRep.addElement(fmxIndexFileAnchor);
132
190
  }
@@ -137,49 +195,56 @@ class EntityDictionary {
137
195
  * @returns The Famix model of the source file
138
196
  */
139
197
  createOrGetFamixFile(f, isModule) {
140
- let fmxFile;
198
+ let fmxFile; // | Famix.Module;
141
199
  const fileName = f.getBaseName();
142
200
  const fullyQualifiedFilename = f.getFilePath();
143
- if (!this.fmxFileMap.has(fullyQualifiedFilename)) {
201
+ const foundFileName = this.fmxFileMap.get(fullyQualifiedFilename);
202
+ if (!foundFileName) {
144
203
  if (isModule) {
145
204
  fmxFile = new Famix.Module();
146
205
  }
147
206
  else {
148
207
  fmxFile = new Famix.ScriptEntity();
149
208
  }
150
- fmxFile.setName(fileName);
151
- fmxFile.setNumberOfLinesOfText(f.getEndLineNumber() - f.getStartLineNumber());
152
- fmxFile.setNumberOfCharacters(f.getFullText().length);
209
+ fmxFile.name = fileName;
210
+ fmxFile.numberOfLinesOfText = f.getEndLineNumber() - f.getStartLineNumber();
211
+ fmxFile.numberOfCharacters = f.getFullText().length;
212
+ initFQN(f, fmxFile);
153
213
  this.makeFamixIndexFileAnchor(f, fmxFile);
154
214
  this.fmxFileMap.set(fullyQualifiedFilename, fmxFile);
155
215
  this.famixRep.addElement(fmxFile);
156
216
  }
157
217
  else {
158
- fmxFile = this.fmxFileMap.get(fullyQualifiedFilename);
218
+ fmxFile = foundFileName;
159
219
  }
160
220
  this.fmxElementObjectMap.set(fmxFile, f);
161
221
  return fmxFile;
162
222
  }
163
223
  /**
164
- * Creates or gets a Famix namespace
165
- * @param m A namespace
166
- * @returns The Famix model of the namespace
224
+ * Creates or gets a Famix Module
225
+ * @param m A module
226
+ * @returns The Famix model of the module
167
227
  */
168
- createOrGetFamixNamespace(m) {
169
- let fmxNamespace;
170
- const namespaceName = m.getName();
171
- if (!this.fmxNamespaceMap.has(namespaceName)) {
172
- fmxNamespace = new Famix.Namespace();
173
- fmxNamespace.setName(namespaceName);
174
- this.makeFamixIndexFileAnchor(m, fmxNamespace);
175
- this.fmxNamespaceMap.set(namespaceName, fmxNamespace);
176
- this.famixRep.addElement(fmxNamespace);
228
+ createOrGetFamixModule(m) {
229
+ let fmxModule;
230
+ const moduleName = m.getName();
231
+ const foundModuleName = this.fmxModuleMap.get(moduleName);
232
+ if (!foundModuleName) {
233
+ fmxModule = new Famix.Module();
234
+ fmxModule.name = moduleName;
235
+ fmxModule.isAmbient = (0, process_functions_1.isAmbient)(m);
236
+ fmxModule.isNamespace = (0, process_functions_1.isNamespace)(m);
237
+ fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;
238
+ initFQN(m, fmxModule);
239
+ this.makeFamixIndexFileAnchor(m, fmxModule);
240
+ this.fmxModuleMap.set(moduleName, fmxModule);
241
+ this.famixRep.addElement(fmxModule);
177
242
  }
178
243
  else {
179
- fmxNamespace = this.fmxNamespaceMap.get(namespaceName);
244
+ fmxModule = foundModuleName;
180
245
  }
181
- this.fmxElementObjectMap.set(fmxNamespace, m);
182
- return fmxNamespace;
246
+ this.fmxElementObjectMap.set(fmxModule, m);
247
+ return fmxModule;
183
248
  }
184
249
  /**
185
250
  * Creates a Famix alias
@@ -190,19 +255,21 @@ class EntityDictionary {
190
255
  let fmxAlias;
191
256
  const aliasName = a.getName();
192
257
  const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);
193
- if (!this.fmxAliasMap.has(aliasFullyQualifiedName)) {
258
+ const foundAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);
259
+ if (!foundAlias) {
194
260
  fmxAlias = new Famix.Alias();
195
- fmxAlias.setName(a.getName());
261
+ fmxAlias.name = a.getName();
196
262
  const aliasNameWithGenerics = aliasName + (a.getTypeParameters().length ? ("<" + a.getTypeParameters().map(tp => tp.getName()).join(", ") + ">") : "");
197
263
  analyze_1.logger.debug(`> NOTE: alias ${aliasName} has fully qualified name ${aliasFullyQualifiedName} and name with generics ${aliasNameWithGenerics}.`);
198
264
  const fmxType = this.createOrGetFamixType(aliasNameWithGenerics, a);
199
- fmxAlias.setAliasedEntity(fmxType);
265
+ fmxAlias.aliasedEntity = fmxType;
266
+ initFQN(a, fmxAlias);
200
267
  this.makeFamixIndexFileAnchor(a, fmxAlias);
201
268
  this.fmxAliasMap.set(aliasFullyQualifiedName, fmxAlias);
202
269
  this.famixRep.addElement(fmxAlias);
203
270
  }
204
271
  else {
205
- fmxAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);
272
+ fmxAlias = foundAlias;
206
273
  }
207
274
  this.fmxElementObjectMap.set(fmxAlias, a);
208
275
  return fmxAlias;
@@ -216,26 +283,27 @@ class EntityDictionary {
216
283
  let fmxClass;
217
284
  const isAbstract = cls.isAbstract();
218
285
  const classFullyQualifiedName = FQNFunctions.getFQN(cls);
219
- const clsName = cls.getName();
220
- if (!this.fmxClassMap.has(classFullyQualifiedName)) {
221
- const isGeneric = cls.getTypeParameters().length;
286
+ const clsName = cls.getName() || this.UNKNOWN_VALUE;
287
+ const isGeneric = cls.getTypeParameters().length;
288
+ const foundClass = this.fmxClassMap.get(classFullyQualifiedName);
289
+ if (!foundClass) {
222
290
  if (isGeneric) {
223
- fmxClass = new Famix.ParameterizableClass();
291
+ fmxClass = new Famix.ParametricClass();
224
292
  }
225
293
  else {
226
294
  fmxClass = new Famix.Class();
227
295
  }
228
- fmxClass.setName(clsName);
229
- fmxClass.setFullyQualifiedName(classFullyQualifiedName);
230
- fmxClass.setIsAbstract(isAbstract);
296
+ fmxClass.name = clsName;
297
+ fmxClass.fullyQualifiedName = classFullyQualifiedName;
298
+ fmxClass.isAbstract = isAbstract;
231
299
  this.makeFamixIndexFileAnchor(cls, fmxClass);
232
300
  this.fmxClassMap.set(classFullyQualifiedName, fmxClass);
233
301
  this.famixRep.addElement(fmxClass);
302
+ this.fmxElementObjectMap.set(fmxClass, cls);
234
303
  }
235
304
  else {
236
- fmxClass = this.fmxClassMap.get(classFullyQualifiedName);
305
+ fmxClass = foundClass;
237
306
  }
238
- this.fmxElementObjectMap.set(fmxClass, cls);
239
307
  return fmxClass;
240
308
  }
241
309
  /**
@@ -247,25 +315,84 @@ class EntityDictionary {
247
315
  let fmxInterface;
248
316
  const interName = inter.getName();
249
317
  const interFullyQualifiedName = FQNFunctions.getFQN(inter);
250
- if (!this.fmxInterfaceMap.has(interName)) {
318
+ const foundInterface = this.fmxInterfaceMap.get(interFullyQualifiedName);
319
+ if (!foundInterface) {
251
320
  const isGeneric = inter.getTypeParameters().length;
252
321
  if (isGeneric) {
253
- fmxInterface = new Famix.ParameterizableInterface();
322
+ fmxInterface = new Famix.ParametricInterface();
254
323
  }
255
324
  else {
256
325
  fmxInterface = new Famix.Interface();
257
326
  }
258
- fmxInterface.setName(interName);
327
+ fmxInterface.name = interName;
328
+ initFQN(inter, fmxInterface);
259
329
  this.makeFamixIndexFileAnchor(inter, fmxInterface);
260
330
  this.fmxInterfaceMap.set(interFullyQualifiedName, fmxInterface);
261
331
  this.famixRep.addElement(fmxInterface);
332
+ this.fmxElementObjectMap.set(fmxInterface, inter);
262
333
  }
263
334
  else {
264
- fmxInterface = this.fmxInterfaceMap.get(interName);
335
+ fmxInterface = foundInterface;
265
336
  }
266
- this.fmxElementObjectMap.set(fmxInterface, inter);
267
337
  return fmxInterface;
268
338
  }
339
+ /**
340
+ * Creates or gets a Famix concrete element
341
+ * @param concreteElement A parametric Element
342
+ * @param concreteElementDeclaration the element declaration
343
+ * @param concreteArguments concrete arguments
344
+ * @returns A parametric Element
345
+ */
346
+ createOrGetFamixConcreteElement(concreteElement, concreteElementDeclaration, concreteArguments) {
347
+ let fullyQualifiedFilename = concreteElement.fullyQualifiedName;
348
+ let params = "";
349
+ concreteArguments.map((param) => {
350
+ params = params + param.getText() + ',';
351
+ });
352
+ params = params.substring(0, params.length - 1);
353
+ fullyQualifiedFilename = Helpers.replaceLastBetweenTags(fullyQualifiedFilename, params);
354
+ let concElement;
355
+ if (!this.fmxInterfaceMap.has(fullyQualifiedFilename) &&
356
+ !this.fmxClassMap.has(fullyQualifiedFilename) &&
357
+ !this.fmxFunctionAndMethodMap.has(fullyQualifiedFilename)) {
358
+ concElement = lodash_1.default.cloneDeep(concreteElement);
359
+ concElement.fullyQualifiedName = fullyQualifiedFilename;
360
+ concElement.clearGenericParameters();
361
+ concreteArguments.map((param) => {
362
+ const parameter = this.createOrGetFamixConcreteType(param);
363
+ concElement.addConcreteParameter(parameter);
364
+ });
365
+ if (concreteElement instanceof Famix.ParametricClass) {
366
+ this.fmxClassMap.set(fullyQualifiedFilename, concElement);
367
+ }
368
+ else if (concreteElement instanceof Famix.ParametricInterface) {
369
+ this.fmxInterfaceMap.set(fullyQualifiedFilename, concElement);
370
+ }
371
+ else if (concreteElement instanceof Famix.ParametricFunction) {
372
+ this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement);
373
+ }
374
+ else { // if (concreteElement instanceof Famix.ParametricMethod) {
375
+ this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement);
376
+ }
377
+ this.famixRep.addElement(concElement);
378
+ this.fmxElementObjectMap.set(concElement, concreteElementDeclaration);
379
+ }
380
+ else {
381
+ if (concreteElement instanceof Famix.ParametricClass) {
382
+ concElement = this.fmxClassMap.get(fullyQualifiedFilename);
383
+ }
384
+ else if (concreteElement instanceof Famix.ParametricInterface) {
385
+ concElement = this.fmxInterfaceMap.get(fullyQualifiedFilename);
386
+ }
387
+ else if (concreteElement instanceof Famix.ParametricFunction) {
388
+ concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename);
389
+ }
390
+ else { // if (concreteElement instanceof Famix.ParametricMethod) {
391
+ concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename);
392
+ }
393
+ }
394
+ return concElement;
395
+ }
269
396
  /**
270
397
  * Creates a Famix property
271
398
  * @param property A property
@@ -274,7 +401,7 @@ class EntityDictionary {
274
401
  createFamixProperty(property) {
275
402
  const fmxProperty = new Famix.Property();
276
403
  const isSignature = property instanceof ts_morph_1.PropertySignature;
277
- fmxProperty.setName(property.getName());
404
+ fmxProperty.name = property.getName();
278
405
  let propTypeName = this.UNKNOWN_VALUE;
279
406
  try {
280
407
  propTypeName = property.getType().getText().trim();
@@ -283,23 +410,40 @@ class EntityDictionary {
283
410
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for property: ${property.getName()}. Continuing...`);
284
411
  }
285
412
  const fmxType = this.createOrGetFamixType(propTypeName, property);
286
- fmxProperty.setDeclaredType(fmxType);
287
- property.getModifiers().forEach(m => fmxProperty.addModifier(m.getText()));
413
+ fmxProperty.declaredType = fmxType;
414
+ // add the visibility (public, private, etc.) to the fmxProperty
415
+ fmxProperty.visibility = "";
416
+ property.getModifiers().forEach(m => {
417
+ switch (m.getText()) {
418
+ case ts_morph_1.Scope.Public:
419
+ fmxProperty.visibility = "public";
420
+ break;
421
+ case ts_morph_1.Scope.Protected:
422
+ fmxProperty.visibility = "protected";
423
+ break;
424
+ case ts_morph_1.Scope.Private:
425
+ fmxProperty.visibility = "private";
426
+ break;
427
+ case "static":
428
+ fmxProperty.isClassSide = true;
429
+ break;
430
+ case "readonly":
431
+ fmxProperty.readOnly = true;
432
+ break;
433
+ default:
434
+ break;
435
+ }
436
+ });
288
437
  if (!isSignature && property.getExclamationTokenNode()) {
289
- fmxProperty.addModifier("!");
438
+ fmxProperty.isDefinitelyAssigned = true;
290
439
  }
291
440
  if (property.getQuestionTokenNode()) {
292
- fmxProperty.addModifier("?");
441
+ fmxProperty.isOptional = true;
293
442
  }
294
443
  if (property.getName().substring(0, 1) === "#") {
295
- fmxProperty.addModifier("#");
296
- }
297
- if (fmxProperty.getModifiers().has("static")) {
298
- fmxProperty.setIsClassSide(true);
299
- }
300
- else {
301
- fmxProperty.setIsClassSide(false);
444
+ fmxProperty.isJavaScriptPrivate = true;
302
445
  }
446
+ initFQN(property, fmxProperty);
303
447
  this.makeFamixIndexFileAnchor(property, fmxProperty);
304
448
  this.famixRep.addElement(fmxProperty);
305
449
  this.fmxElementObjectMap.set(fmxProperty, property);
@@ -311,86 +455,98 @@ class EntityDictionary {
311
455
  * @param currentCC The cyclomatic complexity metrics of the current source file
312
456
  * @returns The Famix model of the method or the accessor
313
457
  */
314
- createFamixMethod(method, currentCC) {
458
+ createOrGetFamixMethod(method, currentCC) {
315
459
  let fmxMethod;
316
- if (method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) {
317
- fmxMethod = new Famix.Accessor();
318
- const isGetter = method instanceof ts_morph_1.GetAccessorDeclaration;
319
- const isSetter = method instanceof ts_morph_1.SetAccessorDeclaration;
320
- if (isGetter) {
321
- fmxMethod.setKind("getter");
460
+ const isGeneric = method.getTypeParameters().length > 0;
461
+ const functionFullyQualifiedName = FQNFunctions.getFQN(method);
462
+ if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {
463
+ if (method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) {
464
+ fmxMethod = new Famix.Accessor();
465
+ const isGetter = method instanceof ts_morph_1.GetAccessorDeclaration;
466
+ const isSetter = method instanceof ts_morph_1.SetAccessorDeclaration;
467
+ if (isGetter) {
468
+ fmxMethod.kind = "getter";
469
+ }
470
+ if (isSetter) {
471
+ fmxMethod.kind = "setter";
472
+ }
473
+ this.famixRep.addElement(fmxMethod);
322
474
  }
323
- if (isSetter) {
324
- fmxMethod.setKind("setter");
475
+ else {
476
+ if (isGeneric) {
477
+ fmxMethod = new Famix.ParametricMethod();
478
+ }
479
+ else {
480
+ fmxMethod = new Famix.Method();
481
+ }
482
+ this.famixRep.addElement(fmxMethod);
325
483
  }
326
- this.famixRep.addElement(fmxMethod);
327
- }
328
- else {
329
- fmxMethod = new Famix.Method();
330
- this.famixRep.addElement(fmxMethod);
331
- }
332
- const isConstructor = method instanceof ts_morph_1.ConstructorDeclaration;
333
- const isSignature = method instanceof ts_morph_1.MethodSignature;
334
- const isGeneric = method.getTypeParameters().length > 0;
335
- fmxMethod.setIsGeneric(isGeneric);
336
- let isAbstract = false;
337
- let isStatic = false;
338
- if (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) {
339
- isAbstract = method.isAbstract();
340
- isStatic = method.isStatic();
341
- }
342
- if (isConstructor) {
343
- fmxMethod.setKind("constructor");
344
- }
345
- fmxMethod.setIsAbstract(isAbstract);
346
- fmxMethod.setIsClassSide(isStatic);
347
- fmxMethod.setIsPrivate((method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'private')) !== undefined : false);
348
- fmxMethod.setIsProtected((method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'protected')) !== undefined : false);
349
- fmxMethod.setSignature(Helpers.computeSignature(method.getText()));
350
- let methodName;
351
- if (isConstructor) {
352
- methodName = "constructor";
353
- }
354
- else {
355
- methodName = method.getName();
356
- }
357
- fmxMethod.setName(methodName);
358
- if (!isConstructor) {
359
- if (method.getName().substring(0, 1) === "#") {
360
- fmxMethod.setIsPrivate(true);
484
+ const isConstructor = method instanceof ts_morph_1.ConstructorDeclaration;
485
+ const isSignature = method instanceof ts_morph_1.MethodSignature;
486
+ let isAbstract = false;
487
+ let isStatic = false;
488
+ if (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) {
489
+ isAbstract = method.isAbstract();
490
+ isStatic = method.isStatic();
361
491
  }
362
- }
363
- if (!fmxMethod.getIsPrivate() && !fmxMethod.getIsProtected()) {
364
- fmxMethod.setIsPublic(true);
365
- }
366
- else {
367
- fmxMethod.setIsPublic(false);
368
- }
369
- if (!isSignature) {
370
- fmxMethod.setCyclomaticComplexity(currentCC[fmxMethod.getName()]);
371
- }
372
- else {
373
- fmxMethod.setCyclomaticComplexity(0);
374
- }
375
- let methodTypeName = this.UNKNOWN_VALUE;
376
- try {
377
- methodTypeName = method.getReturnType().getText().trim();
378
- }
379
- catch (error) {
380
- analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of method: ${fmxMethod.getName()}. Continuing...`);
381
- }
382
- const fmxType = this.createOrGetFamixType(methodTypeName, method);
383
- fmxMethod.setDeclaredType(fmxType);
384
- fmxMethod.setNumberOfLinesOfCode(method.getEndLineNumber() - method.getStartLineNumber());
385
- const parameters = method.getParameters();
386
- fmxMethod.setNumberOfParameters(parameters.length);
387
- if (!isSignature) {
388
- fmxMethod.setNumberOfStatements(method.getStatements().length);
492
+ if (isConstructor) {
493
+ fmxMethod.kind = "constructor";
494
+ }
495
+ fmxMethod.isAbstract = isAbstract;
496
+ fmxMethod.isClassSide = isStatic;
497
+ fmxMethod.isPrivate = (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'private')) !== undefined : false;
498
+ fmxMethod.isProtected = (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'protected')) !== undefined : false;
499
+ fmxMethod.signature = Helpers.computeSignature(method.getText());
500
+ let methodName;
501
+ if (isConstructor) {
502
+ methodName = "constructor";
503
+ }
504
+ else {
505
+ methodName = method.getName();
506
+ }
507
+ fmxMethod.name = methodName;
508
+ if (!isConstructor) {
509
+ if (method.getName().substring(0, 1) === "#") {
510
+ fmxMethod.isPrivate = true;
511
+ }
512
+ }
513
+ if (!fmxMethod.isPrivate && !fmxMethod.isProtected) {
514
+ fmxMethod.isPublic = true;
515
+ }
516
+ else {
517
+ fmxMethod.isPublic = false;
518
+ }
519
+ if (!isSignature) {
520
+ fmxMethod.cyclomaticComplexity = currentCC[fmxMethod.name];
521
+ }
522
+ else {
523
+ fmxMethod.cyclomaticComplexity = 0;
524
+ }
525
+ let methodTypeName = this.UNKNOWN_VALUE;
526
+ try {
527
+ methodTypeName = method.getReturnType().getText().trim();
528
+ }
529
+ catch (error) {
530
+ analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of method: ${fmxMethod.name}. Continuing...`);
531
+ }
532
+ const fmxType = this.createOrGetFamixType(methodTypeName, method);
533
+ fmxMethod.declaredType = fmxType;
534
+ fmxMethod.numberOfLinesOfCode = method.getEndLineNumber() - method.getStartLineNumber();
535
+ const parameters = method.getParameters();
536
+ fmxMethod.numberOfParameters = parameters.length;
537
+ if (!isSignature) {
538
+ fmxMethod.numberOfStatements = method.getStatements().length;
539
+ }
540
+ else {
541
+ fmxMethod.numberOfStatements = 0;
542
+ }
543
+ initFQN(method, fmxMethod);
544
+ this.makeFamixIndexFileAnchor(method, fmxMethod);
545
+ this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxMethod);
389
546
  }
390
547
  else {
391
- fmxMethod.setNumberOfStatements(0);
548
+ fmxMethod = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName);
392
549
  }
393
- this.makeFamixIndexFileAnchor(method, fmxMethod);
394
550
  this.fmxElementObjectMap.set(fmxMethod, method);
395
551
  return fmxMethod;
396
552
  }
@@ -400,34 +556,48 @@ class EntityDictionary {
400
556
  * @param currentCC The cyclomatic complexity metrics of the current source file
401
557
  * @returns The Famix model of the function
402
558
  */
403
- createFamixFunction(func, currentCC) {
404
- const fmxFunction = new Famix.Function();
405
- if (func.getName()) {
406
- fmxFunction.setName(func.getName());
559
+ createOrGetFamixFunction(func, currentCC) {
560
+ let fmxFunction;
561
+ const isGeneric = func.getTypeParameters().length > 0;
562
+ const functionFullyQualifiedName = FQNFunctions.getFQN(func);
563
+ if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {
564
+ if (isGeneric) {
565
+ fmxFunction = new Famix.ParametricFunction();
566
+ }
567
+ else {
568
+ fmxFunction = new Famix.Function();
569
+ }
570
+ const name = func.getName();
571
+ if (name) {
572
+ fmxFunction.name = name;
573
+ }
574
+ else {
575
+ fmxFunction.name = "anonymous";
576
+ }
577
+ fmxFunction.signature = Helpers.computeSignature(func.getText());
578
+ fmxFunction.cyclomaticComplexity = currentCC[fmxFunction.name];
579
+ fmxFunction.fullyQualifiedName = functionFullyQualifiedName;
580
+ let functionTypeName = this.UNKNOWN_VALUE;
581
+ try {
582
+ functionTypeName = func.getReturnType().getText().trim();
583
+ }
584
+ catch (error) {
585
+ analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${func.getName()}. Continuing...`);
586
+ }
587
+ const fmxType = this.createOrGetFamixType(functionTypeName, func);
588
+ fmxFunction.declaredType = fmxType;
589
+ fmxFunction.numberOfLinesOfCode = func.getEndLineNumber() - func.getStartLineNumber();
590
+ const parameters = func.getParameters();
591
+ fmxFunction.numberOfParameters = parameters.length;
592
+ fmxFunction.numberOfStatements = func.getStatements().length;
593
+ this.makeFamixIndexFileAnchor(func, fmxFunction);
594
+ this.famixRep.addElement(fmxFunction);
595
+ this.fmxElementObjectMap.set(fmxFunction, func);
596
+ this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxFunction);
407
597
  }
408
598
  else {
409
- fmxFunction.setName("anonymous");
599
+ fmxFunction = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName);
410
600
  }
411
- fmxFunction.setSignature(Helpers.computeSignature(func.getText()));
412
- fmxFunction.setCyclomaticComplexity(currentCC[fmxFunction.getName()]);
413
- const isGeneric = func.getTypeParameters().length > 0;
414
- fmxFunction.setIsGeneric(isGeneric);
415
- let functionTypeName = this.UNKNOWN_VALUE;
416
- try {
417
- functionTypeName = func.getReturnType().getText().trim();
418
- }
419
- catch (error) {
420
- analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${func.getName()}. Continuing...`);
421
- }
422
- const fmxType = this.createOrGetFamixType(functionTypeName, func);
423
- fmxFunction.setDeclaredType(fmxType);
424
- fmxFunction.setNumberOfLinesOfCode(func.getEndLineNumber() - func.getStartLineNumber());
425
- const parameters = func.getParameters();
426
- fmxFunction.setNumberOfParameters(parameters.length);
427
- fmxFunction.setNumberOfStatements(func.getStatements().length);
428
- this.makeFamixIndexFileAnchor(func, fmxFunction);
429
- this.famixRep.addElement(fmxFunction);
430
- this.fmxElementObjectMap.set(fmxFunction, func);
431
601
  return fmxFunction;
432
602
  }
433
603
  /**
@@ -445,8 +615,9 @@ class EntityDictionary {
445
615
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for parameter: ${param.getName()}. Continuing...`);
446
616
  }
447
617
  const fmxType = this.createOrGetFamixType(paramTypeName, param);
448
- fmxParam.setDeclaredType(fmxType);
449
- fmxParam.setName(param.getName());
618
+ fmxParam.declaredType = fmxType;
619
+ fmxParam.name = param.getName();
620
+ initFQN(param, fmxParam);
450
621
  this.makeFamixIndexFileAnchor(param, fmxParam);
451
622
  this.famixRep.addElement(fmxParam);
452
623
  this.fmxElementObjectMap.set(fmxParam, param);
@@ -459,12 +630,84 @@ class EntityDictionary {
459
630
  */
460
631
  createFamixParameterType(tp) {
461
632
  const fmxParameterType = new Famix.ParameterType();
462
- fmxParameterType.setName(tp.getName());
633
+ fmxParameterType.name = tp.getName();
634
+ initFQN(tp, fmxParameterType);
463
635
  this.makeFamixIndexFileAnchor(tp, fmxParameterType);
464
636
  this.famixRep.addElement(fmxParameterType);
465
637
  this.fmxElementObjectMap.set(fmxParameterType, tp);
466
638
  return fmxParameterType;
467
639
  }
640
+ /**
641
+ * Creates a Famix type parameter
642
+ * @param tp A type parameter
643
+ * @returns The Famix model of the type parameter
644
+ */
645
+ createOrGetFamixConcreteType(param) {
646
+ const typeParameterDeclaration = param.getSymbol()?.getDeclarations()[0];
647
+ const parameterTypeName = param.getText();
648
+ let fmxParameterType = undefined;
649
+ let isClassOrInterface = false;
650
+ if (this.fmxClassMap.has(parameterTypeName)) {
651
+ this.fmxClassMap.forEach((obj, name) => {
652
+ if (obj instanceof Famix.ParametricClass) {
653
+ if (name === param.getText() && obj.genericParameters.size > 0) {
654
+ fmxParameterType = obj;
655
+ isClassOrInterface = true;
656
+ }
657
+ }
658
+ else {
659
+ if (name === param.getText()) {
660
+ fmxParameterType = obj;
661
+ isClassOrInterface = true;
662
+ }
663
+ }
664
+ });
665
+ }
666
+ if (this.fmxInterfaceMap.has(parameterTypeName)) {
667
+ this.fmxInterfaceMap.forEach((obj, name) => {
668
+ if (obj instanceof Famix.ParametricInterface) {
669
+ if (name === param.getText() && obj.genericParameters.size > 0) {
670
+ fmxParameterType = obj;
671
+ isClassOrInterface = true;
672
+ }
673
+ }
674
+ else {
675
+ if (name === param.getText()) {
676
+ fmxParameterType = obj;
677
+ isClassOrInterface = true;
678
+ }
679
+ }
680
+ });
681
+ }
682
+ if (!isClassOrInterface) {
683
+ if (!this.fmxTypeMap.has(parameterTypeName)) {
684
+ if (parameterTypeName === "number" || parameterTypeName === "string" || parameterTypeName === "boolean" || parameterTypeName === "bigint" || parameterTypeName === "symbol" || parameterTypeName === "undefined" || parameterTypeName === "null" || parameterTypeName === "any" || parameterTypeName === "unknown" || parameterTypeName === "never" || parameterTypeName === "void") {
685
+ fmxParameterType = new Famix.PrimitiveType();
686
+ fmxParameterType.isStub = true;
687
+ }
688
+ else {
689
+ fmxParameterType = new Famix.ParameterType();
690
+ }
691
+ fmxParameterType.name = parameterTypeName;
692
+ this.famixRep.addElement(fmxParameterType);
693
+ this.fmxTypeMap.set(parameterTypeName, fmxParameterType);
694
+ this.fmxElementObjectMap.set(fmxParameterType, typeParameterDeclaration);
695
+ }
696
+ else {
697
+ const result = this.fmxTypeMap.get(parameterTypeName);
698
+ if (result) {
699
+ fmxParameterType = result;
700
+ }
701
+ else {
702
+ throw new Error(`Famix type ${parameterTypeName} is not found in the Type map.`);
703
+ }
704
+ }
705
+ }
706
+ if (!fmxParameterType) {
707
+ throw new Error(`fmxParameterType was undefined for parameterTypeName ${parameterTypeName}`);
708
+ }
709
+ return fmxParameterType;
710
+ }
468
711
  /**
469
712
  * Creates a Famix variable
470
713
  * @param variable A variable
@@ -480,8 +723,9 @@ class EntityDictionary {
480
723
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for variable: ${variable.getName()}. Continuing...`);
481
724
  }
482
725
  const fmxType = this.createOrGetFamixType(variableTypeName, variable);
483
- fmxVariable.setDeclaredType(fmxType);
484
- fmxVariable.setName(variable.getName());
726
+ fmxVariable.declaredType = fmxType;
727
+ fmxVariable.name = variable.getName();
728
+ initFQN(variable, fmxVariable);
485
729
  this.makeFamixIndexFileAnchor(variable, fmxVariable);
486
730
  this.famixRep.addElement(fmxVariable);
487
731
  this.fmxElementObjectMap.set(fmxVariable, variable);
@@ -494,7 +738,8 @@ class EntityDictionary {
494
738
  */
495
739
  createFamixEnum(enumEntity) {
496
740
  const fmxEnum = new Famix.Enum();
497
- fmxEnum.setName(enumEntity.getName());
741
+ fmxEnum.name = enumEntity.getName();
742
+ initFQN(enumEntity, fmxEnum);
498
743
  this.makeFamixIndexFileAnchor(enumEntity, fmxEnum);
499
744
  this.famixRep.addElement(fmxEnum);
500
745
  this.fmxElementObjectMap.set(fmxEnum, enumEntity);
@@ -515,8 +760,9 @@ class EntityDictionary {
515
760
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for enum value: ${enumMember.getName()}. Continuing...`);
516
761
  }
517
762
  const fmxType = this.createOrGetFamixType(enumValueTypeName, enumMember);
518
- fmxEnumValue.setDeclaredType(fmxType);
519
- fmxEnumValue.setName(enumMember.getName());
763
+ fmxEnumValue.declaredType = fmxType;
764
+ fmxEnumValue.name = enumMember.getName();
765
+ initFQN(enumMember, fmxEnumValue);
520
766
  this.makeFamixIndexFileAnchor(enumMember, fmxEnumValue);
521
767
  this.famixRep.addElement(fmxEnumValue);
522
768
  this.fmxElementObjectMap.set(fmxEnumValue, enumMember);
@@ -532,11 +778,12 @@ class EntityDictionary {
532
778
  const fmxDecorator = new Famix.Decorator();
533
779
  const decoratorName = "@" + decorator.getName();
534
780
  const decoratorExpression = decorator.getText().substring(1);
535
- fmxDecorator.setName(decoratorName);
536
- fmxDecorator.setDecoratorExpression(decoratorExpression);
781
+ fmxDecorator.name = decoratorName;
782
+ fmxDecorator.decoratorExpression = decoratorExpression;
537
783
  const decoratedEntityFullyQualifiedName = FQNFunctions.getFQN(decoratedEntity);
538
784
  const fmxDecoratedEntity = this.famixRep.getFamixEntityByFullyQualifiedName(decoratedEntityFullyQualifiedName);
539
- fmxDecorator.setDecoratedEntity(fmxDecoratedEntity);
785
+ fmxDecorator.decoratedEntity = fmxDecoratedEntity;
786
+ initFQN(decorator, fmxDecorator);
540
787
  this.makeFamixIndexFileAnchor(decorator, fmxDecorator);
541
788
  this.famixRep.addElement(fmxDecorator);
542
789
  this.fmxElementObjectMap.set(fmxDecorator, decorator);
@@ -550,10 +797,10 @@ class EntityDictionary {
550
797
  * @returns The Famix model of the comment
551
798
  */
552
799
  createFamixComment(comment, fmxScope, isJSDoc) {
553
- analyze_1.logger.debug(`> NOTE: creating comment ${comment.getText()} in scope ${fmxScope.getName()}.`);
800
+ analyze_1.logger.debug(`> NOTE: creating comment ${comment.getText()} in scope ${fmxScope.name}.`);
554
801
  const fmxComment = new Famix.Comment();
555
- fmxComment.setContainer(fmxScope); // adds comment to the container's comments collection
556
- fmxComment.setIsJSDoc(isJSDoc);
802
+ fmxComment.container = fmxScope; // adds comment to the container's comments collection
803
+ fmxComment.isJSDoc = isJSDoc;
557
804
  this.makeFamixIndexFileAnchor(comment, fmxComment);
558
805
  this.famixRep.addElement(fmxComment);
559
806
  this.fmxElementObjectMap.set(fmxComment, comment);
@@ -568,30 +815,34 @@ class EntityDictionary {
568
815
  createOrGetFamixType(typeName, element) {
569
816
  let fmxType;
570
817
  let isPrimitiveType = false;
571
- let isParameterizedType = false;
572
- analyze_1.logger.debug("Creating (or getting) type: '" + typeName + "' of element: " + (element === null || element === void 0 ? void 0 : element.getText()) + " of kind: " + (element === null || element === void 0 ? void 0 : element.getKindName()));
573
- let ancestor;
818
+ let isParameterType = false;
819
+ analyze_1.logger.debug("Creating (or getting) type: '" + typeName + "' of element: " + element?.getText() + " of kind: " + element?.getKindName());
820
+ let ancestor = undefined;
574
821
  if (element !== undefined) {
575
822
  const typeAncestor = Helpers.findTypeAncestor(element);
823
+ if (!typeAncestor) {
824
+ throw new Error(`Ancestor not found for element ${element.getText()}.`);
825
+ }
576
826
  const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);
577
827
  ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
578
828
  if (!ancestor) {
579
- throw new Error(`Ancestor ${ancestorFullyQualifiedName} not found.`);
829
+ analyze_1.logger.debug(`Ancestor ${FQNFunctions.getFQN(typeAncestor)} not found. Adding the new type.`);
830
+ ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor);
580
831
  }
581
832
  }
582
833
  if (typeName === "number" || typeName === "string" || typeName === "boolean" || typeName === "bigint" || typeName === "symbol" || typeName === "undefined" || typeName === "null" || typeName === "any" || typeName === "unknown" || typeName === "never" || typeName === "void") {
583
834
  isPrimitiveType = true;
584
835
  }
585
836
  if (!isPrimitiveType && typeName.includes("<") && typeName.includes(">") && !(typeName.includes("=>"))) {
586
- isParameterizedType = true;
837
+ isParameterType = true;
587
838
  }
588
839
  if (!this.fmxTypeMap.has(typeName)) {
589
840
  if (isPrimitiveType) {
590
841
  fmxType = new Famix.PrimitiveType();
591
- fmxType.setIsStub(true);
842
+ fmxType.isStub = true;
592
843
  }
593
- else if (isParameterizedType) {
594
- fmxType = new Famix.ParameterizedType();
844
+ else if (isParameterType) {
845
+ fmxType = new Famix.ParameterType();
595
846
  const parameterTypeNames = typeName.substring(typeName.indexOf("<") + 1, typeName.indexOf(">")).split(",").map(s => s.trim());
596
847
  const baseTypeName = typeName.substring(0, typeName.indexOf("<")).trim();
597
848
  parameterTypeNames.forEach(parameterTypeName => {
@@ -599,19 +850,29 @@ class EntityDictionary {
599
850
  fmxType.addArgument(fmxParameterType);
600
851
  });
601
852
  const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);
602
- fmxType.setBaseType(fmxBaseType);
853
+ fmxType.baseType = fmxBaseType;
603
854
  }
604
855
  else {
605
856
  fmxType = new Famix.Type();
606
857
  }
607
- fmxType.setName(typeName);
608
- fmxType.setContainer(ancestor);
858
+ fmxType.name = typeName;
859
+ if (!ancestor) {
860
+ throw new Error(`Ancestor not found for type ${typeName}.`);
861
+ }
862
+ fmxType.container = ancestor;
863
+ initFQN(element, fmxType);
609
864
  this.makeFamixIndexFileAnchor(element, fmxType);
610
865
  this.famixRep.addElement(fmxType);
611
866
  this.fmxTypeMap.set(typeName, fmxType);
612
867
  }
613
868
  else {
614
- fmxType = this.fmxTypeMap.get(typeName);
869
+ const result = this.fmxTypeMap.get(typeName);
870
+ if (result) {
871
+ fmxType = result;
872
+ }
873
+ else {
874
+ throw new Error(`Famix type ${typeName} is not found in the Type map.`);
875
+ }
615
876
  }
616
877
  this.fmxElementObjectMap.set(fmxType, element);
617
878
  return fmxType;
@@ -623,13 +884,20 @@ class EntityDictionary {
623
884
  */
624
885
  createFamixAccess(node, id) {
625
886
  const fmxVar = this.famixRep.getFamixEntityById(id);
887
+ if (!fmxVar) {
888
+ throw new Error(`Famix entity with id ${id} not found, for node ${node.getText()} in ${node.getSourceFile().getBaseName()} at line ${node.getStartLineNumber()}.`);
889
+ }
890
+ analyze_1.logger.debug(`Creating FamixAccess. Node: [${node.getKindName()}] '${node.getText()}' at line ${node.getStartLineNumber()} in ${node.getSourceFile().getBaseName()}, id: ${id} refers to fmxVar '${fmxVar.fullyQualifiedName}'.`);
626
891
  const nodeReferenceAncestor = Helpers.findAncestor(node);
627
892
  const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);
628
- const accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
893
+ let accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
894
+ if (!accessor) {
895
+ analyze_1.logger.error(`Ancestor ${ancestorFullyQualifiedName} of kind ${nodeReferenceAncestor.getKindName()} not found.`);
896
+ // accessor = this.createOrGetFamixType(ancestorFullyQualifiedName, nodeReferenceAncestor as TypeDeclaration);
897
+ }
629
898
  const fmxAccess = new Famix.Access();
630
- fmxAccess.setAccessor(accessor);
631
- fmxAccess.setVariable(fmxVar);
632
- this.makeFamixIndexFileAnchor(node, fmxAccess);
899
+ fmxAccess.accessor = accessor;
900
+ fmxAccess.variable = fmxVar;
633
901
  this.famixRep.addElement(fmxAccess);
634
902
  this.fmxElementObjectMap.set(fmxAccess, node);
635
903
  }
@@ -647,11 +915,10 @@ class EntityDictionary {
647
915
  const receiverFullyQualifiedName = FQNFunctions.getFQN(m.getParent());
648
916
  const receiver = this.famixRep.getFamixEntityByFullyQualifiedName(receiverFullyQualifiedName);
649
917
  const fmxInvocation = new Famix.Invocation();
650
- fmxInvocation.setSender(sender);
651
- fmxInvocation.setReceiver(receiver);
918
+ fmxInvocation.sender = sender;
919
+ fmxInvocation.receiver = receiver;
652
920
  fmxInvocation.addCandidate(fmxMethodOrFunction);
653
- fmxInvocation.setSignature(fmxMethodOrFunction.getSignature());
654
- this.makeFamixIndexFileAnchor(node, fmxInvocation);
921
+ fmxInvocation.signature = fmxMethodOrFunction.signature;
655
922
  this.famixRep.addElement(fmxInvocation);
656
923
  this.fmxElementObjectMap.set(fmxInvocation, node);
657
924
  }
@@ -672,11 +939,17 @@ class EntityDictionary {
672
939
  else {
673
940
  subClass = this.fmxInterfaceMap.get(classFullyQualifiedName);
674
941
  }
942
+ if (!subClass) {
943
+ throw new Error(`Subclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
944
+ }
675
945
  let inhClassName;
676
946
  let inhClassFullyQualifiedName;
677
947
  let superClass;
678
948
  if (inhClass instanceof ts_morph_1.ClassDeclaration || inhClass instanceof ts_morph_1.InterfaceDeclaration) {
679
949
  inhClassName = inhClass.getName();
950
+ if (!inhClassName) {
951
+ throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);
952
+ }
680
953
  inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);
681
954
  if (inhClass instanceof ts_morph_1.ClassDeclaration) {
682
955
  superClass = this.fmxClassMap.get(inhClassFullyQualifiedName);
@@ -684,6 +957,9 @@ class EntityDictionary {
684
957
  else {
685
958
  superClass = this.fmxInterfaceMap.get(inhClassFullyQualifiedName);
686
959
  }
960
+ if (!superClass) {
961
+ throw new Error(`Superclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
962
+ }
687
963
  }
688
964
  else {
689
965
  // inhClass is an ExpressionWithTypeArguments
@@ -701,17 +977,24 @@ class EntityDictionary {
701
977
  this.fmxInterfaceMap.set(inhClassFullyQualifiedName, superClass);
702
978
  }
703
979
  this.fmxElementObjectMap.set(superClass, inhClass);
704
- superClass.setName(inhClassName);
705
- superClass.setFullyQualifiedName(inhClassFullyQualifiedName);
706
- superClass.setIsStub(true);
980
+ superClass.name = inhClassName;
981
+ superClass.fullyQualifiedName = inhClassFullyQualifiedName;
982
+ superClass.isStub = true;
707
983
  this.makeFamixIndexFileAnchor(inhClass, superClass);
708
984
  this.famixRep.addElement(superClass);
709
985
  }
710
- fmxInheritance.setSubclass(subClass);
711
- fmxInheritance.setSuperclass(superClass);
712
- this.makeFamixIndexFileAnchor(null, fmxInheritance);
986
+ fmxInheritance.subclass = subClass;
987
+ fmxInheritance.superclass = superClass;
713
988
  this.famixRep.addElement(fmxInheritance);
714
- this.fmxElementObjectMap.set(fmxInheritance, null);
989
+ // We don't map inheritance to the source code element because there are two elements (super, sub)
990
+ // this.fmxElementObjectMap.set(fmxInheritance, null);
991
+ }
992
+ createFamixImportClause(importedEntity, importingEntity) {
993
+ const fmxImportClause = new Famix.ImportClause();
994
+ fmxImportClause.importedEntity = importedEntity;
995
+ fmxImportClause.importingEntity = importingEntity;
996
+ importingEntity.addOutgoingImport(fmxImportClause);
997
+ this.famixRep.addElement(fmxImportClause);
715
998
  }
716
999
  /**
717
1000
  * Creates a Famix import clause
@@ -723,12 +1006,11 @@ class EntityDictionary {
723
1006
  * @param isInExports A boolean indicating if the imported entity is in the exports
724
1007
  * @param isDefaultExport A boolean indicating if the imported entity is a default export
725
1008
  */
726
- createFamixImportClause(importClauseInfo) {
727
- var _a, _b;
728
- const { importDeclaration, importer, moduleSpecifierFilePath, importElement, isInExports, isDefaultExport } = importClauseInfo;
1009
+ oldCreateFamixImportClause(importClauseInfo) {
1010
+ const { importDeclaration, importerSourceFile: importer, moduleSpecifierFilePath, importElement, isInExports, isDefaultExport } = importClauseInfo;
729
1011
  analyze_1.logger.debug(`createFamixImportClause: Creating import clause:`);
730
1012
  const fmxImportClause = new Famix.ImportClause();
731
- let importedEntity;
1013
+ let importedEntity = undefined;
732
1014
  let importedEntityName;
733
1015
  const absolutePathProject = this.famixRep.getAbsolutePath();
734
1016
  const absolutePath = path_1.default.normalize(moduleSpecifierFilePath);
@@ -736,6 +1018,7 @@ class EntityDictionary {
736
1018
  const pathInProject = this.convertToRelativePath(absolutePath, absolutePathProject).replace(/\\/g, "/");
737
1019
  let pathName = "{" + pathInProject + "}.";
738
1020
  // Named imports, e.g. import { ClassW } from "./complexExportModule";
1021
+ // Start with simple import clause (without referring to the actual variable)
739
1022
  if (importDeclaration instanceof ts_morph_1.ImportDeclaration
740
1023
  && importElement instanceof ts_morph_1.ImportSpecifier) {
741
1024
  importedEntityName = importElement.getName();
@@ -745,54 +1028,404 @@ class EntityDictionary {
745
1028
  }
746
1029
  if (importedEntity === undefined) {
747
1030
  importedEntity = new Famix.NamedEntity();
748
- importedEntity.setName(importedEntityName);
1031
+ importedEntity.name = importedEntityName;
749
1032
  if (!isInExports) {
750
- importedEntity.setIsStub(true);
1033
+ importedEntity.isStub = true;
751
1034
  }
1035
+ importedEntity.fullyQualifiedName = pathName;
752
1036
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
753
- importedEntity.setFullyQualifiedName(pathName);
1037
+ // must add entity to repository
1038
+ this.famixRep.addElement(importedEntity);
754
1039
  }
755
1040
  }
756
1041
  // handle import equals declarations, e.g. import myModule = require("./complexExportModule");
757
1042
  // TypeScript can't determine the type of the imported module, so we create a Module entity
758
1043
  else if (importDeclaration instanceof ts_morph_1.ImportEqualsDeclaration) {
759
- importedEntityName = importDeclaration === null || importDeclaration === void 0 ? void 0 : importDeclaration.getName();
1044
+ importedEntityName = importDeclaration?.getName();
760
1045
  pathName = pathName + importedEntityName;
761
1046
  importedEntity = new Famix.StructuralEntity();
762
- importedEntity.setName(importedEntityName);
1047
+ importedEntity.name = importedEntityName;
1048
+ initFQN(importDeclaration, importedEntity);
763
1049
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
764
- importedEntity.setFullyQualifiedName(pathName);
765
- const anyType = this.createOrGetFamixType('any', undefined);
766
- importedEntity.setDeclaredType(anyType);
1050
+ importedEntity.fullyQualifiedName = pathName;
1051
+ const anyType = this.createOrGetFamixType('any', importDeclaration);
1052
+ importedEntity.declaredType = anyType;
767
1053
  }
768
1054
  else { // default imports, e.g. import ClassW from "./complexExportModule";
769
1055
  importedEntityName = importElement.getText();
770
1056
  pathName = pathName + (isDefaultExport ? "defaultExport" : "namespaceExport");
771
1057
  importedEntity = new Famix.NamedEntity();
772
- importedEntity.setName(importedEntityName);
1058
+ importedEntity.name = importedEntityName;
1059
+ importedEntity.fullyQualifiedName = pathName;
773
1060
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
774
- importedEntity.setFullyQualifiedName(pathName);
775
1061
  }
776
- this.famixRep.addElement(importedEntity);
1062
+ // I don't think it should be added to the repository if it exists already
1063
+ if (!isInExports)
1064
+ this.famixRep.addElement(importedEntity);
777
1065
  const importerFullyQualifiedName = FQNFunctions.getFQN(importer);
778
1066
  const fmxImporter = this.famixRep.getFamixEntityByFullyQualifiedName(importerFullyQualifiedName);
779
- fmxImportClause.setImportingEntity(fmxImporter);
780
- fmxImportClause.setImportedEntity(importedEntity);
1067
+ fmxImportClause.importingEntity = fmxImporter;
1068
+ fmxImportClause.importedEntity = importedEntity;
781
1069
  if (importDeclaration instanceof ts_morph_1.ImportEqualsDeclaration) {
782
- fmxImportClause.setModuleSpecifier(importDeclaration === null || importDeclaration === void 0 ? void 0 : importDeclaration.getModuleReference().getText());
1070
+ fmxImportClause.moduleSpecifier = importDeclaration?.getModuleReference().getText();
783
1071
  }
784
1072
  else {
785
- fmxImportClause.setModuleSpecifier(importDeclaration === null || importDeclaration === void 0 ? void 0 : importDeclaration.getModuleSpecifierValue());
1073
+ fmxImportClause.moduleSpecifier = importDeclaration?.getModuleSpecifierValue();
786
1074
  }
787
- analyze_1.logger.debug(`createFamixImportClause: ${(_a = fmxImportClause.getImportedEntity()) === null || _a === void 0 ? void 0 : _a.getName()} (of type ${Helpers.getSubTypeName(fmxImportClause.getImportedEntity())}) is imported by ${(_b = fmxImportClause.getImportingEntity()) === null || _b === void 0 ? void 0 : _b.getName()}`);
788
- // make an index file anchor for the import clause
789
- this.makeFamixIndexFileAnchor(importDeclaration, fmxImportClause);
1075
+ analyze_1.logger.debug(`createFamixImportClause: ${fmxImportClause.importedEntity?.name} (of type ${Helpers.getSubTypeName(fmxImportClause.importedEntity)}) is imported by ${fmxImportClause.importingEntity?.name}`);
790
1076
  fmxImporter.addOutgoingImport(fmxImportClause);
791
1077
  this.famixRep.addElement(fmxImportClause);
792
- this.fmxElementObjectMap.set(fmxImportClause, importDeclaration);
1078
+ if (importDeclaration)
1079
+ this.fmxElementObjectMap.set(fmxImportClause, importDeclaration);
1080
+ }
1081
+ /**
1082
+ * Creates a Famix Arrow Function
1083
+ * @param arrowExpression An Expression
1084
+ * @returns The Famix model of the variable
1085
+ */
1086
+ createFamixArrowFunction(arrowExpression, currentCC) {
1087
+ let fmxArrowFunction;
1088
+ const arrowFunction = arrowExpression.asKindOrThrow(ts_morph_1.SyntaxKind.ArrowFunction);
1089
+ const isGeneric = arrowFunction.getTypeParameters().length > 0;
1090
+ if (isGeneric) {
1091
+ fmxArrowFunction = new Famix.ParametricArrowFunction();
1092
+ }
1093
+ else {
1094
+ fmxArrowFunction = new Famix.ArrowFunction();
1095
+ }
1096
+ // Get the parent of the arrow function (the variable declaration)
1097
+ const parent = arrowFunction.getParentIfKind(ts_morph_1.SyntaxKind.VariableDeclaration);
1098
+ let functionName = '(NO_NAME)';
1099
+ if (parent && parent instanceof ts_morph_1.VariableDeclaration) {
1100
+ // Get the name of the variable
1101
+ functionName = parent.getName();
1102
+ }
1103
+ if (functionName) {
1104
+ fmxArrowFunction.name = functionName;
1105
+ }
1106
+ else {
1107
+ fmxArrowFunction.name = "anonymous";
1108
+ }
1109
+ // Signature of an arrow function is (parameters) => return_type
1110
+ const parametersSignature = arrowFunction.getParameters().map(p => p.getText()).join(", ");
1111
+ const returnTypeSignature = arrowFunction.getReturnType().getText();
1112
+ fmxArrowFunction.signature = `(${parametersSignature}) => ${returnTypeSignature}`;
1113
+ fmxArrowFunction.cyclomaticComplexity = currentCC[fmxArrowFunction.name];
1114
+ let functionTypeName = this.UNKNOWN_VALUE;
1115
+ try {
1116
+ functionTypeName = arrowFunction.getReturnType().getText().trim();
1117
+ }
1118
+ catch (error) {
1119
+ analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${functionName}. Continuing...`);
1120
+ }
1121
+ const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction);
1122
+ fmxArrowFunction.declaredType = fmxType;
1123
+ fmxArrowFunction.numberOfLinesOfCode = arrowFunction.getEndLineNumber() - arrowFunction.getStartLineNumber();
1124
+ const parameters = arrowFunction.getParameters();
1125
+ fmxArrowFunction.numberOfParameters = parameters.length;
1126
+ fmxArrowFunction.numberOfStatements = arrowFunction.getStatements().length;
1127
+ initFQN(arrowExpression, fmxArrowFunction);
1128
+ this.makeFamixIndexFileAnchor(arrowExpression, fmxArrowFunction);
1129
+ this.famixRep.addElement(fmxArrowFunction);
1130
+ this.fmxElementObjectMap.set(fmxArrowFunction, arrowFunction);
1131
+ return fmxArrowFunction;
1132
+ }
1133
+ /**
1134
+ * Creates a Famix concretisation
1135
+ * @param cls A class
1136
+ * @returns The Famix model of the concretisation
1137
+ */
1138
+ createFamixConcretisation(conEntity, genEntity) {
1139
+ const fmxConcretisation = new Famix.Concretisation();
1140
+ fmxConcretisation.concreteEntity = conEntity;
1141
+ fmxConcretisation.genericEntity = genEntity;
1142
+ // this.fmxElementObjectMap.set(fmxConcretisation,null);
1143
+ this.famixRep.addElement(fmxConcretisation);
1144
+ const parameterConcretisation = this.createFamixParameterConcretisation(fmxConcretisation);
1145
+ return fmxConcretisation;
1146
+ }
1147
+ /**
1148
+ * Creates a Famix concretisation
1149
+ * @param concretisation A FamixConcretisation
1150
+ * @returns The Famix model of the ParameterConcrestisation
1151
+ */
1152
+ createFamixParameterConcretisation(concretisation) {
1153
+ const conClass = concretisation.concreteEntity;
1154
+ const genClass = concretisation.genericEntity;
1155
+ analyze_1.logger.debug(`Creating parameter concretisation between ${conClass.fullyQualifiedName} and ${genClass.fullyQualifiedName}`);
1156
+ const parameterConcretisations = this.famixRep._getAllEntitiesWithType("ParameterConcretisation");
1157
+ const concreteParameters = conClass.concreteParameters;
1158
+ const genericParameters = genClass.genericParameters;
1159
+ let conClassTypeParametersIterator = concreteParameters.values();
1160
+ let genClassTypeParametersIterator = genericParameters.values();
1161
+ let fmxParameterConcretisation = undefined;
1162
+ for (let i = 0; i < genericParameters.size; i++) {
1163
+ const conClassTypeParameter = conClassTypeParametersIterator.next().value;
1164
+ const genClassTypeParameter = genClassTypeParametersIterator.next().value;
1165
+ let createParameterConcretisation = true;
1166
+ if (conClassTypeParameter && genClassTypeParameter && conClassTypeParameter.name != genClassTypeParameter.name) {
1167
+ parameterConcretisations.forEach((param) => {
1168
+ if (conClassTypeParameter.name == param.concreteParameter.name && genClassTypeParameter.name == param.genericParameter.name) {
1169
+ createParameterConcretisation = false;
1170
+ fmxParameterConcretisation = param;
1171
+ }
1172
+ });
1173
+ if (createParameterConcretisation) {
1174
+ fmxParameterConcretisation = new Famix.ParameterConcretisation();
1175
+ fmxParameterConcretisation.genericParameter = genClassTypeParameter;
1176
+ fmxParameterConcretisation.concreteParameter = conClassTypeParameter;
1177
+ fmxParameterConcretisation.addConcretisation(concretisation);
1178
+ // this.fmxElementObjectMap.set(fmxParameterConcretisation,null);
1179
+ }
1180
+ else {
1181
+ if (!fmxParameterConcretisation) {
1182
+ throw new Error(`fmxParameterConcretisation was undefined for concretisation with generic parameter ${genClassTypeParameter.name} and concrete parameter ${conClassTypeParameter.name}`);
1183
+ }
1184
+ fmxParameterConcretisation.addConcretisation(concretisation);
1185
+ }
1186
+ this.famixRep.addElement(fmxParameterConcretisation);
1187
+ }
1188
+ }
1189
+ if (!fmxParameterConcretisation) {
1190
+ analyze_1.logger.error(`fmxParameterConcretisation was undefined for concretisation with concrete entity ${conClass.fullyQualifiedName} and generic entity ${genClass.fullyQualifiedName}`);
1191
+ }
1192
+ return fmxParameterConcretisation;
1193
+ }
1194
+ /**
1195
+ * Creates a Famix concretisation between two classes or two interfaces
1196
+ * @param element A class or an Interface
1197
+ */
1198
+ createFamixConcretisationClassOrInterfaceSpecialisation(element) {
1199
+ const superEntity = element.getExtends();
1200
+ let superEntityArray;
1201
+ if (superEntity) {
1202
+ superEntityArray = Array.isArray(superEntity) ? superEntity : [superEntity];
1203
+ }
1204
+ if (superEntityArray && superEntityArray.length > 0) {
1205
+ superEntityArray.forEach(entity => {
1206
+ let entityIsGeneric;
1207
+ const superEntitySymbol = entity.getExpression().getSymbolOrThrow();
1208
+ let superEntityDeclaration;
1209
+ if (superEntity instanceof ts_morph_1.ExpressionWithTypeArguments) {
1210
+ superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts_morph_1.ts.SyntaxKind.ClassDeclaration);
1211
+ }
1212
+ else {
1213
+ superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts_morph_1.ts.SyntaxKind.InterfaceDeclaration);
1214
+ }
1215
+ if (superEntityDeclaration) {
1216
+ entityIsGeneric = superEntityDeclaration.getTypeParameters().length > 0;
1217
+ }
1218
+ if (entityIsGeneric) {
1219
+ let EntityDeclaration;
1220
+ let genEntity;
1221
+ if (superEntity instanceof ts_morph_1.ExpressionWithTypeArguments) {
1222
+ EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0];
1223
+ genEntity = this.createOrGetFamixClass(EntityDeclaration);
1224
+ }
1225
+ else {
1226
+ EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0];
1227
+ genEntity = this.createOrGetFamixInterface(EntityDeclaration);
1228
+ }
1229
+ const genParams = EntityDeclaration.getTypeParameters().map((param) => param.getText());
1230
+ const args = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1231
+ const conParams = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1232
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1233
+ let conEntity;
1234
+ conEntity = this.createOrGetFamixConcreteElement(genEntity, EntityDeclaration, args);
1235
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1236
+ let createConcretisation = true;
1237
+ concretisations.forEach((conc) => {
1238
+ if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName) {
1239
+ createConcretisation = false;
1240
+ }
1241
+ });
1242
+ if (createConcretisation) {
1243
+ const fmxConcretisation = this.createFamixConcretisation(conEntity, genEntity);
1244
+ }
1245
+ }
1246
+ }
1247
+ });
1248
+ }
1249
+ // TODO: This function seems unfinished
1250
+ }
1251
+ /**
1252
+ * Creates a Famix concretisation between a class and its instanciations
1253
+ * @param cls A class
1254
+ */
1255
+ createFamixConcretisationGenericInstantiation(cls) {
1256
+ const isGeneric = cls.getTypeParameters().length > 0;
1257
+ if (isGeneric) {
1258
+ const instances = cls.getSourceFile().getDescendantsOfKind(ts_morph_1.ts.SyntaxKind.NewExpression)
1259
+ .filter(newExpr => {
1260
+ const expression = newExpr.getExpression();
1261
+ return expression.getText() === cls.getName();
1262
+ });
1263
+ instances.forEach(instance => {
1264
+ const instanceIsGeneric = instance.getTypeArguments().length > 0;
1265
+ if (instanceIsGeneric) {
1266
+ const conParams = instance.getTypeArguments().map((param) => param.getText());
1267
+ const genEntity = this.createOrGetFamixClass(cls);
1268
+ const genParams = cls.getTypeParameters().map((param) => param.getText());
1269
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1270
+ let conEntity;
1271
+ conEntity = this.createOrGetFamixConcreteElement(genEntity, cls, instance.getTypeArguments());
1272
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1273
+ let createConcretisation = true;
1274
+ concretisations.forEach((conc) => {
1275
+ if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName) {
1276
+ createConcretisation = false;
1277
+ }
1278
+ });
1279
+ if (createConcretisation) {
1280
+ const fmxConcretisation = this.createFamixConcretisation(conEntity, genEntity);
1281
+ }
1282
+ }
1283
+ }
1284
+ });
1285
+ }
1286
+ // TODO: This function seems unfinished
1287
+ }
1288
+ /**
1289
+ * Creates a Famix concretisation between a class and its instanciations
1290
+ * @param func A function
1291
+ */
1292
+ createFamixConcretisationFunctionInstantiation(element) {
1293
+ const isGeneric = element.getTypeParameters().length > 0;
1294
+ if (isGeneric) {
1295
+ const genParams = element.getTypeParameters().map(param => param.getText());
1296
+ const uses = element.findReferencesAsNodes();
1297
+ uses.forEach(usage => {
1298
+ let currentNode = usage;
1299
+ while (currentNode) {
1300
+ if (currentNode.getKind() === ts_morph_1.SyntaxKind.CallExpression) {
1301
+ const callExpression = currentNode.asKind(ts_morph_1.SyntaxKind.CallExpression);
1302
+ if (!callExpression) {
1303
+ throw new Error(`CallExpression not found for ${currentNode.getText()}`);
1304
+ }
1305
+ const instanceIsGeneric = callExpression.getTypeArguments().length > 0;
1306
+ if (instanceIsGeneric) {
1307
+ const args = callExpression.getTypeArguments();
1308
+ const conParams = callExpression.getTypeArguments().map(param => param.getText());
1309
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1310
+ let genElement;
1311
+ if (element instanceof ts_morph_1.FunctionDeclaration) {
1312
+ genElement = this.createOrGetFamixFunction(element, {});
1313
+ }
1314
+ else {
1315
+ genElement = this.createOrGetFamixMethod(element, {});
1316
+ }
1317
+ let concElement;
1318
+ concElement = this.createOrGetFamixConcreteElement(genElement, element, args);
1319
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1320
+ let createConcretisation = true;
1321
+ concretisations.forEach((conc) => {
1322
+ if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName) {
1323
+ createConcretisation = false;
1324
+ }
1325
+ });
1326
+ if (createConcretisation) {
1327
+ const fmxConcretisation = this.createFamixConcretisation(concElement, genElement);
1328
+ }
1329
+ }
1330
+ }
1331
+ break;
1332
+ }
1333
+ // Remonter à l'élément parent (utile si le nœud de référence est un enfant)
1334
+ currentNode = currentNode.getParent();
1335
+ }
1336
+ });
1337
+ }
1338
+ }
1339
+ /**
1340
+ * Creates a Famix concretisation between a class and an interface
1341
+ * @param cls A class
1342
+ */
1343
+ createFamixConcretisationInterfaceClass(cls) {
1344
+ const superInterfaces = cls.getImplements();
1345
+ superInterfaces.forEach(interfaceType => {
1346
+ const interfaceIsGeneric = interfaceType.getTypeArguments().length > 0;
1347
+ if (interfaceIsGeneric) {
1348
+ const interfaceDeclaration = interfaceType.getExpression().getSymbol()?.getDeclarations()[0];
1349
+ const genParams = interfaceDeclaration.getTypeParameters().map((param) => param.getText());
1350
+ const conParams = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1351
+ const args = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1352
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1353
+ const genInterface = this.createOrGetFamixInterface(interfaceDeclaration);
1354
+ const conInterface = this.createOrGetFamixConcreteElement(genInterface, interfaceDeclaration, args);
1355
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1356
+ let createConcretisation = true;
1357
+ concretisations.forEach((conc) => {
1358
+ if (genInterface.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conInterface.fullyQualifiedName) {
1359
+ createConcretisation = false;
1360
+ }
1361
+ });
1362
+ if (createConcretisation) {
1363
+ const fmxConcretisation = this.createFamixConcretisation(conInterface, genInterface);
1364
+ }
1365
+ }
1366
+ }
1367
+ });
1368
+ }
1369
+ /**
1370
+ * Creates a Famix concretisation between an interface and a Type
1371
+ * @param element A variable or a function
1372
+ * @param inter An interface
1373
+ */
1374
+ createFamixConcretisationTypeInstanciation(element) {
1375
+ const isGeneric = element.getTypeParameters().length > 0;
1376
+ if (isGeneric) {
1377
+ const genParams = element.getTypeParameters().map(param => param.getText());
1378
+ const uses = element.findReferencesAsNodes();
1379
+ uses.forEach(use => {
1380
+ let parentNode = use.getParent();
1381
+ while (parentNode) {
1382
+ if (parentNode.getKind() === ts_morph_1.SyntaxKind.TypeReference) {
1383
+ const typeReferenceNode = parentNode.asKind(ts_morph_1.SyntaxKind.TypeReference);
1384
+ if (!typeReferenceNode) {
1385
+ throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);
1386
+ }
1387
+ const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;
1388
+ if (typeReferenceNodeIsGeneric) { }
1389
+ const args = typeReferenceNode.getTypeArguments();
1390
+ const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());
1391
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1392
+ let genElement;
1393
+ if (element instanceof ts_morph_1.ClassDeclaration) {
1394
+ genElement = this.createOrGetFamixClass(element);
1395
+ }
1396
+ else {
1397
+ genElement = this.createOrGetFamixInterface(element);
1398
+ }
1399
+ let concElement;
1400
+ concElement = this.createOrGetFamixConcreteElement(genElement, element, args);
1401
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1402
+ let createConcretisation = true;
1403
+ concretisations.forEach((conc) => {
1404
+ if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName) {
1405
+ createConcretisation = false;
1406
+ }
1407
+ });
1408
+ if (createConcretisation) {
1409
+ const fmxConcretisation = this.createFamixConcretisation(concElement, genElement);
1410
+ }
1411
+ }
1412
+ break;
1413
+ }
1414
+ parentNode = parentNode.getParent();
1415
+ }
1416
+ });
1417
+ }
793
1418
  }
794
1419
  convertToRelativePath(absolutePath, absolutePathProject) {
795
1420
  return absolutePath.replace(absolutePathProject, "").slice(1);
796
1421
  }
797
1422
  }
798
1423
  exports.EntityDictionary = EntityDictionary;
1424
+ function initFQN(sourceElement, famixElement) {
1425
+ if (!(sourceElement instanceof ts_morph_1.CommentRange)) {
1426
+ const fqn = FQNFunctions.getFQN(sourceElement);
1427
+ analyze_1.logger.debug("Setting fully qualified name for " + famixElement.getJSON() + " to " + fqn);
1428
+ famixElement.fullyQualifiedName = fqn;
1429
+ }
1430
+ }
1431
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRW50aXR5RGljdGlvbmFyeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9mYW1peF9mdW5jdGlvbnMvRW50aXR5RGljdGlvbmFyeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHVDQUE2bEI7QUFDN2xCLDhFQUFnRjtBQUNoRixnRUFBa0Q7QUFDbEQsb0VBQWdFO0FBQ2hFLHdDQUE0QztBQUM1QywwRUFBaUQ7QUFDakQsNERBQThDO0FBQzlDLHFEQUF1QztBQUN2QyxnREFBd0I7QUFDeEIsb0RBQXVCO0FBVXZCLE1BQWEsZ0JBQWdCO0lBY3pCO1FBWk8sYUFBUSxHQUFHLElBQUksa0NBQWUsRUFBRSxDQUFDO1FBQ2hDLGdCQUFXLEdBQUcsSUFBSSxHQUFHLEVBQXVCLENBQUMsQ0FBQyw0Q0FBNEM7UUFDMUYsZ0JBQVcsR0FBRyxJQUFJLEdBQUcsRUFBK0MsQ0FBQyxDQUFDLDREQUE0RDtRQUNsSSxvQkFBZSxHQUFHLElBQUksR0FBRyxFQUF1RCxDQUFDLENBQUMsZ0RBQWdEO1FBQ2xJLGlCQUFZLEdBQUcsSUFBSSxHQUFHLEVBQXdCLENBQUMsQ0FBQyxnREFBZ0Q7UUFDaEcsZUFBVSxHQUFHLElBQUksR0FBRyxFQUE2QyxDQUFDLENBQUMsa0RBQWtEO1FBQ3JILGVBQVUsR0FBRyxJQUFJLEdBQUcsRUFBa0UsQ0FBQyxDQUFDLDJDQUEyQztRQUNuSSw0QkFBdUIsR0FBRyxJQUFJLEdBQThGLENBQUEsQ0FBQywrQ0FBK0M7UUFDNUssa0JBQWEsR0FBRyxnQ0FBZ0MsQ0FBQyxDQUFDLDZDQUE2QztRQUNoRyx3QkFBbUIsR0FBRyxJQUFJLEdBQUcsRUFBa0MsQ0FBQztRQUNoRSw0QkFBdUIsR0FBRyxJQUFJLEdBQUcsRUFBa0MsQ0FBQztRQUd2RSxJQUFJLENBQUMsUUFBUSxDQUFDLHNCQUFzQixDQUFDLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO0lBQ25FLENBQUM7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RyxLQUFLLENBQUM7Z0NBQ2pDLENBQUM7NEJBQ0wsQ0FBQyxDQUFDLENBQUM7NEJBRUgsSUFBSSxvQkFBb0IsRUFBRSxDQUFDO2dDQUN2QixNQUFNLGlCQUFpQixHQUEwQixJQUFJLENBQUMseUJBQXlCLENBQUMsV0FBVyxFQUFDLFVBQVUsQ0FBQyxDQUFDOzRCQUM1RyxDQUFDO3dCQUNMLENBQUM7d0JBQ0wsTUFBTTtvQkFDVixDQUFDO29CQUNELFVBQVUsR0FBRyxVQUFVLENBQUMsU0FBUyxFQUFFLENBQUM7Z0JBQ3hDLENBQUM7WUFDTCxDQUFDLENBQUMsQ0FBQztRQUNQLENBQUM7SUFDTCxDQUFDO0lBRU0scUJBQXFCLENBQUMsWUFBb0IsRUFBRSxtQkFBMkI7UUFDMUUsT0FBTyxZQUFZLENBQUMsT0FBTyxDQUFDLG1CQUFtQixFQUFFLEVBQUUsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsRSxDQUFDO0NBQ0o7QUE3aERELDRDQTZoREM7QUFDRCxTQUFTLE9BQU8sQ0FBQyxhQUFnQyxFQUFFLFlBQWlDO0lBQ2hGLElBQUksQ0FBQyxDQUFDLGFBQWEsWUFBWSx1QkFBWSxDQUFDLEVBQUUsQ0FBQztRQUMzQyxNQUFNLEdBQUcsR0FBRyxZQUFZLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1FBQy9DLGdCQUFNLENBQUMsS0FBSyxDQUFDLG1DQUFtQyxHQUFHLFlBQVksQ0FBQyxPQUFPLEVBQUUsR0FBRyxNQUFNLEdBQUcsR0FBRyxDQUFDLENBQUM7UUFDekYsWUFBa0MsQ0FBQyxrQkFBa0IsR0FBRyxHQUFHLENBQUM7SUFDakUsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDbGFzc0RlY2xhcmF0aW9uLCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uLCBGdW5jdGlvbkRlY2xhcmF0aW9uLCBJZGVudGlmaWVyLCBJbnRlcmZhY2VEZWNsYXJhdGlvbiwgTWV0aG9kRGVjbGFyYXRpb24sIE1ldGhvZFNpZ25hdHVyZSwgTW9kdWxlRGVjbGFyYXRpb24sIFByb3BlcnR5RGVjbGFyYXRpb24sIFByb3BlcnR5U2lnbmF0dXJlLCBTb3VyY2VGaWxlLCBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb24sIFZhcmlhYmxlRGVjbGFyYXRpb24sIFBhcmFtZXRlckRlY2xhcmF0aW9uLCBEZWNvcmF0b3IsIEdldEFjY2Vzc29yRGVjbGFyYXRpb24sIFNldEFjY2Vzc29yRGVjbGFyYXRpb24sIEltcG9ydFNwZWNpZmllciwgQ29tbWVudFJhbmdlLCBFbnVtRGVjbGFyYXRpb24sIEVudW1NZW1iZXIsIFR5cGVBbGlhc0RlY2xhcmF0aW9uLCBGdW5jdGlvbkV4cHJlc3Npb24sIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cywgSW1wb3J0RGVjbGFyYXRpb24sIEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uLCBTeW50YXhLaW5kLCBFeHByZXNzaW9uLCBUeXBlTm9kZSwgTm9kZSwgdHMsIFNjb3BlLCBUeXBlLCBBcnJvd0Z1bmN0aW9uIH0gZnJvbSBcInRzLW1vcnBoXCI7XG5pbXBvcnQgeyBpc0FtYmllbnQsIGlzTmFtZXNwYWNlIH0gZnJvbSBcIi4uL2FuYWx5emVfZnVuY3Rpb25zL3Byb2Nlc3NfZnVuY3Rpb25zXCI7XG5pbXBvcnQgKiBhcyBGYW1peCBmcm9tIFwiLi4vbGliL2ZhbWl4L21vZGVsL2ZhbWl4XCI7XG5pbXBvcnQgeyBGYW1peFJlcG9zaXRvcnkgfSBmcm9tIFwiLi4vbGliL2ZhbWl4L2ZhbWl4X3JlcG9zaXRvcnlcIjtcbmltcG9ydCB7IGxvZ2dlciwgY29uZmlnIH0gZnJvbSBcIi4uL2FuYWx5emVcIjtcbmltcG9ydCBHcmFwaGVtZVNwbGl0dGVyIGZyb20gXCJncmFwaGVtZS1zcGxpdHRlclwiO1xuaW1wb3J0ICogYXMgSGVscGVycyBmcm9tIFwiLi9oZWxwZXJzX2NyZWF0aW9uXCI7XG5pbXBvcnQgKiBhcyBGUU5GdW5jdGlvbnMgZnJvbSBcIi4uL2ZxblwiO1xuaW1wb3J0IHBhdGggZnJvbSBcInBhdGhcIjtcbmltcG9ydCBfIGZyb20gJ2xvZGFzaCc7XG5cbmV4cG9ydCB0eXBlIFRTTW9ycGhPYmplY3RUeXBlID0gSW1wb3J0RGVjbGFyYXRpb24gfCBJbXBvcnRFcXVhbHNEZWNsYXJhdGlvbiB8IFNvdXJjZUZpbGUgfCBNb2R1bGVEZWNsYXJhdGlvbiB8IENsYXNzRGVjbGFyYXRpb24gfCBJbnRlcmZhY2VEZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uIHwgQ29uc3RydWN0b3JEZWNsYXJhdGlvbiB8IE1ldGhvZFNpZ25hdHVyZSB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24gfCBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFZhcmlhYmxlRGVjbGFyYXRpb24gfCBQcm9wZXJ0eURlY2xhcmF0aW9uIHwgUHJvcGVydHlTaWduYXR1cmUgfCBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb24gfCBJZGVudGlmaWVyIHwgRGVjb3JhdG9yIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBJbXBvcnRTcGVjaWZpZXIgfCBDb21tZW50UmFuZ2UgfCBFbnVtRGVjbGFyYXRpb24gfCBFbnVtTWVtYmVyIHwgVHlwZUFsaWFzRGVjbGFyYXRpb24gfCBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHM7XG5cbmV4cG9ydCB0eXBlIFR5cGVEZWNsYXJhdGlvbiA9IFR5cGVBbGlhc0RlY2xhcmF0aW9uIHwgUHJvcGVydHlEZWNsYXJhdGlvbiB8IFByb3BlcnR5U2lnbmF0dXJlIHwgTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgRnVuY3Rpb25FeHByZXNzaW9uIHwgUGFyYW1ldGVyRGVjbGFyYXRpb24gfCBWYXJpYWJsZURlY2xhcmF0aW9uIHwgRW51bU1lbWJlciB8IEltcG9ydEVxdWFsc0RlY2xhcmF0aW9uO1xuXG50eXBlIFBhcmFtZXRyaWNWYXJpYW50VHlwZSA9IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kO1xuXG50eXBlIENvbmNyZXRlRWxlbWVudFRTTW9ycGhUeXBlID0gQ2xhc3NEZWNsYXJhdGlvbiB8IEludGVyZmFjZURlY2xhcmF0aW9uIHwgRnVuY3Rpb25EZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uO1xuXG5leHBvcnQgY2xhc3MgRW50aXR5RGljdGlvbmFyeSB7XG4gICAgXG4gICAgcHVibGljIGZhbWl4UmVwID0gbmV3IEZhbWl4UmVwb3NpdG9yeSgpO1xuICAgIHByaXZhdGUgZm14QWxpYXNNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguQWxpYXM+KCk7IC8vIE1hcHMgdGhlIGFsaWFzIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXG4gICAgcHJpdmF0ZSBmbXhDbGFzc01hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5DbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNDbGFzcz4oKTsgLy8gTWFwcyB0aGUgZnVsbHkgcXVhbGlmaWVkIGNsYXNzIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXG4gICAgcHJpdmF0ZSBmbXhJbnRlcmZhY2VNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguSW50ZXJmYWNlIHwgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZT4oKTsgLy8gTWFwcyB0aGUgaW50ZXJmYWNlIG5hbWVzIHRvIHRoZWlyIEZhbWl4IG1vZGVsXG4gICAgcHJpdmF0ZSBmbXhNb2R1bGVNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguTW9kdWxlPigpOyAvLyBNYXBzIHRoZSBuYW1lc3BhY2UgbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcbiAgICBwcml2YXRlIGZteEZpbGVNYXAgPSBuZXcgTWFwPHN0cmluZywgRmFtaXguU2NyaXB0RW50aXR5IHwgRmFtaXguTW9kdWxlPigpOyAvLyBNYXBzIHRoZSBzb3VyY2UgZmlsZSBuYW1lcyB0byB0aGVpciBGYW1peCBtb2RlbFxuICAgIHByaXZhdGUgZm14VHlwZU1hcCA9IG5ldyBNYXA8c3RyaW5nLCBGYW1peC5UeXBlIHwgRmFtaXguUHJpbWl0aXZlVHlwZSB8IEZhbWl4LlBhcmFtZXRlclR5cGU+KCk7IC8vIE1hcHMgdGhlIHR5cGUgbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcbiAgICBwcml2YXRlIGZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwID0gbmV3IE1hcDxzdHJpbmcsIEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uIHwgRmFtaXguTWV0aG9kIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZD4gLy8gTWFwcyB0aGUgZnVuY3Rpb24gbmFtZXMgdG8gdGhlaXIgRmFtaXggbW9kZWxcbiAgICBwcml2YXRlIFVOS05PV05fVkFMVUUgPSAnKHVua25vd24gZHVlIHRvIHBhcnNpbmcgZXJyb3IpJzsgLy8gVGhlIHZhbHVlIHRvIHVzZSB3aGVuIGEgbmFtZSBpcyBub3QgdXNhYmxlXG4gICAgcHVibGljIGZteEVsZW1lbnRPYmplY3RNYXAgPSBuZXcgTWFwPEZhbWl4LkVudGl0eSxUU01vcnBoT2JqZWN0VHlwZT4oKTtcbiAgICBwdWJsaWMgdHNNb3JwaEVsZW1lbnRPYmplY3RNYXAgPSBuZXcgTWFwPFRTTW9ycGhPYmplY3RUeXBlLEZhbWl4LkVudGl0eT4oKTtcbiAgICAgICAgICAgIFxuICAgIGNvbnN0cnVjdG9yKCkge1xuICAgICAgICB0aGlzLmZhbWl4UmVwLnNldEZteEVsZW1lbnRPYmplY3RNYXAodGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwKTsgICAgICBcbiAgICB9XG5cbiAgICBwdWJsaWMgYWRkU291cmNlQW5jaG9yKGZteDogRmFtaXguU291cmNlZEVudGl0eSwgbm9kZTogVFNNb3JwaE9iamVjdFR5cGUpOiBGYW1peC5JbmRleGVkRmlsZUFuY2hvciB7XG4gICAgICAgIGNvbnN0IHNvdXJjZUFuY2hvcjogRmFtaXguSW5kZXhlZEZpbGVBbmNob3IgPSBuZXcgRmFtaXguSW5kZXhlZEZpbGVBbmNob3IoKTtcbiAgICAgICAgbGV0IHNvdXJjZVN0YXJ0LCBzb3VyY2VFbmQ6IG51bWJlcjtcbiAgICAgICAgaWYgKGZteCAmJiBub2RlKSB7XG4gICAgICAgICAgICAvLyBmaW5kIHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGUgc291cmNlIGVsZW1lbnRcbiAgICAgICAgICAgIGlmICghKG5vZGUgaW5zdGFuY2VvZiBDb21tZW50UmFuZ2UpKSB7XG4gICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBub2RlLmdldFN0YXJ0KCk7XG4gICAgICAgICAgICAgICAgc291cmNlRW5kID0gbm9kZS5nZXRFbmQoKTtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgc291cmNlU3RhcnQgPSBub2RlLmdldFBvcygpO1xuICAgICAgICAgICAgICAgIHNvdXJjZUVuZCA9IG5vZGUuZ2V0RW5kKCk7XG4gICAgICAgICAgICB9XG4gICAgXG4gICAgICAgICAgICBpZiAoY29uZmlnLmV4cGVjdEdyYXBoZW1lcykge1xuICAgICAgICAgICAgICAgIC8qKlxuICAgICAgICAgICAgICAgICAqIFRoZSBmb2xsb3dpbmcgbG9naWMgaGFuZGxlcyB0aGUgY2FzZSBvZiBtdWx0aS1jb2RlIHBvaW50IGNoYXJhY3RlcnMgKGUuZy4gZW1vamkpIGluIHRoZSBzb3VyY2UgdGV4dC5cbiAgICAgICAgICAgICAgICAgKiBUaGlzIGlzIG5lZWRlZCBiZWNhdXNlIFBoYXJvL1NtYWxsdGFsayB0cmVhdHMgbXVsdGktY29kZSBwb2ludCBjaGFyYWN0ZXJzIGFzIGEgc2luZ2xlIGNoYXJhY3RlciwgXG4gICAgICAgICAgICAgICAgICogYnV0IEphdmFTY3JpcHQgdHJlYXRzIHRoZW0gYXMgbXVsdGlwbGUgY2hhcmFjdGVycy4gVGhpcyBtZWFucyB0aGF0IHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9uc1xuICAgICAgICAgICAgICAgICAqIG9mIGEgc291cmNlIGVsZW1lbnQgaW4gUGhhcm8vU21hbGx0YWxrIHdpbGwgYmUgZGlmZmVyZW50IHRoYW4gdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZVxuICAgICAgICAgICAgICAgICAqIHNhbWUgc291cmNlIGVsZW1lbnQgaW4gSmF2YVNjcmlwdC4gVGhpcyBsb2dpYyBmaW5kcyB0aGUgc3RhcnQgYW5kIGVuZCBwb3NpdGlvbnMgb2YgdGhlIHNvdXJjZVxuICAgICAgICAgICAgICAgICAqIGVsZW1lbnQgaW4gSmF2YVNjcmlwdCBhbmQgdGhlbiB1c2VzIHRob3NlIHBvc2l0aW9ucyB0byBzZXQgdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZVxuICAgICAgICAgICAgICAgICAqIEZhbWl4IGluZGV4IGZpbGUgYW5jaG9yLlxuICAgICAgICAgICAgICAgICAqIEl0IGRlcGVuZHMgb24gY29kZSBpbiB0aGUgJ2dyYXBoZW1lLXNwbGl0dGVyJyBwYWNrYWdlIGluIG5wbS5cbiAgICAgICAgICAgICAgICAgKi9cbiAgICAgICAgICAgICAgICBjb25zdCBzcGxpdHRlciA9IG5ldyBHcmFwaGVtZVNwbGl0dGVyKCk7XG4gICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRmlsZVRleHQgPSBub2RlLmdldFNvdXJjZUZpbGUoKS5nZXRGdWxsVGV4dCgpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGhhc0dyYXBoZW1lQ2x1c3RlcnMgPSBzcGxpdHRlci5jb3VudEdyYXBoZW1lcyhzb3VyY2VGaWxlVGV4dCkgPiAxO1xuICAgICAgICAgICAgICAgIGlmIChoYXNHcmFwaGVtZUNsdXN0ZXJzKSB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUVsZW1lbnRUZXh0ID0gc291cmNlRmlsZVRleHQuc3Vic3RyaW5nKHNvdXJjZVN0YXJ0LCBzb3VyY2VFbmQpO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcyA9IHNwbGl0dGVyLnNwbGl0R3JhcGhlbWVzKHNvdXJjZUVsZW1lbnRUZXh0KTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRmlsZVRleHRHcmFwaGVtZXMgPSBzcGxpdHRlci5zcGxpdEdyYXBoZW1lcyhzb3VyY2VGaWxlVGV4dCk7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IG51bWJlck9mR3JhcGhlbWVDbHVzdGVyc0JlZm9yZVN0YXJ0ID0gc3BsaXR0ZXIuY291bnRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQuc3Vic3RyaW5nKDAsIHNvdXJjZVN0YXJ0KSk7XG4gICAgICAgIFxuICAgICAgICAgICAgICAgICAgICAvLyBmaW5kIHRoZSBzdGFydCBvZiB0aGUgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMgYXJyYXkgaW4gdGhlIHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzIGFycmF5XG4gICAgICAgICAgICAgICAgICAgIHNvdXJjZVN0YXJ0ID0gSGVscGVycy5pbmRleE9mU3BsaXRBcnJheSh7c2VhcmNoQXJyYXk6IHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGFyZ2V0QXJyYXk6IHNvdXJjZUVsZW1lbnRUZXh0R3JhcGhlbWVzLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3RhcnQ6IHNvdXJjZVN0YXJ0IC0gbnVtYmVyT2ZHcmFwaGVtZUNsdXN0ZXJzQmVmb3JlU3RhcnR9KTtcbiAgICAgICAgICAgICAgICAgICAgc291cmNlRW5kID0gc291cmNlU3RhcnQgKyBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcy5sZW5ndGg7XG4gICAgICAgICAgICAgICAgfSBcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gVGhlICsxIGlzIGJlY2F1c2UgdGhlIHNvdXJjZSBhbmNob3IgKFBoYXJvKSBpcyAxLWJhc2VkLCBidXQgdHMtbW9ycGggaXMgMC1iYXNlZFxuICAgICAgICAgICAgc291cmNlQW5jaG9yLnN0YXJ0UG9zID0gc291cmNlU3RhcnQgKyAxO1xuICAgICAgICAgICAgc291cmNlQW5jaG9yLmVuZFBvcyA9IHNvdXJjZUVuZCArIDE7XG5cbiAgICAgICAgICAgIGNvbnN0IGZpbGVOYW1lID0gbm9kZS5nZXRTb3VyY2VGaWxlKCkuZ2V0RmlsZVBhdGgoKTtcblxuICAgICAgICAgICAgc291cmNlQW5jaG9yLmVsZW1lbnQgPSBmbXg7XG4gICAgICAgICAgICBzb3VyY2VBbmNob3IuZmlsZU5hbWUgPSBmaWxlTmFtZTtcbiAgICAgICAgICAgIGZteC5zb3VyY2VBbmNob3IgPSBzb3VyY2VBbmNob3I7XG4gICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoc291cmNlQW5jaG9yKTtcblxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBzb3VyY2VBbmNob3I7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogTWFrZXMgYSBGYW1peCBpbmRleCBmaWxlIGFuY2hvclxuICAgICAqIEBwYXJhbSBzb3VyY2VFbGVtZW50IEEgc291cmNlIGVsZW1lbnRcbiAgICAgKiBAcGFyYW0gZmFtaXhFbGVtZW50IFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgc291cmNlIGVsZW1lbnRcbiAgICAgKi9cbiAgICBwdWJsaWMgbWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHNvdXJjZUVsZW1lbnQ6IFRTTW9ycGhPYmplY3RUeXBlLCBmYW1peEVsZW1lbnQ6IEZhbWl4LlNvdXJjZWRFbnRpdHkpOiB2b2lkIHtcbiAgICAgICAgLy8gY2hlY2sgaWYgZmFtaXhFbGVtZW50IGRvZXNuJ3QgaGF2ZSBhIHZhbGlkIGZ1bGx5UXVhbGlmaWVkTmFtZVxuICAgICAgICBpZiAodHlwZW9mIChmYW1peEVsZW1lbnQgYXMgYW55KS5nZXRGdWxseVF1YWxpZmllZE5hbWUgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgICAgIC8vIFRoZSBtZXRob2QgZXhpc3RzXG4gICAgICAgICAgICBjb25zdCBmdWxseVF1YWxpZmllZE5hbWUgPSAoZmFtaXhFbGVtZW50IGFzIGFueSkuZnVsbHlRdWFsaWZpZWROYW1lO1xuICAgICAgICAgICAgaWYgKCFmdWxseVF1YWxpZmllZE5hbWUgfHwgZnVsbHlRdWFsaWZpZWROYW1lID09PSB0aGlzLlVOS05PV05fVkFMVUUpIHtcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhbWl4IGVsZW1lbnQgJHtmYW1peEVsZW1lbnQuY29uc3RydWN0b3IubmFtZX0gaGFzIG5vIHZhbGlkIGZ1bGx5UXVhbGlmaWVkTmFtZS5gKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhcIm1ha2luZyBpbmRleCBmaWxlIGFuY2hvciBmb3IgJ1wiICsgc291cmNlRWxlbWVudD8uZ2V0VGV4dCgpICsgXCInIHdpdGggZmFtaXhFbGVtZW50IFwiICsgZmFtaXhFbGVtZW50LmdldEpTT04oKSk7XG4gICAgICAgIGNvbnN0IGZteEluZGV4RmlsZUFuY2hvciA9IG5ldyBGYW1peC5JbmRleGVkRmlsZUFuY2hvcigpO1xuICAgICAgICBmbXhJbmRleEZpbGVBbmNob3IuZWxlbWVudCA9IGZhbWl4RWxlbWVudDtcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmYW1peEVsZW1lbnQsIHNvdXJjZUVsZW1lbnQpO1xuXG4gICAgICAgIGlmIChzb3VyY2VFbGVtZW50ICE9PSBudWxsKSB7XG4gICAgICAgICAgICBjb25zdCBhYnNvbHV0ZVBhdGhQcm9qZWN0ID0gdGhpcy5mYW1peFJlcC5nZXRBYnNvbHV0ZVBhdGgoKTtcbiAgICAgICAgXG4gICAgICAgICAgICBjb25zdCBhYnNvbHV0ZVBhdGggPSBwYXRoLm5vcm1hbGl6ZShzb3VyY2VFbGVtZW50LmdldFNvdXJjZUZpbGUoKS5nZXRGaWxlUGF0aCgpKTtcblxuICAgICAgICAgICAgY29uc3QgcG9zaXRpb25Ob2RlTW9kdWxlcyA9IGFic29sdXRlUGF0aC5pbmRleE9mKCdub2RlX21vZHVsZXMnKTtcblxuICAgICAgICAgICAgbGV0IHBhdGhJblByb2plY3Q6IHN0cmluZyA9IFwiXCI7XG5cbiAgICAgICAgICAgIGlmIChwb3NpdGlvbk5vZGVNb2R1bGVzICE9PSAtMSkge1xuICAgICAgICAgICAgICAgIGNvbnN0IHBhdGhGcm9tTm9kZU1vZHVsZXMgPSBhYnNvbHV0ZVBhdGguc3Vic3RyaW5nKHBvc2l0aW9uTm9kZU1vZHVsZXMpO1xuICAgICAgICAgICAgICAgIHBhdGhJblByb2plY3QgPSBwYXRoRnJvbU5vZGVNb2R1bGVzO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICBwYXRoSW5Qcm9qZWN0ID0gdGhpcy5jb252ZXJ0VG9SZWxhdGl2ZVBhdGgoYWJzb2x1dGVQYXRoLCBhYnNvbHV0ZVBhdGhQcm9qZWN0KTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgLy8gcmV2ZXJ0IGFueSBiYWNrc2xhc2hlcyB0byBmb3J3YXJkIHNsYXNoZXMgKHBhdGgubm9ybWFsaXplIG9uIHdpbmRvd3MgaW50cm9kdWNlcyB0aGVtKVxuICAgICAgICAgICAgcGF0aEluUHJvamVjdCA9IHBhdGhJblByb2plY3QucmVwbGFjZSgvXFxcXC9nLCBcIi9cIik7XG5cbiAgICAgICAgICAgIGZteEluZGV4RmlsZUFuY2hvci5maWxlTmFtZSA9IHBhdGhJblByb2plY3Q7XG4gICAgICAgICAgICBsZXQgc291cmNlU3RhcnQsIHNvdXJjZUVuZCwgc291cmNlTGluZVN0YXJ0LCBzb3VyY2VMaW5lRW5kOiBudW1iZXI7XG4gICAgICAgICAgICBpZiAoIShzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgQ29tbWVudFJhbmdlKSkge1xuICAgICAgICAgICAgICAgIHNvdXJjZVN0YXJ0ID0gc291cmNlRWxlbWVudC5nZXRTdGFydCgpO1xuICAgICAgICAgICAgICAgIHNvdXJjZUVuZCA9IHNvdXJjZUVsZW1lbnQuZ2V0RW5kKCk7XG4gICAgICAgICAgICAgICAgc291cmNlTGluZVN0YXJ0ID0gc291cmNlRWxlbWVudC5nZXRTdGFydExpbmVOdW1iZXIoKTtcbiAgICAgICAgICAgICAgICBzb3VyY2VMaW5lRW5kID0gc291cmNlRWxlbWVudC5nZXRFbmRMaW5lTnVtYmVyKCk7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIHNvdXJjZVN0YXJ0ID0gc291cmNlRWxlbWVudC5nZXRQb3MoKTtcbiAgICAgICAgICAgICAgICBzb3VyY2VFbmQgPSBzb3VyY2VFbGVtZW50LmdldEVuZCgpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaWYgKGNvbmZpZy5leHBlY3RHcmFwaGVtZXMpIHtcbiAgICAgICAgICAgICAgICAvKipcbiAgICAgICAgICAgICAgICAgKiBUaGUgZm9sbG93aW5nIGxvZ2ljIGhhbmRsZXMgdGhlIGNhc2Ugb2YgbXVsdGktY29kZSBwb2ludCBjaGFyYWN0ZXJzIChlLmcuIGVtb2ppKSBpbiB0aGUgc291cmNlIHRleHQuXG4gICAgICAgICAgICAgICAgICogVGhpcyBpcyBuZWVkZWQgYmVjYXVzZSBQaGFyby9TbWFsbHRhbGsgdHJlYXRzIG11bHRpLWNvZGUgcG9pbnQgY2hhcmFjdGVycyBhcyBhIHNpbmdsZSBjaGFyYWN0ZXIsIFxuICAgICAgICAgICAgICAgICAqIGJ1dCBKYXZhU2NyaXB0IHRyZWF0cyB0aGVtIGFzIG11bHRpcGxlIGNoYXJhY3RlcnMuIFRoaXMgbWVhbnMgdGhhdCB0aGUgc3RhcnQgYW5kIGVuZCBwb3NpdGlvbnNcbiAgICAgICAgICAgICAgICAgKiBvZiBhIHNvdXJjZSBlbGVtZW50IGluIFBoYXJvL1NtYWxsdGFsayB3aWxsIGJlIGRpZmZlcmVudCB0aGFuIHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGVcbiAgICAgICAgICAgICAgICAgKiBzYW1lIHNvdXJjZSBlbGVtZW50IGluIEphdmFTY3JpcHQuIFRoaXMgbG9naWMgZmluZHMgdGhlIHN0YXJ0IGFuZCBlbmQgcG9zaXRpb25zIG9mIHRoZSBzb3VyY2VcbiAgICAgICAgICAgICAgICAgKiBlbGVtZW50IGluIEphdmFTY3JpcHQgYW5kIHRoZW4gdXNlcyB0aG9zZSBwb3NpdGlvbnMgdG8gc2V0IHRoZSBzdGFydCBhbmQgZW5kIHBvc2l0aW9ucyBvZiB0aGVcbiAgICAgICAgICAgICAgICAgKiBGYW1peCBpbmRleCBmaWxlIGFuY2hvci5cbiAgICAgICAgICAgICAgICAgKiBJdCBkZXBlbmRzIG9uIGNvZGUgaW4gdGhlICdncmFwaGVtZS1zcGxpdHRlcicgcGFja2FnZSBpbiBucG0uXG4gICAgICAgICAgICAgICAgICovXG4gICAgICAgICAgICAgICAgY29uc3Qgc3BsaXR0ZXIgPSBuZXcgR3JhcGhlbWVTcGxpdHRlcigpO1xuICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUZpbGVUZXh0ID0gc291cmNlRWxlbWVudC5nZXRTb3VyY2VGaWxlKCkuZ2V0RnVsbFRleHQoKTtcbiAgICAgICAgICAgICAgICBjb25zdCBoYXNHcmFwaGVtZUNsdXN0ZXJzID0gc3BsaXR0ZXIuY291bnRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQpID4gMTtcbiAgICAgICAgICAgICAgICBpZiAoaGFzR3JhcGhlbWVDbHVzdGVycykge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBzb3VyY2VFbGVtZW50VGV4dCA9IHNvdXJjZUZpbGVUZXh0LnN1YnN0cmluZyhzb3VyY2VTdGFydCwgc291cmNlRW5kKTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3Qgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMgPSBzcGxpdHRlci5zcGxpdEdyYXBoZW1lcyhzb3VyY2VFbGVtZW50VGV4dCk7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IHNvdXJjZUZpbGVUZXh0R3JhcGhlbWVzID0gc3BsaXR0ZXIuc3BsaXRHcmFwaGVtZXMoc291cmNlRmlsZVRleHQpO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBudW1iZXJPZkdyYXBoZW1lQ2x1c3RlcnNCZWZvcmVTdGFydCA9IHNwbGl0dGVyLmNvdW50R3JhcGhlbWVzKHNvdXJjZUZpbGVUZXh0LnN1YnN0cmluZygwLCBzb3VyY2VTdGFydCkpO1xuICAgICAgICBcbiAgICAgICAgICAgICAgICAgICAgLy8gZmluZCB0aGUgc3RhcnQgb2YgdGhlIHNvdXJjZUVsZW1lbnRUZXh0R3JhcGhlbWVzIGFycmF5IGluIHRoZSBzb3VyY2VGaWxlVGV4dEdyYXBoZW1lcyBhcnJheVxuICAgICAgICAgICAgICAgICAgICBzb3VyY2VTdGFydCA9IEhlbHBlcnMuaW5kZXhPZlNwbGl0QXJyYXkoe3NlYXJjaEFycmF5OiBzb3VyY2VGaWxlVGV4dEdyYXBoZW1lcywgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRhcmdldEFycmF5OiBzb3VyY2VFbGVtZW50VGV4dEdyYXBoZW1lcywgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXJ0OiBzb3VyY2VTdGFydCAtIG51bWJlck9mR3JhcGhlbWVDbHVzdGVyc0JlZm9yZVN0YXJ0fSk7XG4gICAgICAgICAgICAgICAgICAgIHNvdXJjZUVuZCA9IHNvdXJjZVN0YXJ0ICsgc291cmNlRWxlbWVudFRleHRHcmFwaGVtZXMubGVuZ3RoO1xuICAgICAgICAgICAgICAgIH0gXG4gICAgICAgICAgICB9XG4gICAgICAgICAgICAvLyBub3RlOiB0aGUgKzEgaXMgYmVjYXVzZSB0aGUgc291cmNlIGFuY2hvciBpcyAxLWJhc2VkLCBidXQgdHMtbW9ycGggaXMgMC1iYXNlZFxuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLnN0YXJ0UG9zID0gc291cmNlU3RhcnQgKyAxO1xuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmVuZFBvcyA9IHNvdXJjZUVuZCArIDE7XG5cbiAgICAgICAgICAgIC8vIGlmICghKGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LkltcG9ydENsYXVzZSB8fCBmYW1peEVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5BY2Nlc3MgfHwgZmFtaXhFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguUmVmZXJlbmNlIHx8IGZhbWl4RWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4Lkludm9jYXRpb24gfHwgZmFtaXhFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguSW5oZXJpdGFuY2UpICYmICEoZmFtaXhFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguQ29tbWVudCkgJiYgIShzb3VyY2VFbGVtZW50IGluc3RhbmNlb2YgQ29tbWVudFJhbmdlKSAmJiAhKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBJZGVudGlmaWVyKSAmJiAhKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBJbXBvcnRTcGVjaWZpZXIpICYmICEoc291cmNlRWxlbWVudCBpbnN0YW5jZW9mIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cykpIHtcbiAgICAgICAgICAgIC8vICAgIGluaXRGUU4oc291cmNlRWxlbWVudCwgZmFtaXhFbGVtZW50KTtcbiAgICAgICAgICAgIC8vIH1cbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIC8vIHNvdXJjZUVsZW1lbnQgaXMgbnVsbFxuICAgICAgICAgICAgbG9nZ2VyLndhcm4oXCJzb3VyY2VFbGVtZW50IGlzIG51bGwgZm9yIGZhbWl4RWxlbWVudCBcIiArIGZhbWl4RWxlbWVudC5nZXRKU09OKCkpO1xuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmZpbGVOYW1lID0gXCJ1bmtub3duXCI7XG4gICAgICAgICAgICBmbXhJbmRleEZpbGVBbmNob3Iuc3RhcnRQb3MgPSAwO1xuICAgICAgICAgICAgZm14SW5kZXhGaWxlQW5jaG9yLmVuZFBvcyA9IDA7XG4gICAgICAgIH1cblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW5kZXhGaWxlQW5jaG9yKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBzY3JpcHQgZW50aXR5IG9yIG1vZHVsZVxuICAgICAqIEBwYXJhbSBmIEEgc291cmNlIGZpbGVcbiAgICAgKiBAcGFyYW0gaXNNb2R1bGUgQSBib29sZWFuIGluZGljYXRpbmcgaWYgdGhlIHNvdXJjZSBmaWxlIGlzIGEgbW9kdWxlXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBzb3VyY2UgZmlsZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4RmlsZShmOiBTb3VyY2VGaWxlLCBpc01vZHVsZTogYm9vbGVhbik6IEZhbWl4LlNjcmlwdEVudGl0eSB8IEZhbWl4Lk1vZHVsZSB7XG4gICAgICAgIGxldCBmbXhGaWxlOiBGYW1peC5TY3JpcHRFbnRpdHk7IC8vIHwgRmFtaXguTW9kdWxlO1xuXG4gICAgICAgIGNvbnN0IGZpbGVOYW1lID0gZi5nZXRCYXNlTmFtZSgpO1xuICAgICAgICBjb25zdCBmdWxseVF1YWxpZmllZEZpbGVuYW1lID0gZi5nZXRGaWxlUGF0aCgpO1xuICAgICAgICBjb25zdCBmb3VuZEZpbGVOYW1lID0gdGhpcy5mbXhGaWxlTWFwLmdldChmdWxseVF1YWxpZmllZEZpbGVuYW1lKTtcbiAgICAgICAgaWYgKCFmb3VuZEZpbGVOYW1lKSB7XG4gICAgICAgICAgICBpZiAoaXNNb2R1bGUpIHtcbiAgICAgICAgICAgICAgICBmbXhGaWxlID0gbmV3IEZhbWl4Lk1vZHVsZSgpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14RmlsZSA9IG5ldyBGYW1peC5TY3JpcHRFbnRpdHkoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGZteEZpbGUubmFtZSA9IGZpbGVOYW1lO1xuICAgICAgICAgICAgZm14RmlsZS5udW1iZXJPZkxpbmVzT2ZUZXh0ID0gZi5nZXRFbmRMaW5lTnVtYmVyKCkgLSBmLmdldFN0YXJ0TGluZU51bWJlcigpO1xuICAgICAgICAgICAgZm14RmlsZS5udW1iZXJPZkNoYXJhY3RlcnMgPSBmLmdldEZ1bGxUZXh0KCkubGVuZ3RoO1xuXG4gICAgICAgICAgICBpbml0RlFOKGYsIGZteEZpbGUpO1xuXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihmLCBmbXhGaWxlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhGaWxlTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBmbXhGaWxlKTtcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhGaWxlKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteEZpbGUgPSBmb3VuZEZpbGVOYW1lO1xuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhGaWxlLGYpO1xuICAgICAgICByZXR1cm4gZm14RmlsZTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBNb2R1bGVcbiAgICAgKiBAcGFyYW0gbSBBIG1vZHVsZVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgbW9kdWxlXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhNb2R1bGUobTogTW9kdWxlRGVjbGFyYXRpb24pOiBGYW1peC5Nb2R1bGUge1xuICAgICAgICBsZXQgZm14TW9kdWxlOiBGYW1peC5Nb2R1bGU7XG4gICAgICAgIGNvbnN0IG1vZHVsZU5hbWUgPSBtLmdldE5hbWUoKTtcbiAgICAgICAgY29uc3QgZm91bmRNb2R1bGVOYW1lID0gdGhpcy5mbXhNb2R1bGVNYXAuZ2V0KG1vZHVsZU5hbWUpO1xuICAgICAgICBpZiAoIWZvdW5kTW9kdWxlTmFtZSkge1xuICAgICAgICAgICAgZm14TW9kdWxlID0gbmV3IEZhbWl4Lk1vZHVsZSgpO1xuICAgICAgICAgICAgZm14TW9kdWxlLm5hbWUgPSBtb2R1bGVOYW1lO1xuICAgICAgICAgICAgZm14TW9kdWxlLmlzQW1iaWVudCA9IGlzQW1iaWVudChtKTtcbiAgICAgICAgICAgIGZteE1vZHVsZS5pc05hbWVzcGFjZSA9IGlzTmFtZXNwYWNlKG0pO1xuICAgICAgICAgICAgZm14TW9kdWxlLmlzTW9kdWxlID0gIWZteE1vZHVsZS5pc05hbWVzcGFjZSAmJiAhZm14TW9kdWxlLmlzQW1iaWVudDtcblxuICAgICAgICAgICAgaW5pdEZRTihtLCBmbXhNb2R1bGUpO1xuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IobSwgZm14TW9kdWxlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhNb2R1bGVNYXAuc2V0KG1vZHVsZU5hbWUsIGZteE1vZHVsZSk7XG5cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhNb2R1bGUpO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14TW9kdWxlID0gZm91bmRNb2R1bGVOYW1lO1xuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhNb2R1bGUsbSk7XG4gICAgICAgIHJldHVybiBmbXhNb2R1bGU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGFsaWFzXG4gICAgICogQHBhcmFtIGEgQW4gYWxpYXNcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGFsaWFzXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4QWxpYXMoYTogVHlwZUFsaWFzRGVjbGFyYXRpb24pOiBGYW1peC5BbGlhcyB7XG4gICAgICAgIGxldCBmbXhBbGlhczogRmFtaXguQWxpYXM7XG4gICAgICAgIGNvbnN0IGFsaWFzTmFtZSA9IGEuZ2V0TmFtZSgpO1xuICAgICAgICBjb25zdCBhbGlhc0Z1bGx5UXVhbGlmaWVkTmFtZSA9IGEuZ2V0VHlwZSgpLmdldFRleHQoKTsgLy8gRlFORnVuY3Rpb25zLmdldEZRTihhKTtcbiAgICAgICAgY29uc3QgZm91bmRBbGlhcyA9IHRoaXMuZm14QWxpYXNNYXAuZ2V0KGFsaWFzRnVsbHlRdWFsaWZpZWROYW1lKTtcbiAgICAgICAgaWYgKCFmb3VuZEFsaWFzKSB7XG4gICAgICAgICAgICBmbXhBbGlhcyA9IG5ldyBGYW1peC5BbGlhcygpO1xuICAgICAgICAgICAgZm14QWxpYXMubmFtZSA9IGEuZ2V0TmFtZSgpO1xuICAgICAgICAgICAgY29uc3QgYWxpYXNOYW1lV2l0aEdlbmVyaWNzID0gYWxpYXNOYW1lICsgKGEuZ2V0VHlwZVBhcmFtZXRlcnMoKS5sZW5ndGggPyAoXCI8XCIgKyBhLmdldFR5cGVQYXJhbWV0ZXJzKCkubWFwKHRwID0+IHRwLmdldE5hbWUoKSkuam9pbihcIiwgXCIpICsgXCI+XCIpIDogXCJcIik7XG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoYD4gTk9URTogYWxpYXMgJHthbGlhc05hbWV9IGhhcyBmdWxseSBxdWFsaWZpZWQgbmFtZSAke2FsaWFzRnVsbHlRdWFsaWZpZWROYW1lfSBhbmQgbmFtZSB3aXRoIGdlbmVyaWNzICR7YWxpYXNOYW1lV2l0aEdlbmVyaWNzfS5gKTtcblxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoYWxpYXNOYW1lV2l0aEdlbmVyaWNzLCBhKTtcbiAgICAgICAgICAgIGZteEFsaWFzLmFsaWFzZWRFbnRpdHkgPSBmbXhUeXBlO1xuICAgICAgICAgICAgaW5pdEZRTihhLCBmbXhBbGlhcyk7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihhLCBmbXhBbGlhcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZm14QWxpYXNNYXAuc2V0KGFsaWFzRnVsbHlRdWFsaWZpZWROYW1lLCBmbXhBbGlhcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhBbGlhcyk7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICBmbXhBbGlhcyA9IGZvdW5kQWxpYXM7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhBbGlhcyxhKTtcblxuICAgICAgICByZXR1cm4gZm14QWxpYXM7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBvciBnZXRzIGEgRmFtaXggY2xhc3Mgb3IgcGFyYW1ldGVyaXphYmxlIGNsYXNzXG4gICAgICogQHBhcmFtIGNscyBBIGNsYXNzXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjbGFzc1xuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4Q2xhc3MoY2xzOiBDbGFzc0RlY2xhcmF0aW9uKTogRmFtaXguQ2xhc3MgfCBGYW1peC5QYXJhbWV0cmljQ2xhc3Mge1xuICAgICAgICBsZXQgZm14Q2xhc3M6IEZhbWl4LkNsYXNzIHwgRmFtaXguUGFyYW1ldHJpY0NsYXNzO1xuICAgICAgICBjb25zdCBpc0Fic3RyYWN0ID0gY2xzLmlzQWJzdHJhY3QoKTtcbiAgICAgICAgY29uc3QgY2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGNscyk7XG4gICAgICAgIGNvbnN0IGNsc05hbWUgPSBjbHMuZ2V0TmFtZSgpIHx8IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gY2xzLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoO1xuICAgICAgICBjb25zdCBmb3VuZENsYXNzID0gdGhpcy5mbXhDbGFzc01hcC5nZXQoY2xhc3NGdWxseVF1YWxpZmllZE5hbWUpO1xuICAgICAgICBpZiAoIWZvdW5kQ2xhc3MpIHtcbiAgICAgICAgICAgIGlmIChpc0dlbmVyaWMpIHtcbiAgICAgICAgICAgICAgICBmbXhDbGFzcyA9IG5ldyBGYW1peC5QYXJhbWV0cmljQ2xhc3MoKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgIGZteENsYXNzID0gbmV3IEZhbWl4LkNsYXNzKCk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGZteENsYXNzLm5hbWUgPSBjbHNOYW1lO1xuICAgICAgICAgICAgZm14Q2xhc3MuZnVsbHlRdWFsaWZpZWROYW1lID0gY2xhc3NGdWxseVF1YWxpZmllZE5hbWU7XG4gICAgICAgICAgICBmbXhDbGFzcy5pc0Fic3RyYWN0ID0gaXNBYnN0cmFjdDtcblxuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IoY2xzLCBmbXhDbGFzcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZm14Q2xhc3NNYXAuc2V0KGNsYXNzRnVsbHlRdWFsaWZpZWROYW1lLCBmbXhDbGFzcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhDbGFzcyk7XG5cbiAgICAgICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14Q2xhc3MsY2xzKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteENsYXNzID0gZm91bmRDbGFzcztcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBmbXhDbGFzcztcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBpbnRlcmZhY2Ugb3IgcGFyYW1ldGVyaXphYmxlIGludGVyZmFjZVxuICAgICAqIEBwYXJhbSBpbnRlciBBbiBpbnRlcmZhY2VcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGludGVyZmFjZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4SW50ZXJmYWNlKGludGVyOiBJbnRlcmZhY2VEZWNsYXJhdGlvbik6IEZhbWl4LkludGVyZmFjZSB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2Uge1xuXG4gICAgICAgIGxldCBmbXhJbnRlcmZhY2U6IEZhbWl4LkludGVyZmFjZSB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgIGNvbnN0IGludGVyTmFtZSA9IGludGVyLmdldE5hbWUoKTtcbiAgICAgICAgY29uc3QgaW50ZXJGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGludGVyKTtcbiAgICAgICAgY29uc3QgZm91bmRJbnRlcmZhY2UgPSB0aGlzLmZteEludGVyZmFjZU1hcC5nZXQoaW50ZXJGdWxseVF1YWxpZmllZE5hbWUpO1xuICAgICAgICBpZiAoIWZvdW5kSW50ZXJmYWNlKSB7XG4gICAgICAgICAgICBjb25zdCBpc0dlbmVyaWMgPSBpbnRlci5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aDtcbiAgICAgICAgICAgIGlmIChpc0dlbmVyaWMpIHtcbiAgICAgICAgICAgICAgICBmbXhJbnRlcmZhY2UgPSBuZXcgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSgpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14SW50ZXJmYWNlID0gbmV3IEZhbWl4LkludGVyZmFjZSgpO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBmbXhJbnRlcmZhY2UubmFtZSA9IGludGVyTmFtZTtcbiAgICAgICAgICAgIGluaXRGUU4oaW50ZXIsIGZteEludGVyZmFjZSk7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbnRlciwgZm14SW50ZXJmYWNlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhJbnRlcmZhY2VNYXAuc2V0KGludGVyRnVsbHlRdWFsaWZpZWROYW1lLCBmbXhJbnRlcmZhY2UpO1xuXG4gICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW50ZXJmYWNlKTtcblxuICAgICAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhJbnRlcmZhY2UsaW50ZXIpO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14SW50ZXJmYWNlID0gZm91bmRJbnRlcmZhY2U7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIGZteEludGVyZmFjZTtcbiAgICB9XG5cbiAgICBcbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBjb25jcmV0ZSBlbGVtZW50XG4gICAgICogQHBhcmFtIGNvbmNyZXRlRWxlbWVudCBBIHBhcmFtZXRyaWMgRWxlbWVudCAgIFxuICAgICAqIEBwYXJhbSBjb25jcmV0ZUVsZW1lbnREZWNsYXJhdGlvbiB0aGUgZWxlbWVudCBkZWNsYXJhdGlvblxuICAgICAqIEBwYXJhbSBjb25jcmV0ZUFyZ3VtZW50cyBjb25jcmV0ZSBhcmd1bWVudHNcbiAgICAgKiBAcmV0dXJucyBBIHBhcmFtZXRyaWMgRWxlbWVudCAgXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZUVsZW1lbnQoY29uY3JldGVFbGVtZW50IDogUGFyYW1ldHJpY1ZhcmlhbnRUeXBlLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0ZUVsZW1lbnREZWNsYXJhdGlvbiA6IENvbmNyZXRlRWxlbWVudFRTTW9ycGhUeXBlLCBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0ZUFyZ3VtZW50czogVHlwZU5vZGVbXSk6IFBhcmFtZXRyaWNWYXJpYW50VHlwZSB7XG4gICAgICAgIFxuICAgICAgICBsZXQgZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSA9IGNvbmNyZXRlRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWU7XG4gICAgICAgIGxldCBwYXJhbXMgPSBcIlwiO1xuICAgICAgICBcbiAgICAgICAgY29uY3JldGVBcmd1bWVudHMubWFwKChwYXJhbSkgPT4ge1xuICAgICAgICAgICAgcGFyYW1zID0gcGFyYW1zK3BhcmFtLmdldFRleHQoKSsnLCdcbiAgICAgICAgfSlcbiAgICAgICAgXG4gICAgICAgIHBhcmFtcyA9IHBhcmFtcy5zdWJzdHJpbmcoMCwgcGFyYW1zLmxlbmd0aCAtIDEpXG4gICAgICAgICAgICAgICAgXG4gICAgICAgIGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUgPSBIZWxwZXJzLnJlcGxhY2VMYXN0QmV0d2VlblRhZ3MoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSxwYXJhbXMpO1xuXG4gICAgICAgIGxldCBjb25jRWxlbWVudDogUGFyYW1ldHJpY1ZhcmlhbnRUeXBlO1xuXG4gICAgICAgIGlmICghdGhpcy5mbXhJbnRlcmZhY2VNYXAuaGFzKGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpICYmIFxuICAgICAgICAgICAgIXRoaXMuZm14Q2xhc3NNYXAuaGFzKGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpICYmIFxuICAgICAgICAgICAgIXRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuaGFzKGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpKXtcbiAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gXy5jbG9uZURlZXAoY29uY3JldGVFbGVtZW50KTsgXG4gICAgICAgICAgICBjb25jRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUgPSBmdWxseVF1YWxpZmllZEZpbGVuYW1lO1xuICAgICAgICAgICAgY29uY0VsZW1lbnQuY2xlYXJHZW5lcmljUGFyYW1ldGVycygpO1xuICAgICAgICAgICAgY29uY3JldGVBcmd1bWVudHMubWFwKChwYXJhbSkgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlciA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlVHlwZShwYXJhbSk7XG4gICAgICAgICAgICAgICAgY29uY0VsZW1lbnQuYWRkQ29uY3JldGVQYXJhbWV0ZXIocGFyYW1ldGVyKTtcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgICBcbiAgICAgICAgICAgIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljQ2xhc3MpIHtcbiAgICAgICAgICAgICAgICB0aGlzLmZteENsYXNzTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBjb25jRWxlbWVudCBhcyBGYW1peC5QYXJhbWV0cmljQ2xhc3MpO1xuICAgICAgICAgICAgfSBlbHNlIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlKSB7XG4gICAgICAgICAgICAgICAgdGhpcy5mbXhJbnRlcmZhY2VNYXAuc2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUsIGNvbmNFbGVtZW50IGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UpO1xuICAgICAgICAgICAgfSBlbHNlIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24pIHtcbiAgICAgICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBjb25jRWxlbWVudCBhcyBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24pO1xuICAgICAgICAgICAgfSBlbHNlIHsgLy8gaWYgKGNvbmNyZXRlRWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNNZXRob2QpIHtcbiAgICAgICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdWxseVF1YWxpZmllZEZpbGVuYW1lLCBjb25jRWxlbWVudCBhcyBGYW1peC5QYXJhbWV0cmljTWV0aG9kKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChjb25jRWxlbWVudCk7XG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGNvbmNFbGVtZW50LGNvbmNyZXRlRWxlbWVudERlY2xhcmF0aW9uKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljQ2xhc3MpIHtcbiAgICAgICAgICAgICAgICBjb25jRWxlbWVudCA9IHRoaXMuZm14Q2xhc3NNYXAuZ2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcbiAgICAgICAgICAgIH0gZWxzZSBpZiAoY29uY3JldGVFbGVtZW50IGluc3RhbmNlb2YgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZSkge1xuICAgICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5mbXhJbnRlcmZhY2VNYXAuZ2V0KGZ1bGx5UXVhbGlmaWVkRmlsZW5hbWUpIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgICAgICB9IGVsc2UgaWYgKGNvbmNyZXRlRWxlbWVudCBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbikge1xuICAgICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5mbXhGdW5jdGlvbkFuZE1ldGhvZE1hcC5nZXQoZnVsbHlRdWFsaWZpZWRGaWxlbmFtZSkgYXMgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uO1xuICAgICAgICAgICAgfSBlbHNlIHsgIC8vIGlmIChjb25jcmV0ZUVsZW1lbnQgaW5zdGFuY2VvZiBGYW1peC5QYXJhbWV0cmljTWV0aG9kKSB7XG4gICAgICAgICAgICAgICAgY29uY0VsZW1lbnQgPSB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmdldChmdWxseVF1YWxpZmllZEZpbGVuYW1lKSBhcyBGYW1peC5QYXJhbWV0cmljTWV0aG9kO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBjb25jRWxlbWVudDtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggcHJvcGVydHlcbiAgICAgKiBAcGFyYW0gcHJvcGVydHkgQSBwcm9wZXJ0eVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgcHJvcGVydHlcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhQcm9wZXJ0eShwcm9wZXJ0eTogUHJvcGVydHlEZWNsYXJhdGlvbiB8IFByb3BlcnR5U2lnbmF0dXJlKTogRmFtaXguUHJvcGVydHkge1xuICAgICAgICBjb25zdCBmbXhQcm9wZXJ0eSA9IG5ldyBGYW1peC5Qcm9wZXJ0eSgpO1xuICAgICAgICBjb25zdCBpc1NpZ25hdHVyZSA9IHByb3BlcnR5IGluc3RhbmNlb2YgUHJvcGVydHlTaWduYXR1cmU7XG4gICAgICAgIGZteFByb3BlcnR5Lm5hbWUgPSBwcm9wZXJ0eS5nZXROYW1lKCk7XG5cbiAgICAgICAgbGV0IHByb3BUeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIHByb3BUeXBlTmFtZSA9IHByb3BlcnR5LmdldFR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHByb3BlcnR5OiAke3Byb3BlcnR5LmdldE5hbWUoKX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKHByb3BUeXBlTmFtZSwgcHJvcGVydHkpO1xuICAgICAgICBmbXhQcm9wZXJ0eS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuXG4gICAgICAgIC8vIGFkZCB0aGUgdmlzaWJpbGl0eSAocHVibGljLCBwcml2YXRlLCBldGMuKSB0byB0aGUgZm14UHJvcGVydHlcbiAgICAgICAgZm14UHJvcGVydHkudmlzaWJpbGl0eSA9IFwiXCI7XG5cbiAgICAgICAgcHJvcGVydHkuZ2V0TW9kaWZpZXJzKCkuZm9yRWFjaChtID0+IHtcbiAgICAgICAgICAgIHN3aXRjaCAobS5nZXRUZXh0KCkpIHtcbiAgICAgICAgICAgICAgICBjYXNlIFNjb3BlLlB1YmxpYzpcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkudmlzaWJpbGl0eSA9IFwicHVibGljXCI7XG4gICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIGNhc2UgU2NvcGUuUHJvdGVjdGVkOlxuICAgICAgICAgICAgICAgICAgICBmbXhQcm9wZXJ0eS52aXNpYmlsaXR5ID0gXCJwcm90ZWN0ZWRcIjtcbiAgICAgICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICAgICAgY2FzZSBTY29wZS5Qcml2YXRlOlxuICAgICAgICAgICAgICAgICAgICBmbXhQcm9wZXJ0eS52aXNpYmlsaXR5ID0gXCJwcml2YXRlXCI7XG4gICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIGNhc2UgXCJzdGF0aWNcIjpcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkuaXNDbGFzc1NpZGUgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgICAgICBjYXNlIFwicmVhZG9ubHlcIjpcbiAgICAgICAgICAgICAgICAgICAgZm14UHJvcGVydHkucmVhZE9ubHkgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgaWYgKCFpc1NpZ25hdHVyZSAmJiBwcm9wZXJ0eS5nZXRFeGNsYW1hdGlvblRva2VuTm9kZSgpKSB7XG4gICAgICAgICAgICBmbXhQcm9wZXJ0eS5pc0RlZmluaXRlbHlBc3NpZ25lZCA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHByb3BlcnR5LmdldFF1ZXN0aW9uVG9rZW5Ob2RlKCkpIHtcbiAgICAgICAgICAgIGZteFByb3BlcnR5LmlzT3B0aW9uYWwgPSB0cnVlO1xuICAgICAgICB9XG4gICAgICAgIGlmIChwcm9wZXJ0eS5nZXROYW1lKCkuc3Vic3RyaW5nKDAsIDEpID09PSBcIiNcIikge1xuICAgICAgICAgICAgZm14UHJvcGVydHkuaXNKYXZhU2NyaXB0UHJpdmF0ZSA9IHRydWU7XG4gICAgICAgIH1cblxuICAgICAgICBpbml0RlFOKHByb3BlcnR5LCBmbXhQcm9wZXJ0eSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHByb3BlcnR5LCBmbXhQcm9wZXJ0eSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFByb3BlcnR5KTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteFByb3BlcnR5LHByb3BlcnR5KTtcblxuICAgICAgICByZXR1cm4gZm14UHJvcGVydHk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IG1ldGhvZCBvciBhY2Nlc3NvclxuICAgICAqIEBwYXJhbSBtZXRob2QgQSBtZXRob2Qgb3IgYW4gYWNjZXNzb3JcbiAgICAgKiBAcGFyYW0gY3VycmVudENDIFRoZSBjeWNsb21hdGljIGNvbXBsZXhpdHkgbWV0cmljcyBvZiB0aGUgY3VycmVudCBzb3VyY2UgZmlsZVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgbWV0aG9kIG9yIHRoZSBhY2Nlc3NvclxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVPckdldEZhbWl4TWV0aG9kKG1ldGhvZDogTWV0aG9kRGVjbGFyYXRpb24gfCBDb25zdHJ1Y3RvckRlY2xhcmF0aW9uIHwgTWV0aG9kU2lnbmF0dXJlIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24sIGN1cnJlbnRDQzogeyBba2V5OiBzdHJpbmddOiBudW1iZXIgfSk6IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LkFjY2Vzc29yIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZCB7XG4gICAgICAgIGxldCBmbXhNZXRob2Q6IEZhbWl4Lk1ldGhvZCB8IEZhbWl4LkFjY2Vzc29yIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZDtcbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gbWV0aG9kLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgY29uc3QgZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKG1ldGhvZCk7XG4gICAgICAgIGlmICghdGhpcy5mbXhGdW5jdGlvbkFuZE1ldGhvZE1hcC5oYXMoZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUpKSB7XG5cbiAgICAgICAgICAgIGlmIChtZXRob2QgaW5zdGFuY2VvZiBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHx8IG1ldGhvZCBpbnN0YW5jZW9mIFNldEFjY2Vzc29yRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QgPSBuZXcgRmFtaXguQWNjZXNzb3IoKTtcbiAgICAgICAgICAgICAgICBjb25zdCBpc0dldHRlciA9IG1ldGhvZCBpbnN0YW5jZW9mIEdldEFjY2Vzc29yRGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgY29uc3QgaXNTZXR0ZXIgPSBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uO1xuICAgICAgICAgICAgICAgIGlmIChpc0dldHRlcikgeyhmbXhNZXRob2QgYXMgRmFtaXguQWNjZXNzb3IpLmtpbmQgPSBcImdldHRlclwiO31cbiAgICAgICAgICAgICAgICBpZiAoaXNTZXR0ZXIpIHsoZm14TWV0aG9kIGFzIEZhbWl4LkFjY2Vzc29yKS5raW5kID0gXCJzZXR0ZXJcIjt9XG4gICAgICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteE1ldGhvZCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBpZiAoaXNHZW5lcmljKSB7XG4gICAgICAgICAgICAgICAgICAgIGZteE1ldGhvZCA9IG5ldyBGYW1peC5QYXJhbWV0cmljTWV0aG9kKCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBmbXhNZXRob2QgPSBuZXcgRmFtaXguTWV0aG9kKCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhNZXRob2QpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29uc3QgaXNDb25zdHJ1Y3RvciA9IG1ldGhvZCBpbnN0YW5jZW9mIENvbnN0cnVjdG9yRGVjbGFyYXRpb247XG4gICAgICAgICAgICBjb25zdCBpc1NpZ25hdHVyZSA9IG1ldGhvZCBpbnN0YW5jZW9mIE1ldGhvZFNpZ25hdHVyZTtcblxuICAgICAgICAgICAgbGV0IGlzQWJzdHJhY3QgPSBmYWxzZTtcbiAgICAgICAgICAgIGxldCBpc1N0YXRpYyA9IGZhbHNlO1xuICAgICAgICAgICAgaWYgKG1ldGhvZCBpbnN0YW5jZW9mIE1ldGhvZERlY2xhcmF0aW9uIHx8IG1ldGhvZCBpbnN0YW5jZW9mIEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgICAgIGlzQWJzdHJhY3QgPSBtZXRob2QuaXNBYnN0cmFjdCgpO1xuICAgICAgICAgICAgICAgIGlzU3RhdGljID0gbWV0aG9kLmlzU3RhdGljKCk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGlmIChpc0NvbnN0cnVjdG9yKSB7KGZteE1ldGhvZCBhcyBGYW1peC5BY2Nlc3Nvcikua2luZCA9IFwiY29uc3RydWN0b3JcIjt9XG4gICAgICAgICAgICBmbXhNZXRob2QuaXNBYnN0cmFjdCA9IGlzQWJzdHJhY3Q7XG4gICAgICAgICAgICBmbXhNZXRob2QuaXNDbGFzc1NpZGUgPSBpc1N0YXRpYztcbiAgICAgICAgICAgIGZteE1ldGhvZC5pc1ByaXZhdGUgPSAobWV0aG9kIGluc3RhbmNlb2YgTWV0aG9kRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fCBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKSA/IChtZXRob2QuZ2V0TW9kaWZpZXJzKCkuZmluZCh4ID0+IHguZ2V0VGV4dCgpID09PSAncHJpdmF0ZScpKSAhPT0gdW5kZWZpbmVkIDogZmFsc2U7XG4gICAgICAgICAgICBmbXhNZXRob2QuaXNQcm90ZWN0ZWQgPSAobWV0aG9kIGluc3RhbmNlb2YgTWV0aG9kRGVjbGFyYXRpb24gfHwgbWV0aG9kIGluc3RhbmNlb2YgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8fCBtZXRob2QgaW5zdGFuY2VvZiBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKSA/IChtZXRob2QuZ2V0TW9kaWZpZXJzKCkuZmluZCh4ID0+IHguZ2V0VGV4dCgpID09PSAncHJvdGVjdGVkJykpICE9PSB1bmRlZmluZWQgOiBmYWxzZTtcbiAgICAgICAgICAgIGZteE1ldGhvZC5zaWduYXR1cmUgPSBIZWxwZXJzLmNvbXB1dGVTaWduYXR1cmUobWV0aG9kLmdldFRleHQoKSk7XG5cbiAgICAgICAgICAgIGxldCBtZXRob2ROYW1lOiBzdHJpbmc7XG4gICAgICAgICAgICBpZiAoaXNDb25zdHJ1Y3Rvcikge1xuICAgICAgICAgICAgICAgIG1ldGhvZE5hbWUgPSBcImNvbnN0cnVjdG9yXCI7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBtZXRob2ROYW1lID0gKG1ldGhvZCBhcyBNZXRob2REZWNsYXJhdGlvbiB8IE1ldGhvZFNpZ25hdHVyZSB8IEdldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBTZXRBY2Nlc3NvckRlY2xhcmF0aW9uKS5nZXROYW1lKCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBmbXhNZXRob2QubmFtZSA9IG1ldGhvZE5hbWU7XG5cbiAgICAgICAgICAgIGlmICghaXNDb25zdHJ1Y3Rvcikge1xuICAgICAgICAgICAgICAgIGlmIChtZXRob2QuZ2V0TmFtZSgpLnN1YnN0cmluZygwLCAxKSA9PT0gXCIjXCIpIHtcbiAgICAgICAgICAgICAgICAgICAgZm14TWV0aG9kLmlzUHJpdmF0ZSA9IHRydWU7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZiAoIWZteE1ldGhvZC5pc1ByaXZhdGUgJiYgIWZteE1ldGhvZC5pc1Byb3RlY3RlZCkge1xuICAgICAgICAgICAgICAgIGZteE1ldGhvZC5pc1B1YmxpYyA9IHRydWU7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QuaXNQdWJsaWMgPSBmYWxzZTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgaWYgKCFpc1NpZ25hdHVyZSkge1xuICAgICAgICAgICAgICAgIGZteE1ldGhvZC5jeWNsb21hdGljQ29tcGxleGl0eSA9IGN1cnJlbnRDQ1tmbXhNZXRob2QubmFtZV07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBmbXhNZXRob2QuY3ljbG9tYXRpY0NvbXBsZXhpdHkgPSAwO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBsZXQgbWV0aG9kVHlwZU5hbWUgPSB0aGlzLlVOS05PV05fVkFMVUU7IFxuICAgICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgICAgICBtZXRob2RUeXBlTmFtZSA9IG1ldGhvZC5nZXRSZXR1cm5UeXBlKCkuZ2V0VGV4dCgpLnRyaW0oKTsgICAgICAgICAgICBcbiAgICAgICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHJldHVybiB0eXBlIG9mIG1ldGhvZDogJHtmbXhNZXRob2QubmFtZX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUobWV0aG9kVHlwZU5hbWUsIG1ldGhvZCk7XG4gICAgICAgICAgICBmbXhNZXRob2QuZGVjbGFyZWRUeXBlID0gZm14VHlwZTtcbiAgICAgICAgICAgIGZteE1ldGhvZC5udW1iZXJPZkxpbmVzT2ZDb2RlID0gbWV0aG9kLmdldEVuZExpbmVOdW1iZXIoKSAtIG1ldGhvZC5nZXRTdGFydExpbmVOdW1iZXIoKTtcbiAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPSBtZXRob2QuZ2V0UGFyYW1ldGVycygpO1xuICAgICAgICAgICAgZm14TWV0aG9kLm51bWJlck9mUGFyYW1ldGVycyA9IHBhcmFtZXRlcnMubGVuZ3RoO1xuXG4gICAgICAgICAgICBpZiAoIWlzU2lnbmF0dXJlKSB7XG4gICAgICAgICAgICAgICAgZm14TWV0aG9kLm51bWJlck9mU3RhdGVtZW50cyA9IG1ldGhvZC5nZXRTdGF0ZW1lbnRzKCkubGVuZ3RoO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14TWV0aG9kLm51bWJlck9mU3RhdGVtZW50cyA9IDA7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBcbiAgICAgICAgICAgIGluaXRGUU4obWV0aG9kLCBmbXhNZXRob2QpO1xuICAgICAgICAgICAgdGhpcy5tYWtlRmFtaXhJbmRleEZpbGVBbmNob3IobWV0aG9kLCBmbXhNZXRob2QpO1xuXG4gICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSwgZm14TWV0aG9kKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteE1ldGhvZCA9IHRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuZ2V0KGZ1bmN0aW9uRnVsbHlRdWFsaWZpZWROYW1lKSBhcyAoRmFtaXguTWV0aG9kIHwgRmFtaXguQWNjZXNzb3IgfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14TWV0aG9kLG1ldGhvZCk7XG4gICAgICAgIFxuICAgICAgICByZXR1cm4gZm14TWV0aG9kO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBmdW5jdGlvblxuICAgICAqIEBwYXJhbSBmdW5jIEEgZnVuY3Rpb25cbiAgICAgKiBAcGFyYW0gY3VycmVudENDIFRoZSBjeWNsb21hdGljIGNvbXBsZXhpdHkgbWV0cmljcyBvZiB0aGUgY3VycmVudCBzb3VyY2UgZmlsZVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgZnVuY3Rpb25cbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peEZ1bmN0aW9uKGZ1bmM6IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24sIGN1cnJlbnRDQzogeyBba2V5OiBzdHJpbmddOiBudW1iZXIgfSk6IEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uIHtcbiAgICAgICAgbGV0IGZteEZ1bmN0aW9uOiBGYW1peC5GdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbjtcbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gZnVuYy5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7ICAgICAgICBcbiAgICAgICAgY29uc3QgZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGZ1bmMpO1xuICAgICAgICBpZiAoIXRoaXMuZm14RnVuY3Rpb25BbmRNZXRob2RNYXAuaGFzKGZ1bmN0aW9uRnVsbHlRdWFsaWZpZWROYW1lKSkge1xuICAgICAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgICAgIGZteEZ1bmN0aW9uID0gbmV3IEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbigpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14RnVuY3Rpb24gPSBuZXcgRmFtaXguRnVuY3Rpb24oKTtcbiAgICAgICAgICAgIH1cbiAgICBcbiAgICAgICAgICAgIGNvbnN0IG5hbWUgPSBmdW5jLmdldE5hbWUoKTtcbiAgICAgICAgICAgIGlmIChuYW1lKSB7XG4gICAgICAgICAgICAgICAgZm14RnVuY3Rpb24ubmFtZSA9IG5hbWU7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBmbXhGdW5jdGlvbi5uYW1lID0gXCJhbm9ueW1vdXNcIjtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgZm14RnVuY3Rpb24uc2lnbmF0dXJlID0gSGVscGVycy5jb21wdXRlU2lnbmF0dXJlKGZ1bmMuZ2V0VGV4dCgpKTtcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLmN5Y2xvbWF0aWNDb21wbGV4aXR5ID0gY3VycmVudENDW2ZteEZ1bmN0aW9uLm5hbWVdO1xuICAgICAgICAgICAgZm14RnVuY3Rpb24uZnVsbHlRdWFsaWZpZWROYW1lID0gZnVuY3Rpb25GdWxseVF1YWxpZmllZE5hbWU7XG4gICAgXG4gICAgICAgICAgICBsZXQgZnVuY3Rpb25UeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgICAgZnVuY3Rpb25UeXBlTmFtZSA9IGZ1bmMuZ2V0UmV0dXJuVHlwZSgpLmdldFRleHQoKS50cmltKCk7XG4gICAgICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgICAgIGxvZ2dlci5lcnJvcihgPiBXQVJOSU5HOiBnb3QgZXhjZXB0aW9uICR7ZXJyb3J9LiBGYWlsZWQgdG8gZ2V0IHVzYWJsZSBuYW1lIGZvciByZXR1cm4gdHlwZSBvZiBmdW5jdGlvbjogJHtmdW5jLmdldE5hbWUoKX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgICAgIH1cbiAgICBcbiAgICAgICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKGZ1bmN0aW9uVHlwZU5hbWUsIGZ1bmMpO1xuICAgICAgICAgICAgZm14RnVuY3Rpb24uZGVjbGFyZWRUeXBlID0gZm14VHlwZTtcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLm51bWJlck9mTGluZXNPZkNvZGUgPSBmdW5jLmdldEVuZExpbmVOdW1iZXIoKSAtIGZ1bmMuZ2V0U3RhcnRMaW5lTnVtYmVyKCk7XG4gICAgICAgICAgICBjb25zdCBwYXJhbWV0ZXJzID0gZnVuYy5nZXRQYXJhbWV0ZXJzKCk7XG4gICAgICAgICAgICBmbXhGdW5jdGlvbi5udW1iZXJPZlBhcmFtZXRlcnMgPSBwYXJhbWV0ZXJzLmxlbmd0aDtcbiAgICAgICAgICAgIGZteEZ1bmN0aW9uLm51bWJlck9mU3RhdGVtZW50cyA9IGZ1bmMuZ2V0U3RhdGVtZW50cygpLmxlbmd0aDtcbiAgICAgICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGZ1bmMsIGZteEZ1bmN0aW9uKTtcbiAgICBcbiAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhGdW5jdGlvbik7XG4gICAgXG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEZ1bmN0aW9uLGZ1bmMpO1xuXG4gICAgICAgICAgICB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLnNldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSwgZm14RnVuY3Rpb24pO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14RnVuY3Rpb24gPSB0aGlzLmZteEZ1bmN0aW9uQW5kTWV0aG9kTWFwLmdldChmdW5jdGlvbkZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgKEZhbWl4LkZ1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiBmbXhGdW5jdGlvbjtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggcGFyYW1ldGVyXG4gICAgICogQHBhcmFtIHBhcmFtIEEgcGFyYW1ldGVyXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBwYXJhbWV0ZXJcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhQYXJhbWV0ZXIocGFyYW06IFBhcmFtZXRlckRlY2xhcmF0aW9uKTogRmFtaXguUGFyYW1ldGVyIHtcbiAgICAgICAgY29uc3QgZm14UGFyYW0gPSBuZXcgRmFtaXguUGFyYW1ldGVyKCk7XG5cbiAgICAgICAgbGV0IHBhcmFtVHlwZU5hbWUgPSB0aGlzLlVOS05PV05fVkFMVUU7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgICBwYXJhbVR5cGVOYW1lID0gcGFyYW0uZ2V0VHlwZSgpLmdldFRleHQoKS50cmltKCk7XG4gICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgICAgICBsb2dnZXIuZXJyb3IoYD4gV0FSTklORzogZ290IGV4Y2VwdGlvbiAke2Vycm9yfS4gRmFpbGVkIHRvIGdldCB1c2FibGUgbmFtZSBmb3IgcGFyYW1ldGVyOiAke3BhcmFtLmdldE5hbWUoKX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKHBhcmFtVHlwZU5hbWUsIHBhcmFtKTtcbiAgICAgICAgZm14UGFyYW0uZGVjbGFyZWRUeXBlID0gZm14VHlwZTtcbiAgICAgICAgZm14UGFyYW0ubmFtZSA9IHBhcmFtLmdldE5hbWUoKTtcblxuICAgICAgICBpbml0RlFOKHBhcmFtLCBmbXhQYXJhbSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHBhcmFtLCBmbXhQYXJhbSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtKTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteFBhcmFtLHBhcmFtKTtcblxuICAgICAgICByZXR1cm4gZm14UGFyYW07XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IHR5cGUgcGFyYW1ldGVyXG4gICAgICogQHBhcmFtIHRwIEEgdHlwZSBwYXJhbWV0ZXJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHR5cGUgcGFyYW1ldGVyXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4UGFyYW1ldGVyVHlwZSh0cDogVHlwZVBhcmFtZXRlckRlY2xhcmF0aW9uKTogRmFtaXguUGFyYW1ldGVyVHlwZSB7XG4gICAgICAgIFxuICAgICAgICBjb25zdCBmbXhQYXJhbWV0ZXJUeXBlID0gbmV3IEZhbWl4LlBhcmFtZXRlclR5cGUoKTtcbiAgIFxuICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlLm5hbWUgPSB0cC5nZXROYW1lKCk7ICAgICAgXG4gICAgICAgIGluaXRGUU4odHAsIGZteFBhcmFtZXRlclR5cGUpO1xuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcih0cCwgZm14UGFyYW1ldGVyVHlwZSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtZXRlclR5cGUpO1xuXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UGFyYW1ldGVyVHlwZSx0cCk7XG5cbiAgICAgICAgcmV0dXJuIGZteFBhcmFtZXRlclR5cGU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IHR5cGUgcGFyYW1ldGVyXG4gICAgICogQHBhcmFtIHRwIEEgdHlwZSBwYXJhbWV0ZXJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHR5cGUgcGFyYW1ldGVyXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhDb25jcmV0ZVR5cGUocGFyYW06IFR5cGVOb2RlKTogRmFtaXguUGFyYW1ldGVyVHlwZSB8IEZhbWl4LlByaW1pdGl2ZVR5cGUgfCBGYW1peC5DbGFzcyB8IEZhbWl4LkludGVyZmFjZSB7XG4gICAgICAgIGNvbnN0IHR5cGVQYXJhbWV0ZXJEZWNsYXJhdGlvbiA9IHBhcmFtLmdldFN5bWJvbCgpPy5nZXREZWNsYXJhdGlvbnMoKVswXSBhcyBUeXBlUGFyYW1ldGVyRGVjbGFyYXRpb247XG4gICAgICAgIGNvbnN0IHBhcmFtZXRlclR5cGVOYW1lIDogc3RyaW5nID0gcGFyYW0uZ2V0VGV4dCgpO1xuICAgICAgICBsZXQgZm14UGFyYW1ldGVyVHlwZTogRmFtaXguVHlwZSB8IEZhbWl4LkNsYXNzIHwgRmFtaXguSW50ZXJmYWNlIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkO1xuXG4gICAgICAgIGxldCBpc0NsYXNzT3JJbnRlcmZhY2UgPSBmYWxzZTtcbiAgICAgICAgaWYgKHRoaXMuZm14Q2xhc3NNYXAuaGFzKHBhcmFtZXRlclR5cGVOYW1lKSl7XG4gICAgICAgICAgICB0aGlzLmZteENsYXNzTWFwLmZvckVhY2goKG9iaiwgbmFtZSkgPT4ge1xuICAgICAgICAgICAgICAgIGlmKG9iaiBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNDbGFzcyl7XG4gICAgICAgICAgICAgICAgICAgIGlmIChuYW1lID09PSBwYXJhbS5nZXRUZXh0KCkgJiYgb2JqLmdlbmVyaWNQYXJhbWV0ZXJzLnNpemU+MCkge1xuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IG9iajtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlzQ2xhc3NPckludGVyZmFjZSA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgIH0gXG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgaWYgKG5hbWUgPT09IHBhcmFtLmdldFRleHQoKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZSA9IG9iajtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlzQ2xhc3NPckludGVyZmFjZSA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgIH0gXG4gICAgICAgICAgICAgICAgfSAgIFxuICAgICAgICAgICAgfSlcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh0aGlzLmZteEludGVyZmFjZU1hcC5oYXMocGFyYW1ldGVyVHlwZU5hbWUpKXtcbiAgICAgICAgICAgIHRoaXMuZm14SW50ZXJmYWNlTWFwLmZvckVhY2goKG9iaiwgbmFtZSkgPT4ge1xuICAgICAgICAgICAgICAgIGlmKG9iaiBpbnN0YW5jZW9mIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2Upe1xuICAgICAgICAgICAgICAgICAgICBpZiAobmFtZSA9PT0gcGFyYW0uZ2V0VGV4dCgpICYmIG9iai5nZW5lcmljUGFyYW1ldGVycy5zaXplPjApIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlclR5cGUgPSBvYmo7XG4gICAgICAgICAgICAgICAgICAgICAgICBpc0NsYXNzT3JJbnRlcmZhY2UgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICB9IFxuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIGlmIChuYW1lID09PSBwYXJhbS5nZXRUZXh0KCkpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlclR5cGUgPSBvYmo7XG4gICAgICAgICAgICAgICAgICAgICAgICBpc0NsYXNzT3JJbnRlcmZhY2UgPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICB9IFxuICAgICAgICAgICAgICAgIH0gICBcbiAgICAgICAgICAgIH0pXG4gICAgICAgIH1cblxuICAgICAgICBpZighaXNDbGFzc09ySW50ZXJmYWNlKXtcbiAgICAgICAgICAgIGlmICghdGhpcy5mbXhUeXBlTWFwLmhhcyhwYXJhbWV0ZXJUeXBlTmFtZSkpIHsgICAgICAgICAgIFxuICAgICAgICAgICAgICAgIGlmIChwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJudW1iZXJcIiB8fCBwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJzdHJpbmdcIiB8fCBwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJib29sZWFuXCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwiYmlnaW50XCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwic3ltYm9sXCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwidW5kZWZpbmVkXCIgfHwgcGFyYW1ldGVyVHlwZU5hbWUgPT09IFwibnVsbFwiIHx8IHBhcmFtZXRlclR5cGVOYW1lID09PSBcImFueVwiIHx8IHBhcmFtZXRlclR5cGVOYW1lID09PSBcInVua25vd25cIiB8fCBwYXJhbWV0ZXJUeXBlTmFtZSA9PT0gXCJuZXZlclwiIHx8IHBhcmFtZXRlclR5cGVOYW1lID09PSBcInZvaWRcIikge1xuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlID0gbmV3IEZhbWl4LlByaW1pdGl2ZVR5cGUoKTtcbiAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZS5pc1N0dWIgPSB0cnVlO1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlclR5cGUgPSBuZXcgRmFtaXguUGFyYW1ldGVyVHlwZSgpO1xuICAgICAgICAgICAgICAgIH0gXG4gICAgXG4gICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyVHlwZS5uYW1lID0gcGFyYW1ldGVyVHlwZU5hbWU7XG4gICAgICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFBhcmFtZXRlclR5cGUpO1xuICAgICAgICAgICAgICAgIHRoaXMuZm14VHlwZU1hcC5zZXQocGFyYW1ldGVyVHlwZU5hbWUsIGZteFBhcmFtZXRlclR5cGUpO1xuICAgICAgICAgICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UGFyYW1ldGVyVHlwZSx0eXBlUGFyYW1ldGVyRGVjbGFyYXRpb24pO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gdGhpcy5mbXhUeXBlTWFwLmdldChwYXJhbWV0ZXJUeXBlTmFtZSk7XG4gICAgICAgICAgICAgICAgaWYgKHJlc3VsdCkge1xuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJUeXBlID0gcmVzdWx0O1xuICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgRmFtaXggdHlwZSAke3BhcmFtZXRlclR5cGVOYW1lfSBpcyBub3QgZm91bmQgaW4gdGhlIFR5cGUgbWFwLmApO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICBpZiAoIWZteFBhcmFtZXRlclR5cGUpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgZm14UGFyYW1ldGVyVHlwZSB3YXMgdW5kZWZpbmVkIGZvciBwYXJhbWV0ZXJUeXBlTmFtZSAke3BhcmFtZXRlclR5cGVOYW1lfWApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBmbXhQYXJhbWV0ZXJUeXBlO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCB2YXJpYWJsZVxuICAgICAqIEBwYXJhbSB2YXJpYWJsZSBBIHZhcmlhYmxlXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSB2YXJpYWJsZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peFZhcmlhYmxlKHZhcmlhYmxlOiBWYXJpYWJsZURlY2xhcmF0aW9uKTogRmFtaXguVmFyaWFibGUge1xuICAgICAgICBjb25zdCBmbXhWYXJpYWJsZSA9IG5ldyBGYW1peC5WYXJpYWJsZSgpO1xuICAgIFxuICAgICAgICBsZXQgdmFyaWFibGVUeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIHZhcmlhYmxlVHlwZU5hbWUgPSB2YXJpYWJsZS5nZXRUeXBlKCkuZ2V0VGV4dCgpLnRyaW0oKTtcbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAgIGxvZ2dlci5lcnJvcihgPiBXQVJOSU5HOiBnb3QgZXhjZXB0aW9uICR7ZXJyb3J9LiBGYWlsZWQgdG8gZ2V0IHVzYWJsZSBuYW1lIGZvciB2YXJpYWJsZTogJHt2YXJpYWJsZS5nZXROYW1lKCl9LiBDb250aW51aW5nLi4uYCk7XG4gICAgICAgIH1cbiAgICBcbiAgICAgICAgY29uc3QgZm14VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUodmFyaWFibGVUeXBlTmFtZSwgdmFyaWFibGUpO1xuICAgICAgICBmbXhWYXJpYWJsZS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuICAgICAgICBmbXhWYXJpYWJsZS5uYW1lID0gdmFyaWFibGUuZ2V0TmFtZSgpO1xuICAgICAgICBpbml0RlFOKHZhcmlhYmxlLCBmbXhWYXJpYWJsZSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKHZhcmlhYmxlLCBmbXhWYXJpYWJsZSk7XG4gICAgXG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhWYXJpYWJsZSk7XG4gICAgXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14VmFyaWFibGUsdmFyaWFibGUpO1xuICAgIFxuICAgICAgICByZXR1cm4gZm14VmFyaWFibGU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGVudW1cbiAgICAgKiBAcGFyYW0gZW51bUVudGl0eSBBbiBlbnVtXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBlbnVtXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4RW51bShlbnVtRW50aXR5OiBFbnVtRGVjbGFyYXRpb24pOiBGYW1peC5FbnVtIHtcbiAgICAgICAgY29uc3QgZm14RW51bSA9IG5ldyBGYW1peC5FbnVtKCk7XG4gICAgICAgIGZteEVudW0ubmFtZSA9IGVudW1FbnRpdHkuZ2V0TmFtZSgpO1xuICAgICAgICBpbml0RlFOKGVudW1FbnRpdHksIGZteEVudW0pO1xuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihlbnVtRW50aXR5LCBmbXhFbnVtKTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14RW51bSk7XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhFbnVtLGVudW1FbnRpdHkpO1xuXG4gICAgICAgIHJldHVybiBmbXhFbnVtO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBlbnVtIHZhbHVlXG4gICAgICogQHBhcmFtIGVudW1NZW1iZXIgQW4gZW51bSBtZW1iZXJcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIGVudW0gbWVtYmVyXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4RW51bVZhbHVlKGVudW1NZW1iZXI6IEVudW1NZW1iZXIpOiBGYW1peC5FbnVtVmFsdWUge1xuICAgICAgICBjb25zdCBmbXhFbnVtVmFsdWUgPSBuZXcgRmFtaXguRW51bVZhbHVlKCk7XG5cbiAgICAgICAgbGV0IGVudW1WYWx1ZVR5cGVOYW1lID0gdGhpcy5VTktOT1dOX1ZBTFVFO1xuICAgICAgICB0cnkge1xuICAgICAgICAgICAgZW51bVZhbHVlVHlwZU5hbWUgPSBlbnVtTWVtYmVyLmdldFR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIGVudW0gdmFsdWU6ICR7ZW51bU1lbWJlci5nZXROYW1lKCl9LiBDb250aW51aW5nLi4uYCk7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBmbXhUeXBlID0gdGhpcy5jcmVhdGVPckdldEZhbWl4VHlwZShlbnVtVmFsdWVUeXBlTmFtZSwgZW51bU1lbWJlcik7XG4gICAgICAgIGZteEVudW1WYWx1ZS5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuICAgICAgICBmbXhFbnVtVmFsdWUubmFtZSA9IGVudW1NZW1iZXIuZ2V0TmFtZSgpO1xuICAgICAgICBpbml0RlFOKGVudW1NZW1iZXIsIGZteEVudW1WYWx1ZSk7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGVudW1NZW1iZXIsIGZteEVudW1WYWx1ZSk7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEVudW1WYWx1ZSk7XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhFbnVtVmFsdWUsZW51bU1lbWJlcik7XG5cbiAgICAgICAgcmV0dXJuIGZteEVudW1WYWx1ZTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIG9yIGdldHMgYSBGYW1peCBkZWNvcmF0b3JcbiAgICAgKiBAcGFyYW0gZGVjb3JhdG9yIEEgZGVjb3JhdG9yXG4gICAgICogQHBhcmFtIGRlY29yYXRlZEVudGl0eSBBIGNsYXNzLCBhIG1ldGhvZCwgYSBwYXJhbWV0ZXIgb3IgYSBwcm9wZXJ0eVxuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgZGVjb3JhdG9yXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZU9yR2V0RmFtaXhEZWNvcmF0b3IoZGVjb3JhdG9yOiBEZWNvcmF0b3IsIGRlY29yYXRlZEVudGl0eTogQ2xhc3NEZWNsYXJhdGlvbiB8IE1ldGhvZERlY2xhcmF0aW9uIHwgR2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IFNldEFjY2Vzc29yRGVjbGFyYXRpb24gfCBQYXJhbWV0ZXJEZWNsYXJhdGlvbiB8IFByb3BlcnR5RGVjbGFyYXRpb24pOiBGYW1peC5EZWNvcmF0b3Ige1xuICAgICAgICBjb25zdCBmbXhEZWNvcmF0b3IgPSBuZXcgRmFtaXguRGVjb3JhdG9yKCk7XG4gICAgICAgIGNvbnN0IGRlY29yYXRvck5hbWUgPSBcIkBcIiArIGRlY29yYXRvci5nZXROYW1lKCk7XG4gICAgICAgIGNvbnN0IGRlY29yYXRvckV4cHJlc3Npb24gPSBkZWNvcmF0b3IuZ2V0VGV4dCgpLnN1YnN0cmluZygxKTtcblxuICAgICAgICBmbXhEZWNvcmF0b3IubmFtZSA9IGRlY29yYXRvck5hbWU7XG4gICAgICAgIGZteERlY29yYXRvci5kZWNvcmF0b3JFeHByZXNzaW9uID0gZGVjb3JhdG9yRXhwcmVzc2lvbjtcbiAgICAgICAgY29uc3QgZGVjb3JhdGVkRW50aXR5RnVsbHlRdWFsaWZpZWROYW1lID0gRlFORnVuY3Rpb25zLmdldEZRTihkZWNvcmF0ZWRFbnRpdHkpO1xuICAgICAgICBjb25zdCBmbXhEZWNvcmF0ZWRFbnRpdHkgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUoZGVjb3JhdGVkRW50aXR5RnVsbHlRdWFsaWZpZWROYW1lKSBhcyBGYW1peC5OYW1lZEVudGl0eTtcbiAgICAgICAgZm14RGVjb3JhdG9yLmRlY29yYXRlZEVudGl0eSA9IGZteERlY29yYXRlZEVudGl0eTtcbiAgICAgICAgaW5pdEZRTihkZWNvcmF0b3IsIGZteERlY29yYXRvcik7XG4gICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGRlY29yYXRvciwgZm14RGVjb3JhdG9yKTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14RGVjb3JhdG9yKTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteERlY29yYXRvcixkZWNvcmF0b3IpO1xuXG4gICAgICAgIHJldHVybiBmbXhEZWNvcmF0b3I7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGNvbW1lbnRcbiAgICAgKiBAcGFyYW0gY29tbWVudCBBIGNvbW1lbnRcbiAgICAgKiBAcGFyYW0gZm14U2NvcGUgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjb21tZW50J3MgY29udGFpbmVyXG4gICAgICogQHBhcmFtIGlzSlNEb2MgQSBib29sZWFuIGluZGljYXRpbmcgaWYgdGhlIGNvbW1lbnQgaXMgYSBKU0RvY1xuICAgICAqIEByZXR1cm5zIFRoZSBGYW1peCBtb2RlbCBvZiB0aGUgY29tbWVudFxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbW1lbnQoY29tbWVudDogQ29tbWVudFJhbmdlLCBmbXhTY29wZTogRmFtaXguTmFtZWRFbnRpdHksIGlzSlNEb2M6IGJvb2xlYW4pOiBGYW1peC5Db21tZW50IHtcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGA+IE5PVEU6IGNyZWF0aW5nIGNvbW1lbnQgJHtjb21tZW50LmdldFRleHQoKX0gaW4gc2NvcGUgJHtmbXhTY29wZS5uYW1lfS5gKTtcbiAgICAgICAgY29uc3QgZm14Q29tbWVudCA9IG5ldyBGYW1peC5Db21tZW50KCk7XG4gICAgICAgIGZteENvbW1lbnQuY29udGFpbmVyID0gZm14U2NvcGU7ICAvLyBhZGRzIGNvbW1lbnQgdG8gdGhlIGNvbnRhaW5lcidzIGNvbW1lbnRzIGNvbGxlY3Rpb25cbiAgICAgICAgZm14Q29tbWVudC5pc0pTRG9jID0gaXNKU0RvYztcblxuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihjb21tZW50LCBmbXhDb21tZW50KTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14Q29tbWVudCk7XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhDb21tZW50LGNvbW1lbnQpO1xuXG4gICAgICAgIHJldHVybiBmbXhDb21tZW50O1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgb3IgZ2V0cyBhIEZhbWl4IHR5cGVcbiAgICAgKiBAcGFyYW0gdHlwZU5hbWUgQSB0eXBlIG5hbWVcbiAgICAgKiBAcGFyYW0gZWxlbWVudCBBIHRzLW1vcnBoIGVsZW1lbnRcbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIHR5cGVcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlT3JHZXRGYW1peFR5cGUodHlwZU5hbWU6IHN0cmluZywgZWxlbWVudDogVHlwZURlY2xhcmF0aW9uKTogRmFtaXguVHlwZSB8IEZhbWl4LlByaW1pdGl2ZVR5cGUgfCBGYW1peC5QYXJhbWV0ZXJUeXBlIHtcbiAgICAgICAgbGV0IGZteFR5cGU6IEZhbWl4LlR5cGUgfCBGYW1peC5QcmltaXRpdmVUeXBlIHwgRmFtaXguUGFyYW1ldGVyVHlwZTtcbiAgICAgICAgbGV0IGlzUHJpbWl0aXZlVHlwZSA9IGZhbHNlO1xuICAgICAgICBsZXQgaXNQYXJhbWV0ZXJUeXBlID0gZmFsc2U7XG5cbiAgICAgICAgbG9nZ2VyLmRlYnVnKFwiQ3JlYXRpbmcgKG9yIGdldHRpbmcpIHR5cGU6ICdcIiArIHR5cGVOYW1lICsgXCInIG9mIGVsZW1lbnQ6IFwiICsgZWxlbWVudD8uZ2V0VGV4dCgpICsgXCIgb2Yga2luZDogXCIgKyBlbGVtZW50Py5nZXRLaW5kTmFtZSgpKTtcbiAgICAgICAgbGV0IGFuY2VzdG9yOiBGYW1peC5Db250YWluZXJFbnRpdHkgfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ7XG4gICAgICAgIGlmIChlbGVtZW50ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGNvbnN0IHR5cGVBbmNlc3RvciA9IEhlbHBlcnMuZmluZFR5cGVBbmNlc3RvcihlbGVtZW50KTtcbiAgICAgICAgICAgIGlmICghdHlwZUFuY2VzdG9yKSB7XG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBBbmNlc3RvciBub3QgZm91bmQgZm9yIGVsZW1lbnQgJHtlbGVtZW50LmdldFRleHQoKX0uYCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjb25zdCBhbmNlc3RvckZ1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4odHlwZUFuY2VzdG9yKTtcbiAgICAgICAgICAgIGFuY2VzdG9yID0gdGhpcy5mYW1peFJlcC5nZXRGYW1peEVudGl0eUJ5RnVsbHlRdWFsaWZpZWROYW1lKGFuY2VzdG9yRnVsbHlRdWFsaWZpZWROYW1lKSBhcyBGYW1peC5Db250YWluZXJFbnRpdHk7XG4gICAgICAgICAgICBpZiAoIWFuY2VzdG9yKSB7XG4gICAgICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKGBBbmNlc3RvciAke0ZRTkZ1bmN0aW9ucy5nZXRGUU4odHlwZUFuY2VzdG9yKX0gbm90IGZvdW5kLiBBZGRpbmcgdGhlIG5ldyB0eXBlLmApO1xuICAgICAgICAgICAgICAgIGFuY2VzdG9yID0gdGhpcy5jcmVhdGVPckdldEZhbWl4VHlwZSh0eXBlQW5jZXN0b3IuZ2V0VGV4dCgpLCB0eXBlQW5jZXN0b3IgYXMgVHlwZURlY2xhcmF0aW9uKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh0eXBlTmFtZSA9PT0gXCJudW1iZXJcIiB8fCB0eXBlTmFtZSA9PT0gXCJzdHJpbmdcIiB8fCB0eXBlTmFtZSA9PT0gXCJib29sZWFuXCIgfHwgdHlwZU5hbWUgPT09IFwiYmlnaW50XCIgfHwgdHlwZU5hbWUgPT09IFwic3ltYm9sXCIgfHwgdHlwZU5hbWUgPT09IFwidW5kZWZpbmVkXCIgfHwgdHlwZU5hbWUgPT09IFwibnVsbFwiIHx8IHR5cGVOYW1lID09PSBcImFueVwiIHx8IHR5cGVOYW1lID09PSBcInVua25vd25cIiB8fCB0eXBlTmFtZSA9PT0gXCJuZXZlclwiIHx8IHR5cGVOYW1lID09PSBcInZvaWRcIikge1xuICAgICAgICAgICAgaXNQcmltaXRpdmVUeXBlID0gdHJ1ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmKCFpc1ByaW1pdGl2ZVR5cGUgJiYgdHlwZU5hbWUuaW5jbHVkZXMoXCI8XCIpICYmIHR5cGVOYW1lLmluY2x1ZGVzKFwiPlwiKSAmJiAhKHR5cGVOYW1lLmluY2x1ZGVzKFwiPT5cIikpKSB7XG4gICAgICAgICAgICBpc1BhcmFtZXRlclR5cGUgPSB0cnVlO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKCF0aGlzLmZteFR5cGVNYXAuaGFzKHR5cGVOYW1lKSkge1xuICAgICAgICAgICAgaWYgKGlzUHJpbWl0aXZlVHlwZSkge1xuICAgICAgICAgICAgICAgIGZteFR5cGUgPSBuZXcgRmFtaXguUHJpbWl0aXZlVHlwZSgpO1xuICAgICAgICAgICAgICAgIGZteFR5cGUuaXNTdHViID0gdHJ1ZTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2UgaWYgKGlzUGFyYW1ldGVyVHlwZSkge1xuICAgICAgICAgICAgICAgIGZteFR5cGUgPSBuZXcgRmFtaXguUGFyYW1ldGVyVHlwZSgpO1xuICAgICAgICAgICAgICAgIGNvbnN0IHBhcmFtZXRlclR5cGVOYW1lcyA9IHR5cGVOYW1lLnN1YnN0cmluZyh0eXBlTmFtZS5pbmRleE9mKFwiPFwiKSArIDEsIHR5cGVOYW1lLmluZGV4T2YoXCI+XCIpKS5zcGxpdChcIixcIikubWFwKHMgPT4gcy50cmltKCkpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGJhc2VUeXBlTmFtZSA9IHR5cGVOYW1lLnN1YnN0cmluZygwLCB0eXBlTmFtZS5pbmRleE9mKFwiPFwiKSkudHJpbSgpO1xuICAgICAgICAgICAgICAgIHBhcmFtZXRlclR5cGVOYW1lcy5mb3JFYWNoKHBhcmFtZXRlclR5cGVOYW1lID0+IHtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZm14UGFyYW1ldGVyVHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUocGFyYW1ldGVyVHlwZU5hbWUsIGVsZW1lbnQpO1xuICAgICAgICAgICAgICAgICAgICAoZm14VHlwZSBhcyBGYW1peC5QYXJhbWV0ZXJUeXBlKS5hZGRBcmd1bWVudChmbXhQYXJhbWV0ZXJUeXBlKTtcbiAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICBjb25zdCBmbXhCYXNlVHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoYmFzZVR5cGVOYW1lLCBlbGVtZW50KTtcbiAgICAgICAgICAgICAgICAoZm14VHlwZSBhcyBGYW1peC5QYXJhbWV0ZXJUeXBlKS5iYXNlVHlwZSA9IGZteEJhc2VUeXBlO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgZm14VHlwZSA9IG5ldyBGYW1peC5UeXBlKCk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIGZteFR5cGUubmFtZSA9IHR5cGVOYW1lO1xuICAgICAgICAgICAgaWYgKCFhbmNlc3Rvcikge1xuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgQW5jZXN0b3Igbm90IGZvdW5kIGZvciB0eXBlICR7dHlwZU5hbWV9LmApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZm14VHlwZS5jb250YWluZXIgPSBhbmNlc3RvcjtcbiAgICAgICAgICAgIGluaXRGUU4oZWxlbWVudCwgZm14VHlwZSk7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihlbGVtZW50LCBmbXhUeXBlKTtcblxuICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteFR5cGUpO1xuXG4gICAgICAgICAgICB0aGlzLmZteFR5cGVNYXAuc2V0KHR5cGVOYW1lLCBmbXhUeXBlKTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGNvbnN0IHJlc3VsdCA9IHRoaXMuZm14VHlwZU1hcC5nZXQodHlwZU5hbWUpO1xuICAgICAgICAgICAgaWYgKHJlc3VsdCkge1xuICAgICAgICAgICAgICAgIGZteFR5cGUgPSByZXN1bHQ7XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgRmFtaXggdHlwZSAke3R5cGVOYW1lfSBpcyBub3QgZm91bmQgaW4gdGhlIFR5cGUgbWFwLmApO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhUeXBlLGVsZW1lbnQpO1xuXG4gICAgICAgIHJldHVybiBmbXhUeXBlO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBhY2Nlc3NcbiAgICAgKiBAcGFyYW0gbm9kZSBBIG5vZGVcbiAgICAgKiBAcGFyYW0gaWQgQW4gaWQgb2YgYSBwYXJhbWV0ZXIsIGEgdmFyaWFibGUsIGEgcHJvcGVydHkgb3IgYW4gZW51bSBtZW1iZXJcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhBY2Nlc3Mobm9kZTogSWRlbnRpZmllciwgaWQ6IG51bWJlcik6IHZvaWQge1xuICAgICAgICBjb25zdCBmbXhWYXIgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlJZChpZCkgYXMgRmFtaXguU3RydWN0dXJhbEVudGl0eTtcbiAgICAgICAgaWYgKCFmbXhWYXIpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgRmFtaXggZW50aXR5IHdpdGggaWQgJHtpZH0gbm90IGZvdW5kLCBmb3Igbm9kZSAke25vZGUuZ2V0VGV4dCgpfSBpbiAke25vZGUuZ2V0U291cmNlRmlsZSgpLmdldEJhc2VOYW1lKCl9IGF0IGxpbmUgJHtub2RlLmdldFN0YXJ0TGluZU51bWJlcigpfS5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgQ3JlYXRpbmcgRmFtaXhBY2Nlc3MuIE5vZGU6IFske25vZGUuZ2V0S2luZE5hbWUoKX1dICcke25vZGUuZ2V0VGV4dCgpfScgYXQgbGluZSAke25vZGUuZ2V0U3RhcnRMaW5lTnVtYmVyKCl9IGluICR7bm9kZS5nZXRTb3VyY2VGaWxlKCkuZ2V0QmFzZU5hbWUoKX0sIGlkOiAke2lkfSByZWZlcnMgdG8gZm14VmFyICcke2ZteFZhci5mdWxseVF1YWxpZmllZE5hbWV9Jy5gKTtcblxuICAgICAgICBjb25zdCBub2RlUmVmZXJlbmNlQW5jZXN0b3IgPSBIZWxwZXJzLmZpbmRBbmNlc3Rvcihub2RlKTtcbiAgICAgICAgY29uc3QgYW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKG5vZGVSZWZlcmVuY2VBbmNlc3Rvcik7XG4gICAgICAgIGxldCBhY2Nlc3NvciA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUZ1bGx5UXVhbGlmaWVkTmFtZShhbmNlc3RvckZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgRmFtaXguQ29udGFpbmVyRW50aXR5O1xuICAgICAgICBpZiAoIWFjY2Vzc29yKSB7XG4gICAgICAgICAgICBsb2dnZXIuZXJyb3IoYEFuY2VzdG9yICR7YW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWV9IG9mIGtpbmQgJHtub2RlUmVmZXJlbmNlQW5jZXN0b3IuZ2V0S2luZE5hbWUoKX0gbm90IGZvdW5kLmApO1xuICAgICAgICAgICAgLy8gYWNjZXNzb3IgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKGFuY2VzdG9yRnVsbHlRdWFsaWZpZWROYW1lLCBub2RlUmVmZXJlbmNlQW5jZXN0b3IgYXMgVHlwZURlY2xhcmF0aW9uKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteEFjY2VzcyA9IG5ldyBGYW1peC5BY2Nlc3MoKTtcbiAgICAgICAgZm14QWNjZXNzLmFjY2Vzc29yID0gYWNjZXNzb3I7XG4gICAgICAgIGZteEFjY2Vzcy52YXJpYWJsZSA9IGZteFZhcjtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14QWNjZXNzKTtcblxuICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEFjY2Vzcyxub2RlKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW52b2NhdGlvblxuICAgICAqIEBwYXJhbSBub2RlIEEgbm9kZVxuICAgICAqIEBwYXJhbSBtIEEgbWV0aG9kIG9yIGEgZnVuY3Rpb25cbiAgICAgKiBAcGFyYW0gaWQgVGhlIGlkIG9mIHRoZSBtZXRob2Qgb3IgdGhlIGZ1bmN0aW9uXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4SW52b2NhdGlvbihub2RlOiBJZGVudGlmaWVyLCBtOiBNZXRob2REZWNsYXJhdGlvbiB8IENvbnN0cnVjdG9yRGVjbGFyYXRpb24gfCBHZXRBY2Nlc3NvckRlY2xhcmF0aW9uIHwgU2V0QWNjZXNzb3JEZWNsYXJhdGlvbiB8IEZ1bmN0aW9uRGVjbGFyYXRpb24gfCBGdW5jdGlvbkV4cHJlc3Npb24sIGlkOiBudW1iZXIpOiB2b2lkIHtcbiAgICAgICAgY29uc3QgZm14TWV0aG9kT3JGdW5jdGlvbiA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUlkKGlkKSBhcyBGYW1peC5CZWhhdmlvcmFsRW50aXR5O1xuICAgICAgICBjb25zdCBub2RlUmVmZXJlbmNlQW5jZXN0b3IgPSBIZWxwZXJzLmZpbmRBbmNlc3Rvcihub2RlKTtcbiAgICAgICAgY29uc3QgYW5jZXN0b3JGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKG5vZGVSZWZlcmVuY2VBbmNlc3Rvcik7XG4gICAgICAgIGNvbnN0IHNlbmRlciA9IHRoaXMuZmFtaXhSZXAuZ2V0RmFtaXhFbnRpdHlCeUZ1bGx5UXVhbGlmaWVkTmFtZShhbmNlc3RvckZ1bGx5UXVhbGlmaWVkTmFtZSkgYXMgRmFtaXguQ29udGFpbmVyRW50aXR5O1xuICAgICAgICBjb25zdCByZWNlaXZlckZ1bGx5UXVhbGlmaWVkTmFtZSA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4obS5nZXRQYXJlbnQoKSk7XG4gICAgICAgIGNvbnN0IHJlY2VpdmVyID0gdGhpcy5mYW1peFJlcC5nZXRGYW1peEVudGl0eUJ5RnVsbHlRdWFsaWZpZWROYW1lKHJlY2VpdmVyRnVsbHlRdWFsaWZpZWROYW1lKSBhcyBGYW1peC5OYW1lZEVudGl0eTtcblxuICAgICAgICBjb25zdCBmbXhJbnZvY2F0aW9uID0gbmV3IEZhbWl4Lkludm9jYXRpb24oKTtcbiAgICAgICAgZm14SW52b2NhdGlvbi5zZW5kZXIgPSBzZW5kZXI7XG4gICAgICAgIGZteEludm9jYXRpb24ucmVjZWl2ZXIgPSByZWNlaXZlcjtcbiAgICAgICAgZm14SW52b2NhdGlvbi5hZGRDYW5kaWRhdGUoZm14TWV0aG9kT3JGdW5jdGlvbik7XG4gICAgICAgIGZteEludm9jYXRpb24uc2lnbmF0dXJlID0gZm14TWV0aG9kT3JGdW5jdGlvbi5zaWduYXR1cmU7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEludm9jYXRpb24pO1xuXG4gICAgICAgIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14SW52b2NhdGlvbixub2RlKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW5oZXJpdGFuY2VcbiAgICAgKiBAcGFyYW0gY2xzIEEgY2xhc3Mgb3IgYW4gaW50ZXJmYWNlIChzdWJjbGFzcylcbiAgICAgKiBAcGFyYW0gaW5oQ2xhc3MgVGhlIGluaGVyaXRlZCBjbGFzcyBvciBpbnRlcmZhY2UgKHN1cGVyY2xhc3MpXG4gICAgICovXG4gICAgcHVibGljIGNyZWF0ZUZhbWl4SW5oZXJpdGFuY2UoY2xzOiBDbGFzc0RlY2xhcmF0aW9uIHwgSW50ZXJmYWNlRGVjbGFyYXRpb24sIGluaENsYXNzOiBDbGFzc0RlY2xhcmF0aW9uIHwgSW50ZXJmYWNlRGVjbGFyYXRpb24gfCBFeHByZXNzaW9uV2l0aFR5cGVBcmd1bWVudHMpOiB2b2lkIHtcbiAgICAgICAgY29uc3QgZm14SW5oZXJpdGFuY2UgPSBuZXcgRmFtaXguSW5oZXJpdGFuY2UoKTtcbiAgICAgICAgLy8gY29uc3QgY2xzTmFtZSA9IGNscy5nZXROYW1lKCk7XG4gICAgICAgIGNvbnN0IGNsYXNzRnVsbHlRdWFsaWZpZWROYW1lID0gRlFORnVuY3Rpb25zLmdldEZRTihjbHMpO1xuICAgICAgICBsb2dnZXIuZGVidWcoYGNyZWF0ZUZhbWl4SW5oZXJpdGFuY2U6IGNsYXNzRnVsbHlRdWFsaWZpZWROYW1lOiBjbGFzcyBmcW4gPSAke2NsYXNzRnVsbHlRdWFsaWZpZWROYW1lfWApO1xuICAgICAgICBsZXQgc3ViQ2xhc3M6IEZhbWl4LkNsYXNzIHwgRmFtaXguSW50ZXJmYWNlIHwgdW5kZWZpbmVkO1xuICAgICAgICBpZiAoY2xzIGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgc3ViQ2xhc3MgPSB0aGlzLmZteENsYXNzTWFwLmdldChjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XG4gICAgICAgIH1cbiAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICBzdWJDbGFzcyA9IHRoaXMuZm14SW50ZXJmYWNlTWFwLmdldChjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKCFzdWJDbGFzcykge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBTdWJjbGFzcyAke2NsYXNzRnVsbHlRdWFsaWZpZWROYW1lfSBub3QgZm91bmQgaW4gQ2xhc3Mgb3IgSW50ZXJmYWNlIG1hcHMuYCk7XG4gICAgICAgIH1cbiAgICAgICAgXG4gICAgICAgIGxldCBpbmhDbGFzc05hbWU6IHN0cmluZyB8IHVuZGVmaW5lZDtcbiAgICAgICAgbGV0IGluaENsYXNzRnVsbHlRdWFsaWZpZWROYW1lOiBzdHJpbmc7XG4gICAgICAgIGxldCBzdXBlckNsYXNzOiBGYW1peC5DbGFzcyB8IEZhbWl4LkludGVyZmFjZSB8IHVuZGVmaW5lZDtcbiAgICAgICAgaWYgKGluaENsYXNzIGluc3RhbmNlb2YgQ2xhc3NEZWNsYXJhdGlvbiB8fCBpbmhDbGFzcyBpbnN0YW5jZW9mIEludGVyZmFjZURlY2xhcmF0aW9uKSB7XG4gICAgICAgICAgICBpbmhDbGFzc05hbWUgPSBpbmhDbGFzcy5nZXROYW1lKCk7XG4gICAgICAgICAgICBpZiAoIWluaENsYXNzTmFtZSkge1xuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgSW5oZXJpdGVkIGNsYXNzIG9yIGludGVyZmFjZSBuYW1lIG5vdCBmb3VuZCBmb3IgJHtpbmhDbGFzcy5nZXRUZXh0KCl9LmApO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGluaENsYXNzKTtcbiAgICAgICAgICAgIGlmIChpbmhDbGFzcyBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgICAgICBzdXBlckNsYXNzID0gdGhpcy5mbXhDbGFzc01hcC5nZXQoaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgc3VwZXJDbGFzcyA9IHRoaXMuZm14SW50ZXJmYWNlTWFwLmdldChpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBpZiAoIXN1cGVyQ2xhc3MpIHtcbiAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFN1cGVyY2xhc3MgJHtjbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZX0gbm90IGZvdW5kIGluIENsYXNzIG9yIEludGVyZmFjZSBtYXBzLmApO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgLy8gaW5oQ2xhc3MgaXMgYW4gRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzXG4gICAgICAgICAgICBpbmhDbGFzc05hbWUgPSBpbmhDbGFzcy5nZXRFeHByZXNzaW9uKCkuZ2V0VGV4dCgpO1xuICAgICAgICAgICAgLy8gd2hhdCBpcyBpbmhDbGFzc0Z1bGx5UXVhbGlmaWVkTmFtZT8gVE9ET1xuICAgICAgICAgICAgaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUgPSAnVW5kZWZpbmVkX1Njb3BlX2Zyb21faW1wb3J0ZXIuJyArIGluaENsYXNzTmFtZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChzdXBlckNsYXNzID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGlmIChpbmhDbGFzcyBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgICAgICBzdXBlckNsYXNzID0gbmV3IEZhbWl4LkNsYXNzKCk7XG4gICAgICAgICAgICAgICAgdGhpcy5mbXhDbGFzc01hcC5zZXQoaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUsIHN1cGVyQ2xhc3MpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgc3VwZXJDbGFzcyA9IG5ldyBGYW1peC5JbnRlcmZhY2UoKTtcbiAgICAgICAgICAgICAgICB0aGlzLmZteEludGVyZmFjZU1hcC5zZXQoaW5oQ2xhc3NGdWxseVF1YWxpZmllZE5hbWUsIHN1cGVyQ2xhc3MpO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KHN1cGVyQ2xhc3MsaW5oQ2xhc3MpO1xuXG4gICAgICAgICAgICBzdXBlckNsYXNzLm5hbWUgPSBpbmhDbGFzc05hbWU7XG4gICAgICAgICAgICBzdXBlckNsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZSA9IGluaENsYXNzRnVsbHlRdWFsaWZpZWROYW1lO1xuICAgICAgICAgICAgc3VwZXJDbGFzcy5pc1N0dWIgPSB0cnVlO1xuXG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbmhDbGFzcywgc3VwZXJDbGFzcyk7XG4gICAgICAgIFxuICAgICAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KHN1cGVyQ2xhc3MpO1xuICAgICAgICB9XG5cbiAgICAgICAgZm14SW5oZXJpdGFuY2Uuc3ViY2xhc3MgPSBzdWJDbGFzcztcbiAgICAgICAgZm14SW5oZXJpdGFuY2Uuc3VwZXJjbGFzcyA9IHN1cGVyQ2xhc3M7XG5cbiAgICAgICAgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGZteEluaGVyaXRhbmNlKTtcblxuICAgICAgICAvLyBXZSBkb24ndCBtYXAgaW5oZXJpdGFuY2UgdG8gdGhlIHNvdXJjZSBjb2RlIGVsZW1lbnQgYmVjYXVzZSB0aGVyZSBhcmUgdHdvIGVsZW1lbnRzIChzdXBlciwgc3ViKVxuICAgICAgICAvLyB0aGlzLmZteEVsZW1lbnRPYmplY3RNYXAuc2V0KGZteEluaGVyaXRhbmNlLCBudWxsKTtcblxuICAgIH1cblxuICAgIHB1YmxpYyBjcmVhdGVGYW1peEltcG9ydENsYXVzZShpbXBvcnRlZEVudGl0eTogRmFtaXguTmFtZWRFbnRpdHksIGltcG9ydGluZ0VudGl0eTogRmFtaXguTW9kdWxlKSB7XG4gICAgICAgIGNvbnN0IGZteEltcG9ydENsYXVzZSA9IG5ldyBGYW1peC5JbXBvcnRDbGF1c2UoKTtcbiAgICAgICAgZm14SW1wb3J0Q2xhdXNlLmltcG9ydGVkRW50aXR5ID0gaW1wb3J0ZWRFbnRpdHk7XG4gICAgICAgIGZteEltcG9ydENsYXVzZS5pbXBvcnRpbmdFbnRpdHkgPSBpbXBvcnRpbmdFbnRpdHk7XG4gICAgICAgIGltcG9ydGluZ0VudGl0eS5hZGRPdXRnb2luZ0ltcG9ydChmbXhJbXBvcnRDbGF1c2UpO1xuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW1wb3J0Q2xhdXNlKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggaW1wb3J0IGNsYXVzZVxuICAgICAqIEBwYXJhbSBpbXBvcnRDbGF1c2VJbmZvIFRoZSBpbmZvcm1hdGlvbiBuZWVkZWQgdG8gY3JlYXRlIGEgRmFtaXggaW1wb3J0IGNsYXVzZVxuICAgICAqIEBwYXJhbSBpbXBvcnREZWNsYXJhdGlvbiBUaGUgaW1wb3J0IGRlY2xhcmF0aW9uXG4gICAgICogQHBhcmFtIGltcG9ydGVyIEEgc291cmNlIGZpbGUgd2hpY2ggaXMgYSBtb2R1bGVcbiAgICAgKiBAcGFyYW0gbW9kdWxlU3BlY2lmaWVyRmlsZVBhdGggVGhlIHBhdGggb2YgdGhlIG1vZHVsZSB3aGVyZSB0aGUgZXhwb3J0IGRlY2xhcmF0aW9uIGlzXG4gICAgICogQHBhcmFtIGltcG9ydEVsZW1lbnQgVGhlIGltcG9ydGVkIGVudGl0eVxuICAgICAqIEBwYXJhbSBpc0luRXhwb3J0cyBBIGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgaW1wb3J0ZWQgZW50aXR5IGlzIGluIHRoZSBleHBvcnRzXG4gICAgICogQHBhcmFtIGlzRGVmYXVsdEV4cG9ydCBBIGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgaW1wb3J0ZWQgZW50aXR5IGlzIGEgZGVmYXVsdCBleHBvcnRcbiAgICAgKi9cbiAgICBwdWJsaWMgb2xkQ3JlYXRlRmFtaXhJbXBvcnRDbGF1c2UoaW1wb3J0Q2xhdXNlSW5mbzoge2ltcG9ydERlY2xhcmF0aW9uPzogSW1wb3J0RGVjbGFyYXRpb24gfCBJbXBvcnRFcXVhbHNEZWNsYXJhdGlvbiwgaW1wb3J0ZXJTb3VyY2VGaWxlOiBTb3VyY2VGaWxlLCBtb2R1bGVTcGVjaWZpZXJGaWxlUGF0aDogc3RyaW5nLCBpbXBvcnRFbGVtZW50OiBJbXBvcnRTcGVjaWZpZXIgfCBJZGVudGlmaWVyLCBpc0luRXhwb3J0czogYm9vbGVhbiwgaXNEZWZhdWx0RXhwb3J0OiBib29sZWFufSk6IHZvaWQge1xuICAgICAgICBjb25zdCB7aW1wb3J0RGVjbGFyYXRpb24sIGltcG9ydGVyU291cmNlRmlsZTogaW1wb3J0ZXIsIG1vZHVsZVNwZWNpZmllckZpbGVQYXRoLCBpbXBvcnRFbGVtZW50LCBpc0luRXhwb3J0cywgaXNEZWZhdWx0RXhwb3J0fSA9IGltcG9ydENsYXVzZUluZm87XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgY3JlYXRlRmFtaXhJbXBvcnRDbGF1c2U6IENyZWF0aW5nIGltcG9ydCBjbGF1c2U6YCk7XG4gICAgICAgIGNvbnN0IGZteEltcG9ydENsYXVzZSA9IG5ldyBGYW1peC5JbXBvcnRDbGF1c2UoKTtcblxuICAgICAgICBsZXQgaW1wb3J0ZWRFbnRpdHk6IEZhbWl4Lk5hbWVkRW50aXR5IHwgRmFtaXguU3RydWN0dXJhbEVudGl0eSB8IHVuZGVmaW5lZCA9IHVuZGVmaW5lZDtcbiAgICAgICAgbGV0IGltcG9ydGVkRW50aXR5TmFtZTogc3RyaW5nO1xuXG4gICAgICAgIGNvbnN0IGFic29sdXRlUGF0aFByb2plY3QgPSB0aGlzLmZhbWl4UmVwLmdldEFic29sdXRlUGF0aCgpO1xuICAgICAgICBcbiAgICAgICAgY29uc3QgYWJzb2x1dGVQYXRoID0gcGF0aC5ub3JtYWxpemUobW9kdWxlU3BlY2lmaWVyRmlsZVBhdGgpO1xuICAgICAgICAvLyBjb252ZXJ0IHRoZSBwYXRoIGFuZCByZW1vdmUgYW55IHdpbmRvd3MgYmFja3NsYXNoZXMgaW50cm9kdWNlZCBieSBwYXRoLm5vcm1hbGl6ZVxuICAgICAgICBjb25zdCBwYXRoSW5Qcm9qZWN0OiBzdHJpbmcgPSB0aGlzLmNvbnZlcnRUb1JlbGF0aXZlUGF0aChhYnNvbHV0ZVBhdGgsIGFic29sdXRlUGF0aFByb2plY3QpLnJlcGxhY2UoL1xcXFwvZywgXCIvXCIpO1xuICAgICAgICBsZXQgcGF0aE5hbWUgPSBcIntcIiArIHBhdGhJblByb2plY3QgKyBcIn0uXCI7XG5cbiAgICAgICAgLy8gTmFtZWQgaW1wb3J0cywgZS5nLiBpbXBvcnQgeyBDbGFzc1cgfSBmcm9tIFwiLi9jb21wbGV4RXhwb3J0TW9kdWxlXCI7XG5cbiAgICAgICAgLy8gU3RhcnQgd2l0aCBzaW1wbGUgaW1wb3J0IGNsYXVzZSAod2l0aG91dCByZWZlcnJpbmcgdG8gdGhlIGFjdHVhbCB2YXJpYWJsZSlcblxuICAgICAgICBpZiAoaW1wb3J0RGVjbGFyYXRpb24gaW5zdGFuY2VvZiBJbXBvcnREZWNsYXJhdGlvbiBcbiAgICAgICAgICAgICYmIGltcG9ydEVsZW1lbnQgaW5zdGFuY2VvZiBJbXBvcnRTcGVjaWZpZXIpIHsgXG4gICAgICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHlOYW1lID0gaW1wb3J0RWxlbWVudC5nZXROYW1lKCk7XG4gICAgICAgICAgICBwYXRoTmFtZSA9IHBhdGhOYW1lICsgaW1wb3J0ZWRFbnRpdHlOYW1lO1xuICAgICAgICAgICAgaWYgKGlzSW5FeHBvcnRzKSB7XG4gICAgICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUocGF0aE5hbWUpIGFzIEZhbWl4Lk5hbWVkRW50aXR5O1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaWYgKGltcG9ydGVkRW50aXR5ID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eSA9IG5ldyBGYW1peC5OYW1lZEVudGl0eSgpO1xuICAgICAgICAgICAgICAgIGltcG9ydGVkRW50aXR5Lm5hbWUgPSBpbXBvcnRlZEVudGl0eU5hbWU7XG4gICAgICAgICAgICAgICAgaWYgKCFpc0luRXhwb3J0cykge1xuICAgICAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5pc1N0dWIgPSB0cnVlO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPSBwYXRoTmFtZTtcbiAgICAgICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbXBvcnRFbGVtZW50LCBpbXBvcnRlZEVudGl0eSk7XG4gICAgICAgICAgICAgICAgLy8gbXVzdCBhZGQgZW50aXR5IHRvIHJlcG9zaXRvcnlcbiAgICAgICAgICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoaW1wb3J0ZWRFbnRpdHkpO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIC8vIGhhbmRsZSBpbXBvcnQgZXF1YWxzIGRlY2xhcmF0aW9ucywgZS5nLiBpbXBvcnQgbXlNb2R1bGUgPSByZXF1aXJlKFwiLi9jb21wbGV4RXhwb3J0TW9kdWxlXCIpO1xuICAgICAgICAvLyBUeXBlU2NyaXB0IGNhbid0IGRldGVybWluZSB0aGUgdHlwZSBvZiB0aGUgaW1wb3J0ZWQgbW9kdWxlLCBzbyB3ZSBjcmVhdGUgYSBNb2R1bGUgZW50aXR5XG4gICAgICAgIGVsc2UgaWYgKGltcG9ydERlY2xhcmF0aW9uIGluc3RhbmNlb2YgSW1wb3J0RXF1YWxzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgIGltcG9ydGVkRW50aXR5TmFtZSA9IGltcG9ydERlY2xhcmF0aW9uPy5nZXROYW1lKCk7XG4gICAgICAgICAgICBwYXRoTmFtZSA9IHBhdGhOYW1lICsgaW1wb3J0ZWRFbnRpdHlOYW1lO1xuICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkgPSBuZXcgRmFtaXguU3RydWN0dXJhbEVudGl0eSgpO1xuICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkubmFtZSA9IGltcG9ydGVkRW50aXR5TmFtZTtcbiAgICAgICAgICAgIGluaXRGUU4oaW1wb3J0RGVjbGFyYXRpb24sIGltcG9ydGVkRW50aXR5KTtcbiAgICAgICAgICAgIHRoaXMubWFrZUZhbWl4SW5kZXhGaWxlQW5jaG9yKGltcG9ydEVsZW1lbnQsIGltcG9ydGVkRW50aXR5KTtcbiAgICAgICAgICAgIGltcG9ydGVkRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9IHBhdGhOYW1lO1xuICAgICAgICAgICAgY29uc3QgYW55VHlwZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peFR5cGUoJ2FueScsIGltcG9ydERlY2xhcmF0aW9uKTtcbiAgICAgICAgICAgIChpbXBvcnRlZEVudGl0eSBhcyBGYW1peC5TdHJ1Y3R1cmFsRW50aXR5KS5kZWNsYXJlZFR5cGUgPSBhbnlUeXBlO1xuICAgICAgICB9IGVsc2UgeyAgLy8gZGVmYXVsdCBpbXBvcnRzLCBlLmcuIGltcG9ydCBDbGFzc1cgZnJvbSBcIi4vY29tcGxleEV4cG9ydE1vZHVsZVwiOyAgXG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eU5hbWUgPSBpbXBvcnRFbGVtZW50LmdldFRleHQoKTtcbiAgICAgICAgICAgIHBhdGhOYW1lID0gcGF0aE5hbWUgKyAoaXNEZWZhdWx0RXhwb3J0ID8gXCJkZWZhdWx0RXhwb3J0XCIgOiBcIm5hbWVzcGFjZUV4cG9ydFwiKTtcbiAgICAgICAgICAgIGltcG9ydGVkRW50aXR5ID0gbmV3IEZhbWl4Lk5hbWVkRW50aXR5KCk7XG4gICAgICAgICAgICBpbXBvcnRlZEVudGl0eS5uYW1lID0gaW1wb3J0ZWRFbnRpdHlOYW1lO1xuICAgICAgICAgICAgaW1wb3J0ZWRFbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lID0gcGF0aE5hbWU7XG4gICAgICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihpbXBvcnRFbGVtZW50LCBpbXBvcnRlZEVudGl0eSk7XG4gICAgICAgIH1cbiAgICAgICAgLy8gSSBkb24ndCB0aGluayBpdCBzaG91bGQgYmUgYWRkZWQgdG8gdGhlIHJlcG9zaXRvcnkgaWYgaXQgZXhpc3RzIGFscmVhZHlcbiAgICAgICAgaWYgKCFpc0luRXhwb3J0cykgdGhpcy5mYW1peFJlcC5hZGRFbGVtZW50KGltcG9ydGVkRW50aXR5KTtcbiAgICAgICAgY29uc3QgaW1wb3J0ZXJGdWxseVF1YWxpZmllZE5hbWUgPSBGUU5GdW5jdGlvbnMuZ2V0RlFOKGltcG9ydGVyKTtcbiAgICAgICAgY29uc3QgZm14SW1wb3J0ZXIgPSB0aGlzLmZhbWl4UmVwLmdldEZhbWl4RW50aXR5QnlGdWxseVF1YWxpZmllZE5hbWUoaW1wb3J0ZXJGdWxseVF1YWxpZmllZE5hbWUpIGFzIEZhbWl4Lk1vZHVsZTtcbiAgICAgICAgZm14SW1wb3J0Q2xhdXNlLmltcG9ydGluZ0VudGl0eSA9IGZteEltcG9ydGVyO1xuICAgICAgICBmbXhJbXBvcnRDbGF1c2UuaW1wb3J0ZWRFbnRpdHkgPSBpbXBvcnRlZEVudGl0eTtcbiAgICAgICAgaWYgKGltcG9ydERlY2xhcmF0aW9uIGluc3RhbmNlb2YgSW1wb3J0RXF1YWxzRGVjbGFyYXRpb24pIHtcbiAgICAgICAgICAgIGZteEltcG9ydENsYXVzZS5tb2R1bGVTcGVjaWZpZXIgPSBpbXBvcnREZWNsYXJhdGlvbj8uZ2V0TW9kdWxlUmVmZXJlbmNlKCkuZ2V0VGV4dCgpIGFzIHN0cmluZztcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGZteEltcG9ydENsYXVzZS5tb2R1bGVTcGVjaWZpZXIgPSBpbXBvcnREZWNsYXJhdGlvbj8uZ2V0TW9kdWxlU3BlY2lmaWVyVmFsdWUoKSBhcyBzdHJpbmc7XG4gICAgICAgIH1cbiAgICBcbiAgICAgICAgbG9nZ2VyLmRlYnVnKGBjcmVhdGVGYW1peEltcG9ydENsYXVzZTogJHtmbXhJbXBvcnRDbGF1c2UuaW1wb3J0ZWRFbnRpdHk/Lm5hbWV9IChvZiB0eXBlICR7XG4gICAgICAgICAgICBIZWxwZXJzLmdldFN1YlR5cGVOYW1lKGZteEltcG9ydENsYXVzZS5pbXBvcnRlZEVudGl0eSl9KSBpcyBpbXBvcnRlZCBieSAke2ZteEltcG9ydENsYXVzZS5pbXBvcnRpbmdFbnRpdHk/Lm5hbWV9YCk7XG5cbiAgICAgICAgZm14SW1wb3J0ZXIuYWRkT3V0Z29pbmdJbXBvcnQoZm14SW1wb3J0Q2xhdXNlKTtcblxuICAgICAgICB0aGlzLmZhbWl4UmVwLmFkZEVsZW1lbnQoZm14SW1wb3J0Q2xhdXNlKTtcblxuICAgICAgICBpZiAoaW1wb3J0RGVjbGFyYXRpb24pIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14SW1wb3J0Q2xhdXNlLCBpbXBvcnREZWNsYXJhdGlvbik7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IEFycm93IEZ1bmN0aW9uXG4gICAgICogQHBhcmFtIGFycm93RXhwcmVzc2lvbiBBbiBFeHByZXNzaW9uXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSB2YXJpYWJsZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peEFycm93RnVuY3Rpb24oYXJyb3dFeHByZXNzaW9uOiBFeHByZXNzaW9uLCBjdXJyZW50Q0M6IHsgW2tleTogc3RyaW5nXTogbnVtYmVyIH0gKTogRmFtaXguQXJyb3dGdW5jdGlvbiB8IEZhbWl4LlBhcmFtZXRyaWNBcnJvd0Z1bmN0aW9uIHtcbiAgICAgICAgXG4gICAgICAgIGxldCBmbXhBcnJvd0Z1bmN0aW9uOiBGYW1peC5BcnJvd0Z1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY0Fycm93RnVuY3Rpb247XG5cbiAgICAgICAgY29uc3QgYXJyb3dGdW5jdGlvbiA9IGFycm93RXhwcmVzc2lvbi5hc0tpbmRPclRocm93KFN5bnRheEtpbmQuQXJyb3dGdW5jdGlvbik7XG5cbiAgICAgICAgY29uc3QgaXNHZW5lcmljID0gYXJyb3dGdW5jdGlvbi5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XG5cbiAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgZm14QXJyb3dGdW5jdGlvbiA9IG5ldyBGYW1peC5QYXJhbWV0cmljQXJyb3dGdW5jdGlvbigpO1xuICAgICAgICB9XG4gICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgZm14QXJyb3dGdW5jdGlvbiA9IG5ldyBGYW1peC5BcnJvd0Z1bmN0aW9uKCk7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBHZXQgdGhlIHBhcmVudCBvZiB0aGUgYXJyb3cgZnVuY3Rpb24gKHRoZSB2YXJpYWJsZSBkZWNsYXJhdGlvbilcbiAgICAgICAgY29uc3QgcGFyZW50ID0gYXJyb3dGdW5jdGlvbi5nZXRQYXJlbnRJZktpbmQoU3ludGF4S2luZC5WYXJpYWJsZURlY2xhcmF0aW9uKTtcbiAgICAgICAgbGV0IGZ1bmN0aW9uTmFtZSA9ICcoTk9fTkFNRSknO1xuXG4gICAgICAgIGlmIChwYXJlbnQgJiYgcGFyZW50IGluc3RhbmNlb2YgVmFyaWFibGVEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgLy8gR2V0IHRoZSBuYW1lIG9mIHRoZSB2YXJpYWJsZVxuICAgICAgICAgICAgZnVuY3Rpb25OYW1lID0gcGFyZW50LmdldE5hbWUoKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChmdW5jdGlvbk5hbWUpIHtcbiAgICAgICAgICAgIGZteEFycm93RnVuY3Rpb24ubmFtZSA9IGZ1bmN0aW9uTmFtZTtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIGZteEFycm93RnVuY3Rpb24ubmFtZSA9IFwiYW5vbnltb3VzXCI7XG4gICAgICAgIH1cblxuICAgICAgICAvLyBTaWduYXR1cmUgb2YgYW4gYXJyb3cgZnVuY3Rpb24gaXMgKHBhcmFtZXRlcnMpID0+IHJldHVybl90eXBlXG4gICAgICAgIGNvbnN0IHBhcmFtZXRlcnNTaWduYXR1cmUgPSBhcnJvd0Z1bmN0aW9uLmdldFBhcmFtZXRlcnMoKS5tYXAocCA9PiBwLmdldFRleHQoKSkuam9pbihcIiwgXCIpO1xuICAgICAgICBjb25zdCByZXR1cm5UeXBlU2lnbmF0dXJlID0gYXJyb3dGdW5jdGlvbi5nZXRSZXR1cm5UeXBlKCkuZ2V0VGV4dCgpO1xuICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLnNpZ25hdHVyZSA9IGAoJHtwYXJhbWV0ZXJzU2lnbmF0dXJlfSkgPT4gJHtyZXR1cm5UeXBlU2lnbmF0dXJlfWA7XG4gICAgICAgIGZteEFycm93RnVuY3Rpb24uY3ljbG9tYXRpY0NvbXBsZXhpdHkgPSBjdXJyZW50Q0NbZm14QXJyb3dGdW5jdGlvbi5uYW1lXTtcblxuICAgICAgICBsZXQgZnVuY3Rpb25UeXBlTmFtZSA9IHRoaXMuVU5LTk9XTl9WQUxVRTtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGZ1bmN0aW9uVHlwZU5hbWUgPSBhcnJvd0Z1bmN0aW9uLmdldFJldHVyblR5cGUoKS5nZXRUZXh0KCkudHJpbSgpO1xuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGA+IFdBUk5JTkc6IGdvdCBleGNlcHRpb24gJHtlcnJvcn0uIEZhaWxlZCB0byBnZXQgdXNhYmxlIG5hbWUgZm9yIHJldHVybiB0eXBlIG9mIGZ1bmN0aW9uOiAke2Z1bmN0aW9uTmFtZX0uIENvbnRpbnVpbmcuLi5gKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZteFR5cGUgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhUeXBlKGZ1bmN0aW9uVHlwZU5hbWUsIGFycm93RnVuY3Rpb24gYXMgdW5rbm93biBhcyBGdW5jdGlvbkRlY2xhcmF0aW9uKTtcbiAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5kZWNsYXJlZFR5cGUgPSBmbXhUeXBlO1xuICAgICAgICBmbXhBcnJvd0Z1bmN0aW9uLm51bWJlck9mTGluZXNPZkNvZGUgPSBhcnJvd0Z1bmN0aW9uLmdldEVuZExpbmVOdW1iZXIoKSAtIGFycm93RnVuY3Rpb24uZ2V0U3RhcnRMaW5lTnVtYmVyKCk7XG4gICAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPSBhcnJvd0Z1bmN0aW9uLmdldFBhcmFtZXRlcnMoKTtcbiAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5udW1iZXJPZlBhcmFtZXRlcnMgPSBwYXJhbWV0ZXJzLmxlbmd0aDtcbiAgICAgICAgZm14QXJyb3dGdW5jdGlvbi5udW1iZXJPZlN0YXRlbWVudHMgPSBhcnJvd0Z1bmN0aW9uLmdldFN0YXRlbWVudHMoKS5sZW5ndGg7XG4gICAgICAgIGluaXRGUU4oYXJyb3dFeHByZXNzaW9uIGFzIHVua25vd24gYXMgVFNNb3JwaE9iamVjdFR5cGUsIGZteEFycm93RnVuY3Rpb24pO1xuICAgICAgICB0aGlzLm1ha2VGYW1peEluZGV4RmlsZUFuY2hvcihhcnJvd0V4cHJlc3Npb24gYXMgdW5rbm93biBhcyBUU01vcnBoT2JqZWN0VHlwZSwgZm14QXJyb3dGdW5jdGlvbik7XG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhBcnJvd0Z1bmN0aW9uKTtcbiAgICAgICAgdGhpcy5mbXhFbGVtZW50T2JqZWN0TWFwLnNldChmbXhBcnJvd0Z1bmN0aW9uLGFycm93RnVuY3Rpb24gYXMgdW5rbm93biBhcyBUU01vcnBoT2JqZWN0VHlwZSk7XG5cbiAgICAgICAgcmV0dXJuIGZteEFycm93RnVuY3Rpb247XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQ3JlYXRlcyBhIEZhbWl4IGNvbmNyZXRpc2F0aW9uXG4gICAgICogQHBhcmFtIGNscyBBIGNsYXNzXG4gICAgICogQHJldHVybnMgVGhlIEZhbWl4IG1vZGVsIG9mIHRoZSBjb25jcmV0aXNhdGlvblxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uKGNvbkVudGl0eSA6IEZhbWl4LlBhcmFtZXRyaWNDbGFzcyB8IEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2UgfCBGYW1peC5QYXJhbWV0cmljRnVuY3Rpb24gfCBGYW1peC5QYXJhbWV0cmljTWV0aG9kICxnZW5FbnRpdHkgOiBGYW1peC5QYXJhbWV0cmljQ2xhc3MgfCBGYW1peC5QYXJhbWV0cmljSW50ZXJmYWNlIHwgRmFtaXguUGFyYW1ldHJpY0Z1bmN0aW9uIHwgRmFtaXguUGFyYW1ldHJpY01ldGhvZCk6IEZhbWl4LkNvbmNyZXRpc2F0aW9uIHtcbiAgICAgICAgXG4gICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSBuZXcgRmFtaXguQ29uY3JldGlzYXRpb24oKTsgICAgICAgICAgICAgIFxuICAgICAgICBcbiAgICAgICAgZm14Q29uY3JldGlzYXRpb24uY29uY3JldGVFbnRpdHkgPSBjb25FbnRpdHk7XG4gICAgICAgIGZteENvbmNyZXRpc2F0aW9uLmdlbmVyaWNFbnRpdHkgPSBnZW5FbnRpdHk7XG4gICAgICAgIC8vIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14Q29uY3JldGlzYXRpb24sbnVsbCk7XG4gICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhDb25jcmV0aXNhdGlvbik7ICAgIFxuICAgICAgICBjb25zdCBwYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbihmbXhDb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICBcbiAgICAgICAgcmV0dXJuIGZteENvbmNyZXRpc2F0aW9uO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvblxuICAgICAqIEBwYXJhbSBjb25jcmV0aXNhdGlvbiBBIEZhbWl4Q29uY3JldGlzYXRpb25cbiAgICAgKiBAcmV0dXJucyBUaGUgRmFtaXggbW9kZWwgb2YgdGhlIFBhcmFtZXRlckNvbmNyZXN0aXNhdGlvblxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKGNvbmNyZXRpc2F0aW9uOiBGYW1peC5Db25jcmV0aXNhdGlvbik6IEZhbWl4LlBhcmFtZXRlckNvbmNyZXRpc2F0aW9uIHwgdW5kZWZpbmVke1xuICAgICAgICBjb25zdCBjb25DbGFzcyA9IGNvbmNyZXRpc2F0aW9uLmNvbmNyZXRlRW50aXR5O1xuICAgICAgICBjb25zdCBnZW5DbGFzcyA9IGNvbmNyZXRpc2F0aW9uLmdlbmVyaWNFbnRpdHk7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhgQ3JlYXRpbmcgcGFyYW1ldGVyIGNvbmNyZXRpc2F0aW9uIGJldHdlZW4gJHtjb25DbGFzcy5mdWxseVF1YWxpZmllZE5hbWV9IGFuZCAke2dlbkNsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZX1gKTtcbiAgICAgICAgY29uc3QgcGFyYW1ldGVyQ29uY3JldGlzYXRpb25zID0gdGhpcy5mYW1peFJlcC5fZ2V0QWxsRW50aXRpZXNXaXRoVHlwZShcIlBhcmFtZXRlckNvbmNyZXRpc2F0aW9uXCIpIGFzIFNldDxGYW1peC5QYXJhbWV0ZXJDb25jcmV0aXNhdGlvbj47XG4gICAgICAgIGNvbnN0IGNvbmNyZXRlUGFyYW1ldGVycyA9IGNvbkNsYXNzLmNvbmNyZXRlUGFyYW1ldGVycztcbiAgICAgICAgY29uc3QgZ2VuZXJpY1BhcmFtZXRlcnMgPSBnZW5DbGFzcy5nZW5lcmljUGFyYW1ldGVycztcbiAgICAgICAgXG4gICAgICAgIGxldCBjb25DbGFzc1R5cGVQYXJhbWV0ZXJzSXRlcmF0b3IgPSBjb25jcmV0ZVBhcmFtZXRlcnMudmFsdWVzKCk7XG4gICAgICAgIGxldCBnZW5DbGFzc1R5cGVQYXJhbWV0ZXJzSXRlcmF0b3IgPSBnZW5lcmljUGFyYW1ldGVycy52YWx1ZXMoKTtcbiAgICAgICAgbGV0IGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uIDogRmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24gfCB1bmRlZmluZWQgPSB1bmRlZmluZWQ7XG5cbiAgICAgICAgZm9yIChsZXQgaSA9IDA7IGkgPCBnZW5lcmljUGFyYW1ldGVycy5zaXplOyBpKyspIHtcbiAgICAgICAgICAgIGNvbnN0IGNvbkNsYXNzVHlwZVBhcmFtZXRlciA9IGNvbkNsYXNzVHlwZVBhcmFtZXRlcnNJdGVyYXRvci5uZXh0KCkudmFsdWUgYXMgRmFtaXguUGFyYW1ldGVyVHlwZTtcbiAgICAgICAgICAgIGNvbnN0IGdlbkNsYXNzVHlwZVBhcmFtZXRlciA9IGdlbkNsYXNzVHlwZVBhcmFtZXRlcnNJdGVyYXRvci5uZXh0KCkudmFsdWUgYXMgRmFtaXguUGFyYW1ldGVyVHlwZTtcbiAgICAgICAgICAgIGxldCBjcmVhdGVQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xuICAgICAgICAgICAgaWYoY29uQ2xhc3NUeXBlUGFyYW1ldGVyICYmIGdlbkNsYXNzVHlwZVBhcmFtZXRlciAmJiBjb25DbGFzc1R5cGVQYXJhbWV0ZXIubmFtZSAhPSBnZW5DbGFzc1R5cGVQYXJhbWV0ZXIubmFtZSl7XG4gICAgICAgICAgICAgICAgcGFyYW1ldGVyQ29uY3JldGlzYXRpb25zLmZvckVhY2goKHBhcmFtIDogRmFtaXguUGFyYW1ldGVyQ29uY3JldGlzYXRpb24pID0+IHtcbiAgICAgICAgICAgICAgICAgICAgaWYgKGNvbkNsYXNzVHlwZVBhcmFtZXRlci5uYW1lID09IHBhcmFtLmNvbmNyZXRlUGFyYW1ldGVyLm5hbWUgJiYgZ2VuQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWUgPT0gcGFyYW0uZ2VuZXJpY1BhcmFtZXRlci5uYW1lKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjcmVhdGVQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICAgICAgZm14UGFyYW1ldGVyQ29uY3JldGlzYXRpb24gPSBwYXJhbTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgaWYgKGNyZWF0ZVBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKSB7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uID0gbmV3IEZhbWl4LlBhcmFtZXRlckNvbmNyZXRpc2F0aW9uKCk7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLmdlbmVyaWNQYXJhbWV0ZXIgPSBnZW5DbGFzc1R5cGVQYXJhbWV0ZXI7XG4gICAgICAgICAgICAgICAgICAgIGZteFBhcmFtZXRlckNvbmNyZXRpc2F0aW9uLmNvbmNyZXRlUGFyYW1ldGVyID0gY29uQ2xhc3NUeXBlUGFyYW1ldGVyO1xuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbi5hZGRDb25jcmV0aXNhdGlvbihjb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICAgICAgICAgIC8vIHRoaXMuZm14RWxlbWVudE9iamVjdE1hcC5zZXQoZm14UGFyYW1ldGVyQ29uY3JldGlzYXRpb24sbnVsbCk7XG4gICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgaWYgKCFmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiB3YXMgdW5kZWZpbmVkIGZvciBjb25jcmV0aXNhdGlvbiB3aXRoIGdlbmVyaWMgcGFyYW1ldGVyICR7Z2VuQ2xhc3NUeXBlUGFyYW1ldGVyLm5hbWV9IGFuZCBjb25jcmV0ZSBwYXJhbWV0ZXIgJHtjb25DbGFzc1R5cGVQYXJhbWV0ZXIubmFtZX1gKTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbi5hZGRDb25jcmV0aXNhdGlvbihjb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIHRoaXMuZmFtaXhSZXAuYWRkRWxlbWVudChmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbik7XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgICAgaWYgKCFmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbiB3YXMgdW5kZWZpbmVkIGZvciBjb25jcmV0aXNhdGlvbiB3aXRoIGNvbmNyZXRlIGVudGl0eSAke2NvbkNsYXNzLmZ1bGx5UXVhbGlmaWVkTmFtZX0gYW5kIGdlbmVyaWMgZW50aXR5ICR7Z2VuQ2xhc3MuZnVsbHlRdWFsaWZpZWROYW1lfWApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBmbXhQYXJhbWV0ZXJDb25jcmV0aXNhdGlvbjtcblxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIHR3byBjbGFzc2VzIG9yIHR3byBpbnRlcmZhY2VzXG4gICAgICogQHBhcmFtIGVsZW1lbnQgQSBjbGFzcyBvciBhbiBJbnRlcmZhY2VcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkNsYXNzT3JJbnRlcmZhY2VTcGVjaWFsaXNhdGlvbihlbGVtZW50OiBDbGFzc0RlY2xhcmF0aW9uIHwgSW50ZXJmYWNlRGVjbGFyYXRpb24pe1xuICAgICAgICBcbiAgICAgICAgY29uc3Qgc3VwZXJFbnRpdHkgPSBlbGVtZW50LmdldEV4dGVuZHMoKTtcbiAgICAgICAgbGV0IHN1cGVyRW50aXR5QXJyYXk7XG4gICAgICAgIGlmIChzdXBlckVudGl0eSl7XG4gICAgICAgICAgICBzdXBlckVudGl0eUFycmF5ID0gQXJyYXkuaXNBcnJheShzdXBlckVudGl0eSkgPyBzdXBlckVudGl0eSA6IFtzdXBlckVudGl0eV07XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHN1cGVyRW50aXR5QXJyYXkgJiYgc3VwZXJFbnRpdHlBcnJheS5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICBzdXBlckVudGl0eUFycmF5LmZvckVhY2goZW50aXR5ID0+IHtcbiAgICAgICAgICAgICAgICBsZXQgZW50aXR5SXNHZW5lcmljO1xuICAgICAgICAgICAgICAgIGNvbnN0IHN1cGVyRW50aXR5U3ltYm9sID0gZW50aXR5LmdldEV4cHJlc3Npb24oKS5nZXRTeW1ib2xPclRocm93KCk7XG4gICAgICAgICAgICAgICAgbGV0IHN1cGVyRW50aXR5RGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgaWYgKHN1cGVyRW50aXR5IGluc3RhbmNlb2YgRXhwcmVzc2lvbldpdGhUeXBlQXJndW1lbnRzKSB7XG4gICAgICAgICAgICAgICAgICAgIHN1cGVyRW50aXR5RGVjbGFyYXRpb24gPSBzdXBlckVudGl0eVN5bWJvbC5nZXREZWNsYXJhdGlvbnMoKVswXS5hc0tpbmQodHMuU3ludGF4S2luZC5DbGFzc0RlY2xhcmF0aW9uKTtcbiAgICAgICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBzdXBlckVudGl0eURlY2xhcmF0aW9uID0gc3VwZXJFbnRpdHlTeW1ib2wuZ2V0RGVjbGFyYXRpb25zKClbMF0uYXNLaW5kKHRzLlN5bnRheEtpbmQuSW50ZXJmYWNlRGVjbGFyYXRpb24pO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBpZiAoc3VwZXJFbnRpdHlEZWNsYXJhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICBlbnRpdHlJc0dlbmVyaWMgPSBzdXBlckVudGl0eURlY2xhcmF0aW9uLmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgaWYgKGVudGl0eUlzR2VuZXJpYykge1xuICAgICAgICAgICAgICAgICAgICBsZXQgRW50aXR5RGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgICAgIGxldCBnZW5FbnRpdHk7XG4gICAgICAgICAgICAgICAgICAgIGlmIChzdXBlckVudGl0eSBpbnN0YW5jZW9mIEV4cHJlc3Npb25XaXRoVHlwZUFyZ3VtZW50cykge1xuICAgICAgICAgICAgICAgICAgICAgICAgRW50aXR5RGVjbGFyYXRpb24gPSBlbnRpdHkuZ2V0RXhwcmVzc2lvbigpLmdldFN5bWJvbCgpPy5nZXREZWNsYXJhdGlvbnMoKVswXSBhcyBDbGFzc0RlY2xhcmF0aW9uO1xuICAgICAgICAgICAgICAgICAgICAgICAgZ2VuRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q2xhc3MoRW50aXR5RGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcbiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIEVudGl0eURlY2xhcmF0aW9uID0gZW50aXR5LmdldEV4cHJlc3Npb24oKS5nZXRTeW1ib2woKT8uZ2V0RGVjbGFyYXRpb25zKClbMF0gYXMgSW50ZXJmYWNlRGVjbGFyYXRpb247XG4gICAgICAgICAgICAgICAgICAgICAgICBnZW5FbnRpdHkgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoRW50aXR5RGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gRW50aXR5RGVjbGFyYXRpb24uZ2V0VHlwZVBhcmFtZXRlcnMoKS5tYXAoKHBhcmFtKSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBhcmdzID0gZWxlbWVudC5nZXRIZXJpdGFnZUNsYXVzZXMoKVswXS5nZXRUeXBlTm9kZXMoKVswXS5nZXRUeXBlQXJndW1lbnRzKClcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gZWxlbWVudC5nZXRIZXJpdGFnZUNsYXVzZXMoKVswXS5nZXRUeXBlTm9kZXMoKVswXS5nZXRUeXBlQXJndW1lbnRzKCkubWFwKChwYXJhbSkgPT4gcGFyYW0uZ2V0VGV4dCgpKTtcbiAgICAgICAgICAgICAgICAgICAgaWYgKCFIZWxwZXJzLmFycmF5c0FyZUVxdWFsKGNvblBhcmFtcyxnZW5QYXJhbXMpKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBsZXQgY29uRW50aXR5O1xuICAgICAgICAgICAgICAgICAgICAgICAgY29uRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q29uY3JldGVFbGVtZW50KGdlbkVudGl0eSxFbnRpdHlEZWNsYXJhdGlvbixhcmdzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xuICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNyZWF0ZUNvbmNyZXRpc2F0aW9uIDogYm9vbGVhbiA9IHRydWU7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0aXNhdGlvbnMuZm9yRWFjaCgoY29uYyA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25FbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lKXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY3JlYXRlQ29uY3JldGlzYXRpb24gPSBmYWxzZTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIFxuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGNyZWF0ZUNvbmNyZXRpc2F0aW9uKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZm14Q29uY3JldGlzYXRpb24gOiBGYW1peC5Db25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbihjb25FbnRpdHksZ2VuRW50aXR5KTtcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICAgIC8vIFRPRE86IFRoaXMgZnVuY3Rpb24gc2VlbXMgdW5maW5pc2hlZFxuICAgIH0gICAgXG4gICAgXG5cbiAgICAvKipcbiAgICAgKiBDcmVhdGVzIGEgRmFtaXggY29uY3JldGlzYXRpb24gYmV0d2VlbiBhIGNsYXNzIGFuZCBpdHMgaW5zdGFuY2lhdGlvbnNcbiAgICAgKiBAcGFyYW0gY2xzIEEgY2xhc3NcbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkdlbmVyaWNJbnN0YW50aWF0aW9uKGNsczogQ2xhc3NEZWNsYXJhdGlvbil7XG4gICAgICAgXG4gICAgICAgIGNvbnN0IGlzR2VuZXJpYyA9IGNscy5nZXRUeXBlUGFyYW1ldGVycygpLmxlbmd0aCA+IDA7XG4gICAgICAgIGlmIChpc0dlbmVyaWMpIHtcbiAgICAgICAgICAgIGNvbnN0IGluc3RhbmNlcyA9IGNscy5nZXRTb3VyY2VGaWxlKCkuZ2V0RGVzY2VuZGFudHNPZktpbmQodHMuU3ludGF4S2luZC5OZXdFeHByZXNzaW9uKVxuICAgICAgICAgICAgICAgIC5maWx0ZXIobmV3RXhwciA9PiB7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGV4cHJlc3Npb24gPSBuZXdFeHByLmdldEV4cHJlc3Npb24oKTtcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuIGV4cHJlc3Npb24uZ2V0VGV4dCgpID09PSBjbHMuZ2V0TmFtZSgpO1xuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIGluc3RhbmNlcy5mb3JFYWNoKGluc3RhbmNlID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCBpbnN0YW5jZUlzR2VuZXJpYyA9IGluc3RhbmNlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGggPiAwO1xuICAgICAgICAgICAgICAgIGlmIChpbnN0YW5jZUlzR2VuZXJpYykge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBjb25QYXJhbXMgPSBpbnN0YW5jZS5nZXRUeXBlQXJndW1lbnRzKCkubWFwKChwYXJhbSkgPT4gcGFyYW0uZ2V0VGV4dCgpKTtcbiAgICAgICAgICAgICAgICAgICAgY29uc3QgZ2VuRW50aXR5ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q2xhc3MoY2xzKSBhcyBGYW1peC5QYXJhbWV0cmljQ2xhc3M7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGdlblBhcmFtcyA9IGNscy5nZXRUeXBlUGFyYW1ldGVycygpLm1hcCgocGFyYW0pID0+IHBhcmFtLmdldFRleHQoKSk7XG4gICAgICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNvbkVudGl0eTtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbkVudGl0eSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChnZW5FbnRpdHksY2xzLGluc3RhbmNlLmdldFR5cGVBcmd1bWVudHMoKSk7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjb25jcmV0aXNhdGlvbnMgPSB0aGlzLmZhbWl4UmVwLl9nZXRBbGxFbnRpdGllc1dpdGhUeXBlKFwiQ29uY3JldGlzYXRpb25cIikgYXMgU2V0PEZhbWl4LkNvbmNyZXRpc2F0aW9uPjtcbiAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjcmVhdGVDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICAgICAgY29uY3JldGlzYXRpb25zLmZvckVhY2goKGNvbmMgOiBGYW1peC5Db25jcmV0aXNhdGlvbikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChnZW5FbnRpdHkuZnVsbHlRdWFsaWZpZWROYW1lID09IGNvbmMuZ2VuZXJpY0VudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgJiYgY29uYy5jb25jcmV0ZUVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSl7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbmNyZXRpc2F0aW9uID0gZmFsc2U7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICBcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjcmVhdGVDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSB0aGlzLmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb24oY29uRW50aXR5LGdlbkVudGl0eSk7XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KVxuICAgICAgICB9XG4gICAgICAgIC8vIFRPRE86IFRoaXMgZnVuY3Rpb24gc2VlbXMgdW5maW5pc2hlZFxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGEgY2xhc3MgYW5kIGl0cyBpbnN0YW5jaWF0aW9uc1xuICAgICAqIEBwYXJhbSBmdW5jIEEgZnVuY3Rpb25cbiAgICAgKi9cbiAgICBwdWJsaWMgY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbkZ1bmN0aW9uSW5zdGFudGlhdGlvbihlbGVtZW50OiBGdW5jdGlvbkRlY2xhcmF0aW9uIHwgTWV0aG9kRGVjbGFyYXRpb24pe1xuICAgICAgICBjb25zdCBpc0dlbmVyaWMgPSBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gZWxlbWVudC5nZXRUeXBlUGFyYW1ldGVycygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgY29uc3QgdXNlcyA9IGVsZW1lbnQuZmluZFJlZmVyZW5jZXNBc05vZGVzKCk7ICAgIFxuICAgICAgICAgICAgdXNlcy5mb3JFYWNoKHVzYWdlID0+IHtcbiAgICAgICAgICAgICAgICBsZXQgY3VycmVudE5vZGU6IE5vZGUgfCB1bmRlZmluZWQgPSB1c2FnZTtcblxuICAgICAgICAgICAgICAgIHdoaWxlIChjdXJyZW50Tm9kZSkge1xuICAgICAgICAgICAgICAgICAgICBpZiAoY3VycmVudE5vZGUuZ2V0S2luZCgpID09PSBTeW50YXhLaW5kLkNhbGxFeHByZXNzaW9uKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjYWxsRXhwcmVzc2lvbiA9IGN1cnJlbnROb2RlLmFzS2luZChTeW50YXhLaW5kLkNhbGxFeHByZXNzaW9uKTtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmICghY2FsbEV4cHJlc3Npb24pIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYENhbGxFeHByZXNzaW9uIG5vdCBmb3VuZCBmb3IgJHtjdXJyZW50Tm9kZS5nZXRUZXh0KCl9YCk7XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBpbnN0YW5jZUlzR2VuZXJpYyA9IGNhbGxFeHByZXNzaW9uLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGggPiAwO1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGluc3RhbmNlSXNHZW5lcmljKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgYXJncyA9IGNhbGxFeHByZXNzaW9uLmdldFR5cGVBcmd1bWVudHMoKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjb25QYXJhbXMgPSBjYWxsRXhwcmVzc2lvbi5nZXRUeXBlQXJndW1lbnRzKCkubWFwKHBhcmFtID0+IHBhcmFtLmdldFRleHQoKSk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCFIZWxwZXJzLmFycmF5c0FyZUVxdWFsKGNvblBhcmFtcyxnZW5QYXJhbXMpKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBnZW5FbGVtZW50O1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZihlbGVtZW50IGluc3RhbmNlb2YgRnVuY3Rpb25EZWNsYXJhdGlvbil7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBnZW5FbGVtZW50ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4RnVuY3Rpb24oZWxlbWVudCwge30pIGFzIEZhbWl4LlBhcmFtZXRyaWNGdW5jdGlvbjtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdlbkVsZW1lbnQgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhNZXRob2QoZWxlbWVudCwge30pIGFzIEZhbWl4LlBhcmFtZXRyaWNNZXRob2Q7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbGV0IGNvbmNFbGVtZW50O1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jRWxlbWVudCA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChnZW5FbGVtZW50LGVsZW1lbnQsYXJncyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgY3JlYXRlQ29uY3JldGlzYXRpb24gOiBib29sZWFuID0gdHJ1ZTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uY3JldGlzYXRpb25zLmZvckVhY2goKGNvbmMgOiBGYW1peC5Db25jcmV0aXNhdGlvbikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkVsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lID09IGNvbmMuZ2VuZXJpY0VudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgJiYgY29uYy5jb25jcmV0ZUVudGl0eS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uY0VsZW1lbnQuZnVsbHlRdWFsaWZpZWROYW1lKXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjcmVhdGVDb25jcmV0aXNhdGlvbiA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjcmVhdGVDb25jcmV0aXNhdGlvbikge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgZm14Q29uY3JldGlzYXRpb24gOiBGYW1peC5Db25jcmV0aXNhdGlvbiA9IHRoaXMuY3JlYXRlRmFtaXhDb25jcmV0aXNhdGlvbihjb25jRWxlbWVudCxnZW5FbGVtZW50KTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIC8vIFJlbW9udGVyIMOgIGwnw6lsw6ltZW50IHBhcmVudCAodXRpbGUgc2kgbGUgbsWTdWQgZGUgcsOpZsOpcmVuY2UgZXN0IHVuIGVuZmFudClcbiAgICAgICAgICAgICAgICAgICAgY3VycmVudE5vZGUgPSBjdXJyZW50Tm9kZS5nZXRQYXJlbnQoKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGEgY2xhc3MgYW5kIGFuIGludGVyZmFjZVxuICAgICAqIEBwYXJhbSBjbHMgQSBjbGFzc1xuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uSW50ZXJmYWNlQ2xhc3MoY2xzOiBDbGFzc0RlY2xhcmF0aW9uKXtcbiAgICBcbiAgICAgICAgY29uc3Qgc3VwZXJJbnRlcmZhY2VzID0gY2xzLmdldEltcGxlbWVudHMoKTtcbiAgICAgICAgc3VwZXJJbnRlcmZhY2VzLmZvckVhY2goaW50ZXJmYWNlVHlwZSA9PiB7XG4gICAgICAgICAgICBjb25zdCBpbnRlcmZhY2VJc0dlbmVyaWMgPSBpbnRlcmZhY2VUeXBlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGg+MDtcbiAgICAgICAgICAgIGlmIChpbnRlcmZhY2VJc0dlbmVyaWMpIHtcbiAgICAgICAgICAgICAgICBjb25zdCBpbnRlcmZhY2VEZWNsYXJhdGlvbiA9IGludGVyZmFjZVR5cGUuZ2V0RXhwcmVzc2lvbigpLmdldFN5bWJvbCgpPy5nZXREZWNsYXJhdGlvbnMoKVswXSBhcyBJbnRlcmZhY2VEZWNsYXJhdGlvbjtcbiAgICAgICAgICAgICAgICBjb25zdCBnZW5QYXJhbXMgPSBpbnRlcmZhY2VEZWNsYXJhdGlvbi5nZXRUeXBlUGFyYW1ldGVycygpLm1hcCgocGFyYW0pID0+IHBhcmFtLmdldFRleHQoKSk7XG4gICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gY2xzLmdldEhlcml0YWdlQ2xhdXNlcygpWzBdLmdldFR5cGVOb2RlcygpWzBdLmdldFR5cGVBcmd1bWVudHMoKS5tYXAoKHBhcmFtKSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGFyZ3MgPSBjbHMuZ2V0SGVyaXRhZ2VDbGF1c2VzKClbMF0uZ2V0VHlwZU5vZGVzKClbMF0uZ2V0VHlwZUFyZ3VtZW50cygpO1xuICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xuICAgICAgICAgICAgICAgICAgICBjb25zdCBnZW5JbnRlcmZhY2UgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoaW50ZXJmYWNlRGVjbGFyYXRpb24pIGFzIEZhbWl4LlBhcmFtZXRyaWNJbnRlcmZhY2U7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbkludGVyZmFjZSA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENvbmNyZXRlRWxlbWVudChnZW5JbnRlcmZhY2UsaW50ZXJmYWNlRGVjbGFyYXRpb24sYXJncyk7XG4gICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbmNyZXRpc2F0aW9ucyA9IHRoaXMuZmFtaXhSZXAuX2dldEFsbEVudGl0aWVzV2l0aFR5cGUoXCJDb25jcmV0aXNhdGlvblwiKSBhcyBTZXQ8RmFtaXguQ29uY3JldGlzYXRpb24+O1xuICAgICAgICAgICAgICAgICAgICBsZXQgY3JlYXRlQ29uY3JldGlzYXRpb24gOiBib29sZWFuID0gdHJ1ZTtcbiAgICAgICAgICAgICAgICAgICAgY29uY3JldGlzYXRpb25zLmZvckVhY2goKGNvbmMgOiBGYW1peC5Db25jcmV0aXNhdGlvbikgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGdlbkludGVyZmFjZS5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25JbnRlcmZhY2UuZnVsbHlRdWFsaWZpZWROYW1lKXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjcmVhdGVDb25jcmV0aXNhdGlvbiA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIFxuICAgICAgICAgICAgICAgICAgICBpZiAoY3JlYXRlQ29uY3JldGlzYXRpb24pIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGZteENvbmNyZXRpc2F0aW9uIDogRmFtaXguQ29uY3JldGlzYXRpb24gPSB0aGlzLmNyZWF0ZUZhbWl4Q29uY3JldGlzYXRpb24oY29uSW50ZXJmYWNlLGdlbkludGVyZmFjZSk7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgIH0pO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIENyZWF0ZXMgYSBGYW1peCBjb25jcmV0aXNhdGlvbiBiZXR3ZWVuIGFuIGludGVyZmFjZSBhbmQgYSBUeXBlXG4gICAgICogQHBhcmFtIGVsZW1lbnQgQSB2YXJpYWJsZSBvciBhIGZ1bmN0aW9uXG4gICAgICogQHBhcmFtIGludGVyIEFuIGludGVyZmFjZVxuICAgICAqL1xuICAgIHB1YmxpYyBjcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uVHlwZUluc3RhbmNpYXRpb24oZWxlbWVudDogSW50ZXJmYWNlRGVjbGFyYXRpb24gfCBDbGFzc0RlY2xhcmF0aW9uKXtcblxuICAgICAgICBjb25zdCBpc0dlbmVyaWMgPSBlbGVtZW50LmdldFR5cGVQYXJhbWV0ZXJzKCkubGVuZ3RoID4gMDtcbiAgICAgICAgaWYgKGlzR2VuZXJpYykge1xuICAgICAgICAgICAgY29uc3QgZ2VuUGFyYW1zID0gZWxlbWVudC5nZXRUeXBlUGFyYW1ldGVycygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgY29uc3QgdXNlcyA9IGVsZW1lbnQuZmluZFJlZmVyZW5jZXNBc05vZGVzKCk7XG4gICAgICAgICAgICB1c2VzLmZvckVhY2godXNlID0+IHsgICAgICAgIFxuICAgICAgICAgICAgICAgIGxldCBwYXJlbnROb2RlID0gdXNlLmdldFBhcmVudCgpO1xuICAgICAgICAgICAgICAgIHdoaWxlIChwYXJlbnROb2RlKSB7XG4gICAgICAgICAgICAgICAgICAgIGlmIChwYXJlbnROb2RlLmdldEtpbmQoKSA9PT0gU3ludGF4S2luZC5UeXBlUmVmZXJlbmNlKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB0eXBlUmVmZXJlbmNlTm9kZSA9IHBhcmVudE5vZGUuYXNLaW5kKFN5bnRheEtpbmQuVHlwZVJlZmVyZW5jZSk7XG4gICAgICAgICAgICAgICAgICAgICAgICBpZiAoIXR5cGVSZWZlcmVuY2VOb2RlKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBUeXBlUmVmZXJlbmNlTm9kZSBub3QgZm91bmQgZm9yICR7cGFyZW50Tm9kZS5nZXRUZXh0KCl9YCk7XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB0eXBlUmVmZXJlbmNlTm9kZUlzR2VuZXJpYyA9IHR5cGVSZWZlcmVuY2VOb2RlLmdldFR5cGVBcmd1bWVudHMoKS5sZW5ndGggPiAwO1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHR5cGVSZWZlcmVuY2VOb2RlSXNHZW5lcmljKSB7fVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGFyZ3MgPSB0eXBlUmVmZXJlbmNlTm9kZS5nZXRUeXBlQXJndW1lbnRzKCk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uUGFyYW1zID0gdHlwZVJlZmVyZW5jZU5vZGUuZ2V0VHlwZUFyZ3VtZW50cygpLm1hcChwYXJhbSA9PiBwYXJhbS5nZXRUZXh0KCkpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmICghSGVscGVycy5hcnJheXNBcmVFcXVhbChjb25QYXJhbXMsZ2VuUGFyYW1zKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgZ2VuRWxlbWVudDtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYoZWxlbWVudCBpbnN0YW5jZW9mIENsYXNzRGVjbGFyYXRpb24pe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZ2VuRWxlbWVudCA9IHRoaXMuY3JlYXRlT3JHZXRGYW1peENsYXNzKGVsZW1lbnQpIGFzIEZhbWl4LlBhcmFtZXRyaWNDbGFzcztcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdlbkVsZW1lbnQgPSB0aGlzLmNyZWF0ZU9yR2V0RmFtaXhJbnRlcmZhY2UoZWxlbWVudCkgYXMgRmFtaXguUGFyYW1ldHJpY0ludGVyZmFjZTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsZXQgY29uY0VsZW1lbnQ7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbmNFbGVtZW50ID0gdGhpcy5jcmVhdGVPckdldEZhbWl4Q29uY3JldGVFbGVtZW50KGdlbkVsZW1lbnQsZWxlbWVudCxhcmdzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgY29uY3JldGlzYXRpb25zID0gdGhpcy5mYW1peFJlcC5fZ2V0QWxsRW50aXRpZXNXaXRoVHlwZShcIkNvbmNyZXRpc2F0aW9uXCIpIGFzIFNldDxGYW1peC5Db25jcmV0aXNhdGlvbj47XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxldCBjcmVhdGVDb25jcmV0aXNhdGlvbiA6IGJvb2xlYW4gPSB0cnVlO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25jcmV0aXNhdGlvbnMuZm9yRWFjaCgoY29uYyA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoZ2VuRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUgPT0gY29uYy5nZW5lcmljRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSAmJiBjb25jLmNvbmNyZXRlRW50aXR5LmZ1bGx5UXVhbGlmaWVkTmFtZSA9PSBjb25jRWxlbWVudC5mdWxseVF1YWxpZmllZE5hbWUpe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbmNyZXRpc2F0aW9uID0gZmFsc2U7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICBcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGNyZWF0ZUNvbmNyZXRpc2F0aW9uKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBmbXhDb25jcmV0aXNhdGlvbiA6IEZhbWl4LkNvbmNyZXRpc2F0aW9uID0gdGhpcy5jcmVhdGVGYW1peENvbmNyZXRpc2F0aW9uKGNvbmNFbGVtZW50LGdlbkVsZW1lbnQpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgcGFyZW50Tm9kZSA9IHBhcmVudE5vZGUuZ2V0UGFyZW50KCk7XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSk7XG4gICAgICAgIH1cbiAgICB9XG5cbiAgICBwdWJsaWMgY29udmVydFRvUmVsYXRpdmVQYXRoKGFic29sdXRlUGF0aDogc3RyaW5nLCBhYnNvbHV0ZVBhdGhQcm9qZWN0OiBzdHJpbmcpIHtcbiAgICAgICAgcmV0dXJuIGFic29sdXRlUGF0aC5yZXBsYWNlKGFic29sdXRlUGF0aFByb2plY3QsIFwiXCIpLnNsaWNlKDEpO1xuICAgIH1cbn1cbmZ1bmN0aW9uIGluaXRGUU4oc291cmNlRWxlbWVudDogVFNNb3JwaE9iamVjdFR5cGUsIGZhbWl4RWxlbWVudDogRmFtaXguU291cmNlZEVudGl0eSkge1xuICAgIGlmICghKHNvdXJjZUVsZW1lbnQgaW5zdGFuY2VvZiBDb21tZW50UmFuZ2UpKSB7XG4gICAgICAgIGNvbnN0IGZxbiA9IEZRTkZ1bmN0aW9ucy5nZXRGUU4oc291cmNlRWxlbWVudCk7XG4gICAgICAgIGxvZ2dlci5kZWJ1ZyhcIlNldHRpbmcgZnVsbHkgcXVhbGlmaWVkIG5hbWUgZm9yIFwiICsgZmFtaXhFbGVtZW50LmdldEpTT04oKSArIFwiIHRvIFwiICsgZnFuKTtcbiAgICAgICAgKGZhbWl4RWxlbWVudCBhcyBGYW1peC5OYW1lZEVudGl0eSkuZnVsbHlRdWFsaWZpZWROYW1lID0gZnFuO1xuICAgIH1cbn1cblxuIl19