月度归档:2024年10月

PHP下载MP3文件文件

新建文件download.php,键入以下代码:

<?php

$fileName = $_GET['file'];

$file = 'music/' . $fileName;

if (!file_exists($file)) {
    http_response_code(404);
    die();
}

header("Cache-Control: private");
header("Content-type: audio/mpeg3");
header("Content-Transfer-Encoding: binary");
header("Content-Disposition: attachment; filename=".$fileName);
//So the browser can display the download progress
header("Content-Length: ".filesize($file));

readfile($file);

给 WordPress 添加文章浏览量统计功能

前几天给网站添加了文章的浏览量统计功能,但统计了几天后发现,统计了个寂寞,来访的除了蜘蛛就是自己,意义不大,索性删除了罢。想要统计,后面可以接入专门的网站统计系统,比如Google Analytics。下面把wordpress文章统计代码分享出来。

下面的代码我是加到functions.php里面的,当然,也可以做成插件。

/**
 * 获取文章阅读量
 *
 * @since 2024.10.25
 *
 */
function getPostViews($postID){
    $count_key = 'post_views_count';
    $count = get_post_meta($postID, $count_key, true);
    if($count==''){
        delete_post_meta($postID, $count_key);
        add_post_meta($postID, $count_key, '0');
        return "0";
    }
    return $count;
}

/**
 * 更新文章阅读量
 *
 * @since 2024.10.25
 *
 */
function setPostViews($postID) {
    // 检查用户是否已登录
    if (is_user_logged_in()) {
        return; // 已登录用户,不执行统计
    }
    $count_key = 'post_views_count';
    $count = get_post_meta($postID, $count_key, true);
    if($count==''){
        $count = 0;
        delete_post_meta($postID, $count_key);
        add_post_meta($postID, $count_key, '0');
    }else{
        $count++;
        update_post_meta($postID, $count_key, $count);
    }
}
// Remove issues with prefetching adding extra views
remove_action( 'wp_head', 'adjacent_posts_rel_link_wp_head', 10, 0);

使用方法:

setPostViews函数加到single.php里面,如果有访问就会调用该函数实现文章阅读统计。然后在适当的地方调用getPostViews函数用于获取文章的阅读量。

当然,也可以完善setPostViews函数,使之不统计蜘蛛的流量,要实现也不难,通过useragent来判断即可。但既然觉得这事没有意义,也就懒得去做了。

补充:

既然去掉了该功能,那么数据库里产生的统计数据就要删除掉:

DELETE FROM wp_postmeta WHERE meta_key = 'post_views_count';

PHP打开错误提示

有时我们希望在页面上能显示PHP的报错信息,有两种方法,一种是通过配置php.ini,但有时候我们只希望在某个页面上显示错误信息,或者由于一些原因修改不了php.ini,此时就需要使用PHP代码来打开错误提示,代码如下:

<?php
error_reporting(E_ALL);
ini_set('display_errors', 'On');

PHP生成WordPress的Nginx跳转规则

建站初期总有一些事情考虑不周,前几天调整了固定链接,从/category/id.html调整为/archive/id,然后总觉得哪里不踏实。趁着周末去考试的时间想了一下,如果固定链接依赖于文章ID或分类,那么一旦ID或分类发生变化,文章的固定链接就会失效。再者,从长远来说,将来如果要改版或重构网站,基于ID或分类的链接将变得不稳定。

由此,可以得出结论,固定链接最好不要依赖文章ID和分类,而是要依赖于文章的内容。这样的固定链接显得更稳定,而且具有更好的可读性,有利于SEO优化,分享链接时,链接本身也变得更有意义。

所以,决定把本站的固定链接改成基于postname的形式。修改固定链接后,要把之前的文章链接做一个跳转,但是由于文章有点多,手动修改很费时间,于是动手写了一段PHP代码,用于生成Nginx的跳转规则,代码如下:

<?php
// 数据库连接设置
$servername = "localhost"; // 数据库主机
$username = "your_username"; // 数据库用户名
$password = "your_password"; // 数据库密码
$dbname = "your_database"; // 数据库名

// 创建连接
$conn = new mysqli($servername, $username, $password, $dbname);

// 检查连接
if ($conn->connect_error) {
    die("连接失败: " . $conn->connect_error);
}

// 查询已发布的文章
$sql = "SELECT ID, post_title, post_name FROM wp_posts WHERE post_status='publish' AND post_type='post'";
$result = $conn->query($sql);

// 检查查询结果
if ($result->num_rows > 0) {
    while($row = $result->fetch_assoc()) {
        $id = $row['ID'];
        $post_name = $row['post_name'];
        // 生成 Nginx 重定向规则
        echo "rewrite ^/archives/$id$ /$post_name permanent;" . "<br>";
    }
} else {
    echo "没有找到已发布的文章。";
}

// 关闭连接
$conn->close();
?>

运行这段代码,就会在浏览器页面输出重定向规则,我们把它复制到nginx配置里就可以了。

location /archives/ {
    rewrite ^/archives/3681$ /letting-go-of-tech-obsession permanent;
    # 为了便于展示,这里只粘贴了一条规则...
}

后记:这种基于postname的permalink就是写文章的时候麻烦一点,写完后,还要再把标题翻译成英文,看个人取舍了。后面如果我忍受不了这多余的一步,也可能换成基于post_id的形式。使用postname还有一个好处理就是,如果要换成其它形式,WP可以自动实现跳转,而不用再另写跳转规则。

PHP获取当前用户的客户端IP

可以用以下函数来获取访客的客户端IP地址:

<?php
function get_client_ip() {
    $ip = $_SERVER['REMOTE_ADDR'];
    if (isset($_SERVER['HTTP_CLIENT_IP']) && preg_match('/^([0-9]{1,3}\.){3}[0-9]{1,3}$/', $_SERVER['HTTP_CLIENT_IP'])) {
        $ip = $_SERVER['HTTP_CLIENT_IP'];
    } elseif (isset($_SERVER['HTTP_X_FORWARDED_FOR']) and preg_match_all('#\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}#s', $_SERVER['HTTP_X_FORWARDED_FOR'], $matches)) {
        foreach ($matches[0] as $xip) {
            if (!preg_match('#^(10|172\.16|192\.168)\.#', $xip)) {
                $ip = $xip;
                break;
            }
        }
    }
    return $ip;
}

再来一个版本的代码,这个版本增加了对使用代理情况下的判断,根据需要选一个用就可以了:

<?php
/**
 * 获取客户端IP
 */
function getClientIp() {
    $ip = 'unknown';
    $unknown = 'unknown';

    if (isset($_SERVER['HTTP_X_FORWARDED_FOR']) && $_SERVER['HTTP_X_FORWARDED_FOR'] && strcasecmp($_SERVER['HTTP_X_FORWARDED_FOR'], $unknown)) {
        // 使用透明代理、欺骗性代理的情况
        $ip = $_SERVER['HTTP_X_FORWARDED_FOR'];

    } elseif (isset($_SERVER['REMOTE_ADDR']) && $_SERVER['REMOTE_ADDR'] && strcasecmp($_SERVER['REMOTE_ADDR'], $unknown)) {
        // 没有代理、使用普通匿名代理和高匿代理的情况
        $ip = $_SERVER['REMOTE_ADDR'];
    }

    // 处理多层代理的情况
    if (strpos($ip, ',') !== false) {
        // 输出第一个IP
        $ip = reset(explode(',', $ip));
    }

    return str_replace('.', '_', $ip);
}  

PHP处理用户名,中间用星号表示

有时需要将用户名进行隐私处理,可以用以下代码将用户名中间字母用星号遮盖:

<?php
//将用户名进行处理,中间用星号表示
function substr_cut($user_name){
    //获取字符串长度
    $strlen = mb_strlen($user_name, 'utf-8');
    //如果字符创长度小于2,不做任何处理
    if($strlen<2){
        return $user_name;
    }else{
        //mb_substr — 获取字符串的部分
        $firstStr = mb_substr($user_name, 0, 1, 'utf-8');
        $lastStr = mb_substr($user_name, -1, 1, 'utf-8')
        //str_repeat — 重复一个字符串
        return $strlen == 2 ? $firstStr . str_repeat('*', mb_strlen($user_name, 'utf-8') - 1) : $firstStr . str_repeat("*", $strlen - 2) . $lastStr;
    }

}

PHP利用CURL实现 http post 和 get

PHP利用CURL实现HTTP GET和HTTP POST,用于HTTP请求,具体代码如下:

<?php
//初始化
$curl = curl_init();
//设置抓取的url
curl_setopt($curl, CURLOPT_URL, 'http://www.baidu.com');
//设置头文件的信息作为数据流输出,如果不想打印出http头,可以把这行去掉
curl_setopt($curl, CURLOPT_HEADER, 1);
//设置获取的信息以文件流的形式返回,而不是直接输出。
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
//执行命令
$data = curl_exec($curl);
//关闭URL请求
curl_close($curl);
//显示获得的数据
print_r($data);

HTTP POST:

<?php

$ch = curl_init();

curl_setopt($ch, CURLOPT_URL,$notify_url);
curl_setopt($ch, CURLOPT_POST, 1);


//curl_setopt($ch, CURLOPT_POSTFIELDS, "postvar1=value1&postvar2=value2&postvar3=value3");

$notifyData = [];

// In real life you should use something like:
curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($notifyData));

// Receive server response ...
// curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);

curl_exec($ch);

curl_close($ch);

// echo $server_output;

PHP日志类

新建Log.class.php,复制以下代码放入其中:

<?php
interface ILogHandler {
    public function write($msg);
}

// PHP日志类
class CLogFileHandler implements ILogHandler {
    private $handle = null;

    public function __construct($file = '') {
        $this->handle = fopen($file, 'a');
    }

    public function write($msg) {
        fwrite($this->handle, $msg, 4096);
    }

    public function __destruct() {
        fclose($this->handle);
    }
}
class Log {
    private $handler = null;
    private $level = 15;

    private static $instance = null;

    private function __construct() {
    }
    private function __clone() {
    }

    public static function Init($handler = null, $level = 15) {
        if (!self::$instance instanceof self) {
            self::$instance = new self();
            self::$instance->__setHandle($handler);
            self::$instance->__setLevel($level);
        }
        return self::$instance;
    }


    private function __setHandle($handler) {
        $this->handler = $handler;
    }

    private function __setLevel($level) {
        $this->level = $level;
    }

    public static function DEBUG($msg) {
        self::$instance->write(1, $msg);
    }

    public static function WARN($msg) {
        self::$instance->write(4, $msg);
    }

    public static function ERROR($msg) {
        $debugInfo = debug_backtrace();
        $stack = "[";
        foreach ($debugInfo as $key => $val) {
            if (array_key_exists("file", $val)) {
                $stack .= ",file:" . $val["file"];
            }
            if (array_key_exists("line", $val)) {
                $stack .= ",line:" . $val["line"];
            }
            if (array_key_exists("function", $val)) {
                $stack .= ",function:" . $val["function"];
            }
        }
        $stack .= "]";
        self::$instance->write(8, $stack . $msg);
    }

    public static function INFO($msg) {
        self::$instance->write(2, $msg);
    }

    private function getLevelStr($level) {
        switch ($level) {
            case 1:
                return 'debug';
                break;
            case 2:
                return 'info';
                break;
            case 4:
                return 'warn';
                break;
            case 8:
                return 'error';
                break;
            default:
        }
    }

    protected function write($level, $msg) {
        if (($level & $this->level) == $level) {
            $msg = '[' . date('Y-m-d H:i:s') . '][' . $this->getLevelStr($level) . '] ' . $msg . "\n";
            $this->handler->write($msg);
        }
    }
}

调用方式:

<?php
require 'Log.class.php'; // 导入日志类文件
define('DS', DIRECTORY_SEPARATOR); // 设置目录分隔符
define('LOG_PATH', dirname(__FILE__) . DS . 'log' . DS); // 日志文件目录

$logHandler = new CLogFileHandler(LOG_PATH . date('Y-m-d') . '.log');
$log = Log::Init($logHandler, 15);

Log::DEBUG('111111');

天翼云和华为云初体验

今天博客园送的华为云服务器开通了,晚上花了一个小时的时间简单体验了一下,感觉还不错。

我安装的是 Debian 系统,华为云直接有最新的12版本,只要apt update && apt upgrade一下就可以一键升级到的最新的12.7版本了,这点比天翼云方便,天翼云只有 Debian11。

华为云服务器 Debian 镜像已经设置好了华为云的源,使用起来非常方便,不用自己再进行调整。而天翼云就没有,要自己动手修改。

另外,天翼云的 Debian 镜像配置有点小问题,/etc/hosts 里面没有和 hostname 关联的记录,导致使用起来有点小问题,具体什么问题当时忘记记录了,反正是有。还有就是系统的区域配置有问题,只有 en_US,没有 en_DK,导致对日期格式的支持有些问题,apt 安装包时有警告。还有一些小问题记不太清了。反观华为云就用的很顺畅,命令敲下去一气呵成、酣畅淋漓。

还有,天翼云比较严格,常用的端口都是不能用的,直接用 IP 都不行。要用的话只能迁移备案,但众所周知,这是个麻烦事,这台37块钱的服务器大概率要吃灰。

总体来说,华为云用的比较安逸,体验比天翼云要好。

注:本文始发于我的博客园 https://www.cnblogs.com/art ,之后重新发于本站。

phpMyAdmin 初次安装配置

将 config.sample.inc.php 复制一份,把复制后的文件改名为 config.inc.php,把$cfg['blowfish_secret']修改为

$cfg['blowfish_secret'] = '419f9149e486203888b17fee459c2912'; /* YOU MUST FILL IN THIS FOR COOKIE AUTH! */