addons-linter/tests/test.utils.js

364 строки
10 KiB
JavaScript

import * as utils from 'utils';
import { unexpectedSuccess } from './helpers';
import { oneLine } from 'common-tags';
describe('utils.getRootExpression()', () => {
const node = {
type: 'CallExpression',
callee: { // <-- bar()
type: 'MemberExpression',
object: {
type: 'CallExpression',
callee: { // <-- foo()
type: 'MemberExpression',
object: {
type: 'CallExpression',
callee: { // <-- pref()
type: 'Identifier',
name: 'pref',
},
},
property: {
type: 'Identifier',
name: 'foo',
},
},
},
property: {
type: 'Identifier',
name: 'bar',
},
},
};
it('should verify that the root node is what was expected', () => {
const root = utils.getRootExpression(node);
expect(root.name).toEqual('pref');
});
});
describe('utils.getNodeReference()', () => {
// Represents scope for following code:
// const foo = window; foo = bar;
const context = {
getScope: () => {
// TODO: Look into generating these AST nodes using ESPrima
return {
variables: [{
name: 'foo', // Reference name
type: 'Identifier',
defs: [{
parent: {
parent: {
body: [{
type: 'VariableDeclaration',
declarations: [{
init: {
name: 'window',
},
}],
}, {
type: 'ExpressionStatement',
expression: {
type: 'AssignmentExpression',
right: {
name: 'bar',
},
},
},
],
},
},
}],
}],
};
},
};
it('should return the name of the referenced variable', () => {
const ref = { name: 'foo' };
const val = utils.getNodeReference(context, ref);
expect(val.name).toEqual('bar');
});
it('should return the name of the reference if not in scope', () => {
const ref = { name: 'doesNotExist' };
const val = utils.getNodeReference(context, ref);
expect(val.name).toEqual(ref.name);
});
});
describe('utils.getVariable()', () => {
// This is the expected schema from eslint
const context = {
getScope: () => {
return {
variables: [{
name: 'foo',
defs: [{
type: 'Variable',
name: {
parent: {
init: {
type: 'Literal',
value: 'bar',
},
},
},
}],
}],
};
},
};
const contextWithoutParent = {
getScope: () => {
return {
variables: [{
name: 'foo',
defs: [{
type: 'Variable',
name: {},
}],
}],
};
},
};
it('should return the correct variable in the given context.', () => {
const foo = utils.getVariable(context, 'foo');
expect(foo.type).toEqual('Literal');
expect(foo.value).toEqual('bar');
});
it("should return undefined if the variable doesn't exist.", () => {
const undef = utils.getVariable(context, 'doesNotExist');
expect(typeof undef).toEqual('undefined');
});
it("should return undefined if the init property isn't on the parent", () => {
const undef = utils.getVariable(contextWithoutParent, 'foo');
expect(typeof undef).toEqual('undefined');
});
});
describe('utils.checkOtherReferences', () => {
const context = {
getScope: () => {
return {
variables: [],
};
},
};
it('should return the node if reference is a Literal', () => {
const literal = utils.getNodeReference(context, { type: 'Literal' });
expect(literal.type).toEqual('Literal');
});
it('should return the node if reference is undefined', () => {
const undef = utils.getNodeReference(context, { type: 'undefined' });
expect(undef.type).toEqual('undefined');
});
});
describe('utils.ensureFilenameExists()', () => {
it('should throw error when filename is not a string', () => {
expect(() => {
utils.ensureFilenameExists();
}).toThrow('Filename is required');
expect(() => {
utils.ensureFilenameExists(0);
}).toThrow('Filename is required');
expect(() => {
utils.ensureFilenameExists(undefined);
}).toThrow('Filename is required');
expect(() => {
utils.ensureFilenameExists(null);
}).toThrow('Filename is required');
});
it('should throw error when filename is empty', () => {
expect(() => {
utils.ensureFilenameExists('');
}).toThrow('Filename is required');
});
it('should accept filenames', () => {
expect(() => {
utils.ensureFilenameExists('foo.js');
utils.ensureFilenameExists('0');
}).not.toThrow();
});
});
describe('utils.checkMinNodeVersion()', () => {
it('should reject if version is not high enough', () => {
const fakeProcess = {
version: 'v0.12.4',
};
return utils.checkMinNodeVersion('0.12.7', fakeProcess)
.then(unexpectedSuccess)
.catch((err) => {
expect(err.message).toContain('Node version must be 0.12.7 or greater');
});
});
it('should not reject if version is not high enough', () => {
const fakeProcess = {
version: 'v4.1.2',
};
return utils.checkMinNodeVersion('0.12.7', fakeProcess);
});
});
describe('utils.ignorePrivateFunctions()', () => {
it('should return only "public" functions', () => {
const listOfRuleFunctions = {
checkForEval: sinon.stub(),
_parseEvalPossibility: sinon.stub(),
checkForCurlyBraces: sinon.stub(),
__checkForFunctions: sinon.stub(),
i_am_an_underscore_function: sinon.stub(),
};
const publicFunctions = utils.ignorePrivateFunctions(listOfRuleFunctions);
expect(typeof publicFunctions).toBe('object');
expect(Object.keys(publicFunctions).length).toBe(3);
expect(Object.keys(publicFunctions)).not.toContain('_parseEvalPossibility');
expect(Object.keys(publicFunctions)).not.toContain('__checkForFunctions');
});
it('should return an empty object when given only private functions', () => {
const listOfRuleFunctions = {
_parseEvalPossibility: sinon.stub(),
__checkForFunctions: sinon.stub(),
};
const publicFunctions = utils.ignorePrivateFunctions(listOfRuleFunctions);
expect(typeof publicFunctions).toBe('object');
expect(Object.keys(publicFunctions).length).toBe(0);
});
it('should return only functions', () => {
const listOfRuleFunctions = {
iAmARule: sinon.stub(),
_privateMethod: sinon.stub(),
IAMCONSTANT: 'foo',
};
const publicFunctions = utils.ignorePrivateFunctions(listOfRuleFunctions);
Object.keys(publicFunctions).forEach((functionName) => {
expect(typeof publicFunctions[functionName]).toEqual('function');
});
});
});
describe('utils.getPackageTypeAsString()', () => {
it('should look up a package type when passed a number', () => {
expect(utils.getPackageTypeAsString(2)).toEqual('PACKAGE_THEME');
});
it('should look up a package type when passed a string', () => {
expect(utils.getPackageTypeAsString('2')).toEqual('PACKAGE_THEME');
});
it('should throw if given a non-existent package type value', () => {
expect(() => {
utils.getPackageTypeAsString(127);
}).toThrow('Invalid package type constant "127"');
});
it('should throw if given a bogus package type value', () => {
expect(() => {
utils.getPackageTypeAsString('whatevs');
}).toThrow('Invalid package type constant "whatevs"');
});
});
describe('utils.isLocalUrl', () => {
it('should not match remote urls', () => {
expect(utils.isLocalUrl('http://foo.com')).toBeFalsy();
expect(utils.isLocalUrl('https://foo.com')).toBeFalsy();
expect(utils.isLocalUrl('ftp://foo.com')).toBeFalsy();
expect(utils.isLocalUrl('//foo.com')).toBeFalsy();
});
it('should not match data uri', () => {
expect(utils.isLocalUrl('' +
'lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAAAIBRAA7')).toBeFalsy();
});
it('should match chrome protocol', () => {
expect(utils.isLocalUrl('chrome://bar/foo')).toBeTruthy();
});
it('should match resource protocol', () => {
expect(utils.isLocalUrl('resource://bar/foo')).toBeTruthy();
});
it('should match non-remote urls starting with /', () => {
expect(utils.isLocalUrl('/bar/foo')).toBeTruthy();
});
it('should match non-remote urls starting with alpha', () => {
expect(utils.isLocalUrl('bar')).toBeTruthy();
});
});
describe('utils.isBrowserNamespace', () => {
it('is true for browser', () => {
expect(utils.isBrowserNamespace('browser')).toEqual(true);
});
it('is true for chrome', () => {
expect(utils.isBrowserNamespace('chrome')).toEqual(true);
});
it('is not true for other strings', () => {
expect(utils.isBrowserNamespace('foo')).toEqual(false);
expect(utils.isBrowserNamespace('bar')).toEqual(false);
expect(utils.isBrowserNamespace('BROWSER')).toEqual(false);
expect(utils.isBrowserNamespace('chrOme')).toEqual(false);
});
});
describe('utils.parseCspPolicy', () => {
it('should allow empty policies', () => {
expect(utils.parseCspPolicy('')).toEqual({});
expect(utils.parseCspPolicy(null)).toEqual({});
expect(utils.parseCspPolicy(undefined)).toEqual({});
});
it('should parse directives correctly', () => {
const rawPolicy = oneLine`
default-src 'none'; script-src 'self'; connect-src https: 'self';
img-src 'self'; style-src 'self';
`;
const parsedPolicy = utils.parseCspPolicy(rawPolicy);
expect(parsedPolicy['script-src']).toEqual(['\'self\'']);
expect(parsedPolicy['default-src']).toEqual(['\'none\'']);
expect(parsedPolicy['connect-src']).toEqual(['https:', '\'self\'']);
expect(parsedPolicy['img-src']).toEqual(['\'self\'']);
expect(parsedPolicy['style-src']).toEqual(['\'self\'']);
});
it('should handle upper case correctly', () => {
const parsedPolicy = utils.parseCspPolicy('DEFAULT-SRC \'NoNe\'');
expect(parsedPolicy['default-src']).toEqual(['\'none\'']);
});
});