DeadCodeInjectionTransformer.ts 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  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 { TDeadNodeInjectionCustomNodeFactory } from '../../types/container/custom-nodes/TDeadNodeInjectionCustomNodeFactory';
  6. import { TNodeWithBlockScope } from '../../types/node/TNodeWithBlockScope';
  7. import { ICustomNode } from '../../interfaces/custom-nodes/ICustomNode';
  8. import { IOptions } from '../../interfaces/options/IOptions';
  9. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  10. import { ITransformersRunner } from '../../interfaces/node-transformers/ITransformersRunner';
  11. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  12. import { DeadCodeInjectionCustomNode } from '../../enums/custom-nodes/DeadCodeInjectionCustomNode';
  13. import { NodeTransformer } from '../../enums/node-transformers/NodeTransformer';
  14. import { NodeType } from '../../enums/node/NodeType';
  15. import { TransformationStage } from '../../enums/node-transformers/TransformationStage';
  16. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  17. import { NodeGuards } from '../../node/NodeGuards';
  18. import { Nodes } from '../../node/Nodes';
  19. import { NodeUtils } from '../../node/NodeUtils';
  20. @injectable()
  21. export class DeadCodeInjectionTransformer extends AbstractNodeTransformer {
  22. /**
  23. * @type {string}
  24. */
  25. private static deadCodeInjectionRootAstHostNodeName: string = 'deadCodeInjectionRootAstHostNode';
  26. /**
  27. * @type {number}
  28. */
  29. private static readonly maxNestedBlockStatementsCount: number = 4;
  30. /**
  31. * @type {number}
  32. */
  33. private static readonly minCollectedBlockStatementsCount: number = 5;
  34. /**
  35. * @type {NodeTransformer[]}
  36. */
  37. private static readonly transformersToRenameBlockScopeIdentifiers: NodeTransformer[] = [
  38. NodeTransformer.CatchClauseTransformer,
  39. NodeTransformer.ClassDeclarationTransformer,
  40. NodeTransformer.FunctionDeclarationTransformer,
  41. NodeTransformer.FunctionTransformer,
  42. NodeTransformer.LabeledStatementTransformer,
  43. NodeTransformer.VariableDeclarationTransformer
  44. ];
  45. /**
  46. * @type {Set <BlockStatement>}
  47. */
  48. private readonly deadCodeInjectionRootAstHostNodeSet: Set <ESTree.BlockStatement> = new Set();
  49. /**
  50. * @type {ESTree.BlockStatement[]}
  51. */
  52. private readonly collectedBlockStatements: ESTree.BlockStatement[] = [];
  53. /**
  54. * @type {number}
  55. */
  56. private collectedBlockStatementsTotalLength: number = 0;
  57. /**
  58. * @type {TDeadNodeInjectionCustomNodeFactory}
  59. */
  60. private readonly deadCodeInjectionCustomNodeFactory: TDeadNodeInjectionCustomNodeFactory;
  61. /**
  62. * @type {ITransformersRunner}
  63. */
  64. private readonly transformersRunner: ITransformersRunner;
  65. /**
  66. * @param {TControlFlowCustomNodeFactory} deadCodeInjectionCustomNodeFactory
  67. * @param {ITransformersRunner} transformersRunner
  68. * @param {IRandomGenerator} randomGenerator
  69. * @param {IOptions} options
  70. */
  71. constructor (
  72. @inject(ServiceIdentifiers.Factory__IDeadCodeInjectionCustomNode)
  73. deadCodeInjectionCustomNodeFactory: TDeadNodeInjectionCustomNodeFactory,
  74. @inject(ServiceIdentifiers.ITransformersRunner) transformersRunner: ITransformersRunner,
  75. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  76. @inject(ServiceIdentifiers.IOptions) options: IOptions
  77. ) {
  78. super(randomGenerator, options);
  79. this.deadCodeInjectionCustomNodeFactory = deadCodeInjectionCustomNodeFactory;
  80. this.transformersRunner = transformersRunner;
  81. }
  82. /**
  83. * @param {Node} blockStatementNode
  84. * @returns {boolean}
  85. */
  86. private static isValidBlockStatementNode (blockStatementNode: ESTree.Node): boolean {
  87. const isProhibitedNode: (node: ESTree.Node) => boolean =
  88. (node: ESTree.Node): boolean => NodeGuards.isBreakStatementNode(node) ||
  89. NodeGuards.isContinueStatementNode(node) ||
  90. NodeGuards.isAwaitExpressionNode(node) ||
  91. NodeGuards.isSuperNode(node);
  92. let nestedBlockStatementsCount: number = 0,
  93. isValidBlockStatementNode: boolean = true;
  94. estraverse.traverse(blockStatementNode, {
  95. enter: (node: ESTree.Node): estraverse.VisitorOption | void => {
  96. if (NodeGuards.isBlockStatementNode(node)) {
  97. nestedBlockStatementsCount++;
  98. }
  99. if (
  100. nestedBlockStatementsCount > DeadCodeInjectionTransformer.maxNestedBlockStatementsCount ||
  101. isProhibitedNode(node)
  102. ) {
  103. isValidBlockStatementNode = false;
  104. return estraverse.VisitorOption.Break;
  105. }
  106. }
  107. });
  108. return isValidBlockStatementNode;
  109. }
  110. /**
  111. * @param {TransformationStage} transformationStage
  112. * @returns {IVisitor | null}
  113. */
  114. public getVisitor (transformationStage: TransformationStage): IVisitor | null {
  115. switch (transformationStage) {
  116. case TransformationStage.DeadCodeInjection:
  117. return {
  118. enter: (node: ESTree.Node, parentNode: ESTree.Node | null) => {
  119. if (parentNode && NodeGuards.isProgramNode(node)) {
  120. this.analyzeNode(node, parentNode);
  121. return node;
  122. }
  123. },
  124. leave: (node: ESTree.Node, parentNode: ESTree.Node | null) => {
  125. if (parentNode && NodeGuards.isBlockStatementNode(node)) {
  126. return this.transformNode(node, parentNode);
  127. }
  128. }
  129. };
  130. case TransformationStage.Finalizing:
  131. if (!this.deadCodeInjectionRootAstHostNodeSet.size) {
  132. return null;
  133. }
  134. return {
  135. enter: (node: ESTree.Node, parentNode: ESTree.Node | null) => {
  136. if (parentNode && this.isDeadCodeInjectionRootAstHostNode(node)) {
  137. return this.restoreNode(node, parentNode);
  138. }
  139. }
  140. };
  141. default:
  142. return null;
  143. }
  144. }
  145. /**
  146. * @param {NodeGuards} programNode
  147. * @param {NodeGuards} parentNode
  148. */
  149. public analyzeNode (programNode: ESTree.Node, parentNode: ESTree.Node): void {
  150. estraverse.traverse(programNode, {
  151. enter: (node: ESTree.Node): void => {
  152. if (!NodeGuards.isBlockStatementNode(node)) {
  153. return;
  154. }
  155. let clonedBlockStatementNode: ESTree.BlockStatement = NodeUtils.clone(node);
  156. if (!DeadCodeInjectionTransformer.isValidBlockStatementNode(clonedBlockStatementNode)) {
  157. return;
  158. }
  159. /**
  160. * We should transform identifiers in the dead code block statement to avoid conflicts with original code
  161. */
  162. NodeUtils.parentizeNode(clonedBlockStatementNode, clonedBlockStatementNode);
  163. clonedBlockStatementNode = this.transformersRunner.transform(
  164. clonedBlockStatementNode,
  165. DeadCodeInjectionTransformer.transformersToRenameBlockScopeIdentifiers,
  166. TransformationStage.Obfuscating
  167. );
  168. this.collectedBlockStatements.push(clonedBlockStatementNode);
  169. }
  170. });
  171. this.collectedBlockStatementsTotalLength = this.collectedBlockStatements.length;
  172. }
  173. /**
  174. * @param {BlockStatement} blockStatementNode
  175. * @param {NodeGuards} parentNode
  176. * @returns {NodeGuards | VisitorOption}
  177. */
  178. public transformNode (blockStatementNode: ESTree.BlockStatement, parentNode: ESTree.Node): ESTree.Node | estraverse.VisitorOption {
  179. if (this.collectedBlockStatementsTotalLength < DeadCodeInjectionTransformer.minCollectedBlockStatementsCount) {
  180. return estraverse.VisitorOption.Break;
  181. }
  182. if (!this.collectedBlockStatements.length) {
  183. return estraverse.VisitorOption.Break;
  184. }
  185. if (this.randomGenerator.getMathRandom() > this.options.deadCodeInjectionThreshold) {
  186. return blockStatementNode;
  187. }
  188. const blockScopeOfBlockStatementNode: TNodeWithBlockScope = NodeUtils
  189. .getBlockScopesOfNode(blockStatementNode)[0];
  190. if (blockScopeOfBlockStatementNode.type === NodeType.Program) {
  191. return blockStatementNode;
  192. }
  193. const minInteger: number = 0;
  194. const maxInteger: number = this.collectedBlockStatements.length - 1;
  195. const randomIndex: number = this.randomGenerator.getRandomInteger(minInteger, maxInteger);
  196. const randomBlockStatementNode: ESTree.BlockStatement = this.collectedBlockStatements.splice(randomIndex, 1)[0];
  197. if (randomBlockStatementNode === blockStatementNode) {
  198. return blockStatementNode;
  199. }
  200. return this.replaceBlockStatementNode(blockStatementNode, randomBlockStatementNode, parentNode);
  201. }
  202. /**
  203. * @param {FunctionExpression} deadCodeInjectionRootAstHostNode
  204. * @param {Node} parentNode
  205. * @returns {Node}
  206. */
  207. public restoreNode (deadCodeInjectionRootAstHostNode: ESTree.BlockStatement, parentNode: ESTree.Node): ESTree.Node {
  208. const hostNodeFirstStatement: ESTree.Statement = deadCodeInjectionRootAstHostNode.body[0];
  209. if (!NodeGuards.isFunctionDeclarationNode(hostNodeFirstStatement)) {
  210. throw new Error('Wrong dead code injection root AST host node. Host node should contain `FunctionDeclaration` node');
  211. }
  212. return hostNodeFirstStatement.body;
  213. }
  214. /**
  215. * @param {Node} node
  216. * @returns {boolean}
  217. */
  218. private isDeadCodeInjectionRootAstHostNode (node: ESTree.Node): node is ESTree.BlockStatement {
  219. return NodeGuards.isBlockStatementNode(node) && this.deadCodeInjectionRootAstHostNodeSet.has(node);
  220. }
  221. /**
  222. * @param {BlockStatement} blockStatementNode
  223. * @param {BlockStatement} randomBlockStatementNode
  224. * @param {Node} parentNode
  225. * @returns {BlockStatement}
  226. */
  227. private replaceBlockStatementNode (
  228. blockStatementNode: ESTree.BlockStatement,
  229. randomBlockStatementNode: ESTree.BlockStatement,
  230. parentNode: ESTree.Node
  231. ): ESTree.BlockStatement {
  232. /**
  233. * Should wrap original random block statement node into the parent block statement node (ast root host node)
  234. * with function declaration node. This function declaration node will create block scope for all identifiers
  235. * inside random block statement node and this identifiers won't affect identifiers of the rest AST tree.
  236. */
  237. const deadCodeInjectionRootAstHostNode: ESTree.BlockStatement = Nodes.getBlockStatementNode([
  238. Nodes.getFunctionDeclarationNode(
  239. DeadCodeInjectionTransformer.deadCodeInjectionRootAstHostNodeName,
  240. [],
  241. randomBlockStatementNode
  242. )
  243. ]);
  244. /**
  245. * Should store that host node and then extract random block statement node on the `finalizing` stage
  246. */
  247. this.deadCodeInjectionRootAstHostNodeSet.add(deadCodeInjectionRootAstHostNode);
  248. const blockStatementDeadCodeInjectionCustomNode: ICustomNode = this.deadCodeInjectionCustomNodeFactory(
  249. DeadCodeInjectionCustomNode.BlockStatementDeadCodeInjectionNode
  250. );
  251. blockStatementDeadCodeInjectionCustomNode.initialize(blockStatementNode, deadCodeInjectionRootAstHostNode);
  252. const newBlockStatementNode: ESTree.BlockStatement = <ESTree.BlockStatement>blockStatementDeadCodeInjectionCustomNode.getNode()[0];
  253. NodeUtils.parentizeNode(newBlockStatementNode, parentNode);
  254. return newBlockStatementNode;
  255. }
  256. }