-
Notifications
You must be signed in to change notification settings - Fork 30k
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
Default http[2s] server timeout is problematic #27556
Comments
Timing out and closing the socket after two minutes have elapsed is surprising and problematic for users. This behavior was specific to Node.js, and doesn't seem to be common in other language runtimes. Fixes: nodejs#27556
#27558 removes the default value on the I propose that we add an environment variable ( |
To elaborate, it would be simpler to add the environment variable across all branches, but that has the following downsides:
|
@ofrobots Does it need to be its own environment variable? It could be a command line option that could be allowed under |
^--- I'm not sure if the config option should be exposed outside of the JS API, but if it is, I'd prefer a CLI option that can go in NODE_OPTIONS. |
The reason for allowing configurability outside of the JS API is that hosting environments (e.g. serverless environment, specially serverless containers) usually don't have control over user code, but they do have control over the process environment. If it is exposed only via the JS API, then all we can do is document this for the user as a pitfall. @sam-github @richardlau cli flag is probably be fine. |
#27704 is a semver minor change to 12.x to fix this. Once that lands, it can be further back-ported to 10.x and 8.x. |
Make it possible to override the default http server timeout. Ideally there should be no server timeout - as done on the master branch. This is a non-breaking way to enable platform providers to override the value. Ref: nodejs#27558 Ref: nodejs#27556
Make it possible to override the default http server timeout. Ideally there should be no server timeout - as done on the master branch. This is a non-breaking way to enable platform providers to override the value. Ref: #27558 Ref: #27556 PR-URL: #27704 Refs: #27558 Refs: #27556 Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: Myles Borins <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
Make it possible to override the default http server timeout. Ideally there should be no server timeout - as done on the master branch. This is a non-breaking way to enable platform providers to override the value. PR-URL: nodejs#27704 Refs: nodejs#27558 Refs: nodejs#27556 Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: Myles Borins <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
Make it possible to override the default http server timeout. Ideally there should be no server timeout - as done on the master branch. This is a non-breaking way to enable platform providers to override the value. Backport-PR-URL: #27939 PR-URL: #27704 Refs: #27558 Refs: #27556 Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: Myles Borins <[email protected]> Reviewed-By: Ruben Bridgewater <[email protected]>
As pointed out by @natlibfi-arlehiko in #31378, LTS documentation on this topic is misleading: Lines 886 to 896 in d617fdf
Same on v10: Line 829 in 308b8b9
|
Prior to Node.js v13, http[2s] have a specific default timeout value which should not be confused with net.Socket default timeout. Fixes: nodejs#31378 Fixes: nodejs#27556 Refs: nodejs#27704
Hello, I see the change has been merged already I wonder how this got merged into upstream without being caught, If the timeout is removed this opens up a possible security vulnerability. Consider the following scenario I have node server running on linux with default file descriptors which is 1024, all an adversary has to do open 1024 connection and not send any data to break your system. There is an issue open on golang issue tracker (golang/go#16100) to add timeout support to avoid this problem since author seems to compare this with go as one of the language. |
I just want to note that this comment was addressed in https://nodejs.org/en/blog/vulnerability/september-2020-security-releases/. There is a new |
Having no timeout appears to have the consequence that a session without any streams can linger in memory and never close. I think it's rare for any application code (handler) interact with sessions directly. They're likely to interact with the streams. What can happen is that a session can either never create a stream, thus application never knowing a session was created. Or, an application can handle and close a stream, but the session just stays in memory because the client never sent something to go away. In other words:
OR
I see multiple solutions:
If I had more low level experience, I could probably create an exploit that would bankrupt an HTTP2 server of its RAM (DOS attack) by spawning sessions with 0 requests, or bad requests and just leaving. Go has a 15-second ping interval if no frames have been exchanged: golang/net#55 .NET has 20 seconds and a default option to only ping by default if there's an active request (aka stream), or always: dotnet/runtime#40257 Chrome uses a 10 second ping interval: https://chromium.googlesource.com/chromium/src/+/c9ec5e3acbfaacef722bbe5505d7998dfe39edf1/net/spdy/spdy_session.cc#109 |
Today, Node.js http servers timeout and close the socket after 2 minutes. This means that if a request needs to do work that takes longer than 2 minutes to complete; the socket would be unceremoniously closed and the client will get an empty response.
This is a common problem (search results) that users have to work-around (but most don't know about it). This is specially problematic in serverless environments where users may be doing non-interactive event processing (e.g. photo / image transcoding, uploading large datasets somewhere). Users have to know that this problem is possible and must adjust the timeout to make sure the service works correctly.
This is problematic for PaaS hosting providers where the best we can do is document to our users that timeout must be adjusted. This is specially a problem for serverless containers (e.g. Google Cloud Run).
IMO the default value should be removed:
I would like propose that we remove the timeout. I am working on a PR to do so. Alternatively, If there is strong reason for us to have a default timeout value, let's clearly articulate the reason, and at least provide environment knobs to allow e.g. Serverless hosts to configure a different default value.
Thoughts? Opinions?
/cc @nodejs/http @nodejs/http2
The text was updated successfully, but these errors were encountered: