0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

MySQL 8.0/8.4执行DDL丢数据有什么影响

OSC开源社区 来源:OSC开源社区 2024-12-24 09:27 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

问题是有,但好在规避办法也比较简单,影响也有限。

先说解决办法,从简单到麻烦:

执行 ALTER TABLE 时,显式指定ALGORITHM=INSTANT/COPY,反正不要使用 INPLACE。

适当调大 innodb_ddl_buffer_size 参数值,其默认值1MB,例如调大到100MB就可以应对大部分业务表的DDL操作场景。

利用 pt-osc 或 gh-ost 等工具进行 Online DDL 操作。

在业务低谷时段执行DDL操作,有条件的话甚至可以在业务维护期间再执行DDL操作。

升级版本到已修复的 Percona 分支版本(下文会提到)。

问题来源

在 MySQL 8.0.27 版本中新增并行DDL功能后才“引入”了这个问题。目前在最新的 8.1.x/8.3.x/8.3.x/8.4.x/9.0.x/9.1.x 等版本中依然存在,预计到 MySQL 8.0.41 新版本会修复。

For online DDL operations, storage is usually the bottleneck. To address this issue, CPU utilization and index building has been improved. Indexes can now be built simultaneously instead of serially. Memory management has also been tightened to respect memory configuration limits set by the user.

详见:https://dev.mysql.com/doc/relnotes/mysql/8.0/en/news-8-0-27.html

触发原因:在INPLACE模式的DDL操作中重建主键索引时,因错误处理会略过部分记录,导致数据丢失。

触发条件:只影响INPLACE模式的DDL操作,不影响COPY和INSTANT模式的DDL操作。以下是几种常见的可能触发问题的DDL操作场景:

场景1:ALTER TABLE ENGINE=INNODB 重整表空间操作,需要重建主键索引。

场景2:ALTER TABLE ADD NEW-COL ...,ALGORITHM=INPLACE,新增列操作,因指定了INPLACE模式,需要重建主键索引。

其他例如INSTANT模式加新字段,增删索引则不会触发该问题。

关于该问题的详细解读详见几篇文章:

八怪老师推文 https://www.jianshu.com/p/c66fe0349345?v=1734349439280 。

Rex老师推文 MySQL 8.4-LTS DDL会导致数据丢失。

丁奇老师推文 丢数据风险 @ MySQL官方最新版。

Percona 推文 Who Ate My MySQL Table Rows?。

涉及到2个MySQL bug:

DDL 丢数风险:https://bugs.mysql.com/bug.php?id=115608

DDL 重复行报错:https://bugs.mysql.com/bug.php?id=115511

该问题核心就存在于如果涉及到需要用INPLACE算法重建主键索引的DDL操作,就需要在 innodb_ddl_buffer_size 用满后直接插入到 #sql-ibXXX 数据文件中,这个时候可能正在page的中间的某个位置,插入的时候会暂时放弃page上的mutex,并且保存游标到持久游标,然后插入数据,插入完成后再从持久游标恢复游标。这样做的目的可能是为了提高page修改的并发,但是这里保存和恢复持久游标却出了问题,主要是page中的数据可能出现修改,这种修改对应了前面的2个BUG:

Purge线程,清理del flag。

其他线程INSERT了数据。

具体游标的保存和恢复出现的问题,可以参考Rex老师的文章 MySQL 8.4-LTS DDL会导致数据丢失。

问题影响

目前该问题已知影响的版本列表如下:

MySQL 8.0.x 系列版本中,所有 >= 8.0.27 的 MySQL 8.0.x 版本;

所有 8.4.x 系列 LTS 版本;

Percona Server for MySQL 中从 8.0.27-18 至 8.0.37-29,以及 8.4.0-1 版本。

Percona XtraDB Cluster 中从 8.0.27-18.1 至 8.0.37-29,以及 8.4.0-1 版本。

未受影响或已修复的版本列表如下:

所有早于 MySQL 8.0 的版本,及 MySQL 5.6、5.7 等版本,以及 Percona 5.6、5.7 版本;

Percona 8.0 系列中 8.0.39-30 及更高版本;

Percona 8.4 系列中 8.4.2-2 及更高版本;

Percona XtraDB Cluster 8.0 系列中 8.0.39-30 及更高版本。

目前所有活跃的 MySQL 版本均未修复,已安排在MySQL 8.0.41版本修复该问题。GreatSQL也会在下一个新版本中修复该问题。

问题复现/模拟

模拟测例1

经过测试,该问题触发概率和 update/delete 并发负载有关,结合 MySQL bug #113812 提供的案例,我进行了简化和改造,测试用例如下:

#/bin/sh
#bugtest.sh,测例1
#需要先安装mysql_random_data_load测试工具
#通过socket方式连接MySQL时用root密码并且是空密码
MYSQL="mysql-N-s-uroot-S/data/MySQL/mysql.sock"
HOST=127.0.0.1
PORT=3306
USER="yejr"
PWD="yejr"

echo"1.Preparework"

read-r-d''bugSQL<<-EOSQL || true
CREATE DATABASE IF NOT EXISTS test;
USE test;
DROP TABLE IF EXISTS t1;
CREATE TABLE IF NOT EXISTS t1(
 id int not null,
 c1 varchar(20) not null,
 c2 varchar(30) not null,
 c3 datetime not null,
 c4 varchar(30) not null,
 PRIMARY KEY (id),
 KEY idx_c3 (c3)
) ENGINE=InnoDB;

CREATE USER IF NOT EXISTS '${USER}'@'%';
ALTER USER '${USER}'@'%' IDENTIFIED BY '${PWD}';
GRANT ALL PRIVILEGES ON test.t1 TO '${USER}'@'%';
EOSQL

${MYSQL} -f -e "${bugSQL}"

echo "2. Starting run test"

${MYSQL} -e "truncate table test.t1;"

for i in {1..1000}
do
 mysql_random_data_load -u${USER} -p${PWD} -h${HOST} -P${PORT} --max-threads=2 test t1 1000 >/dev/null2>&1
c_before_del=`${MYSQL}-e"selectcount(*)fromtest.t1;"`
c_delete=`${MYSQL}-e"selectcount(*)fromtest.t1wherec3< curdate() - interval 7 day;"`
 ${MYSQL} -e "delete from test.t1 where c3 < curdate() - interval 7 day;"
 c_before_alter=`${MYSQL} -e "select count(*) from test.t1;"`
 ${MYSQL} -e "alter table test.t1 engine=innodb;"
 c_after_alter=`${MYSQL} -e "select count(*) from test.t1;"`
 if [ ${c_before_alter} -ne ${c_after_alter} ] ; then
  echo "run ${i} times, delete: ${c_delete}, before alter: ${c_before_alter}, after alter: ${c_after_alter}"
  exit
 fi
 if [ `expr ${i} % 10` -eq 0 ] ; then
  echo "run ${i} times"
 fi
done

执行该测试用例脚本,当发现有问题时,结果显式如下:

$sh./bugtest.sh
1.Preparework
2.Startingruntest
run10times
run20times
run30times
...
run175times,delete:979,beforealter:3436,afteralter:3435

这就表示执行到第175次后触发问题,发现丢了一条记录。在这个测例中,如果加大 innodb_ddl_buffer_size 参数值到10MB,则不再触发问题。

模拟测例2

对上面的测试用例再进行调整后,改成下面这个测例,在执行完1000次后仍未触发问题(可见并不总是会触发问题,只有个别情况下会踩雷):

#!/bin/sh
#bugtest.sh,测例2
#需要先安装mysql_random_data_load测试工具
#通过socket方式连接MySQL时用root密码并且是空密码
MYSQL="mysql-N-s-uroot-S/nvme/GreatSQL/mysql.sock"
HOST=127.0.0.1
PORT=3306
USER="yejr"
PWD="yejr"

echo"1.Preparework"

read-r-d''bugSQL<<-EOSQL || true
CREATE DATABASE IF NOT EXISTS test;
USE test;
DROP TABLE IF EXISTS t1;
CREATE TABLE IF NOT EXISTS t1(
 id int not null,
 c1 varchar(20) not null,
 c2 varchar(30) not null,
 c3 int not null,
 c4 varchar(30) not null,
 PRIMARY KEY (id),
 KEY idx_c3 (c3)
) ENGINE=InnoDB;

CREATE USER IF NOT EXISTS '${USER}'@'%';
ALTER USER '${USER}'@'%' IDENTIFIED BY '${PWD}';
GRANT ALL PRIVILEGES ON test.t1 TO '${USER}'@'%';
EOSQL

${MYSQL} -f -e "${bugSQL}"

echo "2. Starting run test"

${MYSQL} -e "truncate table test.t1;"

for i in {1..300}
do
 mysql_random_data_load -u${USER} -p${PWD} -h${HOST} -P${PORT} --max-threads=2 test t1 1000 >/dev/null2>&1
c_before_del=`${MYSQL}-e"selectcount(*)fromtest.t1;"`
${MYSQL}-e"deletefromtest.t1LIMIT980;"
c_before_alter=`${MYSQL}-e"selectcount(*)fromtest.t1;"`
${MYSQL}-e"altertabletest.t1engine=innodb;"
c_after_alter=`${MYSQL}-e"selectcount(*)fromtest.t1;"`
if[${c_before_alter}-ne${c_after_alter}];then
echo"run${i}times,beforealter:${c_before_alter},afteralter:${c_after_alter}"
exit
fi
if[`expr${i}%10`-eq0];then
echo"run${i}times"
fi
done

从多次反复测试的结果来看,大致的规律是当执行 ALTER TABLE 操作特别频繁时,就可能会在表重建时遇到被 Purge 的记录还没来得及被抹掉,这就比较容易触发问题。试着把上面的测例1做些微调,把 ALTER TABLE 这部分的处理逻辑修改成下面这样:

...
47if[`expr${i}%20`-eq0];then
48sleep2
49${MYSQL}-e"altertabletest.t1engine=innodb;"
50fi
...

即每完成20轮测试后再执行 ALTER TABLE 操作,并且在此之前还要先休眠等待2秒。改用新逻辑后,就没再触发问题。

模拟测例3

提示:该测例需要改成MySQL debug版本运行(平时使用的是release二进制包,是无法复现的)。

准备测试数据

CREATETABLEt1(pkCHAR(5)PRIMARYKEY);
INSERTINTOt1VALUES('aaaaa'),('bbbbb'),('bbbcc'),('ccccc'),('ddddd'),('eeeee');

测试方法

S1 S2
这一步的目的是2行数据key buffer就满
SET DEBUG='+d,ddl_buf_add_two';
set global innodb_purge_stop_now=ON;
DELETE FROM t1 WHERE pk = 'bbbcc';
进行DDL,并且来到ddl0par-scan.cc:238 行
ALTER TABLE t1 ENGINE=InnoDB, ALGORITHM=INPLACE
SET GLOBAL innodb_purge_run_now=ON;
DDL继续进程(丢数据)

测试结果

285831a8-c0df-11ef-9310-92fbcf53809c.jpg

写在后面

在线上生产环境中,除了必要的增删字段、增删索引、修改字段定义外,直接执行 ALTER TABLE ... ENGINE=InnoDB 或 OPTIMIZE TABLE 重建整个表空间的行为还是比较少的,尤其是操作大表时,也基本上都习惯了用类似 gt-osc 之类的第三方辅助工具来完成。

此外,调大 innodb_ddl_buffer_size 参数值也可以应对大部分业务表的DDL操作需求,在我的测试中,调大到10MB就可以保证上述测试表有几十万行数据时不出问题,调大到100MB则可以保证上述测试表有千万行数据时不出问题。如果是更大、更宽的表就需要进一步测试验证了。

总的来看,这个问题在线上生产环境中并不是百分百会触发,只是存在一定较低的几率,在文章一开始也提到了几个可以规避的方法,所以说其影响其实也是有限的,不必过于紧张。先采用紧急办法规避问题,后面再择机升级版本就好。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • MySQL
    +关注

    关注

    1

    文章

    930

    浏览量

    29748
  • DDL
    DDL
    +关注

    关注

    0

    文章

    14

    浏览量

    6570

原文标题:MySQL 8.0/8.4执行DDL会丢数据?是,但影响有限

文章出处:【微信号:OSC开源社区,微信公众号:OSC开源社区】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    MySQL数据库慢查询分析与优化实战

    在讨论MySQL慢查询之前,需要先明确一个关键前提:什么是慢查询? 不同业务场景下,慢查询的定义差异巨大。一个数据报表后台的SQL执行30秒可能属于正常范围,但一个订单创建的数据库操作
    的头像 发表于 04-02 09:38 167次阅读

    NineData 新增支持 MySQL 到 openGauss PostgreSQL 数据复制链路

    MySQL 到 openGauss PostgreSQL 兼容版的迁移,真正难的从来不是“把数据搬过去”,而是如何在业务不停、数据持续变化、结果需要验证、问题需要及时发现的前提下,把整个迁移过程稳稳
    的头像 发表于 03-19 11:44 198次阅读
    NineData 新增支持 <b class='flag-5'>MySQL</b> 到 openGauss PostgreSQL <b class='flag-5'>数据</b>复制链路

    MySQL主从延迟排查全流程

    复制延迟一上来,很多人先盯 Seconds_Behind_Master。这个指标当然要看,但它只能告诉你“延迟已经发生了”,不能告诉你是网络拉取慢、Relay Log 堆积、SQL 线程执行慢、并行复制没吃满,还是下游被长事务、DDL、热点表拖住了。
    的头像 发表于 03-11 09:49 367次阅读

    恒讯科技解析:如何安装MySQL并创建数据

    安装和管理MySQL不必复杂。只需几分钟,你就能在Linux服务器上搭建MySQL,创建第一个数据库,甚至自动化备份——同时确保数据安全有序。 什么是
    的头像 发表于 01-14 14:25 351次阅读

    工业数据中台支持接入MySQL数据库吗

    工业数据中台完全支持接入MySQL数据库 ,且通过数据同步、集成与治理等技术手段,能够充分发挥MySQL
    的头像 发表于 12-04 11:23 506次阅读
    工业<b class='flag-5'>数据</b>中台支持接入<b class='flag-5'>MySQL</b><b class='flag-5'>数据</b>库吗

    Mysql数据恢复—Windows Server下MySQL(InnoDB)全表误删数据恢复案例

    本地服务器,操作系统为windows server。服务器上部署mysql单实例,innodb引擎,独立表空间。未进行数据库备份,未开启binlog。 人为误操作使用Delete命令删除数据时未添加where子句,导致全表
    的头像 发表于 09-23 15:56 866次阅读
    <b class='flag-5'>Mysql</b><b class='flag-5'>数据</b>恢复—Windows Server下<b class='flag-5'>MySQL</b>(InnoDB)全表误删<b class='flag-5'>数据</b>恢复案例

    mysql数据恢复—mysql数据库表被truncate的数据恢复案例

    某云ECS网站服务器,linux操作系统,部署了mysql数据库。工作人员在执行数据库版本更新测试时,错误地将本应在测试库执行的sql脚本在生产库上
    的头像 发表于 09-11 09:28 1176次阅读
    <b class='flag-5'>mysql</b><b class='flag-5'>数据</b>恢复—<b class='flag-5'>mysql</b><b class='flag-5'>数据</b>库表被truncate的<b class='flag-5'>数据</b>恢复案例

    CentOS 7下MySQL 8双主热备高可用架构全解

    MySQL主节点2 核心逻辑: 通过Keepalived实现VIP漂移 双向GTID同步保证数据一致性 双写模式需配合应用层冲突解决机制 MySQL 8部署流程 ▶ 步骤1:官方源配置 wget
    的头像 发表于 08-12 17:08 983次阅读

    0.1 至 8.0 GHz SP3T 开关 skyworksinc

    电子发烧友网为你提供()0.1 至 8.0 GHz SP3T 开关相关产品参数、数据手册,更有0.1 至 8.0 GHz SP3T 开关的引脚图、接线图、封装手册、中文资料、英文资料,0.1 至
    发表于 08-07 18:31
    0.1 至 <b class='flag-5'>8.0</b> GHz SP3T 开关 skyworksinc

    MySQL 8.0性能优化实战指南

    作为一名运维工程师,MySQL数据库优化是我们日常工作中最具挑战性的任务之一。MySQL 8.0作为当前主流版本,在性能、安全性和功能上都有了显著提升,但如何充分发挥其潜力,仍需要我们
    的头像 发表于 07-24 11:48 1045次阅读

    MySQL数据备份与恢复策略

    数据是企业的核心资产,MySQL作为主流的关系型数据库管理系统,其数据的安全性和可靠性至关重要。本文将深入探讨MySQL
    的头像 发表于 07-14 11:11 881次阅读

    企业级MySQL数据库管理指南

    在当今数字化时代,MySQL作为全球最受欢迎的开源关系型数据库,承载着企业核心业务数据的存储与处理。作为数据库管理员(DBA),掌握MySQL
    的头像 发表于 07-09 09:50 886次阅读

    MYSQL集群高可用和数据监控平台实现方案

    该项目共分为2个子项目,由MYSQL集群高可用和数据监控平台两部分组成。
    的头像 发表于 05-28 10:10 1453次阅读
    <b class='flag-5'>MYSQL</b>集群高可用和<b class='flag-5'>数据</b>监控平台实现方案

    MySQL数据库采集网关是什么?有什么功能?

    MySQL数据库采集网关是一种用于连接、采集、处理并传输数据MySQL数据库的中间设备或软件系统,通常部署在
    的头像 发表于 05-26 15:20 814次阅读

    MySQL数据库是什么

    MySQL数据库是一种 开源的关系型数据库管理系统(RDBMS) ,由瑞典MySQL AB公司开发,后被Oracle公司收购。它通过结构化查询语言(SQL)进行
    的头像 发表于 05-23 09:18 1438次阅读