const { NetUtil } = ChromeUtils.importESModule(
"resource://gre/modules/NetUtil.sys.mjs"
);
const { HttpServer } = ChromeUtils.importESModule(
"resource://testing-common/httpd.sys.mjs"
);
const ReferrerInfo = Components.Constructor(
"@mozilla.org/referrer-info;1",
"nsIReferrerInfo",
"init"
);
var server =
new HttpServer();
server.registerPathHandler(
"/image.png", imageHandler);
server.start(-1);
/* import-globals-from image_load_helpers.js */
load(
"image_load_helpers.js");
var gHits = 0;
var gIoService = Services.io;
var gPublicLoader = Cc[
"@mozilla.org/image/loader;1"].createInstance(
Ci.imgILoader
);
var gPrivateLoader = Cc[
"@mozilla.org/image/loader;1"].createInstance(
Ci.imgILoader
);
gPrivateLoader.QueryInterface(Ci.imgICache).respectPrivacyNotifications();
var nonPrivateLoadContext = Cu.createLoadContext();
var privateLoadContext = Cu.createPrivateLoadContext();
function imageHandler(metadata, response) {
gHits++;
response.setHeader(
"Cache-Control",
"max-age=10000",
false);
response.setStatusLine(metadata.httpVersion, 200,
"OK");
response.setHeader(
"Content-Type",
"image/png",
false);
var body = atob(
"iVBORw0KGgoAAAANSUhEUgAAAAMAAAADCAIAAADZSiLoAAAAEUlEQVQImWP4z8AAQTAamQkAhpcI+DeMzFcAAAAASUVORK5CYII="
);
response.bodyOutputStream.write(body, body.length);
}
var requests = [];
var listeners = [];
var gImgPath =
"http://localhost:" + server.identity.primaryPort + "/image.png";
function setup_chan(path, isPrivate, callback) {
var uri = NetUtil.newURI(gImgPath);
var securityFlags = Ci.nsILoadInfo.SEC_ALLOW_CROSS_ORIGIN_SEC_CONTEXT_IS_NULL;
var principal = Services.scriptSecurityManager.createContentPrincipal(uri, {
privateBrowsingId: isPrivate ? 1 : 0,
});
var chan = NetUtil.newChannel({
uri,
loadingPrincipal: principal,
securityFlags,
contentPolicyType: Ci.nsIContentPolicy.TYPE_INTERNAL_IMAGE,
});
chan.notificationCallbacks = isPrivate
? privateLoadContext
: nonPrivateLoadContext;
var channelListener =
new ChannelListener();
chan.asyncOpen(channelListener);
var listener =
new ImageListener(
null, callback);
var outlistener = {};
var loader = isPrivate ? gPrivateLoader : gPublicLoader;
var outer = Cc[
"@mozilla.org/image/tools;1"]
.getService(Ci.imgITools)
.createScriptedObserver(listener);
listeners.push(outer);
requests.push(
loader.loadImageWithChannelXPCOM(chan, outer,
null, outlistener)
);
channelListener.outputListener = outlistener.value;
listener.synchronous =
false;
}
function loadImage(isPrivate, callback) {
var listener =
new ImageListener(
null, callback);
var outer = Cc[
"@mozilla.org/image/tools;1"]
.getService(Ci.imgITools)
.createScriptedObserver(listener);
var uri = gIoService.newURI(gImgPath);
var loadGroup = Cc[
"@mozilla.org/network/load-group;1"].createInstance(
Ci.nsILoadGroup
);
loadGroup.notificationCallbacks = isPrivate
? privateLoadContext
: nonPrivateLoadContext;
var loader = isPrivate ? gPrivateLoader : gPublicLoader;
var referrerInfo =
new ReferrerInfo(
Ci.nsIReferrerInfo.NO_REFERRER_WHEN_DOWNGRADE,
true,
null
);
requests.push(
loader.loadImageXPCOM(
uri,
null,
referrerInfo,
null,
loadGroup,
outer,
null,
0,
null
)
);
listener.synchronous =
false;
}
function run_loadImage_tests() {
function observer() {
Services.obs.removeObserver(observer,
"cacheservice:empty-cache");
gHits = 0;
loadImage(
false,
function () {
loadImage(
false,
function () {
loadImage(
true,
function () {
loadImage(
true,
function () {
Assert.equal(gHits, 2);
server.stop(do_test_finished);
});
});
});
});
}
for (let loader of [gPublicLoader, gPrivateLoader]) {
loader.QueryInterface(Ci.imgICache).clearCache(
true);
loader.QueryInterface(Ci.imgICache).clearCache(
false);
}
Services.obs.addObserver(observer,
"cacheservice:empty-cache");
let cs = Services.cache2;
cs.clear();
}
function cleanup() {
for (
var i = 0; i < requests.length; ++i) {
requests[i].cancelAndForgetObserver(0);
}
}
function run_test() {
registerCleanupFunction(cleanup);
do_test_pending();
Services.prefs.setBoolPref(
"network.http.rcwn.enabled",
false);
// We create a public channel that loads an image, then an identical
// one that should cause a cache read. We then create a private channel
// and load the same image, and do that a second time to ensure a cache
// read. In total, we should cause two separate http responses to occur,
// since the private channels shouldn't be able to use the public cache.
setup_chan(
"/image.png",
false,
function () {
setup_chan(
"/image.png",
false,
function () {
setup_chan(
"/image.png",
true,
function () {
setup_chan(
"/image.png",
true,
function () {
Assert.equal(gHits, 2);
run_loadImage_tests();
});
});
});
});
}