StringArrayScopeCallsWrapperTransformer.ts 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. import { inject, injectable, } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import * as ESTree from 'estree';
  4. import { TInitialData } from '../../types/TInitialData';
  5. import { TNodeWithLexicalScope } from '../../types/node/TNodeWithLexicalScope';
  6. import { TNodeWithLexicalScopeAndStatements } from '../../types/node/TNodeWithLexicalScopeAndStatements';
  7. import { TStringArrayEncoding } from '../../types/options/TStringArrayEncoding';
  8. import { TStringArrayScopeCallsWrapperDataByEncoding } from '../../types/node-transformers/string-array-transformers/TStringArrayScopeCallsWrapperDataByEncoding';
  9. import { TStringArrayTransformerCustomNodeFactory } from '../../types/container/custom-nodes/TStringArrayTransformerCustomNodeFactory';
  10. import { ICustomNode } from '../../interfaces/custom-nodes/ICustomNode';
  11. import { IOptions } from '../../interfaces/options/IOptions';
  12. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  13. import { IStringArrayScopeCallsWrapperData } from '../../interfaces/node-transformers/string-array-transformers/IStringArrayScopeCallsWrapperData';
  14. import { IStringArrayScopeCallsWrapperDataStorage } from '../../interfaces/storages/string-array-transformers/IStringArrayScopeCallsWrapperDataStorage';
  15. import { IStringArrayStorage } from '../../interfaces/storages/string-array-transformers/IStringArrayStorage';
  16. import { IVisitedLexicalScopeNodesStackStorage } from '../../interfaces/storages/string-array-transformers/IVisitedLexicalScopeNodesStackStorage';
  17. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  18. import { NodeTransformationStage } from '../../enums/node-transformers/NodeTransformationStage';
  19. import { StringArrayTransformerCustomNode } from '../../enums/custom-nodes/StringArrayTransformerCustomNode';
  20. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  21. import { NodeAppender } from '../../node/NodeAppender';
  22. import { NodeGuards } from '../../node/NodeGuards';
  23. import { StringArrayScopeCallsWrapperNode } from '../../custom-nodes/string-array-nodes/StringArrayScopeCallsWrapperNode';
  24. @injectable()
  25. export class StringArrayScopeCallsWrapperTransformer extends AbstractNodeTransformer {
  26. /**
  27. * @type {IStringArrayStorage}
  28. */
  29. private readonly stringArrayStorage: IStringArrayStorage;
  30. /**
  31. * @type {IStringArrayScopeCallsWrapperDataStorage}
  32. */
  33. private readonly stringArrayScopeCallsWrapperDataStorage: IStringArrayScopeCallsWrapperDataStorage;
  34. /**
  35. * @type {TStringArrayTransformerCustomNodeFactory}
  36. */
  37. private readonly stringArrayTransformerCustomNodeFactory: TStringArrayTransformerCustomNodeFactory;
  38. /**
  39. * @type {IVisitedLexicalScopeNodesStackStorage}
  40. */
  41. private readonly visitedLexicalScopeNodesStackStorage: IVisitedLexicalScopeNodesStackStorage;
  42. /**
  43. * @param {IRandomGenerator} randomGenerator
  44. * @param {IOptions} options
  45. * @param {IVisitedLexicalScopeNodesStackStorage} visitedLexicalScopeNodesStackStorage
  46. * @param {IStringArrayStorage} stringArrayStorage
  47. * @param {IStringArrayScopeCallsWrapperDataStorage} stringArrayScopeCallsWrapperDataStorage
  48. * @param {TStringArrayTransformerCustomNodeFactory} stringArrayTransformerCustomNodeFactory
  49. */
  50. public constructor (
  51. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  52. @inject(ServiceIdentifiers.IOptions) options: IOptions,
  53. @inject(ServiceIdentifiers.IVisitedLexicalScopeNodesStackStorage) visitedLexicalScopeNodesStackStorage: IVisitedLexicalScopeNodesStackStorage,
  54. @inject(ServiceIdentifiers.IStringArrayStorage) stringArrayStorage: IStringArrayStorage,
  55. @inject(ServiceIdentifiers.IStringArrayScopeCallsWrapperDataStorage) stringArrayScopeCallsWrapperDataStorage: IStringArrayScopeCallsWrapperDataStorage,
  56. @inject(ServiceIdentifiers.Factory__IStringArrayTransformerCustomNode)
  57. stringArrayTransformerCustomNodeFactory: TStringArrayTransformerCustomNodeFactory
  58. ) {
  59. super(randomGenerator, options);
  60. this.visitedLexicalScopeNodesStackStorage = visitedLexicalScopeNodesStackStorage;
  61. this.stringArrayStorage = stringArrayStorage;
  62. this.stringArrayScopeCallsWrapperDataStorage = stringArrayScopeCallsWrapperDataStorage;
  63. this.stringArrayTransformerCustomNodeFactory = stringArrayTransformerCustomNodeFactory;
  64. }
  65. /**
  66. * @param {NodeTransformationStage} nodeTransformationStage
  67. * @returns {IVisitor | null}
  68. */
  69. public getVisitor (nodeTransformationStage: NodeTransformationStage): IVisitor | null {
  70. switch (nodeTransformationStage) {
  71. case NodeTransformationStage.StringArray:
  72. return {
  73. enter: (node: ESTree.Node): void => {
  74. if (NodeGuards.isNodeWithLexicalScopeAndStatements(node)) {
  75. this.onLexicalScopeNodeEnter(node);
  76. }
  77. },
  78. leave: (node: ESTree.Node): ESTree.Node | undefined => {
  79. if (NodeGuards.isNodeWithLexicalScopeAndStatements(node)) {
  80. this.onLexicalScopeNodeLeave();
  81. return this.transformNode(node);
  82. }
  83. }
  84. };
  85. default:
  86. return null;
  87. }
  88. }
  89. /**
  90. * @param {TNodeWithLexicalScopeAndStatements} lexicalScopeNode
  91. * @returns {TNodeWithLexicalScopeAndStatements}
  92. */
  93. public transformNode (lexicalScopeNode: TNodeWithLexicalScopeAndStatements): TNodeWithLexicalScopeAndStatements {
  94. if (!this.options.stringArrayWrappersCount) {
  95. return lexicalScopeNode;
  96. }
  97. const lexicalScopeBodyNode: ESTree.Program | ESTree.BlockStatement =
  98. NodeGuards.isProgramNode(lexicalScopeNode)
  99. ? lexicalScopeNode
  100. : lexicalScopeNode.body;
  101. const stringArrayScopeCallsWrapperDataByEncoding: TStringArrayScopeCallsWrapperDataByEncoding | null =
  102. this.stringArrayScopeCallsWrapperDataStorage.get(lexicalScopeNode) ?? null;
  103. if (!stringArrayScopeCallsWrapperDataByEncoding) {
  104. return lexicalScopeNode;
  105. }
  106. const stringArrayScopeCallsWrapperDataList: (IStringArrayScopeCallsWrapperData | undefined)[] =
  107. Object.values(stringArrayScopeCallsWrapperDataByEncoding);
  108. // iterates over data for each encoding type
  109. for (const stringArrayScopeCallsWrapperData of stringArrayScopeCallsWrapperDataList) {
  110. if (!stringArrayScopeCallsWrapperData) {
  111. continue;
  112. }
  113. const {encoding, names} = stringArrayScopeCallsWrapperData;
  114. const namesLength: number = names.length;
  115. /**
  116. * Iterates over each name of scope wrapper name
  117. * Reverse iteration appends wrappers at index `0` at the correct order
  118. */
  119. for (let i = namesLength - 1; i >= 0; i--) {
  120. const stringArrayScopeCallsWrapperName: string = names[i];
  121. const upperStringArrayCallsWrapperName: string = this.getUpperStringArrayCallsWrapperName(encoding);
  122. const stringArrayScopeCallsWrapperNode: ICustomNode<TInitialData<StringArrayScopeCallsWrapperNode>> =
  123. this.stringArrayTransformerCustomNodeFactory(
  124. StringArrayTransformerCustomNode.StringArrayScopeCallsWrapperNode
  125. );
  126. stringArrayScopeCallsWrapperNode.initialize(
  127. stringArrayScopeCallsWrapperName,
  128. upperStringArrayCallsWrapperName
  129. );
  130. NodeAppender.prepend(
  131. lexicalScopeBodyNode,
  132. stringArrayScopeCallsWrapperNode.getNode()
  133. );
  134. }
  135. }
  136. return lexicalScopeNode;
  137. }
  138. /**
  139. * @param {TStringArrayEncoding} encoding
  140. * @returns {string}
  141. */
  142. private getRootStringArrayCallsWrapperName (encoding: TStringArrayEncoding): string {
  143. return this.stringArrayStorage.getStorageCallsWrapperName(encoding);
  144. }
  145. /**
  146. * @param {TStringArrayEncoding} encoding
  147. * @returns {string}
  148. */
  149. private getUpperStringArrayCallsWrapperName (encoding: TStringArrayEncoding): string {
  150. const rootStringArrayCallsWrapperName: string = this.getRootStringArrayCallsWrapperName(encoding);
  151. if (!this.options.stringArrayWrappersChainedCalls) {
  152. return rootStringArrayCallsWrapperName;
  153. }
  154. const parentLexicalScope: TNodeWithLexicalScope | undefined = this.visitedLexicalScopeNodesStackStorage.getLastElement();
  155. if (!parentLexicalScope) {
  156. return rootStringArrayCallsWrapperName;
  157. }
  158. const parentLexicalScopeDataByEncoding = this.stringArrayScopeCallsWrapperDataStorage
  159. .get(parentLexicalScope) ?? null;
  160. const parentLexicalScopeNames: string[] | null = parentLexicalScopeDataByEncoding?.[encoding]?.names ?? null;
  161. return parentLexicalScopeNames?.length
  162. ? this.randomGenerator
  163. .getRandomGenerator()
  164. .pickone(parentLexicalScopeNames)
  165. : rootStringArrayCallsWrapperName;
  166. }
  167. /**
  168. * @param {TNodeWithLexicalScopeAndStatements} lexicalScopeNode
  169. */
  170. private onLexicalScopeNodeEnter (lexicalScopeNode: TNodeWithLexicalScopeAndStatements): void {
  171. this.visitedLexicalScopeNodesStackStorage.push(lexicalScopeNode);
  172. }
  173. private onLexicalScopeNodeLeave (): void {
  174. this.visitedLexicalScopeNodesStackStorage.pop();
  175. }
  176. }