import { assert } from 'chai'; import { IObfuscationResult } from '../../../../../../src/interfaces/IObfuscationResult'; import { NO_ADDITIONAL_NODES_PRESET } from '../../../../../../src/options/presets/NoCustomNodes'; import { readFileAsString } from '../../../../../helpers/readFileAsString'; import { JavaScriptObfuscator } from '../../../../../../src/JavaScriptObfuscatorFacade'; describe('BinaryExpressionControlFlowReplacer', function () { this.timeout(100000); describe('replace (binaryExpressionNode: ESTree.BinaryExpression,parentNode: ESTree.Node,controlFlowStorage: IStorage )', () => { describe('Variant #1 - single binary expression', () => { const controlFlowStorageCallRegExp: RegExp = /var *_0x([a-f0-9]){4,6} *= *_0x([a-f0-9]){4,6}\['\w{5}'\]\(0x1, *0x2\);/; let obfuscatedCode: string; before(() => { const code: string = readFileAsString(__dirname + '/fixtures/input-1.js'); const obfuscationResult: IObfuscationResult = JavaScriptObfuscator.obfuscate( code, { ...NO_ADDITIONAL_NODES_PRESET, controlFlowFlattening: true, controlFlowFlatteningThreshold: 1 } ); obfuscatedCode = obfuscationResult.getObfuscatedCode(); }); it('should replace binary expression node with call to control flow storage node', () => { assert.match(obfuscatedCode, controlFlowStorageCallRegExp); }); }); describe('Variant #2 - multiple binary expressions with threshold = 1', () => { const expectedMatchErrorsCount: number = 0; const expectedChance: number = 0.5; const samplesCount: number = 1000; const delta: number = 0.1; const controlFlowStorageCallRegExp1: RegExp = /var *_0x(?:[a-f0-9]){4,6} *= *(_0x([a-f0-9]){4,6}\['\w{5}'\])\(0x1, *0x2\);/; const controlFlowStorageCallRegExp2: RegExp = /var *_0x(?:[a-f0-9]){4,6} *= *(_0x([a-f0-9]){4,6}\['\w{5}'\])\(0x2, *0x3\);/; let matchErrorsCount: number = 0, usingExistingIdentifierChance: number; before(() => { const code: string = readFileAsString(__dirname + '/fixtures/input-2.js'); let obfuscationResult: IObfuscationResult, obfuscatedCode: string, firstMatchArray: RegExpMatchArray | null, secondMatchArray: RegExpMatchArray | null, firstMatch: string | undefined, secondMatch: string | undefined, equalsValue: number = 0; for (let i = 0; i < samplesCount; i++) { obfuscationResult = JavaScriptObfuscator.obfuscate( code, { ...NO_ADDITIONAL_NODES_PRESET, controlFlowFlattening: true, controlFlowFlatteningThreshold: 1 } ); obfuscatedCode = obfuscationResult.getObfuscatedCode(); firstMatchArray = obfuscatedCode.match(controlFlowStorageCallRegExp1); secondMatchArray = obfuscatedCode.match(controlFlowStorageCallRegExp2); if (!firstMatchArray || !secondMatchArray) { matchErrorsCount++; continue; } firstMatch = firstMatchArray ? firstMatchArray[1] : undefined; secondMatch = secondMatchArray ? secondMatchArray[1] : undefined; if (firstMatch === secondMatch) { equalsValue++; } } usingExistingIdentifierChance = equalsValue / samplesCount; }); it('should replace binary expression node with call to control flow storage node', () => { assert.equal(matchErrorsCount, expectedMatchErrorsCount); }); it('should use existing identifier for control flow storage with expected chance', () => { assert.closeTo(usingExistingIdentifierChance, expectedChance, delta); }); }); }); });