WEB应用

php平滑重启nginx,彻底清除WordPress的静态缓存

Jager · 12月6日 · 2014年 · · 6783次已读

每一次分享技术文章,都是基于自己的痛点,基于自己的需求。这次也一样,所以分享具体方法之前,我先说一下我这次的需求与痛点:

一、需求痛点

在博客集成了代码版缓存功能之后,为了方便在前台清理页面缓存,我特意写了ajax 清理缓存的功能相关文章)。这个功能写好之后确实可以正常工作。

但是,为了让网站加载速度提升到极致,我还在 nginx 里面加了类似于 WP Super Cache 的 mod_rewrite 机制:当存在页面缓存时,会绕过 PHP 解析,而直接调取缓存在前台展示。

这样就发现了一个问题:当我在前台点击缓存清理后,后台的页面缓存文件确实是删除了,但是 nginx 却在内存里面缓存了一份!!从而导致一段时间内怎么刷新页面,展示的依然是缓存内容!这样一来,不管是我还是用户,点击前台这个清理按钮根本就不能实时看到效果,明显就鸡肋了!

我发现这个问题的做法是,点击按钮删除缓存,然后进入 Linux 系统去 reload 一下 nginx,才能彻底刷新缓存!我勒个去,每次我调试代码的时候,真心能把人累死(虽然我可以关闭缓存功能,但是我就是要享受一下自己写的清理功能嘛!)。

需求都有了,怎能让技术成为瓶颈?

所以,根据以上需求,很容易得出一个解决方案:当点击前台清理按钮时,php 先删除缓存文件,然后 reload 平滑重启 nginx 就可以实现彻底清除缓存了!

测试了半天,发现难点是 php 如何才能执行 Linux 命令。经过不断测试,终于搞定这个问题,下面开始分享!

二、执行权限

php 执行 Linux 命令有几个前提条件:

①、php 必须开放一些执行外部命令的函数,比如 exec()、system()等;

②、必须赋予 WEB 启动帐号(比如 www 帐号)执行特殊命令(比如 .../nginx -s reload)的权限。

对于问题 ①:

i. 修改 php 配置文件 php.ini,先找到 safe_mode 配置,确认 safe_mode=off,即关闭 php 安全模式(lnmp 一键安装包默认已经是关闭的了);

ii. 继续找到 disable_functions 配置,将其中的 exec 删除,即允许执行 exec()函数;

ii. 最后重载 php-fpm 或 php 即可生效,比如 lnmp 环境可以执行 service php-fpm reload 命令。

Ps:开启 exec 函数存在被恶意注入的风险,不过我这种小博客就没什么好惧怕的,况且我的备份及防护都很完善!

对于问题②:

我们需要在 Linux 中赋予 WEB 帐号使用 sudo 执行指定命令的权限,在这个需求中,我们可以这样做:

#编辑/etc/sudoers 文件:
vim /etc/sudoers

#找到 Defaults    requiretty,并注释掉:
#Defaults    requiretty

#接着在文件最后加上一行允许 www 帐号以 root 身份无密码执行 reload nginx 的命令:
www ALL=(root) NOPASSWD:/usr/local/nginx/sbin/nginx -s reload

#最后按下 ESC 退出编辑模式,键入 :x! 或 :wp! 强行保存并退出 vim 即可。

Ps:操作 vim 需要一定的 Linux 基础知识,不会的童鞋先脑补一下吧!

三、部署代码

①、新增平滑重启 nginx 的脚本

#在 Linux 的 opt 目录新增 reload_nginx.sh 脚本:

[root@Mars_Server ~]# vim /opt/reload_nginx.sh 
#脚本内容(注意 nginx 的实际路径):
#!/bin/bash
/usr/bin/sudo /usr/local/nginx/sbin/nginx -s reload

#保存脚本后,赋读取和执行权限:
chmod +xr /opt/reload_nginx.sh

②、PHP 代码

php 执行这个脚本的代码很简单:

<?php exec(EscapeShellCmd("/opt/reload_nginx.sh")); ?>

Ps:实际上,使用 exec(/opt/reload_nginx.sh"") ; 也是完全可以的。多套了一层 EscapeShellCmd 是为了安全考虑(其实这里貌似没啥必要,算是掩耳盗铃吧!)。

既然知道 php 代码了,那么只要修改上次分享的缓存清理代码,如下新增 26 行和 33 行即可:

<?php
//缓存清理代码(实际使用,请自行修改缓存路径!)
if(isset($_POST['action'])){
    if($_POST['action'] == 'delcache'){
        if($_POST['page_type'] == 'single'){
            $post = $_POST['post_id'];
            $cachefile = "/home/wwwroot/zhang.ge/cache/zhang.ge/".$post.".html/index.html";
            $cachedir = "/home/wwwroot/zhang.ge/cache/zhang.ge/".$post.".html";
        } else if($_POST['page_type'] == 'page') {
            $post = $_POST['slug'];
            $cachefile = "/home/wwwroot/zhang.ge/cache/zhang.ge/".$post."/index.html";
            $cachedir = "/home/wwwroot/zhang.ge/cache/zhang.ge/".$post;
        } else if($_POST['page_type'] == 'category') {
            $post = $_POST['slug'];
            $cachefile = "/home/wwwroot/zhang.ge/cache/zhang.ge/".$post."/index.html";
            $cachedir = "/home/wwwroot/zhang.ge/cache/zhang.ge/".$post;
        } else if($_POST['page_type'] == 'home') { 
            $cachefile = "/home/wwwroot/zhang.ge/cache/zhang.ge/index.html";
        } else {
            exit();
        }
        if($_POST['page_type'] == 'home'){
            if (file_exists($cachefile)) {
                unlink($cachefile);
                //删除缓存后平滑重启 nginx:
                exec(EscapeShellCmd("/opt/reload_nginx.sh"));
            }
        } else if($_POST['page_type'] != 'null') {
            if (file_exists($cachefile)) {
                unlink($cachefile);
                rmdir($cachedir);
                //删除缓存后平滑重启 nginx:
                exec(EscapeShellCmd("/opt/reload_nginx.sh"));
            }
        } else {
           exit(); 
        }
    }
    exit();
}
?>

Ps:若对以上代码有任何疑问,请务必参考上一篇相关文章

全部完成之后,现在在前台使用缓存清理功能,将会先删除缓存文件,然后平滑重启 nginx,从而真正实现在纯静态的前台实时清理缓存,显示最新内容!

四、写在最后

这篇教程算是我这种强迫症&发骚友学习实验之作,而且整个教程并未过多考虑安全问题。所以,如果不是和我这种既不在意被人攻击,又清楚个中原理的人,还是不建议做这种强迫症设置(实际上也就是解禁了 exec 函数存在隐患而已)!

总之,这篇文章分享的方案,还是有一定的参考价值的,根据本文思路,就可以实现在 WEB 界面任意操作服务器了,其实也就和大部分站长用的 Linux 系统面板差不多,希望能给有需要的人提供一些参考,有任何相关问题记得给我留言。

9 条回应
  1. 极品飞鸽 2014-12-7 · 21:27

    这个不错,我用的主机宝linux系统,自带平滑启动nginx,不知道效果一不一样

    • avatar
      Jager 2014-12-7 · 21:32

      原理一样,目的不一样。

  2. 健康之水 2014-12-8 · 10:24

    每一次分享技术文章,都是基于自己的痛点,基于自己的需求。。怎么和我的感觉这么像呢?

  3. 外汇站 2014-12-15 · 14:33

    http://m.zhang.ge/4780.html 手机访问这个页面咋打不开呢,是我plus问题,还是缓存页面问题。

    • avatar
      Jager 2014-12-15 · 16:50

      服务器缓存问题,已解决。

  4. 成航先森 2016-8-17 · 10:59

    生产环境,想用平滑重启,不知道可行不?

  5. showfuli 2017-3-16 · 9:48

    签到成功!签到时间:上午9:48:57,每日签到,生活更精彩哦~

  6. 普通网友 2017-10-24 · 23:54

    用这种方是感觉安全性有问题,你为什么不直接在linux下面写一个定时自动清理的脚本呢?

  7. pigm 2018-3-29 · 2:28

    嘿,我倒是真遇到这个问题了,无论选择第一个还是第二个清除方式,等一会准502,然后测试发现
    我非要service php-fpm restart ,才能恢复访问,这是什么情况...?