import { inject, injectable, } from 'inversify'; import { ServiceIdentifiers } from '../../container/ServiceIdentifiers'; import * as ESTree from 'estree'; import { TLiteralObfuscatingReplacerFactory } from '../../types/container/node-transformers/TLiteralObfuscatingReplacerFactory'; import { IEscapeSequenceEncoder } from '../../interfaces/utils/IEscapeSequenceEncoder'; import { IOptions } from '../../interfaces/options/IOptions'; import { IRandomGenerator } from '../../interfaces/utils/IRandomGenerator'; import { IStringArrayStorageAnalyzer } from '../../interfaces/analyzers/string-array-storage-analyzer/IStringArrayStorageAnalyzer'; import { IVisitor } from '../../interfaces/node-transformers/IVisitor'; import { LiteralObfuscatingReplacer } from '../../enums/node-transformers/obfuscating-transformers/obfuscating-replacers/LiteralObfuscatingReplacer'; import { NodeTransformationStage } from '../../enums/node-transformers/NodeTransformationStage'; import { AbstractNodeTransformer } from '../AbstractNodeTransformer'; import { NodeFactory } from '../../node/NodeFactory'; import { NodeGuards } from '../../node/NodeGuards'; import { NodeLiteralUtils } from '../../node/NodeLiteralUtils'; import { NodeMetadata } from '../../node/NodeMetadata'; import { NodeUtils } from '../../node/NodeUtils'; @injectable() export class LiteralTransformer extends AbstractNodeTransformer { /** * @type {IEscapeSequenceEncoder} */ private readonly escapeSequenceEncoder: IEscapeSequenceEncoder; /** * @type {TLiteralObfuscatingReplacerFactory} */ private readonly literalObfuscatingReplacerFactory: TLiteralObfuscatingReplacerFactory; /** * @type {IStringArrayStorageAnalyzer} */ private readonly stringArrayStorageAnalyzer: IStringArrayStorageAnalyzer; /** * @param {TLiteralObfuscatingReplacerFactory} literalObfuscatingReplacerFactory * @param {IRandomGenerator} randomGenerator * @param {IOptions} options * @param {IStringArrayStorageAnalyzer} stringArrayStorageAnalyzer * @param {IEscapeSequenceEncoder} escapeSequenceEncoder */ public constructor ( @inject(ServiceIdentifiers.Factory__IObfuscatingReplacer) literalObfuscatingReplacerFactory: TLiteralObfuscatingReplacerFactory, @inject(ServiceIdentifiers.IRandomGenerator) randomGenerator: IRandomGenerator, @inject(ServiceIdentifiers.IOptions) options: IOptions, @inject(ServiceIdentifiers.IStringArrayStorageAnalyzer) stringArrayStorageAnalyzer: IStringArrayStorageAnalyzer, @inject(ServiceIdentifiers.IEscapeSequenceEncoder) escapeSequenceEncoder: IEscapeSequenceEncoder ) { super(randomGenerator, options); this.literalObfuscatingReplacerFactory = literalObfuscatingReplacerFactory; this.stringArrayStorageAnalyzer = stringArrayStorageAnalyzer; this.escapeSequenceEncoder = escapeSequenceEncoder; } /** * @param {NodeTransformationStage} nodeTransformationStage * @returns {IVisitor | null} */ public getVisitor (nodeTransformationStage: NodeTransformationStage): IVisitor | null { switch (nodeTransformationStage) { case NodeTransformationStage.Obfuscating: return { enter: (node: ESTree.Node, parentNode: ESTree.Node | null): ESTree.Node | undefined => { if (NodeGuards.isProgramNode(node)) { this.analyzeNode(node); } if (parentNode && NodeGuards.isLiteralNode(node) && !NodeMetadata.isReplacedLiteral(node)) { return this.transformNode(node, parentNode); } } }; case NodeTransformationStage.Finalizing: return { enter: (node: ESTree.Node, parentNode: ESTree.Node | null): ESTree.Node | undefined => { if (parentNode && NodeGuards.isLiteralNode(node)) { return this.encodeLiteralNodeToEscapeSequence(node, parentNode); } } }; default: return null; } } public analyzeNode (programNode: ESTree.Program): void { this.stringArrayStorageAnalyzer.analyze(programNode); } /** * @param {Literal} literalNode * @param {NodeGuards} parentNode * @returns {NodeGuards} */ public transformNode (literalNode: ESTree.Literal, parentNode: ESTree.Node): ESTree.Node { if (NodeLiteralUtils.isProhibitedLiteralNode(literalNode, parentNode)) { return literalNode; } let newLiteralNode: ESTree.Node; switch (typeof literalNode.value) { case 'boolean': newLiteralNode = this.literalObfuscatingReplacerFactory( LiteralObfuscatingReplacer.BooleanLiteralObfuscatingReplacer ).replace(literalNode); break; case 'number': case 'bigint': newLiteralNode = this.literalObfuscatingReplacerFactory( LiteralObfuscatingReplacer.NumberLiteralObfuscatingReplacer ).replace(literalNode); break; case 'string': newLiteralNode = this.literalObfuscatingReplacerFactory( LiteralObfuscatingReplacer.StringLiteralObfuscatingReplacer ).replace(literalNode); break; default: newLiteralNode = literalNode; } NodeUtils.parentizeNode(newLiteralNode, parentNode); return newLiteralNode; } /** * @param {Literal} literalNode * @param {Node} parentNode * @returns {Literal} */ private encodeLiteralNodeToEscapeSequence ( literalNode: ESTree.Literal, parentNode: ESTree.Node ): ESTree.Literal { if (typeof literalNode.value !== 'string') { return literalNode; } return NodeFactory.literalNode( this.escapeSequenceEncoder.encode(literalNode.value, this.options.unicodeEscapeSequence) ); } }