2013-10-26 01:56:51 +04:00
|
|
|
var Cc = Components.classes;
|
|
|
|
var Ci = Components.interfaces;
|
|
|
|
var Cu = Components.utils;
|
|
|
|
var Cr = Components.results;
|
|
|
|
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
2015-06-18 12:23:00 +03:00
|
|
|
|
|
|
|
var running_single_process = false;
|
2013-10-26 01:56:51 +04:00
|
|
|
|
2014-06-04 00:37:46 +04:00
|
|
|
var predictor = null;
|
2015-06-18 12:23:00 +03:00
|
|
|
|
|
|
|
function is_child_process() {
|
|
|
|
return Cc["@mozilla.org/xre/app-info;1"].getService(Ci.nsIXULRuntime).processType == Ci.nsIXULRuntime.PROCESS_TYPE_CONTENT;
|
|
|
|
}
|
2013-10-26 01:56:51 +04:00
|
|
|
|
|
|
|
function extract_origin(uri) {
|
|
|
|
var o = uri.scheme + "://" + uri.asciiHost;
|
|
|
|
if (uri.port !== -1) {
|
|
|
|
o = o + ":" + uri.port;
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
var LoadContext = function _loadContext() {
|
|
|
|
};
|
|
|
|
|
|
|
|
LoadContext.prototype = {
|
|
|
|
usePrivateBrowsing: false,
|
|
|
|
|
|
|
|
getInterface: function loadContext_getInterface(iid) {
|
|
|
|
return this.QueryInterface(iid);
|
|
|
|
},
|
|
|
|
|
|
|
|
QueryInterface: function loadContext_QueryInterface(iid) {
|
2014-06-04 00:37:46 +04:00
|
|
|
if (iid.equals(Ci.nsINetworkPredictorVerifier) ||
|
2013-10-26 01:56:51 +04:00
|
|
|
iid.equals(Ci.nsILoadContext)) {
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
2015-09-23 11:10:21 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
originAttributes: {}
|
2013-10-26 01:56:51 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
var load_context = new LoadContext();
|
|
|
|
|
|
|
|
var Verifier = function _verifier(testing, expected_preconnects, expected_preresolves) {
|
|
|
|
this.verifying = testing;
|
|
|
|
this.expected_preconnects = expected_preconnects;
|
|
|
|
this.expected_preresolves = expected_preresolves;
|
|
|
|
};
|
|
|
|
|
|
|
|
Verifier.prototype = {
|
2015-01-15 00:59:04 +03:00
|
|
|
complete: false,
|
2013-10-26 01:56:51 +04:00
|
|
|
verifying: null,
|
|
|
|
expected_preconnects: null,
|
|
|
|
expected_preresolves: null,
|
|
|
|
|
|
|
|
getInterface: function verifier_getInterface(iid) {
|
|
|
|
return this.QueryInterface(iid);
|
|
|
|
},
|
|
|
|
|
|
|
|
QueryInterface: function verifier_QueryInterface(iid) {
|
2014-06-04 00:37:46 +04:00
|
|
|
if (iid.equals(Ci.nsINetworkPredictorVerifier) ||
|
2013-10-26 01:56:51 +04:00
|
|
|
iid.equals(Ci.nsISupports)) {
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
|
|
},
|
|
|
|
|
|
|
|
maybe_run_next_test: function verifier_maybe_run_next_test() {
|
|
|
|
if (this.expected_preconnects.length === 0 &&
|
2015-01-15 00:59:04 +03:00
|
|
|
this.expected_preresolves.length === 0 &&
|
|
|
|
!this.complete) {
|
|
|
|
this.complete = true;
|
2013-10-26 01:56:51 +04:00
|
|
|
do_check_true(true, "Well this is unexpected...");
|
2015-01-15 00:59:04 +03:00
|
|
|
// This kicks off the ability to run the next test
|
|
|
|
reset_predictor();
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onPredictPreconnect: function verifier_onPredictPreconnect(uri) {
|
|
|
|
var origin = extract_origin(uri);
|
|
|
|
var index = this.expected_preconnects.indexOf(origin);
|
2015-01-15 00:59:04 +03:00
|
|
|
if (index == -1 && !this.complete) {
|
2013-10-26 01:56:51 +04:00
|
|
|
do_check_true(false, "Got preconnect for unexpected uri " + origin);
|
|
|
|
} else {
|
|
|
|
this.expected_preconnects.splice(index, 1);
|
|
|
|
}
|
|
|
|
this.maybe_run_next_test();
|
|
|
|
},
|
|
|
|
|
|
|
|
onPredictDNS: function verifier_onPredictDNS(uri) {
|
|
|
|
var origin = extract_origin(uri);
|
|
|
|
var index = this.expected_preresolves.indexOf(origin);
|
2015-01-15 00:59:04 +03:00
|
|
|
if (index == -1 && !this.complete) {
|
2013-10-26 01:56:51 +04:00
|
|
|
do_check_true(false, "Got preresolve for unexpected uri " + origin);
|
|
|
|
} else {
|
|
|
|
this.expected_preresolves.splice(index, 1);
|
|
|
|
}
|
|
|
|
this.maybe_run_next_test();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-06-04 00:37:46 +04:00
|
|
|
function reset_predictor() {
|
2015-06-18 12:23:00 +03:00
|
|
|
if (running_single_process || is_child_process()) {
|
|
|
|
predictor.reset();
|
|
|
|
} else {
|
|
|
|
sendCommand("predictor.reset();");
|
|
|
|
}
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
function newURI(s) {
|
2015-06-18 12:23:00 +03:00
|
|
|
return Services.io.newURI(s, null, null);
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
2015-01-15 00:59:04 +03:00
|
|
|
var prepListener = {
|
|
|
|
numEntriesToOpen: 0,
|
|
|
|
numEntriesOpened: 0,
|
|
|
|
continueCallback: null,
|
|
|
|
|
|
|
|
QueryInterface: function (iid) {
|
|
|
|
if (iid.equals(Ci.nsICacheEntryOpenCallback)) {
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
|
|
},
|
|
|
|
|
|
|
|
init: function (entriesToOpen, cb) {
|
|
|
|
this.numEntriesOpened = 0;
|
|
|
|
this.numEntriesToOpen = entriesToOpen;
|
|
|
|
this.continueCallback = cb;
|
|
|
|
},
|
|
|
|
|
|
|
|
onCacheEntryCheck: function (entry, appCache) {
|
|
|
|
return Ci.nsICacheEntryOpenCallback.ENTRY_WANTED;
|
|
|
|
},
|
|
|
|
|
|
|
|
onCacheEntryAvailable: function (entry, isNew, appCache, result) {
|
|
|
|
do_check_eq(result, Cr.NS_OK);
|
|
|
|
entry.setMetaDataElement("predictor_test", "1");
|
|
|
|
entry.metaDataReady();
|
|
|
|
this.numEntriesOpened++;
|
|
|
|
if (this.numEntriesToOpen == this.numEntriesOpened) {
|
|
|
|
this.continueCallback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
function open_and_continue(uris, continueCallback) {
|
|
|
|
var lci = {
|
|
|
|
QueryInterface: function (iid) {
|
|
|
|
if (iid.equals(Ci.nsILoadContextInfo)) {
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
|
|
},
|
|
|
|
|
|
|
|
isPrivate: false,
|
|
|
|
appId: Ci.nsILoadContextInfo.NO_APP_ID,
|
|
|
|
isInBrowserElement: false,
|
|
|
|
isAnonymous: false
|
|
|
|
};
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
var ds = Services.cache2.diskCacheStorage(lci, false);
|
2015-01-15 00:59:04 +03:00
|
|
|
|
|
|
|
prepListener.init(uris.length, continueCallback);
|
|
|
|
for (var i = 0; i < uris.length; ++i) {
|
|
|
|
ds.asyncOpenURI(uris[i], "", Ci.nsICacheStorage.OPEN_NORMALLY,
|
|
|
|
prepListener);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-26 01:56:51 +04:00
|
|
|
function test_link_hover() {
|
2015-06-18 12:23:00 +03:00
|
|
|
if (!running_single_process && !is_child_process()) {
|
|
|
|
// This one we can just proxy to the child and be done with, no extra setup
|
|
|
|
// is necessary.
|
|
|
|
sendCommand("test_link_hover();");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-10-26 01:56:51 +04:00
|
|
|
var uri = newURI("http://localhost:4444/foo/bar");
|
|
|
|
var referrer = newURI("http://localhost:4444/foo");
|
|
|
|
var preconns = ["http://localhost:4444"];
|
|
|
|
|
|
|
|
var verifier = new Verifier("hover", preconns, []);
|
2014-06-04 00:37:46 +04:00
|
|
|
predictor.predict(uri, referrer, predictor.PREDICT_LINK, load_context, verifier);
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
const pageload_toplevel = newURI("http://localhost:4444/index.html");
|
|
|
|
|
|
|
|
function continue_test_pageload() {
|
2013-10-26 01:56:51 +04:00
|
|
|
var subresources = [
|
|
|
|
"http://localhost:4444/style.css",
|
|
|
|
"http://localhost:4443/jquery.js",
|
|
|
|
"http://localhost:4444/image.png"
|
|
|
|
];
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
// This is necessary to learn the origin stuff
|
|
|
|
predictor.learn(pageload_toplevel, null, predictor.LEARN_LOAD_TOPLEVEL, load_context);
|
|
|
|
var preconns = [];
|
|
|
|
for (var i = 0; i < subresources.length; i++) {
|
|
|
|
var sruri = newURI(subresources[i]);
|
|
|
|
predictor.learn(sruri, pageload_toplevel, predictor.LEARN_LOAD_SUBRESOURCE, load_context);
|
|
|
|
preconns.push(extract_origin(sruri));
|
|
|
|
}
|
|
|
|
|
|
|
|
var verifier = new Verifier("pageload", preconns, []);
|
|
|
|
predictor.predict(pageload_toplevel, null, predictor.PREDICT_LOAD, load_context, verifier);
|
|
|
|
}
|
2013-10-26 01:56:51 +04:00
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
function test_pageload() {
|
|
|
|
open_and_continue([pageload_toplevel], function () {
|
|
|
|
if (running_single_process) {
|
|
|
|
continue_test_pageload();
|
|
|
|
} else {
|
|
|
|
sendCommand("continue_test_pageload();");
|
|
|
|
}
|
2015-01-15 00:59:04 +03:00
|
|
|
});
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
const redirect_inituri = newURI("http://localhost:4443/redirect");
|
|
|
|
const redirect_targeturi = newURI("http://localhost:4444/index.html");
|
|
|
|
|
|
|
|
function continue_test_redrect() {
|
2013-10-26 01:56:51 +04:00
|
|
|
var subresources = [
|
|
|
|
"http://localhost:4444/style.css",
|
|
|
|
"http://localhost:4443/jquery.js",
|
|
|
|
"http://localhost:4444/image.png"
|
|
|
|
];
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
predictor.learn(redirect_inituri, null, predictor.LEARN_LOAD_TOPLEVEL, load_context);
|
|
|
|
predictor.learn(redirect_targeturi, null, predictor.LEARN_LOAD_TOPLEVEL, load_context);
|
|
|
|
predictor.learn(redirect_targeturi, redirect_inituri, predictor.LEARN_LOAD_REDIRECT, load_context);
|
|
|
|
|
|
|
|
var preconns = [];
|
|
|
|
preconns.push(extract_origin(redirect_targeturi));
|
|
|
|
for (var i = 0; i < subresources.length; i++) {
|
|
|
|
var sruri = newURI(subresources[i]);
|
|
|
|
predictor.learn(sruri, redirect_targeturi, predictor.LEARN_LOAD_SUBRESOURCE, load_context);
|
|
|
|
preconns.push(extract_origin(sruri));
|
|
|
|
}
|
|
|
|
|
|
|
|
var verifier = new Verifier("redirect", preconns, []);
|
|
|
|
predictor.predict(redirect_inituri, null, predictor.PREDICT_LOAD, load_context, verifier);
|
|
|
|
}
|
2013-10-26 01:56:51 +04:00
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
function test_redirect() {
|
|
|
|
open_and_continue([redirect_inituri, redirect_targeturi], function () {
|
|
|
|
if (running_single_process) {
|
|
|
|
continue_test_redirect();
|
|
|
|
} else {
|
|
|
|
sendCommand("continue_test_redirect();");
|
|
|
|
}
|
2015-01-15 00:59:04 +03:00
|
|
|
});
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
function test_startup() {
|
2015-06-18 12:23:00 +03:00
|
|
|
if (!running_single_process && !is_child_process()) {
|
|
|
|
// This one we can just proxy to the child and be done with, no extra setup
|
|
|
|
// is necessary.
|
|
|
|
sendCommand("test_startup();");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-10-26 01:56:51 +04:00
|
|
|
var uris = [
|
|
|
|
"http://localhost:4444/startup",
|
|
|
|
"http://localhost:4443/startup"
|
|
|
|
];
|
|
|
|
var preconns = [];
|
|
|
|
for (var i = 0; i < uris.length; i++) {
|
|
|
|
var uri = newURI(uris[i]);
|
2014-06-04 00:37:46 +04:00
|
|
|
predictor.learn(uri, null, predictor.LEARN_STARTUP, load_context);
|
2013-10-26 01:56:51 +04:00
|
|
|
preconns.push(extract_origin(uri));
|
|
|
|
}
|
|
|
|
|
|
|
|
var verifier = new Verifier("startup", preconns, []);
|
2014-06-04 00:37:46 +04:00
|
|
|
predictor.predict(null, null, predictor.PREDICT_STARTUP, load_context, verifier);
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
const dns_toplevel = newURI("http://localhost:4444/index.html");
|
|
|
|
|
|
|
|
function continue_test_dns() {
|
2013-10-26 01:56:51 +04:00
|
|
|
var subresource = "http://localhost:4443/jquery.js";
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
predictor.learn(dns_toplevel, null, predictor.LEARN_LOAD_TOPLEVEL, load_context);
|
|
|
|
var sruri = newURI(subresource);
|
|
|
|
predictor.learn(sruri, dns_toplevel, predictor.LEARN_LOAD_SUBRESOURCE, load_context);
|
2015-01-15 00:59:04 +03:00
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
var preresolves = [extract_origin(sruri)];
|
|
|
|
var verifier = new Verifier("dns", [], preresolves);
|
|
|
|
predictor.predict(dns_toplevel, null, predictor.PREDICT_LOAD, load_context, verifier);
|
|
|
|
}
|
2015-01-15 00:59:04 +03:00
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
function test_dns() {
|
|
|
|
open_and_continue([dns_toplevel], function () {
|
|
|
|
// Ensure that this will do preresolves
|
|
|
|
Services.prefs.setIntPref("network.predictor.preconnect-min-confidence", 101);
|
|
|
|
if (running_single_process) {
|
|
|
|
continue_test_dns();
|
|
|
|
} else {
|
|
|
|
sendCommand("continue_test_dns();");
|
|
|
|
}
|
2015-01-15 00:59:04 +03:00
|
|
|
});
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
const origin_toplevel = newURI("http://localhost:4444/index.html");
|
|
|
|
|
|
|
|
function continue_test_origin() {
|
2013-10-26 01:56:51 +04:00
|
|
|
var subresources = [
|
|
|
|
"http://localhost:4444/style.css",
|
|
|
|
"http://localhost:4443/jquery.js",
|
|
|
|
"http://localhost:4444/image.png"
|
|
|
|
];
|
2015-06-18 12:23:00 +03:00
|
|
|
predictor.learn(origin_toplevel, null, predictor.LEARN_LOAD_TOPLEVEL, load_context);
|
|
|
|
var preconns = [];
|
|
|
|
for (var i = 0; i < subresources.length; i++) {
|
|
|
|
var sruri = newURI(subresources[i]);
|
|
|
|
predictor.learn(sruri, origin_toplevel, predictor.LEARN_LOAD_SUBRESOURCE, load_context);
|
|
|
|
var origin = extract_origin(sruri);
|
|
|
|
if (preconns.indexOf(origin) === -1) {
|
|
|
|
preconns.push(origin);
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
2015-06-18 12:23:00 +03:00
|
|
|
}
|
2013-10-26 01:56:51 +04:00
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
var loaduri = newURI("http://localhost:4444/anotherpage.html");
|
|
|
|
var verifier = new Verifier("origin", preconns, []);
|
|
|
|
predictor.predict(loaduri, null, predictor.PREDICT_LOAD, load_context, verifier);
|
2013-10-26 01:56:51 +04:00
|
|
|
}
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
function test_origin() {
|
|
|
|
open_and_continue([origin_toplevel], function () {
|
|
|
|
if (running_single_process) {
|
|
|
|
continue_test_origin();
|
|
|
|
} else {
|
|
|
|
sendCommand("continue_test_origin();");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2014-05-08 00:03:00 +04:00
|
|
|
|
|
|
|
function cleanup() {
|
2015-01-15 00:59:04 +03:00
|
|
|
observer.cleaningUp = true;
|
2015-06-18 12:23:00 +03:00
|
|
|
reset_predictor();
|
2014-05-08 00:03:00 +04:00
|
|
|
}
|
|
|
|
|
2013-10-26 01:56:51 +04:00
|
|
|
var tests = [
|
2015-01-15 00:59:04 +03:00
|
|
|
// This must ALWAYS come first, to ensure a clean slate
|
|
|
|
reset_predictor,
|
2013-10-26 01:56:51 +04:00
|
|
|
test_link_hover,
|
|
|
|
test_pageload,
|
2015-01-15 00:59:04 +03:00
|
|
|
// TODO: These are disabled until the features are re-written
|
|
|
|
//test_redirect,
|
|
|
|
//test_startup,
|
|
|
|
// END DISABLED TESTS
|
|
|
|
test_origin,
|
2013-10-26 01:56:51 +04:00
|
|
|
test_dns,
|
2015-01-15 00:59:04 +03:00
|
|
|
// This must ALWAYS come last, to ensure we clean up after ourselves
|
|
|
|
cleanup
|
2013-10-26 01:56:51 +04:00
|
|
|
];
|
|
|
|
|
2015-01-15 00:59:04 +03:00
|
|
|
var observer = {
|
|
|
|
cleaningUp: false,
|
|
|
|
|
|
|
|
QueryInterface: function (iid) {
|
|
|
|
if (iid.equals(Ci.nsIObserver) ||
|
|
|
|
iid.equals(Ci.nsISupports)) {
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw Cr.NS_ERROR_NO_INTERFACE;
|
|
|
|
},
|
|
|
|
|
|
|
|
observe: function (subject, topic, data) {
|
|
|
|
if (topic != "predictor-reset-complete") {
|
|
|
|
return;
|
|
|
|
}
|
2015-06-18 12:23:00 +03:00
|
|
|
|
2015-01-15 00:59:04 +03:00
|
|
|
if (this.cleaningUp) {
|
|
|
|
unregisterObserver();
|
|
|
|
}
|
2015-06-18 12:23:00 +03:00
|
|
|
|
2015-01-15 00:59:04 +03:00
|
|
|
run_next_test();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
function registerObserver() {
|
2015-06-18 12:23:00 +03:00
|
|
|
Services.obs.addObserver(observer, "predictor-reset-complete", false);
|
2015-01-15 00:59:04 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
function unregisterObserver() {
|
2015-06-18 12:23:00 +03:00
|
|
|
Services.obs.removeObserver(observer, "predictor-reset-complete");
|
2015-01-15 00:59:04 +03:00
|
|
|
}
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
function run_test_real() {
|
2013-10-26 01:56:51 +04:00
|
|
|
tests.forEach(add_test);
|
2015-07-21 23:20:20 +03:00
|
|
|
do_get_profile();
|
2015-06-18 12:23:00 +03:00
|
|
|
|
|
|
|
Services.prefs.setBoolPref("network.predictor.enabled", true);
|
|
|
|
Services.prefs.setBoolPref("network.predictor.cleaned-up", true);
|
|
|
|
Services.prefs.setBoolPref("browser.cache.use_new_backend_temp", true);
|
|
|
|
Services.prefs.setIntPref("browser.cache.use_new_backend", 1);
|
2015-07-21 23:20:20 +03:00
|
|
|
|
|
|
|
predictor = Cc["@mozilla.org/network/predictor;1"].getService(Ci.nsINetworkPredictor);
|
|
|
|
|
|
|
|
registerObserver();
|
|
|
|
|
2015-06-18 12:23:00 +03:00
|
|
|
do_register_cleanup(() => {
|
|
|
|
Services.prefs.clearUserPref("network.predictor.preconnect-min-confidence");
|
|
|
|
Services.prefs.clearUserPref("network.predictor.enabled");
|
|
|
|
Services.prefs.clearUserPref("network.predictor.cleaned-up");
|
|
|
|
Services.prefs.clearUserPref("browser.cache.use_new_backend_temp");
|
|
|
|
Services.prefs.clearUserPref("browser.cache.use_new_backend");
|
|
|
|
});
|
|
|
|
|
2013-10-26 01:56:51 +04:00
|
|
|
run_next_test();
|
|
|
|
}
|
2015-06-18 12:23:00 +03:00
|
|
|
|
|
|
|
function run_test() {
|
|
|
|
// This indirection is necessary to make e10s tests work as expected
|
|
|
|
running_single_process = true;
|
|
|
|
run_test_real();
|
|
|
|
}
|