FunctionControlFlowTransformer.ts 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  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 { TControlFlowStorageFactory } from '../../types/container/node-transformers/TControlFlowStorageFactory';
  8. import {
  9. TControlFlowStorageFactoryCreator
  10. } from '../../types/container/node-transformers/TControlFlowStorageFactoryCreator';
  11. import { TInitialData } from '../../types/TInitialData';
  12. import { TNodeWithStatements } from '../../types/node/TNodeWithStatements';
  13. import { IControlFlowStorage } from '../../interfaces/storages/control-flow-transformers/IControlFlowStorage';
  14. import { ICustomNode } from '../../interfaces/custom-nodes/ICustomNode';
  15. import { IOptions } from '../../interfaces/options/IOptions';
  16. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  17. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  18. import { ControlFlowCustomNode } from '../../enums/custom-nodes/ControlFlowCustomNode';
  19. import {
  20. ControlFlowReplacer
  21. } from '../../enums/node-transformers/control-flow-transformers/control-flow-replacers/ControlFlowReplacer';
  22. import { ControlFlowStorage } from '../../enums/storages/ControlFlowStorage';
  23. import { NodeType } from '../../enums/node/NodeType';
  24. import { NodeTransformationStage } from '../../enums/node-transformers/NodeTransformationStage';
  25. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  26. import {
  27. ControlFlowStorageNode
  28. } from '../../custom-nodes/control-flow-flattening-nodes/control-flow-storage-nodes/ControlFlowStorageNode';
  29. import { NodeAppender } from '../../node/NodeAppender';
  30. import { NodeGuards } from '../../node/NodeGuards';
  31. import { NodeMetadata } from '../../node/NodeMetadata';
  32. import { NodeStatementUtils } from '../../node/NodeStatementUtils';
  33. import { NodeUtils } from '../../node/NodeUtils';
  34. @injectable()
  35. export class FunctionControlFlowTransformer extends AbstractNodeTransformer {
  36. /**
  37. * @type {number}
  38. */
  39. private static readonly hostNodeSearchMinDepth: number = 0;
  40. /**
  41. * @type {number}
  42. */
  43. private static readonly hostNodeSearchMaxDepth: number = 2;
  44. /**
  45. * @type {Map <string, ControlFlowReplacer>}
  46. */
  47. protected readonly controlFlowReplacersMap: Map <string, ControlFlowReplacer> = new Map([
  48. [NodeType.BinaryExpression, ControlFlowReplacer.BinaryExpressionControlFlowReplacer],
  49. [NodeType.CallExpression, ControlFlowReplacer.CallExpressionControlFlowReplacer],
  50. [NodeType.LogicalExpression, ControlFlowReplacer.LogicalExpressionControlFlowReplacer],
  51. [NodeType.Literal, ControlFlowReplacer.StringLiteralControlFlowReplacer]
  52. ]);
  53. /**
  54. * @type {WeakMap<TNodeWithStatements, IControlFlowStorage>}
  55. */
  56. protected readonly controlFlowData: WeakMap <TNodeWithStatements, IControlFlowStorage> = new WeakMap();
  57. /**
  58. * @type {WeakMap<TNodeWithStatements, VariableDeclaration>}
  59. */
  60. protected readonly hostNodesWithControlFlowNode: WeakMap<TNodeWithStatements, ESTree.VariableDeclaration> = new WeakMap();
  61. /**
  62. * @type {TControlFlowReplacerFactory}
  63. */
  64. protected readonly controlFlowReplacerFactory: TControlFlowReplacerFactory;
  65. /**
  66. * @type {TControlFlowStorageFactory}
  67. */
  68. protected controlFlowStorageFactory: TControlFlowStorageFactory;
  69. /**
  70. * @type {TControlFlowCustomNodeFactory}
  71. */
  72. protected readonly controlFlowCustomNodeFactory: TControlFlowCustomNodeFactory;
  73. /**
  74. * @type {WeakSet<ESTree.Function>}
  75. */
  76. protected readonly visitedFunctionNodes: WeakSet<ESTree.Function> = new WeakSet();
  77. /**
  78. * @param {TControlFlowStorageFactoryCreator} controlFlowStorageFactoryCreator
  79. * @param {TControlFlowReplacerFactory} controlFlowReplacerFactory
  80. * @param {TControlFlowCustomNodeFactory} controlFlowCustomNodeFactory
  81. * @param {IRandomGenerator} randomGenerator
  82. * @param {IOptions} options
  83. */
  84. public constructor (
  85. @inject(ServiceIdentifiers.Factory__TControlFlowStorage)
  86. controlFlowStorageFactoryCreator: TControlFlowStorageFactoryCreator,
  87. @inject(ServiceIdentifiers.Factory__IControlFlowReplacer)
  88. controlFlowReplacerFactory: TControlFlowReplacerFactory,
  89. @inject(ServiceIdentifiers.Factory__IControlFlowCustomNode)
  90. controlFlowCustomNodeFactory: TControlFlowCustomNodeFactory,
  91. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  92. @inject(ServiceIdentifiers.IOptions) options: IOptions
  93. ) {
  94. super(randomGenerator, options);
  95. this.controlFlowStorageFactory = controlFlowStorageFactoryCreator(ControlFlowStorage.FunctionControlFlowStorage);
  96. this.controlFlowReplacerFactory = controlFlowReplacerFactory;
  97. this.controlFlowCustomNodeFactory = controlFlowCustomNodeFactory;
  98. }
  99. /**
  100. * @param {NodeTransformationStage} nodeTransformationStage
  101. * @returns {IVisitor | null}
  102. */
  103. public getVisitor (nodeTransformationStage: NodeTransformationStage): IVisitor | null {
  104. if (!this.options.controlFlowFlattening) {
  105. return null;
  106. }
  107. switch (nodeTransformationStage) {
  108. case NodeTransformationStage.ControlFlowFlattening:
  109. return {
  110. leave: (
  111. node: ESTree.Node,
  112. parentNode: ESTree.Node | null
  113. ): ESTree.Node | estraverse.VisitorOption | void => {
  114. if (parentNode && NodeGuards.isFunctionNode(node)) {
  115. return this.transformNode(node, parentNode);
  116. }
  117. }
  118. };
  119. default:
  120. return null;
  121. }
  122. }
  123. /**
  124. * @param {Function} functionNode
  125. * @param {Node} parentNode
  126. * @returns {Function}
  127. */
  128. public transformNode (functionNode: ESTree.Function, parentNode: ESTree.Node): ESTree.Function {
  129. this.visitedFunctionNodes.add(functionNode);
  130. if (!NodeGuards.isBlockStatementNode(functionNode.body)) {
  131. return functionNode;
  132. }
  133. const hostNode: TNodeWithStatements = this.getHostNode(functionNode.body);
  134. const controlFlowStorage: IControlFlowStorage = this.getControlFlowStorage(hostNode);
  135. this.transformFunctionBody(functionNode, controlFlowStorage);
  136. if (!controlFlowStorage.getLength()) {
  137. return functionNode;
  138. }
  139. const controlFlowStorageNode: ESTree.VariableDeclaration = this.getControlFlowStorageNode(controlFlowStorage);
  140. this.appendControlFlowStorageNode(hostNode, controlFlowStorageNode);
  141. return functionNode;
  142. }
  143. /**
  144. * @param {BlockStatement} functionNode
  145. * @param {IControlFlowStorage} controlFlowStorage
  146. */
  147. protected transformFunctionBody (functionNode: ESTree.Function, controlFlowStorage: IControlFlowStorage): void {
  148. estraverse.replace(functionNode.body, {
  149. enter: (node: ESTree.Node, parentNode: ESTree.Node | null): estraverse.VisitorOption | ESTree.Node =>
  150. this.transformFunctionBodyNode(node, parentNode, functionNode, controlFlowStorage)
  151. });
  152. }
  153. /**
  154. * @param {Node} node
  155. * @param {Node | null} parentNode
  156. * @param {Function} functionNode
  157. * @param {IControlFlowStorage} controlFlowStorage
  158. * @returns {ESTraverse.VisitorOption | Node}
  159. */
  160. protected transformFunctionBodyNode (
  161. node: ESTree.Node,
  162. parentNode: ESTree.Node | null,
  163. functionNode: ESTree.Function,
  164. controlFlowStorage: IControlFlowStorage
  165. ): estraverse.VisitorOption | ESTree.Node {
  166. const shouldSkipTraverse = !parentNode
  167. || NodeMetadata.isIgnoredNode(node)
  168. || this.isVisitedFunctionNode(node);
  169. if (shouldSkipTraverse) {
  170. return estraverse.VisitorOption.Skip;
  171. }
  172. const controlFlowReplacerName: ControlFlowReplacer | null = this.controlFlowReplacersMap.get(node.type)
  173. ?? null;
  174. if (!controlFlowReplacerName) {
  175. return node;
  176. }
  177. if (!this.isAllowedTransformationByThreshold()) {
  178. return node;
  179. }
  180. const replacedNode: ESTree.Node = this.controlFlowReplacerFactory(controlFlowReplacerName)
  181. .replace(
  182. node,
  183. parentNode,
  184. controlFlowStorage
  185. );
  186. NodeUtils.parentizeNode(replacedNode, parentNode);
  187. return replacedNode;
  188. }
  189. /**
  190. * @param {BlockStatement} functionNodeBody
  191. * @returns {TNodeWithStatements}
  192. */
  193. protected getHostNode (functionNodeBody: ESTree.BlockStatement): TNodeWithStatements {
  194. const blockScopesOfNode: TNodeWithStatements[] = NodeStatementUtils.getParentNodesWithStatements(functionNodeBody);
  195. if (blockScopesOfNode.length === 1) {
  196. return functionNodeBody;
  197. } else {
  198. blockScopesOfNode.pop();
  199. }
  200. if (blockScopesOfNode.length > FunctionControlFlowTransformer.hostNodeSearchMinDepth) {
  201. blockScopesOfNode.splice(0, FunctionControlFlowTransformer.hostNodeSearchMinDepth);
  202. }
  203. if (blockScopesOfNode.length > FunctionControlFlowTransformer.hostNodeSearchMaxDepth) {
  204. blockScopesOfNode.length = FunctionControlFlowTransformer.hostNodeSearchMaxDepth;
  205. }
  206. return this.randomGenerator.getRandomGenerator().pickone(blockScopesOfNode);
  207. }
  208. /**
  209. * @param {TNodeWithStatements} hostNode
  210. * @returns {TControlFlowStorage}
  211. */
  212. protected getControlFlowStorage (hostNode: TNodeWithStatements): IControlFlowStorage {
  213. let controlFlowStorage: IControlFlowStorage;
  214. const hostControlFlowStorage: IControlFlowStorage | null = this.controlFlowData.get(hostNode) ?? null;
  215. if (!hostControlFlowStorage) {
  216. controlFlowStorage = this.controlFlowStorageFactory();
  217. } else {
  218. const existingControlFlowStorageNode: ESTree.VariableDeclaration | null =
  219. this.hostNodesWithControlFlowNode.get(hostNode) ?? null;
  220. if (existingControlFlowStorageNode) {
  221. NodeAppender.remove(hostNode, existingControlFlowStorageNode);
  222. }
  223. controlFlowStorage = hostControlFlowStorage;
  224. }
  225. this.controlFlowData.set(hostNode, controlFlowStorage);
  226. return controlFlowStorage;
  227. }
  228. /**
  229. * @param {IControlFlowStorage} controlFlowStorage
  230. * @returns {VariableDeclaration}
  231. */
  232. protected getControlFlowStorageNode (controlFlowStorage: IControlFlowStorage): ESTree.VariableDeclaration {
  233. const controlFlowStorageCustomNode: ICustomNode<TInitialData<ControlFlowStorageNode>> =
  234. this.controlFlowCustomNodeFactory(ControlFlowCustomNode.ControlFlowStorageNode);
  235. controlFlowStorageCustomNode.initialize(controlFlowStorage);
  236. const controlFlowStorageNode: ESTree.Node = controlFlowStorageCustomNode.getNode()[0];
  237. if (!NodeGuards.isVariableDeclarationNode(controlFlowStorageNode)) {
  238. throw new Error('`controlFlowStorageNode` should contain `VariableDeclaration` node with control flow storage object');
  239. }
  240. return controlFlowStorageNode;
  241. }
  242. /**
  243. * @param {TNodeWithStatements} hostNode
  244. * @param {VariableDeclaration} controlFlowStorageNode
  245. */
  246. protected appendControlFlowStorageNode (
  247. hostNode: TNodeWithStatements,
  248. controlFlowStorageNode: ESTree.VariableDeclaration
  249. ): void {
  250. NodeUtils.parentizeAst(controlFlowStorageNode);
  251. NodeAppender.prepend(hostNode, [controlFlowStorageNode]);
  252. this.hostNodesWithControlFlowNode.set(hostNode, controlFlowStorageNode);
  253. }
  254. /**
  255. * @param {NodeGuards} node
  256. * @returns {boolean}
  257. */
  258. protected isVisitedFunctionNode (node: ESTree.Node): boolean {
  259. return NodeGuards.isFunctionNode(node) && this.visitedFunctionNodes.has(node);
  260. }
  261. /**
  262. * @returns {boolean}
  263. */
  264. protected isAllowedTransformationByThreshold (): boolean {
  265. return this.randomGenerator.getMathRandom() <= this.options.controlFlowFlatteningThreshold;
  266. }
  267. }