FunctionControlFlowTransformer.ts 10.0 KB

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