import * as estraverse from 'estraverse'; import * as ESTree from 'estree'; import { TNodeGroup } from './types/TNodeGroup'; import { TNodeObfuscator } from './types/TNodeObfuscator'; import { ICustomNode } from './interfaces/custom-nodes/ICustomNode'; import { INodeObfuscator } from './interfaces/INodeObfuscator'; import { IObfuscator } from './interfaces/IObfuscator'; import { IOptions } from './interfaces/IOptions'; import { IStackTraceData } from './interfaces/stack-trace-analyzer/IStackTraceData'; import { AppendState } from './enums/AppendState'; import { NodeType } from './enums/NodeType'; import { CatchClauseObfuscator } from './node-obfuscators/CatchClauseObfuscator'; import { ConsoleOutputNodesGroup } from './node-groups/ConsoleOutputNodesGroup'; import { DebugProtectionNodesGroup } from './node-groups/DebugProtectionNodesGroup'; import { DomainLockNodesGroup } from './node-groups/DomainLockNodesGroup'; import { FunctionDeclarationObfuscator } from './node-obfuscators/FunctionDeclarationObfuscator'; import { FunctionObfuscator } from './node-obfuscators/FunctionObfuscator'; import { LiteralObfuscator } from './node-obfuscators/LiteralObfuscator'; import { MemberExpressionObfuscator } from './node-obfuscators/MemberExpressionObfuscator'; import { MethodDefinitionObfuscator } from './node-obfuscators/MethodDefinitionObfuscator'; import { Node } from './node/Node'; import { NodeUtils } from './node/NodeUtils'; import { ObjectExpressionObfuscator } from './node-obfuscators/ObjectExpressionObfuscator'; import { SelfDefendingNodesGroup } from './node-groups/SelfDefendingNodesGroup'; import { UnicodeArrayNodesGroup } from './node-groups/UnicodeArrayNodesGroup'; import { VariableDeclarationObfuscator } from './node-obfuscators/VariableDeclarationObfuscator'; import { StackTraceAnalyzer } from './stack-trace-analyzer/StackTraceAnalyzer'; export class Obfuscator implements IObfuscator { /** * @type {TNodeGroup[]} */ private static nodeGroups: TNodeGroup[] = [ DomainLockNodesGroup, SelfDefendingNodesGroup, ConsoleOutputNodesGroup, DebugProtectionNodesGroup, UnicodeArrayNodesGroup ]; /** * @type {Map} */ private static nodeObfuscators: Map = new Map ([ [NodeType.ArrowFunctionExpression, [FunctionObfuscator]], [NodeType.ClassDeclaration, [FunctionDeclarationObfuscator]], [NodeType.CatchClause, [CatchClauseObfuscator]], [NodeType.FunctionDeclaration, [ FunctionDeclarationObfuscator, FunctionObfuscator ]], [NodeType.FunctionExpression, [FunctionObfuscator]], [NodeType.MemberExpression, [MemberExpressionObfuscator]], [NodeType.MethodDefinition, [MethodDefinitionObfuscator]], [NodeType.ObjectExpression, [ObjectExpressionObfuscator]], [NodeType.VariableDeclaration, [VariableDeclarationObfuscator]], [NodeType.Literal, [LiteralObfuscator]] ]); /** * @type {Map} */ private customNodes: Map = new Map (); /** * @type {Map} */ private obfuscatorsCache: Map = new Map (); /** * @type {IOptions} */ private options: IOptions; /** * @param options */ constructor (options: IOptions) { this.options = options; } /** * @param node * @returns {ESTree.Node} */ public obfuscateNode (node: ESTree.Program): ESTree.Node { if (Node.isProgramNode(node) && !node.body.length) { return node; } NodeUtils.parentize(node); const stackTraceData: IStackTraceData[] = new StackTraceAnalyzer(node.body).analyze(); this.initializeCustomNodes(stackTraceData); this.beforeObfuscation(node); this.obfuscate(node); this.afterObfuscation(node); return node; } /** * @param astTree */ private afterObfuscation (astTree: ESTree.Node): void { this.customNodes.forEach((node: ICustomNode) => { if (node.getAppendState() === AppendState.AfterObfuscation) { node.appendNode(astTree); } }); } /** * @param astTree */ private beforeObfuscation (astTree: ESTree.Node): void { this.customNodes.forEach((node: ICustomNode) => { if (node.getAppendState() === AppendState.BeforeObfuscation) { node.appendNode(astTree); } }); }; /** * @param stackTraceData */ private initializeCustomNodes (stackTraceData: IStackTraceData[]): void { let customNodes: [string, ICustomNode][] = []; Obfuscator.nodeGroups.forEach((nodeGroupConstructor: TNodeGroup) => { const nodeGroupNodes: Map | undefined = new nodeGroupConstructor( stackTraceData, this.options ).getNodes(); if (!nodeGroupNodes) { return; } customNodes.push(...nodeGroupNodes); }); this.customNodes = new Map (customNodes); } /** * @param node * @param parentNode */ private initializeNodeObfuscators (node: ESTree.Node, parentNode: ESTree.Node): void { let nodeObfuscators: TNodeObfuscator[] | undefined = Obfuscator.nodeObfuscators.get(node.type); if (!nodeObfuscators) { return; } nodeObfuscators.forEach((obfuscator: TNodeObfuscator) => { let cachedObfuscator: INodeObfuscator|undefined = this.obfuscatorsCache.get(obfuscator); if (!cachedObfuscator) { cachedObfuscator = new obfuscator(this.customNodes, this.options); this.obfuscatorsCache.set(obfuscator, cachedObfuscator); } cachedObfuscator.obfuscateNode(node, parentNode); }); } /** * @param node */ private obfuscate (node: ESTree.Node): void { estraverse.traverse(node, { enter: (node: ESTree.Node, parentNode: ESTree.Node): void => { this.initializeNodeObfuscators(node, parentNode); } }); } }