对于mysql的不完全恢复,我们可以借助于innobackupex的多重备份加上binlog来将数据库恢复到任意时刻。这里的不完全恢复(也叫时点恢复)是相对于完全恢复。本文主要演示了基于innobackupex如何做一个不完全恢复,供大家参考。a、创建演示环境 robin@localhost[(none)]> show variables like 'version'; --当前mysql版本 +---------------+------------+ | variable_name | value | +---------------+------------+ | version | 5.6.12-log | +---------------+------------+ robin@localhost[(none)]> reset master; query ok, 0 rows affected (0.03 sec) robin@localhost[(none)]> use tempdb; robin@localhost[tempdb]> create table tb(id smallint,val varchar(20)); robin@localhost[tempdb]> insert into tb values(1,'fullbak'); --创建一个全备 shell> innobackupex --user=robin -password=xxx --port=3606 --socket=/tmp/mysql3606.sock --defaults-file=/etc/my3606.cnf \ > /hotbak/full --no-timestamp b、创建一个增备 --在创建增备前插入一条记录到tb robin@localhost[tempdb]> insert into tb values(2,'incbak'); shell> innobackupex --user=robin -password=xxx --port=3606 --socket=/tmp/mysql3606.sock --defaults-file=/etc/my3606.cnf \ > --incremental /hotbak/inc --incremental-basedir=/hotbak/full --no-timestamp --再次新增一条记录 robin@localhost[tempdb]> insert into tb values(3,'pointrecover'); query ok, 1 row affected (0.01 sec) --记下当前的时间点用于后续的不完全恢复 robin@localhost[tempdb]> system date; thu dec 25 11:53:54 cst 2014 --模拟误操作 robin@localhost[tempdb]> truncate table tb; query ok, 0 rows affected (0.01 sec) c、再次全备 shell> innobackupex --user=robin -password=xxx --port=3606 --socket=/tmp/mysql3606.sock --defaults-file=/etc/my3606.cnf \ > /hotbak/full2 --no-timestamp --全备后新增一张表 robin@localhost[tempdb]> create table tb_after_truncate(id int,val varchar(20)); query ok, 0 rows affected (0.02 sec)3、演示恢复过程--下面理清一下思路: --当前备份情况: 全备+增备+全备 --我们在增备之后truncate了表tb,然后又创建了一个全备,新建了一个表tb_after_truncate。 --此时我们需要将数据库恢复到truncate(误操作)之前 --解决方案:我们需要利用第一次的全备+增备+binglog来恢复到truncate前,当前第二次全备用不上。 a、先做基于全备的apply,注意,此时使用了--redo-only shell> i【本文来自鸿网互联 (http://www.68idc.cn)】nnobackupex --apply-log --redo-only --user=robin -password=xxx --port=3606 \ > --defaults-file=/etc/my3606.cnf /hotbak/full b、基于增备的apply, --此时没有--redo-only,如果有多个增备,仅仅最后一个增备无需指定--redo-only shell> innobackupex --apply-log --user=robin -password=xxx --port=3606 --defaults-file=/etc/my3606.cnf \ > /hotbak/full --incremental-dir=/hotbak/inc c、进行copy back shell> mysqldown -p3606 --copy back前关闭实例 shell> netstat -nltp|grep mysql|grep 3606 shell> mv /data/inst3606/data3606 /data/inst3606/data3606bk shell> mkdir -p /data/inst3606/data3606 shell> innobackupex --user=robin -password=xxx --port=3606 --copy-back /hotbak/full --defaults-file=/etc/my3606.cnf shell> chown -r mysql:mysql /data/inst3606/data3606 d、启动恢复后的实例 shell> mysqld_safe --defaults-file=/etc/my3606.cnf & shell> mysql -uroot -pxxx -p3606 -s /tmp/mysql3606.sock \ > -e select * from tempdb.tb warning: using a password on the command line interface can be insecure. +------+---------+ | id | val | +------+---------+ | 1 | fullbak | | 2 | incbak | +------+---------+ --获取增量之后的log position shell> cd /hotbak/inc/ shell> more xtrabackup_binlog_info inst3606bin.000001 774 --这里使用了stop-datetime去将日志追加到truncate之前 shell> mysqlbinlog /data/inst3606/log/bin/inst3606bin.000001 --start-position=774 --stop-datetime=2014-12-25 11:53:54 \ > |mysql -urobin -pxxx -p3606 -s /tmp/mysql3606.sock --验证结果如下,可以看到已经恢复到truncate之前了 shell> mysql -uroot -pxxx -p3606 -s /tmp/mysql3606.sock \ > -e select * from tempdb.tb warning: using a password on the command line interface can be insecure. +------+--------------+ | id | val | +------+--------------+ | 1 | fullbak | | 2 | incbak | | 3 | pointrecover | +------+--------------+ --如果我们需要继续恢复后面的事务,我们可以找出truncate前后位置,然后跳过这个position shell> mysqlbinlog /data/inst3606/log/bin/inst3606bin.000001 --start-datetime=2014-12-25 11:53:54|grep truncate -a5 truncate table tb /*!*/; # at 1180 #141225 11:55:35 server id 3606 end_log_pos 1260 crc32 0x12f55fc5 query thread_id=928 exec_time=0 error_code=0 set timestamp=1419479735/*!*/; /*!\c latin1 *//*!*/; -- create table tb_after_truncate(id int,val varchar(20)) /*!*/; # at 1392 #141225 13:06:47 server id 3606 end_log_pos 1415 crc32 0xf956f311 stop delimiter ; # end of log file --我们找出的position为1260,跳过1260之前的继续追加binlog shell> mysqlbinlog /data/inst3606/log/bin/inst3606bin.000001 --start-position=1260 \ > |mysql -urobin -pxxx -p3606 -s /tmp/mysql3606.sock --验证追加后的结果,可以看到表tb_after_truncate存在 [mysql@app ~]$ mysql -uroot -pxxx -p3606 -s /tmp/mysql3606.sock \ > -e desc tempdb.tb_after_truncate warning: using a password on the command line interface can be insecure. +-------+-------------+------+-----+---------+-------+ | field | type | null | key | default | extra | +-------+-------------+------+-----+---------+-------+ | id | int(11) | yes | | null | | | val | varchar(20) | yes | | null | | +-------+-------------+------+-----+---------+-------+4、小结f、也可以跳过故障点,继续追加后面的binlog日志至最新,如本文尾部的演示
