/* 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"; let httpsUri; registerCleanupFunction(async () => { Services.prefs.clearUserPref("network.http.http3.enable"); Services.prefs.clearUserPref("network.dns.localDomains"); Services.prefs.clearUserPref("network.dns.disableIPv6"); Services.prefs.clearUserPref( "network.http.http3.alt-svc-mapping-for-testing" ); Services.prefs.clearUserPref("network.http.http3.backup_timer_delay"); dump("cleanup done\n"); }); let Http3FailedListener = function() {}; Http3FailedListener.prototype = { onStartRequest: function testOnStartRequest(request) {}, onDataAvailable: function testOnDataAvailable(request, stream, off, cnt) { this.amount += cnt; read_stream(stream, cnt); }, onStopRequest: function testOnStopRequest(request, status) { if (Components.isSuccessCode(status)) { // This is still HTTP2 connection let httpVersion = ""; try { httpVersion = request.protocolVersion; } catch (e) {} Assert.equal(httpVersion, "h2"); this.finish(false); } else { Assert.equal(status, Cr.NS_ERROR_NET_PARTIAL_TRANSFER); this.finish(true); } }, }; function makeChan() { let chan = NetUtil.newChannel({ uri: httpsUri, loadUsingSystemPrincipal: true, }).QueryInterface(Ci.nsIHttpChannel); chan.loadFlags = Ci.nsIChannel.LOAD_INITIAL_DOCUMENT_URI; return chan; } function altsvcSetupPromise(chan, listener) { return new Promise(resolve => { function finish(result) { resolve(result); } listener.finish = finish; chan.asyncOpen(listener); }); } add_task(async function test_fatal_error() { let h2Port = Services.env.get("MOZHTTP2_PORT"); Assert.notEqual(h2Port, null); let h3Port = Services.env.get("MOZHTTP3_PORT_FAILED"); Assert.notEqual(h3Port, null); Assert.notEqual(h3Port, ""); Services.prefs.setBoolPref("network.http.http3.enable", true); Services.prefs.setCharPref("network.dns.localDomains", "foo.example.com"); Services.prefs.setBoolPref("network.dns.disableIPv6", true); Services.prefs.setCharPref( "network.http.http3.alt-svc-mapping-for-testing", "foo.example.com;h3-29=:" + h3Port ); Services.prefs.setIntPref("network.http.http3.backup_timer_delay", 0); let certdb = Cc["@mozilla.org/security/x509certdb;1"].getService( Ci.nsIX509CertDB ); addCertFromFile(certdb, "http2-ca.pem", "CTu,u,u"); httpsUri = "https://foo.example.com:" + h2Port + "/"; }); add_task(async function test_fatal_stream_error() { let result = false; // We need to loop here because we need to wait for AltSvc storage to // to be started. do { // We need to close HTTP2 connections, otherwise our connection pooling // will dispatch the request over already existing HTTP2 connection. Services.obs.notifyObservers(null, "net:prune-all-connections"); let chan = makeChan(); let listener = new Http3FailedListener(); result = await altsvcSetupPromise(chan, listener); } while (result === false); }); let CheckOnlyHttp2Listener = function() {}; CheckOnlyHttp2Listener.prototype = { onStartRequest: function testOnStartRequest(request) {}, onDataAvailable: function testOnDataAvailable(request, stream, off, cnt) { read_stream(stream, cnt); }, onStopRequest: function testOnStopRequest(request, status) { Assert.equal(status, Cr.NS_OK); let httpVersion = ""; try { httpVersion = request.protocolVersion; } catch (e) {} Assert.equal(httpVersion, "h2"); this.finish(false); }, }; add_task(async function test_no_http3_after_error() { let chan = makeChan(); let listener = new CheckOnlyHttp2Listener(); await altsvcSetupPromise(chan, listener); }); // also after all connections are closed. add_task(async function test_no_http3_after_error2() { Services.obs.notifyObservers(null, "net:prune-all-connections"); let chan = makeChan(); let listener = new CheckOnlyHttp2Listener(); await altsvcSetupPromise(chan, listener); });