Maintaining a constant response time on a server with high network traffic while using PHP is probably the hardest & most annoying thing I've done in my career. Switching from Apache to Nginx was a huge performance upgrade... that lasted for a solid half hour before the RPS (requests-per-second) was causing the stack to overflow once again.
I started browsing Stack Overflow & Google (devs best & worst friend/enemy) for answers, I found a few posts that showed how to fix the issue (PHP-FPM conf) but didn't explain anything. As a hands-on developer, I wanted to know how things worked.
Originally my PHP-FPM conf looked something like this
[www]
user = apache
group = apache
listen = 127.0.0.1:9000
listen.allowed_clients = 127.0.0.1
pm = dynamic
pm.max_children = 200
pm.start_servers = 20
pm.min_spare_servers = 10
pm.max_spare_servers = 20
pm.max_requests = 1000
I started reading into what pm = dynamic
meant and I found this (in another conf file haha)
dynamic - the number of child processes are set dynamically based on the following directives. With this process management, there will be always at least 1 children.
Hold on... are all 200 children being spawned on startup? Are they always idle?
Yes & Yes (i think). I started viewing and recording memory usage of the apache
pool using ps aux |grep apache
. No matter how many requests were being processed (0 requests - 1000 requests) there were always 200 children alive. Don't get me wrong, I love kids but 200 at once for no reason is a bit much.
After spending a few hours screwing with my PHP-FPM conf and running stress tests I came up with this
[www]
user = apache
group = apache
listen = 127.0.0.1:9001
listen.allowed_clients = 127.0.0.1
pm = ondemand
pm.max_children = 200
pm.process_idle_timeout = 1s
pm.max_requests = 1000
I ran another stress test: 2000 RPS (requests-per-second) for one min and the average response time went from 1000MS to 120MS.
What happened?
PHP-FPM's PM (Pool Manager) was spawning 200 children on startup, even though those children were idle the PM was using unnecessary resources to manage the children. The switch from dynamic
to ondemand
allowed children to be spawned when needed and killed them after 1s of inactivity.
Feel free to critique this post, I have little knowledge of how PHP-FPM's pool manager works. I felt that it might be helpful to someone in a bind with PHP-FPM.
EDIT: Some info in this article is incorrect e.g I confused dynamic
with static
spawning 200 children on startup. However, I still use this same setup on new servers and it performs so much better than any other config I've used.
Top comments (8)
Manually configuring PHP-FPM with nginx and apache is quite difficult for someone who doesn't have sysadmin experience. They might not even know what is php-fpm. I think it is better to use some tool that automates this process or at least makes it easier.
This article wasn’t directed at users who don’t know what PHP-FPM is.
Got curious with your comment, what are the automation tools?
^ I’d like to know them as well. I use Laravel Forge to provision the server and setup PHP-FPM but I still make configuration changes.
Your article is full of factual errors about
dynamic
,ondemand
,static
. Please fix them.RTFM:
What is your setup?
RAM and CPU?
Really depends on the project, I’d always run at least 4GB of RAM and 2 CPU’s. That works great for small-med projects, I believe the server that I referenced in this post had 18GB of RAM and 8 cpu’s I’m not 100% sure.
so
wht is the best value for
PHP-FPM Pool
Options Max Requests =
Process Idle =
Timeout Max Children=
??