小石头
Excellence in any department can be attained only by the labor of a lifetime; it is not to be purchased at a lesser price.
posts - 91,comments - 22,trackbacks - 0
Subversion备份
作者: rocksun   
2006-10-26

作者:Rock Sun, Subversion中文站。
如有转发请注明出处:http://www.subversion.org.cn/ind ... ;id=85&Itemid=9

版本控制最关键的一件事是保证数据的安全性,不能因为磁盘损坏,程序故障造成版本库无可挽回的错误,为此必须制定较完备的备份策略。在Subversion中,我们有三种备份方式:完全备份,增量备份和同步版本库。
1, 完全备份

最常见和简单的备份就是直接使用拷贝命令,将版本库目录拷贝到备份目录上,就可以了。但是这样不是很安全的方式,因为如果在拷贝时版本库发生变化,将会造成备份的结果不够准确,失去备份的作用,为此Subversion提供了“svnadmin hotcopy”命令,可以防止这种问题。

还记得我们的版本库目录吗?

    D:\SVNROOT
    ├─project1
    │  ├─conf
    │  ├─dav
    │  ├─db
    │  │  ├─revprops
    │  │  ├─revs
    │  │  └─transactions
    │  ├─hooks
    │  └─locks
    └─project2
        ├─conf
        ├─dav
        ├─db
        │  ├─revprops
        │  ├─revs
        │  └─transactions
        ├─hooks
        └─locks
      

我们在D:\SVNROOT下创建了两个文件,simpleBackup.bat:

    @echo 正在备份版本库%1......
    @%SVN_HOME%\bin\svnadmin hotcopy %1 %BACKUP_DIRECTORY%\%2
    @echo 版本库%1成功备份到了%2!

这个文件仅仅是对“svnadmin hotcopy”的包装,然后是backup.bat:

    echo off

    rem Subversion的安装目录
    set SVN_HOME="D:\Subversion"

    rem 所有版本库的父目录
    set SVN_ROOT=D:\svnroot

    rem 备份的目录
    set BACKUP_SVN_ROOT=D:\svnrootbak

    set BACKUP_DIRECTORY=%BACKUP_SVN_ROOT%\%date:~0,10%
    if exist %BACKUP_DIRECTORY% goto checkBack
    echo 建立备份目录%BACKUP_DIRECTORY%>>%SVN_ROOT%/backup.log

    mkdir %BACKUP_DIRECTORY%
    for /r %SVN_ROOT% %%I in (.) do @if exist "%%I\conf\svnserve.conf" %SVN_ROOT%\simpleBackup.bat "%%~fI" %%~nI
    goto end

    :checkBack
    echo 备份目录%BACKUP_DIRECTORY%已经存在,请清空。
    goto end

    :end

根据以上的配置,你只需要运行backup.bat,就可以把“SVN_ROOT”下的版本库都备份到“BACKUP_SVN_ROOT”里,并且存放在备份所在日的目录里,例如“D:\svnrootbak\2006-10-22”。

虽然这部分工作很简单,可是必须有人定时地去执行这个操作(例如每周五),为了避免发生遗忘的情况,我们可以将这个操作加入到系统的at人物当中去,例如还是上面的环境,为了安装at,我们运行:

    at 1:00 /every:M D:\svnroot\backup.bat

这样在每周一凌晨1:00都会执行这个备份过程。当然备份在本机也是不安全的,你也许需要上传到别的机器,这个就要靠你自己去实现了。

2, 增量备份

尽管完全备份非常简单,但是也是有代价的,当版本库非常巨大时,经常进行完全备份是不现实的,也并不必要,但是一旦版本库在备份之间发生问题,该如何呢,这里我们就用到了增量备份。

增量备份通常要与完全备份结合使用,就像oracle数据库的归档日志,记录着每次Subversion提交的变化,然后在需要恢复时能够回到最新的可用状态。

为了记录每次提交的结果,我们需要使用一项Subversion的特性--钩子(hook),看看我们的project1目录:

    ├─project1
    │  ├─conf
    │  ├─dav
    │  ├─db
    │  │  ├─revprops
    │  │  ├─revs
    │  │  └─transactions
    │  ├─hooks
    │  └─locks

其中的hooks目录里存放的就是钩子脚本,我们在此处只使用post-commit钩子,这个钩子会在每次提交之后执行,为了实现我们的备份功能,我们在hooks下建立一个文件post-commit.bat,内容如下:

    echo off
    set SVN_HOME="C:\Program Files\Subversion"
    set SVN_ROOT=D:\svnroot
    set UNIX_SVN_ROOT=D:/svnroot
    set DELTA_BACKUP_SVN_ROOT=D:\svnrootbak\delta
    set LOG_FILE=%1\backup.log
    echo backup revision %2 >> %LOG_FILE%
    for /r %SVN_ROOT% %%I in (.) do if D:/svnroot/%%~nI == %1 %SVN_ROOT%\%%~nI\hooks\deltaBackup.bat %%~nI %2
    goto end
    :end

通过这个脚本,可以实现D:\svnroot下的版本库提交时自动增量备份到D:\svnrootbak\delta(确定这个目录存在),其中使用的deltaBackup.bat其实可以放在任何地方,只是对脚本的svnadmin dump的包装,内容如下:

    @echo 正在备份版本库%2......
    %SVN_HOME%\bin\svnadmin dump %SVN_ROOT%\%1 --incremental --revision %2 >> %DELTA_BACKUP_SVN_ROOT%\%1.dump
    @echo 版本库%2成功备份到了%3!

以上两个脚本可以直接拷贝到project2的hooks目录下,不需要修改就可以实现project2的自动备份。

以上的操作已经OK了,现在需要做的是将完全备份和增量备份结合起来,也就是在完全备份后清理增量备份的结果,使只保存完全备份后的结果。

最后的结果,可以下载附件,将之解压缩到d:\下,然后修改几个bat文件的SVN_HOME就可以使用了。
3, 版本库同步

Subversion 1.4增加了同步机制,可以实现一个版本库同另一个版本库的同步(但好像只是单向的),我们可以用来实现版本库的备份或镜像。
3.1. 对目标库初始化

    svnsync init svn://localhost/project2 svn://localhost/project1
     

其中project2是目标的版本库,而project1是源版本库。其中的目标版本库必须为空,而且必须训育修订版本属性的修改,也就是在目标的版本库的hooks目录里添加一个文件pre-revprop-change.bat,内容为空即可。
3.2. 同步project2到project1

    svnsync sync svn://localhost/project2
     

这时候你update一下你的project2的一个工作拷贝,就会发现有了project1的所有内容。如果project1又有提交,这时候 project2的版本库无法看到最新的变化,还需要再运行一遍sync操作,这样才能将最新的变化同步。需要注意的是,目标版本库只能做成只读的,如果目标版本库发生了变更,则无法继续同步了。
3.3. 同步历史属性的修改

因为同步不会更新对历史属性的修改,所以svnsync还有子命令copy-revprops,可以同步某个版本的属性。
3.4. 钩子自动同步

希望在每次提交时同步,则需要在源版本库增加post-commit脚本,内容如下:

echo off
set SVN_HOME="D:\Subversion"
%SVN_HOME%\bin\svnsync sync  --non-interactive svn://localhost/project2


把以上内容存放为post-commit.bat,然后放到版本库project1下的hooks目录下,这样project1每次提交,都会引起project12的同步。

8 评论

好!

nannanRank: 9Rank: 9Rank: 92006-10-27 15:52

版本库同步
3.1-3.3都可以实现。
但是"3.4. 钩子自动同步"不行。
我的post-commit.bat是这样的:
echo off
set SVN_HOME="C:\Program Files\Subversion"
%SVN_HOME%\bin\svnsync sync --non-interactive http://127.0.0.1/svn/MyData2
这是什么原因呢?

JeffreyRank: 12006-11-10 10:54

你单独执行一下post-commit.bat,看看报告什么错误。

rocksunRank: 9Rank: 9Rank: 92006-11-10 11:06

我在.bat文件中加了pause。
执行的情况是

F:\XMDKSVN\MyData\hooks>echo off
请按任意键继续. . .

[ 本帖最后由 Jeffrey 于 2006-11-10 11:51 编辑 ]

JeffreyRank: 12006-11-10 11:45

嗯,应该没问题把?

rocksunRank: 9Rank: 9Rank: 92006-11-10 12:09

先谢谢rocksun。
嗯。我单打开post-commit.bat,如果MyData有变化了,MyData@是可以和它同步的。但就是在提交时,两个没办法同步。
这是怎么回事呢。

JeffreyRank: 12006-11-10 12:37

应该没问题吧?

post-commit和提交是完全独立的,不会互相影响

rocksunRank: 9Rank: 9Rank: 92006-11-10 14:17

我还是不明白耶。

dentyRank: 12006-12-18 09:34

posted on 2007-03-05 10:52 小石头 阅读(587) 评论(0)  编辑  收藏 所属分类: Subversion

只有注册用户登录后才能发表评论。


网站导航: