-
-
Notifications
You must be signed in to change notification settings - Fork 145
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 client: Expose maximum time to keep alive idle connection with persistent connections #488
Comments
Hey @pbrisson! So I fully agree this should be a configuration option. However, there are a few things at play why we didn't go for it out of the box. First of all, as @clue mentioned in #486, we first want to gather some feedback before enabling this by default. The other side of this is we need to come up with a decent and future proof way of passing in configuration. While reviewing this I also flagged this with @clue that for now and he agreed we need to do something about this. But for now to first get it in and gather production feedback on this feature before tweaking it. If you lower the number to 1s or 0.5s. At which point do you gain/lose that 100ms improvement? Another thing is that if we raise it to a certain level the loop stays active and we probably should add a close method to the Browser to instruct it to close those idle connections. |
I agree what @WyriHaximus said here, it makes a lot of sense to introduce a configuration for the
@clue mentioned a reason for this in #486, saying:
Using a small time of 1ms made sense as a first step in order to get the feature out without bloating the whole ticket, because raising the currently set 1ms would involve a lot more test and edge cases that need to be covered. I think we'll keep the short timeout as a default in the future, while allowing to adjust the timer to your needs, so to conclude, this is definitely on our roadmap for HTTP, also mentioned by @clue in his pull request:
I hope this answers your questions and shows some of our plans for further development of HTTP keep-alive 👍 |
I also agree that 1ms is too short a default and the value should at least configurable. |
@WyriHaximus @SimonFrings Thanks for your answers. I think users of the library can only benefit of larger timeouts as multiple requests being made to the same URL are just reusing the existing connection then (whereas the 100ms improvement I mentioned). This said, it totally makes sense that you wanted to minimize the impact of that feature while it's being tested! I'm happy to help testing that feature when it's implemented. I will refrain from submitting a PR as I said, I'm aware it represents a lot of refactoring to expose the variable that will make the timeout configurable. For now I switched to a pure React\Socket implementation, which is more flexible for my use case. |
@pbrisson Good input and I completely agree that the maximum time to keep alive idle connections should be configurable 👍 I have indeed implemented the HTTP keep alive feature with a 1ms default in #486 as a starting point only and to add more options in follow-up PRs. The 1ms default was specifically chosen as a reasonable default to get more feedback how people plan to use this feature and because the 1ms default already significantly improves common use cases like following redirects and handling queued requests as described in https://github.com/reactphp/http#concurrency without exhibiting too many side-effects. Of course, there are plenty use cases where a higher maximum time to keep alive idle connections can be useful. For example, if you constantly poll a remote API every 3s, you're probably okay with reusing the same connection and would want to avoid the overhead of recreating the underlying connection for each request. However, higher timeout values also introduce new problems that are next to non-existent with the current 1ms default. Idle HTTP/1.x connections may be closed by either side without notice which means there's has an inherent race condition that might happen if the server closes the connection exactly when the client sends the next request. Both sides may specify the HTTP On top of this, ReactPHP's event loop will keep running as long as there is something to do. This means that if we would keep alive idle connections for 60s, the following example would keep running for 1s + 1 request roundtrip + 60s. $browser = new React\Http\Browser();
$response = React\Async\await($browser->get('https://example.com/'));
assert($response instanceof Psr\Http\Message\ResponseInterface);
var_dump($response->getStatusCode());
Loop::addTimer(1, function () use ($browser) {
$response = React\Async\await($browser->get('https://example.com/'));
assert($response instanceof Psr\Http\Message\ResponseInterface);
var_dump($response->getStatusCode());
}); For long running applications such as server daemons this might not be a problem, but this clearly demonstrates why the current default of 1ms is less problematic (see clue/reactphp-redis#130 and others for similar issues). From the HTTP client perspective, this situation is rather hard to detect, as there's currently no reasonable way to know no further request will (or can) be sent. That said, I agree that the maximum time to keep alive idle connections should be configurable to give consumers of this project more control for more specific requirements. I've looked into adding a new Long story short: This is definitely going to happen, but requires significant work. For HTTP keep-alive as implemented in #486 and referenced tickets, you're already looking at several weeks(!) of work, so I think this might easily be one of the largest feature additions in ReactPHP lately. We now have a solid base for this feature, so I'd like to thank our sponsors for making this possible! If you'd like to support this development, please reach out or consider sponsoring ReactPHP! ❤️ |
Hi there!
Good job with taking care of #468. Having the ability to use persistent connections is definitely a game changer for some use cases!
I was wondering if it would be a good idea to expose the $maximumTimeToKeepAliveIdleConnection variable (https://github.com/reactphp/http/blob/1.x/src/Io/ClientConnectionManager.php line 39) in order to support ongoing communication with a webserver, especially with REST API and architectures such as OpenResty.
I'm aware it represents some work as a lot of objects have to be exposed (client, transaction, sender, etc.), but I was wondering if it would be worth it for that particular use case. I noticed an improvement of 100ms per chained request on average by changing the timeout from 1ms to 10s.
Is there a reason why connections are considered idle only under 1ms? I think it might be too low (which is the reason why I'm flagging this as an issue and not a suggestion).
Thanks a lot for ReactPHP!
The text was updated successfully, but these errors were encountered: