123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360 |
- import * as commander from 'commander';
- import * as packageJson from 'pjson';
- import * as path from 'path';
- import { TInputCLIOptions } from '../types/options/TInputCLIOptions';
- import { TInputOptions } from '../types/options/TInputOptions';
- import { TObject } from '../types/TObject';
- import { TSourceCodeData } from '../types/cli/TSourceCodeData';
- import { IFileData } from '../interfaces/cli/IFileData';
- import { IInitializable } from '../interfaces/IInitializable';
- import { IObfuscationResult } from '../interfaces/IObfuscationResult';
- import { initializable } from '../decorators/Initializable';
- import { DEFAULT_PRESET } from '../options/presets/Default';
- import { ArraySanitizer } from './sanitizers/ArraySanitizer';
- import { BooleanSanitizer } from './sanitizers/BooleanSanitizer';
- import { IdentifierNamesGeneratorSanitizer } from './sanitizers/IdentifierNamesGeneratorSanitizer';
- import { ObfuscationTargetSanitizer } from './sanitizers/ObfuscatingTargetSanitizer';
- import { SourceMapModeSanitizer } from './sanitizers/SourceMapModeSanitizer';
- import { StringArrayEncodingSanitizer } from './sanitizers/StringArrayEncodingSanitizer';
- import { CLIUtils } from './utils/CLIUtils';
- import { JavaScriptObfuscator } from '../JavaScriptObfuscatorFacade';
- import { SourceCodeReader } from './utils/SourceCodeReader';
- export class JavaScriptObfuscatorCLI implements IInitializable {
- /**
- * @type {string[]}
- */
- public static readonly availableInputExtensions: string[] = [
- '.js'
- ];
- /**
- * @type {BufferEncoding}
- */
- public static readonly encoding: BufferEncoding = 'utf8';
- /**
- * @type {string}
- */
- public static obfuscatedFilePrefix: string = '-obfuscated';
- /**
- * @type {string[]}
- */
- private readonly arguments: string[];
- /**
- * @type {string[]}
- */
- private readonly rawArguments: string[];
- /**
- * @type {commander.CommanderStatic}
- */
- @initializable()
- private commands: commander.CommanderStatic;
- /**
- * @type {TInputCLIOptions}
- */
- @initializable()
- private inputCLIOptions: TInputCLIOptions;
- /**
- * @type {string}
- */
- @initializable()
- private inputPath: string;
- /**
- * @param {string[]} argv
- */
- constructor (argv: string[]) {
- this.rawArguments = argv;
- this.arguments = argv.slice(2);
- }
- /**
- * @param {TObject} options
- * @returns {TInputOptions}
- */
- private static filterOptions (options: TObject): TInputOptions {
- const filteredOptions: TInputOptions = {};
- for (const option in options) {
- if (!options.hasOwnProperty(option) || options[option] === undefined) {
- continue;
- }
- filteredOptions[option] = options[option];
- }
- return filteredOptions;
- }
- /**
- * @param {string} sourceCode
- * @param {string} outputCodePath
- * @param {TInputOptions} options
- */
- private static processSourceCodeWithoutSourceMap (
- sourceCode: string,
- outputCodePath: string,
- options: TInputOptions
- ): void {
- const obfuscatedCode: string = JavaScriptObfuscator.obfuscate(sourceCode, options).getObfuscatedCode();
- CLIUtils.writeFile(outputCodePath, obfuscatedCode);
- }
- /**
- * @param {string} sourceCode
- * @param {string} outputCodePath
- * @param {TInputOptions} options
- */
- private static processSourceCodeWithSourceMap (
- sourceCode: string,
- outputCodePath: string,
- options: TInputOptions
- ): void {
- const outputSourceMapPath: string = CLIUtils.getOutputSourceMapPath(
- outputCodePath,
- options.sourceMapFileName || ''
- );
- options = {
- ...options,
- sourceMapFileName: path.basename(outputSourceMapPath)
- };
- const obfuscationResult: IObfuscationResult = JavaScriptObfuscator.obfuscate(sourceCode, options);
- CLIUtils.writeFile(outputCodePath, obfuscationResult.getObfuscatedCode());
- if (options.sourceMapMode === 'separate' && obfuscationResult.getSourceMap()) {
- CLIUtils.writeFile(outputSourceMapPath, obfuscationResult.getSourceMap());
- }
- }
- public initialize (): void {
- this.inputPath = this.arguments[0] || '';
- this.commands = <commander.CommanderStatic>(new commander.Command());
- this.configureCommands();
- this.configureHelp();
- this.inputCLIOptions = this.commands.opts();
- }
- public run (): void {
- const canShowHelp: boolean = !this.arguments.length || this.arguments.includes('--help');
- if (canShowHelp) {
- return this.commands.outputHelp();
- }
- const sourceCodeData: TSourceCodeData = SourceCodeReader.readSourceCode(this.inputPath);
- this.processSourceCodeData(sourceCodeData);
- }
- /**
- * @returns {TInputOptions}
- */
- private buildOptions (): TInputOptions {
- const inputCLIOptions: TInputOptions = JavaScriptObfuscatorCLI.filterOptions(this.inputCLIOptions);
- const configFilePath: string | undefined = this.inputCLIOptions.config;
- const configFileLocation: string = configFilePath ? path.resolve(configFilePath, '.') : '';
- const configFileOptions: TInputOptions = configFileLocation ? CLIUtils.getUserConfig(configFileLocation) : {};
- return {
- ...DEFAULT_PRESET,
- ...configFileOptions,
- ...inputCLIOptions
- };
- }
- private configureCommands (): void {
- this.commands
- .usage('<inputPath> [options]')
- .version(
- packageJson.version,
- '-v, --version'
- )
- .option(
- '-o, --output <path>',
- 'Output path for obfuscated code'
- )
- .option(
- '--compact <boolean>',
- 'Disable one line output code compacting',
- BooleanSanitizer
- )
- .option(
- '--config <boolean>',
- 'Name of js / json config file'
- )
- .option(
- '--control-flow-flattening <boolean>',
- 'Enables control flow flattening',
- BooleanSanitizer
- )
- .option(
- '--control-flow-flattening-threshold <number>',
- 'The probability that the control flow flattening transformation will be applied to the node',
- parseFloat
- )
- .option(
- '--dead-code-injection <boolean>',
- 'Enables dead code injection',
- BooleanSanitizer
- )
- .option(
- '--dead-code-injection-threshold <number>',
- 'The probability that the dead code injection transformation will be applied to the node',
- parseFloat
- )
- .option(
- '--debug-protection <boolean>',
- 'Disable browser Debug panel (can cause DevTools enabled browser freeze)',
- BooleanSanitizer
- )
- .option(
- '--debug-protection-interval <boolean>',
- 'Disable browser Debug panel even after page was loaded (can cause DevTools enabled browser freeze)',
- BooleanSanitizer
- )
- .option(
- '--disable-console-output <boolean>',
- 'Allow console.log, console.info, console.error and console.warn messages output into browser console',
- BooleanSanitizer
- )
- .option(
- '--domain-lock <list> (comma separated, without whitespaces)',
- 'Blocks the execution of the code in domains that do not match the passed RegExp patterns (comma separated)',
- ArraySanitizer
- )
- .option(
- '--identifier-names-generator <string> [hexadecimal, mangled]', 'Sets identifier names generator (Default: hexadecimal)',
- IdentifierNamesGeneratorSanitizer
- )
- .option(
- '--log <boolean>', 'Enables logging of the information to the console',
- BooleanSanitizer
- )
- .option(
- '--reserved-names <list> (comma separated, without whitespaces)',
- 'Disable obfuscation of variable names, function names and names of function parameters that match the passed RegExp patterns (comma separated)',
- ArraySanitizer
- )
- .option(
- '--rename-globals <boolean>', 'Allows to enable obfuscation of global variable and function names with declaration.',
- BooleanSanitizer
- )
- .option(
- '--rotate-string-array <boolean>', 'Disable rotation of unicode array values during obfuscation',
- BooleanSanitizer
- )
- .option(
- '--seed <number>',
- 'Sets seed for random generator. This is useful for creating repeatable results.',
- parseFloat
- )
- .option(
- '--self-defending <boolean>',
- 'Disables self-defending for obfuscated code',
- BooleanSanitizer
- )
- .option(
- '--source-map <boolean>',
- 'Enables source map generation',
- BooleanSanitizer
- )
- .option(
- '--source-map-base-url <string>',
- 'Sets base url to the source map import url when `--source-map-mode=separate`'
- )
- .option(
- '--source-map-file-name <string>',
- 'Sets file name for output source map when `--source-map-mode=separate`'
- )
- .option(
- '--source-map-mode <string> [inline, separate]',
- 'Specify source map output mode',
- SourceMapModeSanitizer
- )
- .option(
- '--string-array <boolean>',
- 'Disables gathering of all literal strings into an array and replacing every literal string with an array call',
- BooleanSanitizer
- )
- .option(
- '--string-array-encoding <boolean|string> [true, false, base64, rc4]',
- 'Encodes all strings in strings array using base64 or rc4 (this option can slow down your code speed',
- StringArrayEncodingSanitizer
- )
- .option(
- '--string-array-threshold <number>',
- 'The probability that the literal string will be inserted into stringArray (Default: 0.8, Min: 0, Max: 1)',
- parseFloat
- )
- .option(
- '--target <string>',
- 'Allows to set target environment for obfuscated code.',
- ObfuscationTargetSanitizer
- )
- .option(
- '--unicode-escape-sequence <boolean>',
- 'Allows to enable/disable string conversion to unicode escape sequence',
- BooleanSanitizer
- )
- .parse(this.rawArguments);
- }
- private configureHelp (): void {
- this.commands.on('--help', () => {
- console.log(' Examples:\n');
- console.log(' %> javascript-obfuscator input_file_name.js --compact true --self-defending false');
- console.log(' %> javascript-obfuscator input_file_name.js --output output_file_name.js --compact true --self-defending false');
- console.log(' %> javascript-obfuscator input_directory_name --compact true --self-defending false');
- console.log('');
- });
- }
- /**
- * @param {TSourceCodeData} sourceCodeData
- */
- private processSourceCodeData (sourceCodeData: TSourceCodeData): void {
- if (!Array.isArray(sourceCodeData)) {
- const outputCodePath: string = this.inputCLIOptions.output || CLIUtils.getOutputCodePath(this.inputPath);
- this.processSourceCode(sourceCodeData, outputCodePath);
- } else {
- sourceCodeData.forEach(({ filePath, content }: IFileData) => {
- const outputCodePath: string = CLIUtils.getOutputCodePath(filePath);
- this.processSourceCode(content, outputCodePath);
- });
- }
- }
- /**
- * @param {string} sourceCode
- * @param {string} outputCodePath
- */
- private processSourceCode (sourceCode: string, outputCodePath: string): void {
- const options: TInputOptions = this.buildOptions();
- if (options.sourceMap) {
- JavaScriptObfuscatorCLI.processSourceCodeWithSourceMap(sourceCode, outputCodePath, options);
- } else {
- JavaScriptObfuscatorCLI.processSourceCodeWithoutSourceMap(sourceCode, outputCodePath, options);
- }
- }
- }
|