同事删库跑路后,我连表名都不能修改了?

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 同事删库跑路后,我连表名都不能修改了?


事情是这样的,前几天隔壁部门的哥们在生产环境的数据库上,执行了一下drop命令,好嘛,活生生的删库跑路的例子居然真的在我身边发生了,好在运维同学给力,后来恢复了数据。事后听说这哥们虽然没被开除,但也吃了个公司的警告。

再然后,运维那边回收了所有环境下数据库的drop命令的权限,甚至包括了开发环境,本来觉得对我们也没啥影响,一般我们也没有啥需要删表的需求。但是隔了没几天,我在重命名一个表的时候,突然弹出了这样一个报错:

仔细看了一眼报错:

1142 - DROP command denied to user 'hydra'@'localhost' for table 't_orders'

什么情况,重命名表和drop命令还有什么关系?本着怀疑的态度,就想探究一下没有drop权限后,对我们的日常数据库操作都有什么影响,于是就有了后面一系列在本地进行的测试。

首先需要一个没有drop权限的 mysql 用户,我们先在本地环境使用 root 用户登录 mysql,取消用户 hydra 的drop权限。和grant授权命令相对应的,可以使用revoke命令取消对用户的授权:

revoke drop on *.* from hydra@'localhost';

好了,准备工作做完了,It's show time~

修改表名

前面直接使用 navicat 来修改表名失败,那我们再用 sql 命令来尝试一下:

上面测试了两种重命名表的命令,无论是ALTER还是RENAME都不能正常使用,看来drop的权限确实会对修改表名造成影响。至于重命名失败的原因,看一下官方文档(链接:https://dev.mysql.com/doc/refman/5.7/en)的说明:

RENAME TABLE renames one or more tables. You must have ALTER and DROP privileges for the original table, and CREATE and INSERT privileges for the new table.

简单来说就是在重命名表时,必须有原始表的ALTERDROP权限,以及新表的CREATEINSERT权限。

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

truncate

当我需要清空一张表、顺带把AUTO_INCREMENT的主键置为初始值时,突然发现truncate命令也无法执行了:

有了上面的经验,还是看一下官方文档(链接:https://dev.mysql.com/doc/refman/5.7/en)的说明:

Although TRUNCATE TABLE is similar to DELETE, it is classified as a DDL statement rather than a DML statement. It differs from DELETE in the following ways:

Truncate operations drop and re-create the table, which is much faster than deleting rows one by one, particularly for large tables.

文档给出的解释是,尽管truncatedelete的功能很像,但是truncate被归类为 DDL 语言,而delete则是 DML 语言。相对于delete一行行删除数据,truncate删除 表后重新新建表 ,这一操作相对delete会快很多,尤其是对大表而言。

从分类也可以看出两者之间的不同,DML(data manipulation language)作为数据操作语言,主要是针对数据进行一些操作,例如常用的增删改查。而DDL(data definition language)则是数据定义语言,主要应用于定义或改变表的结构等操作,并且这一操作过程是隐性提交的,不能回滚。

truncate无法使用的情况下,来执行一下delete试试:

虽然说不带where条件的delete删除语句很不推荐使用,但是在功能上还是可以执行成功的。那么再看看另一个问题,表中的自增id重置了吗?

我们知道,如果执行了truncate的话,那么自增列id的值会被重置为 1。下面看看delete执行后的情况,插入一条数据并查询:

通过上面的结果,可以看到使用delete清表后,自增列的值还是在原先的基础上进行自增。如果需要重置这个值的话,需要我们手动在表上执行alter命令修改:

alter table t_orders auto_increment= 1;

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

drop 作用范围

那么,是否存在即使在没有权限的情况下,也可以执行成功的drop指令?我们对不同对象分别进行测试,首先尝试对数据库、表、视图的drop操作:

drop DATABASE mall;
> 1044 - Access denied for user 'hydra'@'localhost' to database 'mall'
> 时间: 0.005s
drop TABLE t_orders;
> 1142 - DROP command denied to user 'hydra'@'localhost' for table 't_orders'
> 时间: 0s
drop VIEW order_view;
> 1142 - DROP command denied to user 'hydra'@'localhost' for table 'order_view'
> 时间: 0.001s

上面这些命令理所当然没有执行成功,但是在尝试到使用drop删除存储过程时,意料之外的结果出现了。在没有drop权限的情况下,对存储过程的drop操作,居然可以执行成功:

翻到官方文档(链接:https://dev.mysql.com/doc/refman/5.7/en)中授权这一章节,看一下这张图就明白了:

上面的表进行了解释,drop命令的作用范围仅仅是数据库、表以及视图,而存储过程的权限被单独放在alter routine中了,因此即使没有drop权限,我们仍可以用drop命令来删除存储过程。

delete 后如何恢复数据

通过前面的实验可以看到,虽然在回收drop权限后不能使用truncate清空数据表了,但我们仍然可以使用delete语句达到相同的效果,那么为什么delete就不害怕删库的风险呢?

前面我们提到过,delete语句属于 DML 语言,其实在实际的删除过程中是一行行地进行删除的,并且会将每行数据的删除日志记录在日志中,下面我们就看看如何利用binlog来恢复删除的数据。

首先要求数据库开启binlog,使用下面的语句来查询是否开启:

show variables like '%log_bin%';

在值为ON的情况下,表示开启了binglog

确保开启了binlog后,我们使用delete来删除表中的全部数据:

delete from t_orders;

在恢复删除的数据前,需要先找到存放数据文件的目录:

在该目录下,存在若干名称为mysql-bin.**** *的文件,我们需要根据删除操作发生的时间找到临近的binglog文件:

找到目标binlog文件后,这里先将它拷贝到D:\tmp目录下,然后到 mysql 安装目录的bin目录下,执行下面的指令:

mysqlbinlog --base64-output=decode-rows -v 
  --database=mall 
  --start-datetime="2021-09-17 20:50:00" 
  --stop-datetime="2021-09-17 21:30:00" 
  D:\tmp\mysql-bin.000001 > mysqllog.sql

对参数进行一下说明:

  • base64-output=decode-rows:基于行事件解析成 sql 语句,并将数据转换正常的字符。
  • database:数据库名。
  • start-datetime:从 binlog 中第一个等于或晚于该时间戳的事件开始读取,也就是恢复数据的起始时间。
  • stop-datetime:与上面对应的,是恢复数据的结束时间。
  • D:\tmp\mysql-bin.000001:恢复数据的日志文件。
  • mysqllog.sql:恢复数据的输出文件。

执行完成后,在bin目录下会生成一个mysqllog.sql的文件,打开文件看一下,可以找到删除时执行的delete语句:

从语句中可以拿到delete命令执行时每一行数据的值,这样就可以进行数据的恢复了。如果需要恢复的数据量非常大的话,建议使用脚本批量将delete语句转换为insert语句,减轻恢复数据的工作量。

所以,答应我,以后删库前,先看一下有没有开启 binlog 好吗?



相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
2月前
|
SQL Java 关系型数据库
删除几条数据怎么搞
删除几条数据怎么搞
|
4月前
|
存储 关系型数据库 MySQL
(十五)MySQL命令大全:以后再也不用担心忘记SQL该怎么写啦~
相信大家在编写SQL时一定有一个困扰,就是明明记得数据库中有个命令/函数,可以实现自己需要的功能,但偏偏不记得哪个命令该怎么写了,这时只能靠盲目的去百度,以此来寻找自己需要的命令。
155 28
|
SQL 存储 安全
又有程序员删库跑路?还好我早有准备
又有程序员删库跑路?还好我早有准备
83 0
|
运维 安全 固态存储
不需要的binlog如何手动干掉?放心,这不是删库更不用跑路。
不需要的binlog如何手动干掉?放心,这不是删库更不用跑路。
196 0
|
运维 前端开发 JavaScript
删库跑路后的现场还原
遭遇删库跑路怎么办?可观测性是研发质量和产品的试金石,是企业城墙的基石,这里拿删库跑路举一个栗子,说明可观测性的重要程度,用好可观测性,能更了解系统,扩宽业务。
202 0
咱们也聊聊删库跑路吧!
咱们也聊聊删库跑路吧!
|
SQL 关系型数据库 MySQL
开学实验里要用到mysql,忘记基本的select语句怎么玩啦?补救来啦(1)
开学实验里要用到mysql,忘记基本的select语句怎么玩啦?补救来啦(1)
128 0
开学实验里要用到mysql,忘记基本的select语句怎么玩啦?补救来啦(1)
|
SQL 关系型数据库 MySQL
开学实验里要用到mysql,忘记基本的select语句怎么玩啦?补救来啦(2)
开学实验里要用到mysql,忘记基本的select语句怎么玩啦?补救来啦(2)
98 0
开学实验里要用到mysql,忘记基本的select语句怎么玩啦?补救来啦(2)
|
SQL 安全 NoSQL
加班到2点,一不小心我把MySQL删了
大家好,我是Leo。目前在常州从事Java后端。上一篇文章我们介绍了线上数据库挂了一个节点之后,应该如何排查节点宕机问题。从select 1 ,外部统计,内部统计等一系列流程方案的介绍。这一篇我们介绍一下线上数据库误删数据后,到底是跑路还是该如何解决!
加班到2点,一不小心我把MySQL删了
|
弹性计算 Java 关系型数据库
离开小厂进大厂的第一周,mysql修改列名的语法
离开小厂进大厂的第一周,mysql修改列名的语法
离开小厂进大厂的第一周,mysql修改列名的语法