解决宝塔面板主主复制主键冲突的问题(苹果CMS)
直接操作:
我是四个服务器做的主主复制,
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();
}
}