331 lines
11 KiB
JavaScript
331 lines
11 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
"use strict";
|
|
|
|
/* import-globals-from head_cache.js */
|
|
/* import-globals-from head_cookies.js */
|
|
/* import-globals-from head_channels.js */
|
|
/* import-globals-from head_servers.js */
|
|
|
|
// We don't normally allow localhost channels to be proxied, but this
|
|
// is easier than updating all the certs and/or domains.
|
|
Services.prefs.setBoolPref("network.proxy.allow_hijacking_localhost", true);
|
|
registerCleanupFunction(() => {
|
|
Services.prefs.clearUserPref("network.proxy.allow_hijacking_localhost");
|
|
});
|
|
|
|
const { HttpServer } = ChromeUtils.importESModule(
|
|
"resource://testing-common/httpd.sys.mjs"
|
|
);
|
|
|
|
function makeChan(uri) {
|
|
let chan = NetUtil.newChannel({
|
|
uri,
|
|
loadUsingSystemPrincipal: true,
|
|
}).QueryInterface(Ci.nsIHttpChannel);
|
|
chan.loadFlags = Ci.nsIChannel.LOAD_INITIAL_DOCUMENT_URI;
|
|
return chan;
|
|
}
|
|
|
|
function channelOpenPromise(chan, flags) {
|
|
return new Promise(resolve => {
|
|
function finish(req, buffer) {
|
|
resolve([req, buffer]);
|
|
}
|
|
chan.asyncOpen(new ChannelListener(finish, null, flags));
|
|
});
|
|
}
|
|
|
|
function registerSimplePathHandler(server, path) {
|
|
return server.registerPathHandler(path, (req, resp) => {
|
|
resp.writeHead(200);
|
|
resp.end("done");
|
|
});
|
|
}
|
|
|
|
function regiisterServerNamePathHandler(server, path) {
|
|
return server.registerPathHandler(path, (req, resp) => {
|
|
resp.writeHead(200);
|
|
resp.end(global.server_name);
|
|
});
|
|
}
|
|
|
|
add_task(async function test_dual_stack() {
|
|
let httpserv = new HttpServer();
|
|
let content = "ok";
|
|
httpserv.registerPathHandler("/", function handler(metadata, response) {
|
|
response.setHeader("Content-Length", `${content.length}`);
|
|
response.bodyOutputStream.write(content, content.length);
|
|
});
|
|
httpserv.start_dualStack(-1);
|
|
|
|
let chan = makeChan(`http://127.0.0.1:${httpserv.identity.primaryPort}/`);
|
|
let [, response] = await channelOpenPromise(chan);
|
|
Assert.equal(response, content);
|
|
|
|
chan = makeChan(`http://[::1]:${httpserv.identity.primaryPort}/`);
|
|
[, response] = await channelOpenPromise(chan);
|
|
Assert.equal(response, content);
|
|
await new Promise(resolve => httpserv.stop(resolve));
|
|
});
|
|
|
|
add_task(async function test_http() {
|
|
let server = new NodeHTTPServer();
|
|
await server.start();
|
|
registerCleanupFunction(async () => {
|
|
await server.stop();
|
|
});
|
|
let chan = makeChan(`http://localhost:${server.port()}/test`);
|
|
let req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 404);
|
|
await registerSimplePathHandler(server, "/test");
|
|
chan = makeChan(`http://localhost:${server.port()}/test`);
|
|
req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 200);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).protocolVersion, "http/1.1");
|
|
equal(req.QueryInterface(Ci.nsIHttpChannelInternal).isProxyUsed, false);
|
|
|
|
await server.stop();
|
|
});
|
|
|
|
add_task(async function test_https() {
|
|
let certdb = Cc["@mozilla.org/security/x509certdb;1"].getService(
|
|
Ci.nsIX509CertDB
|
|
);
|
|
addCertFromFile(certdb, "http2-ca.pem", "CTu,u,u");
|
|
|
|
let server = new NodeHTTPSServer();
|
|
await server.start();
|
|
registerCleanupFunction(async () => {
|
|
await server.stop();
|
|
});
|
|
let chan = makeChan(`https://localhost:${server.port()}/test`);
|
|
let req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 404);
|
|
await registerSimplePathHandler(server, "/test");
|
|
chan = makeChan(`https://localhost:${server.port()}/test`);
|
|
req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 200);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).protocolVersion, "http/1.1");
|
|
|
|
await server.stop();
|
|
});
|
|
|
|
add_task(async function test_http2() {
|
|
let certdb = Cc["@mozilla.org/security/x509certdb;1"].getService(
|
|
Ci.nsIX509CertDB
|
|
);
|
|
addCertFromFile(certdb, "http2-ca.pem", "CTu,u,u");
|
|
|
|
let server = new NodeHTTP2Server();
|
|
await server.start();
|
|
registerCleanupFunction(async () => {
|
|
await server.stop();
|
|
});
|
|
let chan = makeChan(`https://localhost:${server.port()}/test`);
|
|
let req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 404);
|
|
await registerSimplePathHandler(server, "/test");
|
|
chan = makeChan(`https://localhost:${server.port()}/test`);
|
|
req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 200);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).protocolVersion, "h2");
|
|
|
|
await server.stop();
|
|
});
|
|
|
|
add_task(async function test_http1_proxy() {
|
|
let certdb = Cc["@mozilla.org/security/x509certdb;1"].getService(
|
|
Ci.nsIX509CertDB
|
|
);
|
|
addCertFromFile(certdb, "http2-ca.pem", "CTu,u,u");
|
|
|
|
let proxy = new NodeHTTPProxyServer();
|
|
await proxy.start();
|
|
registerCleanupFunction(async () => {
|
|
await proxy.stop();
|
|
});
|
|
|
|
let chan = makeChan(`http://localhost:${proxy.port()}/test`);
|
|
let req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 405);
|
|
|
|
await with_node_servers(
|
|
[NodeHTTPServer, NodeHTTPSServer, NodeHTTP2Server],
|
|
async server => {
|
|
await server.execute(
|
|
`global.server_name = "${server.constructor.name}";`
|
|
);
|
|
await regiisterServerNamePathHandler(server, "/test");
|
|
let [req1, buff] = await channelOpenPromise(
|
|
makeChan(`${server.origin()}/test`),
|
|
CL_ALLOW_UNKNOWN_CL
|
|
);
|
|
equal(req1.status, Cr.NS_OK);
|
|
equal(req1.QueryInterface(Ci.nsIHttpChannel).responseStatus, 200);
|
|
equal(buff, server.constructor.name);
|
|
//Bug 1792187: Check if proxy is set to true when a proxy is used.
|
|
equal(req1.QueryInterface(Ci.nsIHttpChannelInternal).isProxyUsed, true);
|
|
equal(
|
|
req1.QueryInterface(Ci.nsIHttpChannel).protocolVersion,
|
|
server.constructor.name == "NodeHTTP2Server" ? "h2" : "http/1.1"
|
|
);
|
|
}
|
|
);
|
|
|
|
await proxy.stop();
|
|
});
|
|
|
|
add_task(async function test_https_proxy() {
|
|
let certdb = Cc["@mozilla.org/security/x509certdb;1"].getService(
|
|
Ci.nsIX509CertDB
|
|
);
|
|
addCertFromFile(certdb, "http2-ca.pem", "CTu,u,u");
|
|
addCertFromFile(certdb, "proxy-ca.pem", "CTu,u,u");
|
|
|
|
let proxy = new NodeHTTPSProxyServer();
|
|
await proxy.start();
|
|
registerCleanupFunction(async () => {
|
|
await proxy.stop();
|
|
});
|
|
|
|
let chan = makeChan(`https://localhost:${proxy.port()}/test`);
|
|
let req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 405);
|
|
|
|
await with_node_servers(
|
|
[NodeHTTPServer, NodeHTTPSServer, NodeHTTP2Server],
|
|
async server => {
|
|
await server.execute(
|
|
`global.server_name = "${server.constructor.name}";`
|
|
);
|
|
await regiisterServerNamePathHandler(server, "/test");
|
|
|
|
let [req1, buff] = await channelOpenPromise(
|
|
makeChan(`${server.origin()}/test`),
|
|
CL_ALLOW_UNKNOWN_CL
|
|
);
|
|
equal(req1.status, Cr.NS_OK);
|
|
equal(req1.QueryInterface(Ci.nsIHttpChannel).responseStatus, 200);
|
|
equal(buff, server.constructor.name);
|
|
}
|
|
);
|
|
|
|
await proxy.stop();
|
|
});
|
|
|
|
add_task(async function test_http2_proxy() {
|
|
let certdb = Cc["@mozilla.org/security/x509certdb;1"].getService(
|
|
Ci.nsIX509CertDB
|
|
);
|
|
addCertFromFile(certdb, "http2-ca.pem", "CTu,u,u");
|
|
addCertFromFile(certdb, "proxy-ca.pem", "CTu,u,u");
|
|
|
|
let proxy = new NodeHTTP2ProxyServer();
|
|
await proxy.start();
|
|
registerCleanupFunction(async () => {
|
|
await proxy.stop();
|
|
});
|
|
|
|
let chan = makeChan(`https://localhost:${proxy.port()}/test`);
|
|
let req = await new Promise(resolve => {
|
|
chan.asyncOpen(new ChannelListener(resolve, null, CL_ALLOW_UNKNOWN_CL));
|
|
});
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 405);
|
|
|
|
await with_node_servers(
|
|
[NodeHTTPServer, NodeHTTPSServer, NodeHTTP2Server],
|
|
async server => {
|
|
await server.execute(
|
|
`global.server_name = "${server.constructor.name}";`
|
|
);
|
|
await regiisterServerNamePathHandler(server, "/test");
|
|
let [req1, buff] = await channelOpenPromise(
|
|
makeChan(`${server.origin()}/test`),
|
|
CL_ALLOW_UNKNOWN_CL
|
|
);
|
|
equal(req1.status, Cr.NS_OK);
|
|
equal(req1.QueryInterface(Ci.nsIHttpChannel).responseStatus, 200);
|
|
equal(buff, server.constructor.name);
|
|
}
|
|
);
|
|
|
|
await proxy.stop();
|
|
});
|
|
|
|
add_task(async function test_proxy_with_redirects() {
|
|
let certdb = Cc["@mozilla.org/security/x509certdb;1"].getService(
|
|
Ci.nsIX509CertDB
|
|
);
|
|
addCertFromFile(certdb, "http2-ca.pem", "CTu,u,u");
|
|
|
|
let proxies = [
|
|
NodeHTTPProxyServer,
|
|
NodeHTTPSProxyServer,
|
|
NodeHTTP2ProxyServer,
|
|
];
|
|
for (let p of proxies) {
|
|
let proxy = new p();
|
|
await proxy.start();
|
|
registerCleanupFunction(async () => {
|
|
await proxy.stop();
|
|
});
|
|
|
|
await with_node_servers(
|
|
[NodeHTTPServer, NodeHTTPSServer, NodeHTTP2Server],
|
|
async server => {
|
|
info(`Testing ${p.name} with ${server.constructor.name}`);
|
|
await server.execute(
|
|
`global.server_name = "${server.constructor.name}";`
|
|
);
|
|
await server.registerPathHandler("/redirect", (req, resp) => {
|
|
resp.writeHead(302, {
|
|
Location: "/test",
|
|
});
|
|
resp.end(global.server_name);
|
|
});
|
|
await server.registerPathHandler("/test", (req, resp) => {
|
|
resp.writeHead(200);
|
|
resp.end(global.server_name);
|
|
});
|
|
|
|
let chan = makeChan(`${server.origin()}/redirect`);
|
|
let [req, buff] = await channelOpenPromise(chan, CL_ALLOW_UNKNOWN_CL);
|
|
equal(req.status, Cr.NS_OK);
|
|
equal(req.QueryInterface(Ci.nsIHttpChannel).responseStatus, 200);
|
|
equal(buff, server.constructor.name);
|
|
req.QueryInterface(Ci.nsIProxiedChannel);
|
|
ok(!!req.proxyInfo);
|
|
notEqual(req.proxyInfo.type, "direct");
|
|
}
|
|
);
|
|
await proxy.stop();
|
|
}
|
|
});
|