FunctionControlFlowTransformer.ts 11 KB

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