-
Notifications
You must be signed in to change notification settings - Fork 30.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
HTTP 1.1 fallback for HTTP 2 client #31759
Comments
This might be a duplicate of #16256 but it’s unclear if that was actually resolved, it certainly doesn’t seem to be documented yet. |
It's definitely possible but the flow is a bit difficult. Essentially there are three possible approaches:
Also consider that this is going to be even more difficult with the addition of quic/http3. There's no way to upgrade or failover since it's udp instead of tcp. The thing that has made this supremely difficult is how much existing http frameworks on top of node.js base apis monkeypatch and work around the core API. It was impossible to get these working together without breaking existing stuff or making things far more complicated (and slow) |
Can we get an example, maybe as a working test, for this? Once I know the flow I can get the right abstractions in place to make it easier.
Ya, that’s fine, one problem at a time 😉 What is the plan for browsers in the regard? |
Ok... so, first, you need to decide if your client is going to permit using plaintext HTTP/2. The spec says it's ok, browsers require it, and I wouldn't be surprised if most middleboxes that support it require it as a result. Next, you need to decide: are you wanting to discover HTTP/2 support (and upgrade if available) or assume HTTP/2 support (and downgrade if not available). The flows differ for each choice. Plaintext HTTP/1 to HTTP/2 Upgrade (Discover): Start by preparing an normal HTTP/1 request. GET or OPTIONS works best here. Include in the request an When that request is sent to the server, if the server supports HTTP/2, it should upgrade the connection just as it would any upgrade request. If you're familiar with the upgrade to WebSockets then you know what to expect here. Assuming the server upgrades the connection, it will apply the Plaintext HTTP/2 to HTTP/1 Fallback: You can choose to optimistically initiate an HTTP/2 request with the server. The initial HTTP/2 handshake is an intentionally malformed HTTP/1 method. If the server supports HTTP/2, then it will understand that you want to use HTTP/2 and will just continue working. If the server only supports HTTP/1, then it will respond with an HTTP/1 Unsupported Method error, if you get that, you need to fall back to using HTTP/1. TLS HTTP/2 Negotiation: If you are using TLS, the client uses the TLS ALPN extension to identify whether it wants to use HTTP/1 or HTTP/2. This is the easiest choice but the current implementation of the HTTP/1 and HTTP/2 APIs in Node.js would require you to first set up the TLS connection, establish which protocol you're using, then attach the TLS socket to either the HTTP/1 or HTTP/2 API. It's a bit painful. ALTSVC: With the Altsvc flow, any server that supports HTTP/2 (or in the future HTTP/3) can advertise that fact in the Altsvc header (or the Altsvc HTTP/2 extension frame). A client can send an initial HTTP/1 Options or GET request to solicit a response that contains the header, then use that to determine the location of the HTTP/2 server. |
Since all the browsers are Chromium now, and since Chromium is supporting QUIC, it'll just work. I'd need to check what their strategy for detecting availability is tho. Fairly certain they're going with the Altsvc option tho. Will try to verify. |
There's this, but I don't know how up to date it is: https://www.chromium.org/quic/quic-faq ... use of |
There's also this in the HTTP/3 spec itself... essentially, Altsvc is the only currently standard discovery mechanism for QUIC, but that may change: https://tools.ietf.org/html/draft-ietf-quic-http-25#section-3.2 |
nope, just TLS is fine. How common is the ALTSVC support? Are we all that likely to encounter HTTP/2 servers that only support the other discovery method?
That seems like it would be the only option for supporting all 3.
With the Altsvc method are you not able to re-use the TLS connection when it’s used for HTTP/2? It’s a bit much to open and then throw away a connection like this. |
Yes, you can so long as you are absolutely certain that (a) the altsvc advertises the same port and (b) you're absolutely certain there's no other traffic going over that connection. (The HTTP/2 implementation needs to take complete control over the socket). I don't currently have stats on how common it is to share both implementations over a single port. I would say that at this point in time, Altsvc and ALPN are the two most common approaches. Plaintext HTTP/2 is rare to find and likely not worth optimizing for. I've been kicking around the idea of adding an Altsvc client implementation to core. Essentially something like: const { getAltSvc } = require('http')
async function discover() {
const services = await getAltSvc('http://example.com')
if (services.h2) console.log('http2 available at ', services.h2)
if (services.h3) console.log('http3 available at ', services.h3)
} (this can just as easily be an npm module.. I'd be shocked it if didn't exist already) If the (Oh, and to make things even more complicated down the road: there's been talk about making a DNS ALTSVC record and there's work ongoing to implement DNS-over-HTTP2 and DNS-over-QUIC 😄 ) |
Here's the alt-svc header returned when you send a request to google.com: Notice that it's using the same port for each. All of these are QUIC/HTTP3 so you wouldn't be able to reuse the same connection. Where reusing the same TLS connection would be problematic is when the implementation requires use of ALPN to specify the protocol that would be used on that connection. That is, if you create a TLS Connection on port 443 assuming HTTP/1 to get the Altsvc header, and it advertises HTTP/2 also on port 443, reusing the same connection may not work given that the connection was not started with the right ALPN identifier. It depends entirely on how the server side was implemented to behave and I'd be super surprised to find many that behave well. |
I would use this module 😉 |
any movement on this? i’d like to find a way to add support for all this to is there a module to get the |
I've been working on a package that does ALPN negotiation (http2 / http1). We already have this in Got via the https://github.com/szmarczak/http2-wrapper#http2autourl-options-callback |
I discussed this a while back with @jasnell and I forget where it all landed and what was needed.
I’d like to make it so that
bent
uses HTTP2 when available and falls back to HTTP 1.1 transparently when it’s not available.This should be possible, the standard was certainly designed that way, but I can’t find any examples of doing this in Node.js and the core http client libraries are totally separate. It could just be a documentation issue but I think there might actually be some missing API in order to make it work.
The text was updated successfully, but these errors were encountered: