import * as commander from 'commander'; import * as path from 'path'; import { TStringArrayEncoding } from '../types/TStringArrayEncoding'; import { IObfuscationResult } from '../interfaces/IObfuscationResult'; import { IObfuscatorOptions } from '../interfaces/IObfuscatorOptions'; import { SourceMapMode } from '../enums/SourceMapMode'; import { StringArrayEncoding } from '../enums/StringArrayEncoding'; import { DEFAULT_PRESET } from '../preset-options/DefaultPreset'; import { CLIUtils } from './CLIUtils'; import { JavaScriptObfuscator } from '../JavaScriptObfuscator'; import { Utils } from '../Utils'; export class JavaScriptObfuscatorCLI { /** * @type {string[]} */ private arguments: string[]; /** * @type {commander.ICommand} */ private commands: commander.ICommand; /** * @type {string} */ private data: string = ''; /** * @type {string} */ private inputPath: string; /** * @type {string[]} */ private rawArguments: string[]; /** * @param argv */ constructor (argv: string[]) { this.rawArguments = argv; this.arguments = this.rawArguments.slice(2); } /** * @returns {string} */ private static getBuildVersion (): string { return CLIUtils.getPackageConfig().version; } /** * @param value * @returns {boolean} */ private static parseBoolean (value: string): boolean { return value === 'true' || value === '1'; } /** * @param value * @returns {string} */ private static parseSourceMapMode (value: string): string { let availableMode: boolean = Object .keys(SourceMapMode) .some((key: string): boolean => { return SourceMapMode[key] === value; }); if (!availableMode) { throw new ReferenceError('Invalid value of `--sourceMapMode` option'); } return value; } /** * @param value * @returns {TStringArrayEncoding} */ private static parseStringArrayEncoding (value: string): TStringArrayEncoding { switch (value) { case 'true': case '1': case StringArrayEncoding.base64: return true; case StringArrayEncoding.rc4: return StringArrayEncoding.rc4; default: return false; } } public run (): void { this.configureCommands(); if (!this.arguments.length || Utils.arrayContains(this.arguments, '--help')) { this.commands.outputHelp(); return; } this.inputPath = this.arguments[0]; CLIUtils.validateInputPath(this.inputPath); this.getData(); this.processData(); } /** * @returns {IObfuscatorOptions} */ private buildOptions (): IObfuscatorOptions { let obfuscatorOptions: IObfuscatorOptions = {}, availableOptions: string[] = Object.keys(DEFAULT_PRESET); for (const option in this.commands) { if (!this.commands.hasOwnProperty(option)) { continue; } if (!Utils.arrayContains(availableOptions, option)) { continue; } obfuscatorOptions[option] = (this.commands)[option]; } return Object.assign({}, DEFAULT_PRESET, obfuscatorOptions); } private configureCommands (): void { this.commands = new commander.Command() .version(JavaScriptObfuscatorCLI.getBuildVersion(), '-v, --version') .usage(' [options]') .option('-o, --output ', 'Output path for obfuscated code') .option('--compact ', 'Disable one line output code compacting', JavaScriptObfuscatorCLI.parseBoolean) .option('--controlFlowFlattening ', 'Enables control flow flattening', JavaScriptObfuscatorCLI.parseBoolean) .option('--debugProtection ', 'Disable browser Debug panel (can cause DevTools enabled browser freeze)', JavaScriptObfuscatorCLI.parseBoolean) .option('--debugProtectionInterval ', 'Disable browser Debug panel even after page was loaded (can cause DevTools enabled browser freeze)', JavaScriptObfuscatorCLI.parseBoolean) .option('--disableConsoleOutput ', 'Allow console.log, console.info, console.error and console.warn messages output into browser console', JavaScriptObfuscatorCLI.parseBoolean) .option('--domainLock ', 'Blocks the execution of the code in domains that do not match the passed RegExp patterns (comma separated)', (val: string) => val.split(',')) .option('--reservedNames ', 'Disable obfuscation of variable names, function names and names of function parameters that match the passed RegExp patterns (comma separated)', (val: string) => val.split(',')) .option('--rotateStringArray ', 'Disable rotation of unicode array values during obfuscation', JavaScriptObfuscatorCLI.parseBoolean) .option('--seed ', 'Sets seed for random generator. This is useful for creating repeatable results.', parseFloat) .option('--selfDefending ', 'Disables self-defending for obfuscated code', JavaScriptObfuscatorCLI.parseBoolean) .option('--sourceMap ', 'Enables source map generation', JavaScriptObfuscatorCLI.parseBoolean) .option('--sourceMapBaseUrl ', 'Sets base url to the source map import url when `--sourceMapMode=separate`') .option('--sourceMapFileName ', 'Sets file name for output source map when `--sourceMapMode=separate`') .option( '--sourceMapMode [inline, separate]', 'Specify source map output mode', JavaScriptObfuscatorCLI.parseSourceMapMode ) .option('--stringArray ', 'Disables gathering of all literal strings into an array and replacing every literal string with an array call', JavaScriptObfuscatorCLI.parseBoolean) .option('--stringArrayEncoding [true, false, base64, rc4]', 'Encodes all strings in strings array using base64 or rc4 (this option can slow down your code speed', JavaScriptObfuscatorCLI.parseStringArrayEncoding) .option('--stringArrayThreshold ', 'The probability that the literal string will be inserted into stringArray (Default: 0.8, Min: 0, Max: 1)', parseFloat) .option('--unicodeEscapeSequence ', 'Allows to enable/disable string conversion to unicode escape sequence', JavaScriptObfuscatorCLI.parseBoolean) .parse(this.rawArguments); this.commands.on('--help', () => { console.log(' Examples:\n'); console.log(' %> javascript-obfuscator in.js --compact true --selfDefending false'); console.log(' %> javascript-obfuscator in.js --output out.js --compact true --selfDefending false'); console.log(''); }); } private getData (): void { this.data = CLIUtils.readFile(this.inputPath); } private processData (): void { let options: IObfuscatorOptions = this.buildOptions(), outputCodePath: string = CLIUtils.getOutputCodePath((this.commands).output, this.inputPath); if (options.sourceMap) { this.processDataWithSourceMap(outputCodePath, options); } else { this.processDataWithoutSourceMap(outputCodePath, options); } } /** * @param outputCodePath * @param options */ private processDataWithoutSourceMap (outputCodePath: string, options: IObfuscatorOptions): void { let obfuscatedCode: string = JavaScriptObfuscator.obfuscate(this.data, options).getObfuscatedCode(); CLIUtils.writeFile(outputCodePath, obfuscatedCode); } /** * @param outputCodePath * @param options */ private processDataWithSourceMap (outputCodePath: string, options: IObfuscatorOptions): void { let outputSourceMapPath: string = CLIUtils.getOutputSourceMapPath( outputCodePath, options.sourceMapFileName || '' ); options.sourceMapFileName = path.basename(outputSourceMapPath); const obfuscationResult: IObfuscationResult = JavaScriptObfuscator.obfuscate(this.data, options); CLIUtils.writeFile(outputCodePath, obfuscationResult.getObfuscatedCode()); if (options.sourceMapMode === 'separate' && obfuscationResult.getSourceMap()) { CLIUtils.writeFile(outputSourceMapPath, obfuscationResult.getSourceMap()); } } }