ts2famix 2.1.0-beta.2 → 3.1.0

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 (192) hide show
  1. package/.eslintrc.json +24 -24
  2. package/LICENSE +24 -24
  3. package/README.md +78 -78
  4. package/dist/analyze.js +3 -3
  5. package/dist/analyze_functions/process_functions.js +92 -62
  6. package/dist/famix_functions/EntityDictionary.js +674 -599
  7. package/dist/famix_functions/helpers_creation.js +18 -12
  8. package/dist/fqn.js +351 -18
  9. package/dist/lib/famix/famix_JSON_exporter.js +1 -1
  10. package/dist/lib/famix/famix_base_element.js +1 -1
  11. package/dist/lib/famix/famix_repository.js +14 -5
  12. package/dist/lib/famix/index.js +1 -1
  13. package/dist/lib/famix/model/famix/access.js +1 -1
  14. package/dist/lib/famix/model/famix/accessor.js +1 -1
  15. package/dist/lib/famix/model/famix/alias.js +1 -1
  16. package/dist/lib/famix/model/famix/arrow_function.js +1 -1
  17. package/dist/lib/famix/model/famix/behavioral_entity.js +1 -1
  18. package/dist/lib/famix/model/famix/class.js +1 -1
  19. package/dist/lib/famix/model/famix/comment.js +1 -1
  20. package/dist/lib/famix/model/famix/concretisation.js +1 -1
  21. package/dist/lib/famix/model/famix/container_entity.js +1 -1
  22. package/dist/lib/famix/model/famix/decorator.js +1 -1
  23. package/dist/lib/famix/model/famix/entity.js +1 -1
  24. package/dist/lib/famix/model/famix/enum.js +1 -1
  25. package/dist/lib/famix/model/famix/enum_value.js +1 -1
  26. package/dist/lib/famix/model/famix/function.js +1 -1
  27. package/dist/lib/famix/model/famix/import_clause.js +1 -1
  28. package/dist/lib/famix/model/famix/index.js +1 -1
  29. package/dist/lib/famix/model/famix/indexed_file_anchor.js +1 -1
  30. package/dist/lib/famix/model/famix/inheritance.js +1 -1
  31. package/dist/lib/famix/model/famix/interface.js +1 -1
  32. package/dist/lib/famix/model/famix/invocation.js +1 -1
  33. package/dist/lib/famix/model/famix/method.js +1 -1
  34. package/dist/lib/famix/model/famix/module.js +2 -2
  35. package/dist/lib/famix/model/famix/named_entity.js +1 -1
  36. package/dist/lib/famix/model/famix/parameter.js +1 -1
  37. package/dist/lib/famix/model/famix/parameter_concretisation.js +1 -1
  38. package/dist/lib/famix/model/famix/parameter_type.js +1 -1
  39. package/dist/lib/famix/model/famix/parametric_arrow_function.js +1 -1
  40. package/dist/lib/famix/model/famix/parametric_class.js +1 -1
  41. package/dist/lib/famix/model/famix/parametric_function.js +1 -1
  42. package/dist/lib/famix/model/famix/parametric_interface.js +1 -1
  43. package/dist/lib/famix/model/famix/parametric_method.js +1 -1
  44. package/dist/lib/famix/model/famix/primitive_type.js +1 -1
  45. package/dist/lib/famix/model/famix/property.js +1 -1
  46. package/dist/lib/famix/model/famix/reference.js +1 -1
  47. package/dist/lib/famix/model/famix/scoping_entity.js +1 -1
  48. package/dist/lib/famix/model/famix/script_entity.js +1 -1
  49. package/dist/lib/famix/model/famix/source_anchor.js +1 -1
  50. package/dist/lib/famix/model/famix/source_language.js +1 -1
  51. package/dist/lib/famix/model/famix/sourced_entity.js +1 -1
  52. package/dist/lib/famix/model/famix/structural_entity.js +1 -1
  53. package/dist/lib/famix/model/famix/type.js +1 -1
  54. package/dist/lib/famix/model/famix/variable.js +1 -1
  55. package/dist/lib/ts-complex/cyclomatic-service.js +1 -1
  56. package/dist/refactorer/refactor-getter-setter.js +1 -1
  57. package/dist/ts2famix-cli-wrapper.js +1 -1
  58. package/dist/ts2famix-cli.js +3 -1
  59. package/doc-uml/famix-typescript-model.puml +619 -607
  60. package/doc-uml/famix-typescript-model.svg +1 -1
  61. package/eslint.config.mjs +28 -28
  62. package/jest.config.json +1 -1
  63. package/package.json +70 -70
  64. package/src/analyze.ts +120 -120
  65. package/src/analyze_functions/process_functions.ts +1069 -1040
  66. package/src/famix_functions/EntityDictionary.ts +2061 -2016
  67. package/src/famix_functions/helpers_creation.ts +143 -135
  68. package/src/fqn.ts +416 -50
  69. package/src/generate_uml.sh +20 -20
  70. package/src/lib/famix/License.md +22 -22
  71. package/src/lib/famix/famix_JSON_exporter.ts +56 -56
  72. package/src/lib/famix/famix_base_element.ts +22 -22
  73. package/src/lib/famix/famix_repository.ts +288 -278
  74. package/src/lib/famix/index.ts +8 -8
  75. package/src/lib/famix/model/famix/access.ts +50 -50
  76. package/src/lib/famix/model/famix/accessor.ts +15 -15
  77. package/src/lib/famix/model/famix/alias.ts +39 -39
  78. package/src/lib/famix/model/famix/arrow_function.ts +15 -15
  79. package/src/lib/famix/model/famix/behavioral_entity.ts +97 -97
  80. package/src/lib/famix/model/famix/class.ts +85 -85
  81. package/src/lib/famix/model/famix/comment.ts +47 -47
  82. package/src/lib/famix/model/famix/concretisation.ts +40 -40
  83. package/src/lib/famix/model/famix/container_entity.ts +160 -160
  84. package/src/lib/famix/model/famix/decorator.ts +37 -37
  85. package/src/lib/famix/model/famix/entity.ts +15 -15
  86. package/src/lib/famix/model/famix/enum.ts +30 -30
  87. package/src/lib/famix/model/famix/enum_value.ts +28 -28
  88. package/src/lib/famix/model/famix/function.ts +15 -15
  89. package/src/lib/famix/model/famix/import_clause.ts +51 -51
  90. package/src/lib/famix/model/famix/index.ts +41 -41
  91. package/src/lib/famix/model/famix/indexed_file_anchor.ts +46 -46
  92. package/src/lib/famix/model/famix/inheritance.ts +40 -40
  93. package/src/lib/famix/model/famix/interface.ts +75 -75
  94. package/src/lib/famix/model/famix/invocation.ts +65 -65
  95. package/src/lib/famix/model/famix/method.ts +89 -89
  96. package/src/lib/famix/model/famix/module.ts +71 -71
  97. package/src/lib/famix/model/famix/named_entity.ts +95 -95
  98. package/src/lib/famix/model/famix/parameter.ts +28 -28
  99. package/src/lib/famix/model/famix/parameter_concretisation.ts +51 -51
  100. package/src/lib/famix/model/famix/parameter_type.ts +58 -58
  101. package/src/lib/famix/model/famix/parametric_arrow_function.ts +32 -32
  102. package/src/lib/famix/model/famix/parametric_class.ts +49 -49
  103. package/src/lib/famix/model/famix/parametric_function.ts +32 -32
  104. package/src/lib/famix/model/famix/parametric_interface.ts +49 -49
  105. package/src/lib/famix/model/famix/parametric_method.ts +32 -32
  106. package/src/lib/famix/model/famix/primitive_type.ts +15 -15
  107. package/src/lib/famix/model/famix/property.ts +94 -94
  108. package/src/lib/famix/model/famix/reference.ts +40 -40
  109. package/src/lib/famix/model/famix/scoping_entity.ts +35 -35
  110. package/src/lib/famix/model/famix/script_entity.ts +34 -34
  111. package/src/lib/famix/model/famix/source_anchor.ts +30 -30
  112. package/src/lib/famix/model/famix/source_language.ts +35 -35
  113. package/src/lib/famix/model/famix/sourced_entity.ts +70 -70
  114. package/src/lib/famix/model/famix/structural_entity.ts +43 -43
  115. package/src/lib/famix/model/famix/type.ts +87 -87
  116. package/src/lib/famix/model/famix/variable.ts +27 -27
  117. package/src/lib/famix/package.json +28 -28
  118. package/src/lib/ts-complex/cyclomatic-service.ts +83 -83
  119. package/src/refactorer/refactor-getter-setter.ts +140 -140
  120. package/src/ts2famix-cli-wrapper.ts +21 -21
  121. package/src/ts2famix-cli.ts +62 -60
  122. package/tsconfig.check-tests.json +14 -14
  123. package/tsconfig.json +73 -72
  124. package/.vscode/settings.json +0 -4
  125. package/TODO +0 -1
  126. package/arwea-fix.json +0 -1
  127. package/bogus.ts +0 -3
  128. package/class-diagram.puml +0 -792
  129. package/debug.txt +0 -13332
  130. package/debuglog.txt +0 -12073
  131. package/dist/famix2puml.js +0 -126
  132. package/dist/getClasses-arrow-body.js +0 -43
  133. package/dist/lib/famix/src/famix_JSON_exporter.js +0 -55
  134. package/dist/lib/famix/src/famix_base_element.js +0 -18
  135. package/dist/lib/famix/src/famix_repository.js +0 -224
  136. package/dist/lib/famix/src/index.js +0 -31
  137. package/dist/lib/famix/src/model/famix/access.js +0 -40
  138. package/dist/lib/famix/src/model/famix/accessor.js +0 -17
  139. package/dist/lib/famix/src/model/famix/alias.js +0 -33
  140. package/dist/lib/famix/src/model/famix/arrowFunction.js +0 -17
  141. package/dist/lib/famix/src/model/famix/arrow_function.js +0 -17
  142. package/dist/lib/famix/src/model/famix/behavioral_entity.js +0 -79
  143. package/dist/lib/famix/src/model/famix/class.js +0 -71
  144. package/dist/lib/famix/src/model/famix/comment.js +0 -39
  145. package/dist/lib/famix/src/model/famix/concretisation.js +0 -31
  146. package/dist/lib/famix/src/model/famix/container_entity.js +0 -126
  147. package/dist/lib/famix/src/model/famix/decorator.js +0 -32
  148. package/dist/lib/famix/src/model/famix/entity.js +0 -17
  149. package/dist/lib/famix/src/model/famix/enum.js +0 -31
  150. package/dist/lib/famix/src/model/famix/enum_value.js +0 -25
  151. package/dist/lib/famix/src/model/famix/function.js +0 -17
  152. package/dist/lib/famix/src/model/famix/implicit_variable.js +0 -17
  153. package/dist/lib/famix/src/model/famix/import_clause.js +0 -41
  154. package/dist/lib/famix/src/model/famix/index.js +0 -86
  155. package/dist/lib/famix/src/model/famix/indexed_file_anchor.js +0 -38
  156. package/dist/lib/famix/src/model/famix/inheritance.js +0 -33
  157. package/dist/lib/famix/src/model/famix/interface.js +0 -64
  158. package/dist/lib/famix/src/model/famix/invocation.js +0 -54
  159. package/dist/lib/famix/src/model/famix/method.js +0 -67
  160. package/dist/lib/famix/src/model/famix/module.js +0 -60
  161. package/dist/lib/famix/src/model/famix/named_entity.js +0 -78
  162. package/dist/lib/famix/src/model/famix/parameter.js +0 -25
  163. package/dist/lib/famix/src/model/famix/parameterConcretisation.js +0 -44
  164. package/dist/lib/famix/src/model/famix/parameter_concretisation.js +0 -44
  165. package/dist/lib/famix/src/model/famix/parameter_type.js +0 -45
  166. package/dist/lib/famix/src/model/famix/parametricArrowFunction.js +0 -29
  167. package/dist/lib/famix/src/model/famix/parametric_arrow_function.js +0 -31
  168. package/dist/lib/famix/src/model/famix/parametric_class.js +0 -44
  169. package/dist/lib/famix/src/model/famix/parametric_function.js +0 -31
  170. package/dist/lib/famix/src/model/famix/parametric_interface.js +0 -44
  171. package/dist/lib/famix/src/model/famix/parametric_method.js +0 -31
  172. package/dist/lib/famix/src/model/famix/primitive_type.js +0 -17
  173. package/dist/lib/famix/src/model/famix/property.js +0 -73
  174. package/dist/lib/famix/src/model/famix/reference.js +0 -33
  175. package/dist/lib/famix/src/model/famix/scoping_entity.js +0 -36
  176. package/dist/lib/famix/src/model/famix/script_entity.js +0 -29
  177. package/dist/lib/famix/src/model/famix/source_anchor.js +0 -27
  178. package/dist/lib/famix/src/model/famix/source_language.js +0 -35
  179. package/dist/lib/famix/src/model/famix/sourced_entity.js +0 -60
  180. package/dist/lib/famix/src/model/famix/structural_entity.js +0 -39
  181. package/dist/lib/famix/src/model/famix/text_anchor.js +0 -38
  182. package/dist/lib/famix/src/model/famix/type.js +0 -73
  183. package/dist/lib/famix/src/model/famix/variable.js +0 -24
  184. package/fqn-model.json +0 -1
  185. package/iterateGenericTypes.ts +0 -69
  186. package/out/class-diagram/class-diagram.svg +0 -1
  187. package/sample.json +0 -1
  188. package/sample.ts +0 -1
  189. package/stats.txt +0 -3091
  190. package/tabby-debug-output.txt +0 -19433
  191. package/ts2famix.log +0 -22656
  192. package/validate-references.js +0 -103
@@ -43,12 +43,12 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
43
43
  Object.defineProperty(exports, "__esModule", { value: true });
44
44
  exports.EntityDictionary = void 0;
45
45
  exports.isPrimitiveType = isPrimitiveType;
46
+ exports.getPrimitiveTypeName = getPrimitiveTypeName;
46
47
  const ts_morph_1 = require("ts-morph");
47
48
  const process_functions_1 = require("../analyze_functions/process_functions");
48
49
  const Famix = __importStar(require("../lib/famix/model/famix"));
49
50
  const famix_repository_1 = require("../lib/famix/famix_repository");
50
51
  const analyze_1 = require("../analyze");
51
- // import GraphemeSplitter from "grapheme-splitter";
52
52
  // eslint-disable-next-line @typescript-eslint/no-require-imports
53
53
  const GraphemeSplitter = require("grapheme-splitter");
54
54
  const Helpers = __importStar(require("./helpers_creation"));
@@ -117,7 +117,7 @@ class EntityDictionary {
117
117
  }
118
118
  // The +1 is because the source anchor (Pharo) is 1-based, but ts-morph is 0-based
119
119
  sourceAnchor.startPos = sourceStart + 1;
120
- sourceAnchor.endPos = sourceEnd + 1;
120
+ sourceAnchor.endPos = sourceEnd;
121
121
  let fileName = node.getSourceFile().getFilePath();
122
122
  if (fileName.startsWith("/")) {
123
123
  fileName = fileName.substring(1);
@@ -204,7 +204,7 @@ class EntityDictionary {
204
204
  }
205
205
  // note: the +1 is because the source anchor is 1-based, but ts-morph is 0-based
206
206
  fmxIndexFileAnchor.startPos = sourceStart + 1;
207
- fmxIndexFileAnchor.endPos = sourceEnd + 1;
207
+ fmxIndexFileAnchor.endPos = sourceEnd;
208
208
  // 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)) {
209
209
  // initFQN(sourceElement, famixElement);
210
210
  // }
@@ -280,31 +280,31 @@ class EntityDictionary {
280
280
  }
281
281
  /**
282
282
  * Creates a Famix alias
283
- * @param a An alias
283
+ * @param typeAliasDeclaration An alias
284
284
  * @returns The Famix model of the alias
285
285
  */
286
- createFamixAlias(a) {
286
+ createFamixAlias(typeAliasDeclaration) {
287
287
  let fmxAlias;
288
- const aliasName = a.getName();
288
+ const aliasName = typeAliasDeclaration.getName();
289
289
  //const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);
290
- const aliasFullyQualifiedName = FQNFunctions.getFQN(a);
290
+ const aliasFullyQualifiedName = FQNFunctions.getFQN(typeAliasDeclaration);
291
291
  const foundAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);
292
292
  if (!foundAlias) {
293
293
  fmxAlias = new Famix.Alias();
294
- fmxAlias.name = a.getName();
295
- const aliasNameWithGenerics = aliasName + (a.getTypeParameters().length ? ("<" + a.getTypeParameters().map(tp => tp.getName()).join(", ") + ">") : "");
294
+ fmxAlias.name = typeAliasDeclaration.getName();
295
+ const aliasNameWithGenerics = aliasName + (typeAliasDeclaration.getTypeParameters().length ? ("<" + typeAliasDeclaration.getTypeParameters().map(tp => tp.getName()).join(", ") + ">") : "");
296
296
  analyze_1.logger.debug(`> NOTE: alias ${aliasName} has fully qualified name ${aliasFullyQualifiedName} and name with generics ${aliasNameWithGenerics}.`);
297
- const fmxType = this.createOrGetFamixType(aliasNameWithGenerics, a);
297
+ const fmxType = this.createOrGetFamixType(aliasNameWithGenerics, typeAliasDeclaration.getType(), typeAliasDeclaration);
298
298
  fmxAlias.aliasedEntity = fmxType;
299
- initFQN(a, fmxAlias);
300
- this.makeFamixIndexFileAnchor(a, fmxAlias);
299
+ initFQN(typeAliasDeclaration, fmxAlias);
300
+ this.makeFamixIndexFileAnchor(typeAliasDeclaration, fmxAlias);
301
301
  this.fmxAliasMap.set(aliasFullyQualifiedName, fmxAlias);
302
302
  this.famixRep.addElement(fmxAlias);
303
303
  }
304
304
  else {
305
305
  fmxAlias = foundAlias;
306
306
  }
307
- this.fmxElementObjectMap.set(fmxAlias, a);
307
+ this.fmxElementObjectMap.set(fmxAlias, typeAliasDeclaration);
308
308
  return fmxAlias;
309
309
  }
310
310
  /**
@@ -377,51 +377,61 @@ class EntityDictionary {
377
377
  * @param concreteArguments concrete arguments
378
378
  * @returns A parametric Element
379
379
  */
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
- // }
380
+ createOrGetFamixConcreteElement(concreteElement, concreteElementDeclaration, concreteArguments) {
381
+ let fullyQualifiedFilename = concreteElement.fullyQualifiedName;
382
+ let params = "";
383
+ concreteArguments.map((param) => {
384
+ params = params + param.getText() + ',';
385
+ });
386
+ params = params.substring(0, params.length - 1);
387
+ fullyQualifiedFilename = Helpers.replaceLastBetweenTags(fullyQualifiedFilename, params);
388
+ let concElement;
389
+ if (!this.fmxInterfaceMap.has(fullyQualifiedFilename) &&
390
+ !this.fmxClassMap.has(fullyQualifiedFilename) &&
391
+ !this.fmxFunctionAndMethodMap.has(fullyQualifiedFilename)) {
392
+ concElement = lodash_1.default.cloneDeep(concreteElement);
393
+ concElement.fullyQualifiedName = fullyQualifiedFilename;
394
+ concElement.clearGenericParameters();
395
+ concreteArguments.map((param) => {
396
+ if (param instanceof ts_morph_1.TypeParameterDeclaration) {
397
+ const parameter = this.createOrGetFamixType(param.getText(), param.getType(), param);
398
+ concElement.addConcreteParameter(parameter);
399
+ }
400
+ else {
401
+ analyze_1.logger.warn(`> WARNING: concrete argument ${param.getText()} is not a TypeParameterDeclaration. It is a ${param.getKindName()}.`);
402
+ }
403
+ });
404
+ if (concreteElement instanceof Famix.ParametricClass) {
405
+ this.fmxClassMap.set(fullyQualifiedFilename, concElement);
406
+ }
407
+ else if (concreteElement instanceof Famix.ParametricInterface) {
408
+ this.fmxInterfaceMap.set(fullyQualifiedFilename, concElement);
409
+ }
410
+ else if (concreteElement instanceof Famix.ParametricFunction) {
411
+ this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement);
412
+ }
413
+ else { // if (concreteElement instanceof Famix.ParametricMethod) {
414
+ this.fmxFunctionAndMethodMap.set(fullyQualifiedFilename, concElement);
415
+ }
416
+ this.famixRep.addElement(concElement);
417
+ this.fmxElementObjectMap.set(concElement, concreteElementDeclaration);
418
+ }
419
+ else {
420
+ if (concreteElement instanceof Famix.ParametricClass) {
421
+ concElement = this.fmxClassMap.get(fullyQualifiedFilename);
422
+ }
423
+ else if (concreteElement instanceof Famix.ParametricInterface) {
424
+ concElement = this.fmxInterfaceMap.get(fullyQualifiedFilename);
425
+ }
426
+ else if (concreteElement instanceof Famix.ParametricFunction) {
427
+ concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename);
428
+ }
429
+ else { // if (concreteElement instanceof Famix.ParametricMethod) {
430
+ concElement = this.fmxFunctionAndMethodMap.get(fullyQualifiedFilename);
431
+ }
432
+ }
433
+ return concElement;
434
+ }
425
435
  /**
426
436
  * Creates a Famix property
427
437
  * @param property A property
@@ -438,7 +448,7 @@ class EntityDictionary {
438
448
  catch (error) {
439
449
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for property: ${property.getName()}. Continuing...`);
440
450
  }
441
- const fmxType = this.createOrGetFamixType(propTypeName, property);
451
+ const fmxType = this.createOrGetFamixType(propTypeName, property.getType(), property);
442
452
  fmxProperty.declaredType = fmxType;
443
453
  // add the visibility (public, private, etc.) to the fmxProperty
444
454
  fmxProperty.visibility = "";
@@ -485,10 +495,16 @@ class EntityDictionary {
485
495
  * @returns The Famix model of the method or the accessor
486
496
  */
487
497
  createOrGetFamixMethod(method, currentCC) {
488
- let fmxMethod;
489
- const isGeneric = method.getTypeParameters().length > 0;
490
- const functionFullyQualifiedName = FQNFunctions.getFQN(method);
491
- if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {
498
+ // console.log(`\n=== Creating/Getting Method ===`);
499
+ // console.log(`Method kind: ${method.getKindName()}`);
500
+ // console.log(`Method text: ${method.getText().slice(0, 50)}...`);
501
+ const fqn = FQNFunctions.getFQN(method);
502
+ // console.log(`Method FQN: ${fqn}`);
503
+ analyze_1.logger.debug(`Processing method ${fqn}`);
504
+ let fmxMethod = this.fmxFunctionAndMethodMap.get(fqn);
505
+ if (!fmxMethod) {
506
+ // console.log('Method not found in map, creating new');
507
+ const isGeneric = method.getTypeParameters().length > 0;
492
508
  if (method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) {
493
509
  fmxMethod = new Famix.Accessor();
494
510
  const isGetter = method instanceof ts_morph_1.GetAccessorDeclaration;
@@ -499,15 +515,9 @@ class EntityDictionary {
499
515
  if (isSetter) {
500
516
  fmxMethod.kind = "setter";
501
517
  }
502
- this.famixRep.addElement(fmxMethod);
503
518
  }
504
519
  else {
505
- if (isGeneric) {
506
- fmxMethod = new Famix.ParametricMethod();
507
- }
508
- else {
509
- fmxMethod = new Famix.Method();
510
- }
520
+ fmxMethod = isGeneric ? new Famix.ParametricMethod() : new Famix.Method();
511
521
  }
512
522
  const isConstructor = method instanceof ts_morph_1.ConstructorDeclaration;
513
523
  const isSignature = method instanceof ts_morph_1.MethodSignature;
@@ -522,59 +532,41 @@ class EntityDictionary {
522
532
  }
523
533
  fmxMethod.isAbstract = isAbstract;
524
534
  fmxMethod.isClassSide = isStatic;
525
- fmxMethod.isPrivate = (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'private')) !== undefined : false;
526
- fmxMethod.isProtected = (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration) ? (method.getModifiers().find(x => x.getText() === 'protected')) !== undefined : false;
535
+ fmxMethod.isPrivate = (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration)
536
+ ? !!method.getModifiers().find(x => x.getText() === 'private') : false;
537
+ fmxMethod.isProtected = (method instanceof ts_morph_1.MethodDeclaration || method instanceof ts_morph_1.GetAccessorDeclaration || method instanceof ts_morph_1.SetAccessorDeclaration)
538
+ ? !!method.getModifiers().find(x => x.getText() === 'protected') : false;
527
539
  fmxMethod.signature = Helpers.computeSignature(method.getText());
528
- let methodName;
529
- if (isConstructor) {
530
- methodName = "constructor";
531
- }
532
- else {
533
- methodName = method.getName();
534
- }
540
+ const methodName = isConstructor ? "constructor" : method.getName();
535
541
  fmxMethod.name = methodName;
536
- if (!isConstructor) {
537
- if (method.getName().substring(0, 1) === "#") {
538
- fmxMethod.isPrivate = true;
539
- }
540
- }
541
- if (!fmxMethod.isPrivate && !fmxMethod.isProtected) {
542
- fmxMethod.isPublic = true;
543
- }
544
- else {
545
- fmxMethod.isPublic = false;
546
- }
547
- if (!isSignature) {
548
- fmxMethod.cyclomaticComplexity = currentCC[fmxMethod.name];
549
- }
550
- else {
551
- fmxMethod.cyclomaticComplexity = 0;
542
+ if (!isConstructor && methodName.startsWith("#")) {
543
+ fmxMethod.isPrivate = true;
552
544
  }
545
+ fmxMethod.isPublic = !fmxMethod.isPrivate && !fmxMethod.isProtected;
546
+ fmxMethod.cyclomaticComplexity = isSignature ? 0 : (currentCC[methodName] || 0);
553
547
  let methodTypeName = this.UNKNOWN_VALUE;
554
548
  try {
555
549
  methodTypeName = method.getReturnType().getText().trim();
550
+ analyze_1.logger.debug(`Method return type: ${methodTypeName}`);
556
551
  }
557
552
  catch (error) {
558
- analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of method: ${fmxMethod.name}. Continuing...`);
553
+ analyze_1.logger.error(`Failed to get return type for ${fqn}: ${error}`);
559
554
  }
560
- const fmxType = this.createOrGetFamixType(methodTypeName, method);
555
+ const fmxType = this.createOrGetFamixType(methodTypeName, method.getType(), method);
556
+ // console.log(`Created/retrieved return type with FQN: ${fmxType.fullyQualifiedName}`);
561
557
  fmxMethod.declaredType = fmxType;
562
558
  fmxMethod.numberOfLinesOfCode = method.getEndLineNumber() - method.getStartLineNumber();
563
- const parameters = method.getParameters();
564
- fmxMethod.numberOfParameters = parameters.length;
565
- if (!isSignature) {
566
- fmxMethod.numberOfStatements = method.getStatements().length;
567
- }
568
- else {
569
- fmxMethod.numberOfStatements = 0;
570
- }
559
+ fmxMethod.numberOfParameters = method.getParameters().length;
560
+ fmxMethod.numberOfStatements = isSignature ? 0 : method.getStatements().length;
561
+ // Add to famixRep
571
562
  initFQN(method, fmxMethod);
572
563
  this.famixRep.addElement(fmxMethod);
573
564
  this.makeFamixIndexFileAnchor(method, fmxMethod);
574
- this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxMethod);
565
+ this.fmxFunctionAndMethodMap.set(fqn, fmxMethod);
566
+ analyze_1.logger.debug(`Added method ${fqn} to famixRep`);
575
567
  }
576
568
  else {
577
- fmxMethod = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName);
569
+ analyze_1.logger.debug(`Method ${fqn} already exists`);
578
570
  }
579
571
  this.fmxElementObjectMap.set(fmxMethod, method);
580
572
  return fmxMethod;
@@ -614,7 +606,7 @@ class EntityDictionary {
614
606
  catch (error) {
615
607
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${func.getName()}. Continuing...`);
616
608
  }
617
- const fmxType = this.createOrGetFamixType(functionTypeName, func);
609
+ const fmxType = this.createOrGetFamixType(functionTypeName, func.getType(), func);
618
610
  fmxFunction.declaredType = fmxType;
619
611
  fmxFunction.numberOfLinesOfCode = func.getEndLineNumber() - func.getStartLineNumber();
620
612
  const parameters = func.getParameters();
@@ -653,7 +645,7 @@ class EntityDictionary {
653
645
  catch (error) {
654
646
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for parameter: ${param.getName()}. Continuing...`);
655
647
  }
656
- const fmxType = this.createOrGetFamixType(paramTypeName, param);
648
+ const fmxType = this.createOrGetFamixType(paramTypeName, param.getType(), param);
657
649
  fmxParam.declaredType = fmxType;
658
650
  fmxParam.name = param.getName();
659
651
  initFQN(param, fmxParam);
@@ -677,84 +669,88 @@ class EntityDictionary {
677
669
  this.fmxElementObjectMap.set(fmxParameterType, tp);
678
670
  return fmxParameterType;
679
671
  }
680
- /**
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
685
- */
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();
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];
695
- let isClassOrInterface = false;
696
- if (this.fmxClassMap.has(parameterTypeName)) {
697
- this.fmxClassMap.forEach((obj, name) => {
698
- if (obj instanceof Famix.ParametricClass) {
699
- if (name === element.getText() && obj.genericParameters.size > 0) {
700
- fmxParameterType = obj;
701
- isClassOrInterface = true;
702
- }
703
- }
704
- else {
705
- if (name === element.getText()) {
706
- fmxParameterType = obj;
707
- isClassOrInterface = true;
708
- }
709
- }
710
- });
711
- }
712
- if (this.fmxInterfaceMap.has(parameterTypeName)) {
713
- this.fmxInterfaceMap.forEach((obj, name) => {
714
- if (obj instanceof Famix.ParametricInterface) {
715
- if (name === element.getText() && obj.genericParameters.size > 0) {
716
- fmxParameterType = obj;
717
- isClassOrInterface = true;
718
- }
719
- }
720
- else {
721
- if (name === element.getText()) {
722
- fmxParameterType = obj;
723
- isClassOrInterface = true;
724
- }
725
- }
726
- });
727
- }
728
- if (!isClassOrInterface) {
729
- if (!this.fmxTypeMap.has(typeDeclaration)) {
730
- // TODO refactor
731
- if (isPrimitiveType(parameterTypeName)) {
732
- fmxParameterType = new Famix.PrimitiveType();
733
- fmxParameterType.isStub = true;
734
- }
735
- else {
736
- fmxParameterType = new Famix.ParameterType();
737
- }
738
- fmxParameterType.name = parameterTypeName;
739
- this.famixRep.addElement(fmxParameterType);
740
- this.fmxTypeMap.set(typeDeclaration, fmxParameterType);
741
- this.fmxElementObjectMap.set(fmxParameterType, typeParameterDeclaration);
742
- }
743
- else {
744
- const result = this.fmxTypeMap.get(typeDeclaration);
745
- if (result) {
746
- fmxParameterType = result;
747
- }
748
- else {
749
- throw new Error(`Famix type ${typeDeclaration} is not found in the Type map.`);
750
- }
751
- }
752
- }
753
- if (!fmxParameterType) {
754
- throw new Error(`fmxParameterType was undefined for parameterTypeName ${parameterTypeName}`);
755
- }
756
- return fmxParameterType;
757
- }
672
+ // /**
673
+ // * Creates a Famix type in the context of concretizations
674
+ // * @param typeName A type name
675
+ // * @param element An element
676
+ // * @returns The Famix model of the type
677
+ // */
678
+ // public createOrGetFamixConcreteType(element: TypeNode):
679
+ // Famix.ParameterType | Famix.PrimitiveType | Famix.Class | Famix.Interface {
680
+ // if (this.fmxTypeMap.has(element)) {
681
+ // const rType = this.fmxTypeMap.get(element);
682
+ // if (rType) {
683
+ // return rType;
684
+ // } else {
685
+ // throw new Error(`Famix type ${element.getText()} is not found in the type map.`);
686
+ // }
687
+ // }
688
+ // const typeParameterDeclaration = element.getSymbol()?.getDeclarations()[0] as TypeParameterDeclaration;
689
+ // // const parameterTypeName : string = element.getText();
690
+ // const parameterTypeName = getPrimitiveTypeName(element.getType()) || element.getText();
691
+ // let fmxParameterType: Famix.Type | Famix.Class | Famix.Interface | undefined = undefined;
692
+ // // get a TypeReference from a TypeNode
693
+ // const typeReference = element.getType();
694
+ // // get a TypeDeclaration from a TypeReference
695
+ // const typeDeclaration = typeReference.getSymbol()?.getDeclarations()[0] as TSMorphTypeDeclaration;
696
+ // let isClassOrInterface = false;
697
+ // if (this.fmxClassMap.has(parameterTypeName)){
698
+ // this.fmxClassMap.forEach((obj, name) => {
699
+ // if(obj instanceof Famix.ParametricClass){
700
+ // if (name === element.getText() && obj.genericParameters.size>0) {
701
+ // fmxParameterType = obj;
702
+ // isClassOrInterface = true;
703
+ // }
704
+ // } else {
705
+ // if (name === element.getText()) {
706
+ // fmxParameterType = obj;
707
+ // isClassOrInterface = true;
708
+ // }
709
+ // }
710
+ // });
711
+ // }
712
+ // if (this.fmxInterfaceMap.has(parameterTypeName)){
713
+ // this.fmxInterfaceMap.forEach((obj, name) => {
714
+ // if(obj instanceof Famix.ParametricInterface){
715
+ // if (name === element.getText() && obj.genericParameters.size>0) {
716
+ // fmxParameterType = obj;
717
+ // isClassOrInterface = true;
718
+ // }
719
+ // } else {
720
+ // if (name === element.getText()) {
721
+ // fmxParameterType = obj;
722
+ // isClassOrInterface = true;
723
+ // }
724
+ // }
725
+ // });
726
+ // }
727
+ // if(!isClassOrInterface){
728
+ // if (!this.fmxTypeMap.has(typeDeclaration)) {
729
+ // // TODO refactor
730
+ // if (isPrimitiveType(parameterTypeName)) {
731
+ // fmxParameterType = this.createOrGetFamixPrimitiveType(parameterTypeName);
732
+ // } else {
733
+ // fmxParameterType = new Famix.ParameterType();
734
+ // }
735
+ // fmxParameterType.name = parameterTypeName;
736
+ // this.famixRep.addElement(fmxParameterType);
737
+ // this.fmxTypeMap.set(typeDeclaration, fmxParameterType);
738
+ // this.fmxElementObjectMap.set(fmxParameterType,typeParameterDeclaration);
739
+ // }
740
+ // else {
741
+ // const result = this.fmxTypeMap.get(typeDeclaration);
742
+ // if (result) {
743
+ // fmxParameterType = result;
744
+ // } else {
745
+ // throw new Error(`Famix type ${typeDeclaration} is not found in the Type map.`);
746
+ // }
747
+ // }
748
+ // }
749
+ // if (!fmxParameterType) {
750
+ // throw new Error(`fmxParameterType was undefined for parameterTypeName ${parameterTypeName}`);
751
+ // }
752
+ // return fmxParameterType;
753
+ // }
758
754
  /**
759
755
  * Creates a Famix variable
760
756
  * @param variable A variable
@@ -778,7 +774,7 @@ class EntityDictionary {
778
774
  catch (error) {
779
775
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for variable: ${variable.getName()}. Continuing...`);
780
776
  }
781
- const fmxType = this.createOrGetFamixType(variableTypeName, variable);
777
+ const fmxType = this.createOrGetFamixType(variableTypeName, variable.getType(), variable);
782
778
  fmxVariable.declaredType = fmxType;
783
779
  fmxVariable.name = variable.getName();
784
780
  initFQN(variable, fmxVariable);
@@ -826,7 +822,7 @@ class EntityDictionary {
826
822
  catch (error) {
827
823
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for enum value: ${enumMember.getName()}. Continuing...`);
828
824
  }
829
- const fmxType = this.createOrGetFamixType(enumValueTypeName, enumMember);
825
+ const fmxType = this.createOrGetFamixType(enumValueTypeName, enumMember.getType(), enumMember);
830
826
  fmxEnumValue.declaredType = fmxType;
831
827
  fmxEnumValue.name = enumMember.getName();
832
828
  initFQN(enumMember, fmxEnumValue);
@@ -879,59 +875,92 @@ class EntityDictionary {
879
875
  * @param element A ts-morph element
880
876
  * @returns The Famix model of the type
881
877
  */
882
- createOrGetFamixType(typeName, element) {
883
- let fmxType;
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;
892
- analyze_1.logger.debug("Creating (or getting) type: '" + typeName + "' of element: " + element?.getText() + " of kind: " + element?.getKindName());
893
- if (isPrimitive) {
878
+ createOrGetFamixType(typeNameArg, tsMorphType, element) {
879
+ analyze_1.logger.debug(`Creating (or getting) type: '${tsMorphType?.getText() || "undefined"}' of element: '${element?.getText().slice(0, 50)}...' of kind: ${element?.getKindName()}`);
880
+ // convert type to correct primitive name (workaround for unique symbole primitive type)
881
+ // don't convert to primitive if it's a generic type
882
+ const typeName = !typeNameArg.includes("<") && tsMorphType && getPrimitiveTypeName(tsMorphType) || typeNameArg;
883
+ if (isPrimitiveType(typeName)) {
894
884
  return this.createOrGetFamixPrimitiveType(typeName);
895
885
  }
886
+ if (element.isKind(ts_morph_1.SyntaxKind.MethodSignature) || element.isKind(ts_morph_1.SyntaxKind.MethodDeclaration)) {
887
+ const methodFQN = FQNFunctions.getFQN(element);
888
+ const returnTypeFQN = `${methodFQN.replace(/\[Method(Signature|Declaration)\]$/, '')}[ReturnType]`;
889
+ // Check if we already have this return type in the repository
890
+ const existingType = this.famixRep.getFamixEntityByFullyQualifiedName(returnTypeFQN);
891
+ if (existingType) {
892
+ // console.log(`Found existing return type with FQN: ${returnTypeFQN}`);
893
+ return existingType;
894
+ }
895
+ // console.log(`Creating return type with distinct FQN: ${returnTypeFQN}`);
896
+ const fmxType = new Famix.Type();
897
+ fmxType.name = typeName;
898
+ fmxType.fullyQualifiedName = returnTypeFQN;
899
+ // Set container (same as method's container)
900
+ const methodAncestor = Helpers.findTypeAncestor(element);
901
+ if (methodAncestor) {
902
+ const ancestorFQN = FQNFunctions.getFQN(methodAncestor);
903
+ const ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFQN);
904
+ if (ancestor) {
905
+ fmxType.container = ancestor;
906
+ }
907
+ }
908
+ this.famixRep.addElement(fmxType);
909
+ this.fmxTypeMap.set(element, fmxType);
910
+ this.fmxElementObjectMap.set(fmxType, element);
911
+ return fmxType;
912
+ }
913
+ const isParametricType = (element instanceof ts_morph_1.ClassDeclaration && element.getTypeParameters().length > 0) ||
914
+ (element instanceof ts_morph_1.InterfaceDeclaration && element.getTypeParameters().length > 0);
896
915
  if (isParametricType) {
897
- // narrow the type
898
- const parametricElement = element;
899
- return this.createOrGetFamixParametricType(typeName, parametricElement);
916
+ return this.createOrGetFamixParametricType(typeName, element);
900
917
  }
901
918
  if (!this.fmxTypeMap.has(element)) {
902
- let ancestor = undefined;
903
- if (element !== undefined) {
919
+ // console.log('Type not found in map, creating new');
920
+ let ancestor;
921
+ if (element) {
904
922
  const typeAncestor = Helpers.findTypeAncestor(element);
905
- if (!typeAncestor) {
906
- throw new Error(`Ancestor not found for element ${element.getText()}.`);
923
+ // console.log(`Type ancestor found: ${typeAncestor?.getKindName()}`);
924
+ if (typeAncestor) {
925
+ const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);
926
+ // console.log(`Ancestor FQN: ${ancestorFullyQualifiedName}`);
927
+ ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
928
+ if (!ancestor) {
929
+ ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor.getType(), typeAncestor);
930
+ // console.log('Ancestor not found in repo, creating it');
931
+ }
932
+ else {
933
+ console.log(`Found ancestor in famixRep: ${ancestor.fullyQualifiedName}`);
934
+ }
907
935
  }
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);
936
+ else {
937
+ console.log(`No type ancestor found for ${typeName} - proceeding without container`);
913
938
  }
914
939
  }
915
- fmxType = new Famix.Type();
940
+ const fmxType = new Famix.Type();
916
941
  fmxType.name = typeName;
917
- if (!ancestor) {
942
+ // console.log(`Created new type with name: ${typeName}`);
943
+ if (ancestor) {
944
+ fmxType.container = ancestor;
945
+ }
946
+ else {
918
947
  throw new Error(`Ancestor not found for type ${typeName}.`);
919
948
  }
920
- fmxType.container = ancestor;
921
949
  initFQN(element, fmxType);
950
+ // console.log(`Type FQN after init: ${fmxType.fullyQualifiedName}`);
922
951
  this.makeFamixIndexFileAnchor(element, fmxType);
923
952
  this.famixRep.addElement(fmxType);
953
+ // console.log('Added type to repository');
924
954
  this.fmxTypeMap.set(element, fmxType);
925
955
  }
926
956
  else {
927
- const result = this.fmxTypeMap.get(element);
928
- if (result) {
929
- fmxType = result;
930
- }
931
- else {
957
+ const fmxType = this.fmxTypeMap.get(element);
958
+ if (!fmxType) {
932
959
  throw new Error(`Famix type ${typeName} is not found in the Type map.`);
933
960
  }
961
+ return fmxType;
934
962
  }
963
+ const fmxType = this.fmxTypeMap.get(element);
935
964
  this.fmxElementObjectMap.set(fmxType, element);
936
965
  return fmxType;
937
966
  }
@@ -1030,6 +1059,7 @@ class EntityDictionary {
1030
1059
  fmxType = new Famix.PrimitiveType();
1031
1060
  fmxType.isStub = true;
1032
1061
  fmxType.name = typeName;
1062
+ fmxType.fullyQualifiedName = typeName + "[PrimitiveType]";
1033
1063
  this.fmxPrimitiveTypeMap.set(typeName, fmxType);
1034
1064
  this.famixRep.addElement(fmxType);
1035
1065
  }
@@ -1050,22 +1080,25 @@ class EntityDictionary {
1050
1080
  }
1051
1081
  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}'.`);
1052
1082
  const nodeReferenceAncestor = Helpers.findAncestor(node);
1083
+ if (!nodeReferenceAncestor) {
1084
+ analyze_1.logger.error(`No ancestor found for node '${node.getText()}'`);
1085
+ return;
1086
+ }
1053
1087
  const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);
1054
1088
  const accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName);
1055
1089
  if (!accessor) {
1056
1090
  analyze_1.logger.error(`Ancestor ${ancestorFullyQualifiedName} of kind ${nodeReferenceAncestor.getKindName()} not found.`);
1057
- // accessor = this.createOrGetFamixType(ancestorFullyQualifiedName, nodeReferenceAncestor as TypeDeclaration);
1058
- return; // bail out TODO: this is probably wrong
1091
+ return; // Bail out for now
1059
1092
  }
1060
1093
  else {
1061
1094
  analyze_1.logger.debug(`Found accessor to be ${accessor.fullyQualifiedName}.`);
1062
1095
  }
1063
- // make sure accessor is a method, function, script or module
1096
+ // Ensure accessor is a method, function, script, or module
1064
1097
  if (!(accessor instanceof Famix.Method) && !(accessor instanceof Famix.ArrowFunction) && !(accessor instanceof Famix.Function) && !(accessor instanceof Famix.ScriptEntity) && !(accessor instanceof Famix.Module)) {
1065
1098
  analyze_1.logger.error(`Accessor ${accessor.fullyQualifiedName} is not a method, function, etc.`);
1066
1099
  return;
1067
1100
  }
1068
- // don't create any duplicates (e.g. if the same variable is accessed multiple times by same accessor)
1101
+ // Avoid duplicates
1069
1102
  const foundAccess = this.famixRep.getFamixAccessByAccessorAndVariable(accessor, fmxVar);
1070
1103
  if (foundAccess) {
1071
1104
  analyze_1.logger.debug(`FamixAccess already exists for accessor ${accessor.fullyQualifiedName} and variable ${fmxVar.fullyQualifiedName}.`);
@@ -1076,6 +1109,7 @@ class EntityDictionary {
1076
1109
  fmxAccess.variable = fmxVar;
1077
1110
  this.famixRep.addElement(fmxAccess);
1078
1111
  this.fmxElementObjectMap.set(fmxAccess, node);
1112
+ analyze_1.logger.debug(`Created access: ${accessor.fullyQualifiedName} -> ${fmxVar.fullyQualifiedName}`);
1079
1113
  }
1080
1114
  /**
1081
1115
  * Creates a Famix invocation
@@ -1112,98 +1146,70 @@ class EntityDictionary {
1112
1146
  }
1113
1147
  /**
1114
1148
  * Creates a Famix inheritance
1115
- * @param cls A class or an interface (subclass)
1116
- * @param inhClass The inherited class or interface (superclass)
1149
+ * @param baseClassOrInterface A class or an interface (subclass)
1150
+ * @param inheritedClassOrInterface The inherited class or interface (superclass)
1117
1151
  */
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
- // }
1152
+ createOrGetFamixInheritance(baseClassOrInterface, inheritedClassOrInterface) {
1153
+ analyze_1.logger.debug(`Creating FamixInheritance for ${baseClassOrInterface.getText()} and ${inheritedClassOrInterface.getText()} [${inheritedClassOrInterface.constructor.name}].`);
1154
1154
  const fmxInheritance = new Famix.Inheritance();
1155
- // logger.debug(`createFamixInheritance: classFullyQualifiedName: class fqn = ${classFullyQualifiedName}`);
1156
1155
  let subClass;
1157
- if (cls instanceof ts_morph_1.ClassDeclaration) {
1158
- subClass = this.createOrGetFamixClass(cls);
1156
+ if (baseClassOrInterface instanceof ts_morph_1.ClassDeclaration) {
1157
+ subClass = this.createOrGetFamixClass(baseClassOrInterface);
1159
1158
  }
1160
1159
  else {
1161
- subClass = this.createOrGetFamixInterface(cls);
1160
+ subClass = this.createOrGetFamixInterface(baseClassOrInterface);
1162
1161
  }
1163
1162
  if (!subClass) {
1164
- throw new Error(`Subclass ${cls} not found in Class or Interface maps.`);
1163
+ throw new Error(`Subclass ${baseClassOrInterface} not found in Class or Interface maps.`);
1165
1164
  }
1166
1165
  let superClass;
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);
1166
+ if (inheritedClassOrInterface instanceof ts_morph_1.ClassDeclaration) {
1167
+ superClass = this.createOrGetFamixClass(inheritedClassOrInterface);
1168
+ }
1169
+ else if (inheritedClassOrInterface instanceof ts_morph_1.InterfaceDeclaration) {
1170
+ superClass = this.createOrGetFamixInterface(inheritedClassOrInterface);
1194
1171
  }
1195
1172
  else {
1196
- // inhClass instanceof ExpressionWithTypeArguments
1197
- const interfaceDeclaration = getInterfaceDeclarationFromExpression(inhClass);
1198
- if (interfaceDeclaration !== undefined) {
1199
- superClass = this.createOrGetFamixInterface(interfaceDeclaration);
1173
+ // inheritedClassOrInterface instanceof ExpressionWithTypeArguments
1174
+ // must determine if inheritedClassOrInterface is a class or an interface
1175
+ // then find the declaration, else it's a stub
1176
+ const heritageClause = inheritedClassOrInterface.getParent();
1177
+ if (heritageClause instanceof ts_morph_1.HeritageClause) {
1178
+ // cases: 1) class extends class, 2) class implements interface, 3) interface extends interface
1179
+ // class extends class
1180
+ if (heritageClause.getText().startsWith("extends") && baseClassOrInterface instanceof ts_morph_1.ClassDeclaration) {
1181
+ const classDeclaration = getInterfaceOrClassDeclarationFromExpression(inheritedClassOrInterface);
1182
+ if (classDeclaration !== undefined && classDeclaration instanceof ts_morph_1.ClassDeclaration) {
1183
+ superClass = this.createOrGetFamixClass(classDeclaration);
1184
+ }
1185
+ else {
1186
+ analyze_1.logger.error(`Class declaration not found for ${inheritedClassOrInterface.getText()}.`);
1187
+ superClass = this.createOrGetFamixClassStub(inheritedClassOrInterface);
1188
+ }
1189
+ }
1190
+ else if (heritageClause.getText().startsWith("implements") && baseClassOrInterface instanceof ts_morph_1.ClassDeclaration // class implements interface
1191
+ || (heritageClause.getText().startsWith("extends") && baseClassOrInterface instanceof ts_morph_1.InterfaceDeclaration)) { // interface extends interface
1192
+ const interfaceOrClassDeclaration = getInterfaceOrClassDeclarationFromExpression(inheritedClassOrInterface);
1193
+ if (interfaceOrClassDeclaration !== undefined && interfaceOrClassDeclaration instanceof ts_morph_1.InterfaceDeclaration) {
1194
+ superClass = this.createOrGetFamixInterface(interfaceOrClassDeclaration);
1195
+ }
1196
+ else {
1197
+ analyze_1.logger.error(`Interface declaration not found for ${inheritedClassOrInterface.getText()}.`);
1198
+ superClass = this.createOrGetFamixInterfaceStub(inheritedClassOrInterface);
1199
+ }
1200
+ }
1201
+ else {
1202
+ // throw new Error(`Parent of ${inheritedClassOrInterface.getText()} is not a class or an interface.`);
1203
+ analyze_1.logger.error(`Parent of ${inheritedClassOrInterface.getText()} is not a class or an interface.`);
1204
+ superClass = this.createOrGetFamixInterfaceStub(inheritedClassOrInterface);
1205
+ }
1200
1206
  }
1201
1207
  else {
1202
- throw new Error(`Interface declaration not found for ${inhClass.getText()}.`);
1208
+ throw new Error(`Heritage clause not found for ${inheritedClassOrInterface.getText()}.`);
1203
1209
  }
1204
1210
  }
1205
- this.fmxElementObjectMap.set(superClass, inhClass);
1206
- this.makeFamixIndexFileAnchor(inhClass, superClass);
1211
+ this.fmxElementObjectMap.set(superClass, inheritedClassOrInterface);
1212
+ this.makeFamixIndexFileAnchor(inheritedClassOrInterface, superClass);
1207
1213
  this.famixRep.addElement(superClass);
1208
1214
  fmxInheritance.subclass = subClass;
1209
1215
  fmxInheritance.superclass = superClass;
@@ -1212,6 +1218,42 @@ class EntityDictionary {
1212
1218
  // We don't map inheritance to the source code element because there are two elements (super, sub)
1213
1219
  // this.fmxElementObjectMap.set(fmxInheritance, null);
1214
1220
  }
1221
+ createOrGetFamixClassStub(unresolvedInheritedClass) {
1222
+ // make a FQN for the stub
1223
+ const fqn = FQNFunctions.getFQNUnresolvedInheritedClassOrInterface(unresolvedInheritedClass);
1224
+ analyze_1.logger.debug(`createOrGetFamixClassStub: fqn: ${fqn}`);
1225
+ const fmxClass = this.famixRep.getFamixEntityByFullyQualifiedName(fqn);
1226
+ if (fmxClass) {
1227
+ return fmxClass;
1228
+ }
1229
+ else {
1230
+ const stub = new Famix.Class();
1231
+ stub.name = unresolvedInheritedClass.getText();
1232
+ stub.isStub = true;
1233
+ stub.fullyQualifiedName = fqn;
1234
+ this.famixRep.addElement(stub);
1235
+ this.fmxElementObjectMap.set(stub, unresolvedInheritedClass);
1236
+ return stub;
1237
+ }
1238
+ }
1239
+ createOrGetFamixInterfaceStub(unresolvedInheritedInterface) {
1240
+ // make a FQN for the stub
1241
+ const fqn = FQNFunctions.getFQNUnresolvedInheritedClassOrInterface(unresolvedInheritedInterface);
1242
+ analyze_1.logger.debug(`createOrGetFamixInterfaceStub: fqn: ${fqn}`);
1243
+ const fmxInterface = this.famixRep.getFamixEntityByFullyQualifiedName(fqn);
1244
+ if (fmxInterface) {
1245
+ return fmxInterface;
1246
+ }
1247
+ else {
1248
+ const stub = new Famix.Interface();
1249
+ stub.name = unresolvedInheritedInterface.getText();
1250
+ stub.isStub = true;
1251
+ stub.fullyQualifiedName = fqn;
1252
+ this.famixRep.addElement(stub);
1253
+ this.fmxElementObjectMap.set(stub, unresolvedInheritedInterface);
1254
+ return stub;
1255
+ }
1256
+ }
1215
1257
  createFamixImportClause(importedEntity, importingEntity) {
1216
1258
  const fmxImportClause = new Famix.ImportClause();
1217
1259
  fmxImportClause.importedEntity = importedEntity;
@@ -1234,7 +1276,8 @@ class EntityDictionary {
1234
1276
  if (importDeclaration && this.fmxImportClauseMap.has(importDeclaration)) {
1235
1277
  const rImportClause = this.fmxImportClauseMap.get(importDeclaration);
1236
1278
  if (rImportClause) {
1237
- return; // don't do anything
1279
+ analyze_1.logger.debug(`Import clause ${importElement.getText()} already exists in map, skipping.`);
1280
+ return;
1238
1281
  }
1239
1282
  else {
1240
1283
  throw new Error(`Import clause ${importElement.getText()} is not found in the import clause map.`);
@@ -1246,21 +1289,19 @@ class EntityDictionary {
1246
1289
  let importedEntityName;
1247
1290
  const absolutePathProject = this.famixRep.getAbsolutePath();
1248
1291
  const absolutePath = path_1.default.normalize(moduleSpecifierFilePath);
1249
- // convert the path and remove any windows backslashes introduced by path.normalize
1250
1292
  analyze_1.logger.debug(`createFamixImportClause: absolutePath: ${absolutePath}`);
1251
1293
  analyze_1.logger.debug(`createFamixImportClause: convertToRelativePath: ${this.convertToRelativePath(absolutePath, absolutePathProject)}`);
1252
1294
  const pathInProject = this.convertToRelativePath(absolutePath, absolutePathProject).replace(/\\/g, "/");
1253
1295
  analyze_1.logger.debug(`createFamixImportClause: pathInProject: ${pathInProject}`);
1254
1296
  let pathName = "{" + pathInProject + "}.";
1255
1297
  analyze_1.logger.debug(`createFamixImportClause: pathName: ${pathName}`);
1256
- // Named imports, e.g. import { ClassW } from "./complexExportModule";
1257
- // Start with simple import clause (without referring to the actual variable)
1258
1298
  if (importDeclaration instanceof ts_morph_1.ImportDeclaration
1259
1299
  && importElement instanceof ts_morph_1.ImportSpecifier) {
1260
1300
  importedEntityName = importElement.getName();
1261
1301
  pathName = pathName + importedEntityName;
1262
1302
  if (isInExports) {
1263
1303
  importedEntity = this.famixRep.getFamixEntityByFullyQualifiedName(pathName);
1304
+ analyze_1.logger.debug(`Found exported entity: ${pathName}`);
1264
1305
  }
1265
1306
  if (importedEntity === undefined) {
1266
1307
  importedEntity = new Famix.NamedEntity();
@@ -1268,39 +1309,38 @@ class EntityDictionary {
1268
1309
  if (!isInExports) {
1269
1310
  importedEntity.isStub = true;
1270
1311
  }
1271
- // logger.debug(`createFamixImportClause: Creating named entity ${importedEntityName} with FQN ${pathName}`);
1272
- // importedEntity.fullyQualifiedName = pathName;
1312
+ analyze_1.logger.debug(`Creating named entity ${importedEntityName} for ImportSpecifier ${importElement.getText()}`);
1273
1313
  initFQN(importElement, importedEntity);
1314
+ analyze_1.logger.debug(`Assigned FQN to entity: ${importedEntity.fullyQualifiedName}`);
1274
1315
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1275
- // must add entity to repository
1276
1316
  this.famixRep.addElement(importedEntity);
1317
+ analyze_1.logger.debug(`Added entity to repository: ${importedEntity.fullyQualifiedName}`);
1277
1318
  }
1278
1319
  }
1279
- // handle import equals declarations, e.g. import myModule = require("./complexExportModule");
1280
- // TypeScript can't determine the type of the imported module, so we create a Module entity
1281
1320
  else if (importDeclaration instanceof ts_morph_1.ImportEqualsDeclaration) {
1282
1321
  importedEntityName = importDeclaration?.getName();
1283
1322
  pathName = pathName + importedEntityName;
1284
1323
  importedEntity = new Famix.StructuralEntity();
1285
1324
  importedEntity.name = importedEntityName;
1286
1325
  initFQN(importDeclaration, importedEntity);
1326
+ analyze_1.logger.debug(`Assigned FQN to ImportEquals entity: ${importedEntity.fullyQualifiedName}`);
1287
1327
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1288
- // importedEntity.fullyQualifiedName = pathName;
1289
- const anyType = this.createOrGetFamixType('any', importDeclaration);
1328
+ const anyType = this.createOrGetFamixType('any', undefined, importDeclaration);
1290
1329
  importedEntity.declaredType = anyType;
1291
1330
  }
1292
- else { // default imports, e.g. import ClassW from "./complexExportModule";
1331
+ else {
1293
1332
  importedEntityName = importElement.getText();
1294
1333
  pathName = pathName + (isDefaultExport ? "defaultExport" : "namespaceExport");
1295
1334
  importedEntity = new Famix.NamedEntity();
1296
1335
  importedEntity.name = importedEntityName;
1297
- // importedEntity.fullyQualifiedName = pathName;
1298
1336
  initFQN(importElement, importedEntity);
1337
+ analyze_1.logger.debug(`Assigned FQN to default/namespace entity: ${importedEntity.fullyQualifiedName}`);
1299
1338
  this.makeFamixIndexFileAnchor(importElement, importedEntity);
1300
1339
  }
1301
- // I don't think it should be added to the repository if it exists already
1302
- if (!isInExports)
1340
+ if (!isInExports) {
1303
1341
  this.famixRep.addElement(importedEntity);
1342
+ analyze_1.logger.debug(`Added non-exported entity to repository: ${importedEntity.fullyQualifiedName}`);
1343
+ }
1304
1344
  const importerFullyQualifiedName = FQNFunctions.getFQN(importer);
1305
1345
  const fmxImporter = this.famixRep.getFamixEntityByFullyQualifiedName(importerFullyQualifiedName);
1306
1346
  fmxImportClause.importingEntity = fmxImporter;
@@ -1311,7 +1351,7 @@ class EntityDictionary {
1311
1351
  else {
1312
1352
  fmxImportClause.moduleSpecifier = importDeclaration?.getModuleSpecifierValue();
1313
1353
  }
1314
- analyze_1.logger.debug(`createFamixImportClause: ${fmxImportClause.importedEntity?.name} (of type ${Helpers.getSubTypeName(fmxImportClause.importedEntity)}) is imported by ${fmxImportClause.importingEntity?.name}`);
1354
+ analyze_1.logger.debug(`ImportClause: ${fmxImportClause.importedEntity?.name} (type=${Helpers.getSubTypeName(fmxImportClause.importedEntity)}) imported by ${fmxImportClause.importingEntity?.name}`);
1315
1355
  fmxImporter.addOutgoingImport(fmxImportClause);
1316
1356
  this.famixRep.addElement(fmxImportClause);
1317
1357
  if (importDeclaration) {
@@ -1361,7 +1401,7 @@ class EntityDictionary {
1361
1401
  catch (error) {
1362
1402
  analyze_1.logger.error(`> WARNING: got exception ${error}. Failed to get usable name for return type of function: ${functionName}. Continuing...`);
1363
1403
  }
1364
- const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction);
1404
+ const fmxType = this.createOrGetFamixType(functionTypeName, arrowFunction.getReturnType(), arrowFunction);
1365
1405
  fmxArrowFunction.declaredType = fmxType;
1366
1406
  fmxArrowFunction.numberOfLinesOfCode = arrowFunction.getEndLineNumber() - arrowFunction.getStartLineNumber();
1367
1407
  const parameters = arrowFunction.getParameters();
@@ -1383,282 +1423,283 @@ class EntityDictionary {
1383
1423
  * @param cls A class
1384
1424
  * @returns The Famix model of the concretisation
1385
1425
  */
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
- // }
1426
+ createFamixConcretisation(conEntity, genEntity) {
1427
+ const fmxConcretisation = new Famix.Concretisation();
1428
+ fmxConcretisation.concreteEntity = conEntity;
1429
+ fmxConcretisation.genericEntity = genEntity;
1430
+ // this.fmxElementObjectMap.set(fmxConcretisation,null);
1431
+ this.famixRep.addElement(fmxConcretisation);
1432
+ // const parameterConcretisation = this.createFamixParameterConcretisation(fmxConcretisation);
1433
+ return fmxConcretisation;
1434
+ }
1395
1435
  /**
1396
1436
  * Creates a Famix concretisation
1397
1437
  * @param concretisation A FamixConcretisation
1398
1438
  * @returns The Famix model of the ParameterConcrestisation
1399
1439
  */
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
- // }
1440
+ createFamixParameterConcretisation(concretisation) {
1441
+ const conClass = concretisation.concreteEntity;
1442
+ const genClass = concretisation.genericEntity;
1443
+ analyze_1.logger.debug(`Creating parameter concretisation between ${conClass.fullyQualifiedName} and ${genClass.fullyQualifiedName}`);
1444
+ const parameterConcretisations = this.famixRep._getAllEntitiesWithType("ParameterConcretisation");
1445
+ const concreteParameters = conClass.concreteParameters;
1446
+ const genericParameters = genClass.genericParameters;
1447
+ const conClassTypeParametersIterator = concreteParameters.values();
1448
+ const genClassTypeParametersIterator = genericParameters.values();
1449
+ let fmxParameterConcretisation = undefined;
1450
+ for (let i = 0; i < genericParameters.size; i++) {
1451
+ const conClassTypeParameter = conClassTypeParametersIterator.next().value;
1452
+ const genClassTypeParameter = genClassTypeParametersIterator.next().value;
1453
+ let createParameterConcretisation = true;
1454
+ if (conClassTypeParameter && genClassTypeParameter && conClassTypeParameter.name != genClassTypeParameter.name) {
1455
+ parameterConcretisations.forEach((param) => {
1456
+ if (conClassTypeParameter.name == param.concreteParameter.name && genClassTypeParameter.name == param.genericParameter.name) {
1457
+ createParameterConcretisation = false;
1458
+ fmxParameterConcretisation = param;
1459
+ }
1460
+ });
1461
+ if (createParameterConcretisation) {
1462
+ fmxParameterConcretisation = new Famix.ParameterConcretisation();
1463
+ fmxParameterConcretisation.genericParameter = genClassTypeParameter;
1464
+ fmxParameterConcretisation.concreteParameter = conClassTypeParameter;
1465
+ fmxParameterConcretisation.addConcretisation(concretisation);
1466
+ // this.fmxElementObjectMap.set(fmxParameterConcretisation,null);
1467
+ }
1468
+ else {
1469
+ if (!fmxParameterConcretisation) {
1470
+ throw new Error(`fmxParameterConcretisation was undefined for concretisation with generic parameter ${genClassTypeParameter.name} and concrete parameter ${conClassTypeParameter.name}`);
1471
+ }
1472
+ fmxParameterConcretisation.addConcretisation(concretisation);
1473
+ }
1474
+ this.famixRep.addElement(fmxParameterConcretisation);
1475
+ }
1476
+ }
1477
+ if (!fmxParameterConcretisation) {
1478
+ analyze_1.logger.error(`fmxParameterConcretisation was undefined for concretisation with concrete entity ${conClass.fullyQualifiedName} and generic entity ${genClass.fullyQualifiedName}`);
1479
+ }
1480
+ return fmxParameterConcretisation;
1481
+ }
1441
1482
  /**
1442
1483
  * Creates a Famix concretisation between two classes or two interfaces
1443
1484
  * @param element A class or an Interface
1444
1485
  */
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
- // }
1486
+ createFamixConcretisationClassOrInterfaceSpecialisation(element) {
1487
+ const superEntity = element.getExtends();
1488
+ let superEntityArray;
1489
+ if (superEntity) {
1490
+ superEntityArray = Array.isArray(superEntity) ? superEntity : [superEntity];
1491
+ }
1492
+ if (superEntityArray && superEntityArray.length > 0) {
1493
+ superEntityArray.forEach(entity => {
1494
+ let entityIsGeneric;
1495
+ const superEntitySymbol = entity.getExpression().getSymbolOrThrow();
1496
+ let superEntityDeclaration;
1497
+ if (superEntity instanceof ts_morph_1.ExpressionWithTypeArguments) {
1498
+ superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts_morph_1.ts.SyntaxKind.ClassDeclaration);
1499
+ }
1500
+ else {
1501
+ superEntityDeclaration = superEntitySymbol.getDeclarations()[0].asKind(ts_morph_1.ts.SyntaxKind.InterfaceDeclaration);
1502
+ }
1503
+ if (superEntityDeclaration) {
1504
+ entityIsGeneric = superEntityDeclaration.getTypeParameters().length > 0;
1505
+ }
1506
+ if (entityIsGeneric) {
1507
+ let EntityDeclaration;
1508
+ let genEntity;
1509
+ if (superEntity instanceof ts_morph_1.ExpressionWithTypeArguments) {
1510
+ EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0];
1511
+ genEntity = this.createOrGetFamixClass(EntityDeclaration);
1512
+ }
1513
+ else {
1514
+ EntityDeclaration = entity.getExpression().getSymbol()?.getDeclarations()[0];
1515
+ genEntity = this.createOrGetFamixInterface(EntityDeclaration);
1516
+ }
1517
+ const genParams = EntityDeclaration.getTypeParameters().map((param) => param.getText());
1518
+ const args = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1519
+ const conParams = element.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1520
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1521
+ const conEntity = this.createOrGetFamixConcreteElement(genEntity, EntityDeclaration, args);
1522
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1523
+ let createConcretisation = true;
1524
+ concretisations.forEach((conc) => {
1525
+ if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName) {
1526
+ createConcretisation = false;
1527
+ }
1528
+ });
1529
+ if (createConcretisation) {
1530
+ const fmxConcretisation = this.createFamixConcretisation(conEntity, genEntity);
1531
+ }
1532
+ }
1533
+ }
1534
+ });
1535
+ }
1536
+ // TODO: This function seems unfinished
1537
+ }
1496
1538
  /**
1497
1539
  * Creates a Famix concretisation between a class and its instanciations
1498
1540
  * @param cls A class
1499
1541
  */
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
- // }
1542
+ createFamixConcretisationGenericInstantiation(cls) {
1543
+ const isGeneric = cls.getTypeParameters().length > 0;
1544
+ if (isGeneric) {
1545
+ const instances = cls.getSourceFile().getDescendantsOfKind(ts_morph_1.ts.SyntaxKind.NewExpression)
1546
+ .filter(newExpr => {
1547
+ const expression = newExpr.getExpression();
1548
+ return expression.getText() === cls.getName();
1549
+ });
1550
+ instances.forEach(instance => {
1551
+ const instanceIsGeneric = instance.getTypeArguments().length > 0;
1552
+ if (instanceIsGeneric) {
1553
+ const conParams = instance.getTypeArguments().map((param) => param.getText());
1554
+ const genEntity = this.createOrGetFamixClass(cls);
1555
+ const genParams = cls.getTypeParameters().map((param) => param.getText());
1556
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1557
+ const conEntity = this.createOrGetFamixConcreteElement(genEntity, cls, instance.getTypeArguments());
1558
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1559
+ let createConcretisation = true;
1560
+ concretisations.forEach((conc) => {
1561
+ if (genEntity.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conEntity.fullyQualifiedName) {
1562
+ createConcretisation = false;
1563
+ }
1564
+ });
1565
+ if (createConcretisation) {
1566
+ const fmxConcretisation = this.createFamixConcretisation(conEntity, genEntity);
1567
+ }
1568
+ }
1569
+ }
1570
+ });
1571
+ }
1572
+ // TODO: This function seems unfinished
1573
+ }
1533
1574
  /**
1534
1575
  * Creates a Famix concretisation between a class and its instanciations
1535
1576
  * @param func A function
1536
1577
  */
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
- // }
1578
+ createFamixConcretisationFunctionInstantiation(element) {
1579
+ const isGeneric = element.getTypeParameters().length > 0;
1580
+ if (isGeneric) {
1581
+ const genParams = element.getTypeParameters().map(param => param.getText());
1582
+ const uses = element.findReferencesAsNodes();
1583
+ uses.forEach(usage => {
1584
+ let currentNode = usage;
1585
+ while (currentNode) {
1586
+ if (currentNode.getKind() === ts_morph_1.SyntaxKind.CallExpression) {
1587
+ const callExpression = currentNode.asKind(ts_morph_1.SyntaxKind.CallExpression);
1588
+ if (!callExpression) {
1589
+ throw new Error(`CallExpression not found for ${currentNode.getText()}`);
1590
+ }
1591
+ const instanceIsGeneric = callExpression.getTypeArguments().length > 0;
1592
+ if (instanceIsGeneric) {
1593
+ const args = callExpression.getTypeArguments();
1594
+ const conParams = callExpression.getTypeArguments().map(param => param.getText());
1595
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1596
+ let genElement;
1597
+ if (element instanceof ts_morph_1.FunctionDeclaration) {
1598
+ genElement = this.createOrGetFamixFunction(element, {});
1599
+ }
1600
+ else {
1601
+ genElement = this.createOrGetFamixMethod(element, {});
1602
+ }
1603
+ const concElement = this.createOrGetFamixConcreteElement(genElement, element, args);
1604
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1605
+ let createConcretisation = true;
1606
+ concretisations.forEach((conc) => {
1607
+ if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName) {
1608
+ createConcretisation = false;
1609
+ }
1610
+ });
1611
+ if (createConcretisation) {
1612
+ const fmxConcretisation = this.createFamixConcretisation(concElement, genElement);
1613
+ }
1614
+ }
1615
+ }
1616
+ break;
1617
+ }
1618
+ // Remonter à l'élément parent (utile si le nœud de référence est un enfant)
1619
+ currentNode = currentNode.getParent();
1620
+ }
1621
+ });
1622
+ }
1623
+ }
1583
1624
  /**
1584
1625
  * Creates a Famix concretisation between a class and an interface
1585
1626
  * @param cls A class
1586
1627
  */
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
- // }
1628
+ createFamixConcretisationInterfaceClass(cls) {
1629
+ const superInterfaces = cls.getImplements();
1630
+ superInterfaces.forEach(interfaceType => {
1631
+ const interfaceIsGeneric = interfaceType.getTypeArguments().length > 0;
1632
+ if (interfaceIsGeneric) {
1633
+ const interfaceDeclaration = interfaceType.getExpression().getSymbol()?.getDeclarations()[0];
1634
+ const genParams = interfaceDeclaration.getTypeParameters().map((param) => param.getText());
1635
+ const conParams = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments().map((param) => param.getText());
1636
+ const args = cls.getHeritageClauses()[0].getTypeNodes()[0].getTypeArguments();
1637
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1638
+ const genInterface = this.createOrGetFamixInterface(interfaceDeclaration);
1639
+ const conInterface = this.createOrGetFamixConcreteElement(genInterface, interfaceDeclaration, args);
1640
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1641
+ let createConcretisation = true;
1642
+ concretisations.forEach((conc) => {
1643
+ if (genInterface.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == conInterface.fullyQualifiedName) {
1644
+ createConcretisation = false;
1645
+ }
1646
+ });
1647
+ if (createConcretisation) {
1648
+ const fmxConcretisation = this.createFamixConcretisation(conInterface, genInterface);
1649
+ }
1650
+ }
1651
+ }
1652
+ });
1653
+ }
1613
1654
  /**
1614
1655
  * Creates a Famix concretisation between an interface and a Type
1615
1656
  * @param element A variable or a function
1616
1657
  * @param inter An interface
1617
1658
  */
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
- // }
1659
+ createFamixConcretisationTypeInstanciation(element) {
1660
+ const isGeneric = element.getTypeParameters().length > 0;
1661
+ if (isGeneric) {
1662
+ const genParams = element.getTypeParameters().map(param => param.getText());
1663
+ const uses = element.findReferencesAsNodes();
1664
+ uses.forEach(use => {
1665
+ let parentNode = use.getParent();
1666
+ while (parentNode) {
1667
+ if (parentNode.getKind() === ts_morph_1.SyntaxKind.TypeReference) {
1668
+ const typeReferenceNode = parentNode.asKind(ts_morph_1.SyntaxKind.TypeReference);
1669
+ if (!typeReferenceNode) {
1670
+ throw new Error(`TypeReferenceNode not found for ${parentNode.getText()}`);
1671
+ }
1672
+ const typeReferenceNodeIsGeneric = typeReferenceNode.getTypeArguments().length > 0;
1673
+ if (typeReferenceNodeIsGeneric) { }
1674
+ const args = typeReferenceNode.getTypeArguments();
1675
+ const conParams = typeReferenceNode.getTypeArguments().map(param => param.getText());
1676
+ if (!Helpers.arraysAreEqual(conParams, genParams)) {
1677
+ let genElement;
1678
+ if (element instanceof ts_morph_1.ClassDeclaration) {
1679
+ genElement = this.createOrGetFamixClass(element);
1680
+ }
1681
+ else {
1682
+ genElement = this.createOrGetFamixInterface(element);
1683
+ }
1684
+ const concElement = this.createOrGetFamixConcreteElement(genElement, element, args);
1685
+ const concretisations = this.famixRep._getAllEntitiesWithType("Concretisation");
1686
+ let createConcretisation = true;
1687
+ concretisations.forEach((conc) => {
1688
+ if (genElement.fullyQualifiedName == conc.genericEntity.fullyQualifiedName && conc.concreteEntity.fullyQualifiedName == concElement.fullyQualifiedName) {
1689
+ createConcretisation = false;
1690
+ }
1691
+ });
1692
+ if (createConcretisation) {
1693
+ const fmxConcretisation = this.createFamixConcretisation(concElement, genElement);
1694
+ }
1695
+ }
1696
+ break;
1697
+ }
1698
+ parentNode = parentNode.getParent();
1699
+ }
1700
+ });
1701
+ }
1702
+ }
1662
1703
  convertToRelativePath(absolutePath, absolutePathProject) {
1663
1704
  analyze_1.logger.debug(`convertToRelativePath: absolutePath: '${absolutePath}', absolutePathProject: '${absolutePathProject}'`);
1664
1705
  if (absolutePath.startsWith(absolutePathProject)) {
@@ -1679,6 +1720,7 @@ function isPrimitiveType(typeName) {
1679
1720
  typeName === "boolean" ||
1680
1721
  typeName === "bigint" ||
1681
1722
  typeName === "symbol" ||
1723
+ typeName === "unique symbol" ||
1682
1724
  typeName === "undefined" ||
1683
1725
  typeName === "null" ||
1684
1726
  typeName === "any" ||
@@ -1704,29 +1746,32 @@ function initFQN(sourceElement, famixElement) {
1704
1746
  }
1705
1747
  }
1706
1748
  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;
1749
+ // Just keep the existing SyntaxKind set as it is
1750
+ const typeContextKinds = new Set([
1751
+ ts_morph_1.SyntaxKind.Constructor,
1752
+ ts_morph_1.SyntaxKind.MethodDeclaration,
1753
+ ts_morph_1.SyntaxKind.FunctionDeclaration,
1754
+ ts_morph_1.SyntaxKind.FunctionExpression,
1755
+ ts_morph_1.SyntaxKind.ArrowFunction,
1756
+ ts_morph_1.SyntaxKind.Parameter,
1757
+ ts_morph_1.SyntaxKind.VariableDeclaration,
1758
+ ts_morph_1.SyntaxKind.PropertyDeclaration,
1759
+ ts_morph_1.SyntaxKind.PropertySignature,
1760
+ ts_morph_1.SyntaxKind.TypeParameter,
1761
+ ts_morph_1.SyntaxKind.Identifier,
1762
+ ts_morph_1.SyntaxKind.Decorator,
1763
+ ts_morph_1.SyntaxKind.GetAccessor,
1764
+ ts_morph_1.SyntaxKind.SetAccessor,
1765
+ ts_morph_1.SyntaxKind.ImportSpecifier,
1766
+ ts_morph_1.SyntaxKind.EnumDeclaration,
1767
+ ts_morph_1.SyntaxKind.EnumMember,
1768
+ ts_morph_1.SyntaxKind.TypeAliasDeclaration,
1769
+ ts_morph_1.SyntaxKind.ImportDeclaration,
1770
+ ts_morph_1.SyntaxKind.ExpressionWithTypeArguments
1771
+ ]);
1772
+ return typeContextKinds.has(sourceElement.getKind());
1728
1773
  }
1729
- function getInterfaceDeclarationFromExpression(expression) {
1774
+ function getInterfaceOrClassDeclarationFromExpression(expression) {
1730
1775
  // Step 1: Get the type of the expression
1731
1776
  const type = expression.getType();
1732
1777
  // Step 2: Get the symbol associated with the type
@@ -1743,19 +1788,21 @@ function getInterfaceDeclarationFromExpression(expression) {
1743
1788
  }
1744
1789
  }
1745
1790
  // Step 3: Resolve the symbol to find the actual declaration
1746
- const interfaceDeclaration = resolveSymbolToInterfaceDeclaration(symbol);
1791
+ const interfaceDeclaration = resolveSymbolToInterfaceOrClassDeclaration(symbol);
1747
1792
  if (!interfaceDeclaration) {
1748
1793
  analyze_1.logger.error(`Interface declaration not found for ${expression.getText()}.`);
1749
1794
  }
1750
1795
  return interfaceDeclaration;
1751
1796
  }
1752
- function resolveSymbolToInterfaceDeclaration(symbol) {
1797
+ const lodash_1 = __importDefault(require("lodash"));
1798
+ function resolveSymbolToInterfaceOrClassDeclaration(symbol) {
1753
1799
  // Get the declarations associated with the symbol
1754
1800
  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;
1801
+ // Filter for InterfaceDeclaration or ClassDeclaration
1802
+ const interfaceOrClassDeclaration = declarations.find(declaration => declaration instanceof ts_morph_1.InterfaceDeclaration ||
1803
+ declaration instanceof ts_morph_1.ClassDeclaration);
1804
+ if (interfaceOrClassDeclaration) {
1805
+ return interfaceOrClassDeclaration;
1759
1806
  }
1760
1807
  // Handle imports: If the symbol is imported, resolve the import to find the actual declaration
1761
1808
  for (const declaration of declarations) {
@@ -1767,13 +1814,41 @@ function resolveSymbolToInterfaceDeclaration(symbol) {
1767
1814
  const exportedSymbols = moduleSpecifier.getExportSymbols();
1768
1815
  const exportedSymbol = exportedSymbols.find(symbol => symbol.getName() === importSpecifier.getName());
1769
1816
  if (exportedSymbol) {
1770
- return resolveSymbolToInterfaceDeclaration(exportedSymbol);
1817
+ return resolveSymbolToInterfaceOrClassDeclaration(exportedSymbol);
1771
1818
  }
1772
1819
  }
1773
1820
  }
1774
1821
  }
1775
1822
  return undefined;
1776
1823
  }
1824
+ function getPrimitiveTypeName(type) {
1825
+ const flags = type.compilerType.flags;
1826
+ if (flags & ts_morph_1.ts.TypeFlags.String)
1827
+ return "string";
1828
+ if (flags & ts_morph_1.ts.TypeFlags.Number)
1829
+ return "number";
1830
+ if (flags & ts_morph_1.ts.TypeFlags.Boolean)
1831
+ return "boolean";
1832
+ if (flags & ts_morph_1.ts.TypeFlags.BigInt)
1833
+ return "bigint";
1834
+ if (flags & ts_morph_1.ts.TypeFlags.UniqueESSymbol)
1835
+ return "unique symbol";
1836
+ if (flags & ts_morph_1.ts.TypeFlags.ESSymbol)
1837
+ return "symbol";
1838
+ if (flags & ts_morph_1.ts.TypeFlags.Undefined)
1839
+ return "undefined";
1840
+ if (flags & ts_morph_1.ts.TypeFlags.Null)
1841
+ return "null";
1842
+ if (flags & ts_morph_1.ts.TypeFlags.Void)
1843
+ return "void";
1844
+ if (flags & ts_morph_1.ts.TypeFlags.Never)
1845
+ return "never";
1846
+ if (flags & ts_morph_1.ts.TypeFlags.Any)
1847
+ return "any";
1848
+ if (flags & ts_morph_1.ts.TypeFlags.Unknown)
1849
+ return "unknown";
1850
+ return undefined;
1851
+ }
1777
1852
  // function oldGetInterfaceDeclarationFromExpression(expression: ExpressionWithTypeArguments): InterfaceDeclaration | undefined {
1778
1853
  // // Two cases:
1779
1854
  // // class A implements ImportedInterface, DeclaredInterface {}
@@ -1800,4 +1875,4 @@ function resolveSymbolToInterfaceDeclaration(symbol) {
1800
1875
  // }
1801
1876
  // return interfaceDeclaration;
1802
1877
  // }
1803
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"EntityDictionary.js","sourceRoot":"","sources":["../../src/famix_functions/EntityDictionary.ts"],"names":[],"mappings":"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","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"]}
1878
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"EntityDictionary.js","sourceRoot":"","sources":["../../src/famix_functions/EntityDictionary.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * a function getOrCreateXType takes arguments name: string and element: ts-morph-type and returns a Famix.Type\n * The goal is to keep track of the types (e.g., a method's definedType), for the model.\n * The name doesn't need to be fully qualified (it's the name used in the source code, or the Famix model).\n */\n\n\nimport { ClassDeclaration, ConstructorDeclaration, FunctionDeclaration, Identifier, InterfaceDeclaration, MethodDeclaration, MethodSignature, ModuleDeclaration, PropertyDeclaration, PropertySignature, SourceFile, TypeParameterDeclaration, VariableDeclaration, ParameterDeclaration, Decorator, GetAccessorDeclaration, SetAccessorDeclaration, ImportSpecifier, CommentRange, EnumDeclaration, EnumMember, TypeAliasDeclaration, FunctionExpression, ImportDeclaration, ImportEqualsDeclaration, SyntaxKind, Expression, TypeNode, Scope, ArrowFunction, ExpressionWithTypeArguments, HeritageClause, ts, Type } 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\";\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nimport GraphemeSplitter = require('grapheme-splitter');\nimport * as Helpers from \"./helpers_creation\";\nimport * as FQNFunctions from \"../fqn\";\nimport path from \"path\";\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 | TSMorphParametricType;\n\nexport type TSMorphTypeDeclaration = TypeAliasDeclaration | PropertyDeclaration | PropertySignature | ConstructorDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration | FunctionExpression | ParameterDeclaration | VariableDeclaration | EnumMember | ImportEqualsDeclaration | TSMorphParametricType | TypeParameterDeclaration ;\n\nexport type TSMorphParametricType = ClassDeclaration | InterfaceDeclaration | FunctionDeclaration | MethodDeclaration | ArrowFunction;\n\ntype ParametricVariantType = Famix.ParametricClass | Famix.ParametricInterface | Famix.ParametricFunction | Famix.ParametricMethod;\n\ntype ConcreteElementTSMorphType = ClassDeclaration | InterfaceDeclaration | FunctionDeclaration | MethodDeclaration;\n\nexport type InvocableType = MethodDeclaration | ConstructorDeclaration | GetAccessorDeclaration | SetAccessorDeclaration | FunctionDeclaration | FunctionExpression | ArrowFunction;\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<ModuleDeclaration, 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<TSMorphTypeDeclaration, Famix.Type | Famix.ParameterType>(); // Maps the types declarations to their Famix model\n    private fmxPrimitiveTypeMap = new Map<string, Famix.PrimitiveType>(); // Maps the primitive 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 fmxArrowFunctionMap = new Map<string, Famix.ArrowFunction>; // Maps the function names to their Famix model\n    private fmxParameterMap = new Map<ParameterDeclaration, Famix.Parameter>(); // Maps the parameters to their Famix model\n    private fmxVariableMap = new Map<VariableDeclaration, Famix.Variable>(); // Maps the variables to their Famix model\n    private fmxImportClauseMap = new Map<ImportDeclaration | ImportEqualsDeclaration, Famix.ImportClause>(); // Maps the import clauses to their Famix model\n    private fmxEnumMap = new Map<EnumDeclaration, Famix.Enum>(); // Maps the enum names to their Famix model\n    private fmxInheritanceMap = new Map<string, Famix.Inheritance>(); // Maps the inheritance 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;\n\n            let fileName = node.getSourceFile().getFilePath() as string;\n            if (fileName.startsWith(\"/\")) {\n                fileName = fileName.substring(1);\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        // Famix.Comment is not a named entity (does not have a fullyQualifiedName)\n        if (!(famixElement instanceof Famix.Comment)) {  // must be a named entity\n            // insanity check: named entities should have fullyQualifiedName\n            const fullyQualifiedName = (famixElement as Famix.NamedEntity).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            if (pathInProject.startsWith(\"/\")) {\n                pathInProject = pathInProject.substring(1);\n            }\n\n            fmxIndexFileAnchor.fileName = pathInProject;\n            let sourceStart, sourceEnd\n            // ,sourceLineStart, sourceLineEnd\n            : 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;\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 moduleDeclaration A module\n     * @returns The Famix model of the module\n     */\n    public createOrGetFamixModule(moduleDeclaration: ModuleDeclaration): Famix.Module {\n        if (this.fmxModuleMap.has(moduleDeclaration)) {\n            const rModule = this.fmxModuleMap.get(moduleDeclaration);\n            if (rModule) { \n               return rModule;\n            } else {\n                throw new Error(`Famix module ${moduleDeclaration.getName()} is not found in the module map.`);\n            }\n        }\n\n        const fmxModule = new Famix.Module();\n        const moduleName = moduleDeclaration.getName();\n        fmxModule.name = moduleName;\n        fmxModule.isAmbient = isAmbient(moduleDeclaration);\n        fmxModule.isNamespace = isNamespace(moduleDeclaration);\n        fmxModule.isModule = !fmxModule.isNamespace && !fmxModule.isAmbient;\n\n        initFQN(moduleDeclaration, fmxModule);\n        this.makeFamixIndexFileAnchor(moduleDeclaration, fmxModule);\n\n        this.fmxModuleMap.set(moduleDeclaration, fmxModule);\n\n        this.famixRep.addElement(fmxModule);\n\n        this.fmxElementObjectMap.set(fmxModule,moduleDeclaration);\n        return fmxModule;\n    }\n\n    /**\n     * Creates a Famix alias\n     * @param typeAliasDeclaration An alias\n     * @returns The Famix model of the alias\n     */\n    public createFamixAlias(typeAliasDeclaration: TypeAliasDeclaration): Famix.Alias {\n        let fmxAlias: Famix.Alias;\n        const aliasName = typeAliasDeclaration.getName();\n        //const aliasFullyQualifiedName = a.getType().getText(); // FQNFunctions.getFQN(a);\n        const aliasFullyQualifiedName = FQNFunctions.getFQN(typeAliasDeclaration);\n        const foundAlias = this.fmxAliasMap.get(aliasFullyQualifiedName);\n        if (!foundAlias) {\n            fmxAlias = new Famix.Alias();\n            fmxAlias.name = typeAliasDeclaration.getName();\n            const aliasNameWithGenerics = aliasName + (typeAliasDeclaration.getTypeParameters().length ? (\"<\" + typeAliasDeclaration.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, typeAliasDeclaration.getType(), typeAliasDeclaration);\n            fmxAlias.aliasedEntity = fmxType;\n            initFQN(typeAliasDeclaration, fmxAlias);\n            this.makeFamixIndexFileAnchor(typeAliasDeclaration, 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,typeAliasDeclaration);\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            initFQN(cls, fmxClass);\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                if (param instanceof TypeParameterDeclaration) {\n                    const parameter = this.createOrGetFamixType(param.getText(),param.getType(), param);\n                    concElement.addConcreteParameter(parameter);\n                } else {\n                    logger.warn(`> WARNING: concrete argument ${param.getText()} is not a TypeParameterDeclaration. It is a ${param.getKindName()}.`);\n                }\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.getType(), 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(\n        method: MethodDeclaration | ConstructorDeclaration | MethodSignature | GetAccessorDeclaration | SetAccessorDeclaration,\n        currentCC: { [key: string]: number }\n    ): Famix.Method | Famix.Accessor | Famix.ParametricMethod { \n        // console.log(`\\n=== Creating/Getting Method ===`);\n        // console.log(`Method kind: ${method.getKindName()}`);\n        // console.log(`Method text: ${method.getText().slice(0, 50)}...`);\n        const fqn = FQNFunctions.getFQN(method);\n        // console.log(`Method FQN: ${fqn}`);\n        logger.debug(`Processing method ${fqn}`);\n       \n    \n        let fmxMethod = this.fmxFunctionAndMethodMap.get(fqn) as Famix.Method | Famix.Accessor | Famix.ParametricMethod;\n        if (!fmxMethod) {\n            // console.log('Method not found in map, creating new');\n            const isGeneric = method.getTypeParameters().length > 0;\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            } else {\n                fmxMethod = isGeneric ? new Famix.ParametricMethod() : new Famix.Method();\n            }\n    \n            const isConstructor = method instanceof ConstructorDeclaration;\n            const isSignature = method instanceof MethodSignature;\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)\n                ? !!method.getModifiers().find(x => x.getText() === 'private') : false;\n            fmxMethod.isProtected = (method instanceof MethodDeclaration || method instanceof GetAccessorDeclaration || method instanceof SetAccessorDeclaration)\n                ? !!method.getModifiers().find(x => x.getText() === 'protected') : false;\n            fmxMethod.signature = Helpers.computeSignature(method.getText());\n    \n            const methodName = isConstructor ? \"constructor\" : method.getName();\n            fmxMethod.name = methodName;\n    \n            if (!isConstructor && methodName.startsWith(\"#\")) {\n                fmxMethod.isPrivate = true;\n            }\n            fmxMethod.isPublic = !fmxMethod.isPrivate && !fmxMethod.isProtected;\n    \n            fmxMethod.cyclomaticComplexity = isSignature ? 0 : (currentCC[methodName] || 0);\n            let methodTypeName = this.UNKNOWN_VALUE;\n            try {\n                methodTypeName = method.getReturnType().getText().trim();\n                logger.debug(`Method return type: ${methodTypeName}`);\n            } catch (error) {\n                logger.error(`Failed to get return type for ${fqn}: ${error}`);\n            }\n    \n            const fmxType = this.createOrGetFamixType(methodTypeName, method.getType(), method);\n            // console.log(`Created/retrieved return type with FQN: ${fmxType.fullyQualifiedName}`);\n            fmxMethod.declaredType = fmxType;\n            fmxMethod.numberOfLinesOfCode = method.getEndLineNumber() - method.getStartLineNumber();\n            fmxMethod.numberOfParameters = method.getParameters().length;\n            fmxMethod.numberOfStatements = isSignature ? 0 : method.getStatements().length;\n    \n            // Add to famixRep\n            initFQN(method, fmxMethod);\n            this.famixRep.addElement(fmxMethod);\n            this.makeFamixIndexFileAnchor(method, fmxMethod);\n            this.fmxFunctionAndMethodMap.set(fqn, fmxMethod);\n            logger.debug(`Added method ${fqn} to famixRep`);\n        } else {\n            logger.debug(`Method ${fqn} already exists`);\n        }\n    \n        this.fmxElementObjectMap.set(fmxMethod, method);\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            initFQN(func, fmxFunction);\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.getType(), 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 createOrGetFamixParameter(param: ParameterDeclaration): Famix.Parameter {\n        if (this.fmxParameterMap.has(param)) {\n            const rParameter = this.fmxParameterMap.get(param);\n            if (rParameter) { \n               return rParameter;\n            } else {\n                throw new Error(`Famix parameter ${param.getName()} is not found in the parameter map.`);\n            }\n        }\n\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.getType(), 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        this.fmxParameterMap.set(param, fmxParam);\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    // /**\n    //  * Creates a Famix type in the context of concretizations\n    //  * @param typeName A type name\n    //  * @param element An element\n    //  * @returns The Famix model of the type\n    //  */\n    // public createOrGetFamixConcreteType(element: TypeNode): \n    //     Famix.ParameterType | Famix.PrimitiveType | Famix.Class | Famix.Interface {\n    //         if (this.fmxTypeMap.has(element)) {\n    //             const rType = this.fmxTypeMap.get(element);\n    //             if (rType) { \n    //                return rType;\n    //             } else {\n    //                 throw new Error(`Famix type ${element.getText()} is not found in the type map.`);\n    //             }\n    //         }\n\n    //     const typeParameterDeclaration = element.getSymbol()?.getDeclarations()[0] as TypeParameterDeclaration;\n    //     // const parameterTypeName : string = element.getText();\n    //     const parameterTypeName = getPrimitiveTypeName(element.getType()) || element.getText();\n    //     let fmxParameterType: Famix.Type | Famix.Class | Famix.Interface | undefined = undefined;\n\n    //     // get a TypeReference from a TypeNode\n    //     const typeReference = element.getType();\n    //     // get a TypeDeclaration from a TypeReference\n    //     const typeDeclaration = typeReference.getSymbol()?.getDeclarations()[0] as TSMorphTypeDeclaration;\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 === element.getText() && obj.genericParameters.size>0) {\n    //                     fmxParameterType = obj;\n    //                     isClassOrInterface = true;\n    //                 } \n    //             } else {\n    //                 if (name === element.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 === element.getText() && obj.genericParameters.size>0) {\n    //                     fmxParameterType = obj;\n    //                     isClassOrInterface = true;\n    //                 } \n    //             } else {\n    //                 if (name === element.getText()) {\n    //                     fmxParameterType = obj;\n    //                     isClassOrInterface = true;\n    //                 } \n    //             }   \n    //         });\n    //     }\n\n    //     if(!isClassOrInterface){\n    //         if (!this.fmxTypeMap.has(typeDeclaration)) {    \n    //             // TODO refactor \n    //             if (isPrimitiveType(parameterTypeName)) {\n    //                 fmxParameterType = this.createOrGetFamixPrimitiveType(parameterTypeName);\n    //             } else {\n    //                 fmxParameterType = new Famix.ParameterType();\n    //             } \n    \n    //             fmxParameterType.name = parameterTypeName;\n    //             this.famixRep.addElement(fmxParameterType);\n    //             this.fmxTypeMap.set(typeDeclaration, fmxParameterType);\n    //             this.fmxElementObjectMap.set(fmxParameterType,typeParameterDeclaration);\n    //         }\n    //         else {\n    //             const result = this.fmxTypeMap.get(typeDeclaration);\n    //             if (result) {\n    //                 fmxParameterType = result;\n    //             } else {\n    //                 throw new Error(`Famix type ${typeDeclaration} 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 createOrGetFamixVariable(variable: VariableDeclaration): Famix.Variable {\n        if (this.fmxVariableMap.has(variable)) {\n            const rVariable = this.fmxVariableMap.get(variable);\n            if (rVariable) { \n               return rVariable;\n            } else {\n                throw new Error(`Famix parameter ${variable.getName()} is not found in the variable map.`);\n            }\n        }\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.getType(), 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        this.fmxVariableMap.set(variable, fmxVariable);\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 createOrGetFamixEnum(enumEntity: EnumDeclaration): Famix.Enum {\n        if (this.fmxEnumMap.has(enumEntity)) {\n            const rEnum = this.fmxEnumMap.get(enumEntity);\n            if (rEnum) { \n               return rEnum;\n            } else {\n                throw new Error(`Famix enum ${enumEntity.getName()} is not found in the enum map.`);\n            }\n        }\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        this.fmxEnumMap.set(enumEntity, fmxEnum);\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.getType(), 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(typeNameArg: string, tsMorphType: Type | undefined, element: TSMorphTypeDeclaration): Famix.Type {\n        logger.debug(`Creating (or getting) type: '${tsMorphType?.getText() || \"undefined\"}' of element: '${element?.getText().slice(0, 50)}...' of kind: ${element?.getKindName()}`);\n\n        // convert type to correct primitive name (workaround for unique symbole primitive type)\n        // don't convert to primitive if it's a generic type\n        const typeName = !typeNameArg.includes(\"<\") && tsMorphType && getPrimitiveTypeName(tsMorphType) || typeNameArg;\n\n        if (isPrimitiveType(typeName)) {\n            return this.createOrGetFamixPrimitiveType(typeName);\n        }\n\n        if (element.isKind(SyntaxKind.MethodSignature) || element.isKind(SyntaxKind.MethodDeclaration)) {\n            const methodFQN = FQNFunctions.getFQN(element);\n            const returnTypeFQN = `${methodFQN.replace(/\\[Method(Signature|Declaration)\\]$/, '')}[ReturnType]`;\n            \n            // Check if we already have this return type in the repository\n            const existingType = this.famixRep.getFamixEntityByFullyQualifiedName(returnTypeFQN);\n            if (existingType) {\n                // console.log(`Found existing return type with FQN: ${returnTypeFQN}`);\n                return existingType as Famix.Type;\n            }\n    \n            // console.log(`Creating return type with distinct FQN: ${returnTypeFQN}`);\n            const fmxType = new Famix.Type();\n            fmxType.name = typeName;\n            fmxType.fullyQualifiedName = returnTypeFQN;\n            \n            // Set container (same as method's container)\n            const methodAncestor = Helpers.findTypeAncestor(element);\n            if (methodAncestor) {\n                const ancestorFQN = FQNFunctions.getFQN(methodAncestor);\n                const ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFQN) as Famix.ContainerEntity;\n                if (ancestor) {\n                    fmxType.container = ancestor;\n                }\n            }\n    \n            this.famixRep.addElement(fmxType);\n            this.fmxTypeMap.set(element, fmxType);\n            this.fmxElementObjectMap.set(fmxType, element);\n            return fmxType;\n        }\n\n        const isParametricType =\n            (element instanceof ClassDeclaration && element.getTypeParameters().length > 0) ||\n            (element instanceof InterfaceDeclaration && element.getTypeParameters().length > 0);\n            \n        if (isParametricType) {\n            return this.createOrGetFamixParametricType(typeName, element as TSMorphParametricType);\n        }\n    \n        if (!this.fmxTypeMap.has(element)) {\n            // console.log('Type not found in map, creating new'); \n            let ancestor: Famix.ContainerEntity | undefined;    \n            if (element) {\n                const typeAncestor = Helpers.findTypeAncestor(element);\n                // console.log(`Type ancestor found: ${typeAncestor?.getKindName()}`);\n    \n                if (typeAncestor) {\n                    const ancestorFullyQualifiedName = FQNFunctions.getFQN(typeAncestor);\n                    // console.log(`Ancestor FQN: ${ancestorFullyQualifiedName}`);\n                    ancestor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName) as Famix.ContainerEntity;\n                    if (!ancestor) {\n                        ancestor = this.createOrGetFamixType(typeAncestor.getText(), typeAncestor.getType(), typeAncestor as TSMorphTypeDeclaration);\n                        // console.log('Ancestor not found in repo, creating it');\n                    } else {\n                        console.log(`Found ancestor in famixRep: ${ancestor.fullyQualifiedName}`);\n                    }\n                } else {\n                    console.log(`No type ancestor found for ${typeName} - proceeding without container`);\n                }\n            }\n    \n            const fmxType = new Famix.Type();\n            fmxType.name = typeName;\n            // console.log(`Created new type with name: ${typeName}`);    \n            if (ancestor) {\n                fmxType.container = ancestor;\n            } else {\n                throw new Error(`Ancestor not found for type ${typeName}.`);\n            }\n    \n            initFQN(element, fmxType);\n            // console.log(`Type FQN after init: ${fmxType.fullyQualifiedName}`);\n            this.makeFamixIndexFileAnchor(element, fmxType);\n            this.famixRep.addElement(fmxType);\n            // console.log('Added type to repository');\n            this.fmxTypeMap.set(element, fmxType);\n        } else {\n            const fmxType = this.fmxTypeMap.get(element);\n            if (!fmxType) {\n                throw new Error(`Famix type ${typeName} is not found in the Type map.`);\n            }\n            return fmxType;\n        }\n    \n        const fmxType = this.fmxTypeMap.get(element)!;\n        this.fmxElementObjectMap.set(fmxType, element);\n        return fmxType;\n    }\n\n    /**\n     * Creates or gets a Famix type that is parametric\n     * @param typeName A type name\n     * @param element A ts-morph element\n     * @returns The Famix model of the parameter type\n     */\n    createOrGetFamixParametricType(typeName: string, element: TSMorphParametricType): Famix.Type {\n\n        if (this.fmxTypeMap.has(element) === true) {\n            const result = this.fmxTypeMap.get(element);\n            if (result) {\n                return result;\n            } else {\n                throw new Error(`Famix type ${typeName} is not found (undefined) in the Type map.`);\n            }\n        }\n\n        // A parametric type is a type that has type parameters, e.g., List<T>\n        // In TS it can be a class, an interface, a function, an arrow function, or a method\n\n        // create the Famix Parametric Type (maybe it's just an Interface, etc.)\n        let fmxType: Famix.Type;\n\n        if (element instanceof ClassDeclaration) {\n            fmxType = new Famix.ParametricClass();\n        } else if (element instanceof InterfaceDeclaration) {\n            fmxType = new Famix.ParametricInterface();\n        }\n        // functions and methods are not types \n        // else if (element instanceof FunctionDeclaration) {\n        //     fmxType = new Famix.ParametricFunction();\n        // } else if (element instanceof ArrowFunction) {\n        //     fmxType = new Famix.ParametricArrowFunction();\n        // } else if (element instanceof MethodDeclaration) {\n        //     fmxType = new Famix.ParametricMethod();\n        // } \n        else {\n            throw new Error(`Element is not a class, interface, function, arrow function, or method.`);\n        }\n\n        // const parameters = element.getTypeParameters();\n\n        // // for each parameter, getOrCreate the FamixParameterType\n        // for (const parameter of parameters) {\n        //     this.createOrGetFamixParameterType(parameter.getName(), parameter);\n        // }\n\n        // // TODO: the following code is not correct, it is just a placeholder\n        // const parameterTypeNames = typeName.substring(typeName.indexOf(\"<\") + 1, typeName.indexOf(\">\"))\n        //     .split(\",\").map(s => s.trim());\n        // const baseTypeName = typeName.substring(0, typeName.indexOf(\"<\")).trim();\n        // parameterTypeNames.forEach(parameterTypeName => {\n        //     const fmxParameterType = this.createOrGetFamixParameterType(parameterTypeName, element);\n        //     (fmxType as Famix.ParameterType).addArgument(fmxParameterType);\n        // });\n        // const fmxBaseType = this.createOrGetFamixType(baseTypeName, element);\n\n        // (fmxType as Famix.ParameterType).baseType = fmxBaseType;\n\n        fmxType.name = typeName;\n        initFQN(element, fmxType);\n        this.famixRep.addElement(fmxType);\n        this.fmxTypeMap.set(element, fmxType);\n        return fmxType;\n    }\n\n    /**\n     * Creates a type for a parameter in a parametric type, e.g., T in List<T>\n     * @param parameterTypeName \n     * @param element the TypeScript element (TSMorphParametricType) that the type is associated with\n     * @returns \n     */\n    // createOrGetFamixParameterType(parameterTypeName: string, element: ParameterDeclaration) {\n    //     if (this.fmxTypeMap.has(element)) {\n    //         return this.fmxTypeMap.get(element) as Famix.ParameterType;\n    //     }\n\n    //     // determine if element is a \n    //     const fmxType = new Famix.ParameterType();\n    //     // const parameterTypeNames = typeName.substring(typeName.indexOf(\"<\") + 1, typeName.indexOf(\">\"))\n    //     //     .split(\",\").map(s => s.trim());\n    //     // const baseTypeName = typeName.substring(0, typeName.indexOf(\"<\")).trim();\n    //     // parameterTypeNames.forEach(parameterTypeName => {\n    //     //     const fmxParameterType = this.createOrGetFamixParameterType(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    //     initFQN(element, fmxType);\n    //     this.famixRep.addElement(fmxType);\n    //     this.fmxTypeMap.set(element, fmxType);\n    //     return fmxType;\n    // }\n\n    /**\n     * Creates or gets a Famix primitive type\n     * @param typeName A type name\n     * @returns The Famix model of the primitive type\n     */\n    createOrGetFamixPrimitiveType(typeName: string): Famix.PrimitiveType {\n        let fmxType: Famix.PrimitiveType = new Famix.PrimitiveType();\n        if (!this.fmxPrimitiveTypeMap.has(typeName)) {\n            fmxType = new Famix.PrimitiveType();\n            fmxType.isStub = true;\n            fmxType.name = typeName;\n            fmxType.fullyQualifiedName = typeName + \"[PrimitiveType]\";\n            this.fmxPrimitiveTypeMap.set(typeName, fmxType);\n            this.famixRep.addElement(fmxType);\n        } else {\n            fmxType = this.fmxPrimitiveTypeMap.get(typeName) as Famix.PrimitiveType;\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        if (!nodeReferenceAncestor) {\n            logger.error(`No ancestor found for node '${node.getText()}'`);\n            return;\n        }\n    \n        const ancestorFullyQualifiedName = FQNFunctions.getFQN(nodeReferenceAncestor);\n        const accessor = this.famixRep.getFamixEntityByFullyQualifiedName(ancestorFullyQualifiedName) as Famix.ContainerEntity;\n        if (!accessor) {\n            logger.error(`Ancestor ${ancestorFullyQualifiedName} of kind ${nodeReferenceAncestor.getKindName()} not found.`);\n            return; // Bail out for now\n        } else {\n            logger.debug(`Found accessor to be ${accessor.fullyQualifiedName}.`);\n        }\n    \n        // Ensure accessor is a method, function, script, or module\n        if (!(accessor instanceof Famix.Method) && !(accessor instanceof Famix.ArrowFunction) && !(accessor instanceof Famix.Function) && !(accessor instanceof Famix.ScriptEntity) && !(accessor instanceof Famix.Module)) {\n            logger.error(`Accessor ${accessor.fullyQualifiedName} is not a method, function, etc.`);\n            return;\n        }\n    \n        // Avoid duplicates\n        const foundAccess = this.famixRep.getFamixAccessByAccessorAndVariable(accessor, fmxVar);\n        if (foundAccess) {\n            logger.debug(`FamixAccess already exists for accessor ${accessor.fullyQualifiedName} and variable ${fmxVar.fullyQualifiedName}.`);\n            return;\n        }\n    \n        const fmxAccess = new Famix.Access();\n        fmxAccess.accessor = accessor;\n        fmxAccess.variable = fmxVar;\n        this.famixRep.addElement(fmxAccess);\n        this.fmxElementObjectMap.set(fmxAccess, node);\n        logger.debug(`Created access: ${accessor.fullyQualifiedName} -> ${fmxVar.fullyQualifiedName}`);\n    }\n\n    /**\n     * Creates a Famix invocation\n     * @param nodeReferringToInvocable A node\n     * @param invocable A method or a function\n     * @param id The id of the method or the function\n     */\n    public createFamixInvocation(nodeReferringToInvocable: Identifier, invocable: InvocableType, id: number): void {\n        const fmxInvocable = this.famixRep.getFamixEntityById(id) as Famix.BehavioralEntity;\n        // since the node is in the AST, we need to find the ancestor that is in the Famix model\n        const containerOfNode = Helpers.findAncestor(nodeReferringToInvocable);\n        logger.debug(`Found container (ancestor) ${containerOfNode.getKindName()} for AST node ${nodeReferringToInvocable.getText()}.`);\n        const containerFQN = FQNFunctions.getFQN(containerOfNode);\n        logger.debug(`Found containerFQN ${containerFQN}.`);\n        let sender = this.famixRep.getFamixEntityByFullyQualifiedName(containerFQN) as Famix.ContainerEntity;\n        logger.debug(`Found a sender that matches ${sender.fullyQualifiedName}.`);\n        if (sender instanceof Famix.Type) {\n            // TODO this might be an error in getFamixEntityByFullyQualifiedName\n            logger.debug(`Oops! Sender is a type, which is not valid for an Invocation. Trying to find a container for ${sender.fullyQualifiedName}.`);\n            const senderContainer = sender.container;\n            if (senderContainer) {\n                sender = senderContainer;\n            }\n        }\n        const receiverFullyQualifiedName = FQNFunctions.getFQN(invocable.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(fmxInvocable);\n        fmxInvocation.signature = fmxInvocable.signature;\n\n        this.famixRep.addElement(fmxInvocation);\n\n        this.fmxElementObjectMap.set(fmxInvocation,nodeReferringToInvocable);\n    }\n\n    /**\n     * Creates a Famix inheritance\n     * @param baseClassOrInterface A class or an interface (subclass)\n     * @param inheritedClassOrInterface The inherited class or interface (superclass)\n     */\n    public createOrGetFamixInheritance(baseClassOrInterface: ClassDeclaration | InterfaceDeclaration, inheritedClassOrInterface: ClassDeclaration | InterfaceDeclaration | ExpressionWithTypeArguments): void {\n        logger.debug(`Creating FamixInheritance for ${baseClassOrInterface.getText()} and ${inheritedClassOrInterface.getText()} [${inheritedClassOrInterface.constructor.name}].`);\n        const fmxInheritance = new Famix.Inheritance();\n\n        let subClass: Famix.Class | Famix.Interface | undefined;\n        if (baseClassOrInterface instanceof ClassDeclaration) {\n            subClass = this.createOrGetFamixClass(baseClassOrInterface);\n        } else {\n            subClass = this.createOrGetFamixInterface(baseClassOrInterface);\n        }\n\n        if (!subClass) {\n            throw new Error(`Subclass ${baseClassOrInterface} not found in Class or Interface maps.`);\n        }\n\n        let superClass: Famix.Class | Famix.Interface | undefined;\n\n        if (inheritedClassOrInterface instanceof ClassDeclaration) {\n            superClass = this.createOrGetFamixClass(inheritedClassOrInterface);\n        } else if (inheritedClassOrInterface instanceof InterfaceDeclaration) {\n            superClass = this.createOrGetFamixInterface(inheritedClassOrInterface);\n        } else  {\n            // inheritedClassOrInterface instanceof ExpressionWithTypeArguments\n            // must determine if inheritedClassOrInterface is a class or an interface\n            // then find the declaration, else it's a stub\n\n            const heritageClause = inheritedClassOrInterface.getParent();\n            if (heritageClause instanceof HeritageClause) {\n                // cases: 1) class extends class, 2) class implements interface, 3) interface extends interface\n\n                // class extends class\n                if (heritageClause.getText().startsWith(\"extends\") && baseClassOrInterface instanceof ClassDeclaration) {\n                    const classDeclaration = getInterfaceOrClassDeclarationFromExpression(inheritedClassOrInterface);\n                    if (classDeclaration !== undefined && classDeclaration instanceof ClassDeclaration) {\n                        superClass = this.createOrGetFamixClass(classDeclaration);\n                    } else {\n                        logger.error(`Class declaration not found for ${inheritedClassOrInterface.getText()}.`);\n                        superClass = this.createOrGetFamixClassStub(inheritedClassOrInterface);\n                    }\n                } \n                else if (heritageClause.getText().startsWith(\"implements\") && baseClassOrInterface instanceof ClassDeclaration // class implements interface\n                    || (heritageClause.getText().startsWith(\"extends\") && baseClassOrInterface instanceof InterfaceDeclaration)) { // interface extends interface\n\n                    const interfaceOrClassDeclaration = getInterfaceOrClassDeclarationFromExpression(inheritedClassOrInterface);\n                    if (interfaceOrClassDeclaration !== undefined && interfaceOrClassDeclaration instanceof InterfaceDeclaration) {\n                        superClass = this.createOrGetFamixInterface(interfaceOrClassDeclaration);\n                    } else {\n                        logger.error(`Interface declaration not found for ${inheritedClassOrInterface.getText()}.`);\n                        superClass = this.createOrGetFamixInterfaceStub(inheritedClassOrInterface);\n                    }\n                } else {\n                    // throw new Error(`Parent of ${inheritedClassOrInterface.getText()} is not a class or an interface.`);\n                    logger.error(`Parent of ${inheritedClassOrInterface.getText()} is not a class or an interface.`);\n                    superClass = this.createOrGetFamixInterfaceStub(inheritedClassOrInterface);\n                }\n            } else {\n                throw new Error(`Heritage clause not found for ${inheritedClassOrInterface.getText()}.`);\n            }\n\n        }\n\n        this.fmxElementObjectMap.set(superClass, inheritedClassOrInterface);\n\n        this.makeFamixIndexFileAnchor(inheritedClassOrInterface, superClass);\n\n        this.famixRep.addElement(superClass);\n\n        fmxInheritance.subclass = subClass;\n        fmxInheritance.superclass = superClass;\n\n        this.famixRep.addElement(fmxInheritance);\n        // no FQN for inheritance\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    createOrGetFamixClassStub(unresolvedInheritedClass: ExpressionWithTypeArguments): Famix.Class {\n        // make a FQN for the stub\n        const fqn = FQNFunctions.getFQNUnresolvedInheritedClassOrInterface(unresolvedInheritedClass);\n        logger.debug(`createOrGetFamixClassStub: fqn: ${fqn}`);\n        const fmxClass = this.famixRep.getFamixEntityByFullyQualifiedName(fqn) as Famix.Class;\n        if (fmxClass) {\n            return fmxClass;\n        } else {\n            const stub = new Famix.Class();\n            stub.name = unresolvedInheritedClass.getText();\n            stub.isStub = true;\n            stub.fullyQualifiedName = fqn;\n            this.famixRep.addElement(stub);\n            this.fmxElementObjectMap.set(stub, unresolvedInheritedClass);\n            return stub;\n        }\n    }\n\n    createOrGetFamixInterfaceStub(unresolvedInheritedInterface: ExpressionWithTypeArguments): Famix.Interface {\n        // make a FQN for the stub\n        const fqn = FQNFunctions.getFQNUnresolvedInheritedClassOrInterface(unresolvedInheritedInterface);\n        logger.debug(`createOrGetFamixInterfaceStub: fqn: ${fqn}`);\n        const fmxInterface = this.famixRep.getFamixEntityByFullyQualifiedName(fqn) as Famix.Interface;\n        if (fmxInterface) {\n            return fmxInterface;\n        } else {\n            const stub = new Famix.Interface();\n            stub.name = unresolvedInheritedInterface.getText();\n            stub.isStub = true;\n            stub.fullyQualifiedName = fqn;\n            this.famixRep.addElement(stub);\n            this.fmxElementObjectMap.set(stub, unresolvedInheritedInterface);\n            return stub;\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 oldCreateOrGetFamixImportClause(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        if (importDeclaration && this.fmxImportClauseMap.has(importDeclaration)) {\n            const rImportClause = this.fmxImportClauseMap.get(importDeclaration);\n            if (rImportClause) { \n                logger.debug(`Import clause ${importElement.getText()} already exists in map, skipping.`);\n                return;\n            } else {\n                throw new Error(`Import clause ${importElement.getText()} is not found in the import clause map.`);\n            }\n        }\n    \n        logger.info(`creating a new FamixImportClause for ${importDeclaration?.getText()} in ${importer.getBaseName()}.`);\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        logger.debug(`createFamixImportClause: absolutePath: ${absolutePath}`);\n        logger.debug(`createFamixImportClause: convertToRelativePath: ${this.convertToRelativePath(absolutePath, absolutePathProject)}`);\n        const pathInProject: string = this.convertToRelativePath(absolutePath, absolutePathProject).replace(/\\\\/g, \"/\");\n        logger.debug(`createFamixImportClause: pathInProject: ${pathInProject}`);\n        let pathName = \"{\" + pathInProject + \"}.\";\n        logger.debug(`createFamixImportClause: pathName: ${pathName}`);\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                logger.debug(`Found exported entity: ${pathName}`);\n            }\n            if (importedEntity === undefined) {\n                importedEntity = new Famix.NamedEntity();\n                importedEntity.name = importedEntityName;\n                if (!isInExports) {\n                    importedEntity.isStub = true;\n                }\n                logger.debug(`Creating named entity ${importedEntityName} for ImportSpecifier ${importElement.getText()}`);\n                initFQN(importElement, importedEntity);\n                logger.debug(`Assigned FQN to entity: ${importedEntity.fullyQualifiedName}`);\n                this.makeFamixIndexFileAnchor(importElement, importedEntity);\n                this.famixRep.addElement(importedEntity);\n                logger.debug(`Added entity to repository: ${importedEntity.fullyQualifiedName}`);\n            }\n        }\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            logger.debug(`Assigned FQN to ImportEquals entity: ${importedEntity.fullyQualifiedName}`);\n            this.makeFamixIndexFileAnchor(importElement, importedEntity);\n            const anyType = this.createOrGetFamixType('any', undefined, importDeclaration);\n            (importedEntity as Famix.StructuralEntity).declaredType = anyType;\n        } else {  \n            importedEntityName = importElement.getText();\n            pathName = pathName + (isDefaultExport ? \"defaultExport\" : \"namespaceExport\");\n            importedEntity = new Famix.NamedEntity();\n            importedEntity.name = importedEntityName;\n            initFQN(importElement, importedEntity);\n            logger.debug(`Assigned FQN to default/namespace entity: ${importedEntity.fullyQualifiedName}`);\n            this.makeFamixIndexFileAnchor(importElement, importedEntity);\n        }\n        if (!isInExports) {\n            this.famixRep.addElement(importedEntity);\n            logger.debug(`Added non-exported entity to repository: ${importedEntity.fullyQualifiedName}`);\n        }\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(`ImportClause: ${fmxImportClause.importedEntity?.name} (type=${Helpers.getSubTypeName(fmxImportClause.importedEntity)}) imported by ${fmxImportClause.importingEntity?.name}`);\n    \n        fmxImporter.addOutgoingImport(fmxImportClause);\n        this.famixRep.addElement(fmxImportClause);\n    \n        if (importDeclaration) {\n            this.fmxElementObjectMap.set(fmxImportClause, importDeclaration);\n            this.fmxImportClauseMap.set(importDeclaration, fmxImportClause);\n        }\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 createOrGetFamixArrowFunction(arrowExpression: Expression, currentCC: { [key: string]: number } ): Famix.ArrowFunction | Famix.ParametricArrowFunction {\n        \n        let fmxArrowFunction: Famix.ArrowFunction | Famix.ParametricArrowFunction;\n        const functionFullyQualifiedName = FQNFunctions.getFQN(arrowExpression);\n\n        if (!this.fmxFunctionAndMethodMap.has(functionFullyQualifiedName)) {\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.getReturnType(), 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            this.fmxFunctionAndMethodMap.set(functionFullyQualifiedName, fmxArrowFunction);\n        } else {\n            fmxArrowFunction = this.fmxFunctionAndMethodMap.get(functionFullyQualifiedName) as Famix.ArrowFunction;\n        }\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        const conClassTypeParametersIterator = concreteParameters.values();\n        const 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                        const 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                        const 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: TsMorphNode | 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                                const 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                            const 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        logger.debug(`convertToRelativePath: absolutePath: '${absolutePath}', absolutePathProject: '${absolutePathProject}'`);\n        if (absolutePath.startsWith(absolutePathProject)) {\n            return absolutePath.replace(absolutePathProject, \"\").slice(1);\n        } else if (absolutePath.startsWith(\"/\")) {\n            return absolutePath.slice(1);\n        } else {\n            return absolutePath;\n        }\n    }\n}\n\nexport function isPrimitiveType(typeName: string) {\n    return typeName === \"number\" ||\n        typeName === \"string\" ||\n        typeName === \"boolean\" ||\n        typeName === \"bigint\" ||\n        typeName === \"symbol\" ||\n        typeName === \"unique symbol\" ||\n        typeName === \"undefined\" ||\n        typeName === \"null\" ||\n        typeName === \"any\" ||\n        typeName === \"unknown\" ||\n        typeName === \"never\" ||\n        typeName === \"void\";\n}\n\nfunction initFQN(sourceElement: TSMorphObjectType, famixElement: Famix.SourcedEntity) {\n    // handle special cases where an element is a Type -- need to change its name\n    if (famixElement instanceof Famix.Type && !(sourceElement instanceof CommentRange) && isTypeContext(sourceElement)) {\n        let fqn = FQNFunctions.getFQN(sourceElement);\n        // using regex, replace [blah] with [blahType]\n        fqn = fqn.replace(/\\[([^\\]]+)\\]/g, \"[$1Type]\");\n        logger.debug(\"Setting fully qualified name for \" + famixElement.getJSON() + \" to \" + fqn);\n        famixElement.fullyQualifiedName = fqn;\n        return;\n    }\n    // catch all (except comments)\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\nfunction isTypeContext(sourceElement: TSMorphObjectType): boolean {\n    // Just keep the existing SyntaxKind set as it is\n    const typeContextKinds = new Set([\n        SyntaxKind.Constructor,\n        SyntaxKind.MethodDeclaration,\n        SyntaxKind.FunctionDeclaration,\n        SyntaxKind.FunctionExpression,\n        SyntaxKind.ArrowFunction,\n        SyntaxKind.Parameter,\n        SyntaxKind.VariableDeclaration,\n        SyntaxKind.PropertyDeclaration,\n        SyntaxKind.PropertySignature,\n        SyntaxKind.TypeParameter,\n        SyntaxKind.Identifier,\n        SyntaxKind.Decorator,\n        SyntaxKind.GetAccessor,\n        SyntaxKind.SetAccessor,\n        SyntaxKind.ImportSpecifier,\n        SyntaxKind.EnumDeclaration,\n        SyntaxKind.EnumMember,\n        SyntaxKind.TypeAliasDeclaration,\n        SyntaxKind.ImportDeclaration,\n        SyntaxKind.ExpressionWithTypeArguments\n    ]);\n\n    return typeContextKinds.has(sourceElement.getKind());\n}\n\nfunction getInterfaceOrClassDeclarationFromExpression(expression: ExpressionWithTypeArguments): InterfaceDeclaration | ClassDeclaration | undefined {\n    // Step 1: Get the type of the expression\n    const type = expression.getType();\n\n    // Step 2: Get the symbol associated with the type\n    let symbol = type.getSymbol();\n\n    if (!symbol) {\n        // If symbol is not found, try to get the symbol from the identifier\n        const identifier = expression.getFirstDescendantByKind(SyntaxKind.Identifier);\n        if (!identifier) {\n            throw new Error(`Identifier not found for ${expression.getText()}.`);\n        }\n        symbol = identifier.getSymbol();\n        if (!symbol) {\n            throw new Error(`Symbol not found for ${identifier.getText()}.`);\n        }\n    }\n\n    // Step 3: Resolve the symbol to find the actual declaration\n    const interfaceDeclaration = resolveSymbolToInterfaceOrClassDeclaration(symbol);\n\n    if (!interfaceDeclaration) {\n        logger.error(`Interface declaration not found for ${expression.getText()}.`);\n    }\n\n    return interfaceDeclaration;\n}\n\nimport { Symbol as TSMorphSymbol, Node as TsMorphNode } from \"ts-morph\";\nimport _ from \"lodash\";\n\nfunction resolveSymbolToInterfaceOrClassDeclaration(symbol: TSMorphSymbol): InterfaceDeclaration | ClassDeclaration | undefined {\n    // Get the declarations associated with the symbol\n    const declarations = symbol.getDeclarations();\n\n    // Filter for InterfaceDeclaration or ClassDeclaration\n    const interfaceOrClassDeclaration = declarations.find(\n        declaration => \n            declaration instanceof InterfaceDeclaration || \n            declaration instanceof ClassDeclaration) as InterfaceDeclaration | ClassDeclaration | undefined;\n\n    if (interfaceOrClassDeclaration) {\n        return interfaceOrClassDeclaration;\n    }\n\n    // Handle imports: If the symbol is imported, resolve the import to find the actual declaration\n    for (const declaration of declarations) {\n        if (declaration.getKind() === SyntaxKind.ImportSpecifier) {\n            const importSpecifier = declaration as ImportSpecifier;\n            const importDeclaration = importSpecifier.getImportDeclaration();\n            const moduleSpecifier = importDeclaration.getModuleSpecifierSourceFile();\n\n            if (moduleSpecifier) {\n                const exportedSymbols = moduleSpecifier.getExportSymbols();\n                const exportedSymbol = exportedSymbols.find(symbol => symbol.getName() === importSpecifier.getName());\n                if (exportedSymbol) {\n                    return resolveSymbolToInterfaceOrClassDeclaration(exportedSymbol);\n                }\n            }\n        }\n    }\n    return undefined;\n}\n\n\nexport function getPrimitiveTypeName(type: Type): string | undefined {\n  const flags = type.compilerType.flags;\n\n  if (flags & ts.TypeFlags.String) return \"string\";\n  if (flags & ts.TypeFlags.Number) return \"number\";\n  if (flags & ts.TypeFlags.Boolean) return \"boolean\";\n  if (flags & ts.TypeFlags.BigInt) return \"bigint\";\n  if (flags & ts.TypeFlags.UniqueESSymbol) return \"unique symbol\";\n  if (flags & ts.TypeFlags.ESSymbol) return \"symbol\";\n  if (flags & ts.TypeFlags.Undefined) return \"undefined\";\n  if (flags & ts.TypeFlags.Null) return \"null\";\n  if (flags & ts.TypeFlags.Void) return \"void\";\n  if (flags & ts.TypeFlags.Never) return \"never\";\n  if (flags & ts.TypeFlags.Any) return \"any\";\n  if (flags & ts.TypeFlags.Unknown) return \"unknown\";\n\n  return undefined;\n}\n\n// function oldGetInterfaceDeclarationFromExpression(expression: ExpressionWithTypeArguments): InterfaceDeclaration | undefined {\n//     // Two cases:\n//     // class A implements ImportedInterface, DeclaredInterface {}\n//     const type = expression.getType();\n\n//     // ImportedInterface: type will a symbol \n//     let symbol = type.getAliasSymbol();  // will be defined for imported interfaces\n\n//     if (!symbol) {\n//         // DeclaredInterface: type will be an InterfaceDeclaration on Identifier node that is the child of the ExpressionWithTypeArguments\n//         const identifier = expression.getFirstDescendantByKind(SyntaxKind.Identifier);\n//         if (!identifier) {\n//             throw new Error(`Identifier not found for ${expression.getText()}.`);\n//         }\n//         symbol = identifier.getSymbol();\n//         if (!symbol) {\n//             throw new Error(`Symbol not found for ${identifier.getText()}.`);\n//         }\n//     }\n\n//     // Step 3: Get the declarations associated with the symbol\n//     const declarations = symbol.getDeclarations();\n\n//     // Step 4: Filter for InterfaceDeclaration\n//     const interfaceDeclaration = declarations.find(declaration => declaration instanceof InterfaceDeclaration) as InterfaceDeclaration | undefined;\n\n//     if (!interfaceDeclaration) {\n//         throw new Error(`Interface declaration not found for ${expression.getText()}.`);\n//     }\n\n//     return interfaceDeclaration;\n// }\n"]}