升级 PHP 7.4 带来的两个大坑

由于我机器用的滚动更新的 Archlinux,不知不觉 php 已经升级到7.4了,没想到这次更新带来了极大的麻烦。首先是 php-fpm 的新选项 ProtectHome 会导致经典的 File not found 错误,再是 php 解释器会对 null 类型的下标访问直接报错 Trying to access array offset on value of type null

最近在帮一个朋友张罗一个网站,于是把线上代码拉回本地做镜像进行测试。因为web应用有些奇怪的依赖,为了不污染本机的环境,我就把它部署在 Docker 中进行测试。Docker 的基础镜像选择了激进的 Archlinux,搭配上个月底才出炉的 php7.4。于是花了整整一个下午栽在 Debug 大坑中…

首先是一把梭配好了环境后,一跑,报了Php-fpm最经典也是最坑的错误之一:File not found 。配过 php-fpm 的都知道出现这个错误一般是文件权限不对或者文件路径不对,而这两个错误都是比较难找的。于是我又双叒叕体验了一把大眼瞪小眼的路径检查,没问题。文件权限检查,emmm也没问题呀?又返回去检查路径,还是没问题!搞到最后气的 chmod 777一把梭竟然也没能解决问题,有点怀疑人生…

网上搜索 php-fpmFile not found 错误,虽然结果很多,可原因都只有这两个。而这两个原因也都被一一排除了,事情突然向神奇的角度发展起来了...

不知过了多久之后我才想到可能是跟 php 版本有关(因为我本机也跑了其它 php 应用,所以一开始并不觉得 php 有问题)。于是我去搜了一下新版 php7.4 及 php-fpm7.4 的改动,一下就发现了罪魁祸首:php7.4 Commit

这个提交中添加了一个选项:ProtectHome 。顾名思义,开启了之后 php 不会去执行在家目录中的文件 ——而这个新选项的默认值恰好是开启的。使用 systemctl edit php-fpm.service 添加一个选项覆盖,重启服务后,终于一切正常,并迎来第二个大坑错误:

php 中经常使用 inlcude,require 等来包含其它文件。而调试发现在某个 include 之后,php 直接停止执行并报错 Trying to access array offset on value of type null。但是在线上的代码跑起来却一点问题也没有,这就很奇怪了,跟到 include 的文件中之后发现是有个地方在访问数组元素,而数组本身却是 null 。在 php 这种弱类型语言中这种语法一般是支持的,它会整体返回 null ,而在新版的 php7.4 中这个语法却会报告为错误。看来Php也在一点点规范语言的特性,没办法,这个只能自己改代码了。(虽然我目前选择了使用旧版本的 php)

由于 php7.4 在上个月底才刚刚发布,估计还没有大面积更新使用,各个应用的开发者可能也没有针对 php7.4 进行过测试和兼容修改。也正是因此,在网上搜索这些信息时,找不到什么有价值的建议,这篇文章除了记录下被这个新特性坑了一下午之外,也算给其它人留一个解决类似问题的思路吧。

原文地址,作者:sbw Blog

本作品采用《CC 协议》,转载必须注明作者和本文链接
感谢阅读,有收获的话不妨点个赞:smiling_imp:
本帖由系统于 4年前 自动加精
讨论数量: 4

第一次听说有人用 Arch 做 Docker Image 基础镜像😂,「滚着滚着就挂了」2333。

4年前 评论

@Wi1dcard 用 alpine 我都有点心惊胆颤。

4年前 评论

下标问题是开发不严谨,thinkphp文档有句话说得好很:系统产生的异常和错误都是程序的隐患,要尽早排除和解决,而不是掩盖。对于应用自己抛出的异常则做出相应的捕获处理。其实数组下标,未定义变量都会异常,只是很多时候屏蔽了!

4年前 评论

使用姿势不对、代码本身不规范。我这 PHP 7.4 用着就没见需要改啥的

4年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!