123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195 |
- import { inject, injectable, } from 'inversify';
- import { ServiceIdentifiers } from '../../container/ServiceIdentifiers';
- import * as ESTree from 'estree';
- import { IOptions } from '../../interfaces/options/IOptions';
- import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator';
- import { IVisitor } from '../../interfaces/node-transformers/IVisitor';
- import { NodeTransformer } from '../../enums/node-transformers/NodeTransformer';
- import { NodeTransformationStage } from '../../enums/node-transformers/NodeTransformationStage';
- import { AbstractNodeTransformer } from '../AbstractNodeTransformer';
- import { NodeFactory } from '../../node/NodeFactory';
- import { NodeGuards } from '../../node/NodeGuards';
- import { NodeUtils } from '../../node/NodeUtils';
- import { StringUtils } from '../../utils/StringUtils';
- @injectable()
- export class EvalCallExpressionTransformer extends AbstractNodeTransformer {
- /**
- * @type {NodeTransformer.NodeTransformer[]}
- */
- public readonly runAfter: NodeTransformer[] = [
- NodeTransformer.EscapeSequenceTransformer,
- NodeTransformer.ParentificationTransformer,
- NodeTransformer.VariablePreserveTransformer
- ];
- /**
- * @type {Set <FunctionExpression>}
- */
- private readonly evalRootAstHostNodeSet: Set <ESTree.FunctionExpression> = new Set();
- /**
- * @param {IRandomGenerator} randomGenerator
- * @param {IOptions} options
- */
- public constructor (
- @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator,
- @inject(ServiceIdentifiers.IOptions) options: IOptions
- ) {
- super(randomGenerator, options);
- }
- /**
- * @param {Expression | SpreadElement} node
- * @returns {string | null}
- */
- private static extractEvalStringFromCallExpressionArgument (node: ESTree.Expression | ESTree.SpreadElement): string | null {
- if (NodeGuards.isLiteralNode(node)) {
- return EvalCallExpressionTransformer
- .extractEvalStringFromLiteralNode(node);
- }
- if (NodeGuards.isTemplateLiteralNode(node)) {
- return EvalCallExpressionTransformer
- .extractEvalStringFromTemplateLiteralNode(node);
- }
- return null;
- }
- /**
- * @param {Literal} node
- * @returns {string | null}
- */
- private static extractEvalStringFromLiteralNode (node: ESTree.Literal): string | null {
- return typeof node.value === 'string' ? node.value : null;
- }
- /**
- * @param {TemplateLiteral} node
- * @returns {string | null}
- */
- private static extractEvalStringFromTemplateLiteralNode (node: ESTree.TemplateLiteral): string | null {
- const quasis: ESTree.TemplateElement[] = node.quasis;
- const allowedQuasisLength: number = 1;
- if (quasis.length !== allowedQuasisLength || node.expressions.length) {
- return null;
- }
- return quasis[0].value.cooked;
- }
- /**
- * @param {NodeTransformationStage} nodeTransformationStage
- * @returns {IVisitor | null}
- */
- public getVisitor (nodeTransformationStage: NodeTransformationStage): IVisitor | null {
- switch (nodeTransformationStage) {
- case NodeTransformationStage.Preparing:
- return {
- enter: (node: ESTree.Node, parentNode: ESTree.Node | null): ESTree.Node | undefined => {
- if (
- parentNode
- && NodeGuards.isCallExpressionNode(node)
- && NodeGuards.isIdentifierNode(node.callee)
- && node.callee.name === 'eval'
- ) {
- return this.transformNode(node, parentNode);
- }
- }
- };
- case NodeTransformationStage.Finalizing:
- if (!this.evalRootAstHostNodeSet.size) {
- return null;
- }
- return {
- leave: (node: ESTree.Node, parentNode: ESTree.Node | null): ESTree.Node | undefined => {
- if (parentNode && this.isEvalRootAstHostNode(node)) {
- return this.restoreNode(node, parentNode);
- }
- }
- };
- default:
- return null;
- }
- }
- /**
- * @param {CallExpression} callExpressionNode
- * @param {Node} parentNode
- * @returns {Node}
- */
- public transformNode (callExpressionNode: ESTree.CallExpression, parentNode: ESTree.Node): ESTree.Node {
- const callExpressionFirstArgument: ESTree.Expression | ESTree.SpreadElement | undefined = callExpressionNode.arguments[0];
- if (!callExpressionFirstArgument) {
- return callExpressionNode;
- }
- const evalString: string | null = EvalCallExpressionTransformer
- .extractEvalStringFromCallExpressionArgument(callExpressionFirstArgument);
- if (!evalString) {
- return callExpressionNode;
- }
- let ast: ESTree.Statement[];
- // wrapping into try-catch to prevent parsing of incorrect `eval` string
- try {
- ast = NodeUtils.convertCodeToStructure(evalString);
- } catch {
- return callExpressionNode;
- }
- /**
- * we should wrap AST-tree into the parent function expression node (ast root host node).
- * This function expression node will help to correctly transform AST-tree.
- */
- const evalRootAstHostNode: ESTree.FunctionExpression = NodeFactory
- .functionExpressionNode([], NodeFactory.blockStatementNode(ast));
- NodeUtils.parentizeAst(evalRootAstHostNode);
- NodeUtils.parentizeNode(evalRootAstHostNode, parentNode);
- /**
- * we should store that host node and then extract AST-tree on the `finalizing` stage
- */
- this.evalRootAstHostNodeSet.add(evalRootAstHostNode);
- return evalRootAstHostNode;
- }
- /**
- * @param {FunctionExpression} evalRootAstHostNode
- * @param {Node} parentNode
- * @returns {Node}
- */
- public restoreNode (evalRootAstHostNode: ESTree.FunctionExpression, parentNode: ESTree.Node): ESTree.Node {
- const targetAst: ESTree.Statement[] = evalRootAstHostNode.body.body;
- const obfuscatedCode: string = NodeUtils.convertStructureToCode(targetAst);
- return NodeFactory.callExpressionNode(
- NodeFactory.identifierNode('eval'),
- [
- NodeFactory.literalNode(StringUtils.escapeJsString(obfuscatedCode))
- ]
- );
- }
- /**
- * @param {Node} node
- * @returns {boolean}
- */
- private isEvalRootAstHostNode (node: ESTree.Node): node is ESTree.FunctionExpression {
- return NodeGuards.isFunctionExpressionNode(node) && this.evalRootAstHostNodeSet.has(node);
- }
- }
|