前些日子在 破袜子 中 @大致 提到 让WordPress4.9在php8.2下正常运行。 在不知不觉间,PHP已经支持到8.X了。 好快,当初刚接触PHP时还是5.6。 恰好周六,恰好服务器搬家。 索性,在AI的帮助下,尝试让梦幻辰风这个运行在 Emlog5.3.1 版本的部落格,体验到php8的快乐。 通过搜索,在 一年又一年的博客 中找到《Emlog 5.3.1 兼容 PHP 8.0 错误修复》这篇博文,甚至于我还看到今年5月我留下的留言: 感谢大佬分享,我目前还是5.3.1,停留在PHP7.4,想着要不要到8.X的版本。 不过博主没有回我,Ta的更新也停留在2022年2月。 无妨,现在开始,结合 一年又一年的博客 的文章内容和我的操作日志,简单记录一下升级历史。 首先:在2016年或2017年左右,梦幻辰风已经修改版本支持到PHP7.X。 所以,以下所有折腾,都是基于PHP7.0时的版本。而修改Emlog5.3.1支持PHP7.0,我是参考的 星知苑 的博文《PHP7下安装Emlog5.3.1》(注:在本文发布时,Ta的站点已经打不开了。) (注:以下内容好像是CSDN采集的 星知苑 博文,大致方法如下,如果造成影响,后果自负。) 1.修改 include/lib/option.php ,大概11行修改为mysqli //即:默认MySQL连接方式,mysql或mysqli const DEFAULT_MYSQLCONN = 'mysql'; 改为 const DEFAULT_MYSQLCONN = 'mysqli'; //默认链接方式改为mysqli 2.修改 include/lib/database.php ,大概16行删除default: case 'mysql': default ://这边需要删除default: 3.修改 include/lib/cache.php ,大概195行加大括号 $$row['option_name'] = $row['option_value']; 改为 ${$row['option_name']} = $row['option_value']; 4.PHP版本最近更新的比较快,autoload自动加载函数会报错,function.base.php开头的__autoload函数修改如下 spl_autoload_register(function($class) { $class = strtolower($class); if (file_exists(EMLOG_ROOT . '/include/model/' . $class . '.php')) { require_once(EMLOG_ROOT . '/include/model/' . $class . '.php'); } elseif (file_exists(EMLOG_ROOT . '/include/lib/' . $class . '.php')) { require_once(EMLOG_ROOT . '/include/lib/' . $class . '.php'); } elseif (file_exists(EMLOG_ROOT . '/include/controller/' . $class . '.php')) { require_once(EMLOG_ROOT . '/include/controller/' . $class . '.php'); } else { emMsg($class . '加载失败。'); } }); 5.对于部分插件写死了数据库链接方式,需要将 $DB = MySql::getInstance(); 改为 $DB = Database::getInstance(); 等等。 通过上述的修改,Emlog5.3.1(也许)可以支持PHP7.X了。现在,我们将向PHP8进发。 首先,在 init.php 的最开头调用 error_reporting() 设置了错误级别,要排错的话,需要把级别开到 E_ALL 。 *以下方法来自 一年又一年的博客 : 1.修改 /include/lib/function.base.php 文件: 修改 __autoload() 函数: //把 __autoload() 换为匿名函数,然后传给 spl_autoload_register 函数注册 // function __autoload($class) { spl_autoload_register(function ($class) { $class = strtolower($class); if (file_exists(EMLOG_ROOT . '/include/model/' . $class . '.php')) { require_once(EMLOG_ROOT . '/include/model/' . $class . '.php'); } elseif (file_exists(EMLOG_ROOT . '/include/lib/' . $class . '.php')) { require_once(EMLOG_ROOT . '/include/lib/' . $class . '.php'); } elseif (file_exists(EMLOG_ROOT . '/include/controller/' . $class . '.php')) { require_once(EMLOG_ROOT . '/include/controller/' . $class . '.php'); } else { emMsg($class . '加载失败。'); } }); 修补 get_magic_quotes_gpc() 函数 在文件开头 function doStripslashes() {} 函数声明前,补充定义纯用于兼容的桩函数 get_magic_quotes_gpc(),恒返回 FALSE。 if (!function_exists('get_magic_quotes_gpc')) { function get_magic_quotes_gpc() { return false; } } 在 in_array() 前检查 NULL 值 //找到if (!@in_array($actionFunc, $emHooks[$hook])) ,将其修改为: if (!@$emHooks[$hook] || !@in_array($actionFunc, $emHooks[$hook])) { 2.更新 passwordhash.php 密码函数库 前往 phpass 官网下载最新版本的 passwordhash.php ,替换原本的 /include/lib/passwordhash.php 。 现在,程序能跑了,但是会有一些小报错,如: 报错提示:Deprecated: Function get_magic_quotes_gpc() is deprecated in /www/include/lib/function.base.php on line 31 需要修改 include/lib/function.base.php 文件,找到 doStripslashes() 函数,修改如下: function doStripslashes() { // 删除对 get_magic_quotes_gpc() 的调用 // 直接进行 stripslashesDeep 处理 $_GET = stripslashesDeep($_GET); $_POST = stripslashesDeep($_POST); $_COOKIE = stripslashesDeep($_COOKIE); $_REQUEST = stripslashesDeep($_REQUEST); } 同时还需要修改 stripslashesDeep() 函数,修改如下: function stripslashesDeep($value) { if (is_array($value)) { return array_map('stripslashesDeep', $value); } elseif (is_object($value)) { $vars = get_object_vars($value); foreach ($vars as $key => $data) { $value->{$key} = stripslashesDeep($data); } return $value; } else { // 添加对 null 值的检查 return is_string($value) ? stripslashes($value) : $value; } } 同时,元宝AI提示我还有一个完整替代方案:在 init.php 最前面添加全局处理。 // 替换 magic_quotes_gpc 功能 if (version_compare(PHP_VERSION, '7.4', '>=')) { $process = [&$_GET, &$_POST, &$_COOKIE, &$_REQUEST]; while (list($key, $val) = each($process)) { foreach ($val as $k => $v) { unset($process[$key][$k]); if (is_array($v)) { $process[$key][stripslashes($k)] = $v; $process[] = &$process[$key][stripslashes($k)]; } else { $process[$key][stripslashes($k)] = stripslashes($v); } } } unset($process); } 不过我没试,因为已经解决这个报错了。 不过在网站头部出现了: Notice: Undefined variable 。我看了一下代码如下: <?php if ($logid): ?> 修改为: <?php if (isset($logid) && $logid): ?> 即可。 随后,新的报错: Deprecated: Using ${var} in strings is deprecated, use {$var} instead in /www/content/plugins/tpl_options/tpl_options.php on line 987 这个是emlog5.3.1的插件: 模板设置 。 打开 tpl_options.php 第 987 行: // 旧代码(已弃用): $values[$tag['tagname']] = "${tag['tagname']} (${tag['usenum']})"; // 新代码(推荐): $values[$tag['tagname']] = "{$tag['tagname']} ({$tag['usenum']})"; 完整修改后的函数: /** * @param array $option * @return void */ private function renderTag($option) { $tags = Cache::getInstance()->readCache('tags'); $values = array(); foreach ($tags as $tag) { // 修复 PHP 8.2 弃用的字符串插值语法 $values[$tag['tagname']] = "{$tag['tagname']} ({$tag['usenum']})"; } $option['values'] = $values; $this->renderCheckbox($option); } 随后,新的报错: Deprecated: Log_Model::getLogsForHome(): Optional parameter $condition declared before required parameter $perPageNum is implicitly treated as a required parameter in /www/include/model/log_model.php on line 184 修改 log_model.php 第 184 行的函数声明: /** * 前台获取文章列表 * * @param int $perPageNum * @param string $condition * @param int $page * @return array */ function getLogsForHome($perPageNum, $condition = '', $page = 1) { $timezone = Option::get('timezone'); $start_limit = !empty($page) ? ($page - 1) * $perPageNum : 0; $limit = $perPageNum ? "LIMIT $start_limit, $perPageNum" : ''; $sql = "SELECT * FROM " . DB_PREFIX . "blog WHERE type='blog' and hide='n' and checked='y' $condition $limit"; $res = $this->db->query($sql); $logs = array(); while ($row = $this->db->fetch_array($res)) { $row['date'] += $timezone * 3600; $row['log_title'] = htmlspecialchars(trim($row['title'])); $row['log_url'] = Url::log($row['gid']); $row['logid'] = $row['gid']; $cookiePassword = isset($_COOKIE['em_logpwd_' . $row['gid']]) ? addslashes(trim($_COOKIE['em_logpwd_' . $row['gid']])) : ''; if (!empty($row['password']) && $cookiePassword != $row['password']) { $row['excerpt'] = '<p>[该文章已设置加密,请点击标题输入密码访问]</p>'; } else { if (!empty($row['excerpt'])) { $row['excerpt'] .= ''; //$row['excerpt'] .= '<p class="readmore"><a href="' . Url::log($row['logid']) . '">阅读全文>></a></p>'; } } $row['log_description'] = empty($row['excerpt']) ? breakLog($row['content'], $row['gid']) : $row['excerpt']; $row['attachment'] = ''; $row['tag'] = ''; $row['tbcount'] = 0;//兼容未删除引用的模板 $logs[] = $row; } return $logs; } 然后需要搜索整个代码库,找到所有调用 getLogsForHome 函数的地方,并调整参数顺序。 参照如下: // 旧调用方式: $logs = $Log_Model->getLogsForHome('', $page, Option::get('index_lognum')); // 新调用方式: $logs = $Log_Model->getLogsForHome(Option::get('index_lognum'), '', $page); 当然,我没这么干,因为太多了。 我选择保持向后兼容,可以修改函数签名但不改变参数顺序: /** * 前台获取文章列表 * * @param string $condition * @param int $page * @param int|null $perPageNum * @return array */ function getLogsForHome($condition = '', $page = 1, $perPageNum = null) { // 如果 $perPageNum 为 null,使用默认值 if ($perPageNum === null) { $perPageNum = Option::get('index_lognum') ?: 10; } $timezone = Option::get('timezone'); $start_limit = !empty($page) ? ($page - 1) * $perPageNum : 0; $limit = $perPageNum ? "LIMIT $start_limit, $perPageNum" : ''; $sql = "SELECT * FROM " . DB_PREFIX . "blog WHERE type='blog' and hide='n' and checked='y' $condition $limit"; $res = $this->db->query($sql); $logs = array(); while ($row = $this->db->fetch_array($res)) { $row['date'] += $timezone * 3600; $row['log_title'] = htmlspecialchars(trim($row['title'])); $row['log_url'] = Url::log($row['gid']); $row['logid'] = $row['gid']; $cookiePassword = isset($_COOKIE['em_logpwd_' . $row['gid']]) ? addslashes(trim($_COOKIE['em_logpwd_' . $row['gid']])) : ''; if (!empty($row['password']) && $cookiePassword != $row['password']) { $row['excerpt'] = '<p>[该文章已设置加密,请点击标题输入密码访问]</p>'; } else { if (!empty($row['excerpt'])) { $row['excerpt'] .= ''; } } $row['log_description'] = empty($row['excerpt']) ? breakLog($row['content'], $row['gid']) : $row['excerpt']; $row['attachment'] = ''; $row['tag'] = ''; $row['tbcount'] = 0; $logs[] = $row; } return $logs; } 反正没报错了。 然后发现发邮件的插件 kl_sendmail 报错了,而且发送不了邮件。 于是,更新 PHPMailer 的版本,直接更新到7.0.1,将src中所有的文件上传到该插件的目录的class文件中,修改 kl_sendmail.php 文件代码如下: <?php /* Plugin Name: Sendmail Version: 3.8 Plugin URL: (链接已失效) Description: 发送博客留言至E-mail。 Author: 作者:KLLER Author Email: kller@foxmail.com Author URL: (链接已失效) */ !defined('EMLOG_ROOT') && exit('access deined!'); // 使用命名空间引入PHPMailer类 use PHPMailerPHPMailerPHPMailer; use PHPMailerPHPMailerSMTP; use PHPMailerPHPMailerException; require_once(EMLOG_ROOT.'/content/plugins/kl_sendmail/class/PHPMailer.php'); require_once(EMLOG_ROOT.'/content/plugins/kl_sendmail/class/SMTP.php'); require_once(EMLOG_ROOT.'/content/plugins/kl_sendmail/class/Exception.php'); function UBB($content){ $content=preg_replace('!【链接:(.*)】!uU',"<br /><a href="/jump.php?url=1" target="_blank" rel="nofollow">1</a><br />",$content); $content=preg_replace('!【图片链接:(.*)】!uU',"",$content); $content=preg_replace('!【图片地址:(.*)】!uU',"",$content); $content=preg_replace('!【图片:(.*)】!uU',"",$content); $content=preg_replace("!【隐藏评论】([sS]*?)【/隐藏评论】!uU","(小声说)<br /><i>1</i>",$content); $content=preg_replace("!【隐藏内容】([sS]*?)【/隐藏内容】!uU","(小声说)<br /><i>1</i>",$content); $content=preg_replace("!【隐藏内容:([sS]*?)】!uU","(小声说)<br /><i>1</i>",$content); $content=preg_replace("!【隐藏信息:([sS]*?)】!uU","(小声说)<br /><i>1</i>",$content); $content=preg_replace("!【隐藏:([sS]*?)】!uU","(小声说)<br /><i>1</i>",$content); return $content; } function kl_sendmail_do($mailserver, $port, $mailuser, $mailpass, $mailto, $subject, $content, $fromname) { try { $mail = new PHPMailer(true); // 字符设置 $mail->CharSet = "UTF-8"; $mail->Encoding = "base64"; // 服务器设置 if(KL_MAIL_SENDTYPE == 1) { $mail->isSMTP(); } else { $mail->isMail(); } $mail->Host = $mailserver; $mail->Port = $port; $mail->SMTPAuth = true; $mail->Username = $mailuser; $mail->Password = $mailpass; // SSL/TLS设置 if($mailserver == 'smtp.gmail.com' || $mailserver == KL_MAIL_SMTP) { if($port == 465) { $mail->SMTPSecure = PHPMailer::ENCRYPTION_SMTPS; // SSL } else { $mail->SMTPSecure = PHPMailer::ENCRYPTION_STARTTLS; // TLS } } // 发件人 $mail->setFrom($mailuser, $fromname); // 收件人 $mail->addAddress($mailto); // 内容 $mail->isHTML(true); $mail->Subject = $subject; $mail->Body = $content; $mail->AltBody = strip_tags($content); // 纯文本版本 // 发送邮件 $mail->send(); return true; } catch (Exception $e) { error_log("邮件发送失败: " . $mail->ErrorInfo); return false; } } function kl_sendmail_get_comment_mail() { include(EMLOG_ROOT.'/content/plugins/kl_sendmail/kl_sendmail_config.php'); if(KL_IS_SEND_MAIL == 'Y' || KL_IS_REPLY_MAIL == 'Y') { $comname = isset($_POST['comname']) ? addslashes(trim($_POST['comname'])) : ''; $comment = isset($_POST['comment']) ? addslashes(trim($_POST['comment'])) : ''; $commail = isset($_POST['commail']) ? addslashes(trim($_POST['commail'])) : ''; $comurl = isset($_POST['comurl']) ? addslashes(trim($_POST['comurl'])) : ''; $gid = isset($_POST['gid']) ? intval($_POST['gid']) : (isset($_GET['gid']) ? intval($_GET['gid']) : -1); $pid = isset($_POST['pid']) ? intval($_POST['pid']) : 0; $http_referer = empty($_SERVER['HTTP_REFERER']) ? BLOG_URL : $_SERVER['HTTP_REFERER']; $blogname = Option::get('blogname'); $Log_Model = new Log_Model(); $logData = $Log_Model->getOneLogForHome($gid); $log_title = $logData['log_title']; $subject = "文章《{$log_title}》收到了新的评论"; if(!empty($commail)){$commail = $commail;}else{$commail = '未填写';}; if(!empty($comurl)){$comurl = $comurl;}else{$comurl = '未填写';}; if(strpos(KL_MAIL_TOEMAIL, '@139.com') === false) { $content = '这里懒得弄了~'; }else{ $content = $comment; } if(KL_IS_SEND_MAIL == 'Y') { if(ROLE == 'visitor') kl_sendmail_do(KL_MAIL_SMTP, KL_MAIL_PORT, KL_MAIL_SENDEMAIL, KL_MAIL_PASSWORD, KL_MAIL_TOEMAIL, $subject, $content, $blogname); } if(KL_IS_REPLY_MAIL == 'Y') { if($pid > 0) { $DB = Option::EMLOG_VERSION >= '5.3.0' ? Database::getInstance() : MySql::getInstance(); $Comment_Model = new Comment_Model(); $pinfo = $Comment_Model->getOneComment($pid); if(!empty($pinfo['mail'])) { $subject = '您在梦幻辰风发表的评论收到了Ta的回复'; $content = '<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"><div style="width:99%;"> <div style="padding:0 15px;color:#111;background-color:#F5FFFA;border:1px solid #d8e3e8;border-radius:0 0 6px 6px;-moz-border-radius:0 0 6px 6px;-webkit-border-radius:0 0 6px 6px;-khtml-border-radius:0 0 6px 6px;"> <p><strong>'.$pinfo['poster'].'</strong>阁下,您之前在《'.$log_title.'》发表的评论:</p> <p style="padding:10px;background-color:#F0FFF0;">'.$pinfo['comment'].'</p> <p><strong>'.$comname.'</strong> 给您的回复:</p> <p style="padding:10px;background-color:#F0F8FF;">'.$comment.'</p> <p>您可以直接<a href="'.Url::log($gid).'#'.$pid.'" target="_blank">点击这里查看原文</a>,与'.$comname.'继续交流。</p> <p>感谢您关注 梦幻辰风,本通知由自动信箱发出,请勿直接回复本邮件。</p> </div> </div> <!-- 邮件版面格式来自挨踢路 -->'; $content=UBB($content); kl_sendmail_do(KL_MAIL_SMTP, KL_MAIL_PORT, KL_MAIL_SENDEMAIL, KL_MAIL_PASSWORD, $pinfo['mail'], $subject, $content, $blogname); } } } }else{ return; } } addAction('comment_saved', 'kl_sendmail_get_comment_mail'); /* * 微语模块 * */ function kl_sendmail_get_twitter_mail($r, $name, $date, $tid) { include(EMLOG_ROOT.'/content/plugins/kl_sendmail/kl_sendmail_config.php'); if(KL_IS_TWITTER_MAIL == 'Y') { $DB = Option::EMLOG_VERSION >= '5.3.0' ? Database::getInstance() : MySql::getInstance(); $blogname = Option::get('blogname'); $sql = "select a.content, b.username from ".DB_PREFIX."twitter a left join ".DB_PREFIX."user b on b.uid=a.author where a.id={$tid}"; $res = $DB->query($sql); $row = $DB->fetch_array($res); $author = $row['username']; $twitter = $row['content']; $subject = "{$author}发布的碎语收到了新的回复"; if(strpos(KL_MAIL_TOEMAIL, '@139.com') === false) { $content = "{$author}发布的碎语:{$twitter}<br /><br />{$name}对碎语的回复:{$r}<br /><br /><strong>=> 现在就前往<a href="{$_SERVER['HTTP_REFERER']}" target="_blank">碎语页面</a>进行查看</strong><br />"; $content=UBB($content); }else{ $content = $r; } if(ROLE == 'visitor') kl_sendmail_do(KL_MAIL_SMTP, KL_MAIL_PORT, KL_MAIL_SENDEMAIL, KL_MAIL_PASSWORD, KL_MAIL_TOEMAIL, $subject, $content, $blogname); } } addAction('reply_twitter', 'kl_sendmail_get_twitter_mail'); /* * 回复评论 * */ function kl_sendmail_put_reply_mail($commentId, $reply) { global $userData; include(EMLOG_ROOT.'/content/plugins/kl_sendmail/kl_sendmail_config.php'); if(KL_IS_REPLY_MAIL == 'Y') { $DB = Option::EMLOG_VERSION >= '5.3.0' ? Database::getInstance() : MySql::getInstance(); $blogname = Option::get('blogname'); $Comment_Model = new Comment_Model(); $commentArray = $Comment_Model->getOneComment($commentId); extract($commentArray); $subject='您在梦幻辰风发表的评论收到了Ta的回复'; if(strpos($mail, '@139.com') === false) { $emBlog = new Log_Model(); $logData = $emBlog->getOneLogForHome($gid); $log_title = $logData['log_title']; $content = '<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"><div style="width:99%;"> <div style="padding:0 15px;color:#111;background-color:#F5FFFA;border:1px solid #d8e3e8;border-radius:0 0 6px 6px;-moz-border-radius:0 0 6px 6px;-webkit-border-radius:0 0 6px 6px;-khtml-border-radius:0 0 6px 6px;"> <p><strong>'.$poster.'</strong>阁下,您之前在《'.$log_title.'》发表的评论:</p> <p style="padding:10px;background-color:#F0FFF0;">'.$comment.'</p> <p><strong>'.$userData['username'].'</strong> 给您的回复:</p> <p style="padding:10px;background-color:#F0F8FF;">'.$reply.'</p> <p>您可以直接<a href="'.Url::log($gid).'#'.$pid.'" target="_blank">点击这里查看原文</a>,与'.$userData['username'].'继续交流。</p> <p>感谢您关注 梦幻辰风,本通知由自动信箱发出,请勿直接回复本邮件。</p> </div> </div> <!-- 邮件版面格式来自挨踢路 -->'; $content=UBB($content); }else{ $content = $reply; $content=UBB($content); } if($mail != '') kl_sendmail_do(KL_MAIL_SMTP, KL_MAIL_PORT, KL_MAIL_SENDEMAIL, KL_MAIL_PASSWORD, $mail, $subject, $content, $blogname); }else{ return; } } addAction('comment_reply', 'kl_sendmail_put_reply_mail'); function kl_sendmail_menu() { echo '<div class="sidebarsubmenu" id="kl_sendmail"><a href="./plugin.php?plugin=kl_sendmail">邮件设置</a></div>'; } addAction('adm_sidebar_ext', 'kl_sendmail_menu'); ?> 完事。 但是梦幻辰风邮件回复一直失败。 结果发现是今年3月腾讯企业邮箱看我长时间没使用smtp功能,自己给我关了! …… 搞定。 2025年12月7日注: 可能是服务器性能原因,跑PHP8有点卡不愣登的。最后降成PHP7.4,结果反而更快乐了。 欢迎您来到梦幻辰风(www.mhcf.net)来体验更好的阅读!