javascript-obfuscator
Version:
549 lines (431 loc) • 21.9 kB
text/typescript
import { assert } from 'chai';
import { IdentifierNamesGenerator } from '../../../../../src/enums/generators/identifier-names-generators/IdentifierNamesGenerator';
import { StringArrayEncoding } from '../../../../../src/enums/StringArrayEncoding';
import { NO_ADDITIONAL_NODES_PRESET } from '../../../../../src/options/presets/NoCustomNodes';
import { readFileAsString } from '../../../../helpers/readFileAsString';
import { JavaScriptObfuscator } from '../../../../../src/JavaScriptObfuscatorFacade';
describe('LiteralTransformer', () => {
describe('transformation of literal node with string value', () => {
describe('Variant #1: default behaviour', () => {
const stringArrayRegExp: RegExp = /^var *_0x([a-f0-9]){4} *= *\['test'\];/;
const stringArrayCallRegExp: RegExp = /var *test *= *_0x([a-f0-9]){4}\('0x0'\);/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/simple-input.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('match #1: should replace literal node value with value from string array', () => {
assert.match(obfuscatedCode, stringArrayRegExp);
});
it('match #2: should replace literal node value with value from string array', () => {
assert.match(obfuscatedCode, stringArrayCallRegExp);
});
});
describe('Variant #2: `stringArray` option is disabled', () => {
const regExp: RegExp = /^var *test *= *'test';/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/simple-input.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET
}
).getObfuscatedCode();
});
it('shouldn\'t replace literal node value with value from string array', () => {
assert.match(obfuscatedCode, regExp);
});
});
describe('Variant #3: string contains non-latin and non-digit characters and `unicodeEscapeSequence` is disabled', () => {
let testFunc: () => void;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/error-when-non-latin.js');
testFunc = () => JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1
}
);
});
it('should\'t throw an error', () => {
assert.doesNotThrow(testFunc);
});
});
describe('Variant #4: same literal node values', () => {
const stringArrayRegExp: RegExp = /^var *_0x([a-f0-9]){4} *= *\['test'\];/;
const stringArrayCallRegExp: RegExp = /var *test *= *_0x([a-f0-9]){4}\('0x0'\);/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/same-literal-values.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('match #1: should create only one item in string array for same literal node values', () => {
assert.match(obfuscatedCode, stringArrayRegExp);
});
it('match #2: should create only one item in string array for same literal node values', () => {
assert.match(obfuscatedCode, stringArrayCallRegExp);
});
});
describe('Variant #5: `unicodeEscapeSequence` option is enabled', () => {
const regExp: RegExp = /^var *test *= *'\\x74\\x65\\x73\\x74';$/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/simple-input.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
unicodeEscapeSequence: true
}
).getObfuscatedCode();
});
it('should replace literal node value with unicode escape sequence', () => {
assert.match(obfuscatedCode, regExp);
});
});
describe('Variant #6: `unicodeEscapeSequence` and `stringArray` options are enabled', () => {
const stringArrayRegExp: RegExp = /^var *_0x([a-f0-9]){4} *= *\['\\x74\\x65\\x73\\x74'\];/;
const stringArrayCallRegExp: RegExp = /var *test *= *_0x([a-f0-9]){4}\('0x0'\);/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/simple-input.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1,
unicodeEscapeSequence: true
}
).getObfuscatedCode();
});
it('match #1: should replace literal node value with unicode escape sequence from string array', () => {
assert.match(obfuscatedCode, stringArrayRegExp);
});
it('match #1: should replace literal node value with unicode escape sequence from string array', () => {
assert.match(obfuscatedCode, stringArrayCallRegExp);
});
});
describe('Variant #7: short literal node value', () => {
const regExp: RegExp = /var *test *= *'te';/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/short-literal-value.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('shouldn\'t replace short literal node value with value from string array', () => {
assert.match(obfuscatedCode, regExp);
});
});
describe('Variant #8: base64 encoding', () => {
const stringArrayRegExp: RegExp = /^var *_0x([a-f0-9]){4} *= *\['dGVzdA=='\];/;
const stringArrayCallRegExp: RegExp = /var *test *= *_0x([a-f0-9]){4}\('0x0'\);/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/simple-input.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayEncoding: StringArrayEncoding.Base64,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('match #1: should replace literal node value with value from string array encoded using base64', () => {
assert.match(obfuscatedCode, stringArrayRegExp);
});
it('match #2: should replace literal node value with value from string array encoded using base64', () => {
assert.match(obfuscatedCode, stringArrayCallRegExp);
});
});
describe('Variant #9: rc4 encoding', () => {
const regExp: RegExp = /var *test *= *_0x([a-f0-9]){4}\('0x0', *'.{4}'\);/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/simple-input.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayEncoding: StringArrayEncoding.Rc4,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('should replace literal node value with value from string array encoded using rc4', () => {
assert.match(obfuscatedCode, regExp);
});
});
describe('Variant #10: `stringArrayThreshold` option value', () => {
const samples: number = 1000;
const stringArrayThreshold: number = 0.5;
const delta: number = 0.1;
const regExp1: RegExp = /var *test *= *_0x([a-f0-9]){4}\('0x0'\);/g;
const regExp2: RegExp = /var *test *= *'test';/g;
let stringArrayProbability: number,
noStringArrayProbability: number;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/simple-input.js');
const obfuscatedCode: string = JavaScriptObfuscator.obfuscate(
`${code}\n`.repeat(samples),
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: stringArrayThreshold
}
).getObfuscatedCode();
const stringArrayMatchesLength: number = obfuscatedCode
.match(regExp1)!
.length;
const noStringArrayMatchesLength: number = obfuscatedCode
.match(regExp2)!
.length;
stringArrayProbability = stringArrayMatchesLength / samples;
noStringArrayProbability = noStringArrayMatchesLength / samples;
});
it('Variant #1: should replace literal node value with value from string array with `stringArrayThreshold` chance', () => {
assert.closeTo(stringArrayProbability, stringArrayThreshold, delta);
});
it('Variant #2: shouldn\'t replace literal node value with value from string array with `(1 - stringArrayThreshold)` chance', () => {
assert.closeTo(noStringArrayProbability, stringArrayThreshold, delta);
});
});
describe('Variant #11: string array calls wrapper name', () => {
const regExp: RegExp = /console\[b\('0x0'\)]\('a'\);/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/string-array-calls-wrapper-name.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1,
identifierNamesGenerator: IdentifierNamesGenerator.MangledIdentifierNamesGenerator
}
).getObfuscatedCode();
});
it('match #1: should keep identifier with string array calls wrapper name untouched after obfuscation', () => {
assert.match(obfuscatedCode, regExp);
});
});
describe('Variant #12: `reservedStrings` option is enabled', () => {
describe('Variant #1: base `reservedStrings` values', () => {
describe('Variant #1: single reserved string value', () => {
const stringLiteralRegExp1: RegExp = /const foo *= *'foo';/;
const stringLiteralRegExp2: RegExp = /const bar *= *_0x([a-f0-9]){4}\('0x0'\);/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/reserved-strings-option-1.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1,
reservedStrings: ['foo']
}
).getObfuscatedCode();
});
it('match #1: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp1);
});
it('match #2: should transform non-reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp2);
});
});
describe('Variant #2: two reserved string values', () => {
const stringLiteralRegExp1: RegExp = /const foo *= *'foo';/;
const stringLiteralRegExp2: RegExp = /const bar *= *'bar';/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/reserved-strings-option-1.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1,
reservedStrings: ['foo', 'bar']
}
).getObfuscatedCode();
});
it('match #1: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp1);
});
it('match #2: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp2);
});
});
});
describe('Variant #2: RegExp `reservedStrings` values', () => {
describe('Variant #1: single reserved string value', () => {
const stringLiteralRegExp1: RegExp = /const foo *= *_0x([a-f0-9]){4}\('0x0'\);/;
const stringLiteralRegExp2: RegExp = /const bar *= *'bar';/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/reserved-strings-option-1.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1,
reservedStrings: ['ar$']
}
).getObfuscatedCode();
});
it('match #1: should transform non-reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp1);
});
it('match #2: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp2);
});
});
describe('Variant #2: two reserved string values', () => {
const stringLiteralRegExp1: RegExp = /const foo *= *'foo';/;
const stringLiteralRegExp2: RegExp = /const bar *= *'bar';/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/reserved-strings-option-1.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1,
reservedStrings: ['^fo', '.ar']
}
).getObfuscatedCode();
});
it('match #1: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp1);
});
it('match #2: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp2);
});
});
});
describe('Variant #3: `unicodeEscapeSequence` option is enabled', () => {
const stringLiteralRegExp1: RegExp = /const foo *= *'foo';/;
const stringLiteralRegExp2: RegExp = /const bar *= *'\\x62\\x61\\x72';/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/reserved-strings-option-1.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
reservedStrings: ['foo'],
unicodeEscapeSequence: true
}
).getObfuscatedCode();
});
it('match #1: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp1);
});
it('match #2: should transform non-reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp2);
});
});
describe('Variant #4: correct escape of special characters', () => {
const stringLiteralRegExp: RegExp = /var baz *= *'Cannot\\x20find\\x20module\\x20\\x27' *\+ *foo *\+ *'\\x27';/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/reserved-strings-option-2.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
reservedStrings: ['a']
}
).getObfuscatedCode();
});
it('match #1: should ignore reserved strings', () => {
assert.match(obfuscatedCode, stringLiteralRegExp);
});
});
});
});
describe('transformation of literal node with boolean value', () => {
const regExp: RegExp = /^var *test *= *!!\[\];$/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/boolean-value.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('should transform literal node', () => {
assert.match(obfuscatedCode, regExp);
});
});
describe('transformation of literal node with number value', () => {
const regExp: RegExp = /^var *test *= *0x0;$/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/number-value.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('should transform literal node', () => {
assert.match(obfuscatedCode, regExp);
});
});
describe('RegExp literal', () => {
const regExp: RegExp = /^var *regExp *= *\/\(\\d\+\)\/;$/;
let obfuscatedCode: string;
before(() => {
const code: string = readFileAsString(__dirname + '/fixtures/regexp-literal.js');
obfuscatedCode = JavaScriptObfuscator.obfuscate(
code,
{
...NO_ADDITIONAL_NODES_PRESET,
stringArray: true,
stringArrayThreshold: 1
}
).getObfuscatedCode();
});
it('should keep safe value of RegExp literal', () => {
assert.match(obfuscatedCode, regExp);
});
});
});