- 
                Notifications
    You must be signed in to change notification settings 
- Fork 70
Deleting unused variables after Response return #154
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
Conversation
- This is part of this discussion php-pm/php-pm#448 # Rationale HttpKernel handle method is injected as an anonymous function, and because of that, internal memory positions used inside this function is only removed when the GC is applied. In order to avoid memory incremental, we can set to null these variables, and this used memory space will automaticaly be released. # How to check the impact - Create an application using HttpKernel as the bridge - Add this line before the response is returned in the method `handle`. - `syslog(1, memory_get_usage(false));` - Use the application and tailf the syslog. - You will find that the memory increases after each call - By adding these lines, the memory will be stable (you may not see each line the, because syslog group them when are equal)
| Is there any proof that this works? Imho releasing a variable does not return any memory unless gc actually runs. When the anonymous function returns, it will automatically release all local variables. That is- unless there are any other references in the system to those. So imho this is not a cure but treating the symptoms. If this PR really changes the behaviour we should analyse the released variable one by one and check which one triggers the growing memory consumption and then figure out where that is being referenced outside of the closure. | 
| Setting a variable to NULL it releases the memory at the moment. Difference between  | 
| @andig it has nothing to be related with referencies, but a behavior that PHP has with anonymous functions. You can proof that it works by just following the instructions I pointed to you in the PR description. | 
| 
 Could you provide before/after test case and measurements? Is this caused by any specific variable? 
 I would appreciate a link to the docs if possible. | 
| @andig https://stackoverflow.com/a/50931077 I will post the measurements later | 
| ping @mmoreram did you get any proof that this changes the behaviour? Pleas note that the SO link is ages old and php has matured a lot since that time. | 
| @andig You can check a comparision between having this improvements inside the kernel, and without it. https://gist.github.com/mmoreram/d10b40e7745e80617dfc6753ef117016 The test is simple. An active active and a tests battery working against it. In each request / response iteration, a simple syslog print of the current used memory. As you can see, without improvements, the memory grows. With improvements, the memory stays much more stable over the time :) | 
| We should add a comment above those lines explaining that circumstances, so our future self will know what the f was/is wrong with PHP. | 
| I realize I'm a sceptic here as this is beyond what I've experienced with php. Are these tests comparable, i.e. why do they already start at different base memory consumption after the first request? 
 What behavior is that? If nothing holds on to a reference it should be garbage-collected. update are you potentially talking about https://bugs.php.net/bug.php?id=69639? Then  | 
| @andig OK. IMHO, forcing  Now is up to you to accept it or decline, but I'm not going to enter a discussion like this. Good luck. update BTW. Both tests are exactly the same. Same tests battery, same code, just changing the lines I exposed. | 
| 
 It isn't, full ack! 
 Not sure what you mean? You've specifically mentioned 
 and I'd appreciate any reference you have for this claim. Update for reference: there is also a discussion in reactphp/http#289 | 
| I know this PR isn't a cure and probably just treats the symptoms of a deeper issue (maybe in reactphp? maybe in PHP itself?), but if it does work and reduces long-term memory consumption shouldn't we merge it anyway? The changes don't seem very risky and we can always clean it up later if the root cause is discovered and it's no longer needed. | 
| @andig when I say that I don't want to enter a discussion like this, I mean opinions discussions. I exposed some examples and demonstrations that these changes improve the behavior of the package. I really LOVE this project, and improving it is one of my project Apisearch goals. Said that, in Apisearch we have implemented our own Bridge and Bootstrap. This is why I want the maximum performance and avoiding if/else blocks is a must for me. In my Bridge I added these "=null" lines, and my containers are now pretty stable in AWS, instead of being killed each 16 minutes. @dnna I think that there's not a root cause. The cause itself is how PHP and GC works. Deleting variables inside anonymous functions in PHP have always been a MUST if these variables can have a big size (like a Request, or a Response objects). Deleting them is a good practice (has been since there isn't an Apache behing the project killing the PHP thread after each response return). BTW, @andig, I can set my own code in the Bridge and this would improve the memory allocated in each of the slaves, but if I want to change the behavior in the server itself, I should fork the project in order to add my  Thanks for the responses :) | 
| This is not about opinions. My question is about a claim you've made: 
 
 I'd simply appreciate an answer in order to better understand the root cause. If there isn't one then fine, too, and let merge as-is. | 
| @andig ups, I was not talking about this. For example - https://bugs.php.net/bug.php?id=75914 Everyone tell that calling  | 
| So, @mmoreram does alternatively calling  | 
| @marcj never mind. My project is running as I was expecting now with my own Bridge, but seems that in PHP  | 
| @marcj must say that I prefer nulling over  | 
| @andig, you're probably right 👍 | 
| @andig @marcj I did some investigation after reading a bit more about this bug, and the leak seems to be in the block: Commenting out this part causes the memory to be stable in the same way like in @mmoreram 's example. I'll investigate further to see what exactly is causing this leak and make a PR for it. | 
| @dnna, very good find! 🤩 | 
| GREAT! This looks exactly like the referenced bug. Closure is created like which will have access to  | 
| I tried the static part but it still leaked. At the end I just declared it as a normal private function and that seems to solve the issue. | 
| Actually, its not because it isn't a classvar. Mhhhm... | 
| 
 Funny :O /cc @kelunik maybe you could shed some light on this? | 
| Could it be because it uses  | 
| 
 | 
| Big thank you- finally an explanation. My world has just shifted back into order where laws of physics apply. 🌎 It is a private method thanks to @dnna now. No idea why the ever was different... | 
Rationale
HttpKernel handle method is injected as an anonymous function, and because of that, internal memory positions used inside this function is only removed when the GC is applied.
In order to avoid memory incremental, we can set to null these variables, and this used memory space will automaticaly be released.
How to check the impact
handle. -syslog(1, memory_get_usage(false));