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也会在下一个新版本中修复该问题。

问题复现/interwetten与威廉的赔率体系

模拟测例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

    文章

    807

    浏览量

    26547
  • DDL
    DDL
    +关注

    关注

    0

    文章

    13

    浏览量

    6327

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

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

收藏 人收藏

    评论

    相关推荐

    Flexus X 实例搭配华为云 EulerOS,快速部署 MySQL执行读写性能测试

    社区 openEuler 构建的 linux 操作系统,提供云原生、高性能、安全稳定的执行环境来开发和运行应用程序,助力企业客户快速上云及开发者创新 MySQL 安装与启动 原计划是通过指
    的头像 发表于 12-24 12:27 38次阅读
    Flexus X 实例搭配华为云 EulerOS,快速部署 <b class='flag-5'>MySQL</b> 并<b class='flag-5'>执行</b>读写性能测试

    数据数据恢复—Mysql数据库表记录丢失的数据恢复流程

    Mysql数据库故障: Mysql数据库表记录丢失。 Mysql数据库故障表现: 1、
    的头像 发表于 12-16 11:05 127次阅读
    <b class='flag-5'>数据</b>库<b class='flag-5'>数据</b>恢复—<b class='flag-5'>Mysql</b><b class='flag-5'>数据</b>库表记录丢失的<b class='flag-5'>数据</b>恢复流程

    数据数据恢复—MYSQL数据库ibdata1文件损坏的数据恢复案例

    mysql数据库故障: mysql数据库文件ibdata1、MYI、MYD损坏。 故障表现:1、数据库无法进行查询等操作;2、使用my
    的头像 发表于 12-09 11:05 147次阅读

    香港云服务器怎么部署MySQL数据库?

    在香港云服务器上部署MySQL数据库的步骤如下: 步骤 1: 更新软件包列表 首先,确保软件包列表是最新的。在终端中执行以下命令: sudo apt update 步骤 2: 安装 MySQL
    的头像 发表于 11-14 16:15 160次阅读

    用PCM1870采样电话中的声音,采得的数据有时是没有的,为什么?

    我用PCM1870(16位音频AD)采样电话中的声音,但采得的数据有时是没有的,是0,就数据了,但我用示波器看输入的信号,信号确实是输入了的,但输出就是没有啊???
    发表于 11-08 06:43

    适用于MySQL的dbForge架构比较

    dbForge Schema Compare for MySQL 是一种工具,用于轻松有效地比较和部署 MySQL 数据库结构和脚本文件夹差异。该工具提供了 MySQL
    的头像 发表于 10-28 09:41 195次阅读
    适用于<b class='flag-5'>MySQL</b>的dbForge架构比较

    MySQL如何优雅的执行DDL

    一、前言 关于MySQL DDL表结构变更,各个工单平台基本上都支持了pt-osc及Online DDL的方式,但是,我相信仍然有一大部分人,不太了解这两种方式各自的优缺点是啥,以至于实际当中,会
    的头像 发表于 07-18 15:06 258次阅读
    <b class='flag-5'>MySQL</b>如何优雅的<b class='flag-5'>执行</b><b class='flag-5'>DDL</b>

    串口通信的时候怎么避免包的情况?

    1.如何避免在中断里面执行长时间的操作 2.串口通信的时候怎么避免包的情况 3.串口通信为什么不可以一次发送1000bit或者10000bit 也就是说一帧数据为 一位起始位 10000bit
    发表于 07-03 07:00

    如何将MS访问数据转换为MySQL

    借助dbForgeStudio for MySQL,您可以轻松地将数据从MicrosoftAccess迁移到MySQL,并保持数据和功能的完整性。这个过程将允许您利用更具可伸缩性和功能
    的头像 发表于 01-23 13:47 424次阅读
    如何将MS访问<b class='flag-5'>数据</b>转换为<b class='flag-5'>MySQL</b>

    查询SQL在mysql内部是如何执行

    我们知道在mySQL客户端,输入一条查询SQL,然后看到返回查询的结果。这条查询语句在 MySQL 内部到底是如何执行的呢?本文跟大家探讨一下哈,我们先来看下MySQL基本架构~
    的头像 发表于 01-22 14:53 565次阅读
    查询SQL在<b class='flag-5'>mysql</b>内部是如何<b class='flag-5'>执行</b>?

    MES系统需要的主要数据有哪些?

    电子发烧友网站提供《MES系统需要的主要数据有哪些?.docx》资料免费下载
    发表于 01-17 09:09 0次下载

    mysql8.0流程控制介绍

    ,流程就分为三大类:   顺序结构:程序从上往下依次执行    分支结构:程序按条件进行选择执行,从两条或多条路径中选择一条执行    循环结构:程序满足一定条件下,重复执行一组语句针
    的头像 发表于 01-13 10:36 688次阅读

    MySQL密码忘记了怎么办?MySQL密码快速重置方法步骤命令示例!

    MySQL密码忘记了怎么办?MySQL密码快速重置方法步骤命令示例! MySQL是一种常用的关系型数据库管理系统,如果你忘记了MySQL的密
    的头像 发表于 01-12 16:06 744次阅读

    mysql怎么新建一个数据

    mysql怎么新建一个数据库 如何新建一个数据库在MySQL中 创建一个数据库是MySQL中的基
    的头像 发表于 12-28 10:01 887次阅读

    mysql密码忘了怎么重置

    mysql密码忘了怎么重置  MySQL是一种开源的关系型数据库管理系统,密码用于保护数据库的安全性和保密性。如果你忘记了MySQL的密码,
    的头像 发表于 12-27 16:51 6671次阅读