FunctionControlFlowTransformer.ts 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import { injectable, inject } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import * as estraverse from 'estraverse';
  4. import * as ESTree from 'estree';
  5. import { TControlFlowCustomNodeFactory } from '../../types/container/custom-nodes/TControlFlowCustomNodeFactory';
  6. import { TControlFlowReplacerFactory } from '../../types/container/node-transformers/TControlFlowReplacerFactory';
  7. import { TControlFlowStorageFactory } from '../../types/container/node-transformers/TControlFlowStorageFactory';
  8. import { TNodeWithBlockStatement } from '../../types/node/TNodeWithBlockStatement';
  9. import { ICustomNode } from '../../interfaces/custom-nodes/ICustomNode';
  10. import { IOptions } from '../../interfaces/options/IOptions';
  11. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  12. import { IStorage } from '../../interfaces/storages/IStorage';
  13. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  14. import { ControlFlowCustomNode } from '../../enums/container/custom-nodes/ControlFlowCustomNode';
  15. import { NodeType } from '../../enums/node/NodeType';
  16. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  17. import { Node } from '../../node/Node';
  18. import { NodeAppender } from '../../node/NodeAppender';
  19. import { ControlFlowReplacer } from '../../enums/container/node-transformers/ControlFlowReplacer';
  20. import { NodeUtils } from '../../node/NodeUtils';
  21. @injectable()
  22. export class FunctionControlFlowTransformer extends AbstractNodeTransformer {
  23. /**
  24. * @type {Map <string, ControlFlowReplacer>}
  25. */
  26. private static readonly controlFlowReplacersMap: Map <string, ControlFlowReplacer> = new Map([
  27. [NodeType.BinaryExpression, ControlFlowReplacer.BinaryExpressionControlFlowReplacer],
  28. [NodeType.CallExpression, ControlFlowReplacer.CallExpressionControlFlowReplacer],
  29. [NodeType.LogicalExpression, ControlFlowReplacer.LogicalExpressionControlFlowReplacer],
  30. [NodeType.Literal, ControlFlowReplacer.StringLiteralControlFlowReplacer]
  31. ]);
  32. /**
  33. * @type {number}
  34. */
  35. private static readonly hostNodeSearchMinDepth: number = 0;
  36. /**
  37. * @type {number}
  38. */
  39. private static readonly hostNodeSearchMaxDepth: number = 2;
  40. /**
  41. * @type {Map<ESTree.Node, IStorage<ICustomNode>>}
  42. */
  43. private readonly controlFlowData: Map <ESTree.Node, IStorage<ICustomNode>> = new Map();
  44. /**
  45. * @type {Set<ESTree.Function>}
  46. */
  47. private readonly visitedFunctionNodes: Set<ESTree.Function> = new Set();
  48. /**
  49. * @type {Set<TNodeWithBlockStatement>}
  50. */
  51. private readonly hostNodesWithControlFlowNode: Set<TNodeWithBlockStatement> = new Set();
  52. /**
  53. * @type {TControlFlowReplacerFactory}
  54. */
  55. private readonly controlFlowReplacerFactory: TControlFlowReplacerFactory;
  56. /**
  57. * @type {TControlFlowStorageFactory}
  58. */
  59. private readonly controlFlowStorageFactory: TControlFlowStorageFactory;
  60. /**
  61. * @type {TControlFlowCustomNodeFactory}
  62. */
  63. private readonly controlFlowCustomNodeFactory: TControlFlowCustomNodeFactory;
  64. /**
  65. * @param {TControlFlowStorageFactory} controlFlowStorageFactory
  66. * @param {TControlFlowReplacerFactory} controlFlowReplacerFactory
  67. * @param {TControlFlowCustomNodeFactory} controlFlowCustomNodeFactory
  68. * @param {IRandomGenerator} randomGenerator
  69. * @param {IOptions} options
  70. */
  71. constructor (
  72. @inject(ServiceIdentifiers.Factory__TControlFlowStorage)
  73. controlFlowStorageFactory: TControlFlowStorageFactory,
  74. @inject(ServiceIdentifiers.Factory__IControlFlowReplacer)
  75. controlFlowReplacerFactory: TControlFlowReplacerFactory,
  76. @inject(ServiceIdentifiers.Factory__IControlFlowCustomNode)
  77. controlFlowCustomNodeFactory: TControlFlowCustomNodeFactory,
  78. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  79. @inject(ServiceIdentifiers.IOptions) options: IOptions
  80. ) {
  81. super(randomGenerator, options);
  82. this.controlFlowStorageFactory = controlFlowStorageFactory;
  83. this.controlFlowReplacerFactory = controlFlowReplacerFactory;
  84. this.controlFlowCustomNodeFactory = controlFlowCustomNodeFactory;
  85. }
  86. /**
  87. * @return {IVisitor}
  88. */
  89. public getVisitor (): IVisitor {
  90. return {
  91. leave: (node: ESTree.Node, parentNode: ESTree.Node) => {
  92. if (
  93. Node.isFunctionDeclarationNode(node) ||
  94. Node.isFunctionExpressionNode(node) ||
  95. Node.isArrowFunctionExpressionNode(node)
  96. ) {
  97. return this.transformNode(node, parentNode);
  98. }
  99. }
  100. };
  101. }
  102. /**
  103. * @param {Function} functionNode
  104. * @param {Node} parentNode
  105. * @returns {Function}
  106. */
  107. public transformNode (functionNode: ESTree.Function, parentNode: ESTree.Node): ESTree.Function {
  108. this.visitedFunctionNodes.add(functionNode);
  109. if (!Node.isBlockStatementNode(functionNode.body)) {
  110. return functionNode;
  111. }
  112. const hostNode: TNodeWithBlockStatement = this.getHostNode(functionNode.body);
  113. const controlFlowStorage: IStorage<ICustomNode> = this.getControlFlowStorage(hostNode);
  114. this.controlFlowData.set(hostNode, controlFlowStorage);
  115. this.transformFunctionBody(functionNode.body, controlFlowStorage);
  116. if (!controlFlowStorage.getLength()) {
  117. return functionNode;
  118. }
  119. const controlFlowStorageCustomNode: ICustomNode = this.controlFlowCustomNodeFactory(
  120. ControlFlowCustomNode.ControlFlowStorageNode
  121. );
  122. controlFlowStorageCustomNode.initialize(controlFlowStorage);
  123. NodeAppender.prependNode(hostNode, controlFlowStorageCustomNode.getNode());
  124. this.hostNodesWithControlFlowNode.add(hostNode);
  125. return functionNode;
  126. }
  127. /**
  128. * @param {TNodeWithBlockStatement} hostNode
  129. * @returns {IStorage<ICustomNode>}
  130. */
  131. private getControlFlowStorage (hostNode: TNodeWithBlockStatement): IStorage<ICustomNode> {
  132. const controlFlowStorage: IStorage <ICustomNode> = this.controlFlowStorageFactory();
  133. if (this.controlFlowData.has(hostNode)) {
  134. if (this.hostNodesWithControlFlowNode.has(hostNode)) {
  135. hostNode.body.shift();
  136. }
  137. const hostControlFlowStorage: IStorage<ICustomNode> = <IStorage<ICustomNode>>this.controlFlowData.get(hostNode);
  138. controlFlowStorage.mergeWith(hostControlFlowStorage, true);
  139. }
  140. return controlFlowStorage;
  141. }
  142. /**
  143. * @param {BlockStatement} functionNodeBody
  144. * @returns {TNodeWithBlockStatement}
  145. */
  146. private getHostNode (functionNodeBody: ESTree.BlockStatement): TNodeWithBlockStatement {
  147. const blockScopesOfNode: TNodeWithBlockStatement[] = NodeUtils.getBlockScopesOfNode(functionNodeBody);
  148. if (blockScopesOfNode.length === 1) {
  149. return functionNodeBody;
  150. } else {
  151. blockScopesOfNode.pop();
  152. }
  153. if (blockScopesOfNode.length > FunctionControlFlowTransformer.hostNodeSearchMinDepth) {
  154. blockScopesOfNode.splice(0, FunctionControlFlowTransformer.hostNodeSearchMinDepth);
  155. }
  156. if (blockScopesOfNode.length > FunctionControlFlowTransformer.hostNodeSearchMaxDepth) {
  157. blockScopesOfNode.length = FunctionControlFlowTransformer.hostNodeSearchMaxDepth;
  158. }
  159. return this.randomGenerator.getRandomGenerator().pickone(blockScopesOfNode);
  160. }
  161. /**
  162. * @param {Node} node
  163. * @returns {boolean}
  164. */
  165. private isVisitedFunctionNode (node: ESTree.Node): boolean {
  166. return (
  167. Node.isFunctionDeclarationNode(node) ||
  168. Node.isFunctionExpressionNode(node) ||
  169. Node.isArrowFunctionExpressionNode(node)
  170. ) && this.visitedFunctionNodes.has(node);
  171. }
  172. /**
  173. * @param {BlockStatement} functionNodeBody
  174. * @param {IStorage<ICustomNode>} controlFlowStorage
  175. */
  176. private transformFunctionBody (functionNodeBody: ESTree.BlockStatement, controlFlowStorage: IStorage<ICustomNode>): void {
  177. estraverse.replace(functionNodeBody, {
  178. enter: (node: ESTree.Node, parentNode: ESTree.Node): any => {
  179. if (this.isVisitedFunctionNode(node)) {
  180. return estraverse.VisitorOption.Skip;
  181. }
  182. if (!FunctionControlFlowTransformer.controlFlowReplacersMap.has(node.type)) {
  183. return node;
  184. }
  185. if (this.randomGenerator.getMathRandom() > this.options.controlFlowFlatteningThreshold) {
  186. return node;
  187. }
  188. const controlFlowReplacerName: ControlFlowReplacer = <ControlFlowReplacer>FunctionControlFlowTransformer
  189. .controlFlowReplacersMap.get(node.type);
  190. if (controlFlowReplacerName === undefined) {
  191. return node;
  192. }
  193. return {
  194. ...this.controlFlowReplacerFactory(controlFlowReplacerName).replace(node, parentNode, controlFlowStorage),
  195. parentNode
  196. };
  197. }
  198. });
  199. }
  200. }