1247天 咸鱼也有梦想

重要的人越来越少,剩下的人也越来越重要 ​​

解决宝塔面板主主复制主键冲突的问题(苹果CMS)

发布于 11天前 / 19 次围观 / 0 条评论 / 教程 / 咸鱼

直接操作:

我是四个服务器做的主主复制

A->B    B->A

A->C    C->A

A->D    D->A

操作之前首先配置好四个服务器的主主复制

 

配置完之后

 1.分别在每个服务器的MySQL配置页面配置以下参数(宝塔不知道你是有几个服务器做主主复制,所以以下参数需要配置完手动修改)

[mysqld]
# 对于ServerA
auto_increment_increment = 4
auto_increment_offset = 1

# 对于ServerB
auto_increment_increment = 4
auto_increment_offset = 2

# 对于ServerC
auto_increment_increment = 4
auto_increment_offset = 3

# 对于ServerD
auto_increment_increment = 4
auto_increment_offset = 4

2.配置完以上之后发现,还是会报错主键冲突,原来是搜索缓存数据表mac_vod_search的锅

苹果cms默认的搜索插入逻辑不适合主主复制环境,需要将insert() 方法,改成使用原生 SQL 的 INSERT INTO 语句

复制以下完整代码,上传到 网站目录/application/common/model/VodSearch.php 就可以解决

<?php
namespace app\common\model;

use think\Db;
use think\Cache;

class VodSearch extends Base {
    // 设置数据表(不含前缀)
    protected $name = 'vod_search';
    // 最大Id数量,使用IN查询时,超过一定数量,查询不使用索引了
    public $maxIdCount = 1000;
    private $updateTopCount = 50000;

    /**
     * 获取结果Id列表
     */
    public function getResultIdList($search_word, $search_field, $word_multiple = false)
    {
        $search_word = trim($search_word);
        $search_word = str_replace(',,', '', $search_word);
        if (strlen($search_word) == 0 || strlen($search_field) == 0) {
            return [];
        }
        // 如果包含多个关键词,使用递归处理
        if ($word_multiple === true) {
            $id_list = [];
            $search_word_exploded = explode(',', $search_word);
            foreach ($search_word_exploded as $search_word) {
                $id_list += $this->getResultIdList($search_word, $search_field);
            }
            $id_list = array_unique($id_list);
            return $id_list;
        }
        $search_key = md5($search_word . '@' . $search_field);
        $where = ['search_key' => $search_key];
        $search_row = $this->where($where)->field("search_result_ids, search_hit_count")->find();

        if (empty($search_row)) {
            // 查询相关视频ID
            $where_vod = [];
            $where_vod[$search_field] = ['LIKE', '%' . $search_word . '%'];
            $id_list = Db::name('Vod')->where($where_vod)->order("vod_id ASC")->column("vod_id");
            $id_list = is_array($id_list) ? $id_list : [];

            // 插入或更新记录
            $data = [
                'search_key'           => $search_key,
                'search_word'          => mb_substr($search_word, 0, 128),
                'search_field'         => mb_substr($search_field, 0, 64),
                'search_hit_count'     => 1,
                'search_last_hit_time' => time(),
                'search_update_time'   => time(),
                'search_result_count'  => count($id_list),
                'search_result_ids'    => join(',', $id_list),
            ];

            // 使用 ON DUPLICATE KEY UPDATE 避免主键冲突
            $sql = "
                INSERT INTO " . config('database.prefix') . $this->name . " (
                    search_key, search_word, search_field, search_hit_count, search_last_hit_time, 
                    search_update_time, search_result_count, search_result_ids
                ) VALUES (
                    '{$data['search_key']}', '{$data['search_word']}', '{$data['search_field']}', 
                    {$data['search_hit_count']}, {$data['search_last_hit_time']}, 
                    {$data['search_update_time']}, {$data['search_result_count']}, '{$data['search_result_ids']}'
                )
                ON DUPLICATE KEY UPDATE
                    search_word = VALUES(search_word),
                    search_field = VALUES(search_field),
                    search_hit_count = VALUES(search_hit_count),
                    search_last_hit_time = VALUES(search_last_hit_time),
                    search_update_time = VALUES(search_update_time),
                    search_result_count = VALUES(search_result_count),
                    search_result_ids = VALUES(search_result_ids)
            ";
            Db::execute($sql);
        } else {
            // 更新命中次数
            $id_list = explode(',', (string)$search_row['search_result_ids']);
            $id_list = array_filter($id_list);
            $this->where($where)->update([
                'search_hit_count'     => $search_row['search_hit_count'] + 1,
                'search_last_hit_time' => time(),
            ]);
        }

        // 格式化返回的ID列表
        $id_list = array_map('intval', $id_list);
        $id_list = empty($id_list) ? [0] : $id_list;
        return $id_list;
    }

    /**
     * 前端是否开启
     */
    public function isFrontendEnabled()
    {
        $config = config('maccms');
        // 未设置时,默认关闭
        if (!isset($config['app']['vod_search_optimise'])) {
            return false;
        }
        $list = explode('|', $config['app']['vod_search_optimise']);
        return in_array('frontend', $list);
    }

    /**
     * 采集是否开启
     */
    public function isCollectEnabled()
    {
        $config = config('maccms');
        // 未设置时,默认关闭
        if (!isset($config['app']['vod_search_optimise'])) {
            return false;
        }
        $list = explode('|', $config['app']['vod_search_optimise']);
        return in_array('collect', $list);
    }

    /**
     * 检查更新搜索结果
     */
    public function checkAndUpdateTopResults($vod, $force = false)
    {
        static $list;
        if (empty($vod['vod_id'])) {
            return;
        }
        if (is_null($list)) {
            $cach_name = 'vod_search_top_result_v2_' . $this->updateTopCount;
            $list = $force ? [] : Cache::get($cach_name);
            if (empty($list)) {
                $list = $this->field("search_key, search_word, search_field")->order("search_hit_count DESC, search_last_hit_time DESC")->limit("0," . $this->updateTopCount)->select();
                $force === false && Cache::set($cach_name, $list, count($list) < ($this->updateTopCount / 10) ? 3600 : 86400);
                $this->clearOldResult();
            }
        }
        $time_now = time();
        foreach ($list as $row) {
            foreach (explode('|', $row['search_field']) as $field) {
                if (!isset($vod[$field]) || strlen($vod[$field]) == 0) {
                    continue;
                }
                if (stripos($vod[$field], $row['search_word']) === false) {
                    continue;
                }
                Db::execute("UPDATE `" . config('database.prefix') . $this->name . "` SET 
                    search_update_time='{$time_now}',
                    search_result_count=search_result_count+1,
                    search_result_ids=CONCAT(search_result_ids,',','{$vod['vod_id']}')
                WHERE search_key='{$row['search_key']}'");
            }
        }
    }

    /**
     * 获取结果缓存的分钟数,后台配置覆盖默认值
     */
    public function getResultCacheMinutes($config = []) {
        // 默认14天
        $minutes = 20160;
        $config = $config ?: config('maccms');
        if (isset($config['app']['vod_search_optimise_cache_minutes']) && (int)$config['app']['vod_search_optimise_cache_minutes'] > 0) {
            $minutes = (int)$config['app']['vod_search_optimise_cache_minutes'];
        }
        return $minutes;
    }

    /**
     * 清理老的数据
     */
    public function clearOldResult($force = false) 
    {
        // 清理多久前的
        $clear_seconds = $this->getResultCacheMinutes() * 60;
        // 设置间隔,每天最多清理1次
        $cach_name = 'interval_vs_clear_old_v1_' . $clear_seconds;
        $cache_data = Cache::get($cach_name);
        if ($force === false && !empty($cache_data)) {
            return;
        }
        Cache::set($cach_name, 1, min($clear_seconds, 86400));
        // vod_actor在采集的时候可提高效率,暂不清理
        $where = [
            'search_field'       => ['neq', 'vod_actor'],
            'search_update_time' => ['lt', time() - $clear_seconds],
        ];
        // 后台强制清理时,都清掉
        if ($force === true) {
            unset($where['search_field']);
        }
        $this->where($where)->delete();
    }
}