DEV Community

Jake Casto
Jake Casto

Posted on • Edited on

Configuring PHP-FPM For High Network Traffic

php-fpm logo

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


Enter fullscreen mode Exit fullscreen mode

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


Enter fullscreen mode Exit fullscreen mode

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)

Collapse
 
azazqadir profile image
Muhammad Azaz Qadir

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.

Collapse
 
jake profile image
Jake Casto

This article wasn’t directed at users who don’t know what PHP-FPM is.

Collapse
 
aliuadepoju profile image
Wuyi Adepoju

Got curious with your comment, what are the automation tools?

Collapse
 
jake profile image
Jake Casto

^ 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.

Collapse
 
reedzhao profile image
reedzhao

Your article is full of factual errors about dynamic,ondemand,static. Please fix them.

RTFM:

static  - a fixed number (pm.max_children) of child processes;
ondemand - no children are created at startup. Children will be forked when new requests will connect.
dynamic - the number of child processes are set dynamically based on the following directives.
Collapse
 
luissiqueira profile image
Luis Siqueira

What is your setup?
RAM and CPU?

Collapse
 
jake profile image
Jake Casto

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.

Collapse
 
adam_kramer02 profile image
Adam_Kramer ( Test Bank and Solution Manual)

so

wht is the best value for

PHP-FPM Pool
Options Max Requests =

Process Idle =

Timeout Max Children=

??