Percona pt-archiver 出现数据不对等

python 复制代码
#!/bin/env python

import datetime
import subprocess
import time

def __do_archiver(file_name, where_condition):
    file = f'/tmp/{file_name}.txt'
    subprocess.call(["/usr/bin/touch", file])

    command = [
        "/root/percona-toolkit-3.7.0/bin/pt-archiver",
        "--source",
        "t=api_third_receive_record,i=idx_create_time",
        "--database=db0",
        "--password=9527",
        "--user=root",
        "--port=3306",
        "--host=192.168.10.11",
        f'--file={file}',
        f"--where={where_condition}",
        "--charset=utf8",
        "--no-version-check",
        "--no-check-charset",
        "--progress=500",
        "--limit=500",
        "--txn-size=500",
        "--statistics",
        "--no-safe-auto-increment",
        "--bulk-delete",
        "--purge"
    ]
    print('Command', command)
    subprocess.call(command)

def do_archiver(start_date, end_date):
    date_format = '%Y-%m-%d'
    file_format = '%Y_%m_%d'
    start_date_time = datetime.datetime.strptime(start_date, date_format)
    end_date_time = datetime.datetime.strptime(end_date, date_format)
    while start_date_time <= end_date_time:
        file_name = start_date_time.strftime(file_format)

        prev_date_time = start_date_time + datetime.timedelta(days=-1)
        next_date_time = start_date_time + datetime.timedelta(days=1)

        print('Executing', start_date_time.strftime(date_format))
        where_condition = f'create_time>="{start_date_time.strftime(date_format)} 00:00:00" AND create_time<="{start_date_time.strftime(date_format)} 23:59:59"'
        __do_archiver(file_name, where_condition)
        time.sleep(5)
        start_date_time = next_date_time


if __name__ == '__main__':
    do_archiver("2022-11-21", "2022-11-21")

使用 pt-archiver 命令, 按天将 api_third_receive_record 表数据归档到文件里, 并删除表数据. WHERE 条件是按照创建时间筛选, 并使用 idx_create_time 索引.

版本 percona-toolkit-3.7.0

在实际执行时, 会出现表数据已经归档到文件里了, 但是表里还有少许数据没有被删除掉.

通过查询通用日志, 查看到执行的部分SQL语句

sql 复制代码
SELECT * FROM `db0`.`api_third_receive_record` FORCE INDEX(`idx_create_time`) WHERE (create_time>="2022-11-21 00:00:00" AND create_time<="2022-11-21 23:59:59") ORDER BY `create_time` LIMIT 500

DELETE FROM `db0`.`api_third_receive_record` WHERE (((`create_time` >= '2022-11-21 00:03:19.000'))) AND (((`create_time` <= '2022-11-21 15:27:53.000'))) AND (create_time>="2022-11-21 00:00:00" AND create_time<="2022-11-21 23:59:59") LIMIT 500

SELECT * FROM `db0`.`api_third_receive_record` FORCE INDEX(`idx_create_time`) WHERE (create_time>="2022-11-21 00:00:00" AND create_time<="2022-11-21 23:59:59") AND ((`create_time` >= '2022-11-21 15:27:53.000')) ORDER BY `create_time` LIMIT 500

DELETE FROM `db0`.`api_third_receive_record` WHERE (((`create_time` >= '2022-11-21 15:28:20.000'))) AND (((`create_time` <= '2022-11-21 23:56:28.000'))) AND (create_time>="2022-11-21 00:00:00" AND create_time<="2022-11-21 23:59:59") LIMIT 500

SELECT 查询时是使用索引并排序, 但DELETE 时没有使用索引

修改 pt-archiver 脚本文件, 在删除的逻辑里新增 ORDER BY $index_cols 内容

再次执行归档动作, DELETE 语句使用ORDER BY 排序, 数据也和期望一样.

sql 复制代码
DELETE FROM `db0`.`api_third_receive_record` WHERE (((`create_time` >= '2022-11-21 00:03:19.000'))) AND (((`create_time` <= '2022-11-21 15:27:53.000'))) AND (create_time>="2022-11-21 00:00:00" AND create_time<="2022-11-21 23:59:59") ORDER BY `create_time` LIMIT 500

执行计划, 也使用了 idx_create_time 索引

相关推荐
rexling11 分钟前
【Spring Boot】Spring Boot解决循环依赖
java·前端·spring boot
2501_938790077 分钟前
从 0 到 1:解决 VsCode 远程连服务器后 Github Copilot 无法使用问题
服务器·vscode·github
江輕木13 分钟前
如何使用宿主机软件共享网络给CentOS 7
linux·运维·服务器
武子康14 分钟前
Java-163 MongoDB 生产安全加固实战:10 分钟完成认证、最小权限、角色详解
java·数据库·分布式·mongodb·性能优化·系统架构·nosql
zhangyifang_00916 分钟前
PostgreSQL 的表继承与分区
数据库·postgresql
IT曙光16 分钟前
在华为TaiShan 200系列服务器基于CentOS 7.6/7.7创建虚拟机
运维·服务器·centos
ZIM学编程25 分钟前
「学长有话说」作为一个大三学长,我想对大一计算机专业学生说这些!
java·c语言·数据结构·c++·python·学习·php
Dolphin_Home31 分钟前
轻量实用的 XML 与 JSON / 对象互转工具类(Jackson 实现)
xml·java·json
金仓拾光集1 小时前
国产化转型实战:制造业供应链物流系统从MongoDB至金仓数据库迁移全指南
数据库·mongodb·数据库平替用金仓·金仓数据库
Yeniden1 小时前
【设计模式】# 外观模式(Facade)大白话讲解!
java·设计模式·外观模式