EvalCallExpressionTransformer.ts 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. import { inject, injectable, } from 'inversify';
  2. import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
  3. import * as ESTree from 'estree';
  4. import jsStringEscape from 'js-string-escape';
  5. import { IOptions } from '../../interfaces/options/IOptions';
  6. import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
  7. import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
  8. import { NodeTransformer } from '../../enums/node-transformers/NodeTransformer';
  9. import { NodeTransformationStage } from '../../enums/node-transformers/NodeTransformationStage';
  10. import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
  11. import { NodeFactory } from '../../node/NodeFactory';
  12. import { NodeGuards } from '../../node/NodeGuards';
  13. import { NodeUtils } from '../../node/NodeUtils';
  14. @injectable()
  15. export class EvalCallExpressionTransformer extends AbstractNodeTransformer {
  16. /**
  17. * @type {NodeTransformer.ParentificationTransformer[]}
  18. */
  19. public readonly runAfter: NodeTransformer[] = [
  20. NodeTransformer.EscapeSequenceTransformer,
  21. NodeTransformer.ParentificationTransformer,
  22. NodeTransformer.VariablePreserveTransformer
  23. ];
  24. /**
  25. * @type {Set <FunctionExpression>}
  26. */
  27. private readonly evalRootAstHostNodeSet: Set <ESTree.FunctionExpression> = new Set();
  28. /**
  29. * @param {IRandomGenerator} randomGenerator
  30. * @param {IOptions} options
  31. */
  32. public constructor (
  33. @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
  34. @inject(ServiceIdentifiers.IOptions) options: IOptions
  35. ) {
  36. super(randomGenerator, options);
  37. }
  38. /**
  39. * @param {Expression | SpreadElement} node
  40. * @returns {string | null}
  41. */
  42. private static extractEvalStringFromCallExpressionArgument (node: ESTree.Expression | ESTree.SpreadElement): string | null {
  43. if (NodeGuards.isLiteralNode(node)) {
  44. return EvalCallExpressionTransformer
  45. .extractEvalStringFromLiteralNode(node);
  46. }
  47. if (NodeGuards.isTemplateLiteralNode(node)) {
  48. return EvalCallExpressionTransformer
  49. .extractEvalStringFromTemplateLiteralNode(node);
  50. }
  51. return null;
  52. }
  53. /**
  54. * @param {Literal} node
  55. * @returns {string | null}
  56. */
  57. private static extractEvalStringFromLiteralNode (node: ESTree.Literal): string | null {
  58. return typeof node.value === 'string' ? node.value : null;
  59. }
  60. /**
  61. * @param {TemplateLiteral} node
  62. * @returns {string | null}
  63. */
  64. private static extractEvalStringFromTemplateLiteralNode (node: ESTree.TemplateLiteral): string | null {
  65. const quasis: ESTree.TemplateElement[] = node.quasis;
  66. const allowedQuasisLength: number = 1;
  67. if (quasis.length !== allowedQuasisLength || node.expressions.length) {
  68. return null;
  69. }
  70. return quasis[0].value.cooked;
  71. }
  72. /**
  73. * @param {NodeTransformationStage} nodeTransformationStage
  74. * @returns {IVisitor | null}
  75. */
  76. public getVisitor (nodeTransformationStage: NodeTransformationStage): IVisitor | null {
  77. switch (nodeTransformationStage) {
  78. case NodeTransformationStage.Preparing:
  79. return {
  80. enter: (node: ESTree.Node, parentNode: ESTree.Node | null): ESTree.Node | undefined => {
  81. if (
  82. parentNode
  83. && NodeGuards.isCallExpressionNode(node)
  84. && NodeGuards.isIdentifierNode(node.callee)
  85. && node.callee.name === 'eval'
  86. ) {
  87. return this.transformNode(node, parentNode);
  88. }
  89. }
  90. };
  91. case NodeTransformationStage.Finalizing:
  92. if (!this.evalRootAstHostNodeSet.size) {
  93. return null;
  94. }
  95. return {
  96. leave: (node: ESTree.Node, parentNode: ESTree.Node | null): ESTree.Node | undefined => {
  97. if (parentNode && this.isEvalRootAstHostNode(node)) {
  98. return this.restoreNode(node, parentNode);
  99. }
  100. }
  101. };
  102. default:
  103. return null;
  104. }
  105. }
  106. /**
  107. * @param {CallExpression} callExpressionNode
  108. * @param {Node} parentNode
  109. * @returns {Node}
  110. */
  111. public transformNode (callExpressionNode: ESTree.CallExpression, parentNode: ESTree.Node): ESTree.Node {
  112. const callExpressionFirstArgument: ESTree.Expression | ESTree.SpreadElement | undefined = callExpressionNode.arguments[0];
  113. if (!callExpressionFirstArgument) {
  114. return callExpressionNode;
  115. }
  116. const evalString: string | null = EvalCallExpressionTransformer
  117. .extractEvalStringFromCallExpressionArgument(callExpressionFirstArgument);
  118. if (!evalString) {
  119. return callExpressionNode;
  120. }
  121. let ast: ESTree.Statement[];
  122. // wrapping into try-catch to prevent parsing of incorrect `eval` string
  123. try {
  124. ast = NodeUtils.convertCodeToStructure(evalString);
  125. } catch {
  126. return callExpressionNode;
  127. }
  128. /**
  129. * we should wrap AST-tree into the parent function expression node (ast root host node).
  130. * This function expression node will help to correctly transform AST-tree.
  131. */
  132. const evalRootAstHostNode: ESTree.FunctionExpression = NodeFactory
  133. .functionExpressionNode([], NodeFactory.blockStatementNode(ast));
  134. NodeUtils.parentizeAst(evalRootAstHostNode);
  135. NodeUtils.parentizeNode(evalRootAstHostNode, parentNode);
  136. /**
  137. * we should store that host node and then extract AST-tree on the `finalizing` stage
  138. */
  139. this.evalRootAstHostNodeSet.add(evalRootAstHostNode);
  140. return evalRootAstHostNode;
  141. }
  142. /**
  143. * @param {FunctionExpression} evalRootAstHostNode
  144. * @param {Node} parentNode
  145. * @returns {Node}
  146. */
  147. public restoreNode (evalRootAstHostNode: ESTree.FunctionExpression, parentNode: ESTree.Node): ESTree.Node {
  148. const targetAst: ESTree.Statement[] = evalRootAstHostNode.body.body;
  149. const obfuscatedCode: string = NodeUtils.convertStructureToCode(targetAst);
  150. return NodeFactory.callExpressionNode(
  151. NodeFactory.identifierNode('eval'),
  152. [
  153. NodeFactory.literalNode(jsStringEscape(obfuscatedCode))
  154. ]
  155. );
  156. }
  157. /**
  158. * @param {Node} node
  159. * @returns {boolean}
  160. */
  161. private isEvalRootAstHostNode (node: ESTree.Node): node is ESTree.FunctionExpression {
  162. return NodeGuards.isFunctionExpressionNode(node) && this.evalRootAstHostNodeSet.has(node);
  163. }
  164. }