关于 php-fpm reload 会取消正在处理的请求的解决方案

起步

在测试中,发现 php-fpm reload 会强制 kill 掉正在处理的请求。网上查了一下,发现其他人也有这个问题并反馈给了官方:https://bugs.php.net/bug.php?id=75440https://bugs.php.net/bug.php?id=60961,帖子是 2017 和 2012 年的,到现在还没解决。

官方帮助手册还说 reload 是 graceful ,啊哈哈,不要太相信:

man php-fpm
...
SIGINT,SIGTERM
    immediate termination
    SIGQUIT 
graceful stop
    SIGUSR1 
re-open log file
SIGUSR2
    graceful reload of all workers + reload of fpm conf/binary
...

reload 流程简介

php-fpmmaster worker 的工作方式。

php-fpm master 进程通过接受用户发送的 SIGUSR2 信号实现自身服务的 reload:

kill -USR2 <pid>

主进程(master进程)收到 reload 信号,会向所有子进程发送 SIGGUIT 信号,同时注册定时器时间,timeout 的值为 fpm_global_config.process_control_timeout 。在规定时间之内子进程还没有结束,则子进程将被 kill 。比如 timeout 值设为1秒,如果在 1 秒之内还没有结束,则直接向子进程发送 SIGKILL 信号,强制杀死。

最后 master 等待所有的子进程结束后,根据之前保存的启动参数重新启动一个进程,并继承父进程的 socket 文件描述符。

缓兵之计

注意,这只是缓解的方案,依然不能保证请求不会丢失。这个方案在于 process_control_timeout 这个配置选项,配置文件在 php-fpm.conf (我的是在 /usr/local/etc/php-fpm.conf)中,默认值是 0 ,会立即将子进程 kill 掉,这里我改为了 60s 进行测试:

; Time limit for child processes to wait for a reaction on signals from master.
; Available units: s(econds), m(inutes), h(ours), or d(ays)
; Default Unit: seconds
; Default Value: 0
process_control_timeout = 60s

测试结果,正在处理的请求只要在该时间内完成请求,就能正常返回。

这不是 100% 的方案是因为,master 进程要等待所有子进程结束才会重新创建 worker 进程,而 process_control_timeout 等待的时候,worker 进程不接受请求了,因此这段时间内新的请求进不来,这些新请求将由 fpm 排队,nginx 若超时会报 502 给用户,保险起见,nginx 的超时时间的值应该是 process_control_timeout 的两倍。

尽管可能会报 502 ,但这样的处理方式比杀死正在处理的请求让人接受的多了。

总结

尽管设置了 process_control_timeout ,在上述情况之上,PHP-FPM 在 reload 完成之前不会为新请求提供服务。但是,所有这些新请求将由 fpm 排队,并在重新加载完成后立即执行。最终用户的结果是,在此期间,他们看到浏览器显示加载中。另一点是设置的超时,也不能保证请求在这个时间内处理完,还是需要程序员保证自己的脚本运行时间在合理范围内。


编程进阶之路
清茗浮碗,文书闲展,与君俯首三食对案,两相顾颜欢。

宝可梦情怀粉;刀塔手残党;浴室麦霸王。

3.2k 声望
232 粉丝
0 条评论
推荐阅读
如何使用 PHPStorm 进行优雅的项目开发?
PHP Storm 这个开发工具,很多 phper 应该有所耳闻,甚至也有不少人使用其作为生产工具,但是很多人都没有最大限度的使用它,本文就来总结一些优雅开发的小技巧。

唯一丶45阅读 4.8k评论 7

怎样用 PHP 来实现枚举?
在数学和计算机科学理论中,一个集的枚举是列出某些有穷序列集的所有成员的程序,或者是一种特定类型对象的计数。这两种类型经常(但不总是)重叠。枚举是一个被命名的整型常数的集合,枚举在日常生活中很常见,...

唯一丶25阅读 6.4k评论 4

PHP转Go实践:xjson解析神器「开源工具集」
我和劲仔都是PHP转Go,身边越来越多做PHP的朋友也逐渐在用Go进行重构,重构过程中,会发现php的json解析操作(系列化与反序列化)是真的香,弱类型语言的各种隐式类型转换,很大程度的减低了程序的复杂度。

王中阳Go10阅读 2k评论 3

封面图
图片防盗链破解 解决图片防盗链问题 反向代理
当客户端(浏览器)向服务器请求内容的时候,会提交一个header,这个header中包含了如:浏览器信息、cookie等内容,那么有一个叫referer的东东,也包含在这里面。

TANKING7阅读 11.3k评论 5

Git操作不规范,战友提刀来相见!
年终奖都没了,还要扣我绩效,门都没有,哈哈。这波骚Git操作我也是第一次用,担心闪了腰,所以不仅做了备份,也做了笔记,分享给大家。问题描述小A和我在同时开发一个功能模块,他在优化之前的代码逻辑,我在开...

王中阳Go5阅读 2.3k评论 2

封面图
Hyperf 3.0 发布,PHP 新时代
在过去的一年半时间里,Hyperf 2.2 共发布了 35 个小版本,使 Hyperf 达到了一个前所未有的高度,这里也获得了一些不错的数据反馈。

huangzhhui4阅读 1.1k评论 1

封面图
微信公众号开发:自动回复文本/图片/图文消息/关键词回复/上传素材/自定义菜单
对接流程1、申请微信公众号测试账号URL:[链接]2、登录,配置开发者服务器URL和Token开发者服务器配置代码:config.php {代码...} URL是config.php在你服务器的URLToken是上面代码自己设置的Token搞定之后,就能完...

TANKING2阅读 10.1k

宝可梦情怀粉;刀塔手残党;浴室麦霸王。

3.2k 声望
232 粉丝
宣传栏