版本控制最關(guān)鍵的一件事是保證數(shù)據(jù)的安全性,不能因?yàn)榇疟P損壞,程序故障造成版本庫(kù)無(wú)可挽回的錯(cuò)誤,為此必須制定較完備的備份策略。在Subversion中,我們有三種備份方式:完全備份,增量備份和同步版本庫(kù)。

1, 完全備份

最常見(jiàn)和簡(jiǎn)單的備份就是直接使用拷貝命令,將版本庫(kù)目錄拷貝到備份目錄上,就可以了。但是這樣不是很安全的方式,因?yàn)槿绻诳截悤r(shí)版本庫(kù)發(fā)生變化,將會(huì)造成備份的結(jié)果不夠準(zhǔn)確,失去備份的作用,為此Subversion提供了“svnadmin hotcopy”命令,可以防止這種問(wèn)題。

還記得我們的版本庫(kù)目錄嗎?

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

如果要把project1備份到d:\svnrootbak目錄下,只需要運(yùn)行:

svnadmin hotcopy d:\svnroot\project1 d:\svnrootbak\project1

但是我們作為配置管理員,必須想辦法優(yōu)化這個(gè)過(guò)程,如果我們這個(gè)目錄下有許多版本庫(kù),需要為每個(gè)版本庫(kù)寫這樣一條語(yǔ)句備份,為此我寫了下面的腳本,實(shí)現(xiàn)備份一個(gè)目錄下的所有版本庫(kù)。我們?cè)贒:\SVNROOT下創(chuàng)建了兩個(gè)文件,simpleBackup.bat:

@echo 正在備份版本庫(kù)%1......
@%SVN_HOME%\bin\svnadmin hotcopy %1 %BACKUP_DIRECTORY%\%2
@echo 版本庫(kù)%1成功備份到了%2!

這個(gè)文件僅僅是對(duì)“svnadmin hotcopy”的包裝,然后是backup.bat:

echo off

rem Subversion的安裝目錄
set SVN_HOME="D:\Subversion"

rem 所有版本庫(kù)的父目錄
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%

rem 驗(yàn)證目錄是否為版本庫(kù),如果是則取出名稱備份
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%已經(jīng)存在,請(qǐng)清空。
goto end

:end

你在使用的時(shí)候,只需要修改backup.bat開(kāi)頭的三個(gè)路徑,將兩個(gè)腳本拷貝到“SVN_ROOT”下就可以了。根據(jù)以上的配置,你只需要運(yùn)行backup.bat,就可以把“SVN_ROOT”下的版本庫(kù)都備份到“BACKUP_SVN_ROOT”里,并且存放在備份所在日的目錄里,例如“D:\svnrootbak\2006-10-22”。

雖然這部分工作很簡(jiǎn)單,可是必須有人定時(shí)地去執(zhí)行這個(gè)操作(例如每周一凌晨),為了避免發(fā)生遺忘的情況,我們可以將這個(gè)操作加入到系統(tǒng)的at任務(wù)當(dāng)中去,例如還是上面的環(huán)境,為了安裝at任務(wù),我們運(yùn)行:

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

這樣在每周一凌晨1:00都會(huì)執(zhí)行這個(gè)備份過(guò)程。當(dāng)然備份在本機(jī)也是不安全的,你也許需要上傳到別的機(jī)器,這個(gè)就要靠你自己去實(shí)現(xiàn)了。


2, 增量備份

盡管完全備份非常簡(jiǎn)單,但是也是有代價(jià)的,當(dāng)版本庫(kù)非常巨大時(shí),經(jīng)常進(jìn)行完全備份是不現(xiàn)實(shí)的,也并不必要,但是一旦版本庫(kù)在備份之間發(fā)生問(wèn)題,該如何呢,這里我們就用到了增量備份。

增量備份通常要與完全備份結(jié)合使用,就像oracle數(shù)據(jù)庫(kù)的歸檔日志,記錄著每次Subversion提交的變化,然后在需要恢復(fù)時(shí)能夠回到最新的可用狀態(tài)。在我們這個(gè)例子中我們使用的是,svnadmin dump命令進(jìn)行增量的備份,使用方法是:

svnadmin dump?project1?--revision 15 --incremental > dumpfile2

上面的命令實(shí)現(xiàn)了對(duì)修訂版本15進(jìn)行增量的備份,其中的輸出文件dumpfile2只保存了修訂版本15更改的內(nèi)容。

為了記錄每次提交的結(jié)果,我們需要使用一項(xiàng)Subversion的特性--鉤子(hook),看看我們的project1目錄:

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

其中的hooks目錄里存放的就是鉤子腳本,我們?cè)诖颂幹皇褂胮ost-commit鉤子,這個(gè)鉤子會(huì)在每次提交之后執(zhí)行,為了實(shí)現(xiàn)我們的備份功能,我們?cè)趆ooks下建立一個(gè)文件post-commit.bat,內(nèi)容如下:

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

通過(guò)這個(gè)腳本,可以實(shí)現(xiàn)D:\svnroot下的版本庫(kù)提交時(shí)自動(dòng)增量備份到D:\svnrootbak\delta(確定這個(gè)目錄存在),其中使用的deltaBackup.bat其實(shí)可以放在任何地方,只是對(duì)腳本的svnadmin dump的包裝,內(nèi)容如下:

@echo 正在備份版本庫(kù)%2......
%SVN_HOME%\bin\svnadmin dump %SVN_ROOT%\%1 --incremental --revision %2 >> %DELTA_BACKUP_SVN_ROOT%\%1.dump
@echo 版本庫(kù)%2成功備份到了%3!

以上兩個(gè)腳本可以直接拷貝到project2的hooks目錄下,不需要修改就可以實(shí)現(xiàn)project2的自動(dòng)備份。

以上的操作已經(jīng)OK了,現(xiàn)在需要做的是將完全備份和增量備份結(jié)合起來(lái),也就是在完全備份后清理增量備份的結(jié)果,使之只保存完全備份后的結(jié)果。

當(dāng)果真出現(xiàn)版本庫(kù)的故障,就要求我們實(shí)現(xiàn)版本庫(kù)的恢復(fù)操作了,這是用要使用svnadmin load命令,同時(shí)也需要上次的完全備份例如要把上次完全備份backuprepo,和之后的增量備份dumpfile:

svnadmin load?backuprepo < dumpfile

最后的結(jié)果,可以下載svnroot.rar,將之解壓縮到d:\下,然后修改幾個(gè)bat文件的SVN_HOME就可以使用了。

3, 版本庫(kù)同步

Subversion 1.4增加了同步機(jī)制,可以實(shí)現(xiàn)一個(gè)版本庫(kù)同另一個(gè)版本庫(kù)的同步(但好像只是單向的),我們可以用來(lái)實(shí)現(xiàn)版本庫(kù)的備份或鏡像。

3.1. 對(duì)目標(biāo)庫(kù)初始化
svnsync init svn://localhost/project2 svn://localhost/project1 
?

其中project2是目標(biāo)的版本庫(kù),而project1是源版本庫(kù)。其中的目標(biāo)版本庫(kù)必須為空,而且必須允許修訂版本屬性的修改,也就是在目標(biāo)的版本庫(kù)的hooks目錄里添加一個(gè)文件pre-revprop-change.bat,內(nèi)容為空即可。

3.2. 同步project2到project1
svnsync sync svn://localhost/project2 
?

這時(shí)候你update一下你的project2的一個(gè)工作拷貝,就會(huì)發(fā)現(xiàn)有了project1的所有內(nèi)容。如果project1又有提交,這時(shí)候project2的版本庫(kù)無(wú)法看到最新的變化,還需要再運(yùn)行一遍sync操作,這樣才能將最新的變化同步。需要注意的是,目標(biāo)版本庫(kù)只能做成只讀的,如果目標(biāo)版本庫(kù)發(fā)生了變更,則無(wú)法繼續(xù)同步了。

3.3. 同步歷史屬性的修改

因?yàn)橥讲粫?huì)更新對(duì)歷史屬性的修改,所以svnsync還有子命令copy-revprops,可以同步某個(gè)版本的屬性。

3.4. 鉤子自動(dòng)同步

希望在每次提交時(shí)同步,則需要在源版本庫(kù)增加post-commit腳本,內(nèi)容如下:

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

把以上內(nèi)容存放為post-commit.bat,然后放到版本庫(kù)project1下的hooks目錄下,這樣project1每次提交,都會(huì)引起project2的同步。