-
Notifications
You must be signed in to change notification settings - Fork 30.2k
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
Keep alive connection do not get closed with server.close() #2642
Comments
Thanks for opening an issue. I think a similar solution to this issue is being pursued in #2534. |
@kanongil #2534 is about HTTP and HTTPS keep-alive, as indicated in pull-request title |
Rather than an array, you should probably consider a WeakMap now. It is overhead, but that's still better than a broken feature. |
@tshemsedinov Right you are, though the partial fix part still stands. The main problem with the tcp timeout fix is that malicious clients (intentional or otherwise) can keep the connection alive by creating new requests on the socket. Eg. when polling for a value every 2 seconds. |
Just thought I would mention again that the main issue here is that it is possible that sockets never get destroyed. To prevent the overhead @tshemsedinov is talking about, I still think it would be best to do what I mentioned in the description. Which is destroying sockets on the next request to prevent refreshing the timeout for that socket. I think everyone can handle if all sockets are guaranteed to be destroyed within the timeout period. |
Depends what the timeout period is :) It's been several minutes up till now, right? That's too long for my process to shutdown. |
Good point.
I think this callback would be another good place to karate chop these sockets. That should allow all sockets to be destroyed within a second with no extra overhead. |
This should be fixed or noted in the documentation for |
The documentation seems pretty clear to me - "Stops the server from accepting new connections." - but if you think it can be improved, please file a PR with your suggested changes. Do consult CONTRIBUTING.md first, though. |
The clarity of the documentation is not the problem. The problem is the fact that this implementation of the http server as it now stands prevents a person from gracefully shutting down an HTTP server. |
Define 'gracefully'? If you mean 'forcibly close client connections', then no, it doesn't do that, and that's deliberate. OP's suggested change is a no go because it makes it impossible to keep existing connections open indefinitely (which is a use case that should be supported), whereas force-closing can easily be implemented on top of the current behavior - just maintain a list of open connections. There is probably already a npm module for that. |
While tracking the connections is somewhat simple, actually determining the state of these, and acting on it, is not. As far as I am concerned, a graceful server close should:
For 2., the action is simple (a The major pain point is 3., which I don't think is possible to solve using public APIs, and maybe not even using private ones, as some of the state is captured in a closure scope. |
If anyone is interested, I coded up a function that will enable graceful closing on an http server. Heads up that it's written in TypeScript, but it shouldn't be too hard to understand. Hopefully this is helpful to others that have been wrestling with this. |
Spent quite a bit of time debugging this issue today. The Internet seems to believe this is how |
@bnoordhuis I don't think the algorithm described in #2642 (comment) should be the default for close, but I think it should be achievable based on reading the docs. I've poked idly at this a few times without success. One thing with it is that it is not in fact possible to do a "read close" in TCP, only a "write close", so while its possible to do a write close after replying, without somehow corking/discarding incoming data or requests after the currently being read (if any) request is completely read, its not really possible to do. I think we need to trap the request event, check some "closed" state, and then just ignore the request, and somehow get an event for when any currently pending responses are drained. Maybe that's possible with the http API, but its not obvious to me how. Could be just a doc problem, could be a missing feature. @jinxidoru typescript decreases the readability of your example. everyone here reads javascript, not so typescript. If I understand your code, it destroys underlying connections whether they are in the process of handling a request or not, which isn't exactly graceful. |
Just for context on why I (think) I need this close behavior: I've got an API running on heroku that utilizes keep alive. Heroku, at any time, may send a SIGTERM to my app, asking it to gracefully shutdown. If it does not shutdown within 10 seconds, a SIGKILL follows. My API reacts to SIGTERM by cleaning up my timers and database connections, calling What I'm thinking about doing is just calling |
If you have a hard deadline, that's what effectively happens anyway. You (i.e. node) may have written the response to the socket but that doesn't mean the kernel has sent it or that the remote end has received it. They may be on a slow or congested uplink. |
Networks are unreliable, sure. That's why we have TCP. But using the network (or kernel) as an excuse for poor behavior elsewhere only compounds the problem. Just because the message might not get sent doesn't mean we shouldn't try to gracefully shut down incoming connections. |
@davidmurdoch based on your description, I think you're solving the issue well - and also, your app shouldn't be subject to a 10-second downtime during Heroku's restart. The downtime will be however long it takes your app to start (which for node apps tends to be about 1 second): Additionally, during that time your app won't be fully down - instead, Heroku's router will queue requests which will be sent to the new app instance as soon as it listens to |
@nodejs/http |
I wouldn't mind seeing a boolean flag or something that can be passed to |
Only if the server instance tracks socket objects. It's not unthinkable but it seems wasteful when the common case is that it will be unused. Dealing with sockets/servers from the cluster module is also an open problem. In theory we could walk open persistent handles and filter on socket handles belonging to a particular server instance but that is pretty horrid and inefficient. |
@hunterloftis Oh wow, that's awesome! I had ended up deciding to disable keep-alive altogether, so I'm very happy you told me this. Thanks! |
@bnoordhuis I was thinking about the same solution. Seems pretty nasty, but most people are probably fine with close() not being 100% efficient. I'm guessing they will be less fine with typical runtime of an HTTP server slowing down because of the added bookkeeping. Nasty, but if it works... |
when we open these things in libuv we should definitely be doing |
FYI - Here's a simple implementation of tracking & closing keep-alive connections https://github.com/thedillonb/http-shutdown |
@isaacs has a super simple module for this that I use: https://github.com/isaacs/server-destroy |
which node release is this behavior fixed in? |
…/node#2642 has been fixed so server will just do the right thing now.
stoppable seems very inactive and not enough for me, also has issues without reply, so I ended with: |
Added custom destroy function to end all connected socket when closing the server, see nodejs/node/issues/2642 Prevent tests from hanging if connections are still open when server is closing
This is related to nodejs/node#2642 Added stoppable module and stops the server. Also added --exit to the tables test, so it does not hang. Both conditions were holding the test process indefinetely.
This is related to nodejs/node#2642 Added stoppable module and stops the server. Also added --exit to the tables test, so it does not hang. Both conditions were holding the test process indefinitely.
This is related to nodejs/node#2642 Added stoppable module and stops the server. Also added --exit to the tables test, so it does not hang. Both conditions were holding the test process indefinitely.
This is insane.. Also spent 8 hours trying to understand why a server close would take 5 seconds, well that's because of this because the keepAliveTimeout defaults to 5000 Even |
@Tofandel Sounds implausible, it basically calls |
Consider this module: https://github.com/LuKks/graceful-http I based the library on this (plus on real usage): |
I am on server.close((err) => {
if (err) {
console.error(err);
}
console.info("http server closed successfully. Exiting!");
}); best part is I don't have to use |
The following will remain open as long as the client makes a request within one second intervals making it possible that a server will never gracefully close. With the default of 2 minute timeouts a user or api client would need to stop traffic for two minutes before this will take affect. I propose closing sockets using keep alive immediately after the next request. This will prevent needing to keep track of everything that would need to be closed while making the maximum time to gracefully shutdown fixed(and finite).
The text was updated successfully, but these errors were encountered: