源码管理 - SVN的备份和还原

备份策略
==============
svn备份一般采用三种方式:

1)svnadmindump

2)svnadminhotcopy

3)svnsync.

注意,svn备份不宜采用普通的文件拷贝方式(除非你备份的时候将库暂停),如copy命令、rsync命令。

笔者曾经用rsync命令来做增量和全量备份,在季度备份检查审计中,发现备份出来的库大部分都不可用,因此最好是用svn本身提供的功能来进行备份。

优缺点分析

==============

第一种svnadmindump是官方推荐的备份方式,优点是比较灵活,可以全量备份也可以增量备份,并提供了版本恢复机制。

缺点是:如果版本比较大,如版本数增长到数万、数十万,那么dump的过程将非常慢;备份耗时,恢复更耗时;不利于快速进行灾难恢复。

个人建议在版本数比较小的情况下使用这种备份方式。

第二种svnadminhotcopy原设计目的估计不是用来备份的,只能进行全量拷贝,不能进行增量备份;

优点是:备份过程较快,灾难恢复也很快;如果备份机上已经搭建了svn服务,甚至不需要恢复,只需要进行简单配置即可切换到备份库上工作。

缺点是:比较耗费硬盘,需要有较大的硬盘支持(俺的备份机有1TB空间,呵呵)。

第三种svnsync实际上是制作2个镜像库,当一个坏了的时候,可以迅速切换到另一个。不过,必须svn1.4版本以上才支持这个功能。

优点是:当制作成2个镜像库的时候起到双机实时备份的作用;

缺点是:当作为2个镜像库使用时,没办法做到“想完全抛弃今天的修改恢复到昨晚的样子”;而当作为普通备份机制每日备份时,操作又较前2种方法麻烦。
备份的命令
==============
全备份:使用svnadmin dump或svnadmin hotcopy或svnsync来做,

hotcopy:

svnadminhotcopypath/to/repositorypath/to/backup–clean-logs

dump:

svnadmindump版本库路径及名称–revision导出的版本号>导出的命名

增量备份:使用svnadmindump的–incremental选项来实现

svnadmindump版本库路径及名称–revision上次导出的版本号:到本次要导出到的版本号–incremental>导出的命名

一个技巧:如果你有一个较大的Subsersion版本库而你又想用最少的空间来将它备份下来,用这个命令(请将/repo替换成你的版本库路径)吧:

svnadmindump–deltas/repo|bzip2|teedump.bz2|md5sum>dump.md5

分步解释:最重要的一步是-deltas,将消耗更多的CPU资源,但拥有更有效的差异存储办法。

bzip2压缩方案比gzip慢,但换来的更好的压缩率。

更有趣的是,tee方法将压缩的数据流转向到文件dump.bz2,同时将其输出到标准输出,后者有转向给了MD5摘要计算工具。
还原命令
==============
还原版本:svnadmin load 要恢复的版本库路径及名称 < 导出的命名svnadmin hotcopy path/to/repository path/to/backup –clean-logs

svnadmin dump

==============

这是subversion官方推荐的备份方式。 

1)定义备份策略:

备份频度:每周六进行一次全量备份,每周日到周五进行增量备份

备份地点:备份存储路径到/home/backup/svn/

备份命名:全量备份文件名为:weekly_fully_backup.yymmdd,增量备份文件命名为:daily-incremental-backup.yymmdd

备份时间:每晚21点开始

备份检查:每月末进行svnadminload恢复试验。

2)建立全量备份脚本:

在~/下建立一个perl脚本文件,名为weekly_backup.pl,执行全量备份,并压缩备份文件,代码如下(本代码只针对一个库的备份,如果是多个库请做相应改动):

#!/usr/bin/perl-w

my$svn_repos="/home/svn/repos/project1";

my$backup_dir="/home/backup/svn/";

my$next_backup_file="weekly_fully_backup.".`date+%Y%m%d`;

$youngest=`svnlookyoungest$svn_repos`;

chomp$youngest;

print"Backinguptorevision$youngest\n";

my$svnadmin_cmd="svnadmindump--revision0:$youngest$svn_repos>$backup_dir/$next_backup_file";

`$svnadmin_cmd`;

open(LOG,">$backup_dir/last_backed_up");#记录备份的版本号

printLOG$youngest;

closeLOG;

#如果想节约空间,则再执行下面的压缩脚本

print"Compressingdumpfile...\n";

print`gzip-g$backup_dir/$next_backup_file`;

3)建立增量备份脚本:

在全量备份的基础上,进行增量备份:在~/下建立一个perl脚本文件,名为:daily_backup.pl,代码如下:

#!/usr/bin/perl-w

my$svn_repos="/home/svn/repos/project1";

my$backup_dir="/home/backup/svn/";

my$next_backup_file="daily_incremental_backup.".`date+%Y%m%d`;

open(IN,"$backup_dir/last_backed_up");

$previous_youngest=<IN>;

chomp$previous_youngest;

closeIN;

$youngest=`svnlookyoungest$svn_repos`;

chomp$youngest;

if($youngesteq$previous_youngest)

{

print"Nonewrevisionstobackup.\n";

exit0;

}

my$first_rev=$previous_youngest+1;

print"Backinguprevisions$youngest...\n";

my$svnadmin_cmd="svnadmindump--incremental--revision$first_rev:$youngest$svn_repos>$backup_dir/$next_backup_file";

`$svnadmin_cmd`;

open(LOG,">$backup_dir/last_backed_up");#记录备份的版本号

printLOG$youngest;

closeLOG;

#如果想节约空间,则再执行下面的压缩脚本

print"Compressingdumpfile...\n";

print`gzip-g$backup_dir/$next_backup_file`;

4)配置/etc/crontab文件

配置/etc/crontab文件,指定每周六执行weekly_backup.pl,指定周一到周五执行daily_backup.pl;

具体步骤俺就不啰嗦了.

5)备份恢复检查

在月底恢复检查中或者在灾难来临时,请按照如下步骤进行恢复:恢复顺序从低版本逐个恢复到高版本;即,先恢复最近的一次完整备份weekly_full_backup.071201(举例),然后恢复紧挨着这个文件的增量备份daily_incremental_backup.071202,再恢复后一天的备份071203,依次类推。如下:

user1>mkdirnewrepos

user1>svnadmincreatenewrepos

user1>svnadminloadnewrepos<weekly_full_backup.071201

user1>svnadminloadnewrepos<daily_incremental_backup.071202

user1>svnadminloadnewrepos<daily_incremental_backup.071203

....

如果备份时采用了gzip进行压缩,恢复时可将解压缩和恢复命令合并,简单写成:

user1>zcatweekly_full_backup.071201|svnadminloadnewrepos

user1>zcatdaily_incremental_backup.071202|svnadminloadnewrepos

...
 
svnadmin hotcopy整库拷贝方式 
==============
svnadmin hotcopy是将整个库都“热”拷贝一份出来,包括库的钩子脚本、配置文件等;任何时候运行这个脚本都得到一个版本库的安全拷贝,不管是否有其他进程正在使用版本库。 

因此这是俺青睐的备份方式。

1)定义备份策略

备份频度:每天进行一次全量备份,

备份地点:备份目录以日期命名,备份路径到/home/backup/svn/${mmdd}

备份保留时期:保留10天到15天,超过15天的进行删除。

备份时间:每晚21点开始

备份检查:备份完毕后自动运行检查脚本、自动发送报告。

2)建立备份脚本

在自己home目录~/下创建一个文件,backup.sh:

#!/bin/bash

SRCPATH=/home/svn/repos/;#定义仓库parent路径

DISTPATH=/home/backup/svn/`date+\%m%d`/;#定义存放路径;

if[-d"$DISTPATH"]

then

else

mkdir$DISTPATH

chmodg+s$DISTPATH

fi

echo$DISTPATH

svnadminhotcopy$SRCPATH/Project1$DISTPATH/Project1>/home/backup/svn/cpreport.log2>&1;

svnadminhotcopy$SRCPATH/Project2$DISTPATH/Project2

cp$SRCPATH/access$DISTPATH;#备份access文件

cp$SRCPATH/passwd$DISTPATH;#备份passwd文件

perl/home/backup/svn/backup_check.pl#运行检查脚本

perl/home/backup/svn/deletDir.pl#运行删除脚本,对过期备份进行删除。

3)建立检查脚本

在上面指定的地方/home/backup/svn/下建立一个perl脚本:backup_check.pl

备份完整性检查的思路是:对备份的库运行svnlookyoungest,如果能正确打印出最新的版本号,则表明备份文件没有缺失;如果运行报错,则说明备份不完整。我试过如果备份中断,则运行svnlookyoungest会出错。

4)定义删除脚本

由于是全量备份,所以备份不宜保留太多,只需要保留最近10来天的即可,对于超过15天历史的备份基本可以删除了。

   在/home/backup/svn/下建立一个perl脚本:deletDir.pl

5)修改/etc/crontab 文件 

   在该文件中指定每晚21点执行“backup.sh”脚本。

 
svnsync备份
==============
使用svnsync备份很简单,步骤如下: 

1)在备份机上创建一个空库:svnadmincreateProject1

2)更改该库的钩子脚本pre-revprop-change(因为svnsync要改这个库的属性,也就是要将源库的属性备份到这个库,所以要启用这个脚本):

cdSMP/hooks;

cppre-revprop-change.tmplpre-revprop-change;

chmod755pre-revprop-change;

vipre-revprop-change;

将该脚本后面的三句注释掉,或者干脆将它弄成一个空文件。

3)初始化,此时还没有备份任何数据:

svnsyncinitfile:///home/backup/svn/svnsync/Project1/http://svntest.subversion.com/repos/Project1

语法是:svnsyncinit{你刚创建的库url}{源库url}

注意本地url是三个斜杠的:///

4)开始备份(同步):

svnsyncsyncfile:///home/backup/svn/svnsync/Project1

5)建立同步脚本

备份完毕后,建立钩子脚本进行同步。在源库/hooks/下建立/修改post-commit脚本,在其中增加一行,内容如下:

/usr/bin/svnsyncsync--non-interactivefile:///home/backup/svn/svnsync/Project1

  你可能已经注意到上面的备份似乎都是本地备份,不是异地备份。实际上,我是通过将远程的备份机mount(请参阅mount命令)到svn服务器上来实现的,逻辑上看起来是本地备份,物理上实际是异地备份。
 完!

感谢,Thanks!

作者:iTech

出处:http://itech.cnblogs.com/

本文版权归作者iTech所有,转载请包含作者签名和出处,不得用于商业用途,非则追究法律责任!

相关推荐