mirror of
https://github.com/classilla/tenfourfox.git
synced 2025-01-13 11:30:02 +00:00
265 lines
7.8 KiB
JavaScript
265 lines
7.8 KiB
JavaScript
/*
|
|
* Description of the tests:
|
|
* These tests check for conformance to the CSP spec as they relate to Web Manifests.
|
|
*
|
|
* In particular, the tests check that default-src and manifest-src directives are
|
|
* are respected by the ManifestObtainer.
|
|
*/
|
|
/*globals Components*/
|
|
'use strict';
|
|
requestLongerTimeout(10); // e10s tests take time.
|
|
const {
|
|
ManifestObtainer
|
|
} = Cu.import('resource://gre/modules/ManifestObtainer.jsm', {});
|
|
const path = '/tests/dom/security/test/csp/';
|
|
const testFile = `file=${path}file_web_manifest.html`;
|
|
const remoteFile = `file=${path}file_web_manifest_remote.html`;
|
|
const httpsManifest = `file=${path}file_web_manifest_https.html`;
|
|
const mixedContent = `file=${path}file_web_manifest_mixed_content.html`;
|
|
const server = 'file_testserver.sjs';
|
|
const defaultURL = `http://example.org${path}${server}`;
|
|
const remoteURL = `http://mochi.test:8888`;
|
|
const secureURL = `https://example.com${path}${server}`;
|
|
const tests = [
|
|
// CSP block everything, so trying to load a manifest
|
|
// will result in a policy violation.
|
|
{
|
|
expected: `default-src 'none' blocks fetching manifest.`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=default-src 'none'`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(topic) {
|
|
is(topic, 'csp-on-violate-policy', this.expected);
|
|
}
|
|
},
|
|
// CSP allows fetching only from mochi.test:8888,
|
|
// so trying to load a manifest from same origin
|
|
// triggers a CSP violation.
|
|
{
|
|
expected: `default-src mochi.test:8888 blocks manifest fetching.`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=default-src mochi.test:8888`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(topic) {
|
|
is(topic, 'csp-on-violate-policy', this.expected);
|
|
}
|
|
},
|
|
// CSP restricts fetching to 'self', so allowing the manifest
|
|
// to load. The name of the manifest is then checked.
|
|
{
|
|
expected: `CSP default-src 'self' allows fetch of manifest.`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=default-src 'self'`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(manifest) {
|
|
is(manifest.name, 'loaded', this.expected);
|
|
}
|
|
},
|
|
// CSP only allows fetching from mochi.test:8888 and remoteFile
|
|
// requests a manifest from that origin, so manifest should load.
|
|
{
|
|
expected: 'CSP default-src mochi.test:8888 allows fetching manifest.',
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=default-src http://mochi.test:8888`,
|
|
remoteFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(manifest) {
|
|
is(manifest.name, 'loaded', this.expected);
|
|
}
|
|
},
|
|
// default-src blocks everything, so any attempt to
|
|
// fetch a manifest from another origin will trigger a
|
|
// policy violation.
|
|
{
|
|
expected: `default-src 'none' blocks mochi.test:8888`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=default-src 'none'`,
|
|
remoteFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(topic) {
|
|
is(topic, 'csp-on-violate-policy', this.expected);
|
|
}
|
|
},
|
|
// CSP allows fetching from self, so manifest should load.
|
|
{
|
|
expected: `CSP manifest-src allows self`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`manifest-src 'self'`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(manifest) {
|
|
is(manifest.name, 'loaded', this.expected);
|
|
}
|
|
},
|
|
// CSP allows fetching from example.org, so manifest should load.
|
|
{
|
|
expected: `CSP manifest-src allows http://example.org`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`manifest-src http://example.org`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(manifest) {
|
|
is(manifest.name, 'loaded', this.expected);
|
|
}
|
|
},
|
|
// Check interaction with default-src and another origin,
|
|
// CSP allows fetching from example.org, so manifest should load.
|
|
{
|
|
expected: `CSP manifest-src overrides default-src of elsewhere.com`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`default-src: http://elsewhere.com; manifest-src http://example.org`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(manifest) {
|
|
is(manifest.name, 'loaded', this.expected);
|
|
}
|
|
},
|
|
// Check interaction with default-src none,
|
|
// CSP allows fetching manifest from example.org, so manifest should load.
|
|
{
|
|
expected: `CSP manifest-src overrides default-src`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`default-src: 'none'; manifest-src 'self'`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(manifest) {
|
|
is(manifest.name, 'loaded', this.expected);
|
|
}
|
|
},
|
|
// CSP allows fetching from mochi.test:8888, which has a
|
|
// CORS header set to '*'. So the manifest should load.
|
|
{
|
|
expected: `CSP manifest-src allows mochi.test:8888`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=default-src *; manifest-src http://mochi.test:8888`,
|
|
remoteFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(manifest) {
|
|
is(manifest.name, 'loaded', this.expected);
|
|
}
|
|
},
|
|
// CSP restricts fetching to mochi.test:8888, but the test
|
|
// file is at example.org. Hence, a policy violation is
|
|
// triggered.
|
|
{
|
|
expected: `CSP blocks manifest fetching from example.org.`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=manifest-src mochi.test:8888`,
|
|
testFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(topic) {
|
|
is(topic, 'csp-on-violate-policy', this.expected);
|
|
}
|
|
},
|
|
// CSP is set to only allow manifest to be loaded from same origin,
|
|
// but the remote file attempts to load from a different origin. Thus
|
|
// this causes a CSP violation.
|
|
{
|
|
expected: `CSP manifest-src 'self' blocks cross-origin fetch.`,
|
|
get tabURL() {
|
|
let queryParts = [
|
|
`csp=manifest-src 'self'`,
|
|
remoteFile
|
|
];
|
|
return `${defaultURL}?${queryParts.join('&')}`;
|
|
},
|
|
run(topic) {
|
|
is(topic, 'csp-on-violate-policy', this.expected);
|
|
}
|
|
}
|
|
];
|
|
//jscs:disable
|
|
add_task(function* () {
|
|
//jscs:enable
|
|
for (let test of tests) {
|
|
let tabOptions = {
|
|
gBrowser: gBrowser,
|
|
url: test.tabURL,
|
|
};
|
|
yield BrowserTestUtils.withNewTab(
|
|
tabOptions,
|
|
browser => testObtainingManifest(browser, test)
|
|
);
|
|
}
|
|
|
|
function* testObtainingManifest(aBrowser, aTest) {
|
|
const observer = (/blocks/.test(aTest.expected)) ? new NetworkObserver(aTest) : null;
|
|
let manifest;
|
|
// Expect an exception (from promise rejection) if there a content policy
|
|
// that is violated.
|
|
try {
|
|
manifest = yield ManifestObtainer.browserObtainManifest(aBrowser);
|
|
} catch (e) {
|
|
const msg = `Expected promise rejection obtaining.`;
|
|
ok(/blocked the loading of a resource/.test(e.message), msg);
|
|
if (observer) {
|
|
yield observer.finished;
|
|
}
|
|
return;
|
|
}
|
|
// otherwise, we test manifest's content.
|
|
if (manifest) {
|
|
aTest.run(manifest);
|
|
}
|
|
}
|
|
});
|
|
|
|
// Helper object used to observe policy violations. It waits 10 seconds
|
|
// for a response, and then times out causing its associated test to fail.
|
|
function NetworkObserver(test) {
|
|
let finishedTest;
|
|
let success = false;
|
|
this.finished = new Promise((resolver) => {
|
|
finishedTest = resolver;
|
|
})
|
|
this.observe = function observer(subject, topic) {
|
|
SpecialPowers.removeObserver(this, 'csp-on-violate-policy');
|
|
test.run(topic);
|
|
finishedTest();
|
|
success = true;
|
|
};
|
|
SpecialPowers.addObserver(this, 'csp-on-violate-policy', false);
|
|
setTimeout(() => {
|
|
if (!success) {
|
|
test.run('This test timed out.');
|
|
finishedTest();
|
|
}
|
|
}, 1000);
|
|
}
|