ts2famix 2.0.3 → 2.1.0-beta.2

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 (191) hide show
  1. package/.eslintrc.json +24 -24
  2. package/.vscode/settings.json +4 -0
  3. package/LICENSE +24 -24
  4. package/README.md +78 -78
  5. package/TODO +1 -0
  6. package/arwea-fix.json +1 -0
  7. package/bogus.ts +3 -0
  8. package/class-diagram.puml +792 -0
  9. package/debug.txt +13332 -0
  10. package/debuglog.txt +12073 -0
  11. package/dist/analyze.js +19 -9
  12. package/dist/analyze_functions/process_functions.js +101 -71
  13. package/dist/famix2puml.js +126 -0
  14. package/dist/famix_functions/EntityDictionary.js +881 -509
  15. package/dist/famix_functions/helpers_creation.js +18 -8
  16. package/dist/fqn.js +44 -11
  17. package/dist/getClasses-arrow-body.js +43 -0
  18. package/dist/lib/famix/famix_JSON_exporter.js +1 -1
  19. package/dist/lib/famix/famix_base_element.js +1 -1
  20. package/dist/lib/famix/famix_repository.js +67 -2
  21. package/dist/lib/famix/index.js +18 -8
  22. package/dist/lib/famix/model/famix/access.js +1 -1
  23. package/dist/lib/famix/model/famix/accessor.js +1 -1
  24. package/dist/lib/famix/model/famix/alias.js +1 -1
  25. package/dist/lib/famix/model/famix/arrow_function.js +1 -1
  26. package/dist/lib/famix/model/famix/behavioral_entity.js +1 -1
  27. package/dist/lib/famix/model/famix/class.js +1 -1
  28. package/dist/lib/famix/model/famix/comment.js +1 -1
  29. package/dist/lib/famix/model/famix/concretisation.js +1 -1
  30. package/dist/lib/famix/model/famix/container_entity.js +1 -1
  31. package/dist/lib/famix/model/famix/decorator.js +1 -1
  32. package/dist/lib/famix/model/famix/entity.js +1 -1
  33. package/dist/lib/famix/model/famix/enum.js +1 -1
  34. package/dist/lib/famix/model/famix/enum_value.js +1 -1
  35. package/dist/lib/famix/model/famix/function.js +1 -1
  36. package/dist/lib/famix/model/famix/import_clause.js +1 -1
  37. package/dist/lib/famix/model/famix/index.js +1 -1
  38. package/dist/lib/famix/model/famix/indexed_file_anchor.js +1 -1
  39. package/dist/lib/famix/model/famix/inheritance.js +1 -1
  40. package/dist/lib/famix/model/famix/interface.js +1 -1
  41. package/dist/lib/famix/model/famix/invocation.js +1 -1
  42. package/dist/lib/famix/model/famix/method.js +1 -1
  43. package/dist/lib/famix/model/famix/module.js +1 -1
  44. package/dist/lib/famix/model/famix/named_entity.js +1 -1
  45. package/dist/lib/famix/model/famix/parameter.js +1 -1
  46. package/dist/lib/famix/model/famix/parameter_concretisation.js +1 -1
  47. package/dist/lib/famix/model/famix/parameter_type.js +1 -1
  48. package/dist/lib/famix/model/famix/parametric_arrow_function.js +1 -1
  49. package/dist/lib/famix/model/famix/parametric_class.js +1 -1
  50. package/dist/lib/famix/model/famix/parametric_function.js +1 -1
  51. package/dist/lib/famix/model/famix/parametric_interface.js +1 -1
  52. package/dist/lib/famix/model/famix/parametric_method.js +1 -1
  53. package/dist/lib/famix/model/famix/primitive_type.js +1 -1
  54. package/dist/lib/famix/model/famix/property.js +1 -1
  55. package/dist/lib/famix/model/famix/reference.js +1 -1
  56. package/dist/lib/famix/model/famix/scoping_entity.js +1 -1
  57. package/dist/lib/famix/model/famix/script_entity.js +1 -1
  58. package/dist/lib/famix/model/famix/source_anchor.js +1 -1
  59. package/dist/lib/famix/model/famix/source_language.js +1 -1
  60. package/dist/lib/famix/model/famix/sourced_entity.js +1 -1
  61. package/dist/lib/famix/model/famix/structural_entity.js +1 -1
  62. package/dist/lib/famix/model/famix/type.js +1 -1
  63. package/dist/lib/famix/model/famix/variable.js +1 -1
  64. package/dist/lib/famix/src/famix_JSON_exporter.js +55 -0
  65. package/dist/lib/famix/src/famix_base_element.js +18 -0
  66. package/dist/lib/famix/src/famix_repository.js +224 -0
  67. package/dist/lib/famix/src/index.js +31 -0
  68. package/dist/lib/famix/src/model/famix/access.js +40 -0
  69. package/dist/lib/famix/src/model/famix/accessor.js +17 -0
  70. package/dist/lib/famix/src/model/famix/alias.js +33 -0
  71. package/dist/lib/famix/src/model/famix/arrowFunction.js +17 -0
  72. package/dist/lib/famix/src/model/famix/arrow_function.js +17 -0
  73. package/dist/lib/famix/src/model/famix/behavioral_entity.js +79 -0
  74. package/dist/lib/famix/src/model/famix/class.js +71 -0
  75. package/dist/lib/famix/src/model/famix/comment.js +39 -0
  76. package/dist/lib/famix/src/model/famix/concretisation.js +31 -0
  77. package/dist/lib/famix/src/model/famix/container_entity.js +126 -0
  78. package/dist/lib/famix/src/model/famix/decorator.js +32 -0
  79. package/dist/lib/famix/src/model/famix/entity.js +17 -0
  80. package/dist/lib/famix/src/model/famix/enum.js +31 -0
  81. package/dist/lib/famix/src/model/famix/enum_value.js +25 -0
  82. package/dist/lib/famix/src/model/famix/function.js +17 -0
  83. package/dist/lib/famix/src/model/famix/implicit_variable.js +17 -0
  84. package/dist/lib/famix/src/model/famix/import_clause.js +41 -0
  85. package/dist/lib/famix/src/model/famix/index.js +86 -0
  86. package/dist/lib/famix/src/model/famix/indexed_file_anchor.js +38 -0
  87. package/dist/lib/famix/src/model/famix/inheritance.js +33 -0
  88. package/dist/lib/famix/src/model/famix/interface.js +64 -0
  89. package/dist/lib/famix/src/model/famix/invocation.js +54 -0
  90. package/dist/lib/famix/src/model/famix/method.js +67 -0
  91. package/dist/lib/famix/src/model/famix/module.js +60 -0
  92. package/dist/lib/famix/src/model/famix/named_entity.js +78 -0
  93. package/dist/lib/famix/src/model/famix/parameter.js +25 -0
  94. package/dist/lib/famix/src/model/famix/parameterConcretisation.js +44 -0
  95. package/dist/lib/famix/src/model/famix/parameter_concretisation.js +44 -0
  96. package/dist/lib/famix/src/model/famix/parameter_type.js +45 -0
  97. package/dist/lib/famix/src/model/famix/parametricArrowFunction.js +29 -0
  98. package/dist/lib/famix/src/model/famix/parametric_arrow_function.js +31 -0
  99. package/dist/lib/famix/src/model/famix/parametric_class.js +44 -0
  100. package/dist/lib/famix/src/model/famix/parametric_function.js +31 -0
  101. package/dist/lib/famix/src/model/famix/parametric_interface.js +44 -0
  102. package/dist/lib/famix/src/model/famix/parametric_method.js +31 -0
  103. package/dist/lib/famix/src/model/famix/primitive_type.js +17 -0
  104. package/dist/lib/famix/src/model/famix/property.js +73 -0
  105. package/dist/lib/famix/src/model/famix/reference.js +33 -0
  106. package/dist/lib/famix/src/model/famix/scoping_entity.js +36 -0
  107. package/dist/lib/famix/src/model/famix/script_entity.js +29 -0
  108. package/dist/lib/famix/src/model/famix/source_anchor.js +27 -0
  109. package/dist/lib/famix/src/model/famix/source_language.js +35 -0
  110. package/dist/lib/famix/src/model/famix/sourced_entity.js +60 -0
  111. package/dist/lib/famix/src/model/famix/structural_entity.js +39 -0
  112. package/dist/lib/famix/src/model/famix/text_anchor.js +38 -0
  113. package/dist/lib/famix/src/model/famix/type.js +73 -0
  114. package/dist/lib/famix/src/model/famix/variable.js +24 -0
  115. package/dist/lib/ts-complex/cyclomatic-service.js +1 -1
  116. package/dist/refactorer/refactor-getter-setter.js +18 -8
  117. package/dist/ts2famix-cli-wrapper.js +18 -8
  118. package/dist/ts2famix-cli.js +18 -8
  119. package/dist/ts2famix-tsconfig.js +18 -8
  120. package/doc-uml/famix-typescript-model.puml +607 -607
  121. package/eslint.config.mjs +28 -0
  122. package/fqn-model.json +1 -0
  123. package/iterateGenericTypes.ts +69 -0
  124. package/out/class-diagram/class-diagram.svg +1 -0
  125. package/package.json +70 -66
  126. package/sample.json +1 -0
  127. package/sample.ts +1 -0
  128. package/src/analyze.ts +120 -120
  129. package/src/analyze_functions/process_functions.ts +1040 -1019
  130. package/src/famix_functions/EntityDictionary.ts +2016 -1593
  131. package/src/famix_functions/helpers_creation.ts +135 -135
  132. package/src/fqn.ts +50 -16
  133. package/src/generate_uml.sh +20 -20
  134. package/src/lib/famix/License.md +22 -22
  135. package/src/lib/famix/famix_JSON_exporter.ts +56 -56
  136. package/src/lib/famix/famix_base_element.ts +22 -22
  137. package/src/lib/famix/famix_repository.ts +278 -243
  138. package/src/lib/famix/index.ts +8 -8
  139. package/src/lib/famix/model/famix/access.ts +50 -50
  140. package/src/lib/famix/model/famix/accessor.ts +15 -15
  141. package/src/lib/famix/model/famix/alias.ts +39 -39
  142. package/src/lib/famix/model/famix/arrow_function.ts +15 -15
  143. package/src/lib/famix/model/famix/behavioral_entity.ts +97 -97
  144. package/src/lib/famix/model/famix/class.ts +85 -85
  145. package/src/lib/famix/model/famix/comment.ts +47 -47
  146. package/src/lib/famix/model/famix/concretisation.ts +40 -40
  147. package/src/lib/famix/model/famix/container_entity.ts +160 -160
  148. package/src/lib/famix/model/famix/decorator.ts +37 -37
  149. package/src/lib/famix/model/famix/entity.ts +15 -15
  150. package/src/lib/famix/model/famix/enum.ts +30 -30
  151. package/src/lib/famix/model/famix/enum_value.ts +28 -28
  152. package/src/lib/famix/model/famix/function.ts +15 -15
  153. package/src/lib/famix/model/famix/import_clause.ts +51 -51
  154. package/src/lib/famix/model/famix/index.ts +41 -41
  155. package/src/lib/famix/model/famix/indexed_file_anchor.ts +46 -46
  156. package/src/lib/famix/model/famix/inheritance.ts +40 -40
  157. package/src/lib/famix/model/famix/interface.ts +75 -75
  158. package/src/lib/famix/model/famix/invocation.ts +65 -65
  159. package/src/lib/famix/model/famix/method.ts +89 -89
  160. package/src/lib/famix/model/famix/module.ts +71 -71
  161. package/src/lib/famix/model/famix/named_entity.ts +95 -95
  162. package/src/lib/famix/model/famix/parameter.ts +28 -28
  163. package/src/lib/famix/model/famix/parameter_concretisation.ts +51 -51
  164. package/src/lib/famix/model/famix/parameter_type.ts +58 -58
  165. package/src/lib/famix/model/famix/parametric_arrow_function.ts +32 -32
  166. package/src/lib/famix/model/famix/parametric_class.ts +49 -49
  167. package/src/lib/famix/model/famix/parametric_function.ts +32 -32
  168. package/src/lib/famix/model/famix/parametric_interface.ts +49 -49
  169. package/src/lib/famix/model/famix/parametric_method.ts +32 -32
  170. package/src/lib/famix/model/famix/primitive_type.ts +15 -15
  171. package/src/lib/famix/model/famix/property.ts +94 -94
  172. package/src/lib/famix/model/famix/reference.ts +40 -40
  173. package/src/lib/famix/model/famix/scoping_entity.ts +35 -35
  174. package/src/lib/famix/model/famix/script_entity.ts +34 -34
  175. package/src/lib/famix/model/famix/source_anchor.ts +30 -30
  176. package/src/lib/famix/model/famix/source_language.ts +35 -35
  177. package/src/lib/famix/model/famix/sourced_entity.ts +70 -70
  178. package/src/lib/famix/model/famix/structural_entity.ts +43 -43
  179. package/src/lib/famix/model/famix/type.ts +87 -87
  180. package/src/lib/famix/model/famix/variable.ts +27 -27
  181. package/src/lib/famix/package.json +28 -28
  182. package/src/lib/ts-complex/cyclomatic-service.ts +83 -83
  183. package/src/refactorer/refactor-getter-setter.ts +140 -140
  184. package/src/ts2famix-cli-wrapper.ts +21 -21
  185. package/src/ts2famix-cli.ts +60 -60
  186. package/stats.txt +3091 -0
  187. package/tabby-debug-output.txt +19433 -0
  188. package/ts2famix.log +22656 -0
  189. package/tsconfig.check-tests.json +14 -14
  190. package/tsconfig.json +72 -72
  191. package/validate-references.js +103 -0
@@ -1,4 +1,9 @@
1
1
  "use strict";
2
+ /**
3
+ * a function getOrCreateXType takes arguments name: string and element: ts-morph-type and returns a Famix.Type
4
+ * The goal is to keep track of the types (e.g., a method's definedType), for the model.
5
+ * The name doesn't need to be fully qualified (it's the name used in the source code, or the Famix model).
6
+ */
2
7
  var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
8
  if (k2 === undefined) k2 = k;
4
9
  var desc = Object.getOwnPropertyDescriptor(m, k);
@@ -15,28 +20,40 @@ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (
15
20
  }) : function(o, v) {
16
21
  o["default"] = v;
17
22
  });
18
- var __importStar = (this && this.__importStar) || function (mod) {
19
- if (mod && mod.__esModule) return mod;
20
- var result = {};
21
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
- __setModuleDefault(result, mod);
23
- return result;
24
- };
23
+ var __importStar = (this && this.__importStar) || (function () {
24
+ var ownKeys = function(o) {
25
+ ownKeys = Object.getOwnPropertyNames || function (o) {
26
+ var ar = [];
27
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
28
+ return ar;
29
+ };
30
+ return ownKeys(o);
31
+ };
32
+ return function (mod) {
33
+ if (mod && mod.__esModule) return mod;
34
+ var result = {};
35
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
36
+ __setModuleDefault(result, mod);
37
+ return result;
38
+ };
39
+ })();
25
40
  var __importDefault = (this && this.__importDefault) || function (mod) {
26
41
  return (mod && mod.__esModule) ? mod : { "default": mod };
27
42
  };
28
43
  Object.defineProperty(exports, "__esModule", { value: true });
29
44
  exports.EntityDictionary = void 0;
45
+ exports.isPrimitiveType = isPrimitiveType;
30
46
  const ts_morph_1 = require("ts-morph");
31
47
  const process_functions_1 = require("../analyze_functions/process_functions");
32
48
  const Famix = __importStar(require("../lib/famix/model/famix"));
33
49
  const famix_repository_1 = require("../lib/famix/famix_repository");
34
50
  const analyze_1 = require("../analyze");
35
- const grapheme_splitter_1 = __importDefault(require("grapheme-splitter"));
51
+ // import GraphemeSplitter from "grapheme-splitter";
52
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
53
+ const GraphemeSplitter = require("grapheme-splitter");
36
54
  const Helpers = __importStar(require("./helpers_creation"));
37
55
  const FQNFunctions = __importStar(require("../fqn"));
38
56
  const path_1 = __importDefault(require("path"));
39
- const lodash_1 = __importDefault(require("lodash"));
40
57
  class EntityDictionary {
41
58
  constructor() {
42
59
  this.famixRep = new famix_repository_1.FamixRepository();
@@ -45,8 +62,15 @@ class EntityDictionary {
45
62
  this.fmxInterfaceMap = new Map(); // Maps the interface names to their Famix model
46
63
  this.fmxModuleMap = new Map(); // Maps the namespace names to their Famix model
47
64
  this.fmxFileMap = new Map(); // Maps the source file names to their Famix model
48
- this.fmxTypeMap = new Map(); // Maps the type names to their Famix model
65
+ this.fmxTypeMap = new Map(); // Maps the types declarations to their Famix model
66
+ this.fmxPrimitiveTypeMap = new Map(); // Maps the primitive type names to their Famix model
49
67
  this.fmxFunctionAndMethodMap = new Map; // Maps the function names to their Famix model
68
+ this.fmxArrowFunctionMap = new Map; // Maps the function names to their Famix model
69
+ this.fmxParameterMap = new Map(); // Maps the parameters to their Famix model
70
+ this.fmxVariableMap = new Map(); // Maps the variables to their Famix model
71
+ this.fmxImportClauseMap = new Map(); // Maps the import clauses to their Famix model
72
+ this.fmxEnumMap = new Map(); // Maps the enum names to their Famix model
73
+ this.fmxInheritanceMap = new Map(); // Maps the inheritance names to their Famix model
50
74
  this.UNKNOWN_VALUE = '(unknown due to parsing error)'; // The value to use when a name is not usable
51
75
  this.fmxElementObjectMap = new Map();
52
76
  this.tsMorphElementObjectMap = new Map();
@@ -76,7 +100,7 @@ class EntityDictionary {
76
100
  * Famix index file anchor.
77
101
  * It depends on code in the 'grapheme-splitter' package in npm.
78
102
  */
79
- const splitter = new grapheme_splitter_1.default();
103
+ const splitter = new GraphemeSplitter();
80
104
  const sourceFileText = node.getSourceFile().getFullText();
81
105
  const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;
82
106
  if (hasGraphemeClusters) {
@@ -94,7 +118,10 @@ class EntityDictionary {
94
118
  // The +1 is because the source anchor (Pharo) is 1-based, but ts-morph is 0-based
95
119
  sourceAnchor.startPos = sourceStart + 1;
96
120
  sourceAnchor.endPos = sourceEnd + 1;
97
- const fileName = node.getSourceFile().getFilePath();
121
+ let fileName = node.getSourceFile().getFilePath();
122
+ if (fileName.startsWith("/")) {
123
+ fileName = fileName.substring(1);
124
+ }
98
125
  sourceAnchor.element = fmx;
99
126
  sourceAnchor.fileName = fileName;
100
127
  fmx.sourceAnchor = sourceAnchor;
@@ -108,9 +135,9 @@ class EntityDictionary {
108
135
  * @param famixElement The Famix model of the source element
109
136
  */
110
137
  makeFamixIndexFileAnchor(sourceElement, famixElement) {
111
- // check if famixElement doesn't have a valid fullyQualifiedName
112
- if (typeof famixElement.getFullyQualifiedName === 'function') {
113
- // The method exists
138
+ // Famix.Comment is not a named entity (does not have a fullyQualifiedName)
139
+ if (!(famixElement instanceof Famix.Comment)) { // must be a named entity
140
+ // insanity check: named entities should have fullyQualifiedName
114
141
  const fullyQualifiedName = famixElement.fullyQualifiedName;
115
142
  if (!fullyQualifiedName || fullyQualifiedName === this.UNKNOWN_VALUE) {
116
143
  throw new Error(`Famix element ${famixElement.constructor.name} has no valid fullyQualifiedName.`);
@@ -134,13 +161,16 @@ class EntityDictionary {
134
161
  }
135
162
  // revert any backslashes to forward slashes (path.normalize on windows introduces them)
136
163
  pathInProject = pathInProject.replace(/\\/g, "/");
164
+ if (pathInProject.startsWith("/")) {
165
+ pathInProject = pathInProject.substring(1);
166
+ }
137
167
  fmxIndexFileAnchor.fileName = pathInProject;
138
- let sourceStart, sourceEnd, sourceLineStart, sourceLineEnd;
168
+ let sourceStart, sourceEnd;
139
169
  if (!(sourceElement instanceof ts_morph_1.CommentRange)) {
140
170
  sourceStart = sourceElement.getStart();
141
171
  sourceEnd = sourceElement.getEnd();
142
- sourceLineStart = sourceElement.getStartLineNumber();
143
- sourceLineEnd = sourceElement.getEndLineNumber();
172
+ // sourceLineStart = sourceElement.getStartLineNumber();
173
+ // sourceLineEnd = sourceElement.getEndLineNumber();
144
174
  }
145
175
  else {
146
176
  sourceStart = sourceElement.getPos();
@@ -157,7 +187,7 @@ class EntityDictionary {
157
187
  * Famix index file anchor.
158
188
  * It depends on code in the 'grapheme-splitter' package in npm.
159
189
  */
160
- const splitter = new grapheme_splitter_1.default();
190
+ const splitter = new GraphemeSplitter();
161
191
  const sourceFileText = sourceElement.getSourceFile().getFullText();
162
192
  const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;
163
193
  if (hasGraphemeClusters) {
@@ -222,28 +252,30 @@ class EntityDictionary {
222
252
  }
223
253
  /**
224
254
  * Creates or gets a Famix Module
225
- * @param m A module
255
+ * @param moduleDeclaration A module
226
256
  * @returns The Famix model of the module
227
257
  */
228
- createOrGetFamixModule(m) {
229
- let fmxModule;
230
- const moduleName = m.getName();
231
- const foundModuleName = this.fmxModuleMap.get(moduleName);
232
- if (!foundModuleName) {
233
- fmxModule = new Famix.Module();
234
- fmxModule.name = moduleName;
235
- fmxModule.isAmbient = (0, process_functions_1.isAmbient)(m);
236
- fmxModule.isNamespace = (0, process_functions_1.isNamespace)(m);
237
- fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;
238
- initFQN(m, fmxModule);
239
- this.makeFamixIndexFileAnchor(m, fmxModule);
240
- this.fmxModuleMap.set(moduleName, fmxModule);
241
- this.famixRep.addElement(fmxModule);
242
- }
243
- else {
244
- fmxModule = foundModuleName;
245
- }
246
- this.fmxElementObjectMap.set(fmxModule, m);
258
+ createOrGetFamixModule(moduleDeclaration) {
259
+ if (this.fmxModuleMap.has(moduleDeclaration)) {
260
+ const rModule = this.fmxModuleMap.get(moduleDeclaration);
261
+ if (rModule) {
262
+ return rModule;
263
+ }
264
+ else {
265
+ throw new Error(`Famix module ${moduleDeclaration.getName()} is not found in the module map.`);
266
+ }
267
+ }
268
+ const fmxModule = new Famix.Module();
269
+ const moduleName = moduleDeclaration.getName();
270
+ fmxModule.name = moduleName;
271
+ fmxModule.isAmbient = (0, process_functions_1.isAmbient)(moduleDeclaration);
272
+ fmxModule.isNamespace = (0, process_functions_1.isNamespace)(moduleDeclaration);
273
+ fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;
274
+ initFQN(moduleDeclaration, fmxModule);
275
+ this.makeFamixIndexFileAnchor(moduleDeclaration, fmxModule);
276
+ this.fmxModuleMap.set(moduleDeclaration, fmxModule);
277
+ this.famixRep.addElement(fmxModule);
278
+ this.fmxElementObjectMap.set(fmxModule, moduleDeclaration);
247
279
  return fmxModule;
248
280
  }
249
281
  /**
@@ -254,7 +286,8 @@ class EntityDictionary {
254
286
  createFamixAlias(a) {
255
287
  let fmxAlias;
256
288
  const aliasName = a.getName();
257
- const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);
289
+ //const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);
290
+ const aliasFullyQualifiedName = FQNFunctions.getFQN(a);
258
291
  const foundAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);
259
292
  if (!foundAlias) {
260
293
  fmxAlias = new Famix.Alias();
@@ -294,7 +327,8 @@ class EntityDictionary {
294
327
  fmxClass = new Famix.Class();
295
328
  }
296
329
  fmxClass.name = clsName;
297
- fmxClass.fullyQualifiedName = classFullyQualifiedName;
330
+ initFQN(cls, fmxClass);
331
+ // fmxClass.fullyQualifiedName = classFullyQualifiedName;
298
332
  fmxClass.isAbstract = isAbstract;
299
333
  this.makeFamixIndexFileAnchor(cls, fmxClass);
300
334
  this.fmxClassMap.set(classFullyQualifiedName, fmxClass);
@@ -343,56 +377,51 @@ class EntityDictionary {
343
377
  * @param concreteArguments concrete arguments
344
378
  * @returns A parametric Element
345
379
  */
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
- }
380
+ // public createOrGetFamixConcreteElement(concreteElement : ParametricVariantType,
381
+ // concreteElementDeclaration : ConcreteElementTSMorphType,
382
+ // concreteArguments: TypeNode[]): ParametricVariantType {
383
+ // let fullyQualifiedFilename = concreteElement.fullyQualifiedName;
384
+ // let params = "";
385
+ // concreteArguments.map((param) => {
386
+ // params = params+param.getText()+',';
387
+ // });
388
+ // params = params.substring(0, params.length - 1)
389
+ // fullyQualifiedFilename = Helpers.replaceLastBetweenTags(fullyQualifiedFilename,params);
390
+ // let concElement: ParametricVariantType;
391
+ // if (!this.fmxInterfaceMap.has(fullyQualifiedFilename) &&
392
+ // !this.fmxClassMap.has(fullyQualifiedFilename) &&
393
+ // !this.fmxFunctionAndMethodMap.has(fullyQualifiedFilename)){
394
+ // concElement = _.cloneDeep(concreteElement);
395
+ // concElement.fullyQualifiedName = fullyQualifiedFilename;
396
+ // concElement.clearGenericParameters();
397
+ // concreteArguments.map((param) => {
398
+ // const parameter = this.createOrGetFamixConcreteType(param);
399
+ // concElement.addConcreteParameter(parameter);
400
+ // })
401
+ // if (concreteElement instanceof Famix.ParametricClass) {
402
+ // this.fmxClassMap.set(fullyQualifiedFilename, concElement as Famix.ParametricClass);
403
+ // } else if (concreteElement instanceof Famix.ParametricInterface) {
404
+ // this.fmxInterfaceMap.set(fullyQualifiedFilename, concElement as Famix.ParametricInterface);
405
+ // } else if (concreteElement instanceof Famix.ParametricFunction) {
406
+ // this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricFunction);
407
+ // } else { // if (concreteElement instanceof Famix.ParametricMethod) {
408
+ // this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricMethod);
409
+ // }
410
+ // this.famixRep.addElement(concElement);
411
+ // this.fmxElementObjectMap.set(concElement,concreteElementDeclaration);
412
+ // } else {
413
+ // if (concreteElement instanceof Famix.ParametricClass) {
414
+ // concElement = this.fmxClassMap.get(fullyQualifiedFilename) as Famix.ParametricClass;
415
+ // } else if (concreteElement instanceof Famix.ParametricInterface) {
416
+ // concElement = this.fmxInterfaceMap.get(fullyQualifiedFilename) as Famix.ParametricInterface;
417
+ // } else if (concreteElement instanceof Famix.ParametricFunction) {
418
+ // concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricFunction;
419
+ // } else { // if (concreteElement instanceof Famix.ParametricMethod) {
420
+ // concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricMethod;
421
+ // }
422
+ // }
423
+ // return concElement;
424
+ // }
396
425
  /**
397
426
  * Creates a Famix property
398
427
  * @param property A property
@@ -479,7 +508,6 @@ class EntityDictionary {
479
508
  else {
480
509
  fmxMethod = new Famix.Method();
481
510
  }
482
- this.famixRep.addElement(fmxMethod);
483
511
  }
484
512
  const isConstructor = method instanceof ts_morph_1.ConstructorDeclaration;
485
513
  const isSignature = method instanceof ts_morph_1.MethodSignature;
@@ -541,6 +569,7 @@ class EntityDictionary {
541
569
  fmxMethod.numberOfStatements = 0;
542
570
  }
543
571
  initFQN(method, fmxMethod);
572
+ this.famixRep.addElement(fmxMethod);
544
573
  this.makeFamixIndexFileAnchor(method, fmxMethod);
545
574
  this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxMethod);
546
575
  }
@@ -576,7 +605,8 @@ class EntityDictionary {
576
605
  }
577
606
  fmxFunction.signature = Helpers.computeSignature(func.getText());
578
607
  fmxFunction.cyclomaticComplexity = currentCC[fmxFunction.name];
579
- fmxFunction.fullyQualifiedName = functionFullyQualifiedName;
608
+ initFQN(func, fmxFunction);
609
+ // fmxFunction.fullyQualifiedName = functionFullyQualifiedName;
580
610
  let functionTypeName = this.UNKNOWN_VALUE;
581
611
  try {
582
612
  functionTypeName = func.getReturnType().getText().trim();
@@ -605,7 +635,16 @@ class EntityDictionary {
605
635
  * @param param A parameter
606
636
  * @returns The Famix model of the parameter
607
637
  */
608
- createFamixParameter(param) {
638
+ createOrGetFamixParameter(param) {
639
+ if (this.fmxParameterMap.has(param)) {
640
+ const rParameter = this.fmxParameterMap.get(param);
641
+ if (rParameter) {
642
+ return rParameter;
643
+ }
644
+ else {
645
+ throw new Error(`Famix parameter ${param.getName()} is not found in the parameter map.`);
646
+ }
647
+ }
609
648
  const fmxParam = new Famix.Parameter();
610
649
  let paramTypeName = this.UNKNOWN_VALUE;
611
650
  try {
@@ -621,6 +660,7 @@ class EntityDictionary {
621
660
  this.makeFamixIndexFileAnchor(param, fmxParam);
622
661
  this.famixRep.addElement(fmxParam);
623
662
  this.fmxElementObjectMap.set(fmxParam, param);
663
+ this.fmxParameterMap.set(param, fmxParam);
624
664
  return fmxParam;
625
665
  }
626
666
  /**
@@ -638,25 +678,31 @@ class EntityDictionary {
638
678
  return fmxParameterType;
639
679
  }
640
680
  /**
641
- * Creates a Famix type parameter
642
- * @param tp A type parameter
643
- * @returns The Famix model of the type parameter
681
+ * Creates a Famix type in the context of concretizations
682
+ * @param typeName A type name
683
+ * @param element An element
684
+ * @returns The Famix model of the type
644
685
  */
645
- createOrGetFamixConcreteType(param) {
646
- const typeParameterDeclaration = param.getSymbol()?.getDeclarations()[0];
647
- const parameterTypeName = param.getText();
686
+ createOrGetFamixConcreteType(element) {
687
+ // TODO - refactor to stop using names as a key in the maps, use ts-morph element instead
688
+ const typeParameterDeclaration = element.getSymbol()?.getDeclarations()[0];
689
+ const parameterTypeName = element.getText();
648
690
  let fmxParameterType = undefined;
691
+ // get a TypeReference from a TypeNode
692
+ const typeReference = element.getType();
693
+ // get a TypeDeclaration from a TypeReference
694
+ const typeDeclaration = typeReference.getSymbol()?.getDeclarations()[0];
649
695
  let isClassOrInterface = false;
650
696
  if (this.fmxClassMap.has(parameterTypeName)) {
651
697
  this.fmxClassMap.forEach((obj, name) => {
652
698
  if (obj instanceof Famix.ParametricClass) {
653
- if (name === param.getText() && obj.genericParameters.size > 0) {
699
+ if (name === element.getText() && obj.genericParameters.size > 0) {
654
700
  fmxParameterType = obj;
655
701
  isClassOrInterface = true;
656
702
  }
657
703
  }
658
704
  else {
659
- if (name === param.getText()) {
705
+ if (name === element.getText()) {
660
706
  fmxParameterType = obj;
661
707
  isClassOrInterface = true;
662
708
  }
@@ -666,13 +712,13 @@ class EntityDictionary {
666
712
  if (this.fmxInterfaceMap.has(parameterTypeName)) {
667
713
  this.fmxInterfaceMap.forEach((obj, name) => {
668
714
  if (obj instanceof Famix.ParametricInterface) {
669
- if (name === param.getText() && obj.genericParameters.size > 0) {
715
+ if (name === element.getText() && obj.genericParameters.size > 0) {
670
716
  fmxParameterType = obj;
671
717
  isClassOrInterface = true;
672
718
  }
673
719
  }
674
720
  else {
675
- if (name === param.getText()) {
721
+ if (name === element.getText()) {
676
722
  fmxParameterType = obj;
677
723
  isClassOrInterface = true;
678
724
  }
@@ -680,8 +726,9 @@ class EntityDictionary {
680
726
  });
681
727
  }
682
728
  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") {
729
+ if (!this.fmxTypeMap.has(typeDeclaration)) {
730
+ // TODO refactor
731
+ if (isPrimitiveType(parameterTypeName)) {
685
732
  fmxParameterType = new Famix.PrimitiveType();
686
733
  fmxParameterType.isStub = true;
687
734
  }
@@ -690,16 +737,16 @@ class EntityDictionary {
690
737
  }
691
738
  fmxParameterType.name = parameterTypeName;
692
739
  this.famixRep.addElement(fmxParameterType);
693
- this.fmxTypeMap.set(parameterTypeName, fmxParameterType);
740
+ this.fmxTypeMap.set(typeDeclaration, fmxParameterType);
694
741
  this.fmxElementObjectMap.set(fmxParameterType, typeParameterDeclaration);
695
742
  }
696
743
  else {
697
- const result = this.fmxTypeMap.get(parameterTypeName);
744
+ const result = this.fmxTypeMap.get(typeDeclaration);
698
745
  if (result) {
699
746
  fmxParameterType = result;
700
747
  }
701
748
  else {
702
- throw new Error(`Famix type ${parameterTypeName} is not found in the Type map.`);
749
+ throw new Error(`Famix type ${typeDeclaration} is not found in the Type map.`);
703
750
  }
704
751
  }
705
752
  }
@@ -713,7 +760,16 @@ class EntityDictionary {
713
760
  * @param variable A variable
714
761
  * @returns The Famix model of the variable
715
762
  */
716
- createFamixVariable(variable) {
763
+ createOrGetFamixVariable(variable) {
764
+ if (this.fmxVariableMap.has(variable)) {
765
+ const rVariable = this.fmxVariableMap.get(variable);
766
+ if (rVariable) {
767
+ return rVariable;
768
+ }
769
+ else {
770
+ throw new Error(`Famix parameter ${variable.getName()} is not found in the variable map.`);
771
+ }
772
+ }
717
773
  const fmxVariable = new Famix.Variable();
718
774
  let variableTypeName = this.UNKNOWN_VALUE;
719
775
  try {
@@ -729,6 +785,7 @@ class EntityDictionary {
729
785
  this.makeFamixIndexFileAnchor(variable, fmxVariable);
730
786
  this.famixRep.addElement(fmxVariable);
731
787
  this.fmxElementObjectMap.set(fmxVariable, variable);
788
+ this.fmxVariableMap.set(variable, fmxVariable);
732
789
  return fmxVariable;
733
790
  }
734
791
  /**
@@ -736,13 +793,23 @@ class EntityDictionary {
736
793
  * @param enumEntity An enum
737
794
  * @returns The Famix model of the enum
738
795
  */
739
- createFamixEnum(enumEntity) {
796
+ createOrGetFamixEnum(enumEntity) {
797
+ if (this.fmxEnumMap.has(enumEntity)) {
798
+ const rEnum = this.fmxEnumMap.get(enumEntity);
799
+ if (rEnum) {
800
+ return rEnum;
801
+ }
802
+ else {
803
+ throw new Error(`Famix enum ${enumEntity.getName()} is not found in the enum map.`);
804
+ }
805
+ }
740
806
  const fmxEnum = new Famix.Enum();
741
807
  fmxEnum.name = enumEntity.getName();
742
808
  initFQN(enumEntity, fmxEnum);
743
809
  this.makeFamixIndexFileAnchor(enumEntity, fmxEnum);
744
810
  this.famixRep.addElement(fmxEnum);
745
811
  this.fmxElementObjectMap.set(fmxEnum, enumEntity);
812
+ this.fmxEnumMap.set(enumEntity, fmxEnum);
746
813
  return fmxEnum;
747
814
  }
748
815
  /**
@@ -814,47 +881,38 @@ class EntityDictionary {
814
881
  */
815
882
  createOrGetFamixType(typeName, element) {
816
883
  let fmxType;
817
- let isPrimitiveType = false;
818
- let isParameterType = false;
884
+ const isPrimitive = isPrimitiveType(typeName);
885
+ const isParametricType = element instanceof ts_morph_1.ClassDeclaration && element.getTypeParameters().length > 0 ||
886
+ element instanceof ts_morph_1.InterfaceDeclaration && element.getTypeParameters().length > 0;
887
+ // Functions and methods aren't types!
888
+ // ||
889
+ // element instanceof FunctionDeclaration && element.getTypeParameters().length > 0 ||
890
+ // element instanceof MethodDeclaration && element.getTypeParameters().length > 0 ||
891
+ // element instanceof ArrowFunction && element.getTypeParameters().length > 0;
819
892
  analyze_1.logger.debug("Creating (or getting) type: '" + typeName + "' of element: " + element?.getText() + " of kind: " + element?.getKindName());
820
- let ancestor = undefined;
821
- if (element !== undefined) {
822
- const typeAncestor = Helpers.findTypeAncestor(element);
823
- if (!typeAncestor) {
824
- throw new Error(`Ancestor not found for element ${element.getText()}.`);
825
- }
826
- const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);
827
- ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
828
- if (!ancestor) {
829
- analyze_1.logger.debug(`Ancestor ${FQNFunctions.getFQN(typeAncestor)} not found. Adding the new type.`);
830
- ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor);
831
- }
832
- }
833
- if (typeName === "number" || typeName === "string" || typeName === "boolean" || typeName === "bigint" || typeName === "symbol" || typeName === "undefined" || typeName === "null" || typeName === "any" || typeName === "unknown" || typeName === "never" || typeName === "void") {
834
- isPrimitiveType = true;
835
- }
836
- if (!isPrimitiveType && typeName.includes("<") && typeName.includes(">") && !(typeName.includes("=>"))) {
837
- isParameterType = true;
838
- }
839
- if (!this.fmxTypeMap.has(typeName)) {
840
- if (isPrimitiveType) {
841
- fmxType = new Famix.PrimitiveType();
842
- fmxType.isStub = true;
843
- }
844
- else if (isParameterType) {
845
- fmxType = new Famix.ParameterType();
846
- const parameterTypeNames = typeName.substring(typeName.indexOf("<") + 1, typeName.indexOf(">")).split(",").map(s => s.trim());
847
- const baseTypeName = typeName.substring(0, typeName.indexOf("<")).trim();
848
- parameterTypeNames.forEach(parameterTypeName => {
849
- const fmxParameterType = this.createOrGetFamixType(parameterTypeName, element);
850
- fmxType.addArgument(fmxParameterType);
851
- });
852
- const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);
853
- fmxType.baseType = fmxBaseType;
854
- }
855
- else {
856
- fmxType = new Famix.Type();
893
+ if (isPrimitive) {
894
+ return this.createOrGetFamixPrimitiveType(typeName);
895
+ }
896
+ if (isParametricType) {
897
+ // narrow the type
898
+ const parametricElement = element;
899
+ return this.createOrGetFamixParametricType(typeName, parametricElement);
900
+ }
901
+ if (!this.fmxTypeMap.has(element)) {
902
+ let ancestor = undefined;
903
+ if (element !== undefined) {
904
+ const typeAncestor = Helpers.findTypeAncestor(element);
905
+ if (!typeAncestor) {
906
+ throw new Error(`Ancestor not found for element ${element.getText()}.`);
907
+ }
908
+ const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);
909
+ ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
910
+ if (!ancestor) {
911
+ analyze_1.logger.debug(`Ancestor ${FQNFunctions.getFQN(typeAncestor)} not found. Adding the new type.`);
912
+ ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor);
913
+ }
857
914
  }
915
+ fmxType = new Famix.Type();
858
916
  fmxType.name = typeName;
859
917
  if (!ancestor) {
860
918
  throw new Error(`Ancestor not found for type ${typeName}.`);
@@ -863,10 +921,10 @@ class EntityDictionary {
863
921
  initFQN(element, fmxType);
864
922
  this.makeFamixIndexFileAnchor(element, fmxType);
865
923
  this.famixRep.addElement(fmxType);
866
- this.fmxTypeMap.set(typeName, fmxType);
924
+ this.fmxTypeMap.set(element, fmxType);
867
925
  }
868
926
  else {
869
- const result = this.fmxTypeMap.get(typeName);
927
+ const result = this.fmxTypeMap.get(element);
870
928
  if (result) {
871
929
  fmxType = result;
872
930
  }
@@ -877,6 +935,109 @@ class EntityDictionary {
877
935
  this.fmxElementObjectMap.set(fmxType, element);
878
936
  return fmxType;
879
937
  }
938
+ /**
939
+ * Creates or gets a Famix type that is parametric
940
+ * @param typeName A type name
941
+ * @param element A ts-morph element
942
+ * @returns The Famix model of the parameter type
943
+ */
944
+ createOrGetFamixParametricType(typeName, element) {
945
+ if (this.fmxTypeMap.has(element) === true) {
946
+ const result = this.fmxTypeMap.get(element);
947
+ if (result) {
948
+ return result;
949
+ }
950
+ else {
951
+ throw new Error(`Famix type ${typeName} is not found (undefined) in the Type map.`);
952
+ }
953
+ }
954
+ // A parametric type is a type that has type parameters, e.g., List<T>
955
+ // In TS it can be a class, an interface, a function, an arrow function, or a method
956
+ // create the Famix Parametric Type (maybe it's just an Interface, etc.)
957
+ let fmxType;
958
+ if (element instanceof ts_morph_1.ClassDeclaration) {
959
+ fmxType = new Famix.ParametricClass();
960
+ }
961
+ else if (element instanceof ts_morph_1.InterfaceDeclaration) {
962
+ fmxType = new Famix.ParametricInterface();
963
+ }
964
+ // functions and methods are not types
965
+ // else if (element instanceof FunctionDeclaration) {
966
+ // fmxType = new Famix.ParametricFunction();
967
+ // } else if (element instanceof ArrowFunction) {
968
+ // fmxType = new Famix.ParametricArrowFunction();
969
+ // } else if (element instanceof MethodDeclaration) {
970
+ // fmxType = new Famix.ParametricMethod();
971
+ // }
972
+ else {
973
+ throw new Error(`Element is not a class, interface, function, arrow function, or method.`);
974
+ }
975
+ // const parameters = element.getTypeParameters();
976
+ // // for each parameter, getOrCreate the FamixParameterType
977
+ // for (const parameter of parameters) {
978
+ // this.createOrGetFamixParameterType(parameter.getName(), parameter);
979
+ // }
980
+ // // TODO: the following code is not correct, it is just a placeholder
981
+ // const parameterTypeNames = typeName.substring(typeName.indexOf("<") + 1, typeName.indexOf(">"))
982
+ // .split(",").map(s => s.trim());
983
+ // const baseTypeName = typeName.substring(0, typeName.indexOf("<")).trim();
984
+ // parameterTypeNames.forEach(parameterTypeName => {
985
+ // const fmxParameterType = this.createOrGetFamixParameterType(parameterTypeName, element);
986
+ // (fmxType as Famix.ParameterType).addArgument(fmxParameterType);
987
+ // });
988
+ // const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);
989
+ // (fmxType as Famix.ParameterType).baseType = fmxBaseType;
990
+ fmxType.name = typeName;
991
+ initFQN(element, fmxType);
992
+ this.famixRep.addElement(fmxType);
993
+ this.fmxTypeMap.set(element, fmxType);
994
+ return fmxType;
995
+ }
996
+ /**
997
+ * Creates a type for a parameter in a parametric type, e.g., T in List<T>
998
+ * @param parameterTypeName
999
+ * @param element the TypeScript element (TSMorphParametricType) that the type is associated with
1000
+ * @returns
1001
+ */
1002
+ // createOrGetFamixParameterType(parameterTypeName: string, element: ParameterDeclaration) {
1003
+ // if (this.fmxTypeMap.has(element)) {
1004
+ // return this.fmxTypeMap.get(element) as Famix.ParameterType;
1005
+ // }
1006
+ // // determine if element is a
1007
+ // const fmxType = new Famix.ParameterType();
1008
+ // // const parameterTypeNames = typeName.substring(typeName.indexOf("<") + 1, typeName.indexOf(">"))
1009
+ // // .split(",").map(s => s.trim());
1010
+ // // const baseTypeName = typeName.substring(0, typeName.indexOf("<")).trim();
1011
+ // // parameterTypeNames.forEach(parameterTypeName => {
1012
+ // // const fmxParameterType = this.createOrGetFamixParameterType(parameterTypeName, element);
1013
+ // // (fmxType as Famix.ParameterType).addArgument(fmxParameterType);
1014
+ // // });
1015
+ // const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);
1016
+ // (fmxType as Famix.ParameterType).baseType = fmxBaseType;
1017
+ // initFQN(element, fmxType);
1018
+ // this.famixRep.addElement(fmxType);
1019
+ // this.fmxTypeMap.set(element, fmxType);
1020
+ // return fmxType;
1021
+ // }
1022
+ /**
1023
+ * Creates or gets a Famix primitive type
1024
+ * @param typeName A type name
1025
+ * @returns The Famix model of the primitive type
1026
+ */
1027
+ createOrGetFamixPrimitiveType(typeName) {
1028
+ let fmxType = new Famix.PrimitiveType();
1029
+ if (!this.fmxPrimitiveTypeMap.has(typeName)) {
1030
+ fmxType = new Famix.PrimitiveType();
1031
+ fmxType.isStub = true;
1032
+ fmxType.name = typeName;
1033
+ this.fmxPrimitiveTypeMap.set(typeName, fmxType);
1034
+ this.famixRep.addElement(fmxType);
1035
+ }
1036
+ else {
1037
+ fmxType = this.fmxPrimitiveTypeMap.get(typeName);
1038
+ }
1039
+ return fmxType;
1040
+ }
880
1041
  /**
881
1042
  * Creates a Famix access
882
1043
  * @param node A node
@@ -890,10 +1051,25 @@ class EntityDictionary {
890
1051
  analyze_1.logger.debug(`Creating FamixAccess. Node: [${node.getKindName()}] '${node.getText()}' at line ${node.getStartLineNumber()} in ${node.getSourceFile().getBaseName()}, id: ${id} refers to fmxVar '${fmxVar.fullyQualifiedName}'.`);
891
1052
  const nodeReferenceAncestor = Helpers.findAncestor(node);
892
1053
  const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);
893
- let accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
1054
+ const accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
894
1055
  if (!accessor) {
895
1056
  analyze_1.logger.error(`Ancestor ${ancestorFullyQualifiedName} of kind ${nodeReferenceAncestor.getKindName()} not found.`);
896
1057
  // accessor = this.createOrGetFamixType(ancestorFullyQualifiedName, nodeReferenceAncestor as TypeDeclaration);
1058
+ return; // bail out TODO: this is probably wrong
1059
+ }
1060
+ else {
1061
+ analyze_1.logger.debug(`Found accessor to be ${accessor.fullyQualifiedName}.`);
1062
+ }
1063
+ // make sure accessor is a method, function, script or module
1064
+ if (!(accessor instanceof Famix.Method) && !(accessor instanceof Famix.ArrowFunction) && !(accessor instanceof Famix.Function) && !(accessor instanceof Famix.ScriptEntity) && !(accessor instanceof Famix.Module)) {
1065
+ analyze_1.logger.error(`Accessor ${accessor.fullyQualifiedName} is not a method, function, etc.`);
1066
+ return;
1067
+ }
1068
+ // don't create any duplicates (e.g. if the same variable is accessed multiple times by same accessor)
1069
+ const foundAccess = this.famixRep.getFamixAccessByAccessorAndVariable(accessor, fmxVar);
1070
+ if (foundAccess) {
1071
+ analyze_1.logger.debug(`FamixAccess already exists for accessor ${accessor.fullyQualifiedName} and variable ${fmxVar.fullyQualifiedName}.`);
1072
+ return;
897
1073
  }
898
1074
  const fmxAccess = new Famix.Access();
899
1075
  fmxAccess.accessor = accessor;
@@ -903,89 +1079,136 @@ class EntityDictionary {
903
1079
  }
904
1080
  /**
905
1081
  * Creates a Famix invocation
906
- * @param node A node
907
- * @param m A method or a function
1082
+ * @param nodeReferringToInvocable A node
1083
+ * @param invocable A method or a function
908
1084
  * @param id The id of the method or the function
909
1085
  */
910
- createFamixInvocation(node, m, id) {
911
- const fmxMethodOrFunction = this.famixRep.getFamixEntityById(id);
912
- const nodeReferenceAncestor = Helpers.findAncestor(node);
913
- const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);
914
- const sender = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
915
- const receiverFullyQualifiedName = FQNFunctions.getFQN(m.getParent());
1086
+ createFamixInvocation(nodeReferringToInvocable, invocable, id) {
1087
+ const fmxInvocable = this.famixRep.getFamixEntityById(id);
1088
+ // since the node is in the AST, we need to find the ancestor that is in the Famix model
1089
+ const containerOfNode = Helpers.findAncestor(nodeReferringToInvocable);
1090
+ analyze_1.logger.debug(`Found container (ancestor) ${containerOfNode.getKindName()} for AST node ${nodeReferringToInvocable.getText()}.`);
1091
+ const containerFQN = FQNFunctions.getFQN(containerOfNode);
1092
+ analyze_1.logger.debug(`Found containerFQN ${containerFQN}.`);
1093
+ let sender = this.famixRep.getFamixEntityByFullyQualifiedName(containerFQN);
1094
+ analyze_1.logger.debug(`Found a sender that matches ${sender.fullyQualifiedName}.`);
1095
+ if (sender instanceof Famix.Type) {
1096
+ // TODO this might be an error in getFamixEntityByFullyQualifiedName
1097
+ analyze_1.logger.debug(`Oops! Sender is a type, which is not valid for an Invocation. Trying to find a container for ${sender.fullyQualifiedName}.`);
1098
+ const senderContainer = sender.container;
1099
+ if (senderContainer) {
1100
+ sender = senderContainer;
1101
+ }
1102
+ }
1103
+ const receiverFullyQualifiedName = FQNFunctions.getFQN(invocable.getParent());
916
1104
  const receiver = this.famixRep.getFamixEntityByFullyQualifiedName(receiverFullyQualifiedName);
917
1105
  const fmxInvocation = new Famix.Invocation();
918
1106
  fmxInvocation.sender = sender;
919
1107
  fmxInvocation.receiver = receiver;
920
- fmxInvocation.addCandidate(fmxMethodOrFunction);
921
- fmxInvocation.signature = fmxMethodOrFunction.signature;
1108
+ fmxInvocation.addCandidate(fmxInvocable);
1109
+ fmxInvocation.signature = fmxInvocable.signature;
922
1110
  this.famixRep.addElement(fmxInvocation);
923
- this.fmxElementObjectMap.set(fmxInvocation, node);
1111
+ this.fmxElementObjectMap.set(fmxInvocation, nodeReferringToInvocable);
924
1112
  }
925
1113
  /**
926
1114
  * Creates a Famix inheritance
927
1115
  * @param cls A class or an interface (subclass)
928
1116
  * @param inhClass The inherited class or interface (superclass)
929
1117
  */
930
- createFamixInheritance(cls, inhClass) {
1118
+ createOrGetFamixInheritance(cls, inhClass) {
1119
+ // // need a key to see if the inheritance already exists
1120
+ // const classFullyQualifiedName = FQNFunctions.getFQN(cls);
1121
+ // let inKeyword: string;
1122
+ // let inhClassFullyQualifiedName: string;
1123
+ // let inhClassName: string | undefined;
1124
+ // // if inhClass is an ExpressionWithTypeArguments, it is an interface
1125
+ // if (inhClass instanceof ExpressionWithTypeArguments) {
1126
+ // inhClassName = inhClass.getExpression().getText();
1127
+ // // what is inhClassFullyQualifiedName? TODO
1128
+ // inhClassFullyQualifiedName = 'Undefined_Scope_from_importer.' + inhClassName;
1129
+ // } else {
1130
+ // inhClassName = inhClass.getName();
1131
+ // if (!inhClassName) {
1132
+ // throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);
1133
+ // }
1134
+ // inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);
1135
+ // }
1136
+ // // build the unique key of the inheritance
1137
+ // if ((cls instanceof ClassDeclaration && inhClass instanceof ClassDeclaration)
1138
+ // || (cls instanceof InterfaceDeclaration && inhClass instanceof InterfaceDeclaration)) {
1139
+ // inKeyword = " extends ";
1140
+ // } else if (cls instanceof ClassDeclaration && (inhClass instanceof InterfaceDeclaration || inhClass instanceof ExpressionWithTypeArguments)) {
1141
+ // inKeyword = " implements ";
1142
+ // } else {
1143
+ // throw new Error(`Inheritance ${cls.getText()} and ${inhClass.getText()} is not valid.`);
1144
+ // }
1145
+ // const inheritanceFullyQualifiedName = FQNFunctions.getFQN(cls) + inKeyword + inhClassFullyQualifiedName;
1146
+ // if (this.fmxInheritanceMap.has(inheritanceFullyQualifiedName)) {
1147
+ // const rInheritance = this.fmxInheritanceMap.get(inheritanceFullyQualifiedName);
1148
+ // if (rInheritance) {
1149
+ // return; // don't do anything
1150
+ // } else {
1151
+ // throw new Error(`Inheritance ${cls.getText()} is not found in the inheritance map.`);
1152
+ // }
1153
+ // }
931
1154
  const fmxInheritance = new Famix.Inheritance();
932
- // const clsName = cls.getName();
933
- const classFullyQualifiedName = FQNFunctions.getFQN(cls);
934
- analyze_1.logger.debug(`createFamixInheritance: classFullyQualifiedName: class fqn = ${classFullyQualifiedName}`);
1155
+ // logger.debug(`createFamixInheritance: classFullyQualifiedName: class fqn = ${classFullyQualifiedName}`);
935
1156
  let subClass;
936
1157
  if (cls instanceof ts_morph_1.ClassDeclaration) {
937
- subClass = this.fmxClassMap.get(classFullyQualifiedName);
1158
+ subClass = this.createOrGetFamixClass(cls);
938
1159
  }
939
1160
  else {
940
- subClass = this.fmxInterfaceMap.get(classFullyQualifiedName);
1161
+ subClass = this.createOrGetFamixInterface(cls);
941
1162
  }
942
1163
  if (!subClass) {
943
- throw new Error(`Subclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
1164
+ throw new Error(`Subclass ${cls} not found in Class or Interface maps.`);
944
1165
  }
945
- let inhClassName;
946
- let inhClassFullyQualifiedName;
947
1166
  let superClass;
948
- if (inhClass instanceof ts_morph_1.ClassDeclaration || inhClass instanceof ts_morph_1.InterfaceDeclaration) {
949
- inhClassName = inhClass.getName();
950
- if (!inhClassName) {
951
- throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);
952
- }
953
- inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);
954
- if (inhClass instanceof ts_morph_1.ClassDeclaration) {
955
- superClass = this.fmxClassMap.get(inhClassFullyQualifiedName);
956
- }
957
- else {
958
- superClass = this.fmxInterfaceMap.get(inhClassFullyQualifiedName);
959
- }
960
- if (!superClass) {
961
- throw new Error(`Superclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
962
- }
1167
+ // if (inhClass instanceof ClassDeclaration || inhClass instanceof InterfaceDeclaration) {
1168
+ // inhClassName = inhClass.getName();
1169
+ // if (!inhClassName) {
1170
+ // throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);
1171
+ // }
1172
+ // inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);
1173
+ // if (inhClass instanceof ClassDeclaration) {
1174
+ // superClass = this.fmxClassMap.get(inhClassFullyQualifiedName);
1175
+ // }
1176
+ // else {
1177
+ // superClass = this.fmxInterfaceMap.get(inhClassFullyQualifiedName);
1178
+ // }
1179
+ // if (!superClass) {
1180
+ // throw new Error(`Superclass ${classFullyQualifiedName} not found in Class or Interface maps.`);
1181
+ // }
1182
+ // }
1183
+ // // it shouldn't add the class/interface again to the Map, it should use createOrGet (?)
1184
+ // if (superClass === undefined) {
1185
+ if (inhClass instanceof ts_morph_1.ClassDeclaration) {
1186
+ // superClass = new Famix.Class();
1187
+ superClass = this.createOrGetFamixClass(inhClass);
1188
+ // this.fmxClassMap.set(inhClassFullyQualifiedName, superClass);
1189
+ }
1190
+ else if (inhClass instanceof ts_morph_1.InterfaceDeclaration) {
1191
+ // superClass = new Famix.Interface();
1192
+ superClass = this.createOrGetFamixInterface(inhClass);
1193
+ // this.fmxInterfaceMap.set(inhClassFullyQualifiedName, superClass);
963
1194
  }
964
1195
  else {
965
- // inhClass is an ExpressionWithTypeArguments
966
- inhClassName = inhClass.getExpression().getText();
967
- // what is inhClassFullyQualifiedName? TODO
968
- inhClassFullyQualifiedName = 'Undefined_Scope_from_importer.' + inhClassName;
969
- }
970
- if (superClass === undefined) {
971
- if (inhClass instanceof ts_morph_1.ClassDeclaration) {
972
- superClass = new Famix.Class();
973
- this.fmxClassMap.set(inhClassFullyQualifiedName, superClass);
1196
+ // inhClass instanceof ExpressionWithTypeArguments
1197
+ const interfaceDeclaration = getInterfaceDeclarationFromExpression(inhClass);
1198
+ if (interfaceDeclaration !== undefined) {
1199
+ superClass = this.createOrGetFamixInterface(interfaceDeclaration);
974
1200
  }
975
1201
  else {
976
- superClass = new Famix.Interface();
977
- this.fmxInterfaceMap.set(inhClassFullyQualifiedName, superClass);
1202
+ throw new Error(`Interface declaration not found for ${inhClass.getText()}.`);
978
1203
  }
979
- this.fmxElementObjectMap.set(superClass, inhClass);
980
- superClass.name = inhClassName;
981
- superClass.fullyQualifiedName = inhClassFullyQualifiedName;
982
- superClass.isStub = true;
983
- this.makeFamixIndexFileAnchor(inhClass, superClass);
984
- this.famixRep.addElement(superClass);
985
1204
  }
1205
+ this.fmxElementObjectMap.set(superClass, inhClass);
1206
+ this.makeFamixIndexFileAnchor(inhClass, superClass);
1207
+ this.famixRep.addElement(superClass);
986
1208
  fmxInheritance.subclass = subClass;
987
1209
  fmxInheritance.superclass = superClass;
988
1210
  this.famixRep.addElement(fmxInheritance);
1211
+ // no FQN for inheritance
989
1212
  // We don't map inheritance to the source code element because there are two elements (super, sub)
990
1213
  // this.fmxElementObjectMap.set(fmxInheritance, null);
991
1214
  }
@@ -1006,17 +1229,30 @@ class EntityDictionary {
1006
1229
  * @param isInExports A boolean indicating if the imported entity is in the exports
1007
1230
  * @param isDefaultExport A boolean indicating if the imported entity is a default export
1008
1231
  */
1009
- oldCreateFamixImportClause(importClauseInfo) {
1232
+ oldCreateOrGetFamixImportClause(importClauseInfo) {
1010
1233
  const { importDeclaration, importerSourceFile: importer, moduleSpecifierFilePath, importElement, isInExports, isDefaultExport } = importClauseInfo;
1011
- analyze_1.logger.debug(`createFamixImportClause: Creating import clause:`);
1234
+ if (importDeclaration && this.fmxImportClauseMap.has(importDeclaration)) {
1235
+ const rImportClause = this.fmxImportClauseMap.get(importDeclaration);
1236
+ if (rImportClause) {
1237
+ return; // don't do anything
1238
+ }
1239
+ else {
1240
+ throw new Error(`Import clause ${importElement.getText()} is not found in the import clause map.`);
1241
+ }
1242
+ }
1243
+ analyze_1.logger.info(`creating a new FamixImportClause for ${importDeclaration?.getText()} in ${importer.getBaseName()}.`);
1012
1244
  const fmxImportClause = new Famix.ImportClause();
1013
1245
  let importedEntity = undefined;
1014
1246
  let importedEntityName;
1015
1247
  const absolutePathProject = this.famixRep.getAbsolutePath();
1016
1248
  const absolutePath = path_1.default.normalize(moduleSpecifierFilePath);
1017
1249
  // convert the path and remove any windows backslashes introduced by path.normalize
1250
+ analyze_1.logger.debug(`createFamixImportClause: absolutePath: ${absolutePath}`);
1251
+ analyze_1.logger.debug(`createFamixImportClause: convertToRelativePath: ${this.convertToRelativePath(absolutePath, absolutePathProject)}`);
1018
1252
  const pathInProject = this.convertToRelativePath(absolutePath, absolutePathProject).replace(/\\/g, "/");
1253
+ analyze_1.logger.debug(`createFamixImportClause: pathInProject: ${pathInProject}`);
1019
1254
  let pathName = "{" + pathInProject + "}.";
1255
+ analyze_1.logger.debug(`createFamixImportClause: pathName: ${pathName}`);
1020
1256
  // Named imports, e.g. import { ClassW } from "./complexExportModule";
1021
1257
  // Start with simple import clause (without referring to the actual variable)
1022
1258
  if (importDeclaration instanceof ts_morph_1.ImportDeclaration
@@ -1032,7 +1268,9 @@ class EntityDictionary {
1032
1268
  if (!isInExports) {
1033
1269
  importedEntity.isStub = true;
1034
1270
  }
1035
- importedEntity.fullyQualifiedName = pathName;
1271
+ // logger.debug(`createFamixImportClause: Creating named entity ${importedEntityName} with FQN ${pathName}`);
1272
+ // importedEntity.fullyQualifiedName = pathName;
1273
+ initFQN(importElement, importedEntity);
1036
1274
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1037
1275
  // must add entity to repository
1038
1276
  this.famixRep.addElement(importedEntity);
@@ -1047,7 +1285,7 @@ class EntityDictionary {
1047
1285
  importedEntity.name = importedEntityName;
1048
1286
  initFQN(importDeclaration, importedEntity);
1049
1287
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1050
- importedEntity.fullyQualifiedName = pathName;
1288
+ // importedEntity.fullyQualifiedName = pathName;
1051
1289
  const anyType = this.createOrGetFamixType('any', importDeclaration);
1052
1290
  importedEntity.declaredType = anyType;
1053
1291
  }
@@ -1056,7 +1294,8 @@ class EntityDictionary {
1056
1294
  pathName = pathName + (isDefaultExport ? "defaultExport" : "namespaceExport");
1057
1295
  importedEntity = new Famix.NamedEntity();
1058
1296
  importedEntity.name = importedEntityName;
1059
- importedEntity.fullyQualifiedName = pathName;
1297
+ // importedEntity.fullyQualifiedName = pathName;
1298
+ initFQN(importElement, importedEntity);
1060
1299
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1061
1300
  }
1062
1301
  // I don't think it should be added to the repository if it exists already
@@ -1075,59 +1314,68 @@ class EntityDictionary {
1075
1314
  analyze_1.logger.debug(`createFamixImportClause: ${fmxImportClause.importedEntity?.name} (of type ${Helpers.getSubTypeName(fmxImportClause.importedEntity)}) is imported by ${fmxImportClause.importingEntity?.name}`);
1076
1315
  fmxImporter.addOutgoingImport(fmxImportClause);
1077
1316
  this.famixRep.addElement(fmxImportClause);
1078
- if (importDeclaration)
1317
+ if (importDeclaration) {
1079
1318
  this.fmxElementObjectMap.set(fmxImportClause, importDeclaration);
1319
+ this.fmxImportClauseMap.set(importDeclaration, fmxImportClause);
1320
+ }
1080
1321
  }
1081
1322
  /**
1082
1323
  * Creates a Famix Arrow Function
1083
1324
  * @param arrowExpression An Expression
1084
1325
  * @returns The Famix model of the variable
1085
1326
  */
1086
- createFamixArrowFunction(arrowExpression, currentCC) {
1327
+ createOrGetFamixArrowFunction(arrowExpression, currentCC) {
1087
1328
  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;
1329
+ const functionFullyQualifiedName = FQNFunctions.getFQN(arrowExpression);
1330
+ if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {
1331
+ const arrowFunction = arrowExpression.asKindOrThrow(ts_morph_1.SyntaxKind.ArrowFunction);
1332
+ const isGeneric = arrowFunction.getTypeParameters().length > 0;
1333
+ if (isGeneric) {
1334
+ fmxArrowFunction = new Famix.ParametricArrowFunction();
1335
+ }
1336
+ else {
1337
+ fmxArrowFunction = new Famix.ArrowFunction();
1338
+ }
1339
+ // Get the parent of the arrow function (the variable declaration)
1340
+ const parent = arrowFunction.getParentIfKind(ts_morph_1.SyntaxKind.VariableDeclaration);
1341
+ let functionName = '(NO_NAME)';
1342
+ if (parent && parent instanceof ts_morph_1.VariableDeclaration) {
1343
+ // Get the name of the variable
1344
+ functionName = parent.getName();
1345
+ }
1346
+ if (functionName) {
1347
+ fmxArrowFunction.name = functionName;
1348
+ }
1349
+ else {
1350
+ fmxArrowFunction.name = "anonymous";
1351
+ }
1352
+ // Signature of an arrow function is (parameters) => return_type
1353
+ const parametersSignature = arrowFunction.getParameters().map(p => p.getText()).join(", ");
1354
+ const returnTypeSignature = arrowFunction.getReturnType().getText();
1355
+ fmxArrowFunction.signature = `(${parametersSignature}) => ${returnTypeSignature}`;
1356
+ fmxArrowFunction.cyclomaticComplexity = currentCC[fmxArrowFunction.name];
1357
+ let functionTypeName = this.UNKNOWN_VALUE;
1358
+ try {
1359
+ functionTypeName = arrowFunction.getReturnType().getText().trim();
1360
+ }
1361
+ catch (error) {
1362
+ analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${functionName}. Continuing...`);
1363
+ }
1364
+ const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction);
1365
+ fmxArrowFunction.declaredType = fmxType;
1366
+ fmxArrowFunction.numberOfLinesOfCode = arrowFunction.getEndLineNumber() - arrowFunction.getStartLineNumber();
1367
+ const parameters = arrowFunction.getParameters();
1368
+ fmxArrowFunction.numberOfParameters = parameters.length;
1369
+ fmxArrowFunction.numberOfStatements = arrowFunction.getStatements().length;
1370
+ initFQN(arrowExpression, fmxArrowFunction);
1371
+ this.makeFamixIndexFileAnchor(arrowExpression, fmxArrowFunction);
1372
+ this.famixRep.addElement(fmxArrowFunction);
1373
+ this.fmxElementObjectMap.set(fmxArrowFunction, arrowFunction);
1374
+ this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxArrowFunction);
1105
1375
  }
1106
1376
  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();
1377
+ fmxArrowFunction = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName);
1117
1378
  }
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
1379
  return fmxArrowFunction;
1132
1380
  }
1133
1381
  /**
@@ -1135,297 +1383,421 @@ class EntityDictionary {
1135
1383
  * @param cls A class
1136
1384
  * @returns The Famix model of the concretisation
1137
1385
  */
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
- }
1386
+ // public createFamixConcretisation(conEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod ,genEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod): Famix.Concretisation {
1387
+ // const fmxConcretisation : Famix.Concretisation = new Famix.Concretisation();
1388
+ // fmxConcretisation.concreteEntity = conEntity;
1389
+ // fmxConcretisation.genericEntity = genEntity;
1390
+ // // this.fmxElementObjectMap.set(fmxConcretisation,null);
1391
+ // this.famixRep.addElement(fmxConcretisation);
1392
+ // const parameterConcretisation = this.createFamixParameterConcretisation(fmxConcretisation);
1393
+ // return fmxConcretisation;
1394
+ // }
1147
1395
  /**
1148
1396
  * Creates a Famix concretisation
1149
1397
  * @param concretisation A FamixConcretisation
1150
1398
  * @returns The Famix model of the ParameterConcrestisation
1151
1399
  */
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
- }
1400
+ // public createFamixParameterConcretisation(concretisation: Famix.Concretisation): Famix.ParameterConcretisation | undefined{
1401
+ // const conClass = concretisation.concreteEntity;
1402
+ // const genClass = concretisation.genericEntity;
1403
+ // logger.debug(`Creating parameter concretisation between ${conClass.fullyQualifiedName} and ${genClass.fullyQualifiedName}`);
1404
+ // const parameterConcretisations = this.famixRep._getAllEntitiesWithType("ParameterConcretisation") as Set<Famix.ParameterConcretisation>;
1405
+ // const concreteParameters = conClass.concreteParameters;
1406
+ // const genericParameters = genClass.genericParameters;
1407
+ // let conClassTypeParametersIterator = concreteParameters.values();
1408
+ // let genClassTypeParametersIterator = genericParameters.values();
1409
+ // let fmxParameterConcretisation : Famix.ParameterConcretisation | undefined = undefined;
1410
+ // for (let i = 0; i < genericParameters.size; i++) {
1411
+ // const conClassTypeParameter = conClassTypeParametersIterator.next().value as Famix.ParameterType;
1412
+ // const genClassTypeParameter = genClassTypeParametersIterator.next().value as Famix.ParameterType;
1413
+ // let createParameterConcretisation : boolean = true;
1414
+ // if(conClassTypeParameter && genClassTypeParameter && conClassTypeParameter.name != genClassTypeParameter.name){
1415
+ // parameterConcretisations.forEach((param : Famix.ParameterConcretisation) => {
1416
+ // if (conClassTypeParameter.name == param.concreteParameter.name && genClassTypeParameter.name == param.genericParameter.name) {
1417
+ // createParameterConcretisation = false;
1418
+ // fmxParameterConcretisation = param;
1419
+ // }
1420
+ // })
1421
+ // if (createParameterConcretisation) {
1422
+ // fmxParameterConcretisation = new Famix.ParameterConcretisation();
1423
+ // fmxParameterConcretisation.genericParameter = genClassTypeParameter;
1424
+ // fmxParameterConcretisation.concreteParameter = conClassTypeParameter;
1425
+ // fmxParameterConcretisation.addConcretisation(concretisation);
1426
+ // // this.fmxElementObjectMap.set(fmxParameterConcretisation,null);
1427
+ // } else {
1428
+ // if (!fmxParameterConcretisation) {
1429
+ // throw new Error(`fmxParameterConcretisation was undefined for concretisation with generic parameter ${genClassTypeParameter.name} and concrete parameter ${conClassTypeParameter.name}`);
1430
+ // }
1431
+ // fmxParameterConcretisation.addConcretisation(concretisation);
1432
+ // }
1433
+ // this.famixRep.addElement(fmxParameterConcretisation);
1434
+ // }
1435
+ // }
1436
+ // if (!fmxParameterConcretisation) {
1437
+ // logger.error(`fmxParameterConcretisation was undefined for concretisation with concrete entity ${conClass.fullyQualifiedName} and generic entity ${genClass.fullyQualifiedName}`);
1438
+ // }
1439
+ // return fmxParameterConcretisation;
1440
+ // }
1194
1441
  /**
1195
1442
  * Creates a Famix concretisation between two classes or two interfaces
1196
1443
  * @param element A class or an Interface
1197
1444
  */
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
- }
1445
+ // public createFamixConcretisationClassOrInterfaceSpecialisation(element: ClassDeclaration | InterfaceDeclaration){
1446
+ // const superEntity = element.getExtends();
1447
+ // let superEntityArray;
1448
+ // if (superEntity){
1449
+ // superEntityArray = Array.isArray(superEntity) ? superEntity : [superEntity];
1450
+ // }
1451
+ // if (superEntityArray && superEntityArray.length > 0) {
1452
+ // superEntityArray.forEach(entity => {
1453
+ // let entityIsGeneric;
1454
+ // const superEntitySymbol = entity.getExpression().getSymbolOrThrow();
1455
+ // let superEntityDeclaration;
1456
+ // if (superEntity instanceof ExpressionWithTypeArguments) {
1457
+ // superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.ClassDeclaration);
1458
+ // } else {
1459
+ // superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.InterfaceDeclaration);
1460
+ // }
1461
+ // if (superEntityDeclaration) {
1462
+ // entityIsGeneric = superEntityDeclaration.getTypeParameters().length > 0;
1463
+ // }
1464
+ // if (entityIsGeneric) {
1465
+ // let EntityDeclaration;
1466
+ // let genEntity;
1467
+ // if (superEntity instanceof ExpressionWithTypeArguments) {
1468
+ // EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as ClassDeclaration;
1469
+ // genEntity = this.createOrGetFamixClass(EntityDeclaration) as Famix.ParametricClass;
1470
+ // } else {
1471
+ // EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;
1472
+ // genEntity = this.createOrGetFamixInterface(EntityDeclaration) as Famix.ParametricInterface;
1473
+ // }
1474
+ // const genParams = EntityDeclaration.getTypeParameters().map((param) => param.getText());
1475
+ // const args = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments()
1476
+ // const conParams = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1477
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1478
+ // let conEntity;
1479
+ // conEntity = this.createOrGetFamixConcreteElement(genEntity,EntityDeclaration,args);
1480
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1481
+ // let createConcretisation : boolean = true;
1482
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1483
+ // if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){
1484
+ // createConcretisation = false;
1485
+ // }
1486
+ // });
1487
+ // if (createConcretisation) {
1488
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);
1489
+ // }
1490
+ // }
1491
+ // }
1492
+ // });
1493
+ // }
1494
+ // // TODO: This function seems unfinished
1495
+ // }
1251
1496
  /**
1252
1497
  * Creates a Famix concretisation between a class and its instanciations
1253
1498
  * @param cls A class
1254
1499
  */
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
- }
1500
+ // public createFamixConcretisationGenericInstantiation(cls: ClassDeclaration){
1501
+ // const isGeneric = cls.getTypeParameters().length > 0;
1502
+ // if (isGeneric) {
1503
+ // const instances = cls.getSourceFile().getDescendantsOfKind(ts.SyntaxKind.NewExpression)
1504
+ // .filter(newExpr => {
1505
+ // const expression = newExpr.getExpression();
1506
+ // return expression.getText() === cls.getName();
1507
+ // });
1508
+ // instances.forEach(instance => {
1509
+ // const instanceIsGeneric = instance.getTypeArguments().length > 0;
1510
+ // if (instanceIsGeneric) {
1511
+ // const conParams = instance.getTypeArguments().map((param) => param.getText());
1512
+ // const genEntity = this.createOrGetFamixClass(cls) as Famix.ParametricClass;
1513
+ // const genParams = cls.getTypeParameters().map((param) => param.getText());
1514
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1515
+ // let conEntity;
1516
+ // conEntity = this.createOrGetFamixConcreteElement(genEntity,cls,instance.getTypeArguments());
1517
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1518
+ // let createConcretisation : boolean = true;
1519
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1520
+ // if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){
1521
+ // createConcretisation = false;
1522
+ // }
1523
+ // });
1524
+ // if (createConcretisation) {
1525
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);
1526
+ // }
1527
+ // }
1528
+ // }
1529
+ // })
1530
+ // }
1531
+ // // TODO: This function seems unfinished
1532
+ // }
1288
1533
  /**
1289
1534
  * Creates a Famix concretisation between a class and its instanciations
1290
1535
  * @param func A function
1291
1536
  */
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
- }
1537
+ // public createFamixConcretisationFunctionInstantiation(element: FunctionDeclaration | MethodDeclaration){
1538
+ // const isGeneric = element.getTypeParameters().length > 0;
1539
+ // if (isGeneric) {
1540
+ // const genParams = element.getTypeParameters().map(param => param.getText());
1541
+ // const uses = element.findReferencesAsNodes();
1542
+ // uses.forEach(usage => {
1543
+ // let currentNode: Node | undefined = usage;
1544
+ // while (currentNode) {
1545
+ // if (currentNode.getKind() === SyntaxKind.CallExpression) {
1546
+ // const callExpression = currentNode.asKind(SyntaxKind.CallExpression);
1547
+ // if (!callExpression) {
1548
+ // throw new Error(`CallExpression not found for ${currentNode.getText()}`);
1549
+ // }
1550
+ // const instanceIsGeneric = callExpression.getTypeArguments().length > 0;
1551
+ // if (instanceIsGeneric) {
1552
+ // const args = callExpression.getTypeArguments();
1553
+ // const conParams = callExpression.getTypeArguments().map(param => param.getText());
1554
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1555
+ // let genElement;
1556
+ // if(element instanceof FunctionDeclaration){
1557
+ // genElement = this.createOrGetFamixFunction(element, {}) as Famix.ParametricFunction;
1558
+ // } else {
1559
+ // genElement = this.createOrGetFamixMethod(element, {}) as Famix.ParametricMethod;
1560
+ // }
1561
+ // let concElement;
1562
+ // concElement = this.createOrGetFamixConcreteElement(genElement,element,args);
1563
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1564
+ // let createConcretisation : boolean = true;
1565
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1566
+ // if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){
1567
+ // createConcretisation = false;
1568
+ // }
1569
+ // });
1570
+ // if (createConcretisation) {
1571
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);
1572
+ // }
1573
+ // }
1574
+ // }
1575
+ // break;
1576
+ // }
1577
+ // // Remonter à l'élément parent (utile si le nœud de référence est un enfant)
1578
+ // currentNode = currentNode.getParent();
1579
+ // }
1580
+ // });
1581
+ // }
1582
+ // }
1339
1583
  /**
1340
1584
  * Creates a Famix concretisation between a class and an interface
1341
1585
  * @param cls A class
1342
1586
  */
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
- }
1587
+ // public createFamixConcretisationInterfaceClass(cls: ClassDeclaration){
1588
+ // const superInterfaces = cls.getImplements();
1589
+ // superInterfaces.forEach(interfaceType => {
1590
+ // const interfaceIsGeneric = interfaceType.getTypeArguments().length>0;
1591
+ // if (interfaceIsGeneric) {
1592
+ // const interfaceDeclaration = interfaceType.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;
1593
+ // const genParams = interfaceDeclaration.getTypeParameters().map((param) => param.getText());
1594
+ // const conParams = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1595
+ // const args = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1596
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1597
+ // const genInterface = this.createOrGetFamixInterface(interfaceDeclaration) as Famix.ParametricInterface;
1598
+ // const conInterface = this.createOrGetFamixConcreteElement(genInterface,interfaceDeclaration,args);
1599
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1600
+ // let createConcretisation : boolean = true;
1601
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1602
+ // if (genInterface.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conInterface.fullyQualifiedName){
1603
+ // createConcretisation = false;
1604
+ // }
1605
+ // });
1606
+ // if (createConcretisation) {
1607
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conInterface,genInterface);
1608
+ // }
1609
+ // }
1610
+ // }
1611
+ // });
1612
+ // }
1369
1613
  /**
1370
1614
  * Creates a Famix concretisation between an interface and a Type
1371
1615
  * @param element A variable or a function
1372
1616
  * @param inter An interface
1373
1617
  */
1374
- createFamixConcretisationTypeInstanciation(element) {
1375
- const isGeneric = element.getTypeParameters().length > 0;
1376
- if (isGeneric) {
1377
- const genParams = element.getTypeParameters().map(param => param.getText());
1378
- const uses = element.findReferencesAsNodes();
1379
- uses.forEach(use => {
1380
- let parentNode = use.getParent();
1381
- while (parentNode) {
1382
- if (parentNode.getKind() === ts_morph_1.SyntaxKind.TypeReference) {
1383
- const typeReferenceNode = parentNode.asKind(ts_morph_1.SyntaxKind.TypeReference);
1384
- if (!typeReferenceNode) {
1385
- throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);
1386
- }
1387
- const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;
1388
- if (typeReferenceNodeIsGeneric) { }
1389
- const args = typeReferenceNode.getTypeArguments();
1390
- const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());
1391
- if (!Helpers.arraysAreEqual(conParams, genParams)) {
1392
- let genElement;
1393
- if (element instanceof ts_morph_1.ClassDeclaration) {
1394
- genElement = this.createOrGetFamixClass(element);
1395
- }
1396
- else {
1397
- genElement = this.createOrGetFamixInterface(element);
1398
- }
1399
- let concElement;
1400
- concElement = this.createOrGetFamixConcreteElement(genElement, element, args);
1401
- const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1402
- let createConcretisation = true;
1403
- concretisations.forEach((conc) => {
1404
- if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName) {
1405
- createConcretisation = false;
1406
- }
1407
- });
1408
- if (createConcretisation) {
1409
- const fmxConcretisation = this.createFamixConcretisation(concElement, genElement);
1410
- }
1411
- }
1412
- break;
1413
- }
1414
- parentNode = parentNode.getParent();
1415
- }
1416
- });
1417
- }
1418
- }
1618
+ // public createFamixConcretisationTypeInstanciation(element: InterfaceDeclaration | ClassDeclaration){
1619
+ // const isGeneric = element.getTypeParameters().length > 0;
1620
+ // if (isGeneric) {
1621
+ // const genParams = element.getTypeParameters().map(param => param.getText());
1622
+ // const uses = element.findReferencesAsNodes();
1623
+ // uses.forEach(use => {
1624
+ // let parentNode = use.getParent();
1625
+ // while (parentNode) {
1626
+ // if (parentNode.getKind() === SyntaxKind.TypeReference) {
1627
+ // const typeReferenceNode = parentNode.asKind(SyntaxKind.TypeReference);
1628
+ // if (!typeReferenceNode) {
1629
+ // throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);
1630
+ // }
1631
+ // const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;
1632
+ // if (typeReferenceNodeIsGeneric) {}
1633
+ // const args = typeReferenceNode.getTypeArguments();
1634
+ // const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());
1635
+ // if (!Helpers.arraysAreEqual(conParams,genParams)) {
1636
+ // let genElement;
1637
+ // if(element instanceof ClassDeclaration){
1638
+ // genElement = this.createOrGetFamixClass(element) as Famix.ParametricClass;
1639
+ // } else {
1640
+ // genElement = this.createOrGetFamixInterface(element) as Famix.ParametricInterface;
1641
+ // }
1642
+ // let concElement;
1643
+ // concElement = this.createOrGetFamixConcreteElement(genElement,element,args);
1644
+ // const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation") as Set<Famix.Concretisation>;
1645
+ // let createConcretisation : boolean = true;
1646
+ // concretisations.forEach((conc : Famix.Concretisation) => {
1647
+ // if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){
1648
+ // createConcretisation = false;
1649
+ // }
1650
+ // });
1651
+ // if (createConcretisation) {
1652
+ // const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);
1653
+ // }
1654
+ // }
1655
+ // break;
1656
+ // }
1657
+ // parentNode = parentNode.getParent();
1658
+ // }
1659
+ // });
1660
+ // }
1661
+ // }
1419
1662
  convertToRelativePath(absolutePath, absolutePathProject) {
1420
- return absolutePath.replace(absolutePathProject, "").slice(1);
1663
+ analyze_1.logger.debug(`convertToRelativePath: absolutePath: '${absolutePath}', absolutePathProject: '${absolutePathProject}'`);
1664
+ if (absolutePath.startsWith(absolutePathProject)) {
1665
+ return absolutePath.replace(absolutePathProject, "").slice(1);
1666
+ }
1667
+ else if (absolutePath.startsWith("/")) {
1668
+ return absolutePath.slice(1);
1669
+ }
1670
+ else {
1671
+ return absolutePath;
1672
+ }
1421
1673
  }
1422
1674
  }
1423
1675
  exports.EntityDictionary = EntityDictionary;
1676
+ function isPrimitiveType(typeName) {
1677
+ return typeName === "number" ||
1678
+ typeName === "string" ||
1679
+ typeName === "boolean" ||
1680
+ typeName === "bigint" ||
1681
+ typeName === "symbol" ||
1682
+ typeName === "undefined" ||
1683
+ typeName === "null" ||
1684
+ typeName === "any" ||
1685
+ typeName === "unknown" ||
1686
+ typeName === "never" ||
1687
+ typeName === "void";
1688
+ }
1424
1689
  function initFQN(sourceElement, famixElement) {
1690
+ // handle special cases where an element is a Type -- need to change its name
1691
+ if (famixElement instanceof Famix.Type && !(sourceElement instanceof ts_morph_1.CommentRange) && isTypeContext(sourceElement)) {
1692
+ let fqn = FQNFunctions.getFQN(sourceElement);
1693
+ // using regex, replace [blah] with [blahType]
1694
+ fqn = fqn.replace(/\[([^\]]+)\]/g, "[$1Type]");
1695
+ analyze_1.logger.debug("Setting fully qualified name for " + famixElement.getJSON() + " to " + fqn);
1696
+ famixElement.fullyQualifiedName = fqn;
1697
+ return;
1698
+ }
1699
+ // catch all (except comments)
1425
1700
  if (!(sourceElement instanceof ts_morph_1.CommentRange)) {
1426
1701
  const fqn = FQNFunctions.getFQN(sourceElement);
1427
1702
  analyze_1.logger.debug("Setting fully qualified name for " + famixElement.getJSON() + " to " + fqn);
1428
1703
  famixElement.fullyQualifiedName = fqn;
1429
1704
  }
1430
1705
  }
1431
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"EntityDictionary.js","sourceRoot":"","sources":["../../src/famix_functions/EntityDictionary.ts"],"names":[],"mappings":"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","sourcesContent":["import { ClassDeclaration, ConstructorDeclaration, FunctionDeclaration, Identifier, InterfaceDeclaration, MethodDeclaration, MethodSignature, ModuleDeclaration, PropertyDeclaration, PropertySignature, SourceFile, TypeParameterDeclaration, VariableDeclaration, ParameterDeclaration, Decorator, GetAccessorDeclaration, SetAccessorDeclaration, ImportSpecifier, CommentRange, EnumDeclaration, EnumMember, TypeAliasDeclaration, FunctionExpression, ExpressionWithTypeArguments, ImportDeclaration, ImportEqualsDeclaration, SyntaxKind, Expression, TypeNode, Node, ts, Scope, Type, ArrowFunction } from \"ts-morph\";\nimport { isAmbient, isNamespace } from \"../analyze_functions/process_functions\";\nimport * as Famix from \"../lib/famix/model/famix\";\nimport { FamixRepository } from \"../lib/famix/famix_repository\";\nimport { logger, config } from \"../analyze\";\nimport GraphemeSplitter from \"grapheme-splitter\";\nimport * as Helpers from \"./helpers_creation\";\nimport * as FQNFunctions from \"../fqn\";\nimport path from \"path\";\nimport _ from 'lodash';\n\nexport type TSMorphObjectType = ImportDeclaration | ImportEqualsDeclaration | SourceFile | ModuleDeclaration | ClassDeclaration | InterfaceDeclaration | MethodDeclaration | ConstructorDeclaration | MethodSignature | FunctionDeclaration | FunctionExpression | ParameterDeclaration | VariableDeclaration | PropertyDeclaration | PropertySignature | TypeParameterDeclaration | Identifier | Decorator | GetAccessorDeclaration | SetAccessorDeclaration | ImportSpecifier | CommentRange | EnumDeclaration | EnumMember | TypeAliasDeclaration | ExpressionWithTypeArguments;\n\nexport type TypeDeclaration = TypeAliasDeclaration | PropertyDeclaration | PropertySignature | MethodDeclaration | ConstructorDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration | FunctionDeclaration | FunctionExpression | ParameterDeclaration | VariableDeclaration | EnumMember | ImportEqualsDeclaration;\n\ntype ParametricVariantType = Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod;\n\ntype ConcreteElementTSMorphType = ClassDeclaration | InterfaceDeclaration | FunctionDeclaration | MethodDeclaration;\n\nexport class EntityDictionary {\n    \n    public famixRep = new FamixRepository();\n    private fmxAliasMap = new Map<string, Famix.Alias>(); // Maps the alias names to their Famix model\n    private fmxClassMap = new Map<string, Famix.Class | Famix.ParametricClass>(); // Maps the fully qualified class names to their Famix model\n    private fmxInterfaceMap = new Map<string, Famix.Interface | Famix.ParametricInterface>(); // Maps the interface names to their Famix model\n    private fmxModuleMap = new Map<string, Famix.Module>(); // Maps the namespace names to their Famix model\n    private fmxFileMap = new Map<string, Famix.ScriptEntity | Famix.Module>(); // Maps the source file names to their Famix model\n    private fmxTypeMap = new Map<string, Famix.Type | Famix.PrimitiveType | Famix.ParameterType>(); // Maps the type names to their Famix model\n    private fmxFunctionAndMethodMap = new Map<string, Famix.Function | Famix.ParametricFunction | Famix.Method | Famix.ParametricMethod> // Maps the function names to their Famix model\n    private UNKNOWN_VALUE = '(unknown due to parsing error)'; // The value to use when a name is not usable\n    public fmxElementObjectMap = new Map<Famix.Entity,TSMorphObjectType>();\n    public tsMorphElementObjectMap = new Map<TSMorphObjectType,Famix.Entity>();\n            \n    constructor() {\n        this.famixRep.setFmxElementObjectMap(this.fmxElementObjectMap);      \n    }\n\n    public addSourceAnchor(fmx: Famix.SourcedEntity, node: TSMorphObjectType): Famix.IndexedFileAnchor {\n        const sourceAnchor: Famix.IndexedFileAnchor = new Famix.IndexedFileAnchor();\n        let sourceStart, sourceEnd: number;\n        if (fmx && node) {\n            // find the start and end positions of the source element\n            if (!(node instanceof CommentRange)) {\n                sourceStart = node.getStart();\n                sourceEnd = node.getEnd();\n            } else {\n                sourceStart = node.getPos();\n                sourceEnd = node.getEnd();\n            }\n    \n            if (config.expectGraphemes) {\n                /**\n                 * The following logic handles the case of multi-code point characters (e.g. emoji) in the source text.\n                 * This is needed because Pharo/Smalltalk treats multi-code point characters as a single character, \n                 * but JavaScript treats them as multiple characters. This means that the start and end positions\n                 * of a source element in Pharo/Smalltalk will be different than the start and end positions of the\n                 * same source element in JavaScript. This logic finds the start and end positions of the source\n                 * element in JavaScript and then uses those positions to set the start and end positions of the\n                 * Famix index file anchor.\n                 * It depends on code in the 'grapheme-splitter' package in npm.\n                 */\n                const splitter = new GraphemeSplitter();\n                const sourceFileText = node.getSourceFile().getFullText();\n                const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;\n                if (hasGraphemeClusters) {\n                    const sourceElementText = sourceFileText.substring(sourceStart, sourceEnd);\n                    const sourceElementTextGraphemes = splitter.splitGraphemes(sourceElementText);\n                    const sourceFileTextGraphemes = splitter.splitGraphemes(sourceFileText);\n                    const numberOfGraphemeClustersBeforeStart = splitter.countGraphemes(sourceFileText.substring(0, sourceStart));\n        \n                    // find the start of the sourceElementTextGraphemes array in the sourceFileTextGraphemes array\n                    sourceStart = Helpers.indexOfSplitArray({searchArray: sourceFileTextGraphemes, \n                                                        targetArray: sourceElementTextGraphemes, \n                                                        start: sourceStart - numberOfGraphemeClustersBeforeStart});\n                    sourceEnd = sourceStart + sourceElementTextGraphemes.length;\n                } \n            }\n\n            // The +1 is because the source anchor (Pharo) is 1-based, but ts-morph is 0-based\n            sourceAnchor.startPos = sourceStart + 1;\n            sourceAnchor.endPos = sourceEnd + 1;\n\n            const fileName = node.getSourceFile().getFilePath();\n\n            sourceAnchor.element = fmx;\n            sourceAnchor.fileName = fileName;\n            fmx.sourceAnchor = sourceAnchor;\n            this.famixRep.addElement(sourceAnchor);\n\n        }\n        return sourceAnchor;\n    }\n\n    /**\n     * Makes a Famix index file anchor\n     * @param sourceElement A source element\n     * @param famixElement The Famix model of the source element\n     */\n    public makeFamixIndexFileAnchor(sourceElement: TSMorphObjectType, famixElement: Famix.SourcedEntity): void {\n        // check if famixElement doesn't have a valid fullyQualifiedName\n        if (typeof (famixElement as any).getFullyQualifiedName === 'function') {\n            // The method exists\n            const fullyQualifiedName = (famixElement as any).fullyQualifiedName;\n            if (!fullyQualifiedName || fullyQualifiedName === this.UNKNOWN_VALUE) {\n                throw new Error(`Famix element ${famixElement.constructor.name} has no valid fullyQualifiedName.`);\n            }\n        }\n\n        logger.debug(\"making index file anchor for '\" + sourceElement?.getText() + \"' with famixElement \" + famixElement.getJSON());\n        const fmxIndexFileAnchor = new Famix.IndexedFileAnchor();\n        fmxIndexFileAnchor.element = famixElement;\n        this.fmxElementObjectMap.set(famixElement, sourceElement);\n\n        if (sourceElement !== null) {\n            const absolutePathProject = this.famixRep.getAbsolutePath();\n        \n            const absolutePath = path.normalize(sourceElement.getSourceFile().getFilePath());\n\n            const positionNodeModules = absolutePath.indexOf('node_modules');\n\n            let pathInProject: string = \"\";\n\n            if (positionNodeModules !== -1) {\n                const pathFromNodeModules = absolutePath.substring(positionNodeModules);\n                pathInProject = pathFromNodeModules;\n            } else {\n                pathInProject = this.convertToRelativePath(absolutePath, absolutePathProject);\n            }\n\n            // revert any backslashes to forward slashes (path.normalize on windows introduces them)\n            pathInProject = pathInProject.replace(/\\\\/g, \"/\");\n\n            fmxIndexFileAnchor.fileName = pathInProject;\n            let sourceStart, sourceEnd, sourceLineStart, sourceLineEnd: number;\n            if (!(sourceElement instanceof CommentRange)) {\n                sourceStart = sourceElement.getStart();\n                sourceEnd = sourceElement.getEnd();\n                sourceLineStart = sourceElement.getStartLineNumber();\n                sourceLineEnd = sourceElement.getEndLineNumber();\n            } else {\n                sourceStart = sourceElement.getPos();\n                sourceEnd = sourceElement.getEnd();\n            }\n            if (config.expectGraphemes) {\n                /**\n                 * The following logic handles the case of multi-code point characters (e.g. emoji) in the source text.\n                 * This is needed because Pharo/Smalltalk treats multi-code point characters as a single character, \n                 * but JavaScript treats them as multiple characters. This means that the start and end positions\n                 * of a source element in Pharo/Smalltalk will be different than the start and end positions of the\n                 * same source element in JavaScript. This logic finds the start and end positions of the source\n                 * element in JavaScript and then uses those positions to set the start and end positions of the\n                 * Famix index file anchor.\n                 * It depends on code in the 'grapheme-splitter' package in npm.\n                 */\n                const splitter = new GraphemeSplitter();\n                const sourceFileText = sourceElement.getSourceFile().getFullText();\n                const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;\n                if (hasGraphemeClusters) {\n                    const sourceElementText = sourceFileText.substring(sourceStart, sourceEnd);\n                    const sourceElementTextGraphemes = splitter.splitGraphemes(sourceElementText);\n                    const sourceFileTextGraphemes = splitter.splitGraphemes(sourceFileText);\n                    const numberOfGraphemeClustersBeforeStart = splitter.countGraphemes(sourceFileText.substring(0, sourceStart));\n        \n                    // find the start of the sourceElementTextGraphemes array in the sourceFileTextGraphemes array\n                    sourceStart = Helpers.indexOfSplitArray({searchArray: sourceFileTextGraphemes, \n                                                        targetArray: sourceElementTextGraphemes, \n                                                        start: sourceStart - numberOfGraphemeClustersBeforeStart});\n                    sourceEnd = sourceStart + sourceElementTextGraphemes.length;\n                } \n            }\n            // note: the +1 is because the source anchor is 1-based, but ts-morph is 0-based\n            fmxIndexFileAnchor.startPos = sourceStart + 1;\n            fmxIndexFileAnchor.endPos = sourceEnd + 1;\n\n            // 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)) {\n            //    initFQN(sourceElement, famixElement);\n            // }\n        } else {\n            // sourceElement is null\n            logger.warn(\"sourceElement is null for famixElement \" + famixElement.getJSON());\n            fmxIndexFileAnchor.fileName = \"unknown\";\n            fmxIndexFileAnchor.startPos = 0;\n            fmxIndexFileAnchor.endPos = 0;\n        }\n\n        this.famixRep.addElement(fmxIndexFileAnchor);\n    }\n\n    /**\n     * Creates or gets a Famix script entity or module\n     * @param f A source file\n     * @param isModule A boolean indicating if the source file is a module\n     * @returns The Famix model of the source file\n     */\n    public createOrGetFamixFile(f: SourceFile, isModule: boolean): Famix.ScriptEntity | Famix.Module {\n        let fmxFile: Famix.ScriptEntity; // | Famix.Module;\n\n        const fileName = f.getBaseName();\n        const fullyQualifiedFilename = f.getFilePath();\n        const foundFileName = this.fmxFileMap.get(fullyQualifiedFilename);\n        if (!foundFileName) {\n            if (isModule) {\n                fmxFile = new Famix.Module();\n            }\n            else {\n                fmxFile = new Famix.ScriptEntity();\n            }\n            fmxFile.name = fileName;\n            fmxFile.numberOfLinesOfText = f.getEndLineNumber() - f.getStartLineNumber();\n            fmxFile.numberOfCharacters = f.getFullText().length;\n\n            initFQN(f, fmxFile);\n\n            this.makeFamixIndexFileAnchor(f, fmxFile);\n\n            this.fmxFileMap.set(fullyQualifiedFilename, fmxFile);\n            this.famixRep.addElement(fmxFile);\n        }\n        else {\n            fmxFile = foundFileName;\n        }\n\n        this.fmxElementObjectMap.set(fmxFile,f);\n        return fmxFile;\n    }\n\n    /**\n     * Creates or gets a Famix Module\n     * @param m A module\n     * @returns The Famix model of the module\n     */\n    public createOrGetFamixModule(m: ModuleDeclaration): Famix.Module {\n        let fmxModule: Famix.Module;\n        const moduleName = m.getName();\n        const foundModuleName = this.fmxModuleMap.get(moduleName);\n        if (!foundModuleName) {\n            fmxModule = new Famix.Module();\n            fmxModule.name = moduleName;\n            fmxModule.isAmbient = isAmbient(m);\n            fmxModule.isNamespace = isNamespace(m);\n            fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;\n\n            initFQN(m, fmxModule);\n            this.makeFamixIndexFileAnchor(m, fmxModule);\n\n            this.fmxModuleMap.set(moduleName, fmxModule);\n\n            this.famixRep.addElement(fmxModule);\n        }\n        else {\n            fmxModule = foundModuleName;\n        }\n\n        this.fmxElementObjectMap.set(fmxModule,m);\n        return fmxModule;\n    }\n\n    /**\n     * Creates a Famix alias\n     * @param a An alias\n     * @returns The Famix model of the alias\n     */\n    public createFamixAlias(a: TypeAliasDeclaration): Famix.Alias {\n        let fmxAlias: Famix.Alias;\n        const aliasName = a.getName();\n        const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);\n        const foundAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);\n        if (!foundAlias) {\n            fmxAlias = new Famix.Alias();\n            fmxAlias.name = a.getName();\n            const aliasNameWithGenerics = aliasName + (a.getTypeParameters().length ? (\"<\" + a.getTypeParameters().map(tp => tp.getName()).join(\", \") + \">\") : \"\");\n            logger.debug(`> NOTE: alias ${aliasName} has fully qualified name ${aliasFullyQualifiedName} and name with generics ${aliasNameWithGenerics}.`);\n\n            const fmxType = this.createOrGetFamixType(aliasNameWithGenerics, a);\n            fmxAlias.aliasedEntity = fmxType;\n            initFQN(a, fmxAlias);\n            this.makeFamixIndexFileAnchor(a, fmxAlias);\n\n            this.fmxAliasMap.set(aliasFullyQualifiedName, fmxAlias);\n\n            this.famixRep.addElement(fmxAlias);\n        }\n        else {\n            fmxAlias = foundAlias;\n        }\n        this.fmxElementObjectMap.set(fmxAlias,a);\n\n        return fmxAlias;\n    }\n\n    /**\n     * Creates or gets a Famix class or parameterizable class\n     * @param cls A class\n     * @returns The Famix model of the class\n     */\n    public createOrGetFamixClass(cls: ClassDeclaration): Famix.Class | Famix.ParametricClass {\n        let fmxClass: Famix.Class | Famix.ParametricClass;\n        const isAbstract = cls.isAbstract();\n        const classFullyQualifiedName = FQNFunctions.getFQN(cls);\n        const clsName = cls.getName() || this.UNKNOWN_VALUE;\n        const isGeneric = cls.getTypeParameters().length;\n        const foundClass = this.fmxClassMap.get(classFullyQualifiedName);\n        if (!foundClass) {\n            if (isGeneric) {\n                fmxClass = new Famix.ParametricClass();\n            }\n            else {\n                fmxClass = new Famix.Class();\n            }\n\n            fmxClass.name = clsName;\n            fmxClass.fullyQualifiedName = classFullyQualifiedName;\n            fmxClass.isAbstract = isAbstract;\n\n            this.makeFamixIndexFileAnchor(cls, fmxClass);\n\n            this.fmxClassMap.set(classFullyQualifiedName, fmxClass);\n\n            this.famixRep.addElement(fmxClass);\n\n            this.fmxElementObjectMap.set(fmxClass,cls);\n        }\n        else {\n            fmxClass = foundClass;\n        }\n\n        return fmxClass;\n    }\n\n    /**\n     * Creates or gets a Famix interface or parameterizable interface\n     * @param inter An interface\n     * @returns The Famix model of the interface\n     */\n    public createOrGetFamixInterface(inter: InterfaceDeclaration): Famix.Interface | Famix.ParametricInterface {\n\n        let fmxInterface: Famix.Interface | Famix.ParametricInterface;\n        const interName = inter.getName();\n        const interFullyQualifiedName = FQNFunctions.getFQN(inter);\n        const foundInterface = this.fmxInterfaceMap.get(interFullyQualifiedName);\n        if (!foundInterface) {\n            const isGeneric = inter.getTypeParameters().length;\n            if (isGeneric) {\n                fmxInterface = new Famix.ParametricInterface();\n            }\n            else {\n                fmxInterface = new Famix.Interface();\n            }\n\n            fmxInterface.name = interName;\n            initFQN(inter, fmxInterface);\n            this.makeFamixIndexFileAnchor(inter, fmxInterface);\n\n            this.fmxInterfaceMap.set(interFullyQualifiedName, fmxInterface);\n\n            this.famixRep.addElement(fmxInterface);\n\n            this.fmxElementObjectMap.set(fmxInterface,inter);\n        }\n        else {\n            fmxInterface = foundInterface;\n        }\n        return fmxInterface;\n    }\n\n    \n    /**\n     * Creates or gets a Famix concrete element\n     * @param concreteElement A parametric Element   \n     * @param concreteElementDeclaration the element declaration\n     * @param concreteArguments concrete arguments\n     * @returns A parametric Element  \n     */\n    public createOrGetFamixConcreteElement(concreteElement : ParametricVariantType, \n                                           concreteElementDeclaration : ConcreteElementTSMorphType, \n                                           concreteArguments: TypeNode[]): ParametricVariantType {\n        \n        let fullyQualifiedFilename = concreteElement.fullyQualifiedName;\n        let params = \"\";\n        \n        concreteArguments.map((param) => {\n            params = params+param.getText()+','\n        })\n        \n        params = params.substring(0, params.length - 1)\n                \n        fullyQualifiedFilename = Helpers.replaceLastBetweenTags(fullyQualifiedFilename,params);\n\n        let concElement: ParametricVariantType;\n\n        if (!this.fmxInterfaceMap.has(fullyQualifiedFilename) && \n            !this.fmxClassMap.has(fullyQualifiedFilename) && \n            !this.fmxFunctionAndMethodMap.has(fullyQualifiedFilename)){\n            concElement = _.cloneDeep(concreteElement); \n            concElement.fullyQualifiedName = fullyQualifiedFilename;\n            concElement.clearGenericParameters();\n            concreteArguments.map((param) => {\n                const parameter = this.createOrGetFamixConcreteType(param);\n                concElement.addConcreteParameter(parameter);\n            })\n            \n            if (concreteElement instanceof Famix.ParametricClass) {\n                this.fmxClassMap.set(fullyQualifiedFilename, concElement as Famix.ParametricClass);\n            } else if (concreteElement instanceof Famix.ParametricInterface) {\n                this.fmxInterfaceMap.set(fullyQualifiedFilename, concElement as Famix.ParametricInterface);\n            } else if (concreteElement instanceof Famix.ParametricFunction) {\n                this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricFunction);\n            } else { // if (concreteElement instanceof Famix.ParametricMethod) {\n                this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricMethod);\n            }\n            this.famixRep.addElement(concElement);\n            this.fmxElementObjectMap.set(concElement,concreteElementDeclaration);\n        } else {\n            if (concreteElement instanceof Famix.ParametricClass) {\n                concElement = this.fmxClassMap.get(fullyQualifiedFilename) as Famix.ParametricClass;\n            } else if (concreteElement instanceof Famix.ParametricInterface) {\n                concElement = this.fmxInterfaceMap.get(fullyQualifiedFilename) as Famix.ParametricInterface;\n            } else if (concreteElement instanceof Famix.ParametricFunction) {\n                concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricFunction;\n            } else {  // if (concreteElement instanceof Famix.ParametricMethod) {\n                concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricMethod;\n            }\n        }\n        return concElement;\n    }\n\n    /**\n     * Creates a Famix property\n     * @param property A property\n     * @returns The Famix model of the property\n     */\n    public createFamixProperty(property: PropertyDeclaration | PropertySignature): Famix.Property {\n        const fmxProperty = new Famix.Property();\n        const isSignature = property instanceof PropertySignature;\n        fmxProperty.name = property.getName();\n\n        let propTypeName = this.UNKNOWN_VALUE;\n        try {\n            propTypeName = property.getType().getText().trim();\n        } catch (error) {\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for property: ${property.getName()}. Continuing...`);\n        }\n\n        const fmxType = this.createOrGetFamixType(propTypeName, property);\n        fmxProperty.declaredType = fmxType;\n\n        // add the visibility (public, private, etc.) to the fmxProperty\n        fmxProperty.visibility = \"\";\n\n        property.getModifiers().forEach(m => {\n            switch (m.getText()) {\n                case Scope.Public:\n                    fmxProperty.visibility = \"public\";\n                    break;\n                case Scope.Protected:\n                    fmxProperty.visibility = \"protected\";\n                    break;\n                case Scope.Private:\n                    fmxProperty.visibility = \"private\";\n                    break;\n                case \"static\":\n                    fmxProperty.isClassSide = true;\n                    break;\n                case \"readonly\":\n                    fmxProperty.readOnly = true;\n                    break;\n                default:\n                    break;\n            }\n        });\n\n        if (!isSignature && property.getExclamationTokenNode()) {\n            fmxProperty.isDefinitelyAssigned = true;\n        }\n        if (property.getQuestionTokenNode()) {\n            fmxProperty.isOptional = true;\n        }\n        if (property.getName().substring(0, 1) === \"#\") {\n            fmxProperty.isJavaScriptPrivate = true;\n        }\n\n        initFQN(property, fmxProperty);\n        this.makeFamixIndexFileAnchor(property, fmxProperty);\n\n        this.famixRep.addElement(fmxProperty);\n\n        this.fmxElementObjectMap.set(fmxProperty,property);\n\n        return fmxProperty;\n    }\n\n    /**\n     * Creates a Famix method or accessor\n     * @param method A method or an accessor\n     * @param currentCC The cyclomatic complexity metrics of the current source file\n     * @returns The Famix model of the method or the accessor\n     */\n    public createOrGetFamixMethod(method: MethodDeclaration | ConstructorDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration, currentCC: { [key: string]: number }): Famix.Method | Famix.Accessor | Famix.ParametricMethod {\n        let fmxMethod: Famix.Method | Famix.Accessor | Famix.ParametricMethod;\n        const isGeneric = method.getTypeParameters().length > 0;\n        const functionFullyQualifiedName = FQNFunctions.getFQN(method);\n        if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {\n\n            if (method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) {\n                fmxMethod = new Famix.Accessor();\n                const isGetter = method instanceof GetAccessorDeclaration;\n                const isSetter = method instanceof SetAccessorDeclaration;\n                if (isGetter) {(fmxMethod as Famix.Accessor).kind = \"getter\";}\n                if (isSetter) {(fmxMethod as Famix.Accessor).kind = \"setter\";}\n                this.famixRep.addElement(fmxMethod);\n            }\n            else {\n                if (isGeneric) {\n                    fmxMethod = new Famix.ParametricMethod();\n                }\n                else {\n                    fmxMethod = new Famix.Method();\n                }\n                this.famixRep.addElement(fmxMethod);\n            }\n            const isConstructor = method instanceof ConstructorDeclaration;\n            const isSignature = method instanceof MethodSignature;\n\n            let isAbstract = false;\n            let isStatic = false;\n            if (method instanceof MethodDeclaration || method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) {\n                isAbstract = method.isAbstract();\n                isStatic = method.isStatic();\n            }\n\n            if (isConstructor) {(fmxMethod as Famix.Accessor).kind = \"constructor\";}\n            fmxMethod.isAbstract = isAbstract;\n            fmxMethod.isClassSide = isStatic;\n            fmxMethod.isPrivate = (method instanceof MethodDeclaration || method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'private')) !== undefined : false;\n            fmxMethod.isProtected = (method instanceof MethodDeclaration || method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'protected')) !== undefined : false;\n            fmxMethod.signature = Helpers.computeSignature(method.getText());\n\n            let methodName: string;\n            if (isConstructor) {\n                methodName = \"constructor\";\n            }\n            else {\n                methodName = (method as MethodDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration).getName();\n            }\n            fmxMethod.name = methodName;\n\n            if (!isConstructor) {\n                if (method.getName().substring(0, 1) === \"#\") {\n                    fmxMethod.isPrivate = true;\n                }\n            }\n\n            if (!fmxMethod.isPrivate && !fmxMethod.isProtected) {\n                fmxMethod.isPublic = true;\n            }\n            else {\n                fmxMethod.isPublic = false;\n            }\n\n            if (!isSignature) {\n                fmxMethod.cyclomaticComplexity = currentCC[fmxMethod.name];\n            }\n            else {\n                fmxMethod.cyclomaticComplexity = 0;\n            }\n\n            let methodTypeName = this.UNKNOWN_VALUE; \n            try {\n                methodTypeName = method.getReturnType().getText().trim();            \n            } catch (error) {\n                logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of method: ${fmxMethod.name}. Continuing...`);\n            }\n\n            const fmxType = this.createOrGetFamixType(methodTypeName, method);\n            fmxMethod.declaredType = fmxType;\n            fmxMethod.numberOfLinesOfCode = method.getEndLineNumber() - method.getStartLineNumber();\n            const parameters = method.getParameters();\n            fmxMethod.numberOfParameters = parameters.length;\n\n            if (!isSignature) {\n                fmxMethod.numberOfStatements = method.getStatements().length;\n            }\n            else {\n                fmxMethod.numberOfStatements = 0;\n            }\n            \n            initFQN(method, fmxMethod);\n            this.makeFamixIndexFileAnchor(method, fmxMethod);\n\n            this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxMethod);\n        }\n        else {\n            fmxMethod = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName) as (Famix.Method | Famix.Accessor | Famix.ParametricMethod);\n        }\n\n        this.fmxElementObjectMap.set(fmxMethod,method);\n        \n        return fmxMethod;\n    }\n\n    /**\n     * Creates a Famix function\n     * @param func A function\n     * @param currentCC The cyclomatic complexity metrics of the current source file\n     * @returns The Famix model of the function\n     */\n    public createOrGetFamixFunction(func: FunctionDeclaration | FunctionExpression, currentCC: { [key: string]: number }): Famix.Function | Famix.ParametricFunction {\n        let fmxFunction: Famix.Function | Famix.ParametricFunction;\n        const isGeneric = func.getTypeParameters().length > 0;        \n        const functionFullyQualifiedName = FQNFunctions.getFQN(func);\n        if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {\n            if (isGeneric) {\n                fmxFunction = new Famix.ParametricFunction();\n            }\n            else {\n                fmxFunction = new Famix.Function();\n            }\n    \n            const name = func.getName();\n            if (name) {\n                fmxFunction.name = name;\n            }\n            else {\n                fmxFunction.name = \"anonymous\";\n            }\n\n            fmxFunction.signature = Helpers.computeSignature(func.getText());\n            fmxFunction.cyclomaticComplexity = currentCC[fmxFunction.name];\n            fmxFunction.fullyQualifiedName = functionFullyQualifiedName;\n    \n            let functionTypeName = this.UNKNOWN_VALUE;\n            try {\n                functionTypeName = func.getReturnType().getText().trim();\n            } catch (error) {\n                logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${func.getName()}. Continuing...`);\n            }\n    \n            const fmxType = this.createOrGetFamixType(functionTypeName, func);\n            fmxFunction.declaredType = fmxType;\n            fmxFunction.numberOfLinesOfCode = func.getEndLineNumber() - func.getStartLineNumber();\n            const parameters = func.getParameters();\n            fmxFunction.numberOfParameters = parameters.length;\n            fmxFunction.numberOfStatements = func.getStatements().length;\n            this.makeFamixIndexFileAnchor(func, fmxFunction);\n    \n            this.famixRep.addElement(fmxFunction);\n    \n            this.fmxElementObjectMap.set(fmxFunction,func);\n\n            this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxFunction);\n        }\n        else {\n            fmxFunction = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName) as (Famix.Function | Famix.ParametricFunction);\n        }\n\n        return fmxFunction;\n    }\n\n    /**\n     * Creates a Famix parameter\n     * @param param A parameter\n     * @returns The Famix model of the parameter\n     */\n    public createFamixParameter(param: ParameterDeclaration): Famix.Parameter {\n        const fmxParam = new Famix.Parameter();\n\n        let paramTypeName = this.UNKNOWN_VALUE;\n        try {\n            paramTypeName = param.getType().getText().trim();\n        } catch (error) {\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for parameter: ${param.getName()}. Continuing...`);\n        }\n\n        const fmxType = this.createOrGetFamixType(paramTypeName, param);\n        fmxParam.declaredType = fmxType;\n        fmxParam.name = param.getName();\n\n        initFQN(param, fmxParam);\n        this.makeFamixIndexFileAnchor(param, fmxParam);\n\n        this.famixRep.addElement(fmxParam);\n\n        this.fmxElementObjectMap.set(fmxParam,param);\n\n        return fmxParam;\n    }\n\n    /**\n     * Creates a Famix type parameter\n     * @param tp A type parameter\n     * @returns The Famix model of the type parameter\n     */\n    public createFamixParameterType(tp: TypeParameterDeclaration): Famix.ParameterType {\n        \n        const fmxParameterType = new Famix.ParameterType();\n   \n        fmxParameterType.name = tp.getName();      \n        initFQN(tp, fmxParameterType);\n        this.makeFamixIndexFileAnchor(tp, fmxParameterType);\n\n        this.famixRep.addElement(fmxParameterType);\n\n        this.fmxElementObjectMap.set(fmxParameterType,tp);\n\n        return fmxParameterType;\n    }\n\n    /**\n     * Creates a Famix type parameter\n     * @param tp A type parameter\n     * @returns The Famix model of the type parameter\n     */\n    public createOrGetFamixConcreteType(param: TypeNode): Famix.ParameterType | Famix.PrimitiveType | Famix.Class | Famix.Interface {\n        const typeParameterDeclaration = param.getSymbol()?.getDeclarations()[0] as TypeParameterDeclaration;\n        const parameterTypeName : string = param.getText();\n        let fmxParameterType: Famix.Type | Famix.Class | Famix.Interface | undefined = undefined;\n\n        let isClassOrInterface = false;\n        if (this.fmxClassMap.has(parameterTypeName)){\n            this.fmxClassMap.forEach((obj, name) => {\n                if(obj instanceof Famix.ParametricClass){\n                    if (name === param.getText() && obj.genericParameters.size>0) {\n                        fmxParameterType = obj;\n                        isClassOrInterface = true;\n                    } \n                } else {\n                    if (name === param.getText()) {\n                        fmxParameterType = obj;\n                        isClassOrInterface = true;\n                    } \n                }   \n            })\n        }\n\n        if (this.fmxInterfaceMap.has(parameterTypeName)){\n            this.fmxInterfaceMap.forEach((obj, name) => {\n                if(obj instanceof Famix.ParametricInterface){\n                    if (name === param.getText() && obj.genericParameters.size>0) {\n                        fmxParameterType = obj;\n                        isClassOrInterface = true;\n                    } \n                } else {\n                    if (name === param.getText()) {\n                        fmxParameterType = obj;\n                        isClassOrInterface = true;\n                    } \n                }   \n            })\n        }\n\n        if(!isClassOrInterface){\n            if (!this.fmxTypeMap.has(parameterTypeName)) {           \n                if (parameterTypeName === \"number\" || parameterTypeName === \"string\" || parameterTypeName === \"boolean\" || parameterTypeName === \"bigint\" || parameterTypeName === \"symbol\" || parameterTypeName === \"undefined\" || parameterTypeName === \"null\" || parameterTypeName === \"any\" || parameterTypeName === \"unknown\" || parameterTypeName === \"never\" || parameterTypeName === \"void\") {\n                    fmxParameterType = new Famix.PrimitiveType();\n                    fmxParameterType.isStub = true;\n                } else {\n                    fmxParameterType = new Famix.ParameterType();\n                } \n    \n                fmxParameterType.name = parameterTypeName;\n                this.famixRep.addElement(fmxParameterType);\n                this.fmxTypeMap.set(parameterTypeName, fmxParameterType);\n                this.fmxElementObjectMap.set(fmxParameterType,typeParameterDeclaration);\n            }\n            else {\n                const result = this.fmxTypeMap.get(parameterTypeName);\n                if (result) {\n                    fmxParameterType = result;\n                } else {\n                    throw new Error(`Famix type ${parameterTypeName} is not found in the Type map.`);\n                }\n            }\n        }\n        if (!fmxParameterType) {\n            throw new Error(`fmxParameterType was undefined for parameterTypeName ${parameterTypeName}`);\n        }\n        return fmxParameterType;\n    }\n\n    /**\n     * Creates a Famix variable\n     * @param variable A variable\n     * @returns The Famix model of the variable\n     */\n    public createFamixVariable(variable: VariableDeclaration): Famix.Variable {\n        const fmxVariable = new Famix.Variable();\n    \n        let variableTypeName = this.UNKNOWN_VALUE;\n        try {\n            variableTypeName = variable.getType().getText().trim();\n        } catch (error) {\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for variable: ${variable.getName()}. Continuing...`);\n        }\n    \n        const fmxType = this.createOrGetFamixType(variableTypeName, variable);\n        fmxVariable.declaredType = fmxType;\n        fmxVariable.name = variable.getName();\n        initFQN(variable, fmxVariable);\n        this.makeFamixIndexFileAnchor(variable, fmxVariable);\n    \n        this.famixRep.addElement(fmxVariable);\n    \n        this.fmxElementObjectMap.set(fmxVariable,variable);\n    \n        return fmxVariable;\n    }\n\n    /**\n     * Creates a Famix enum\n     * @param enumEntity An enum\n     * @returns The Famix model of the enum\n     */\n    public createFamixEnum(enumEntity: EnumDeclaration): Famix.Enum {\n        const fmxEnum = new Famix.Enum();\n        fmxEnum.name = enumEntity.getName();\n        initFQN(enumEntity, fmxEnum);\n        this.makeFamixIndexFileAnchor(enumEntity, fmxEnum);\n\n        this.famixRep.addElement(fmxEnum);\n\n        this.fmxElementObjectMap.set(fmxEnum,enumEntity);\n\n        return fmxEnum;\n    }\n\n    /**\n     * Creates a Famix enum value\n     * @param enumMember An enum member\n     * @returns The Famix model of the enum member\n     */\n    public createFamixEnumValue(enumMember: EnumMember): Famix.EnumValue {\n        const fmxEnumValue = new Famix.EnumValue();\n\n        let enumValueTypeName = this.UNKNOWN_VALUE;\n        try {\n            enumValueTypeName = enumMember.getType().getText().trim();\n        } catch (error) {\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for enum value: ${enumMember.getName()}. Continuing...`);\n        }\n\n        const fmxType = this.createOrGetFamixType(enumValueTypeName, enumMember);\n        fmxEnumValue.declaredType = fmxType;\n        fmxEnumValue.name = enumMember.getName();\n        initFQN(enumMember, fmxEnumValue);\n        this.makeFamixIndexFileAnchor(enumMember, fmxEnumValue);\n\n        this.famixRep.addElement(fmxEnumValue);\n\n        this.fmxElementObjectMap.set(fmxEnumValue,enumMember);\n\n        return fmxEnumValue;\n    }\n\n    /**\n     * Creates or gets a Famix decorator\n     * @param decorator A decorator\n     * @param decoratedEntity A class, a method, a parameter or a property\n     * @returns The Famix model of the decorator\n     */\n    public createOrGetFamixDecorator(decorator: Decorator, decoratedEntity: ClassDeclaration | MethodDeclaration | GetAccessorDeclaration | SetAccessorDeclaration | ParameterDeclaration | PropertyDeclaration): Famix.Decorator {\n        const fmxDecorator = new Famix.Decorator();\n        const decoratorName = \"@\" + decorator.getName();\n        const decoratorExpression = decorator.getText().substring(1);\n\n        fmxDecorator.name = decoratorName;\n        fmxDecorator.decoratorExpression = decoratorExpression;\n        const decoratedEntityFullyQualifiedName = FQNFunctions.getFQN(decoratedEntity);\n        const fmxDecoratedEntity = this.famixRep.getFamixEntityByFullyQualifiedName(decoratedEntityFullyQualifiedName) as Famix.NamedEntity;\n        fmxDecorator.decoratedEntity = fmxDecoratedEntity;\n        initFQN(decorator, fmxDecorator);\n        this.makeFamixIndexFileAnchor(decorator, fmxDecorator);\n\n        this.famixRep.addElement(fmxDecorator);\n\n        this.fmxElementObjectMap.set(fmxDecorator,decorator);\n\n        return fmxDecorator;\n    }\n\n    /**\n     * Creates a Famix comment\n     * @param comment A comment\n     * @param fmxScope The Famix model of the comment's container\n     * @param isJSDoc A boolean indicating if the comment is a JSDoc\n     * @returns The Famix model of the comment\n     */\n    public createFamixComment(comment: CommentRange, fmxScope: Famix.NamedEntity, isJSDoc: boolean): Famix.Comment {\n        logger.debug(`> NOTE: creating comment ${comment.getText()} in scope ${fmxScope.name}.`);\n        const fmxComment = new Famix.Comment();\n        fmxComment.container = fmxScope;  // adds comment to the container's comments collection\n        fmxComment.isJSDoc = isJSDoc;\n\n        this.makeFamixIndexFileAnchor(comment, fmxComment);\n\n        this.famixRep.addElement(fmxComment);\n\n        this.fmxElementObjectMap.set(fmxComment,comment);\n\n        return fmxComment;\n    }\n\n    /**\n     * Creates or gets a Famix type\n     * @param typeName A type name\n     * @param element A ts-morph element\n     * @returns The Famix model of the type\n     */\n    public createOrGetFamixType(typeName: string, element: TypeDeclaration): Famix.Type | Famix.PrimitiveType | Famix.ParameterType {\n        let fmxType: Famix.Type | Famix.PrimitiveType | Famix.ParameterType;\n        let isPrimitiveType = false;\n        let isParameterType = false;\n\n        logger.debug(\"Creating (or getting) type: '\" + typeName + \"' of element: \" + element?.getText() + \" of kind: \" + element?.getKindName());\n        let ancestor: Famix.ContainerEntity | undefined = undefined;\n        if (element !== undefined) {\n            const typeAncestor = Helpers.findTypeAncestor(element);\n            if (!typeAncestor) {\n                throw new Error(`Ancestor not found for element ${element.getText()}.`);\n            }\n            const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);\n            ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName) as Famix.ContainerEntity;\n            if (!ancestor) {\n                logger.debug(`Ancestor ${FQNFunctions.getFQN(typeAncestor)} not found. Adding the new type.`);\n                ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor as TypeDeclaration);\n            }\n        }\n\n        if (typeName === \"number\" || typeName === \"string\" || typeName === \"boolean\" || typeName === \"bigint\" || typeName === \"symbol\" || typeName === \"undefined\" || typeName === \"null\" || typeName === \"any\" || typeName === \"unknown\" || typeName === \"never\" || typeName === \"void\") {\n            isPrimitiveType = true;\n        }\n\n        if(!isPrimitiveType && typeName.includes(\"<\") && typeName.includes(\">\") && !(typeName.includes(\"=>\"))) {\n            isParameterType = true;\n        }\n\n        if (!this.fmxTypeMap.has(typeName)) {\n            if (isPrimitiveType) {\n                fmxType = new Famix.PrimitiveType();\n                fmxType.isStub = true;\n            }\n            else if (isParameterType) {\n                fmxType = new Famix.ParameterType();\n                const parameterTypeNames = typeName.substring(typeName.indexOf(\"<\") + 1, typeName.indexOf(\">\")).split(\",\").map(s => s.trim());\n                const baseTypeName = typeName.substring(0, typeName.indexOf(\"<\")).trim();\n                parameterTypeNames.forEach(parameterTypeName => {\n                    const fmxParameterType = this.createOrGetFamixType(parameterTypeName, element);\n                    (fmxType as Famix.ParameterType).addArgument(fmxParameterType);\n                });\n                const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);\n                (fmxType as Famix.ParameterType).baseType = fmxBaseType;\n            }\n            else {\n                fmxType = new Famix.Type();\n            }\n\n            fmxType.name = typeName;\n            if (!ancestor) {\n                throw new Error(`Ancestor not found for type ${typeName}.`);\n            }\n            fmxType.container = ancestor;\n            initFQN(element, fmxType);\n            this.makeFamixIndexFileAnchor(element, fmxType);\n\n            this.famixRep.addElement(fmxType);\n\n            this.fmxTypeMap.set(typeName, fmxType);\n        }\n        else {\n            const result = this.fmxTypeMap.get(typeName);\n            if (result) {\n                fmxType = result;\n            } else {\n                throw new Error(`Famix type ${typeName} is not found in the Type map.`);\n            }\n        }\n\n        this.fmxElementObjectMap.set(fmxType,element);\n\n        return fmxType;\n    }\n\n    /**\n     * Creates a Famix access\n     * @param node A node\n     * @param id An id of a parameter, a variable, a property or an enum member\n     */\n    public createFamixAccess(node: Identifier, id: number): void {\n        const fmxVar = this.famixRep.getFamixEntityById(id) as Famix.StructuralEntity;\n        if (!fmxVar) {\n            throw new Error(`Famix entity with id ${id} not found, for node ${node.getText()} in ${node.getSourceFile().getBaseName()} at line ${node.getStartLineNumber()}.`);\n        }\n\n        logger.debug(`Creating FamixAccess. Node: [${node.getKindName()}] '${node.getText()}' at line ${node.getStartLineNumber()} in ${node.getSourceFile().getBaseName()}, id: ${id} refers to fmxVar '${fmxVar.fullyQualifiedName}'.`);\n\n        const nodeReferenceAncestor = Helpers.findAncestor(node);\n        const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);\n        let accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName) as Famix.ContainerEntity;\n        if (!accessor) {\n            logger.error(`Ancestor ${ancestorFullyQualifiedName} of kind ${nodeReferenceAncestor.getKindName()} not found.`);\n            // accessor = this.createOrGetFamixType(ancestorFullyQualifiedName, nodeReferenceAncestor as TypeDeclaration);\n        }\n\n        const fmxAccess = new Famix.Access();\n        fmxAccess.accessor = accessor;\n        fmxAccess.variable = fmxVar;\n\n        this.famixRep.addElement(fmxAccess);\n\n        this.fmxElementObjectMap.set(fmxAccess,node);\n    }\n\n    /**\n     * Creates a Famix invocation\n     * @param node A node\n     * @param m A method or a function\n     * @param id The id of the method or the function\n     */\n    public createFamixInvocation(node: Identifier, m: MethodDeclaration | ConstructorDeclaration | GetAccessorDeclaration | SetAccessorDeclaration | FunctionDeclaration | FunctionExpression, id: number): void {\n        const fmxMethodOrFunction = this.famixRep.getFamixEntityById(id) as Famix.BehavioralEntity;\n        const nodeReferenceAncestor = Helpers.findAncestor(node);\n        const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);\n        const sender = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName) as Famix.ContainerEntity;\n        const receiverFullyQualifiedName = FQNFunctions.getFQN(m.getParent());\n        const receiver = this.famixRep.getFamixEntityByFullyQualifiedName(receiverFullyQualifiedName) as Famix.NamedEntity;\n\n        const fmxInvocation = new Famix.Invocation();\n        fmxInvocation.sender = sender;\n        fmxInvocation.receiver = receiver;\n        fmxInvocation.addCandidate(fmxMethodOrFunction);\n        fmxInvocation.signature = fmxMethodOrFunction.signature;\n\n        this.famixRep.addElement(fmxInvocation);\n\n        this.fmxElementObjectMap.set(fmxInvocation,node);\n    }\n\n    /**\n     * Creates a Famix inheritance\n     * @param cls A class or an interface (subclass)\n     * @param inhClass The inherited class or interface (superclass)\n     */\n    public createFamixInheritance(cls: ClassDeclaration | InterfaceDeclaration, inhClass: ClassDeclaration | InterfaceDeclaration | ExpressionWithTypeArguments): void {\n        const fmxInheritance = new Famix.Inheritance();\n        // const clsName = cls.getName();\n        const classFullyQualifiedName = FQNFunctions.getFQN(cls);\n        logger.debug(`createFamixInheritance: classFullyQualifiedName: class fqn = ${classFullyQualifiedName}`);\n        let subClass: Famix.Class | Famix.Interface | undefined;\n        if (cls instanceof ClassDeclaration) {\n            subClass = this.fmxClassMap.get(classFullyQualifiedName);\n        }\n        else {\n            subClass = this.fmxInterfaceMap.get(classFullyQualifiedName);\n        }\n        if (!subClass) {\n            throw new Error(`Subclass ${classFullyQualifiedName} not found in Class or Interface maps.`);\n        }\n        \n        let inhClassName: string | undefined;\n        let inhClassFullyQualifiedName: string;\n        let superClass: Famix.Class | Famix.Interface | undefined;\n        if (inhClass instanceof ClassDeclaration || inhClass instanceof InterfaceDeclaration) {\n            inhClassName = inhClass.getName();\n            if (!inhClassName) {\n                throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);\n            }\n            inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);\n            if (inhClass instanceof ClassDeclaration) {\n                superClass = this.fmxClassMap.get(inhClassFullyQualifiedName);\n            }\n            else {\n                superClass = this.fmxInterfaceMap.get(inhClassFullyQualifiedName);\n            }\n            if (!superClass) {\n                throw new Error(`Superclass ${classFullyQualifiedName} not found in Class or Interface maps.`);\n            }\n        }\n        else {\n            // inhClass is an ExpressionWithTypeArguments\n            inhClassName = inhClass.getExpression().getText();\n            // what is inhClassFullyQualifiedName? TODO\n            inhClassFullyQualifiedName = 'Undefined_Scope_from_importer.' + inhClassName;\n        }\n\n        if (superClass === undefined) {\n            if (inhClass instanceof ClassDeclaration) {\n                superClass = new Famix.Class();\n                this.fmxClassMap.set(inhClassFullyQualifiedName, superClass);\n            }\n            else {\n                superClass = new Famix.Interface();\n                this.fmxInterfaceMap.set(inhClassFullyQualifiedName, superClass);\n            }\n\n            this.fmxElementObjectMap.set(superClass,inhClass);\n\n            superClass.name = inhClassName;\n            superClass.fullyQualifiedName = inhClassFullyQualifiedName;\n            superClass.isStub = true;\n\n            this.makeFamixIndexFileAnchor(inhClass, superClass);\n        \n            this.famixRep.addElement(superClass);\n        }\n\n        fmxInheritance.subclass = subClass;\n        fmxInheritance.superclass = superClass;\n\n        this.famixRep.addElement(fmxInheritance);\n\n        // We don't map inheritance to the source code element because there are two elements (super, sub)\n        // this.fmxElementObjectMap.set(fmxInheritance, null);\n\n    }\n\n    public createFamixImportClause(importedEntity: Famix.NamedEntity, importingEntity: Famix.Module) {\n        const fmxImportClause = new Famix.ImportClause();\n        fmxImportClause.importedEntity = importedEntity;\n        fmxImportClause.importingEntity = importingEntity;\n        importingEntity.addOutgoingImport(fmxImportClause);\n        this.famixRep.addElement(fmxImportClause);\n    }\n\n    /**\n     * Creates a Famix import clause\n     * @param importClauseInfo The information needed to create a Famix import clause\n     * @param importDeclaration The import declaration\n     * @param importer A source file which is a module\n     * @param moduleSpecifierFilePath The path of the module where the export declaration is\n     * @param importElement The imported entity\n     * @param isInExports A boolean indicating if the imported entity is in the exports\n     * @param isDefaultExport A boolean indicating if the imported entity is a default export\n     */\n    public oldCreateFamixImportClause(importClauseInfo: {importDeclaration?: ImportDeclaration | ImportEqualsDeclaration, importerSourceFile: SourceFile, moduleSpecifierFilePath: string, importElement: ImportSpecifier | Identifier, isInExports: boolean, isDefaultExport: boolean}): void {\n        const {importDeclaration, importerSourceFile: importer, moduleSpecifierFilePath, importElement, isInExports, isDefaultExport} = importClauseInfo;\n        logger.debug(`createFamixImportClause: Creating import clause:`);\n        const fmxImportClause = new Famix.ImportClause();\n\n        let importedEntity: Famix.NamedEntity | Famix.StructuralEntity | undefined = undefined;\n        let importedEntityName: string;\n\n        const absolutePathProject = this.famixRep.getAbsolutePath();\n        \n        const absolutePath = path.normalize(moduleSpecifierFilePath);\n        // convert the path and remove any windows backslashes introduced by path.normalize\n        const pathInProject: string = this.convertToRelativePath(absolutePath, absolutePathProject).replace(/\\\\/g, \"/\");\n        let pathName = \"{\" + pathInProject + \"}.\";\n\n        // Named imports, e.g. import { ClassW } from \"./complexExportModule\";\n\n        // Start with simple import clause (without referring to the actual variable)\n\n        if (importDeclaration instanceof ImportDeclaration \n            && importElement instanceof ImportSpecifier) { \n                importedEntityName = importElement.getName();\n            pathName = pathName + importedEntityName;\n            if (isInExports) {\n                importedEntity = this.famixRep.getFamixEntityByFullyQualifiedName(pathName) as Famix.NamedEntity;\n            }\n            if (importedEntity === undefined) {\n                importedEntity = new Famix.NamedEntity();\n                importedEntity.name = importedEntityName;\n                if (!isInExports) {\n                    importedEntity.isStub = true;\n                }\n                importedEntity.fullyQualifiedName = pathName;\n                this.makeFamixIndexFileAnchor(importElement, importedEntity);\n                // must add entity to repository\n                this.famixRep.addElement(importedEntity);\n            }\n        }\n        // handle import equals declarations, e.g. import myModule = require(\"./complexExportModule\");\n        // TypeScript can't determine the type of the imported module, so we create a Module entity\n        else if (importDeclaration instanceof ImportEqualsDeclaration) {\n            importedEntityName = importDeclaration?.getName();\n            pathName = pathName + importedEntityName;\n            importedEntity = new Famix.StructuralEntity();\n            importedEntity.name = importedEntityName;\n            initFQN(importDeclaration, importedEntity);\n            this.makeFamixIndexFileAnchor(importElement, importedEntity);\n            importedEntity.fullyQualifiedName = pathName;\n            const anyType = this.createOrGetFamixType('any', importDeclaration);\n            (importedEntity as Famix.StructuralEntity).declaredType = anyType;\n        } else {  // default imports, e.g. import ClassW from \"./complexExportModule\";  \n            importedEntityName = importElement.getText();\n            pathName = pathName + (isDefaultExport ? \"defaultExport\" : \"namespaceExport\");\n            importedEntity = new Famix.NamedEntity();\n            importedEntity.name = importedEntityName;\n            importedEntity.fullyQualifiedName = pathName;\n            this.makeFamixIndexFileAnchor(importElement, importedEntity);\n        }\n        // I don't think it should be added to the repository if it exists already\n        if (!isInExports) this.famixRep.addElement(importedEntity);\n        const importerFullyQualifiedName = FQNFunctions.getFQN(importer);\n        const fmxImporter = this.famixRep.getFamixEntityByFullyQualifiedName(importerFullyQualifiedName) as Famix.Module;\n        fmxImportClause.importingEntity = fmxImporter;\n        fmxImportClause.importedEntity = importedEntity;\n        if (importDeclaration instanceof ImportEqualsDeclaration) {\n            fmxImportClause.moduleSpecifier = importDeclaration?.getModuleReference().getText() as string;\n        } else {\n            fmxImportClause.moduleSpecifier = importDeclaration?.getModuleSpecifierValue() as string;\n        }\n    \n        logger.debug(`createFamixImportClause: ${fmxImportClause.importedEntity?.name} (of type ${\n            Helpers.getSubTypeName(fmxImportClause.importedEntity)}) is imported by ${fmxImportClause.importingEntity?.name}`);\n\n        fmxImporter.addOutgoingImport(fmxImportClause);\n\n        this.famixRep.addElement(fmxImportClause);\n\n        if (importDeclaration) this.fmxElementObjectMap.set(fmxImportClause, importDeclaration);\n    }\n\n    /**\n     * Creates a Famix Arrow Function\n     * @param arrowExpression An Expression\n     * @returns The Famix model of the variable\n     */\n    public createFamixArrowFunction(arrowExpression: Expression, currentCC: { [key: string]: number } ): Famix.ArrowFunction | Famix.ParametricArrowFunction {\n        \n        let fmxArrowFunction: Famix.ArrowFunction | Famix.ParametricArrowFunction;\n\n        const arrowFunction = arrowExpression.asKindOrThrow(SyntaxKind.ArrowFunction);\n\n        const isGeneric = arrowFunction.getTypeParameters().length > 0;\n\n        if (isGeneric) {\n            fmxArrowFunction = new Famix.ParametricArrowFunction();\n        }\n        else {\n            fmxArrowFunction = new Famix.ArrowFunction();\n        }\n\n        // Get the parent of the arrow function (the variable declaration)\n        const parent = arrowFunction.getParentIfKind(SyntaxKind.VariableDeclaration);\n        let functionName = '(NO_NAME)';\n\n        if (parent && parent instanceof VariableDeclaration) {\n            // Get the name of the variable\n            functionName = parent.getName();\n        }\n\n        if (functionName) {\n            fmxArrowFunction.name = functionName;\n        }\n        else {\n            fmxArrowFunction.name = \"anonymous\";\n        }\n\n        // Signature of an arrow function is (parameters) => return_type\n        const parametersSignature = arrowFunction.getParameters().map(p => p.getText()).join(\", \");\n        const returnTypeSignature = arrowFunction.getReturnType().getText();\n        fmxArrowFunction.signature = `(${parametersSignature}) => ${returnTypeSignature}`;\n        fmxArrowFunction.cyclomaticComplexity = currentCC[fmxArrowFunction.name];\n\n        let functionTypeName = this.UNKNOWN_VALUE;\n        try {\n            functionTypeName = arrowFunction.getReturnType().getText().trim();\n        } catch (error) {\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${functionName}. Continuing...`);\n        }\n\n        const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction as unknown as FunctionDeclaration);\n        fmxArrowFunction.declaredType = fmxType;\n        fmxArrowFunction.numberOfLinesOfCode = arrowFunction.getEndLineNumber() - arrowFunction.getStartLineNumber();\n        const parameters = arrowFunction.getParameters();\n        fmxArrowFunction.numberOfParameters = parameters.length;\n        fmxArrowFunction.numberOfStatements = arrowFunction.getStatements().length;\n        initFQN(arrowExpression as unknown as TSMorphObjectType, fmxArrowFunction);\n        this.makeFamixIndexFileAnchor(arrowExpression as unknown as TSMorphObjectType, fmxArrowFunction);\n        this.famixRep.addElement(fmxArrowFunction);\n        this.fmxElementObjectMap.set(fmxArrowFunction,arrowFunction as unknown as TSMorphObjectType);\n\n        return fmxArrowFunction;\n    }\n\n    /**\n     * Creates a Famix concretisation\n     * @param cls A class\n     * @returns The Famix model of the concretisation\n     */\n    public createFamixConcretisation(conEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod ,genEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod): Famix.Concretisation {\n        \n        const fmxConcretisation : Famix.Concretisation = new Famix.Concretisation();              \n        \n        fmxConcretisation.concreteEntity = conEntity;\n        fmxConcretisation.genericEntity = genEntity;\n        // this.fmxElementObjectMap.set(fmxConcretisation,null);\n        this.famixRep.addElement(fmxConcretisation);    \n        const parameterConcretisation = this.createFamixParameterConcretisation(fmxConcretisation);\n            \n        return fmxConcretisation;\n    }\n\n    /**\n     * Creates a Famix concretisation\n     * @param concretisation A FamixConcretisation\n     * @returns The Famix model of the ParameterConcrestisation\n     */\n    public createFamixParameterConcretisation(concretisation: Famix.Concretisation): Famix.ParameterConcretisation | undefined{\n        const conClass = concretisation.concreteEntity;\n        const genClass = concretisation.genericEntity;\n        logger.debug(`Creating parameter concretisation between ${conClass.fullyQualifiedName} and ${genClass.fullyQualifiedName}`);\n        const parameterConcretisations = this.famixRep._getAllEntitiesWithType(\"ParameterConcretisation\") as Set<Famix.ParameterConcretisation>;\n        const concreteParameters = conClass.concreteParameters;\n        const genericParameters = genClass.genericParameters;\n        \n        let conClassTypeParametersIterator = concreteParameters.values();\n        let genClassTypeParametersIterator = genericParameters.values();\n        let fmxParameterConcretisation : Famix.ParameterConcretisation | undefined = undefined;\n\n        for (let i = 0; i < genericParameters.size; i++) {\n            const conClassTypeParameter = conClassTypeParametersIterator.next().value as Famix.ParameterType;\n            const genClassTypeParameter = genClassTypeParametersIterator.next().value as Famix.ParameterType;\n            let createParameterConcretisation : boolean = true;\n            if(conClassTypeParameter && genClassTypeParameter && conClassTypeParameter.name != genClassTypeParameter.name){\n                parameterConcretisations.forEach((param : Famix.ParameterConcretisation) => {\n                    if (conClassTypeParameter.name == param.concreteParameter.name && genClassTypeParameter.name == param.genericParameter.name) {\n                        createParameterConcretisation = false;\n                        fmxParameterConcretisation = param;\n                    }\n                })\n                if (createParameterConcretisation) {\n                    fmxParameterConcretisation = new Famix.ParameterConcretisation();\n                    fmxParameterConcretisation.genericParameter = genClassTypeParameter;\n                    fmxParameterConcretisation.concreteParameter = conClassTypeParameter;\n                    fmxParameterConcretisation.addConcretisation(concretisation);\n                    // this.fmxElementObjectMap.set(fmxParameterConcretisation,null);\n                } else {\n                    if (!fmxParameterConcretisation) {\n                        throw new Error(`fmxParameterConcretisation was undefined for concretisation with generic parameter ${genClassTypeParameter.name} and concrete parameter ${conClassTypeParameter.name}`);\n                    }\n                    fmxParameterConcretisation.addConcretisation(concretisation);\n                }\n                this.famixRep.addElement(fmxParameterConcretisation);\n            }\n        }\n        if (!fmxParameterConcretisation) {\n            logger.error(`fmxParameterConcretisation was undefined for concretisation with concrete entity ${conClass.fullyQualifiedName} and generic entity ${genClass.fullyQualifiedName}`);\n        }\n        return fmxParameterConcretisation;\n\n    }\n\n    /**\n     * Creates a Famix concretisation between two classes or two interfaces\n     * @param element A class or an Interface\n     */\n    public createFamixConcretisationClassOrInterfaceSpecialisation(element: ClassDeclaration | InterfaceDeclaration){\n        \n        const superEntity = element.getExtends();\n        let superEntityArray;\n        if (superEntity){\n            superEntityArray = Array.isArray(superEntity) ? superEntity : [superEntity];\n        }\n        if (superEntityArray && superEntityArray.length > 0) {\n            superEntityArray.forEach(entity => {\n                let entityIsGeneric;\n                const superEntitySymbol = entity.getExpression().getSymbolOrThrow();\n                let superEntityDeclaration;\n                if (superEntity instanceof ExpressionWithTypeArguments) {\n                    superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.ClassDeclaration);\n                } else {\n                    superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.InterfaceDeclaration);\n                }\n                if (superEntityDeclaration) {\n                    entityIsGeneric = superEntityDeclaration.getTypeParameters().length > 0;\n                }\n                if (entityIsGeneric) {\n                    let EntityDeclaration;\n                    let genEntity;\n                    if (superEntity instanceof ExpressionWithTypeArguments) {\n                        EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as ClassDeclaration;\n                        genEntity = this.createOrGetFamixClass(EntityDeclaration) as Famix.ParametricClass;\n                    } else {\n                        EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;\n                        genEntity = this.createOrGetFamixInterface(EntityDeclaration) as Famix.ParametricInterface;\n                    }\n                    const genParams = EntityDeclaration.getTypeParameters().map((param) => param.getText());\n                    const args = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments()\n                    const conParams = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());\n                    if (!Helpers.arraysAreEqual(conParams,genParams)) {\n                        let conEntity;\n                        conEntity = this.createOrGetFamixConcreteElement(genEntity,EntityDeclaration,args);\n                        const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\n                        let createConcretisation : boolean = true;\n                        concretisations.forEach((conc : Famix.Concretisation) => {\n                            if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){\n                                createConcretisation = false;\n                            }\n                        });\n            \n                        if (createConcretisation) {\n                            const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);\n                        }\n                    }\n                }\n            });\n        }\n        // TODO: This function seems unfinished\n    }    \n    \n\n    /**\n     * Creates a Famix concretisation between a class and its instanciations\n     * @param cls A class\n     */\n    public createFamixConcretisationGenericInstantiation(cls: ClassDeclaration){\n       \n        const isGeneric = cls.getTypeParameters().length > 0;\n        if (isGeneric) {\n            const instances = cls.getSourceFile().getDescendantsOfKind(ts.SyntaxKind.NewExpression)\n                .filter(newExpr => {\n                    const expression = newExpr.getExpression();\n                    return expression.getText() === cls.getName();\n            });\n\n            instances.forEach(instance => {\n                const instanceIsGeneric = instance.getTypeArguments().length > 0;\n                if (instanceIsGeneric) {\n                    const conParams = instance.getTypeArguments().map((param) => param.getText());\n                    const genEntity = this.createOrGetFamixClass(cls) as Famix.ParametricClass;\n                    const genParams = cls.getTypeParameters().map((param) => param.getText());\n                    if (!Helpers.arraysAreEqual(conParams,genParams)) {\n                        let conEntity;\n                        conEntity = this.createOrGetFamixConcreteElement(genEntity,cls,instance.getTypeArguments());\n                        const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\n                        let createConcretisation : boolean = true;\n                        concretisations.forEach((conc : Famix.Concretisation) => {\n                            if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){\n                                createConcretisation = false;\n                            }\n                        });\n            \n                        if (createConcretisation) {\n                            const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);\n                        }\n                    }\n                }\n            })\n        }\n        // TODO: This function seems unfinished\n    }\n\n    /**\n     * Creates a Famix concretisation between a class and its instanciations\n     * @param func A function\n     */\n    public createFamixConcretisationFunctionInstantiation(element: FunctionDeclaration | MethodDeclaration){\n        const isGeneric = element.getTypeParameters().length > 0;\n        if (isGeneric) {\n            const genParams = element.getTypeParameters().map(param => param.getText());\n            const uses = element.findReferencesAsNodes();    \n            uses.forEach(usage => {\n                let currentNode: Node | undefined = usage;\n\n                while (currentNode) {\n                    if (currentNode.getKind() === SyntaxKind.CallExpression) {\n                        const callExpression = currentNode.asKind(SyntaxKind.CallExpression);\n                        if (!callExpression) {\n                            throw new Error(`CallExpression not found for ${currentNode.getText()}`);\n                        }\n                        const instanceIsGeneric = callExpression.getTypeArguments().length > 0;\n                        if (instanceIsGeneric) {\n                            const args = callExpression.getTypeArguments();\n                            const conParams = callExpression.getTypeArguments().map(param => param.getText());\n                            if (!Helpers.arraysAreEqual(conParams,genParams)) {\n                                let genElement;\n                                if(element instanceof FunctionDeclaration){\n                                    genElement = this.createOrGetFamixFunction(element, {}) as Famix.ParametricFunction;\n                                } else {\n                                    genElement = this.createOrGetFamixMethod(element, {}) as Famix.ParametricMethod;\n                                }\n                                let concElement;\n                                concElement = this.createOrGetFamixConcreteElement(genElement,element,args);\n                                const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\n                                let createConcretisation : boolean = true;\n                                concretisations.forEach((conc : Famix.Concretisation) => {\n                                    if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){\n                                        createConcretisation = false;\n                                    }\n                                });\n        \n                                if (createConcretisation) {\n                                    const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);\n                                }\n                            }\n                        }\n                        break;\n                    }\n                    // Remonter à l'élément parent (utile si le nœud de référence est un enfant)\n                    currentNode = currentNode.getParent();\n                }\n            });\n        }\n    }\n\n    /**\n     * Creates a Famix concretisation between a class and an interface\n     * @param cls A class\n     */\n    public createFamixConcretisationInterfaceClass(cls: ClassDeclaration){\n    \n        const superInterfaces = cls.getImplements();\n        superInterfaces.forEach(interfaceType => {\n            const interfaceIsGeneric = interfaceType.getTypeArguments().length>0;\n            if (interfaceIsGeneric) {\n                const interfaceDeclaration = interfaceType.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;\n                const genParams = interfaceDeclaration.getTypeParameters().map((param) => param.getText());\n                const conParams = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());\n                const args = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();\n                if (!Helpers.arraysAreEqual(conParams,genParams)) {\n                    const genInterface = this.createOrGetFamixInterface(interfaceDeclaration) as Famix.ParametricInterface;\n                    const conInterface = this.createOrGetFamixConcreteElement(genInterface,interfaceDeclaration,args);\n                    const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\n                    let createConcretisation : boolean = true;\n                    concretisations.forEach((conc : Famix.Concretisation) => {\n                        if (genInterface.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conInterface.fullyQualifiedName){\n                            createConcretisation = false;\n                        }\n                    });\n            \n                    if (createConcretisation) {\n                        const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conInterface,genInterface);\n                    }\n                }\n            }\n        });\n    }\n\n    /**\n     * Creates a Famix concretisation between an interface and a Type\n     * @param element A variable or a function\n     * @param inter An interface\n     */\n    public createFamixConcretisationTypeInstanciation(element: InterfaceDeclaration | ClassDeclaration){\n\n        const isGeneric = element.getTypeParameters().length > 0;\n        if (isGeneric) {\n            const genParams = element.getTypeParameters().map(param => param.getText());\n            const uses = element.findReferencesAsNodes();\n            uses.forEach(use => {        \n                let parentNode = use.getParent();\n                while (parentNode) {\n                    if (parentNode.getKind() === SyntaxKind.TypeReference) {\n                        const typeReferenceNode = parentNode.asKind(SyntaxKind.TypeReference);\n                        if (!typeReferenceNode) {\n                            throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);\n                        }\n                        const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;\n                        if (typeReferenceNodeIsGeneric) {}\n                            const args = typeReferenceNode.getTypeArguments();\n                            const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());\n                            if (!Helpers.arraysAreEqual(conParams,genParams)) {\n                                let genElement;\n                                if(element instanceof ClassDeclaration){\n                                    genElement = this.createOrGetFamixClass(element) as Famix.ParametricClass;\n                                } else {\n                                    genElement = this.createOrGetFamixInterface(element) as Famix.ParametricInterface;\n                                }\n                                let concElement;\n                                concElement = this.createOrGetFamixConcreteElement(genElement,element,args);\n                                const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\n                                let createConcretisation : boolean = true;\n                                concretisations.forEach((conc : Famix.Concretisation) => {\n                                    if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){\n                                        createConcretisation = false;\n                                    }\n                                });\n        \n                                if (createConcretisation) {\n                                    const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);\n                                }\n                            }\n                        break;\n                    }\n                    parentNode = parentNode.getParent();\n                }\n            });\n        }\n    }\n\n    public convertToRelativePath(absolutePath: string, absolutePathProject: string) {\n        return absolutePath.replace(absolutePathProject, \"\").slice(1);\n    }\n}\nfunction initFQN(sourceElement: TSMorphObjectType, famixElement: Famix.SourcedEntity) {\n    if (!(sourceElement instanceof CommentRange)) {\n        const fqn = FQNFunctions.getFQN(sourceElement);\n        logger.debug(\"Setting fully qualified name for \" + famixElement.getJSON() + \" to \" + fqn);\n        (famixElement as Famix.NamedEntity).fullyQualifiedName = fqn;\n    }\n}\n\n"]}
1706
+ function isTypeContext(sourceElement) {
1707
+ return sourceElement instanceof ts_morph_1.ConstructorDeclaration
1708
+ || sourceElement instanceof ts_morph_1.MethodDeclaration
1709
+ || sourceElement instanceof ts_morph_1.FunctionDeclaration
1710
+ || sourceElement instanceof ts_morph_1.FunctionExpression
1711
+ || sourceElement instanceof ts_morph_1.ArrowFunction
1712
+ ||
1713
+ sourceElement instanceof ts_morph_1.ParameterDeclaration ||
1714
+ sourceElement instanceof ts_morph_1.VariableDeclaration ||
1715
+ sourceElement instanceof ts_morph_1.PropertyDeclaration ||
1716
+ sourceElement instanceof ts_morph_1.PropertySignature ||
1717
+ sourceElement instanceof ts_morph_1.TypeParameterDeclaration ||
1718
+ sourceElement instanceof ts_morph_1.Identifier ||
1719
+ sourceElement instanceof ts_morph_1.Decorator ||
1720
+ sourceElement instanceof ts_morph_1.GetAccessorDeclaration ||
1721
+ sourceElement instanceof ts_morph_1.SetAccessorDeclaration ||
1722
+ sourceElement instanceof ts_morph_1.ImportSpecifier ||
1723
+ sourceElement instanceof ts_morph_1.EnumDeclaration ||
1724
+ sourceElement instanceof ts_morph_1.EnumMember ||
1725
+ sourceElement instanceof ts_morph_1.TypeAliasDeclaration
1726
+ || sourceElement instanceof ts_morph_1.ImportDeclaration
1727
+ || sourceElement instanceof ts_morph_1.ExpressionWithTypeArguments;
1728
+ }
1729
+ function getInterfaceDeclarationFromExpression(expression) {
1730
+ // Step 1: Get the type of the expression
1731
+ const type = expression.getType();
1732
+ // Step 2: Get the symbol associated with the type
1733
+ let symbol = type.getSymbol();
1734
+ if (!symbol) {
1735
+ // If symbol is not found, try to get the symbol from the identifier
1736
+ const identifier = expression.getFirstDescendantByKind(ts_morph_1.SyntaxKind.Identifier);
1737
+ if (!identifier) {
1738
+ throw new Error(`Identifier not found for ${expression.getText()}.`);
1739
+ }
1740
+ symbol = identifier.getSymbol();
1741
+ if (!symbol) {
1742
+ throw new Error(`Symbol not found for ${identifier.getText()}.`);
1743
+ }
1744
+ }
1745
+ // Step 3: Resolve the symbol to find the actual declaration
1746
+ const interfaceDeclaration = resolveSymbolToInterfaceDeclaration(symbol);
1747
+ if (!interfaceDeclaration) {
1748
+ analyze_1.logger.error(`Interface declaration not found for ${expression.getText()}.`);
1749
+ }
1750
+ return interfaceDeclaration;
1751
+ }
1752
+ function resolveSymbolToInterfaceDeclaration(symbol) {
1753
+ // Get the declarations associated with the symbol
1754
+ const declarations = symbol.getDeclarations();
1755
+ // Filter for InterfaceDeclaration
1756
+ const interfaceDeclaration = declarations.find(declaration => declaration instanceof ts_morph_1.InterfaceDeclaration);
1757
+ if (interfaceDeclaration) {
1758
+ return interfaceDeclaration;
1759
+ }
1760
+ // Handle imports: If the symbol is imported, resolve the import to find the actual declaration
1761
+ for (const declaration of declarations) {
1762
+ if (declaration.getKind() === ts_morph_1.SyntaxKind.ImportSpecifier) {
1763
+ const importSpecifier = declaration;
1764
+ const importDeclaration = importSpecifier.getImportDeclaration();
1765
+ const moduleSpecifier = importDeclaration.getModuleSpecifierSourceFile();
1766
+ if (moduleSpecifier) {
1767
+ const exportedSymbols = moduleSpecifier.getExportSymbols();
1768
+ const exportedSymbol = exportedSymbols.find(symbol => symbol.getName() === importSpecifier.getName());
1769
+ if (exportedSymbol) {
1770
+ return resolveSymbolToInterfaceDeclaration(exportedSymbol);
1771
+ }
1772
+ }
1773
+ }
1774
+ }
1775
+ return undefined;
1776
+ }
1777
+ // function oldGetInterfaceDeclarationFromExpression(expression: ExpressionWithTypeArguments): InterfaceDeclaration | undefined {
1778
+ // // Two cases:
1779
+ // // class A implements ImportedInterface, DeclaredInterface {}
1780
+ // const type = expression.getType();
1781
+ // // ImportedInterface: type will a symbol
1782
+ // let symbol = type.getAliasSymbol(); // will be defined for imported interfaces
1783
+ // if (!symbol) {
1784
+ // // DeclaredInterface: type will be an InterfaceDeclaration on Identifier node that is the child of the ExpressionWithTypeArguments
1785
+ // const identifier = expression.getFirstDescendantByKind(SyntaxKind.Identifier);
1786
+ // if (!identifier) {
1787
+ // throw new Error(`Identifier not found for ${expression.getText()}.`);
1788
+ // }
1789
+ // symbol = identifier.getSymbol();
1790
+ // if (!symbol) {
1791
+ // throw new Error(`Symbol not found for ${identifier.getText()}.`);
1792
+ // }
1793
+ // }
1794
+ // // Step 3: Get the declarations associated with the symbol
1795
+ // const declarations = symbol.getDeclarations();
1796
+ // // Step 4: Filter for InterfaceDeclaration
1797
+ // const interfaceDeclaration = declarations.find(declaration => declaration instanceof InterfaceDeclaration) as InterfaceDeclaration | undefined;
1798
+ // if (!interfaceDeclaration) {
1799
+ // throw new Error(`Interface declaration not found for ${expression.getText()}.`);
1800
+ // }
1801
+ // return interfaceDeclaration;
1802
+ // }
1803
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"EntityDictionary.js","sourceRoot":"","sources":["../../src/famix_functions/EntityDictionary.ts"],"names":[],"mappings":";AAAA;;;;GAIG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAo0DH,0CAYC;AA70DD,uCAA6kB;AAC7kB,8EAAgF;AAChF,gEAAkD;AAClD,oEAAgE;AAChE,wCAA4C;AAC5C,oDAAoD;AACpD,iEAAiE;AACjE,sDAAuD;AACvD,4DAA8C;AAC9C,qDAAuC;AACvC,gDAAwB;AAexB,MAAa,gBAAgB;IAqBzB;QAnBO,aAAQ,GAAG,IAAI,kCAAe,EAAE,CAAC;QAChC,gBAAW,GAAG,IAAI,GAAG,EAAuB,CAAC,CAAC,4CAA4C;QAC1F,gBAAW,GAAG,IAAI,GAAG,EAA+C,CAAC,CAAC,4DAA4D;QAClI,oBAAe,GAAG,IAAI,GAAG,EAAuD,CAAC,CAAC,gDAAgD;QAClI,iBAAY,GAAG,IAAI,GAAG,EAAmC,CAAC,CAAC,gDAAgD;QAC3G,eAAU,GAAG,IAAI,GAAG,EAA6C,CAAC,CAAC,kDAAkD;QACrH,eAAU,GAAG,IAAI,GAAG,EAA4D,CAAC,CAAC,mDAAmD;QACrI,wBAAmB,GAAG,IAAI,GAAG,EAA+B,CAAC,CAAC,qDAAqD;QACnH,4BAAuB,GAAG,IAAI,GAA8F,CAAC,CAAC,+CAA+C;QAC7K,wBAAmB,GAAG,IAAI,GAAgC,CAAC,CAAC,+CAA+C;QAC3G,oBAAe,GAAG,IAAI,GAAG,EAAyC,CAAC,CAAC,2CAA2C;QAC/G,mBAAc,GAAG,IAAI,GAAG,EAAuC,CAAC,CAAC,0CAA0C;QAC3G,uBAAkB,GAAG,IAAI,GAAG,EAAmE,CAAC,CAAC,+CAA+C;QAChJ,eAAU,GAAG,IAAI,GAAG,EAA+B,CAAC,CAAC,2CAA2C;QAChG,sBAAiB,GAAG,IAAI,GAAG,EAA6B,CAAC,CAAC,kDAAkD;QAC5G,kBAAa,GAAG,gCAAgC,CAAC,CAAC,6CAA6C;QAChG,wBAAmB,GAAG,IAAI,GAAG,EAAkC,CAAC;QAChE,4BAAuB,GAAG,IAAI,GAAG,EAAkC,CAAC;QAGvE,IAAI,CAAC,QAAQ,CAAC,sBAAsB,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;IACnE,CAAC;IAEM,eAAe,CAAC,GAAwB,EAAE,IAAuB;QACpE,MAAM,YAAY,GAA4B,IAAI,KAAK,CAAC,iBAAiB,EAAE,CAAC;QAC5E,IAAI,WAAW,EAAE,SAAiB,CAAC;QACnC,IAAI,GAAG,IAAI,IAAI,EAAE,CAAC;YACd,yDAAyD;YACzD,IAAI,CAAC,CAAC,IAAI,YAAY,uBAAY,CAAC,EAAE,CAAC;gBAClC,WAAW,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;gBAC9B,SAAS,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;YAC9B,CAAC;iBAAM,CAAC;gBACJ,WAAW,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;gBAC5B,SAAS,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;YAC9B,CAAC;YAED,IAAI,gBAAM,CAAC,eAAe,EAAE,CAAC;gBACzB;;;;;;;;;mBASG;gBACH,MAAM,QAAQ,GAAG,IAAI,gBAAgB,EAAE,CAAC;gBACxC,MAAM,cAAc,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC,WAAW,EAAE,CAAC;gBAC1D,MAAM,mBAAmB,GAAG,QAAQ,CAAC,cAAc,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;gBACxE,IAAI,mBAAmB,EAAE,CAAC;oBACtB,MAAM,iBAAiB,GAAG,cAAc,CAAC,SAAS,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;oBAC3E,MAAM,0BAA0B,GAAG,QAAQ,CAAC,cAAc,CAAC,iBAAiB,CAAC,CAAC;oBAC9E,MAAM,uBAAuB,GAAG,QAAQ,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;oBACxE,MAAM,mCAAmC,GAAG,QAAQ,CAAC,cAAc,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,CAAC;oBAE9G,8FAA8F;oBAC9F,WAAW,GAAG,OAAO,CAAC,iBAAiB,CAAC,EAAC,WAAW,EAAE,uBAAuB;wBACzC,WAAW,EAAE,0BAA0B;wBACvC,KAAK,EAAE,WAAW,GAAG,mCAAmC,EAAC,CAAC,CAAC;oBAC/F,SAAS,GAAG,WAAW,GAAG,0BAA0B,CAAC,MAAM,CAAC;gBAChE,CAAC;YACL,CAAC;YAED,kFAAkF;YAClF,YAAY,CAAC,QAAQ,GAAG,WAAW,GAAG,CAAC,CAAC;YACxC,YAAY,CAAC,MAAM,GAAG,SAAS,GAAG,CAAC,CAAC;YAEpC,IAAI,QAAQ,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC,WAAW,EAAY,CAAC;YAC5D,IAAI,QAAQ,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;gBAC3B,QAAQ,GAAG,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;YACrC,CAAC;YACD,YAAY,CAAC,OAAO,GAAG,GAAG,CAAC;YAC3B,YAAY,CAAC,QAAQ,GAAG,QAAQ,CAAC;YACjC,GAAG,CAAC,YAAY,GAAG,YAAY,CAAC;YAChC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;QAE3C,CAAC;QACD,OAAO,YAAY,CAAC;IACxB,CAAC;IAED;;;;OAIG;IACI,wBAAwB,CAAC,aAAgC,EAAE,YAAiC;QAC/F,2EAA2E;QAC3E,IAAI,CAAC,CAAC,YAAY,YAAY,KAAK,CAAC,OAAO,CAAC,EAAE,CAAC,CAAE,yBAAyB;YACtE,gEAAgE;YAChE,MAAM,kBAAkB,GAAI,YAAkC,CAAC,kBAAkB,CAAC;YAClF,IAAI,CAAC,kBAAkB,IAAI,kBAAkB,KAAK,IAAI,CAAC,aAAa,EAAE,CAAC;gBACnE,MAAM,IAAI,KAAK,CAAC,iBAAiB,YAAY,CAAC,WAAW,CAAC,IAAI,mCAAmC,CAAC,CAAC;YACvG,CAAC;QACL,CAAC;QAED,gBAAM,CAAC,KAAK,CAAC,gCAAgC,GAAG,aAAa,EAAE,OAAO,EAAE,GAAG,sBAAsB,GAAG,YAAY,CAAC,OAAO,EAAE,CAAC,CAAC;QAC5H,MAAM,kBAAkB,GAAG,IAAI,KAAK,CAAC,iBAAiB,EAAE,CAAC;QACzD,kBAAkB,CAAC,OAAO,GAAG,YAAY,CAAC;QAC1C,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,YAAY,EAAE,aAAa,CAAC,CAAC;QAE1D,IAAI,aAAa,KAAK,IAAI,EAAE,CAAC;YACzB,MAAM,mBAAmB,GAAG,IAAI,CAAC,QAAQ,CAAC,eAAe,EAAE,CAAC;YAE5D,MAAM,YAAY,GAAG,cAAI,CAAC,SAAS,CAAC,aAAa,CAAC,aAAa,EAAE,CAAC,WAAW,EAAE,CAAC,CAAC;YAEjF,MAAM,mBAAmB,GAAG,YAAY,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;YAEjE,IAAI,aAAa,GAAW,EAAE,CAAC;YAE/B,IAAI,mBAAmB,KAAK,CAAC,CAAC,EAAE,CAAC;gBAC7B,MAAM,mBAAmB,GAAG,YAAY,CAAC,SAAS,CAAC,mBAAmB,CAAC,CAAC;gBACxE,aAAa,GAAG,mBAAmB,CAAC;YACxC,CAAC;iBAAM,CAAC;gBACJ,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,mBAAmB,CAAC,CAAC;YAClF,CAAC;YAED,wFAAwF;YACxF,aAAa,GAAG,aAAa,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;YAElD,IAAI,aAAa,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;gBAChC,aAAa,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;YAC/C,CAAC;YAED,kBAAkB,CAAC,QAAQ,GAAG,aAAa,CAAC;YAC5C,IAAI,WAAW,EAAE,SAET,CAAC;YACT,IAAI,CAAC,CAAC,aAAa,YAAY,uBAAY,CAAC,EAAE,CAAC;gBAC3C,WAAW,GAAG,aAAa,CAAC,QAAQ,EAAE,CAAC;gBACvC,SAAS,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC;gBACnC,wDAAwD;gBACxD,oDAAoD;YACxD,CAAC;iBAAM,CAAC;gBACJ,WAAW,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC;gBACrC,SAAS,GAAG,aAAa,CAAC,MAAM,EAAE,CAAC;YACvC,CAAC;YACD,IAAI,gBAAM,CAAC,eAAe,EAAE,CAAC;gBACzB;;;;;;;;;mBASG;gBACH,MAAM,QAAQ,GAAG,IAAI,gBAAgB,EAAE,CAAC;gBACxC,MAAM,cAAc,GAAG,aAAa,CAAC,aAAa,EAAE,CAAC,WAAW,EAAE,CAAC;gBACnE,MAAM,mBAAmB,GAAG,QAAQ,CAAC,cAAc,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;gBACxE,IAAI,mBAAmB,EAAE,CAAC;oBACtB,MAAM,iBAAiB,GAAG,cAAc,CAAC,SAAS,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;oBAC3E,MAAM,0BAA0B,GAAG,QAAQ,CAAC,cAAc,CAAC,iBAAiB,CAAC,CAAC;oBAC9E,MAAM,uBAAuB,GAAG,QAAQ,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;oBACxE,MAAM,mCAAmC,GAAG,QAAQ,CAAC,cAAc,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC,CAAC;oBAE9G,8FAA8F;oBAC9F,WAAW,GAAG,OAAO,CAAC,iBAAiB,CAAC,EAAC,WAAW,EAAE,uBAAuB;wBACzC,WAAW,EAAE,0BAA0B;wBACvC,KAAK,EAAE,WAAW,GAAG,mCAAmC,EAAC,CAAC,CAAC;oBAC/F,SAAS,GAAG,WAAW,GAAG,0BAA0B,CAAC,MAAM,CAAC;gBAChE,CAAC;YACL,CAAC;YACD,gFAAgF;YAChF,kBAAkB,CAAC,QAAQ,GAAG,WAAW,GAAG,CAAC,CAAC;YAC9C,kBAAkB,CAAC,MAAM,GAAG,SAAS,GAAG,CAAC,CAAC;YAE1C,+cAA+c;YAC/c,2CAA2C;YAC3C,IAAI;QACR,CAAC;aAAM,CAAC;YACJ,wBAAwB;YACxB,gBAAM,CAAC,IAAI,CAAC,yCAAyC,GAAG,YAAY,CAAC,OAAO,EAAE,CAAC,CAAC;YAChF,kBAAkB,CAAC,QAAQ,GAAG,SAAS,CAAC;YACxC,kBAAkB,CAAC,QAAQ,GAAG,CAAC,CAAC;YAChC,kBAAkB,CAAC,MAAM,GAAG,CAAC,CAAC;QAClC,CAAC;QAED,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,kBAAkB,CAAC,CAAC;IACjD,CAAC;IAED;;;;;OAKG;IACI,oBAAoB,CAAC,CAAa,EAAE,QAAiB;QACxD,IAAI,OAA2B,CAAC,CAAC,kBAAkB;QAEnD,MAAM,QAAQ,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC;QACjC,MAAM,sBAAsB,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC;QAC/C,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;QAClE,IAAI,CAAC,aAAa,EAAE,CAAC;YACjB,IAAI,QAAQ,EAAE,CAAC;gBACX,OAAO,GAAG,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;YACjC,CAAC;iBACI,CAAC;gBACF,OAAO,GAAG,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;YACvC,CAAC;YACD,OAAO,CAAC,IAAI,GAAG,QAAQ,CAAC;YACxB,OAAO,CAAC,mBAAmB,GAAG,CAAC,CAAC,gBAAgB,EAAE,GAAG,CAAC,CAAC,kBAAkB,EAAE,CAAC;YAC5E,OAAO,CAAC,kBAAkB,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC;YAEpD,OAAO,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;YAEpB,IAAI,CAAC,wBAAwB,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;YAE1C,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,sBAAsB,EAAE,OAAO,CAAC,CAAC;YACrD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QACtC,CAAC;aACI,CAAC;YACF,OAAO,GAAG,aAAa,CAAC;QAC5B,CAAC;QAED,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,OAAO,EAAC,CAAC,CAAC,CAAC;QACxC,OAAO,OAAO,CAAC;IACnB,CAAC;IAED;;;;OAIG;IACI,sBAAsB,CAAC,iBAAoC;QAC9D,IAAI,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,iBAAiB,CAAC,EAAE,CAAC;YAC3C,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;YACzD,IAAI,OAAO,EAAE,CAAC;gBACX,OAAO,OAAO,CAAC;YAClB,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,gBAAgB,iBAAiB,CAAC,OAAO,EAAE,kCAAkC,CAAC,CAAC;YACnG,CAAC;QACL,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;QACrC,MAAM,UAAU,GAAG,iBAAiB,CAAC,OAAO,EAAE,CAAC;QAC/C,SAAS,CAAC,IAAI,GAAG,UAAU,CAAC;QAC5B,SAAS,CAAC,SAAS,GAAG,IAAA,6BAAS,EAAC,iBAAiB,CAAC,CAAC;QACnD,SAAS,CAAC,WAAW,GAAG,IAAA,+BAAW,EAAC,iBAAiB,CAAC,CAAC;QACvD,SAAS,CAAC,QAAQ,GAAG,CAAC,SAAS,CAAC,WAAW,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC;QAEpE,OAAO,CAAC,iBAAiB,EAAE,SAAS,CAAC,CAAC;QACtC,IAAI,CAAC,wBAAwB,CAAC,iBAAiB,EAAE,SAAS,CAAC,CAAC;QAE5D,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,iBAAiB,EAAE,SAAS,CAAC,CAAC;QAEpD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;QAEpC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,SAAS,EAAC,iBAAiB,CAAC,CAAC;QAC1D,OAAO,SAAS,CAAC;IACrB,CAAC;IAED;;;;OAIG;IACI,gBAAgB,CAAC,CAAuB;QAC3C,IAAI,QAAqB,CAAC;QAC1B,MAAM,SAAS,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC;QAC9B,mFAAmF;QACnF,MAAM,uBAAuB,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;QACvD,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;QACjE,IAAI,CAAC,UAAU,EAAE,CAAC;YACd,QAAQ,GAAG,IAAI,KAAK,CAAC,KAAK,EAAE,CAAC;YAC7B,QAAQ,CAAC,IAAI,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC;YAC5B,MAAM,qBAAqB,GAAG,SAAS,GAAG,CAAC,CAAC,CAAC,iBAAiB,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,iBAAiB,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YACvJ,gBAAM,CAAC,KAAK,CAAC,iBAAiB,SAAS,6BAA6B,uBAAuB,2BAA2B,qBAAqB,GAAG,CAAC,CAAC;YAEhJ,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,qBAAqB,EAAE,CAAC,CAAC,CAAC;YACpE,QAAQ,CAAC,aAAa,GAAG,OAAO,CAAC;YACjC,OAAO,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC;YACrB,IAAI,CAAC,wBAAwB,CAAC,CAAC,EAAE,QAAQ,CAAC,CAAC;YAE3C,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,uBAAuB,EAAE,QAAQ,CAAC,CAAC;YAExD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QACvC,CAAC;aACI,CAAC;YACF,QAAQ,GAAG,UAAU,CAAC;QAC1B,CAAC;QACD,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,QAAQ,EAAC,CAAC,CAAC,CAAC;QAEzC,OAAO,QAAQ,CAAC;IACpB,CAAC;IAED;;;;OAIG;IACI,qBAAqB,CAAC,GAAqB;QAC9C,IAAI,QAA6C,CAAC;QAClD,MAAM,UAAU,GAAG,GAAG,CAAC,UAAU,EAAE,CAAC;QACpC,MAAM,uBAAuB,GAAG,YAAY,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACzD,MAAM,OAAO,GAAG,GAAG,CAAC,OAAO,EAAE,IAAI,IAAI,CAAC,aAAa,CAAC;QACpD,MAAM,SAAS,GAAG,GAAG,CAAC,iBAAiB,EAAE,CAAC,MAAM,CAAC;QACjD,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;QACjE,IAAI,CAAC,UAAU,EAAE,CAAC;YACd,IAAI,SAAS,EAAE,CAAC;gBACZ,QAAQ,GAAG,IAAI,KAAK,CAAC,eAAe,EAAE,CAAC;YAC3C,CAAC;iBACI,CAAC;gBACF,QAAQ,GAAG,IAAI,KAAK,CAAC,KAAK,EAAE,CAAC;YACjC,CAAC;YAED,QAAQ,CAAC,IAAI,GAAG,OAAO,CAAC;YACxB,OAAO,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;YACvB,yDAAyD;YACzD,QAAQ,CAAC,UAAU,GAAG,UAAU,CAAC;YAEjC,IAAI,CAAC,wBAAwB,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;YAE7C,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,uBAAuB,EAAE,QAAQ,CAAC,CAAC;YAExD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;YAEnC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,QAAQ,EAAC,GAAG,CAAC,CAAC;QAC/C,CAAC;aACI,CAAC;YACF,QAAQ,GAAG,UAAU,CAAC;QAC1B,CAAC;QAED,OAAO,QAAQ,CAAC;IACpB,CAAC;IAED;;;;OAIG;IACI,yBAAyB,CAAC,KAA2B;QAExD,IAAI,YAAyD,CAAC;QAC9D,MAAM,SAAS,GAAG,KAAK,CAAC,OAAO,EAAE,CAAC;QAClC,MAAM,uBAAuB,GAAG,YAAY,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC3D,MAAM,cAAc,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;QACzE,IAAI,CAAC,cAAc,EAAE,CAAC;YAClB,MAAM,SAAS,GAAG,KAAK,CAAC,iBAAiB,EAAE,CAAC,MAAM,CAAC;YACnD,IAAI,SAAS,EAAE,CAAC;gBACZ,YAAY,GAAG,IAAI,KAAK,CAAC,mBAAmB,EAAE,CAAC;YACnD,CAAC;iBACI,CAAC;gBACF,YAAY,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;YACzC,CAAC;YAED,YAAY,CAAC,IAAI,GAAG,SAAS,CAAC;YAC9B,OAAO,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;YAC7B,IAAI,CAAC,wBAAwB,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;YAEnD,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,uBAAuB,EAAE,YAAY,CAAC,CAAC;YAEhE,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;YAEvC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,YAAY,EAAC,KAAK,CAAC,CAAC;QACrD,CAAC;aACI,CAAC;YACF,YAAY,GAAG,cAAc,CAAC;QAClC,CAAC;QACD,OAAO,YAAY,CAAC;IACxB,CAAC;IAGD;;;;;;OAMG;IACH,mFAAmF;IACnF,mGAAmG;IACnG,iGAAiG;IAEjG,uEAAuE;IACvE,uBAAuB;IAEvB,yCAAyC;IACzC,+CAA+C;IAC/C,UAAU;IAEV,sDAAsD;IAEtD,8FAA8F;IAE9F,8CAA8C;IAE9C,gEAAgE;IAChE,4DAA4D;IAC5D,sEAAsE;IACtE,uDAAuD;IACvD,mEAAmE;IACnE,gDAAgD;IAChD,6CAA6C;IAC7C,0EAA0E;IAC1E,2DAA2D;IAC3D,aAAa;IAEb,kEAAkE;IAClE,kGAAkG;IAClG,6EAA6E;IAC7E,0GAA0G;IAC1G,4EAA4E;IAC5E,iHAAiH;IACjH,+EAA+E;IAC/E,+GAA+G;IAC/G,YAAY;IACZ,iDAAiD;IACjD,gFAAgF;IAChF,eAAe;IACf,kEAAkE;IAClE,mGAAmG;IACnG,6EAA6E;IAC7E,2GAA2G;IAC3G,4EAA4E;IAC5E,kHAAkH;IAClH,gFAAgF;IAChF,gHAAgH;IAChH,YAAY;IACZ,QAAQ;IACR,0BAA0B;IAC1B,IAAI;IAEJ;;;;OAIG;IACI,mBAAmB,CAAC,QAAiD;QACxE,MAAM,WAAW,GAAG,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;QACzC,MAAM,WAAW,GAAG,QAAQ,YAAY,4BAAiB,CAAC;QAC1D,WAAW,CAAC,IAAI,GAAG,QAAQ,CAAC,OAAO,EAAE,CAAC;QAEtC,IAAI,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC;QACtC,IAAI,CAAC;YACD,YAAY,GAAG,QAAQ,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,CAAC;QACvD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACb,gBAAM,CAAC,KAAK,CAAC,4BAA4B,KAAK,6CAA6C,QAAQ,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC;QACpI,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,QAAQ,CAAC,CAAC;QAClE,WAAW,CAAC,YAAY,GAAG,OAAO,CAAC;QAEnC,gEAAgE;QAChE,WAAW,CAAC,UAAU,GAAG,EAAE,CAAC;QAE5B,QAAQ,CAAC,YAAY,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;YAChC,QAAQ,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC;gBAClB,KAAK,gBAAK,CAAC,MAAM;oBACb,WAAW,CAAC,UAAU,GAAG,QAAQ,CAAC;oBAClC,MAAM;gBACV,KAAK,gBAAK,CAAC,SAAS;oBAChB,WAAW,CAAC,UAAU,GAAG,WAAW,CAAC;oBACrC,MAAM;gBACV,KAAK,gBAAK,CAAC,OAAO;oBACd,WAAW,CAAC,UAAU,GAAG,SAAS,CAAC;oBACnC,MAAM;gBACV,KAAK,QAAQ;oBACT,WAAW,CAAC,WAAW,GAAG,IAAI,CAAC;oBAC/B,MAAM;gBACV,KAAK,UAAU;oBACX,WAAW,CAAC,QAAQ,GAAG,IAAI,CAAC;oBAC5B,MAAM;gBACV;oBACI,MAAM;YACd,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,WAAW,IAAI,QAAQ,CAAC,uBAAuB,EAAE,EAAE,CAAC;YACrD,WAAW,CAAC,oBAAoB,GAAG,IAAI,CAAC;QAC5C,CAAC;QACD,IAAI,QAAQ,CAAC,oBAAoB,EAAE,EAAE,CAAC;YAClC,WAAW,CAAC,UAAU,GAAG,IAAI,CAAC;QAClC,CAAC;QACD,IAAI,QAAQ,CAAC,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,GAAG,EAAE,CAAC;YAC7C,WAAW,CAAC,mBAAmB,GAAG,IAAI,CAAC;QAC3C,CAAC;QAED,OAAO,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;QAC/B,IAAI,CAAC,wBAAwB,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;QAErD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;QAEtC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,WAAW,EAAC,QAAQ,CAAC,CAAC;QAEnD,OAAO,WAAW,CAAC;IACvB,CAAC;IAED;;;;;OAKG;IACI,sBAAsB,CAAC,MAAsH,EAAE,SAAoC;QACtL,IAAI,SAAiE,CAAC;QACtE,MAAM,SAAS,GAAG,MAAM,CAAC,iBAAiB,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;QACxD,MAAM,0BAA0B,GAAG,YAAY,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC/D,IAAI,CAAC,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,CAAC,EAAE,CAAC;YAEhE,IAAI,MAAM,YAAY,iCAAsB,IAAI,MAAM,YAAY,iCAAsB,EAAE,CAAC;gBACvF,SAAS,GAAG,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;gBACjC,MAAM,QAAQ,GAAG,MAAM,YAAY,iCAAsB,CAAC;gBAC1D,MAAM,QAAQ,GAAG,MAAM,YAAY,iCAAsB,CAAC;gBAC1D,IAAI,QAAQ,EAAE,CAAC;oBAAC,SAA4B,CAAC,IAAI,GAAG,QAAQ,CAAC;gBAAA,CAAC;gBAC9D,IAAI,QAAQ,EAAE,CAAC;oBAAC,SAA4B,CAAC,IAAI,GAAG,QAAQ,CAAC;gBAAA,CAAC;gBAC9D,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YACxC,CAAC;iBACI,CAAC;gBACF,IAAI,SAAS,EAAE,CAAC;oBACZ,SAAS,GAAG,IAAI,KAAK,CAAC,gBAAgB,EAAE,CAAC;gBAC7C,CAAC;qBACI,CAAC;oBACF,SAAS,GAAG,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;gBACnC,CAAC;YACL,CAAC;YACD,MAAM,aAAa,GAAG,MAAM,YAAY,iCAAsB,CAAC;YAC/D,MAAM,WAAW,GAAG,MAAM,YAAY,0BAAe,CAAC;YAEtD,IAAI,UAAU,GAAG,KAAK,CAAC;YACvB,IAAI,QAAQ,GAAG,KAAK,CAAC;YACrB,IAAI,MAAM,YAAY,4BAAiB,IAAI,MAAM,YAAY,iCAAsB,IAAI,MAAM,YAAY,iCAAsB,EAAE,CAAC;gBAC9H,UAAU,GAAG,MAAM,CAAC,UAAU,EAAE,CAAC;gBACjC,QAAQ,GAAG,MAAM,CAAC,QAAQ,EAAE,CAAC;YACjC,CAAC;YAED,IAAI,aAAa,EAAE,CAAC;gBAAC,SAA4B,CAAC,IAAI,GAAG,aAAa,CAAC;YAAA,CAAC;YACxE,SAAS,CAAC,UAAU,GAAG,UAAU,CAAC;YAClC,SAAS,CAAC,WAAW,GAAG,QAAQ,CAAC;YACjC,SAAS,CAAC,SAAS,GAAG,CAAC,MAAM,YAAY,4BAAiB,IAAI,MAAM,YAAY,iCAAsB,IAAI,MAAM,YAAY,iCAAsB,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,KAAK,SAAS,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC;YACzO,SAAS,CAAC,WAAW,GAAG,CAAC,MAAM,YAAY,4BAAiB,IAAI,MAAM,YAAY,iCAAsB,IAAI,MAAM,YAAY,iCAAsB,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,KAAK,WAAW,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC;YAC7O,SAAS,CAAC,SAAS,GAAG,OAAO,CAAC,gBAAgB,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC;YAEjE,IAAI,UAAkB,CAAC;YACvB,IAAI,aAAa,EAAE,CAAC;gBAChB,UAAU,GAAG,aAAa,CAAC;YAC/B,CAAC;iBACI,CAAC;gBACF,UAAU,GAAI,MAAgG,CAAC,OAAO,EAAE,CAAC;YAC7H,CAAC;YACD,SAAS,CAAC,IAAI,GAAG,UAAU,CAAC;YAE5B,IAAI,CAAC,aAAa,EAAE,CAAC;gBACjB,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,GAAG,EAAE,CAAC;oBAC3C,SAAS,CAAC,SAAS,GAAG,IAAI,CAAC;gBAC/B,CAAC;YACL,CAAC;YAED,IAAI,CAAC,SAAS,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;gBACjD,SAAS,CAAC,QAAQ,GAAG,IAAI,CAAC;YAC9B,CAAC;iBACI,CAAC;gBACF,SAAS,CAAC,QAAQ,GAAG,KAAK,CAAC;YAC/B,CAAC;YAED,IAAI,CAAC,WAAW,EAAE,CAAC;gBACf,SAAS,CAAC,oBAAoB,GAAG,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YAC/D,CAAC;iBACI,CAAC;gBACF,SAAS,CAAC,oBAAoB,GAAG,CAAC,CAAC;YACvC,CAAC;YAED,IAAI,cAAc,GAAG,IAAI,CAAC,aAAa,CAAC;YACxC,IAAI,CAAC;gBACD,cAAc,GAAG,MAAM,CAAC,aAAa,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,CAAC;YAC7D,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACb,gBAAM,CAAC,KAAK,CAAC,4BAA4B,KAAK,0DAA0D,SAAS,CAAC,IAAI,iBAAiB,CAAC,CAAC;YAC7I,CAAC;YAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,EAAE,MAAM,CAAC,CAAC;YAClE,SAAS,CAAC,YAAY,GAAG,OAAO,CAAC;YACjC,SAAS,CAAC,mBAAmB,GAAG,MAAM,CAAC,gBAAgB,EAAE,GAAG,MAAM,CAAC,kBAAkB,EAAE,CAAC;YACxF,MAAM,UAAU,GAAG,MAAM,CAAC,aAAa,EAAE,CAAC;YAC1C,SAAS,CAAC,kBAAkB,GAAG,UAAU,CAAC,MAAM,CAAC;YAEjD,IAAI,CAAC,WAAW,EAAE,CAAC;gBACf,SAAS,CAAC,kBAAkB,GAAG,MAAM,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC;YACjE,CAAC;iBACI,CAAC;gBACF,SAAS,CAAC,kBAAkB,GAAG,CAAC,CAAC;YACrC,CAAC;YAED,OAAO,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;YAC3B,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;YACpC,IAAI,CAAC,wBAAwB,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;YAEjD,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,EAAE,SAAS,CAAC,CAAC;QAC5E,CAAC;aACI,CAAC;YACF,SAAS,GAAG,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,CAA6D,CAAC;QACzI,CAAC;QAED,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,SAAS,EAAC,MAAM,CAAC,CAAC;QAE/C,OAAO,SAAS,CAAC;IACrB,CAAC;IAED;;;;;OAKG;IACI,wBAAwB,CAAC,IAA8C,EAAE,SAAoC;QAChH,IAAI,WAAsD,CAAC;QAC3D,MAAM,SAAS,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;QACtD,MAAM,0BAA0B,GAAG,YAAY,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAC7D,IAAI,CAAC,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,CAAC,EAAE,CAAC;YAChE,IAAI,SAAS,EAAE,CAAC;gBACZ,WAAW,GAAG,IAAI,KAAK,CAAC,kBAAkB,EAAE,CAAC;YACjD,CAAC;iBACI,CAAC;gBACF,WAAW,GAAG,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;YACvC,CAAC;YAED,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;YAC5B,IAAI,IAAI,EAAE,CAAC;gBACP,WAAW,CAAC,IAAI,GAAG,IAAI,CAAC;YAC5B,CAAC;iBACI,CAAC;gBACF,WAAW,CAAC,IAAI,GAAG,WAAW,CAAC;YACnC,CAAC;YAED,WAAW,CAAC,SAAS,GAAG,OAAO,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;YACjE,WAAW,CAAC,oBAAoB,GAAG,SAAS,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;YAC/D,OAAO,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;YAC3B,+DAA+D;YAE/D,IAAI,gBAAgB,GAAG,IAAI,CAAC,aAAa,CAAC;YAC1C,IAAI,CAAC;gBACD,gBAAgB,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,CAAC;YAC7D,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACb,gBAAM,CAAC,KAAK,CAAC,4BAA4B,KAAK,4DAA4D,IAAI,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC;YAC/I,CAAC;YAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,EAAE,IAAI,CAAC,CAAC;YAClE,WAAW,CAAC,YAAY,GAAG,OAAO,CAAC;YACnC,WAAW,CAAC,mBAAmB,GAAG,IAAI,CAAC,gBAAgB,EAAE,GAAG,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACtF,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;YACxC,WAAW,CAAC,kBAAkB,GAAG,UAAU,CAAC,MAAM,CAAC;YACnD,WAAW,CAAC,kBAAkB,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC;YAC7D,IAAI,CAAC,wBAAwB,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;YAEjD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;YAEtC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,WAAW,EAAC,IAAI,CAAC,CAAC;YAE/C,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,EAAE,WAAW,CAAC,CAAC;QAC9E,CAAC;aACI,CAAC;YACF,WAAW,GAAG,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,CAAgD,CAAC;QAC9H,CAAC;QAED,OAAO,WAAW,CAAC;IACvB,CAAC;IAED;;;;OAIG;IACI,yBAAyB,CAAC,KAA2B;QACxD,IAAI,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YAClC,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YACnD,IAAI,UAAU,EAAE,CAAC;gBACd,OAAO,UAAU,CAAC;YACrB,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,mBAAmB,KAAK,CAAC,OAAO,EAAE,qCAAqC,CAAC,CAAC;YAC7F,CAAC;QACL,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;QAEvC,IAAI,aAAa,GAAG,IAAI,CAAC,aAAa,CAAC;QACvC,IAAI,CAAC;YACD,aAAa,GAAG,KAAK,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,CAAC;QACrD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACb,gBAAM,CAAC,KAAK,CAAC,4BAA4B,KAAK,8CAA8C,KAAK,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC;QAClI,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC;QAChE,QAAQ,CAAC,YAAY,GAAG,OAAO,CAAC;QAChC,QAAQ,CAAC,IAAI,GAAG,KAAK,CAAC,OAAO,EAAE,CAAC;QAEhC,OAAO,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;QACzB,IAAI,CAAC,wBAAwB,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;QAE/C,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;QAEnC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;QAC9C,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;QAE1C,OAAO,QAAQ,CAAC;IACpB,CAAC;IAED;;;;OAIG;IACI,wBAAwB,CAAC,EAA4B;QAExD,MAAM,gBAAgB,GAAG,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;QAEnD,gBAAgB,CAAC,IAAI,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC;QACrC,OAAO,CAAC,EAAE,EAAE,gBAAgB,CAAC,CAAC;QAC9B,IAAI,CAAC,wBAAwB,CAAC,EAAE,EAAE,gBAAgB,CAAC,CAAC;QAEpD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,gBAAgB,CAAC,CAAC;QAE3C,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,gBAAgB,EAAC,EAAE,CAAC,CAAC;QAElD,OAAO,gBAAgB,CAAC;IAC5B,CAAC;IAED;;;;;OAKG;IACI,4BAA4B,CAAC,OAAiB;QAEjD,yFAAyF;QACzF,MAAM,wBAAwB,GAAG,OAAO,CAAC,SAAS,EAAE,EAAE,eAAe,EAAE,CAAC,CAAC,CAA6B,CAAC;QACvG,MAAM,iBAAiB,GAAY,OAAO,CAAC,OAAO,EAAE,CAAC;QACrD,IAAI,gBAAgB,GAA2D,SAAS,CAAC;QAEzF,sCAAsC;QACtC,MAAM,aAAa,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC;QACxC,6CAA6C;QAC7C,MAAM,eAAe,GAAG,aAAa,CAAC,SAAS,EAAE,EAAE,eAAe,EAAE,CAAC,CAAC,CAA2B,CAAC;QAElG,IAAI,kBAAkB,GAAG,KAAK,CAAC;QAC/B,IAAI,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,iBAAiB,CAAC,EAAC,CAAC;YACzC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE;gBACnC,IAAG,GAAG,YAAY,KAAK,CAAC,eAAe,EAAC,CAAC;oBACrC,IAAI,IAAI,KAAK,OAAO,CAAC,OAAO,EAAE,IAAI,GAAG,CAAC,iBAAiB,CAAC,IAAI,GAAC,CAAC,EAAE,CAAC;wBAC7D,gBAAgB,GAAG,GAAG,CAAC;wBACvB,kBAAkB,GAAG,IAAI,CAAC;oBAC9B,CAAC;gBACL,CAAC;qBAAM,CAAC;oBACJ,IAAI,IAAI,KAAK,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC;wBAC7B,gBAAgB,GAAG,GAAG,CAAC;wBACvB,kBAAkB,GAAG,IAAI,CAAC;oBAC9B,CAAC;gBACL,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC;QAED,IAAI,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,iBAAiB,CAAC,EAAC,CAAC;YAC7C,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE;gBACvC,IAAG,GAAG,YAAY,KAAK,CAAC,mBAAmB,EAAC,CAAC;oBACzC,IAAI,IAAI,KAAK,OAAO,CAAC,OAAO,EAAE,IAAI,GAAG,CAAC,iBAAiB,CAAC,IAAI,GAAC,CAAC,EAAE,CAAC;wBAC7D,gBAAgB,GAAG,GAAG,CAAC;wBACvB,kBAAkB,GAAG,IAAI,CAAC;oBAC9B,CAAC;gBACL,CAAC;qBAAM,CAAC;oBACJ,IAAI,IAAI,KAAK,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC;wBAC7B,gBAAgB,GAAG,GAAG,CAAC;wBACvB,kBAAkB,GAAG,IAAI,CAAC;oBAC9B,CAAC;gBACL,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC;QAED,IAAG,CAAC,kBAAkB,EAAC,CAAC;YACpB,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,EAAE,CAAC;gBACxC,iBAAiB;gBACjB,IAAI,eAAe,CAAC,iBAAiB,CAAC,EAAE,CAAC;oBACrC,gBAAgB,GAAG,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;oBAC7C,gBAAgB,CAAC,MAAM,GAAG,IAAI,CAAC;gBACnC,CAAC;qBAAM,CAAC;oBACJ,gBAAgB,GAAG,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;gBACjD,CAAC;gBAED,gBAAgB,CAAC,IAAI,GAAG,iBAAiB,CAAC;gBAC1C,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,gBAAgB,CAAC,CAAC;gBAC3C,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,eAAe,EAAE,gBAAgB,CAAC,CAAC;gBACvD,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,gBAAgB,EAAC,wBAAwB,CAAC,CAAC;YAC5E,CAAC;iBACI,CAAC;gBACF,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;gBACpD,IAAI,MAAM,EAAE,CAAC;oBACT,gBAAgB,GAAG,MAAM,CAAC;gBAC9B,CAAC;qBAAM,CAAC;oBACJ,MAAM,IAAI,KAAK,CAAC,cAAc,eAAe,gCAAgC,CAAC,CAAC;gBACnF,CAAC;YACL,CAAC;QACL,CAAC;QACD,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,wDAAwD,iBAAiB,EAAE,CAAC,CAAC;QACjG,CAAC;QACD,OAAO,gBAAgB,CAAC;IAC5B,CAAC;IAED;;;;OAIG;IACI,wBAAwB,CAAC,QAA6B;QACzD,IAAI,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC;YACpC,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;YACpD,IAAI,SAAS,EAAE,CAAC;gBACb,OAAO,SAAS,CAAC;YACpB,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,mBAAmB,QAAQ,CAAC,OAAO,EAAE,oCAAoC,CAAC,CAAC;YAC/F,CAAC;QACL,CAAC;QACD,MAAM,WAAW,GAAG,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;QAEzC,IAAI,gBAAgB,GAAG,IAAI,CAAC,aAAa,CAAC;QAC1C,IAAI,CAAC;YACD,gBAAgB,GAAG,QAAQ,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,CAAC;QAC3D,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACb,gBAAM,CAAC,KAAK,CAAC,4BAA4B,KAAK,6CAA6C,QAAQ,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC;QACpI,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,EAAE,QAAQ,CAAC,CAAC;QACtE,WAAW,CAAC,YAAY,GAAG,OAAO,CAAC;QACnC,WAAW,CAAC,IAAI,GAAG,QAAQ,CAAC,OAAO,EAAE,CAAC;QACtC,OAAO,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;QAC/B,IAAI,CAAC,wBAAwB,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;QAErD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,WAAW,CAAC,CAAC;QAEtC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,WAAW,EAAC,QAAQ,CAAC,CAAC;QACnD,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,QAAQ,EAAE,WAAW,CAAC,CAAC;QAE/C,OAAO,WAAW,CAAC;IACvB,CAAC;IAED;;;;OAIG;IACI,oBAAoB,CAAC,UAA2B;QACnD,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE,CAAC;YAClC,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YAC9C,IAAI,KAAK,EAAE,CAAC;gBACT,OAAO,KAAK,CAAC;YAChB,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,cAAc,UAAU,CAAC,OAAO,EAAE,gCAAgC,CAAC,CAAC;YACxF,CAAC;QACL,CAAC;QACD,MAAM,OAAO,GAAG,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC;QACjC,OAAO,CAAC,IAAI,GAAG,UAAU,CAAC,OAAO,EAAE,CAAC;QACpC,OAAO,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAC7B,IAAI,CAAC,wBAAwB,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAEnD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAElC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,OAAO,EAAC,UAAU,CAAC,CAAC;QACjD,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAEzC,OAAO,OAAO,CAAC;IACnB,CAAC;IAED;;;;OAIG;IACI,oBAAoB,CAAC,UAAsB;QAC9C,MAAM,YAAY,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;QAE3C,IAAI,iBAAiB,GAAG,IAAI,CAAC,aAAa,CAAC;QAC3C,IAAI,CAAC;YACD,iBAAiB,GAAG,UAAU,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,CAAC;QAC9D,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACb,gBAAM,CAAC,KAAK,CAAC,4BAA4B,KAAK,+CAA+C,UAAU,CAAC,OAAO,EAAE,iBAAiB,CAAC,CAAC;QACxI,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,iBAAiB,EAAE,UAAU,CAAC,CAAC;QACzE,YAAY,CAAC,YAAY,GAAG,OAAO,CAAC;QACpC,YAAY,CAAC,IAAI,GAAG,UAAU,CAAC,OAAO,EAAE,CAAC;QACzC,OAAO,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;QAClC,IAAI,CAAC,wBAAwB,CAAC,UAAU,EAAE,YAAY,CAAC,CAAC;QAExD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;QAEvC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,YAAY,EAAC,UAAU,CAAC,CAAC;QAEtD,OAAO,YAAY,CAAC;IACxB,CAAC;IAED;;;;;OAKG;IACI,yBAAyB,CAAC,SAAoB,EAAE,eAAoJ;QACvM,MAAM,YAAY,GAAG,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;QAC3C,MAAM,aAAa,GAAG,GAAG,GAAG,SAAS,CAAC,OAAO,EAAE,CAAC;QAChD,MAAM,mBAAmB,GAAG,SAAS,CAAC,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;QAE7D,YAAY,CAAC,IAAI,GAAG,aAAa,CAAC;QAClC,YAAY,CAAC,mBAAmB,GAAG,mBAAmB,CAAC;QACvD,MAAM,iCAAiC,GAAG,YAAY,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QAC/E,MAAM,kBAAkB,GAAG,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,iCAAiC,CAAsB,CAAC;QACpI,YAAY,CAAC,eAAe,GAAG,kBAAkB,CAAC;QAClD,OAAO,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QACjC,IAAI,CAAC,wBAAwB,CAAC,SAAS,EAAE,YAAY,CAAC,CAAC;QAEvD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;QAEvC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,YAAY,EAAC,SAAS,CAAC,CAAC;QAErD,OAAO,YAAY,CAAC;IACxB,CAAC;IAED;;;;;;OAMG;IACI,kBAAkB,CAAC,OAAqB,EAAE,QAA2B,EAAE,OAAgB;QAC1F,gBAAM,CAAC,KAAK,CAAC,4BAA4B,OAAO,CAAC,OAAO,EAAE,aAAa,QAAQ,CAAC,IAAI,GAAG,CAAC,CAAC;QACzF,MAAM,UAAU,GAAG,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;QACvC,UAAU,CAAC,SAAS,GAAG,QAAQ,CAAC,CAAE,sDAAsD;QACxF,UAAU,CAAC,OAAO,GAAG,OAAO,CAAC;QAE7B,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC;QAEnD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;QAErC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,UAAU,EAAC,OAAO,CAAC,CAAC;QAEjD,OAAO,UAAU,CAAC;IACtB,CAAC;IAED;;;;;OAKG;IACI,oBAAoB,CAAC,QAAgB,EAAE,OAA+B;QACzE,IAAI,OAAmB,CAAC;QACxB,MAAM,WAAW,GAAG,eAAe,CAAC,QAAQ,CAAC,CAAC;QAC9C,MAAM,gBAAgB,GAClB,OAAO,YAAY,2BAAgB,IAAI,OAAO,CAAC,iBAAiB,EAAE,CAAC,MAAM,GAAG,CAAC;YAC7E,OAAO,YAAY,+BAAoB,IAAI,OAAO,CAAC,iBAAiB,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;QAElF,sCAAsC;QACtC,KAAK;QACL,sFAAsF;QACtF,oFAAoF;QACpF,8EAA8E;QAElF,gBAAM,CAAC,KAAK,CAAC,+BAA+B,GAAG,QAAQ,GAAG,gBAAgB,GAAG,OAAO,EAAE,OAAO,EAAE,GAAG,YAAY,GAAG,OAAO,EAAE,WAAW,EAAE,CAAC,CAAC;QAEzI,IAAI,WAAW,EAAE,CAAC;YACd,OAAO,IAAI,CAAC,6BAA6B,CAAC,QAAQ,CAAC,CAAC;QACxD,CAAC;QAED,IAAI,gBAAgB,EAAE,CAAC;YACnB,kBAAkB;YAClB,MAAM,iBAAiB,GAAG,OAAgC,CAAC;YAC3D,OAAO,IAAI,CAAC,8BAA8B,CAAC,QAAQ,EAAE,iBAAiB,CAAC,CAAC;QAC5E,CAAC;QAED,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC;YAChC,IAAI,QAAQ,GAAsC,SAAS,CAAC;YAE5D,IAAI,OAAO,KAAK,SAAS,EAAE,CAAC;gBACxB,MAAM,YAAY,GAAG,OAAO,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC;gBACvD,IAAI,CAAC,YAAY,EAAE,CAAC;oBAChB,MAAM,IAAI,KAAK,CAAC,kCAAkC,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;gBAC5E,CAAC;gBACD,MAAM,0BAA0B,GAAG,YAAY,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;gBACrE,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,0BAA0B,CAA0B,CAAC;gBACjH,IAAI,CAAC,QAAQ,EAAE,CAAC;oBACZ,gBAAM,CAAC,KAAK,CAAC,YAAY,YAAY,CAAC,MAAM,CAAC,YAAY,CAAC,kCAAkC,CAAC,CAAC;oBAC9F,QAAQ,GAAG,IAAI,CAAC,oBAAoB,CAAC,YAAY,CAAC,OAAO,EAAE,EAAE,YAAsC,CAAC,CAAC;gBACzG,CAAC;YACL,CAAC;YAED,OAAO,GAAG,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC;YAC3B,OAAO,CAAC,IAAI,GAAG,QAAQ,CAAC;YAExB,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACZ,MAAM,IAAI,KAAK,CAAC,+BAA+B,QAAQ,GAAG,CAAC,CAAC;YAChE,CAAC;YACD,OAAO,CAAC,SAAS,GAAG,QAAQ,CAAC;YAC7B,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;YAC1B,IAAI,CAAC,wBAAwB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;YAEhD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;YAElC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC1C,CAAC;aACI,CAAC;YACF,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAC5C,IAAI,MAAM,EAAE,CAAC;gBACT,OAAO,GAAG,MAAM,CAAC;YACrB,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,cAAc,QAAQ,gCAAgC,CAAC,CAAC;YAC5E,CAAC;QACL,CAAC;QAED,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,OAAO,EAAC,OAAO,CAAC,CAAC;QAE9C,OAAO,OAAO,CAAC;IACnB,CAAC;IAED;;;;;OAKG;IACH,8BAA8B,CAAC,QAAgB,EAAE,OAA8B;QAE3E,IAAI,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC;YACxC,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAC5C,IAAI,MAAM,EAAE,CAAC;gBACT,OAAO,MAAM,CAAC;YAClB,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,cAAc,QAAQ,4CAA4C,CAAC,CAAC;YACxF,CAAC;QACL,CAAC;QAED,sEAAsE;QACtE,oFAAoF;QAEpF,wEAAwE;QACxE,IAAI,OAAmB,CAAC;QAExB,IAAI,OAAO,YAAY,2BAAgB,EAAE,CAAC;YACtC,OAAO,GAAG,IAAI,KAAK,CAAC,eAAe,EAAE,CAAC;QAC1C,CAAC;aAAM,IAAI,OAAO,YAAY,+BAAoB,EAAE,CAAC;YACjD,OAAO,GAAG,IAAI,KAAK,CAAC,mBAAmB,EAAE,CAAC;QAC9C,CAAC;QACD,uCAAuC;QACvC,qDAAqD;QACrD,gDAAgD;QAChD,iDAAiD;QACjD,qDAAqD;QACrD,qDAAqD;QACrD,8CAA8C;QAC9C,KAAK;aACA,CAAC;YACF,MAAM,IAAI,KAAK,CAAC,yEAAyE,CAAC,CAAC;QAC/F,CAAC;QAED,kDAAkD;QAElD,4DAA4D;QAC5D,wCAAwC;QACxC,0EAA0E;QAC1E,IAAI;QAEJ,uEAAuE;QACvE,kGAAkG;QAClG,sCAAsC;QACtC,4EAA4E;QAC5E,oDAAoD;QACpD,+FAA+F;QAC/F,sEAAsE;QACtE,MAAM;QACN,wEAAwE;QAExE,2DAA2D;QAE3D,OAAO,CAAC,IAAI,GAAG,QAAQ,CAAC;QACxB,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC1B,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAClC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QACtC,OAAO,OAAO,CAAC;IACnB,CAAC;IAED;;;;;OAKG;IACH,4FAA4F;IAC5F,0CAA0C;IAC1C,sEAAsE;IACtE,QAAQ;IAER,oCAAoC;IACpC,iDAAiD;IACjD,yGAAyG;IACzG,6CAA6C;IAC7C,mFAAmF;IACnF,2DAA2D;IAC3D,sGAAsG;IACtG,6EAA6E;IAC7E,aAAa;IACb,4EAA4E;IAC5E,+DAA+D;IAC/D,iCAAiC;IACjC,yCAAyC;IACzC,6CAA6C;IAC7C,sBAAsB;IACtB,IAAI;IAEJ;;;;OAIG;IACH,6BAA6B,CAAC,QAAgB;QAC1C,IAAI,OAAO,GAAwB,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;QAC7D,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC1C,OAAO,GAAG,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;YACpC,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;YACtB,OAAO,CAAC,IAAI,GAAG,QAAQ,CAAC;YACxB,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YAChD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QACtC,CAAC;aAAM,CAAC;YACJ,OAAO,GAAG,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,QAAQ,CAAwB,CAAC;QAC5E,CAAC;QACD,OAAO,OAAO,CAAC;IACnB,CAAC;IAED;;;;OAIG;IACI,iBAAiB,CAAC,IAAgB,EAAE,EAAU;QACjD,MAAM,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,kBAAkB,CAAC,EAAE,CAA2B,CAAC;QAC9E,IAAI,CAAC,MAAM,EAAE,CAAC;YACV,MAAM,IAAI,KAAK,CAAC,wBAAwB,EAAE,wBAAwB,IAAI,CAAC,OAAO,EAAE,OAAO,IAAI,CAAC,aAAa,EAAE,CAAC,WAAW,EAAE,YAAY,IAAI,CAAC,kBAAkB,EAAE,GAAG,CAAC,CAAC;QACvK,CAAC;QAED,gBAAM,CAAC,KAAK,CAAC,gCAAgC,IAAI,CAAC,WAAW,EAAE,MAAM,IAAI,CAAC,OAAO,EAAE,aAAa,IAAI,CAAC,kBAAkB,EAAE,OAAO,IAAI,CAAC,aAAa,EAAE,CAAC,WAAW,EAAE,SAAS,EAAE,sBAAsB,MAAM,CAAC,kBAAkB,IAAI,CAAC,CAAC;QAElO,MAAM,qBAAqB,GAAG,OAAO,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;QACzD,MAAM,0BAA0B,GAAG,YAAY,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAC;QAC9E,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,0BAA0B,CAA0B,CAAC;QACvH,IAAI,CAAC,QAAQ,EAAE,CAAC;YACZ,gBAAM,CAAC,KAAK,CAAC,YAAY,0BAA0B,YAAY,qBAAqB,CAAC,WAAW,EAAE,aAAa,CAAC,CAAC;YACjH,8GAA8G;YAC9G,OAAO,CAAC,wCAAwC;QACpD,CAAC;aAAM,CAAC;YACJ,gBAAM,CAAC,KAAK,CAAC,wBAAwB,QAAQ,CAAC,kBAAkB,GAAG,CAAC,CAAC;QACzE,CAAC;QAGD,6DAA6D;QAC7D,IAAI,CAAC,CAAC,QAAQ,YAAY,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,QAAQ,YAAY,KAAK,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,QAAQ,YAAY,KAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,QAAQ,YAAY,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,QAAQ,YAAY,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC;YACjN,gBAAM,CAAC,KAAK,CAAC,YAAY,QAAQ,CAAC,kBAAkB,kCAAkC,CAAC,CAAC;YACxF,OAAO;QACX,CAAC;QAED,sGAAsG;QACtG,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,mCAAmC,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QACxF,IAAI,WAAW,EAAE,CAAC;YACd,gBAAM,CAAC,KAAK,CAAC,2CAA2C,QAAQ,CAAC,kBAAkB,iBAAiB,MAAM,CAAC,kBAAkB,GAAG,CAAC,CAAC;YAClI,OAAO;QACX,CAAC;QACD,MAAM,SAAS,GAAG,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;QACrC,SAAS,CAAC,QAAQ,GAAG,QAAQ,CAAC;QAC9B,SAAS,CAAC,QAAQ,GAAG,MAAM,CAAC;QAE5B,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;QAEpC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,SAAS,EAAC,IAAI,CAAC,CAAC;IACjD,CAAC;IAED;;;;;OAKG;IACI,qBAAqB,CAAC,wBAAoC,EAAE,SAAwB,EAAE,EAAU;QACnG,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC,kBAAkB,CAAC,EAAE,CAA2B,CAAC;QACpF,wFAAwF;QACxF,MAAM,eAAe,GAAG,OAAO,CAAC,YAAY,CAAC,wBAAwB,CAAC,CAAC;QACvE,gBAAM,CAAC,KAAK,CAAC,8BAA8B,eAAe,CAAC,WAAW,EAAE,iBAAiB,wBAAwB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;QAChI,MAAM,YAAY,GAAG,YAAY,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QAC1D,gBAAM,CAAC,KAAK,CAAC,sBAAsB,YAAY,GAAG,CAAC,CAAC;QACpD,IAAI,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,YAAY,CAA0B,CAAC;QACrG,gBAAM,CAAC,KAAK,CAAC,+BAA+B,MAAM,CAAC,kBAAkB,GAAG,CAAC,CAAC;QAC1E,IAAI,MAAM,YAAY,KAAK,CAAC,IAAI,EAAE,CAAC;YAC/B,oEAAoE;YACpE,gBAAM,CAAC,KAAK,CAAC,gGAAgG,MAAM,CAAC,kBAAkB,GAAG,CAAC,CAAC;YAC3I,MAAM,eAAe,GAAG,MAAM,CAAC,SAAS,CAAC;YACzC,IAAI,eAAe,EAAE,CAAC;gBAClB,MAAM,GAAG,eAAe,CAAC;YAC7B,CAAC;QACL,CAAC;QACD,MAAM,0BAA0B,GAAG,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,SAAS,EAAE,CAAC,CAAC;QAC9E,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,0BAA0B,CAAsB,CAAC;QAEnH,MAAM,aAAa,GAAG,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;QAC7C,aAAa,CAAC,MAAM,GAAG,MAAM,CAAC;QAC9B,aAAa,CAAC,QAAQ,GAAG,QAAQ,CAAC;QAClC,aAAa,CAAC,YAAY,CAAC,YAAY,CAAC,CAAC;QACzC,aAAa,CAAC,SAAS,GAAG,YAAY,CAAC,SAAS,CAAC;QAEjD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;QAExC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,aAAa,EAAC,wBAAwB,CAAC,CAAC;IACzE,CAAC;IAED;;;;OAIG;IACI,2BAA2B,CAAC,GAA4C,EAAE,QAA+E;QAC5J,yDAAyD;QACzD,4DAA4D;QAC5D,yBAAyB;QACzB,0CAA0C;QAC1C,wCAAwC;QACxC,uEAAuE;QACvE,yDAAyD;QACzD,yDAAyD;QACzD,kDAAkD;QAClD,oFAAoF;QACpF,WAAW;QACX,yCAAyC;QACzC,2BAA2B;QAC3B,qGAAqG;QACrG,QAAQ;QACR,kEAAkE;QAClE,IAAI;QACJ,6CAA6C;QAC7C,iFAAiF;QACjF,8FAA8F;QAC9F,+BAA+B;QAC/B,iJAAiJ;QACjJ,kCAAkC;QAClC,WAAW;QACX,+FAA+F;QAC/F,IAAI;QAEJ,2GAA2G;QAC3G,mEAAmE;QACnE,sFAAsF;QACtF,2BAA2B;QAC3B,sCAAsC;QACtC,eAAe;QACf,gGAAgG;QAChG,QAAQ;QACR,IAAI;QAEJ,MAAM,cAAc,GAAG,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC;QAE/C,2GAA2G;QAE3G,IAAI,QAAmD,CAAC;QACxD,IAAI,GAAG,YAAY,2BAAgB,EAAE,CAAC;YAClC,QAAQ,GAAG,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC;QAC/C,CAAC;aACI,CAAC;YACF,QAAQ,GAAG,IAAI,CAAC,yBAAyB,CAAC,GAAG,CAAC,CAAC;QACnD,CAAC;QAED,IAAI,CAAC,QAAQ,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CAAC,YAAY,GAAG,wCAAwC,CAAC,CAAC;QAC7E,CAAC;QAED,IAAI,UAAqD,CAAC;QAC1D,0FAA0F;QAC1F,yCAAyC;QACzC,2BAA2B;QAC3B,qGAAqG;QACrG,QAAQ;QACR,kEAAkE;QAClE,kDAAkD;QAClD,yEAAyE;QACzE,QAAQ;QACR,aAAa;QACb,6EAA6E;QAC7E,QAAQ;QACR,yBAAyB;QACzB,0GAA0G;QAC1G,QAAQ;QACR,IAAI;QAEJ,0FAA0F;QAC1F,kCAAkC;QAElC,IAAI,QAAQ,YAAY,2BAAgB,EAAE,CAAC;YACvC,kCAAkC;YAClC,UAAU,GAAG,IAAI,CAAC,qBAAqB,CAAC,QAAQ,CAAC,CAAC;YAClD,gEAAgE;QACpE,CAAC;aACI,IAAI,QAAQ,YAAY,+BAAoB,EAAE,CAAC;YAChD,sCAAsC;YACtC,UAAU,GAAG,IAAI,CAAC,yBAAyB,CAAC,QAAQ,CAAC,CAAC;YACtD,oEAAoE;QACxE,CAAC;aAAM,CAAC;YACJ,kDAAkD;YAClD,MAAM,oBAAoB,GAAG,qCAAqC,CAAC,QAAQ,CAAC,CAAC;YAC7E,IAAI,oBAAoB,KAAK,SAAS,EAAE,CAAC;gBACrC,UAAU,GAAG,IAAI,CAAC,yBAAyB,CAAC,oBAAoB,CAAC,CAAC;YACtE,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,uCAAuC,QAAQ,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;YAClF,CAAC;QACL,CAAC;QAED,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;QAEnD,IAAI,CAAC,wBAAwB,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC;QAEpD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;QAErC,cAAc,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACnC,cAAc,CAAC,UAAU,GAAG,UAAU,CAAC;QAEvC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;QACzC,yBAAyB;QAEzB,kGAAkG;QAClG,sDAAsD;IAE1D,CAAC;IAEM,uBAAuB,CAAC,cAAiC,EAAE,eAA6B;QAC3F,MAAM,eAAe,GAAG,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;QACjD,eAAe,CAAC,cAAc,GAAG,cAAc,CAAC;QAChD,eAAe,CAAC,eAAe,GAAG,eAAe,CAAC;QAClD,eAAe,CAAC,iBAAiB,CAAC,eAAe,CAAC,CAAC;QACnD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;;;;;OASG;IACI,+BAA+B,CAAC,gBAAiP;QACpR,MAAM,EAAC,iBAAiB,EAAE,kBAAkB,EAAE,QAAQ,EAAE,uBAAuB,EAAE,aAAa,EAAE,WAAW,EAAE,eAAe,EAAC,GAAG,gBAAgB,CAAC;QACjJ,IAAI,iBAAiB,IAAI,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,iBAAiB,CAAC,EAAE,CAAC;YACtE,MAAM,aAAa,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;YACrE,IAAI,aAAa,EAAE,CAAC;gBACjB,OAAO,CAAC,oBAAoB;YAC/B,CAAC;iBAAM,CAAC;gBACJ,MAAM,IAAI,KAAK,CAAC,iBAAiB,aAAa,CAAC,OAAO,EAAE,yCAAyC,CAAC,CAAC;YACvG,CAAC;QACL,CAAC;QAED,gBAAM,CAAC,IAAI,CAAC,wCAAwC,iBAAiB,EAAE,OAAO,EAAE,OAAO,QAAQ,CAAC,WAAW,EAAE,GAAG,CAAC,CAAC;QAClH,MAAM,eAAe,GAAG,IAAI,KAAK,CAAC,YAAY,EAAE,CAAC;QAEjD,IAAI,cAAc,GAA2D,SAAS,CAAC;QACvF,IAAI,kBAA0B,CAAC;QAE/B,MAAM,mBAAmB,GAAG,IAAI,CAAC,QAAQ,CAAC,eAAe,EAAE,CAAC;QAE5D,MAAM,YAAY,GAAG,cAAI,CAAC,SAAS,CAAC,uBAAuB,CAAC,CAAC;QAC7D,mFAAmF;QACnF,gBAAM,CAAC,KAAK,CAAC,0CAA0C,YAAY,EAAE,CAAC,CAAC;QACvE,gBAAM,CAAC,KAAK,CAAC,mDAAmD,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,mBAAmB,CAAC,EAAE,CAAC,CAAC;QACjI,MAAM,aAAa,GAAW,IAAI,CAAC,qBAAqB,CAAC,YAAY,EAAE,mBAAmB,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,CAAC;QAChH,gBAAM,CAAC,KAAK,CAAC,2CAA2C,aAAa,EAAE,CAAC,CAAC;QACzE,IAAI,QAAQ,GAAG,GAAG,GAAG,aAAa,GAAG,IAAI,CAAC;QAC1C,gBAAM,CAAC,KAAK,CAAC,sCAAsC,QAAQ,EAAE,CAAC,CAAC;QAE/D,sEAAsE;QAEtE,6EAA6E;QAE7E,IAAI,iBAAiB,YAAY,4BAAiB;eAC3C,aAAa,YAAY,0BAAe,EAAE,CAAC;YAC1C,kBAAkB,GAAG,aAAa,CAAC,OAAO,EAAE,CAAC;YACjD,QAAQ,GAAG,QAAQ,GAAG,kBAAkB,CAAC;YACzC,IAAI,WAAW,EAAE,CAAC;gBACd,cAAc,GAAG,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,QAAQ,CAAsB,CAAC;YACrG,CAAC;YACD,IAAI,cAAc,KAAK,SAAS,EAAE,CAAC;gBAC/B,cAAc,GAAG,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC;gBACzC,cAAc,CAAC,IAAI,GAAG,kBAAkB,CAAC;gBACzC,IAAI,CAAC,WAAW,EAAE,CAAC;oBACf,cAAc,CAAC,MAAM,GAAG,IAAI,CAAC;gBACjC,CAAC;gBACD,6GAA6G;gBAC7G,gDAAgD;gBAChD,OAAO,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;gBAEvC,IAAI,CAAC,wBAAwB,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;gBAC7D,gCAAgC;gBAChC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;YAC7C,CAAC;QACL,CAAC;QACD,8FAA8F;QAC9F,2FAA2F;aACtF,IAAI,iBAAiB,YAAY,kCAAuB,EAAE,CAAC;YAC5D,kBAAkB,GAAG,iBAAiB,EAAE,OAAO,EAAE,CAAC;YAClD,QAAQ,GAAG,QAAQ,GAAG,kBAAkB,CAAC;YACzC,cAAc,GAAG,IAAI,KAAK,CAAC,gBAAgB,EAAE,CAAC;YAC9C,cAAc,CAAC,IAAI,GAAG,kBAAkB,CAAC;YACzC,OAAO,CAAC,iBAAiB,EAAE,cAAc,CAAC,CAAC;YAC3C,IAAI,CAAC,wBAAwB,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;YAC7D,gDAAgD;YAChD,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,KAAK,EAAE,iBAAiB,CAAC,CAAC;YACnE,cAAyC,CAAC,YAAY,GAAG,OAAO,CAAC;QACtE,CAAC;aAAM,CAAC,CAAE,sEAAsE;YAC5E,kBAAkB,GAAG,aAAa,CAAC,OAAO,EAAE,CAAC;YAC7C,QAAQ,GAAG,QAAQ,GAAG,CAAC,eAAe,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,iBAAiB,CAAC,CAAC;YAC9E,cAAc,GAAG,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC;YACzC,cAAc,CAAC,IAAI,GAAG,kBAAkB,CAAC;YACzC,gDAAgD;YAChD,OAAO,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;YACvC,IAAI,CAAC,wBAAwB,CAAC,aAAa,EAAE,cAAc,CAAC,CAAC;QACjE,CAAC;QACD,0EAA0E;QAC1E,IAAI,CAAC,WAAW;YAAE,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;QAC3D,MAAM,0BAA0B,GAAG,YAAY,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QACjE,MAAM,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC,kCAAkC,CAAC,0BAA0B,CAAiB,CAAC;QACjH,eAAe,CAAC,eAAe,GAAG,WAAW,CAAC;QAC9C,eAAe,CAAC,cAAc,GAAG,cAAc,CAAC;QAChD,IAAI,iBAAiB,YAAY,kCAAuB,EAAE,CAAC;YACvD,eAAe,CAAC,eAAe,GAAG,iBAAiB,EAAE,kBAAkB,EAAE,CAAC,OAAO,EAAY,CAAC;QAClG,CAAC;aAAM,CAAC;YACJ,eAAe,CAAC,eAAe,GAAG,iBAAiB,EAAE,uBAAuB,EAAY,CAAC;QAC7F,CAAC;QAED,gBAAM,CAAC,KAAK,CAAC,4BAA4B,eAAe,CAAC,cAAc,EAAE,IAAI,aACzE,OAAO,CAAC,cAAc,CAAC,eAAe,CAAC,cAAc,CAAC,oBAAoB,eAAe,CAAC,eAAe,EAAE,IAAI,EAAE,CAAC,CAAC;QAEvH,WAAW,CAAC,iBAAiB,CAAC,eAAe,CAAC,CAAC;QAE/C,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;QAE1C,IAAI,iBAAiB,EAAE,CAAC;YACpB,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,eAAe,EAAE,iBAAiB,CAAC,CAAC;YACjE,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,iBAAiB,EAAE,eAAe,CAAC,CAAC;QACpE,CAAC;IACL,CAAC;IAED;;;;OAIG;IACI,6BAA6B,CAAC,eAA2B,EAAE,SAAoC;QAElG,IAAI,gBAAqE,CAAC;QAC1E,MAAM,0BAA0B,GAAG,YAAY,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QAExE,IAAI,CAAC,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,CAAC,EAAE,CAAC;YAEhE,MAAM,aAAa,GAAG,eAAe,CAAC,aAAa,CAAC,qBAAU,CAAC,aAAa,CAAC,CAAC;YAE9E,MAAM,SAAS,GAAG,aAAa,CAAC,iBAAiB,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC;YAE/D,IAAI,SAAS,EAAE,CAAC;gBACZ,gBAAgB,GAAG,IAAI,KAAK,CAAC,uBAAuB,EAAE,CAAC;YAC3D,CAAC;iBACI,CAAC;gBACF,gBAAgB,GAAG,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;YACjD,CAAC;YAED,kEAAkE;YAClE,MAAM,MAAM,GAAG,aAAa,CAAC,eAAe,CAAC,qBAAU,CAAC,mBAAmB,CAAC,CAAC;YAC7E,IAAI,YAAY,GAAG,WAAW,CAAC;YAE/B,IAAI,MAAM,IAAI,MAAM,YAAY,8BAAmB,EAAE,CAAC;gBAClD,+BAA+B;gBAC/B,YAAY,GAAG,MAAM,CAAC,OAAO,EAAE,CAAC;YACpC,CAAC;YAED,IAAI,YAAY,EAAE,CAAC;gBACf,gBAAgB,CAAC,IAAI,GAAG,YAAY,CAAC;YACzC,CAAC;iBACI,CAAC;gBACF,gBAAgB,CAAC,IAAI,GAAG,WAAW,CAAC;YACxC,CAAC;YAED,gEAAgE;YAChE,MAAM,mBAAmB,GAAG,aAAa,CAAC,aAAa,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAC3F,MAAM,mBAAmB,GAAG,aAAa,CAAC,aAAa,EAAE,CAAC,OAAO,EAAE,CAAC;YACpE,gBAAgB,CAAC,SAAS,GAAG,IAAI,mBAAmB,QAAQ,mBAAmB,EAAE,CAAC;YAClF,gBAAgB,CAAC,oBAAoB,GAAG,SAAS,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;YAEzE,IAAI,gBAAgB,GAAG,IAAI,CAAC,aAAa,CAAC;YAC1C,IAAI,CAAC;gBACD,gBAAgB,GAAG,aAAa,CAAC,aAAa,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,CAAC;YACtE,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACb,gBAAM,CAAC,KAAK,CAAC,4BAA4B,KAAK,4DAA4D,YAAY,iBAAiB,CAAC,CAAC;YAC7I,CAAC;YAED,MAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,EAAE,aAA+C,CAAC,CAAC;YAC7G,gBAAgB,CAAC,YAAY,GAAG,OAAO,CAAC;YACxC,gBAAgB,CAAC,mBAAmB,GAAG,aAAa,CAAC,gBAAgB,EAAE,GAAG,aAAa,CAAC,kBAAkB,EAAE,CAAC;YAC7G,MAAM,UAAU,GAAG,aAAa,CAAC,aAAa,EAAE,CAAC;YACjD,gBAAgB,CAAC,kBAAkB,GAAG,UAAU,CAAC,MAAM,CAAC;YACxD,gBAAgB,CAAC,kBAAkB,GAAG,aAAa,CAAC,aAAa,EAAE,CAAC,MAAM,CAAC;YAC3E,OAAO,CAAC,eAA+C,EAAE,gBAAgB,CAAC,CAAC;YAC3E,IAAI,CAAC,wBAAwB,CAAC,eAA+C,EAAE,gBAAgB,CAAC,CAAC;YACjG,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,gBAAgB,CAAC,CAAC;YAC3C,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,gBAAgB,EAAC,aAA6C,CAAC,CAAC;YAC7F,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,EAAE,gBAAgB,CAAC,CAAC;QACnF,CAAC;aAAM,CAAC;YACJ,gBAAgB,GAAG,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,0BAA0B,CAAwB,CAAC;QAC3G,CAAC;QAED,OAAO,gBAAgB,CAAC;IAC5B,CAAC;IAED;;;;OAIG;IACH,iSAAiS;IAEjS,iGAAiG;IAEjG,oDAAoD;IACpD,mDAAmD;IACnD,+DAA+D;IAC/D,uDAAuD;IACvD,kGAAkG;IAElG,gCAAgC;IAChC,IAAI;IAEJ;;;;OAIG;IACH,8HAA8H;IAC9H,sDAAsD;IACtD,qDAAqD;IACrD,mIAAmI;IACnI,+IAA+I;IAC/I,8DAA8D;IAC9D,4DAA4D;IAE5D,wEAAwE;IACxE,uEAAuE;IACvE,8FAA8F;IAE9F,yDAAyD;IACzD,4GAA4G;IAC5G,4GAA4G;IAC5G,8DAA8D;IAC9D,0HAA0H;IAC1H,4FAA4F;IAC5F,iJAAiJ;IACjJ,6DAA6D;IAC7D,0DAA0D;IAC1D,oBAAoB;IACpB,iBAAiB;IACjB,mDAAmD;IACnD,oFAAoF;IACpF,uFAAuF;IACvF,wFAAwF;IACxF,gFAAgF;IAChF,oFAAoF;IACpF,uBAAuB;IACvB,qDAAqD;IACrD,gNAAgN;IAChN,oBAAoB;IACpB,gFAAgF;IAChF,gBAAgB;IAChB,oEAAoE;IACpE,YAAY;IACZ,QAAQ;IACR,yCAAyC;IACzC,6LAA6L;IAC7L,QAAQ;IACR,yCAAyC;IAEzC,IAAI;IAEJ;;;OAGG;IACH,oHAAoH;IAEpH,gDAAgD;IAChD,4BAA4B;IAC5B,wBAAwB;IACxB,uFAAuF;IACvF,QAAQ;IACR,6DAA6D;IAC7D,+CAA+C;IAC/C,mCAAmC;IACnC,mFAAmF;IACnF,0CAA0C;IAC1C,wEAAwE;IACxE,0HAA0H;IAC1H,uBAAuB;IACvB,8HAA8H;IAC9H,gBAAgB;IAChB,4CAA4C;IAC5C,2FAA2F;IAC3F,gBAAgB;IAChB,qCAAqC;IACrC,yCAAyC;IACzC,iCAAiC;IACjC,4EAA4E;IAC5E,wHAAwH;IACxH,0GAA0G;IAC1G,2BAA2B;IAC3B,4HAA4H;IAC5H,kHAAkH;IAClH,oBAAoB;IACpB,2GAA2G;IAC3G,oGAAoG;IACpG,0IAA0I;IAC1I,sEAAsE;IACtE,qCAAqC;IACrC,0GAA0G;IAC1G,oIAAoI;IACpI,iEAAiE;IACjE,iFAAiF;IACjF,gLAAgL;IAChL,4DAA4D;IAC5D,4BAA4B;IAC5B,0BAA0B;IAE1B,kDAAkD;IAClD,gIAAgI;IAChI,wBAAwB;IACxB,oBAAoB;IACpB,gBAAgB;IAChB,cAAc;IACd,QAAQ;IACR,8CAA8C;IAC9C,QAAQ;IAGR;;;OAGG;IACH,+EAA+E;IAE/E,4DAA4D;IAC5D,uBAAuB;IACvB,kGAAkG;IAClG,mCAAmC;IACnC,8DAA8D;IAC9D,iEAAiE;IACjE,cAAc;IAEd,0CAA0C;IAC1C,gFAAgF;IAChF,uCAAuC;IACvC,iGAAiG;IACjG,8FAA8F;IAC9F,6FAA6F;IAC7F,sEAAsE;IACtE,qCAAqC;IACrC,mHAAmH;IACnH,oIAAoI;IACpI,iEAAiE;IACjE,iFAAiF;IACjF,gLAAgL;IAChL,4DAA4D;IAC5D,4BAA4B;IAC5B,0BAA0B;IAE1B,kDAAkD;IAClD,gIAAgI;IAChI,wBAAwB;IACxB,oBAAoB;IACpB,gBAAgB;IAChB,aAAa;IACb,QAAQ;IACR,8CAA8C;IAC9C,IAAI;IAEJ;;;OAGG;IACH,2GAA2G;IAC3G,gEAAgE;IAChE,uBAAuB;IACvB,uFAAuF;IACvF,4DAA4D;IAC5D,kCAAkC;IAClC,yDAAyD;IAEzD,oCAAoC;IACpC,6EAA6E;IAC7E,4FAA4F;IAC5F,6CAA6C;IAC7C,oGAAoG;IACpG,wBAAwB;IACxB,8FAA8F;IAC9F,+CAA+C;IAC/C,0EAA0E;IAC1E,6GAA6G;IAC7G,8EAA8E;IAC9E,8CAA8C;IAC9C,0EAA0E;IAC1E,uHAAuH;IACvH,uCAAuC;IACvC,mHAAmH;IACnH,gCAAgC;IAChC,+CAA+C;IAC/C,2GAA2G;IAC3G,4IAA4I;IAC5I,yEAAyE;IACzE,yFAAyF;IACzF,2LAA2L;IAC3L,oEAAoE;IACpE,oCAAoC;IACpC,kCAAkC;IAElC,0DAA0D;IAC1D,2IAA2I;IAC3I,gCAAgC;IAChC,4BAA4B;IAC5B,wBAAwB;IACxB,6BAA6B;IAC7B,oBAAoB;IACpB,+FAA+F;IAC/F,yDAAyD;IACzD,gBAAgB;IAChB,cAAc;IACd,QAAQ;IACR,IAAI;IAEJ;;;OAGG;IACH,yEAAyE;IAEzE,mDAAmD;IACnD,iDAAiD;IACjD,gFAAgF;IAChF,oCAAoC;IACpC,oIAAoI;IACpI,0GAA0G;IAC1G,kIAAkI;IAClI,6FAA6F;IAC7F,kEAAkE;IAClE,0HAA0H;IAC1H,qHAAqH;IACrH,gIAAgI;IAChI,6DAA6D;IAC7D,6EAA6E;IAC7E,kLAAkL;IAClL,wDAAwD;IACxD,wBAAwB;IACxB,sBAAsB;IAEtB,8CAA8C;IAC9C,kIAAkI;IAClI,oBAAoB;IACpB,gBAAgB;IAChB,YAAY;IACZ,UAAU;IACV,IAAI;IAEJ;;;;OAIG;IACH,uGAAuG;IAEvG,gEAAgE;IAChE,uBAAuB;IACvB,uFAAuF;IACvF,wDAAwD;IACxD,wCAAwC;IACxC,gDAAgD;IAChD,mCAAmC;IACnC,2EAA2E;IAC3E,6FAA6F;IAC7F,gDAAgD;IAChD,sGAAsG;IACtG,wBAAwB;IACxB,0GAA0G;IAC1G,yDAAyD;IACzD,6EAA6E;IAC7E,gHAAgH;IAChH,8EAA8E;IAC9E,8CAA8C;IAC9C,uEAAuE;IACvE,6GAA6G;IAC7G,uCAAuC;IACvC,qHAAqH;IACrH,gCAAgC;IAChC,+CAA+C;IAC/C,2GAA2G;IAC3G,4IAA4I;IAC5I,yEAAyE;IACzE,yFAAyF;IACzF,2LAA2L;IAC3L,oEAAoE;IACpE,oCAAoC;IACpC,kCAAkC;IAElC,0DAA0D;IAC1D,2IAA2I;IAC3I,gCAAgC;IAChC,4BAA4B;IAC5B,6BAA6B;IAC7B,oBAAoB;IACpB,uDAAuD;IACvD,gBAAgB;IAChB,cAAc;IACd,QAAQ;IACR,IAAI;IAEG,qBAAqB,CAAC,YAAoB,EAAE,mBAA2B;QAC1E,gBAAM,CAAC,KAAK,CAAC,yCAAyC,YAAY,4BAA4B,mBAAmB,GAAG,CAAC,CAAC;QACtH,IAAI,YAAY,CAAC,UAAU,CAAC,mBAAmB,CAAC,EAAE,CAAC;YAC/C,OAAO,YAAY,CAAC,OAAO,CAAC,mBAAmB,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAClE,CAAC;aAAM,IAAI,YAAY,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YACtC,OAAO,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACjC,CAAC;aAAM,CAAC;YACJ,OAAO,YAAY,CAAC;QACxB,CAAC;IACL,CAAC;CACJ;AAtyDD,4CAsyDC;AAED,SAAgB,eAAe,CAAC,QAAgB;IAC5C,OAAO,QAAQ,KAAK,QAAQ;QACxB,QAAQ,KAAK,QAAQ;QACrB,QAAQ,KAAK,SAAS;QACtB,QAAQ,KAAK,QAAQ;QACrB,QAAQ,KAAK,QAAQ;QACrB,QAAQ,KAAK,WAAW;QACxB,QAAQ,KAAK,MAAM;QACnB,QAAQ,KAAK,KAAK;QAClB,QAAQ,KAAK,SAAS;QACtB,QAAQ,KAAK,OAAO;QACpB,QAAQ,KAAK,MAAM,CAAC;AAC5B,CAAC;AAED,SAAS,OAAO,CAAC,aAAgC,EAAE,YAAiC;IAChF,6EAA6E;IAC7E,IAAI,YAAY,YAAY,KAAK,CAAC,IAAI,IAAI,CAAC,CAAC,aAAa,YAAY,uBAAY,CAAC,IAAI,aAAa,CAAC,aAAa,CAAC,EAAE,CAAC;QACjH,IAAI,GAAG,GAAG,YAAY,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;QAC7C,8CAA8C;QAC9C,GAAG,GAAG,GAAG,CAAC,OAAO,CAAC,eAAe,EAAE,UAAU,CAAC,CAAC;QAC/C,gBAAM,CAAC,KAAK,CAAC,mCAAmC,GAAG,YAAY,CAAC,OAAO,EAAE,GAAG,MAAM,GAAG,GAAG,CAAC,CAAC;QAC1F,YAAY,CAAC,kBAAkB,GAAG,GAAG,CAAC;QACtC,OAAO;IACX,CAAC;IACD,8BAA8B;IAC9B,IAAI,CAAC,CAAC,aAAa,YAAY,uBAAY,CAAC,EAAE,CAAC;QAC3C,MAAM,GAAG,GAAG,YAAY,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;QAC/C,gBAAM,CAAC,KAAK,CAAC,mCAAmC,GAAG,YAAY,CAAC,OAAO,EAAE,GAAG,MAAM,GAAG,GAAG,CAAC,CAAC;QACzF,YAAkC,CAAC,kBAAkB,GAAG,GAAG,CAAC;IACjE,CAAC;AACL,CAAC;AAGD,SAAS,aAAa,CAAC,aAAsd;IACze,OAAO,aAAa,YAAY,iCAAsB;WAC/C,aAAa,YAAY,4BAAiB;WAC1C,aAAa,YAAY,8BAAmB;WAC5C,aAAa,YAAY,6BAAkB;WAC3C,aAAa,YAAY,wBAAa;;YAEzC,aAAa,YAAY,+BAAoB;QAC7C,aAAa,YAAY,8BAAmB;QAC5C,aAAa,YAAY,8BAAmB;QAC5C,aAAa,YAAY,4BAAiB;QAC1C,aAAa,YAAY,mCAAwB;QACjD,aAAa,YAAY,qBAAU;QACnC,aAAa,YAAY,oBAAS;QAClC,aAAa,YAAY,iCAAsB;QAC/C,aAAa,YAAY,iCAAsB;QAC/C,aAAa,YAAY,0BAAe;QACxC,aAAa,YAAY,0BAAe;QACxC,aAAa,YAAY,qBAAU;QACnC,aAAa,YAAY,+BAAoB;WAC1C,aAAa,YAAY,4BAAiB;WAC1C,aAAa,YAAY,sCAA2B,CAAC;AAChE,CAAC;AAGD,SAAS,qCAAqC,CAAC,UAAuC;IAClF,yCAAyC;IACzC,MAAM,IAAI,GAAG,UAAU,CAAC,OAAO,EAAE,CAAC;IAElC,kDAAkD;IAClD,IAAI,MAAM,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;IAE9B,IAAI,CAAC,MAAM,EAAE,CAAC;QACV,oEAAoE;QACpE,MAAM,UAAU,GAAG,UAAU,CAAC,wBAAwB,CAAC,qBAAU,CAAC,UAAU,CAAC,CAAC;QAC9E,IAAI,CAAC,UAAU,EAAE,CAAC;YACd,MAAM,IAAI,KAAK,CAAC,4BAA4B,UAAU,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;QACzE,CAAC;QACD,MAAM,GAAG,UAAU,CAAC,SAAS,EAAE,CAAC;QAChC,IAAI,CAAC,MAAM,EAAE,CAAC;YACV,MAAM,IAAI,KAAK,CAAC,wBAAwB,UAAU,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;QACrE,CAAC;IACL,CAAC;IAED,4DAA4D;IAC5D,MAAM,oBAAoB,GAAG,mCAAmC,CAAC,MAAM,CAAC,CAAC;IAEzE,IAAI,CAAC,oBAAoB,EAAE,CAAC;QACxB,gBAAM,CAAC,KAAK,CAAC,uCAAuC,UAAU,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;IACjF,CAAC;IAED,OAAO,oBAAoB,CAAC;AAChC,CAAC;AAID,SAAS,mCAAmC,CAAC,MAAqB;IAC9D,kDAAkD;IAClD,MAAM,YAAY,GAAG,MAAM,CAAC,eAAe,EAAE,CAAC;IAE9C,kCAAkC;IAClC,MAAM,oBAAoB,GAAG,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,WAAW,YAAY,+BAAoB,CAAqC,CAAC;IAE/I,IAAI,oBAAoB,EAAE,CAAC;QACvB,OAAO,oBAAoB,CAAC;IAChC,CAAC;IAED,+FAA+F;IAC/F,KAAK,MAAM,WAAW,IAAI,YAAY,EAAE,CAAC;QACrC,IAAI,WAAW,CAAC,OAAO,EAAE,KAAK,qBAAU,CAAC,eAAe,EAAE,CAAC;YACvD,MAAM,eAAe,GAAG,WAA8B,CAAC;YACvD,MAAM,iBAAiB,GAAG,eAAe,CAAC,oBAAoB,EAAE,CAAC;YACjE,MAAM,eAAe,GAAG,iBAAiB,CAAC,4BAA4B,EAAE,CAAC;YAEzE,IAAI,eAAe,EAAE,CAAC;gBAClB,MAAM,eAAe,GAAG,eAAe,CAAC,gBAAgB,EAAE,CAAC;gBAC3D,MAAM,cAAc,GAAG,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,OAAO,EAAE,KAAK,eAAe,CAAC,OAAO,EAAE,CAAC,CAAC;gBACtG,IAAI,cAAc,EAAE,CAAC;oBACjB,OAAO,mCAAmC,CAAC,cAAc,CAAC,CAAC;gBAC/D,CAAC;YACL,CAAC;QACL,CAAC;IACL,CAAC;IAED,OAAO,SAAS,CAAC;AACrB,CAAC;AAED,iIAAiI;AACjI,oBAAoB;AACpB,oEAAoE;AACpE,yCAAyC;AAEzC,gDAAgD;AAChD,sFAAsF;AAEtF,qBAAqB;AACrB,6IAA6I;AAC7I,yFAAyF;AACzF,6BAA6B;AAC7B,oFAAoF;AACpF,YAAY;AACZ,2CAA2C;AAC3C,yBAAyB;AACzB,gFAAgF;AAChF,YAAY;AACZ,QAAQ;AAER,iEAAiE;AACjE,qDAAqD;AAErD,iDAAiD;AACjD,sJAAsJ;AAEtJ,mCAAmC;AACnC,2FAA2F;AAC3F,QAAQ;AAER,mCAAmC;AACnC,IAAI","sourcesContent":["/**\r\n * a function getOrCreateXType takes arguments name: string and element: ts-morph-type and returns a Famix.Type\r\n * The goal is to keep track of the types (e.g., a method's definedType), for the model.\r\n * The name doesn't need to be fully qualified (it's the name used in the source code, or the Famix model).\r\n */\r\n\r\n\r\nimport { ClassDeclaration, ConstructorDeclaration, FunctionDeclaration, Identifier, InterfaceDeclaration, MethodDeclaration, MethodSignature, ModuleDeclaration, PropertyDeclaration, PropertySignature, SourceFile, TypeParameterDeclaration, VariableDeclaration, ParameterDeclaration, Decorator, GetAccessorDeclaration, SetAccessorDeclaration, ImportSpecifier, CommentRange, EnumDeclaration, EnumMember, TypeAliasDeclaration, FunctionExpression, ExpressionWithTypeArguments, ImportDeclaration, ImportEqualsDeclaration, SyntaxKind, Expression, TypeNode, Scope, ArrowFunction } from \"ts-morph\";\r\nimport { isAmbient, isNamespace } from \"../analyze_functions/process_functions\";\r\nimport * as Famix from \"../lib/famix/model/famix\";\r\nimport { FamixRepository } from \"../lib/famix/famix_repository\";\r\nimport { logger, config } from \"../analyze\";\r\n// import GraphemeSplitter from \"grapheme-splitter\";\r\n// eslint-disable-next-line @typescript-eslint/no-require-imports\r\nimport GraphemeSplitter = require('grapheme-splitter');\r\nimport * as Helpers from \"./helpers_creation\";\r\nimport * as FQNFunctions from \"../fqn\";\r\nimport path from \"path\";\r\n// import _ from 'lodash';\r\n\r\nexport type TSMorphObjectType = ImportDeclaration | ImportEqualsDeclaration | SourceFile | ModuleDeclaration | ClassDeclaration | InterfaceDeclaration | MethodDeclaration | ConstructorDeclaration | MethodSignature | FunctionDeclaration | FunctionExpression | ParameterDeclaration | VariableDeclaration | PropertyDeclaration | PropertySignature | TypeParameterDeclaration | Identifier | Decorator | GetAccessorDeclaration | SetAccessorDeclaration | ImportSpecifier | CommentRange | EnumDeclaration | EnumMember | TypeAliasDeclaration | ExpressionWithTypeArguments | TSMorphParametricType;\r\n\r\nexport type TSMorphTypeDeclaration = TypeAliasDeclaration | PropertyDeclaration | PropertySignature | ConstructorDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration | FunctionExpression | ParameterDeclaration | VariableDeclaration | EnumMember | ImportEqualsDeclaration | TSMorphParametricType | TypeParameterDeclaration;\r\n\r\nexport type TSMorphParametricType = ClassDeclaration | InterfaceDeclaration | FunctionDeclaration | MethodDeclaration | ArrowFunction;\r\n\r\n// type ParametricVariantType = Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod;\r\n\r\n// type ConcreteElementTSMorphType = ClassDeclaration | InterfaceDeclaration | FunctionDeclaration | MethodDeclaration;\r\n\r\nexport type InvocableType = MethodDeclaration | ConstructorDeclaration | GetAccessorDeclaration | SetAccessorDeclaration | FunctionDeclaration | FunctionExpression | ArrowFunction;\r\n\r\nexport class EntityDictionary {\r\n    \r\n    public famixRep = new FamixRepository();\r\n    private fmxAliasMap = new Map<string, Famix.Alias>(); // Maps the alias names to their Famix model\r\n    private fmxClassMap = new Map<string, Famix.Class | Famix.ParametricClass>(); // Maps the fully qualified class names to their Famix model\r\n    private fmxInterfaceMap = new Map<string, Famix.Interface | Famix.ParametricInterface>(); // Maps the interface names to their Famix model\r\n    private fmxModuleMap = new Map<ModuleDeclaration, Famix.Module>(); // Maps the namespace names to their Famix model\r\n    private fmxFileMap = new Map<string, Famix.ScriptEntity | Famix.Module>(); // Maps the source file names to their Famix model\r\n    private fmxTypeMap = new Map<TSMorphTypeDeclaration, Famix.Type | Famix.ParameterType>(); // Maps the types declarations to their Famix model\r\n    private fmxPrimitiveTypeMap = new Map<string, Famix.PrimitiveType>(); // Maps the primitive type names to their Famix model\r\n    private fmxFunctionAndMethodMap = new Map<string, Famix.Function | Famix.ParametricFunction | Famix.Method | Famix.ParametricMethod>; // Maps the function names to their Famix model\r\n    private fmxArrowFunctionMap = new Map<string, Famix.ArrowFunction>; // Maps the function names to their Famix model\r\n    private fmxParameterMap = new Map<ParameterDeclaration, Famix.Parameter>(); // Maps the parameters to their Famix model\r\n    private fmxVariableMap = new Map<VariableDeclaration, Famix.Variable>(); // Maps the variables to their Famix model\r\n    private fmxImportClauseMap = new Map<ImportDeclaration | ImportEqualsDeclaration, Famix.ImportClause>(); // Maps the import clauses to their Famix model\r\n    private fmxEnumMap = new Map<EnumDeclaration, Famix.Enum>(); // Maps the enum names to their Famix model\r\n    private fmxInheritanceMap = new Map<string, Famix.Inheritance>(); // Maps the inheritance names to their Famix model\r\n    private UNKNOWN_VALUE = '(unknown due to parsing error)'; // The value to use when a name is not usable\r\n    public fmxElementObjectMap = new Map<Famix.Entity,TSMorphObjectType>();\r\n    public tsMorphElementObjectMap = new Map<TSMorphObjectType,Famix.Entity>();\r\n            \r\n    constructor() {\r\n        this.famixRep.setFmxElementObjectMap(this.fmxElementObjectMap);      \r\n    }\r\n\r\n    public addSourceAnchor(fmx: Famix.SourcedEntity, node: TSMorphObjectType): Famix.IndexedFileAnchor {\r\n        const sourceAnchor: Famix.IndexedFileAnchor = new Famix.IndexedFileAnchor();\r\n        let sourceStart, sourceEnd: number;\r\n        if (fmx && node) {\r\n            // find the start and end positions of the source element\r\n            if (!(node instanceof CommentRange)) {\r\n                sourceStart = node.getStart();\r\n                sourceEnd = node.getEnd();\r\n            } else {\r\n                sourceStart = node.getPos();\r\n                sourceEnd = node.getEnd();\r\n            }\r\n    \r\n            if (config.expectGraphemes) {\r\n                /**\r\n                 * The following logic handles the case of multi-code point characters (e.g. emoji) in the source text.\r\n                 * This is needed because Pharo/Smalltalk treats multi-code point characters as a single character, \r\n                 * but JavaScript treats them as multiple characters. This means that the start and end positions\r\n                 * of a source element in Pharo/Smalltalk will be different than the start and end positions of the\r\n                 * same source element in JavaScript. This logic finds the start and end positions of the source\r\n                 * element in JavaScript and then uses those positions to set the start and end positions of the\r\n                 * Famix index file anchor.\r\n                 * It depends on code in the 'grapheme-splitter' package in npm.\r\n                 */\r\n                const splitter = new GraphemeSplitter();\r\n                const sourceFileText = node.getSourceFile().getFullText();\r\n                const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;\r\n                if (hasGraphemeClusters) {\r\n                    const sourceElementText = sourceFileText.substring(sourceStart, sourceEnd);\r\n                    const sourceElementTextGraphemes = splitter.splitGraphemes(sourceElementText);\r\n                    const sourceFileTextGraphemes = splitter.splitGraphemes(sourceFileText);\r\n                    const numberOfGraphemeClustersBeforeStart = splitter.countGraphemes(sourceFileText.substring(0, sourceStart));\r\n        \r\n                    // find the start of the sourceElementTextGraphemes array in the sourceFileTextGraphemes array\r\n                    sourceStart = Helpers.indexOfSplitArray({searchArray: sourceFileTextGraphemes, \r\n                                                        targetArray: sourceElementTextGraphemes, \r\n                                                        start: sourceStart - numberOfGraphemeClustersBeforeStart});\r\n                    sourceEnd = sourceStart + sourceElementTextGraphemes.length;\r\n                } \r\n            }\r\n\r\n            // The +1 is because the source anchor (Pharo) is 1-based, but ts-morph is 0-based\r\n            sourceAnchor.startPos = sourceStart + 1;\r\n            sourceAnchor.endPos = sourceEnd + 1;\r\n\r\n            let fileName = node.getSourceFile().getFilePath() as string;\r\n            if (fileName.startsWith(\"/\")) {\r\n                fileName = fileName.substring(1);\r\n            }\r\n            sourceAnchor.element = fmx;\r\n            sourceAnchor.fileName = fileName;\r\n            fmx.sourceAnchor = sourceAnchor;\r\n            this.famixRep.addElement(sourceAnchor);\r\n\r\n        }\r\n        return sourceAnchor;\r\n    }\r\n\r\n    /**\r\n     * Makes a Famix index file anchor\r\n     * @param sourceElement A source element\r\n     * @param famixElement The Famix model of the source element\r\n     */\r\n    public makeFamixIndexFileAnchor(sourceElement: TSMorphObjectType, famixElement: Famix.SourcedEntity): void {\r\n        // Famix.Comment is not a named entity (does not have a fullyQualifiedName)\r\n        if (!(famixElement instanceof Famix.Comment)) {  // must be a named entity\r\n            // insanity check: named entities should have fullyQualifiedName\r\n            const fullyQualifiedName = (famixElement as Famix.NamedEntity).fullyQualifiedName;\r\n            if (!fullyQualifiedName || fullyQualifiedName === this.UNKNOWN_VALUE) {\r\n                throw new Error(`Famix element ${famixElement.constructor.name} has no valid fullyQualifiedName.`);\r\n            }\r\n        }\r\n\r\n        logger.debug(\"making index file anchor for '\" + sourceElement?.getText() + \"' with famixElement \" + famixElement.getJSON());\r\n        const fmxIndexFileAnchor = new Famix.IndexedFileAnchor();\r\n        fmxIndexFileAnchor.element = famixElement;\r\n        this.fmxElementObjectMap.set(famixElement, sourceElement);\r\n\r\n        if (sourceElement !== null) {\r\n            const absolutePathProject = this.famixRep.getAbsolutePath();\r\n        \r\n            const absolutePath = path.normalize(sourceElement.getSourceFile().getFilePath());\r\n\r\n            const positionNodeModules = absolutePath.indexOf('node_modules');\r\n\r\n            let pathInProject: string = \"\";\r\n\r\n            if (positionNodeModules !== -1) {\r\n                const pathFromNodeModules = absolutePath.substring(positionNodeModules);\r\n                pathInProject = pathFromNodeModules;\r\n            } else {\r\n                pathInProject = this.convertToRelativePath(absolutePath, absolutePathProject);\r\n            }\r\n\r\n            // revert any backslashes to forward slashes (path.normalize on windows introduces them)\r\n            pathInProject = pathInProject.replace(/\\\\/g, \"/\");\r\n\r\n            if (pathInProject.startsWith(\"/\")) {\r\n                pathInProject = pathInProject.substring(1);\r\n            }\r\n\r\n            fmxIndexFileAnchor.fileName = pathInProject;\r\n            let sourceStart, sourceEnd\r\n            // ,sourceLineStart, sourceLineEnd\r\n            : number;\r\n            if (!(sourceElement instanceof CommentRange)) {\r\n                sourceStart = sourceElement.getStart();\r\n                sourceEnd = sourceElement.getEnd();\r\n                // sourceLineStart = sourceElement.getStartLineNumber();\r\n                // sourceLineEnd = sourceElement.getEndLineNumber();\r\n            } else {\r\n                sourceStart = sourceElement.getPos();\r\n                sourceEnd = sourceElement.getEnd();\r\n            }\r\n            if (config.expectGraphemes) {\r\n                /**\r\n                 * The following logic handles the case of multi-code point characters (e.g. emoji) in the source text.\r\n                 * This is needed because Pharo/Smalltalk treats multi-code point characters as a single character, \r\n                 * but JavaScript treats them as multiple characters. This means that the start and end positions\r\n                 * of a source element in Pharo/Smalltalk will be different than the start and end positions of the\r\n                 * same source element in JavaScript. This logic finds the start and end positions of the source\r\n                 * element in JavaScript and then uses those positions to set the start and end positions of the\r\n                 * Famix index file anchor.\r\n                 * It depends on code in the 'grapheme-splitter' package in npm.\r\n                 */\r\n                const splitter = new GraphemeSplitter();\r\n                const sourceFileText = sourceElement.getSourceFile().getFullText();\r\n                const hasGraphemeClusters = splitter.countGraphemes(sourceFileText) > 1;\r\n                if (hasGraphemeClusters) {\r\n                    const sourceElementText = sourceFileText.substring(sourceStart, sourceEnd);\r\n                    const sourceElementTextGraphemes = splitter.splitGraphemes(sourceElementText);\r\n                    const sourceFileTextGraphemes = splitter.splitGraphemes(sourceFileText);\r\n                    const numberOfGraphemeClustersBeforeStart = splitter.countGraphemes(sourceFileText.substring(0, sourceStart));\r\n        \r\n                    // find the start of the sourceElementTextGraphemes array in the sourceFileTextGraphemes array\r\n                    sourceStart = Helpers.indexOfSplitArray({searchArray: sourceFileTextGraphemes, \r\n                                                        targetArray: sourceElementTextGraphemes, \r\n                                                        start: sourceStart - numberOfGraphemeClustersBeforeStart});\r\n                    sourceEnd = sourceStart + sourceElementTextGraphemes.length;\r\n                } \r\n            }\r\n            // note: the +1 is because the source anchor is 1-based, but ts-morph is 0-based\r\n            fmxIndexFileAnchor.startPos = sourceStart + 1;\r\n            fmxIndexFileAnchor.endPos = sourceEnd + 1;\r\n\r\n            // 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)) {\r\n            //    initFQN(sourceElement, famixElement);\r\n            // }\r\n        } else {\r\n            // sourceElement is null\r\n            logger.warn(\"sourceElement is null for famixElement \" + famixElement.getJSON());\r\n            fmxIndexFileAnchor.fileName = \"unknown\";\r\n            fmxIndexFileAnchor.startPos = 0;\r\n            fmxIndexFileAnchor.endPos = 0;\r\n        }\r\n\r\n        this.famixRep.addElement(fmxIndexFileAnchor);\r\n    }\r\n\r\n    /**\r\n     * Creates or gets a Famix script entity or module\r\n     * @param f A source file\r\n     * @param isModule A boolean indicating if the source file is a module\r\n     * @returns The Famix model of the source file\r\n     */\r\n    public createOrGetFamixFile(f: SourceFile, isModule: boolean): Famix.ScriptEntity | Famix.Module {\r\n        let fmxFile: Famix.ScriptEntity; // | Famix.Module;\r\n\r\n        const fileName = f.getBaseName();\r\n        const fullyQualifiedFilename = f.getFilePath();\r\n        const foundFileName = this.fmxFileMap.get(fullyQualifiedFilename);\r\n        if (!foundFileName) {\r\n            if (isModule) {\r\n                fmxFile = new Famix.Module();\r\n            }\r\n            else {\r\n                fmxFile = new Famix.ScriptEntity();\r\n            }\r\n            fmxFile.name = fileName;\r\n            fmxFile.numberOfLinesOfText = f.getEndLineNumber() - f.getStartLineNumber();\r\n            fmxFile.numberOfCharacters = f.getFullText().length;\r\n\r\n            initFQN(f, fmxFile);\r\n\r\n            this.makeFamixIndexFileAnchor(f, fmxFile);\r\n\r\n            this.fmxFileMap.set(fullyQualifiedFilename, fmxFile);\r\n            this.famixRep.addElement(fmxFile);\r\n        }\r\n        else {\r\n            fmxFile = foundFileName;\r\n        }\r\n\r\n        this.fmxElementObjectMap.set(fmxFile,f);\r\n        return fmxFile;\r\n    }\r\n\r\n    /**\r\n     * Creates or gets a Famix Module\r\n     * @param moduleDeclaration A module\r\n     * @returns The Famix model of the module\r\n     */\r\n    public createOrGetFamixModule(moduleDeclaration: ModuleDeclaration): Famix.Module {\r\n        if (this.fmxModuleMap.has(moduleDeclaration)) {\r\n            const rModule = this.fmxModuleMap.get(moduleDeclaration);\r\n            if (rModule) { \r\n               return rModule;\r\n            } else {\r\n                throw new Error(`Famix module ${moduleDeclaration.getName()} is not found in the module map.`);\r\n            }\r\n        }\r\n\r\n        const fmxModule = new Famix.Module();\r\n        const moduleName = moduleDeclaration.getName();\r\n        fmxModule.name = moduleName;\r\n        fmxModule.isAmbient = isAmbient(moduleDeclaration);\r\n        fmxModule.isNamespace = isNamespace(moduleDeclaration);\r\n        fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;\r\n\r\n        initFQN(moduleDeclaration, fmxModule);\r\n        this.makeFamixIndexFileAnchor(moduleDeclaration, fmxModule);\r\n\r\n        this.fmxModuleMap.set(moduleDeclaration, fmxModule);\r\n\r\n        this.famixRep.addElement(fmxModule);\r\n\r\n        this.fmxElementObjectMap.set(fmxModule,moduleDeclaration);\r\n        return fmxModule;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix alias\r\n     * @param a An alias\r\n     * @returns The Famix model of the alias\r\n     */\r\n    public createFamixAlias(a: TypeAliasDeclaration): Famix.Alias {\r\n        let fmxAlias: Famix.Alias;\r\n        const aliasName = a.getName();\r\n        //const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);\r\n        const aliasFullyQualifiedName = FQNFunctions.getFQN(a);\r\n        const foundAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);\r\n        if (!foundAlias) {\r\n            fmxAlias = new Famix.Alias();\r\n            fmxAlias.name = a.getName();\r\n            const aliasNameWithGenerics = aliasName + (a.getTypeParameters().length ? (\"<\" + a.getTypeParameters().map(tp => tp.getName()).join(\", \") + \">\") : \"\");\r\n            logger.debug(`> NOTE: alias ${aliasName} has fully qualified name ${aliasFullyQualifiedName} and name with generics ${aliasNameWithGenerics}.`);\r\n\r\n            const fmxType = this.createOrGetFamixType(aliasNameWithGenerics, a);\r\n            fmxAlias.aliasedEntity = fmxType;\r\n            initFQN(a, fmxAlias);\r\n            this.makeFamixIndexFileAnchor(a, fmxAlias);\r\n\r\n            this.fmxAliasMap.set(aliasFullyQualifiedName, fmxAlias);\r\n\r\n            this.famixRep.addElement(fmxAlias);\r\n        }\r\n        else {\r\n            fmxAlias = foundAlias;\r\n        }\r\n        this.fmxElementObjectMap.set(fmxAlias,a);\r\n\r\n        return fmxAlias;\r\n    }\r\n\r\n    /**\r\n     * Creates or gets a Famix class or parameterizable class\r\n     * @param cls A class\r\n     * @returns The Famix model of the class\r\n     */\r\n    public createOrGetFamixClass(cls: ClassDeclaration): Famix.Class | Famix.ParametricClass {\r\n        let fmxClass: Famix.Class | Famix.ParametricClass;\r\n        const isAbstract = cls.isAbstract();\r\n        const classFullyQualifiedName = FQNFunctions.getFQN(cls);\r\n        const clsName = cls.getName() || this.UNKNOWN_VALUE;\r\n        const isGeneric = cls.getTypeParameters().length;\r\n        const foundClass = this.fmxClassMap.get(classFullyQualifiedName);\r\n        if (!foundClass) {\r\n            if (isGeneric) {\r\n                fmxClass = new Famix.ParametricClass();\r\n            }\r\n            else {\r\n                fmxClass = new Famix.Class();\r\n            }\r\n\r\n            fmxClass.name = clsName;\r\n            initFQN(cls, fmxClass);\r\n            // fmxClass.fullyQualifiedName = classFullyQualifiedName;\r\n            fmxClass.isAbstract = isAbstract;\r\n\r\n            this.makeFamixIndexFileAnchor(cls, fmxClass);\r\n\r\n            this.fmxClassMap.set(classFullyQualifiedName, fmxClass);\r\n\r\n            this.famixRep.addElement(fmxClass);\r\n\r\n            this.fmxElementObjectMap.set(fmxClass,cls);\r\n        }\r\n        else {\r\n            fmxClass = foundClass;\r\n        }\r\n\r\n        return fmxClass;\r\n    }\r\n\r\n    /**\r\n     * Creates or gets a Famix interface or parameterizable interface\r\n     * @param inter An interface\r\n     * @returns The Famix model of the interface\r\n     */\r\n    public createOrGetFamixInterface(inter: InterfaceDeclaration): Famix.Interface | Famix.ParametricInterface {\r\n\r\n        let fmxInterface: Famix.Interface | Famix.ParametricInterface;\r\n        const interName = inter.getName();\r\n        const interFullyQualifiedName = FQNFunctions.getFQN(inter);\r\n        const foundInterface = this.fmxInterfaceMap.get(interFullyQualifiedName);\r\n        if (!foundInterface) {\r\n            const isGeneric = inter.getTypeParameters().length;\r\n            if (isGeneric) {\r\n                fmxInterface = new Famix.ParametricInterface();\r\n            }\r\n            else {\r\n                fmxInterface = new Famix.Interface();\r\n            }\r\n\r\n            fmxInterface.name = interName;\r\n            initFQN(inter, fmxInterface);\r\n            this.makeFamixIndexFileAnchor(inter, fmxInterface);\r\n\r\n            this.fmxInterfaceMap.set(interFullyQualifiedName, fmxInterface);\r\n\r\n            this.famixRep.addElement(fmxInterface);\r\n\r\n            this.fmxElementObjectMap.set(fmxInterface,inter);\r\n        }\r\n        else {\r\n            fmxInterface = foundInterface;\r\n        }\r\n        return fmxInterface;\r\n    }\r\n\r\n    \r\n    /**\r\n     * Creates or gets a Famix concrete element\r\n     * @param concreteElement A parametric Element   \r\n     * @param concreteElementDeclaration the element declaration\r\n     * @param concreteArguments concrete arguments\r\n     * @returns A parametric Element  \r\n     */\r\n    // public createOrGetFamixConcreteElement(concreteElement : ParametricVariantType, \r\n    //                                        concreteElementDeclaration : ConcreteElementTSMorphType, \r\n    //                                        concreteArguments: TypeNode[]): ParametricVariantType {\r\n        \r\n    //     let fullyQualifiedFilename = concreteElement.fullyQualifiedName;\r\n    //     let params = \"\";\r\n        \r\n    //     concreteArguments.map((param) => {\r\n    //         params = params+param.getText()+',';\r\n    //     });\r\n        \r\n    //     params = params.substring(0, params.length - 1)\r\n                \r\n    //     fullyQualifiedFilename = Helpers.replaceLastBetweenTags(fullyQualifiedFilename,params);\r\n\r\n    //     let concElement: ParametricVariantType;\r\n\r\n    //     if (!this.fmxInterfaceMap.has(fullyQualifiedFilename) && \r\n    //         !this.fmxClassMap.has(fullyQualifiedFilename) && \r\n    //         !this.fmxFunctionAndMethodMap.has(fullyQualifiedFilename)){\r\n    //         concElement = _.cloneDeep(concreteElement); \r\n    //         concElement.fullyQualifiedName = fullyQualifiedFilename;\r\n    //         concElement.clearGenericParameters();\r\n    //         concreteArguments.map((param) => {\r\n    //             const parameter = this.createOrGetFamixConcreteType(param);\r\n    //             concElement.addConcreteParameter(parameter);\r\n    //         })\r\n            \r\n    //         if (concreteElement instanceof Famix.ParametricClass) {\r\n    //             this.fmxClassMap.set(fullyQualifiedFilename, concElement as Famix.ParametricClass);\r\n    //         } else if (concreteElement instanceof Famix.ParametricInterface) {\r\n    //             this.fmxInterfaceMap.set(fullyQualifiedFilename, concElement as Famix.ParametricInterface);\r\n    //         } else if (concreteElement instanceof Famix.ParametricFunction) {\r\n    //             this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricFunction);\r\n    //         } else { // if (concreteElement instanceof Famix.ParametricMethod) {\r\n    //             this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement as Famix.ParametricMethod);\r\n    //         }\r\n    //         this.famixRep.addElement(concElement);\r\n    //         this.fmxElementObjectMap.set(concElement,concreteElementDeclaration);\r\n    //     } else {\r\n    //         if (concreteElement instanceof Famix.ParametricClass) {\r\n    //             concElement = this.fmxClassMap.get(fullyQualifiedFilename) as Famix.ParametricClass;\r\n    //         } else if (concreteElement instanceof Famix.ParametricInterface) {\r\n    //             concElement = this.fmxInterfaceMap.get(fullyQualifiedFilename) as Famix.ParametricInterface;\r\n    //         } else if (concreteElement instanceof Famix.ParametricFunction) {\r\n    //             concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricFunction;\r\n    //         } else {  // if (concreteElement instanceof Famix.ParametricMethod) {\r\n    //             concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename) as Famix.ParametricMethod;\r\n    //         }\r\n    //     }\r\n    //     return concElement;\r\n    // }\r\n\r\n    /**\r\n     * Creates a Famix property\r\n     * @param property A property\r\n     * @returns The Famix model of the property\r\n     */\r\n    public createFamixProperty(property: PropertyDeclaration | PropertySignature): Famix.Property {\r\n        const fmxProperty = new Famix.Property();\r\n        const isSignature = property instanceof PropertySignature;\r\n        fmxProperty.name = property.getName();\r\n\r\n        let propTypeName = this.UNKNOWN_VALUE;\r\n        try {\r\n            propTypeName = property.getType().getText().trim();\r\n        } catch (error) {\r\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for property: ${property.getName()}. Continuing...`);\r\n        }\r\n\r\n        const fmxType = this.createOrGetFamixType(propTypeName, property);\r\n        fmxProperty.declaredType = fmxType;\r\n\r\n        // add the visibility (public, private, etc.) to the fmxProperty\r\n        fmxProperty.visibility = \"\";\r\n\r\n        property.getModifiers().forEach(m => {\r\n            switch (m.getText()) {\r\n                case Scope.Public:\r\n                    fmxProperty.visibility = \"public\";\r\n                    break;\r\n                case Scope.Protected:\r\n                    fmxProperty.visibility = \"protected\";\r\n                    break;\r\n                case Scope.Private:\r\n                    fmxProperty.visibility = \"private\";\r\n                    break;\r\n                case \"static\":\r\n                    fmxProperty.isClassSide = true;\r\n                    break;\r\n                case \"readonly\":\r\n                    fmxProperty.readOnly = true;\r\n                    break;\r\n                default:\r\n                    break;\r\n            }\r\n        });\r\n\r\n        if (!isSignature && property.getExclamationTokenNode()) {\r\n            fmxProperty.isDefinitelyAssigned = true;\r\n        }\r\n        if (property.getQuestionTokenNode()) {\r\n            fmxProperty.isOptional = true;\r\n        }\r\n        if (property.getName().substring(0, 1) === \"#\") {\r\n            fmxProperty.isJavaScriptPrivate = true;\r\n        }\r\n\r\n        initFQN(property, fmxProperty);\r\n        this.makeFamixIndexFileAnchor(property, fmxProperty);\r\n\r\n        this.famixRep.addElement(fmxProperty);\r\n\r\n        this.fmxElementObjectMap.set(fmxProperty,property);\r\n\r\n        return fmxProperty;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix method or accessor\r\n     * @param method A method or an accessor\r\n     * @param currentCC The cyclomatic complexity metrics of the current source file\r\n     * @returns The Famix model of the method or the accessor\r\n     */\r\n    public createOrGetFamixMethod(method: MethodDeclaration | ConstructorDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration, currentCC: { [key: string]: number }): Famix.Method | Famix.Accessor | Famix.ParametricMethod {\r\n        let fmxMethod: Famix.Method | Famix.Accessor | Famix.ParametricMethod;\r\n        const isGeneric = method.getTypeParameters().length > 0;\r\n        const functionFullyQualifiedName = FQNFunctions.getFQN(method);\r\n        if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {\r\n\r\n            if (method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) {\r\n                fmxMethod = new Famix.Accessor();\r\n                const isGetter = method instanceof GetAccessorDeclaration;\r\n                const isSetter = method instanceof SetAccessorDeclaration;\r\n                if (isGetter) {(fmxMethod as Famix.Accessor).kind = \"getter\";}\r\n                if (isSetter) {(fmxMethod as Famix.Accessor).kind = \"setter\";}\r\n                this.famixRep.addElement(fmxMethod);\r\n            }\r\n            else {\r\n                if (isGeneric) {\r\n                    fmxMethod = new Famix.ParametricMethod();\r\n                }\r\n                else {\r\n                    fmxMethod = new Famix.Method();\r\n                }\r\n            }\r\n            const isConstructor = method instanceof ConstructorDeclaration;\r\n            const isSignature = method instanceof MethodSignature;\r\n\r\n            let isAbstract = false;\r\n            let isStatic = false;\r\n            if (method instanceof MethodDeclaration || method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) {\r\n                isAbstract = method.isAbstract();\r\n                isStatic = method.isStatic();\r\n            }\r\n\r\n            if (isConstructor) {(fmxMethod as Famix.Accessor).kind = \"constructor\";}\r\n            fmxMethod.isAbstract = isAbstract;\r\n            fmxMethod.isClassSide = isStatic;\r\n            fmxMethod.isPrivate = (method instanceof MethodDeclaration || method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'private')) !== undefined : false;\r\n            fmxMethod.isProtected = (method instanceof MethodDeclaration || method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'protected')) !== undefined : false;\r\n            fmxMethod.signature = Helpers.computeSignature(method.getText());\r\n\r\n            let methodName: string;\r\n            if (isConstructor) {\r\n                methodName = \"constructor\";\r\n            }\r\n            else {\r\n                methodName = (method as MethodDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration).getName();\r\n            }\r\n            fmxMethod.name = methodName;\r\n\r\n            if (!isConstructor) {\r\n                if (method.getName().substring(0, 1) === \"#\") {\r\n                    fmxMethod.isPrivate = true;\r\n                }\r\n            }\r\n\r\n            if (!fmxMethod.isPrivate && !fmxMethod.isProtected) {\r\n                fmxMethod.isPublic = true;\r\n            }\r\n            else {\r\n                fmxMethod.isPublic = false;\r\n            }\r\n\r\n            if (!isSignature) {\r\n                fmxMethod.cyclomaticComplexity = currentCC[fmxMethod.name];\r\n            }\r\n            else {\r\n                fmxMethod.cyclomaticComplexity = 0;\r\n            }\r\n\r\n            let methodTypeName = this.UNKNOWN_VALUE; \r\n            try {\r\n                methodTypeName = method.getReturnType().getText().trim();            \r\n            } catch (error) {\r\n                logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of method: ${fmxMethod.name}. Continuing...`);\r\n            }\r\n\r\n            const fmxType = this.createOrGetFamixType(methodTypeName, method);\r\n            fmxMethod.declaredType = fmxType;\r\n            fmxMethod.numberOfLinesOfCode = method.getEndLineNumber() - method.getStartLineNumber();\r\n            const parameters = method.getParameters();\r\n            fmxMethod.numberOfParameters = parameters.length;\r\n\r\n            if (!isSignature) {\r\n                fmxMethod.numberOfStatements = method.getStatements().length;\r\n            }\r\n            else {\r\n                fmxMethod.numberOfStatements = 0;\r\n            }\r\n            \r\n            initFQN(method, fmxMethod);\r\n            this.famixRep.addElement(fmxMethod);\r\n            this.makeFamixIndexFileAnchor(method, fmxMethod);\r\n\r\n            this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxMethod);\r\n        }\r\n        else {\r\n            fmxMethod = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName) as (Famix.Method | Famix.Accessor | Famix.ParametricMethod);\r\n        }\r\n\r\n        this.fmxElementObjectMap.set(fmxMethod,method);\r\n        \r\n        return fmxMethod;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix function\r\n     * @param func A function\r\n     * @param currentCC The cyclomatic complexity metrics of the current source file\r\n     * @returns The Famix model of the function\r\n     */\r\n    public createOrGetFamixFunction(func: FunctionDeclaration | FunctionExpression, currentCC: { [key: string]: number }): Famix.Function | Famix.ParametricFunction {\r\n        let fmxFunction: Famix.Function | Famix.ParametricFunction;\r\n        const isGeneric = func.getTypeParameters().length > 0;\r\n        const functionFullyQualifiedName = FQNFunctions.getFQN(func);\r\n        if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {\r\n            if (isGeneric) {\r\n                fmxFunction = new Famix.ParametricFunction();\r\n            }\r\n            else {\r\n                fmxFunction = new Famix.Function();\r\n            }\r\n    \r\n            const name = func.getName();\r\n            if (name) {\r\n                fmxFunction.name = name;\r\n            }\r\n            else {\r\n                fmxFunction.name = \"anonymous\";\r\n            }\r\n\r\n            fmxFunction.signature = Helpers.computeSignature(func.getText());\r\n            fmxFunction.cyclomaticComplexity = currentCC[fmxFunction.name];\r\n            initFQN(func, fmxFunction);\r\n            // fmxFunction.fullyQualifiedName = functionFullyQualifiedName;\r\n    \r\n            let functionTypeName = this.UNKNOWN_VALUE;\r\n            try {\r\n                functionTypeName = func.getReturnType().getText().trim();\r\n            } catch (error) {\r\n                logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${func.getName()}. Continuing...`);\r\n            }\r\n    \r\n            const fmxType = this.createOrGetFamixType(functionTypeName, func);\r\n            fmxFunction.declaredType = fmxType;\r\n            fmxFunction.numberOfLinesOfCode = func.getEndLineNumber() - func.getStartLineNumber();\r\n            const parameters = func.getParameters();\r\n            fmxFunction.numberOfParameters = parameters.length;\r\n            fmxFunction.numberOfStatements = func.getStatements().length;\r\n            this.makeFamixIndexFileAnchor(func, fmxFunction);\r\n    \r\n            this.famixRep.addElement(fmxFunction);\r\n    \r\n            this.fmxElementObjectMap.set(fmxFunction,func);\r\n\r\n            this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxFunction);\r\n        }\r\n        else {\r\n            fmxFunction = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName) as (Famix.Function | Famix.ParametricFunction);\r\n        }\r\n\r\n        return fmxFunction;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix parameter\r\n     * @param param A parameter\r\n     * @returns The Famix model of the parameter\r\n     */\r\n    public createOrGetFamixParameter(param: ParameterDeclaration): Famix.Parameter {\r\n        if (this.fmxParameterMap.has(param)) {\r\n            const rParameter = this.fmxParameterMap.get(param);\r\n            if (rParameter) { \r\n               return rParameter;\r\n            } else {\r\n                throw new Error(`Famix parameter ${param.getName()} is not found in the parameter map.`);\r\n            }\r\n        }\r\n\r\n        const fmxParam = new Famix.Parameter();\r\n\r\n        let paramTypeName = this.UNKNOWN_VALUE;\r\n        try {\r\n            paramTypeName = param.getType().getText().trim();\r\n        } catch (error) {\r\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for parameter: ${param.getName()}. Continuing...`);\r\n        }\r\n\r\n        const fmxType = this.createOrGetFamixType(paramTypeName, param);\r\n        fmxParam.declaredType = fmxType;\r\n        fmxParam.name = param.getName();\r\n\r\n        initFQN(param, fmxParam);\r\n        this.makeFamixIndexFileAnchor(param, fmxParam);\r\n\r\n        this.famixRep.addElement(fmxParam);\r\n\r\n        this.fmxElementObjectMap.set(fmxParam, param);\r\n        this.fmxParameterMap.set(param, fmxParam);\r\n\r\n        return fmxParam;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix type parameter\r\n     * @param tp A type parameter\r\n     * @returns The Famix model of the type parameter\r\n     */\r\n    public createFamixParameterType(tp: TypeParameterDeclaration): Famix.ParameterType {\r\n        \r\n        const fmxParameterType = new Famix.ParameterType();\r\n   \r\n        fmxParameterType.name = tp.getName();      \r\n        initFQN(tp, fmxParameterType);\r\n        this.makeFamixIndexFileAnchor(tp, fmxParameterType);\r\n\r\n        this.famixRep.addElement(fmxParameterType);\r\n\r\n        this.fmxElementObjectMap.set(fmxParameterType,tp);\r\n\r\n        return fmxParameterType;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix type in the context of concretizations\r\n     * @param typeName A type name\r\n     * @param element An element\r\n     * @returns The Famix model of the type\r\n     */\r\n    public createOrGetFamixConcreteType(element: TypeNode): \r\n        Famix.ParameterType | Famix.PrimitiveType | Famix.Class | Famix.Interface {\r\n        // TODO - refactor to stop using names as a key in the maps, use ts-morph element instead\r\n        const typeParameterDeclaration = element.getSymbol()?.getDeclarations()[0] as TypeParameterDeclaration;\r\n        const parameterTypeName : string = element.getText();\r\n        let fmxParameterType: Famix.Type | Famix.Class | Famix.Interface | undefined = undefined;\r\n\r\n        // get a TypeReference from a TypeNode\r\n        const typeReference = element.getType();\r\n        // get a TypeDeclaration from a TypeReference\r\n        const typeDeclaration = typeReference.getSymbol()?.getDeclarations()[0] as TSMorphTypeDeclaration;\r\n\r\n        let isClassOrInterface = false;\r\n        if (this.fmxClassMap.has(parameterTypeName)){\r\n            this.fmxClassMap.forEach((obj, name) => {\r\n                if(obj instanceof Famix.ParametricClass){\r\n                    if (name === element.getText() && obj.genericParameters.size>0) {\r\n                        fmxParameterType = obj;\r\n                        isClassOrInterface = true;\r\n                    } \r\n                } else {\r\n                    if (name === element.getText()) {\r\n                        fmxParameterType = obj;\r\n                        isClassOrInterface = true;\r\n                    } \r\n                }   \r\n            });\r\n        }\r\n\r\n        if (this.fmxInterfaceMap.has(parameterTypeName)){\r\n            this.fmxInterfaceMap.forEach((obj, name) => {\r\n                if(obj instanceof Famix.ParametricInterface){\r\n                    if (name === element.getText() && obj.genericParameters.size>0) {\r\n                        fmxParameterType = obj;\r\n                        isClassOrInterface = true;\r\n                    } \r\n                } else {\r\n                    if (name === element.getText()) {\r\n                        fmxParameterType = obj;\r\n                        isClassOrInterface = true;\r\n                    } \r\n                }   \r\n            });\r\n        }\r\n\r\n        if(!isClassOrInterface){\r\n            if (!this.fmxTypeMap.has(typeDeclaration)) {    \r\n                // TODO refactor \r\n                if (isPrimitiveType(parameterTypeName)) {\r\n                    fmxParameterType = new Famix.PrimitiveType();\r\n                    fmxParameterType.isStub = true;\r\n                } else {\r\n                    fmxParameterType = new Famix.ParameterType();\r\n                } \r\n    \r\n                fmxParameterType.name = parameterTypeName;\r\n                this.famixRep.addElement(fmxParameterType);\r\n                this.fmxTypeMap.set(typeDeclaration, fmxParameterType);\r\n                this.fmxElementObjectMap.set(fmxParameterType,typeParameterDeclaration);\r\n            }\r\n            else {\r\n                const result = this.fmxTypeMap.get(typeDeclaration);\r\n                if (result) {\r\n                    fmxParameterType = result;\r\n                } else {\r\n                    throw new Error(`Famix type ${typeDeclaration} is not found in the Type map.`);\r\n                }\r\n            }\r\n        }\r\n        if (!fmxParameterType) {\r\n            throw new Error(`fmxParameterType was undefined for parameterTypeName ${parameterTypeName}`);\r\n        }\r\n        return fmxParameterType;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix variable\r\n     * @param variable A variable\r\n     * @returns The Famix model of the variable\r\n     */\r\n    public createOrGetFamixVariable(variable: VariableDeclaration): Famix.Variable {\r\n        if (this.fmxVariableMap.has(variable)) {\r\n            const rVariable = this.fmxVariableMap.get(variable);\r\n            if (rVariable) { \r\n               return rVariable;\r\n            } else {\r\n                throw new Error(`Famix parameter ${variable.getName()} is not found in the variable map.`);\r\n            }\r\n        }\r\n        const fmxVariable = new Famix.Variable();\r\n    \r\n        let variableTypeName = this.UNKNOWN_VALUE;\r\n        try {\r\n            variableTypeName = variable.getType().getText().trim();\r\n        } catch (error) {\r\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for variable: ${variable.getName()}. Continuing...`);\r\n        }\r\n    \r\n        const fmxType = this.createOrGetFamixType(variableTypeName, variable);\r\n        fmxVariable.declaredType = fmxType;\r\n        fmxVariable.name = variable.getName();\r\n        initFQN(variable, fmxVariable);\r\n        this.makeFamixIndexFileAnchor(variable, fmxVariable);\r\n    \r\n        this.famixRep.addElement(fmxVariable);\r\n    \r\n        this.fmxElementObjectMap.set(fmxVariable,variable);\r\n        this.fmxVariableMap.set(variable, fmxVariable);\r\n    \r\n        return fmxVariable;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix enum\r\n     * @param enumEntity An enum\r\n     * @returns The Famix model of the enum\r\n     */\r\n    public createOrGetFamixEnum(enumEntity: EnumDeclaration): Famix.Enum {\r\n        if (this.fmxEnumMap.has(enumEntity)) {\r\n            const rEnum = this.fmxEnumMap.get(enumEntity);\r\n            if (rEnum) { \r\n               return rEnum;\r\n            } else {\r\n                throw new Error(`Famix enum ${enumEntity.getName()} is not found in the enum map.`);\r\n            }\r\n        }\r\n        const fmxEnum = new Famix.Enum();\r\n        fmxEnum.name = enumEntity.getName();\r\n        initFQN(enumEntity, fmxEnum);\r\n        this.makeFamixIndexFileAnchor(enumEntity, fmxEnum);\r\n\r\n        this.famixRep.addElement(fmxEnum);\r\n\r\n        this.fmxElementObjectMap.set(fmxEnum,enumEntity);\r\n        this.fmxEnumMap.set(enumEntity, fmxEnum);\r\n\r\n        return fmxEnum;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix enum value\r\n     * @param enumMember An enum member\r\n     * @returns The Famix model of the enum member\r\n     */\r\n    public createFamixEnumValue(enumMember: EnumMember): Famix.EnumValue {\r\n        const fmxEnumValue = new Famix.EnumValue();\r\n\r\n        let enumValueTypeName = this.UNKNOWN_VALUE;\r\n        try {\r\n            enumValueTypeName = enumMember.getType().getText().trim();\r\n        } catch (error) {\r\n            logger.error(`> WARNING: got exception ${error}. Failed to get usable name for enum value: ${enumMember.getName()}. Continuing...`);\r\n        }\r\n\r\n        const fmxType = this.createOrGetFamixType(enumValueTypeName, enumMember);\r\n        fmxEnumValue.declaredType = fmxType;\r\n        fmxEnumValue.name = enumMember.getName();\r\n        initFQN(enumMember, fmxEnumValue);\r\n        this.makeFamixIndexFileAnchor(enumMember, fmxEnumValue);\r\n\r\n        this.famixRep.addElement(fmxEnumValue);\r\n\r\n        this.fmxElementObjectMap.set(fmxEnumValue,enumMember);\r\n\r\n        return fmxEnumValue;\r\n    }\r\n\r\n    /**\r\n     * Creates or gets a Famix decorator\r\n     * @param decorator A decorator\r\n     * @param decoratedEntity A class, a method, a parameter or a property\r\n     * @returns The Famix model of the decorator\r\n     */\r\n    public createOrGetFamixDecorator(decorator: Decorator, decoratedEntity: ClassDeclaration | MethodDeclaration | GetAccessorDeclaration | SetAccessorDeclaration | ParameterDeclaration | PropertyDeclaration): Famix.Decorator {\r\n        const fmxDecorator = new Famix.Decorator();\r\n        const decoratorName = \"@\" + decorator.getName();\r\n        const decoratorExpression = decorator.getText().substring(1);\r\n\r\n        fmxDecorator.name = decoratorName;\r\n        fmxDecorator.decoratorExpression = decoratorExpression;\r\n        const decoratedEntityFullyQualifiedName = FQNFunctions.getFQN(decoratedEntity);\r\n        const fmxDecoratedEntity = this.famixRep.getFamixEntityByFullyQualifiedName(decoratedEntityFullyQualifiedName) as Famix.NamedEntity;\r\n        fmxDecorator.decoratedEntity = fmxDecoratedEntity;\r\n        initFQN(decorator, fmxDecorator);\r\n        this.makeFamixIndexFileAnchor(decorator, fmxDecorator);\r\n\r\n        this.famixRep.addElement(fmxDecorator);\r\n\r\n        this.fmxElementObjectMap.set(fmxDecorator,decorator);\r\n\r\n        return fmxDecorator;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix comment\r\n     * @param comment A comment\r\n     * @param fmxScope The Famix model of the comment's container\r\n     * @param isJSDoc A boolean indicating if the comment is a JSDoc\r\n     * @returns The Famix model of the comment\r\n     */\r\n    public createFamixComment(comment: CommentRange, fmxScope: Famix.NamedEntity, isJSDoc: boolean): Famix.Comment {\r\n        logger.debug(`> NOTE: creating comment ${comment.getText()} in scope ${fmxScope.name}.`);\r\n        const fmxComment = new Famix.Comment();\r\n        fmxComment.container = fmxScope;  // adds comment to the container's comments collection\r\n        fmxComment.isJSDoc = isJSDoc;\r\n\r\n        this.makeFamixIndexFileAnchor(comment, fmxComment);\r\n\r\n        this.famixRep.addElement(fmxComment);\r\n\r\n        this.fmxElementObjectMap.set(fmxComment,comment);\r\n\r\n        return fmxComment;\r\n    }\r\n\r\n    /**\r\n     * Creates or gets a Famix type\r\n     * @param typeName A type name\r\n     * @param element A ts-morph element\r\n     * @returns The Famix model of the type\r\n     */\r\n    public createOrGetFamixType(typeName: string, element: TSMorphTypeDeclaration): Famix.Type {\r\n        let fmxType: Famix.Type;\r\n        const isPrimitive = isPrimitiveType(typeName);\r\n        const isParametricType =\r\n            element instanceof ClassDeclaration && element.getTypeParameters().length > 0 ||\r\n            element instanceof InterfaceDeclaration && element.getTypeParameters().length > 0; \r\n\r\n            // Functions and methods aren't types!\r\n            // ||\r\n            // element instanceof FunctionDeclaration && element.getTypeParameters().length > 0 ||\r\n            // element instanceof MethodDeclaration && element.getTypeParameters().length > 0 ||\r\n            // element instanceof ArrowFunction && element.getTypeParameters().length > 0;\r\n\r\n        logger.debug(\"Creating (or getting) type: '\" + typeName + \"' of element: \" + element?.getText() + \" of kind: \" + element?.getKindName());\r\n\r\n        if (isPrimitive) {\r\n            return this.createOrGetFamixPrimitiveType(typeName);\r\n        }\r\n\r\n        if (isParametricType) {\r\n            // narrow the type\r\n            const parametricElement = element as TSMorphParametricType;\r\n            return this.createOrGetFamixParametricType(typeName, parametricElement);\r\n        }\r\n\r\n        if (!this.fmxTypeMap.has(element)) {\r\n            let ancestor: Famix.ContainerEntity | undefined = undefined;\r\n \r\n            if (element !== undefined) {\r\n                const typeAncestor = Helpers.findTypeAncestor(element);\r\n                if (!typeAncestor) {\r\n                    throw new Error(`Ancestor not found for element ${element.getText()}.`);\r\n                }\r\n                const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);\r\n                ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName) as Famix.ContainerEntity;\r\n                if (!ancestor) {\r\n                    logger.debug(`Ancestor ${FQNFunctions.getFQN(typeAncestor)} not found. Adding the new type.`);\r\n                    ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor as TSMorphTypeDeclaration);\r\n                }\r\n            }\r\n\r\n            fmxType = new Famix.Type();\r\n            fmxType.name = typeName;\r\n\r\n            if (!ancestor) {\r\n                throw new Error(`Ancestor not found for type ${typeName}.`);\r\n            }\r\n            fmxType.container = ancestor;\r\n            initFQN(element, fmxType);\r\n            this.makeFamixIndexFileAnchor(element, fmxType);\r\n\r\n            this.famixRep.addElement(fmxType);\r\n\r\n            this.fmxTypeMap.set(element, fmxType);\r\n        }\r\n        else {\r\n            const result = this.fmxTypeMap.get(element);\r\n            if (result) {\r\n                fmxType = result;\r\n            } else {\r\n                throw new Error(`Famix type ${typeName} is not found in the Type map.`);\r\n            }\r\n        }\r\n\r\n        this.fmxElementObjectMap.set(fmxType,element);\r\n\r\n        return fmxType;\r\n    }\r\n\r\n    /**\r\n     * Creates or gets a Famix type that is parametric\r\n     * @param typeName A type name\r\n     * @param element A ts-morph element\r\n     * @returns The Famix model of the parameter type\r\n     */\r\n    createOrGetFamixParametricType(typeName: string, element: TSMorphParametricType): Famix.Type {\r\n\r\n        if (this.fmxTypeMap.has(element) === true) {\r\n            const result = this.fmxTypeMap.get(element);\r\n            if (result) {\r\n                return result;\r\n            } else {\r\n                throw new Error(`Famix type ${typeName} is not found (undefined) in the Type map.`);\r\n            }\r\n        }\r\n\r\n        // A parametric type is a type that has type parameters, e.g., List<T>\r\n        // In TS it can be a class, an interface, a function, an arrow function, or a method\r\n\r\n        // create the Famix Parametric Type (maybe it's just an Interface, etc.)\r\n        let fmxType: Famix.Type;\r\n\r\n        if (element instanceof ClassDeclaration) {\r\n            fmxType = new Famix.ParametricClass();\r\n        } else if (element instanceof InterfaceDeclaration) {\r\n            fmxType = new Famix.ParametricInterface();\r\n        }\r\n        // functions and methods are not types \r\n        // else if (element instanceof FunctionDeclaration) {\r\n        //     fmxType = new Famix.ParametricFunction();\r\n        // } else if (element instanceof ArrowFunction) {\r\n        //     fmxType = new Famix.ParametricArrowFunction();\r\n        // } else if (element instanceof MethodDeclaration) {\r\n        //     fmxType = new Famix.ParametricMethod();\r\n        // } \r\n        else {\r\n            throw new Error(`Element is not a class, interface, function, arrow function, or method.`);\r\n        }\r\n\r\n        // const parameters = element.getTypeParameters();\r\n\r\n        // // for each parameter, getOrCreate the FamixParameterType\r\n        // for (const parameter of parameters) {\r\n        //     this.createOrGetFamixParameterType(parameter.getName(), parameter);\r\n        // }\r\n\r\n        // // TODO: the following code is not correct, it is just a placeholder\r\n        // const parameterTypeNames = typeName.substring(typeName.indexOf(\"<\") + 1, typeName.indexOf(\">\"))\r\n        //     .split(\",\").map(s => s.trim());\r\n        // const baseTypeName = typeName.substring(0, typeName.indexOf(\"<\")).trim();\r\n        // parameterTypeNames.forEach(parameterTypeName => {\r\n        //     const fmxParameterType = this.createOrGetFamixParameterType(parameterTypeName, element);\r\n        //     (fmxType as Famix.ParameterType).addArgument(fmxParameterType);\r\n        // });\r\n        // const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);\r\n\r\n        // (fmxType as Famix.ParameterType).baseType = fmxBaseType;\r\n\r\n        fmxType.name = typeName;\r\n        initFQN(element, fmxType);\r\n        this.famixRep.addElement(fmxType);\r\n        this.fmxTypeMap.set(element, fmxType);\r\n        return fmxType;\r\n    }\r\n\r\n    /**\r\n     * Creates a type for a parameter in a parametric type, e.g., T in List<T>\r\n     * @param parameterTypeName \r\n     * @param element the TypeScript element (TSMorphParametricType) that the type is associated with\r\n     * @returns \r\n     */\r\n    // createOrGetFamixParameterType(parameterTypeName: string, element: ParameterDeclaration) {\r\n    //     if (this.fmxTypeMap.has(element)) {\r\n    //         return this.fmxTypeMap.get(element) as Famix.ParameterType;\r\n    //     }\r\n\r\n    //     // determine if element is a \r\n    //     const fmxType = new Famix.ParameterType();\r\n    //     // const parameterTypeNames = typeName.substring(typeName.indexOf(\"<\") + 1, typeName.indexOf(\">\"))\r\n    //     //     .split(\",\").map(s => s.trim());\r\n    //     // const baseTypeName = typeName.substring(0, typeName.indexOf(\"<\")).trim();\r\n    //     // parameterTypeNames.forEach(parameterTypeName => {\r\n    //     //     const fmxParameterType = this.createOrGetFamixParameterType(parameterTypeName, element);\r\n    //     //     (fmxType as Famix.ParameterType).addArgument(fmxParameterType);\r\n    //     // });\r\n    //     const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);\r\n    //     (fmxType as Famix.ParameterType).baseType = fmxBaseType;\r\n    //     initFQN(element, fmxType);\r\n    //     this.famixRep.addElement(fmxType);\r\n    //     this.fmxTypeMap.set(element, fmxType);\r\n    //     return fmxType;\r\n    // }\r\n\r\n    /**\r\n     * Creates or gets a Famix primitive type\r\n     * @param typeName A type name\r\n     * @returns The Famix model of the primitive type\r\n     */\r\n    createOrGetFamixPrimitiveType(typeName: string): Famix.PrimitiveType {\r\n        let fmxType: Famix.PrimitiveType = new Famix.PrimitiveType();\r\n        if (!this.fmxPrimitiveTypeMap.has(typeName)) {\r\n            fmxType = new Famix.PrimitiveType();\r\n            fmxType.isStub = true;\r\n            fmxType.name = typeName;\r\n            this.fmxPrimitiveTypeMap.set(typeName, fmxType);\r\n            this.famixRep.addElement(fmxType);\r\n        } else {\r\n            fmxType = this.fmxPrimitiveTypeMap.get(typeName) as Famix.PrimitiveType;\r\n        }\r\n        return fmxType;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix access\r\n     * @param node A node\r\n     * @param id An id of a parameter, a variable, a property or an enum member\r\n     */\r\n    public createFamixAccess(node: Identifier, id: number): void {\r\n        const fmxVar = this.famixRep.getFamixEntityById(id) as Famix.StructuralEntity;\r\n        if (!fmxVar) {\r\n            throw new Error(`Famix entity with id ${id} not found, for node ${node.getText()} in ${node.getSourceFile().getBaseName()} at line ${node.getStartLineNumber()}.`);\r\n        }\r\n\r\n        logger.debug(`Creating FamixAccess. Node: [${node.getKindName()}] '${node.getText()}' at line ${node.getStartLineNumber()} in ${node.getSourceFile().getBaseName()}, id: ${id} refers to fmxVar '${fmxVar.fullyQualifiedName}'.`);\r\n\r\n        const nodeReferenceAncestor = Helpers.findAncestor(node);\r\n        const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);\r\n        const accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName) as Famix.ContainerEntity;\r\n        if (!accessor) {\r\n            logger.error(`Ancestor ${ancestorFullyQualifiedName} of kind ${nodeReferenceAncestor.getKindName()} not found.`);\r\n            // accessor = this.createOrGetFamixType(ancestorFullyQualifiedName, nodeReferenceAncestor as TypeDeclaration);\r\n            return; // bail out TODO: this is probably wrong\r\n        } else {\r\n            logger.debug(`Found accessor to be ${accessor.fullyQualifiedName}.`);\r\n        }\r\n\r\n\r\n        // make sure accessor is a method, function, script or module\r\n        if (!(accessor instanceof Famix.Method) && !(accessor instanceof Famix.ArrowFunction) && !(accessor instanceof Famix.Function) && !(accessor instanceof Famix.ScriptEntity) && !(accessor instanceof Famix.Module)) {\r\n            logger.error(`Accessor ${accessor.fullyQualifiedName} is not a method, function, etc.`);\r\n            return;\r\n        }\r\n\r\n        // don't create any duplicates (e.g. if the same variable is accessed multiple times by same accessor)\r\n        const foundAccess = this.famixRep.getFamixAccessByAccessorAndVariable(accessor, fmxVar);\r\n        if (foundAccess) {\r\n            logger.debug(`FamixAccess already exists for accessor ${accessor.fullyQualifiedName} and variable ${fmxVar.fullyQualifiedName}.`);\r\n            return;\r\n        }\r\n        const fmxAccess = new Famix.Access();\r\n        fmxAccess.accessor = accessor;\r\n        fmxAccess.variable = fmxVar;\r\n\r\n        this.famixRep.addElement(fmxAccess);\r\n\r\n        this.fmxElementObjectMap.set(fmxAccess,node);\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix invocation\r\n     * @param nodeReferringToInvocable A node\r\n     * @param invocable A method or a function\r\n     * @param id The id of the method or the function\r\n     */\r\n    public createFamixInvocation(nodeReferringToInvocable: Identifier, invocable: InvocableType, id: number): void {\r\n        const fmxInvocable = this.famixRep.getFamixEntityById(id) as Famix.BehavioralEntity;\r\n        // since the node is in the AST, we need to find the ancestor that is in the Famix model\r\n        const containerOfNode = Helpers.findAncestor(nodeReferringToInvocable);\r\n        logger.debug(`Found container (ancestor) ${containerOfNode.getKindName()} for AST node ${nodeReferringToInvocable.getText()}.`);\r\n        const containerFQN = FQNFunctions.getFQN(containerOfNode);\r\n        logger.debug(`Found containerFQN ${containerFQN}.`);\r\n        let sender = this.famixRep.getFamixEntityByFullyQualifiedName(containerFQN) as Famix.ContainerEntity;\r\n        logger.debug(`Found a sender that matches ${sender.fullyQualifiedName}.`);\r\n        if (sender instanceof Famix.Type) {\r\n            // TODO this might be an error in getFamixEntityByFullyQualifiedName\r\n            logger.debug(`Oops! Sender is a type, which is not valid for an Invocation. Trying to find a container for ${sender.fullyQualifiedName}.`);\r\n            const senderContainer = sender.container;\r\n            if (senderContainer) {\r\n                sender = senderContainer;\r\n            }\r\n        }\r\n        const receiverFullyQualifiedName = FQNFunctions.getFQN(invocable.getParent());\r\n        const receiver = this.famixRep.getFamixEntityByFullyQualifiedName(receiverFullyQualifiedName) as Famix.NamedEntity;\r\n\r\n        const fmxInvocation = new Famix.Invocation();\r\n        fmxInvocation.sender = sender;\r\n        fmxInvocation.receiver = receiver;\r\n        fmxInvocation.addCandidate(fmxInvocable);\r\n        fmxInvocation.signature = fmxInvocable.signature;\r\n\r\n        this.famixRep.addElement(fmxInvocation);\r\n\r\n        this.fmxElementObjectMap.set(fmxInvocation,nodeReferringToInvocable);\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix inheritance\r\n     * @param cls A class or an interface (subclass)\r\n     * @param inhClass The inherited class or interface (superclass)\r\n     */\r\n    public createOrGetFamixInheritance(cls: ClassDeclaration | InterfaceDeclaration, inhClass: ClassDeclaration | InterfaceDeclaration | ExpressionWithTypeArguments): void {\r\n        // // need a key to see if the inheritance already exists\r\n        // const classFullyQualifiedName = FQNFunctions.getFQN(cls);\r\n        // let inKeyword: string;\r\n        // let inhClassFullyQualifiedName: string;\r\n        // let inhClassName: string | undefined;\r\n        // // if inhClass is an ExpressionWithTypeArguments, it is an interface\r\n        // if (inhClass instanceof ExpressionWithTypeArguments) {\r\n        //     inhClassName = inhClass.getExpression().getText();\r\n        //     // what is inhClassFullyQualifiedName? TODO\r\n        //     inhClassFullyQualifiedName = 'Undefined_Scope_from_importer.' + inhClassName;\r\n        // } else {\r\n        //     inhClassName = inhClass.getName();\r\n        //     if (!inhClassName) {\r\n        //         throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);\r\n        //     }\r\n        //     inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);\r\n        // }\r\n        // // build the unique key of the inheritance\r\n        // if ((cls instanceof ClassDeclaration && inhClass instanceof ClassDeclaration) \r\n        //     || (cls instanceof InterfaceDeclaration && inhClass instanceof InterfaceDeclaration)) {\r\n        //     inKeyword = \" extends \";\r\n        // } else if (cls instanceof ClassDeclaration && (inhClass instanceof InterfaceDeclaration || inhClass instanceof ExpressionWithTypeArguments)) {\r\n        //     inKeyword = \" implements \";\r\n        // } else {\r\n        //     throw new Error(`Inheritance ${cls.getText()} and ${inhClass.getText()} is not valid.`);\r\n        // }\r\n\r\n        // const inheritanceFullyQualifiedName = FQNFunctions.getFQN(cls) + inKeyword + inhClassFullyQualifiedName;\r\n        // if (this.fmxInheritanceMap.has(inheritanceFullyQualifiedName)) {\r\n        //     const rInheritance = this.fmxInheritanceMap.get(inheritanceFullyQualifiedName);\r\n        //     if (rInheritance) { \r\n        //        return; // don't do anything\r\n        //     } else {\r\n        //         throw new Error(`Inheritance ${cls.getText()} is not found in the inheritance map.`);\r\n        //     }\r\n        // }\r\n\r\n        const fmxInheritance = new Famix.Inheritance();\r\n\r\n        // logger.debug(`createFamixInheritance: classFullyQualifiedName: class fqn = ${classFullyQualifiedName}`);\r\n\r\n        let subClass: Famix.Class | Famix.Interface | undefined;\r\n        if (cls instanceof ClassDeclaration) {\r\n            subClass = this.createOrGetFamixClass(cls);\r\n        }\r\n        else {\r\n            subClass = this.createOrGetFamixInterface(cls);\r\n        }\r\n\r\n        if (!subClass) {\r\n            throw new Error(`Subclass ${cls} not found in Class or Interface maps.`);\r\n        }\r\n\r\n        let superClass: Famix.Class | Famix.Interface | undefined;\r\n        // if (inhClass instanceof ClassDeclaration || inhClass instanceof InterfaceDeclaration) {\r\n        //     inhClassName = inhClass.getName();\r\n        //     if (!inhClassName) {\r\n        //         throw new Error(`Inherited class or interface name not found for ${inhClass.getText()}.`);\r\n        //     }\r\n        //     inhClassFullyQualifiedName = FQNFunctions.getFQN(inhClass);\r\n        //     if (inhClass instanceof ClassDeclaration) {\r\n        //         superClass = this.fmxClassMap.get(inhClassFullyQualifiedName);\r\n        //     }\r\n        //     else {\r\n        //         superClass = this.fmxInterfaceMap.get(inhClassFullyQualifiedName);\r\n        //     }\r\n        //     if (!superClass) {\r\n        //         throw new Error(`Superclass ${classFullyQualifiedName} not found in Class or Interface maps.`);\r\n        //     }\r\n        // }\r\n\r\n        // // it shouldn't add the class/interface again to the Map, it should use createOrGet (?)\r\n        // if (superClass === undefined) {\r\n\r\n        if (inhClass instanceof ClassDeclaration) {\r\n            // superClass = new Famix.Class();\r\n            superClass = this.createOrGetFamixClass(inhClass);\r\n            // this.fmxClassMap.set(inhClassFullyQualifiedName, superClass);\r\n        }\r\n        else if (inhClass instanceof InterfaceDeclaration) {\r\n            // superClass = new Famix.Interface();\r\n            superClass = this.createOrGetFamixInterface(inhClass);\r\n            // this.fmxInterfaceMap.set(inhClassFullyQualifiedName, superClass);\r\n        } else {\r\n            // inhClass instanceof ExpressionWithTypeArguments\r\n            const interfaceDeclaration = getInterfaceDeclarationFromExpression(inhClass);\r\n            if (interfaceDeclaration !== undefined) {\r\n                superClass = this.createOrGetFamixInterface(interfaceDeclaration);\r\n            } else {\r\n                throw new Error(`Interface declaration not found for ${inhClass.getText()}.`);\r\n            }\r\n        }\r\n\r\n        this.fmxElementObjectMap.set(superClass, inhClass);\r\n\r\n        this.makeFamixIndexFileAnchor(inhClass, superClass);\r\n\r\n        this.famixRep.addElement(superClass);\r\n\r\n        fmxInheritance.subclass = subClass;\r\n        fmxInheritance.superclass = superClass;\r\n\r\n        this.famixRep.addElement(fmxInheritance);\r\n        // no FQN for inheritance\r\n\r\n        // We don't map inheritance to the source code element because there are two elements (super, sub)\r\n        // this.fmxElementObjectMap.set(fmxInheritance, null);\r\n\r\n    }\r\n\r\n    public createFamixImportClause(importedEntity: Famix.NamedEntity, importingEntity: Famix.Module) {\r\n        const fmxImportClause = new Famix.ImportClause();\r\n        fmxImportClause.importedEntity = importedEntity;\r\n        fmxImportClause.importingEntity = importingEntity;\r\n        importingEntity.addOutgoingImport(fmxImportClause);\r\n        this.famixRep.addElement(fmxImportClause);\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix import clause\r\n     * @param importClauseInfo The information needed to create a Famix import clause\r\n     * @param importDeclaration The import declaration\r\n     * @param importer A source file which is a module\r\n     * @param moduleSpecifierFilePath The path of the module where the export declaration is\r\n     * @param importElement The imported entity\r\n     * @param isInExports A boolean indicating if the imported entity is in the exports\r\n     * @param isDefaultExport A boolean indicating if the imported entity is a default export\r\n     */\r\n    public oldCreateOrGetFamixImportClause(importClauseInfo: {importDeclaration?: ImportDeclaration | ImportEqualsDeclaration, importerSourceFile: SourceFile, moduleSpecifierFilePath: string, importElement: ImportSpecifier | Identifier, isInExports: boolean, isDefaultExport: boolean}): void {\r\n        const {importDeclaration, importerSourceFile: importer, moduleSpecifierFilePath, importElement, isInExports, isDefaultExport} = importClauseInfo;\r\n        if (importDeclaration && this.fmxImportClauseMap.has(importDeclaration)) {\r\n            const rImportClause = this.fmxImportClauseMap.get(importDeclaration);\r\n            if (rImportClause) { \r\n               return; // don't do anything\r\n            } else {\r\n                throw new Error(`Import clause ${importElement.getText()} is not found in the import clause map.`);\r\n            }\r\n        }\r\n\r\n        logger.info(`creating a new FamixImportClause for ${importDeclaration?.getText()} in ${importer.getBaseName()}.`);\r\n        const fmxImportClause = new Famix.ImportClause();\r\n\r\n        let importedEntity: Famix.NamedEntity | Famix.StructuralEntity | undefined = undefined;\r\n        let importedEntityName: string;\r\n\r\n        const absolutePathProject = this.famixRep.getAbsolutePath();\r\n        \r\n        const absolutePath = path.normalize(moduleSpecifierFilePath);\r\n        // convert the path and remove any windows backslashes introduced by path.normalize\r\n        logger.debug(`createFamixImportClause: absolutePath: ${absolutePath}`);\r\n        logger.debug(`createFamixImportClause: convertToRelativePath: ${this.convertToRelativePath(absolutePath, absolutePathProject)}`);\r\n        const pathInProject: string = this.convertToRelativePath(absolutePath, absolutePathProject).replace(/\\\\/g, \"/\");\r\n        logger.debug(`createFamixImportClause: pathInProject: ${pathInProject}`);\r\n        let pathName = \"{\" + pathInProject + \"}.\";\r\n        logger.debug(`createFamixImportClause: pathName: ${pathName}`);\r\n\r\n        // Named imports, e.g. import { ClassW } from \"./complexExportModule\";\r\n\r\n        // Start with simple import clause (without referring to the actual variable)\r\n\r\n        if (importDeclaration instanceof ImportDeclaration \r\n            && importElement instanceof ImportSpecifier) { \r\n                importedEntityName = importElement.getName();\r\n            pathName = pathName + importedEntityName;\r\n            if (isInExports) {\r\n                importedEntity = this.famixRep.getFamixEntityByFullyQualifiedName(pathName) as Famix.NamedEntity;\r\n            }\r\n            if (importedEntity === undefined) {\r\n                importedEntity = new Famix.NamedEntity();\r\n                importedEntity.name = importedEntityName;\r\n                if (!isInExports) {\r\n                    importedEntity.isStub = true;\r\n                }\r\n                // logger.debug(`createFamixImportClause: Creating named entity ${importedEntityName} with FQN ${pathName}`);\r\n                // importedEntity.fullyQualifiedName = pathName;\r\n                initFQN(importElement, importedEntity);\r\n\r\n                this.makeFamixIndexFileAnchor(importElement, importedEntity);\r\n                // must add entity to repository\r\n                this.famixRep.addElement(importedEntity);\r\n            }\r\n        }\r\n        // handle import equals declarations, e.g. import myModule = require(\"./complexExportModule\");\r\n        // TypeScript can't determine the type of the imported module, so we create a Module entity\r\n        else if (importDeclaration instanceof ImportEqualsDeclaration) {\r\n            importedEntityName = importDeclaration?.getName();\r\n            pathName = pathName + importedEntityName;\r\n            importedEntity = new Famix.StructuralEntity();\r\n            importedEntity.name = importedEntityName;\r\n            initFQN(importDeclaration, importedEntity);\r\n            this.makeFamixIndexFileAnchor(importElement, importedEntity);\r\n            // importedEntity.fullyQualifiedName = pathName;\r\n            const anyType = this.createOrGetFamixType('any', importDeclaration);\r\n            (importedEntity as Famix.StructuralEntity).declaredType = anyType;\r\n        } else {  // default imports, e.g. import ClassW from \"./complexExportModule\";  \r\n            importedEntityName = importElement.getText();\r\n            pathName = pathName + (isDefaultExport ? \"defaultExport\" : \"namespaceExport\");\r\n            importedEntity = new Famix.NamedEntity();\r\n            importedEntity.name = importedEntityName;\r\n            // importedEntity.fullyQualifiedName = pathName;\r\n            initFQN(importElement, importedEntity);\r\n            this.makeFamixIndexFileAnchor(importElement, importedEntity);\r\n        }\r\n        // I don't think it should be added to the repository if it exists already\r\n        if (!isInExports) this.famixRep.addElement(importedEntity);\r\n        const importerFullyQualifiedName = FQNFunctions.getFQN(importer);\r\n        const fmxImporter = this.famixRep.getFamixEntityByFullyQualifiedName(importerFullyQualifiedName) as Famix.Module;\r\n        fmxImportClause.importingEntity = fmxImporter;\r\n        fmxImportClause.importedEntity = importedEntity;\r\n        if (importDeclaration instanceof ImportEqualsDeclaration) {\r\n            fmxImportClause.moduleSpecifier = importDeclaration?.getModuleReference().getText() as string;\r\n        } else {\r\n            fmxImportClause.moduleSpecifier = importDeclaration?.getModuleSpecifierValue() as string;\r\n        }\r\n    \r\n        logger.debug(`createFamixImportClause: ${fmxImportClause.importedEntity?.name} (of type ${\r\n            Helpers.getSubTypeName(fmxImportClause.importedEntity)}) is imported by ${fmxImportClause.importingEntity?.name}`);\r\n\r\n        fmxImporter.addOutgoingImport(fmxImportClause);\r\n\r\n        this.famixRep.addElement(fmxImportClause);\r\n\r\n        if (importDeclaration) {\r\n            this.fmxElementObjectMap.set(fmxImportClause, importDeclaration);\r\n            this.fmxImportClauseMap.set(importDeclaration, fmxImportClause);\r\n        }\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix Arrow Function\r\n     * @param arrowExpression An Expression\r\n     * @returns The Famix model of the variable\r\n     */\r\n    public createOrGetFamixArrowFunction(arrowExpression: Expression, currentCC: { [key: string]: number } ): Famix.ArrowFunction | Famix.ParametricArrowFunction {\r\n        \r\n        let fmxArrowFunction: Famix.ArrowFunction | Famix.ParametricArrowFunction;\r\n        const functionFullyQualifiedName = FQNFunctions.getFQN(arrowExpression);\r\n\r\n        if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {\r\n\r\n            const arrowFunction = arrowExpression.asKindOrThrow(SyntaxKind.ArrowFunction);\r\n\r\n            const isGeneric = arrowFunction.getTypeParameters().length > 0;\r\n\r\n            if (isGeneric) {\r\n                fmxArrowFunction = new Famix.ParametricArrowFunction();\r\n            }\r\n            else {\r\n                fmxArrowFunction = new Famix.ArrowFunction();\r\n            }\r\n\r\n            // Get the parent of the arrow function (the variable declaration)\r\n            const parent = arrowFunction.getParentIfKind(SyntaxKind.VariableDeclaration);\r\n            let functionName = '(NO_NAME)';\r\n\r\n            if (parent && parent instanceof VariableDeclaration) {\r\n                // Get the name of the variable\r\n                functionName = parent.getName();\r\n            }\r\n\r\n            if (functionName) {\r\n                fmxArrowFunction.name = functionName;\r\n            }\r\n            else {\r\n                fmxArrowFunction.name = \"anonymous\";\r\n            }\r\n\r\n            // Signature of an arrow function is (parameters) => return_type\r\n            const parametersSignature = arrowFunction.getParameters().map(p => p.getText()).join(\", \");\r\n            const returnTypeSignature = arrowFunction.getReturnType().getText();\r\n            fmxArrowFunction.signature = `(${parametersSignature}) => ${returnTypeSignature}`;\r\n            fmxArrowFunction.cyclomaticComplexity = currentCC[fmxArrowFunction.name];\r\n\r\n            let functionTypeName = this.UNKNOWN_VALUE;\r\n            try {\r\n                functionTypeName = arrowFunction.getReturnType().getText().trim();\r\n            } catch (error) {\r\n                logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${functionName}. Continuing...`);\r\n            }\r\n\r\n            const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction as unknown as FunctionDeclaration);\r\n            fmxArrowFunction.declaredType = fmxType;\r\n            fmxArrowFunction.numberOfLinesOfCode = arrowFunction.getEndLineNumber() - arrowFunction.getStartLineNumber();\r\n            const parameters = arrowFunction.getParameters();\r\n            fmxArrowFunction.numberOfParameters = parameters.length;\r\n            fmxArrowFunction.numberOfStatements = arrowFunction.getStatements().length;\r\n            initFQN(arrowExpression as unknown as TSMorphObjectType, fmxArrowFunction);\r\n            this.makeFamixIndexFileAnchor(arrowExpression as unknown as TSMorphObjectType, fmxArrowFunction);\r\n            this.famixRep.addElement(fmxArrowFunction);\r\n            this.fmxElementObjectMap.set(fmxArrowFunction,arrowFunction as unknown as TSMorphObjectType);\r\n            this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxArrowFunction);\r\n        } else {\r\n            fmxArrowFunction = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName) as Famix.ArrowFunction;\r\n        }\r\n\r\n        return fmxArrowFunction;\r\n    }\r\n\r\n    /**\r\n     * Creates a Famix concretisation\r\n     * @param cls A class\r\n     * @returns The Famix model of the concretisation\r\n     */\r\n    // public createFamixConcretisation(conEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod ,genEntity : Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod): Famix.Concretisation {\r\n        \r\n    //     const fmxConcretisation : Famix.Concretisation = new Famix.Concretisation();              \r\n        \r\n    //     fmxConcretisation.concreteEntity = conEntity;\r\n    //     fmxConcretisation.genericEntity = genEntity;\r\n    //     // this.fmxElementObjectMap.set(fmxConcretisation,null);\r\n    //     this.famixRep.addElement(fmxConcretisation);    \r\n    //     const parameterConcretisation = this.createFamixParameterConcretisation(fmxConcretisation);\r\n            \r\n    //     return fmxConcretisation;\r\n    // }\r\n\r\n    /**\r\n     * Creates a Famix concretisation\r\n     * @param concretisation A FamixConcretisation\r\n     * @returns The Famix model of the ParameterConcrestisation\r\n     */\r\n    // public createFamixParameterConcretisation(concretisation: Famix.Concretisation): Famix.ParameterConcretisation | undefined{\r\n    //     const conClass = concretisation.concreteEntity;\r\n    //     const genClass = concretisation.genericEntity;\r\n    //     logger.debug(`Creating parameter concretisation between ${conClass.fullyQualifiedName} and ${genClass.fullyQualifiedName}`);\r\n    //     const parameterConcretisations = this.famixRep._getAllEntitiesWithType(\"ParameterConcretisation\") as Set<Famix.ParameterConcretisation>;\r\n    //     const concreteParameters = conClass.concreteParameters;\r\n    //     const genericParameters = genClass.genericParameters;\r\n        \r\n    //     let conClassTypeParametersIterator = concreteParameters.values();\r\n    //     let genClassTypeParametersIterator = genericParameters.values();\r\n    //     let fmxParameterConcretisation : Famix.ParameterConcretisation | undefined = undefined;\r\n\r\n    //     for (let i = 0; i < genericParameters.size; i++) {\r\n    //         const conClassTypeParameter = conClassTypeParametersIterator.next().value as Famix.ParameterType;\r\n    //         const genClassTypeParameter = genClassTypeParametersIterator.next().value as Famix.ParameterType;\r\n    //         let createParameterConcretisation : boolean = true;\r\n    //         if(conClassTypeParameter && genClassTypeParameter && conClassTypeParameter.name != genClassTypeParameter.name){\r\n    //             parameterConcretisations.forEach((param : Famix.ParameterConcretisation) => {\r\n    //                 if (conClassTypeParameter.name == param.concreteParameter.name && genClassTypeParameter.name == param.genericParameter.name) {\r\n    //                     createParameterConcretisation = false;\r\n    //                     fmxParameterConcretisation = param;\r\n    //                 }\r\n    //             })\r\n    //             if (createParameterConcretisation) {\r\n    //                 fmxParameterConcretisation = new Famix.ParameterConcretisation();\r\n    //                 fmxParameterConcretisation.genericParameter = genClassTypeParameter;\r\n    //                 fmxParameterConcretisation.concreteParameter = conClassTypeParameter;\r\n    //                 fmxParameterConcretisation.addConcretisation(concretisation);\r\n    //                 // this.fmxElementObjectMap.set(fmxParameterConcretisation,null);\r\n    //             } else {\r\n    //                 if (!fmxParameterConcretisation) {\r\n    //                     throw new Error(`fmxParameterConcretisation was undefined for concretisation with generic parameter ${genClassTypeParameter.name} and concrete parameter ${conClassTypeParameter.name}`);\r\n    //                 }\r\n    //                 fmxParameterConcretisation.addConcretisation(concretisation);\r\n    //             }\r\n    //             this.famixRep.addElement(fmxParameterConcretisation);\r\n    //         }\r\n    //     }\r\n    //     if (!fmxParameterConcretisation) {\r\n    //         logger.error(`fmxParameterConcretisation was undefined for concretisation with concrete entity ${conClass.fullyQualifiedName} and generic entity ${genClass.fullyQualifiedName}`);\r\n    //     }\r\n    //     return fmxParameterConcretisation;\r\n\r\n    // }\r\n\r\n    /**\r\n     * Creates a Famix concretisation between two classes or two interfaces\r\n     * @param element A class or an Interface\r\n     */\r\n    // public createFamixConcretisationClassOrInterfaceSpecialisation(element: ClassDeclaration | InterfaceDeclaration){\r\n        \r\n    //     const superEntity = element.getExtends();\r\n    //     let superEntityArray;\r\n    //     if (superEntity){\r\n    //         superEntityArray = Array.isArray(superEntity) ? superEntity : [superEntity];\r\n    //     }\r\n    //     if (superEntityArray && superEntityArray.length > 0) {\r\n    //         superEntityArray.forEach(entity => {\r\n    //             let entityIsGeneric;\r\n    //             const superEntitySymbol = entity.getExpression().getSymbolOrThrow();\r\n    //             let superEntityDeclaration;\r\n    //             if (superEntity instanceof ExpressionWithTypeArguments) {\r\n    //                 superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.ClassDeclaration);\r\n    //             } else {\r\n    //                 superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts.SyntaxKind.InterfaceDeclaration);\r\n    //             }\r\n    //             if (superEntityDeclaration) {\r\n    //                 entityIsGeneric = superEntityDeclaration.getTypeParameters().length > 0;\r\n    //             }\r\n    //             if (entityIsGeneric) {\r\n    //                 let EntityDeclaration;\r\n    //                 let genEntity;\r\n    //                 if (superEntity instanceof ExpressionWithTypeArguments) {\r\n    //                     EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as ClassDeclaration;\r\n    //                     genEntity = this.createOrGetFamixClass(EntityDeclaration) as Famix.ParametricClass;\r\n    //                 } else {\r\n    //                     EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;\r\n    //                     genEntity = this.createOrGetFamixInterface(EntityDeclaration) as Famix.ParametricInterface;\r\n    //                 }\r\n    //                 const genParams = EntityDeclaration.getTypeParameters().map((param) => param.getText());\r\n    //                 const args = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments()\r\n    //                 const conParams = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());\r\n    //                 if (!Helpers.arraysAreEqual(conParams,genParams)) {\r\n    //                     let conEntity;\r\n    //                     conEntity = this.createOrGetFamixConcreteElement(genEntity,EntityDeclaration,args);\r\n    //                     const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\r\n    //                     let createConcretisation : boolean = true;\r\n    //                     concretisations.forEach((conc : Famix.Concretisation) => {\r\n    //                         if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){\r\n    //                             createConcretisation = false;\r\n    //                         }\r\n    //                     });\r\n            \r\n    //                     if (createConcretisation) {\r\n    //                         const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);\r\n    //                     }\r\n    //                 }\r\n    //             }\r\n    //         });\r\n    //     }\r\n    //     // TODO: This function seems unfinished\r\n    // }    \r\n    \r\n\r\n    /**\r\n     * Creates a Famix concretisation between a class and its instanciations\r\n     * @param cls A class\r\n     */\r\n    // public createFamixConcretisationGenericInstantiation(cls: ClassDeclaration){\r\n       \r\n    //     const isGeneric = cls.getTypeParameters().length > 0;\r\n    //     if (isGeneric) {\r\n    //         const instances = cls.getSourceFile().getDescendantsOfKind(ts.SyntaxKind.NewExpression)\r\n    //             .filter(newExpr => {\r\n    //                 const expression = newExpr.getExpression();\r\n    //                 return expression.getText() === cls.getName();\r\n    //         });\r\n\r\n    //         instances.forEach(instance => {\r\n    //             const instanceIsGeneric = instance.getTypeArguments().length > 0;\r\n    //             if (instanceIsGeneric) {\r\n    //                 const conParams = instance.getTypeArguments().map((param) => param.getText());\r\n    //                 const genEntity = this.createOrGetFamixClass(cls) as Famix.ParametricClass;\r\n    //                 const genParams = cls.getTypeParameters().map((param) => param.getText());\r\n    //                 if (!Helpers.arraysAreEqual(conParams,genParams)) {\r\n    //                     let conEntity;\r\n    //                     conEntity = this.createOrGetFamixConcreteElement(genEntity,cls,instance.getTypeArguments());\r\n    //                     const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\r\n    //                     let createConcretisation : boolean = true;\r\n    //                     concretisations.forEach((conc : Famix.Concretisation) => {\r\n    //                         if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName){\r\n    //                             createConcretisation = false;\r\n    //                         }\r\n    //                     });\r\n            \r\n    //                     if (createConcretisation) {\r\n    //                         const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conEntity,genEntity);\r\n    //                     }\r\n    //                 }\r\n    //             }\r\n    //         })\r\n    //     }\r\n    //     // TODO: This function seems unfinished\r\n    // }\r\n\r\n    /**\r\n     * Creates a Famix concretisation between a class and its instanciations\r\n     * @param func A function\r\n     */\r\n    // public createFamixConcretisationFunctionInstantiation(element: FunctionDeclaration | MethodDeclaration){\r\n    //     const isGeneric = element.getTypeParameters().length > 0;\r\n    //     if (isGeneric) {\r\n    //         const genParams = element.getTypeParameters().map(param => param.getText());\r\n    //         const uses = element.findReferencesAsNodes();    \r\n    //         uses.forEach(usage => {\r\n    //             let currentNode: Node | undefined = usage;\r\n\r\n    //             while (currentNode) {\r\n    //                 if (currentNode.getKind() === SyntaxKind.CallExpression) {\r\n    //                     const callExpression = currentNode.asKind(SyntaxKind.CallExpression);\r\n    //                     if (!callExpression) {\r\n    //                         throw new Error(`CallExpression not found for ${currentNode.getText()}`);\r\n    //                     }\r\n    //                     const instanceIsGeneric = callExpression.getTypeArguments().length > 0;\r\n    //                     if (instanceIsGeneric) {\r\n    //                         const args = callExpression.getTypeArguments();\r\n    //                         const conParams = callExpression.getTypeArguments().map(param => param.getText());\r\n    //                         if (!Helpers.arraysAreEqual(conParams,genParams)) {\r\n    //                             let genElement;\r\n    //                             if(element instanceof FunctionDeclaration){\r\n    //                                 genElement = this.createOrGetFamixFunction(element, {}) as Famix.ParametricFunction;\r\n    //                             } else {\r\n    //                                 genElement = this.createOrGetFamixMethod(element, {}) as Famix.ParametricMethod;\r\n    //                             }\r\n    //                             let concElement;\r\n    //                             concElement = this.createOrGetFamixConcreteElement(genElement,element,args);\r\n    //                             const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\r\n    //                             let createConcretisation : boolean = true;\r\n    //                             concretisations.forEach((conc : Famix.Concretisation) => {\r\n    //                                 if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){\r\n    //                                     createConcretisation = false;\r\n    //                                 }\r\n    //                             });\r\n        \r\n    //                             if (createConcretisation) {\r\n    //                                 const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);\r\n    //                             }\r\n    //                         }\r\n    //                     }\r\n    //                     break;\r\n    //                 }\r\n    //                 // Remonter à l'élément parent (utile si le nœud de référence est un enfant)\r\n    //                 currentNode = currentNode.getParent();\r\n    //             }\r\n    //         });\r\n    //     }\r\n    // }\r\n\r\n    /**\r\n     * Creates a Famix concretisation between a class and an interface\r\n     * @param cls A class\r\n     */\r\n    // public createFamixConcretisationInterfaceClass(cls: ClassDeclaration){\r\n    \r\n    //     const superInterfaces = cls.getImplements();\r\n    //     superInterfaces.forEach(interfaceType => {\r\n    //         const interfaceIsGeneric = interfaceType.getTypeArguments().length>0;\r\n    //         if (interfaceIsGeneric) {\r\n    //             const interfaceDeclaration = interfaceType.getExpression().getSymbol()?.getDeclarations()[0] as InterfaceDeclaration;\r\n    //             const genParams = interfaceDeclaration.getTypeParameters().map((param) => param.getText());\r\n    //             const conParams = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());\r\n    //             const args = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();\r\n    //             if (!Helpers.arraysAreEqual(conParams,genParams)) {\r\n    //                 const genInterface = this.createOrGetFamixInterface(interfaceDeclaration) as Famix.ParametricInterface;\r\n    //                 const conInterface = this.createOrGetFamixConcreteElement(genInterface,interfaceDeclaration,args);\r\n    //                 const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\r\n    //                 let createConcretisation : boolean = true;\r\n    //                 concretisations.forEach((conc : Famix.Concretisation) => {\r\n    //                     if (genInterface.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conInterface.fullyQualifiedName){\r\n    //                         createConcretisation = false;\r\n    //                     }\r\n    //                 });\r\n            \r\n    //                 if (createConcretisation) {\r\n    //                     const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(conInterface,genInterface);\r\n    //                 }\r\n    //             }\r\n    //         }\r\n    //     });\r\n    // }\r\n\r\n    /**\r\n     * Creates a Famix concretisation between an interface and a Type\r\n     * @param element A variable or a function\r\n     * @param inter An interface\r\n     */\r\n    // public createFamixConcretisationTypeInstanciation(element: InterfaceDeclaration | ClassDeclaration){\r\n\r\n    //     const isGeneric = element.getTypeParameters().length > 0;\r\n    //     if (isGeneric) {\r\n    //         const genParams = element.getTypeParameters().map(param => param.getText());\r\n    //         const uses = element.findReferencesAsNodes();\r\n    //         uses.forEach(use => {        \r\n    //             let parentNode = use.getParent();\r\n    //             while (parentNode) {\r\n    //                 if (parentNode.getKind() === SyntaxKind.TypeReference) {\r\n    //                     const typeReferenceNode = parentNode.asKind(SyntaxKind.TypeReference);\r\n    //                     if (!typeReferenceNode) {\r\n    //                         throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);\r\n    //                     }\r\n    //                     const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;\r\n    //                     if (typeReferenceNodeIsGeneric) {}\r\n    //                         const args = typeReferenceNode.getTypeArguments();\r\n    //                         const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());\r\n    //                         if (!Helpers.arraysAreEqual(conParams,genParams)) {\r\n    //                             let genElement;\r\n    //                             if(element instanceof ClassDeclaration){\r\n    //                                 genElement = this.createOrGetFamixClass(element) as Famix.ParametricClass;\r\n    //                             } else {\r\n    //                                 genElement = this.createOrGetFamixInterface(element) as Famix.ParametricInterface;\r\n    //                             }\r\n    //                             let concElement;\r\n    //                             concElement = this.createOrGetFamixConcreteElement(genElement,element,args);\r\n    //                             const concretisations = this.famixRep._getAllEntitiesWithType(\"Concretisation\") as Set<Famix.Concretisation>;\r\n    //                             let createConcretisation : boolean = true;\r\n    //                             concretisations.forEach((conc : Famix.Concretisation) => {\r\n    //                                 if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName){\r\n    //                                     createConcretisation = false;\r\n    //                                 }\r\n    //                             });\r\n        \r\n    //                             if (createConcretisation) {\r\n    //                                 const fmxConcretisation : Famix.Concretisation = this.createFamixConcretisation(concElement,genElement);\r\n    //                             }\r\n    //                         }\r\n    //                     break;\r\n    //                 }\r\n    //                 parentNode = parentNode.getParent();\r\n    //             }\r\n    //         });\r\n    //     }\r\n    // }\r\n\r\n    public convertToRelativePath(absolutePath: string, absolutePathProject: string) {\r\n        logger.debug(`convertToRelativePath: absolutePath: '${absolutePath}', absolutePathProject: '${absolutePathProject}'`);\r\n        if (absolutePath.startsWith(absolutePathProject)) {\r\n            return absolutePath.replace(absolutePathProject, \"\").slice(1);\r\n        } else if (absolutePath.startsWith(\"/\")) {\r\n            return absolutePath.slice(1);\r\n        } else {\r\n            return absolutePath;\r\n        }\r\n    }\r\n}\r\n\r\nexport function isPrimitiveType(typeName: string) {\r\n    return typeName === \"number\" ||\r\n        typeName === \"string\" ||\r\n        typeName === \"boolean\" ||\r\n        typeName === \"bigint\" ||\r\n        typeName === \"symbol\" ||\r\n        typeName === \"undefined\" ||\r\n        typeName === \"null\" ||\r\n        typeName === \"any\" ||\r\n        typeName === \"unknown\" ||\r\n        typeName === \"never\" ||\r\n        typeName === \"void\";\r\n}\r\n\r\nfunction initFQN(sourceElement: TSMorphObjectType, famixElement: Famix.SourcedEntity) {\r\n    // handle special cases where an element is a Type -- need to change its name\r\n    if (famixElement instanceof Famix.Type && !(sourceElement instanceof CommentRange) && isTypeContext(sourceElement)) {\r\n        let fqn = FQNFunctions.getFQN(sourceElement);\r\n        // using regex, replace [blah] with [blahType]\r\n        fqn = fqn.replace(/\\[([^\\]]+)\\]/g, \"[$1Type]\");\r\n        logger.debug(\"Setting fully qualified name for \" + famixElement.getJSON() + \" to \" + fqn);\r\n        famixElement.fullyQualifiedName = fqn;\r\n        return;\r\n    }\r\n    // catch all (except comments)\r\n    if (!(sourceElement instanceof CommentRange)) {\r\n        const fqn = FQNFunctions.getFQN(sourceElement);\r\n        logger.debug(\"Setting fully qualified name for \" + famixElement.getJSON() + \" to \" + fqn);\r\n        (famixElement as Famix.NamedEntity).fullyQualifiedName = fqn;\r\n    } \r\n}\r\n\r\n\r\nfunction isTypeContext(sourceElement: ImportDeclaration | ImportEqualsDeclaration | SourceFile | ModuleDeclaration | ConstructorDeclaration | MethodSignature | FunctionExpression | ParameterDeclaration | VariableDeclaration | PropertyDeclaration | PropertySignature | TypeParameterDeclaration | Identifier | Decorator | GetAccessorDeclaration | SetAccessorDeclaration | ImportSpecifier | EnumDeclaration | EnumMember | TypeAliasDeclaration | ExpressionWithTypeArguments | TSMorphParametricType) {\r\n    return sourceElement instanceof ConstructorDeclaration\r\n        || sourceElement instanceof MethodDeclaration  \r\n        || sourceElement instanceof FunctionDeclaration \r\n        || sourceElement instanceof FunctionExpression \r\n        || sourceElement instanceof ArrowFunction\r\n        ||\r\n        sourceElement instanceof ParameterDeclaration ||\r\n        sourceElement instanceof VariableDeclaration ||\r\n        sourceElement instanceof PropertyDeclaration ||\r\n        sourceElement instanceof PropertySignature ||\r\n        sourceElement instanceof TypeParameterDeclaration ||\r\n        sourceElement instanceof Identifier ||\r\n        sourceElement instanceof Decorator ||\r\n        sourceElement instanceof GetAccessorDeclaration ||\r\n        sourceElement instanceof SetAccessorDeclaration ||\r\n        sourceElement instanceof ImportSpecifier ||\r\n        sourceElement instanceof EnumDeclaration ||\r\n        sourceElement instanceof EnumMember ||\r\n        sourceElement instanceof TypeAliasDeclaration\r\n        || sourceElement instanceof ImportDeclaration\r\n        || sourceElement instanceof ExpressionWithTypeArguments;\r\n}\r\n\r\n\r\nfunction getInterfaceDeclarationFromExpression(expression: ExpressionWithTypeArguments): InterfaceDeclaration | undefined {\r\n    // Step 1: Get the type of the expression\r\n    const type = expression.getType();\r\n\r\n    // Step 2: Get the symbol associated with the type\r\n    let symbol = type.getSymbol();\r\n\r\n    if (!symbol) {\r\n        // If symbol is not found, try to get the symbol from the identifier\r\n        const identifier = expression.getFirstDescendantByKind(SyntaxKind.Identifier);\r\n        if (!identifier) {\r\n            throw new Error(`Identifier not found for ${expression.getText()}.`);\r\n        }\r\n        symbol = identifier.getSymbol();\r\n        if (!symbol) {\r\n            throw new Error(`Symbol not found for ${identifier.getText()}.`);\r\n        }\r\n    }\r\n\r\n    // Step 3: Resolve the symbol to find the actual declaration\r\n    const interfaceDeclaration = resolveSymbolToInterfaceDeclaration(symbol);\r\n\r\n    if (!interfaceDeclaration) {\r\n        logger.error(`Interface declaration not found for ${expression.getText()}.`);\r\n    }\r\n\r\n    return interfaceDeclaration;\r\n}\r\n\r\nimport { Symbol as TSMorphSymbol } from \"ts-morph\";\r\n\r\nfunction resolveSymbolToInterfaceDeclaration(symbol: TSMorphSymbol): InterfaceDeclaration | undefined {\r\n    // Get the declarations associated with the symbol\r\n    const declarations = symbol.getDeclarations();\r\n\r\n    // Filter for InterfaceDeclaration\r\n    const interfaceDeclaration = declarations.find(declaration => declaration instanceof InterfaceDeclaration) as InterfaceDeclaration | undefined;\r\n\r\n    if (interfaceDeclaration) {\r\n        return interfaceDeclaration;\r\n    }\r\n\r\n    // Handle imports: If the symbol is imported, resolve the import to find the actual declaration\r\n    for (const declaration of declarations) {\r\n        if (declaration.getKind() === SyntaxKind.ImportSpecifier) {\r\n            const importSpecifier = declaration as ImportSpecifier;\r\n            const importDeclaration = importSpecifier.getImportDeclaration();\r\n            const moduleSpecifier = importDeclaration.getModuleSpecifierSourceFile();\r\n\r\n            if (moduleSpecifier) {\r\n                const exportedSymbols = moduleSpecifier.getExportSymbols();\r\n                const exportedSymbol = exportedSymbols.find(symbol => symbol.getName() === importSpecifier.getName());\r\n                if (exportedSymbol) {\r\n                    return resolveSymbolToInterfaceDeclaration(exportedSymbol);\r\n                }\r\n            }\r\n        }\r\n    }\r\n\r\n    return undefined;\r\n}\r\n\r\n// function oldGetInterfaceDeclarationFromExpression(expression: ExpressionWithTypeArguments): InterfaceDeclaration | undefined {\r\n//     // Two cases:\r\n//     // class A implements ImportedInterface, DeclaredInterface {}\r\n//     const type = expression.getType();\r\n\r\n//     // ImportedInterface: type will a symbol \r\n//     let symbol = type.getAliasSymbol();  // will be defined for imported interfaces\r\n\r\n//     if (!symbol) {\r\n//         // DeclaredInterface: type will be an InterfaceDeclaration on Identifier node that is the child of the ExpressionWithTypeArguments\r\n//         const identifier = expression.getFirstDescendantByKind(SyntaxKind.Identifier);\r\n//         if (!identifier) {\r\n//             throw new Error(`Identifier not found for ${expression.getText()}.`);\r\n//         }\r\n//         symbol = identifier.getSymbol();\r\n//         if (!symbol) {\r\n//             throw new Error(`Symbol not found for ${identifier.getText()}.`);\r\n//         }\r\n//     }\r\n\r\n//     // Step 3: Get the declarations associated with the symbol\r\n//     const declarations = symbol.getDeclarations();\r\n\r\n//     // Step 4: Filter for InterfaceDeclaration\r\n//     const interfaceDeclaration = declarations.find(declaration => declaration instanceof InterfaceDeclaration) as InterfaceDeclaration | undefined;\r\n\r\n//     if (!interfaceDeclaration) {\r\n//         throw new Error(`Interface declaration not found for ${expression.getText()}.`);\r\n//     }\r\n\r\n//     return interfaceDeclaration;\r\n// }\r\n"]}