說明
此恢復測試腳本,基于rman備份腳本文章使用的fullbak.sh做的備份。
數據庫將被恢復到RESTORE_LO參數設置的位置。
在恢復完成后,執行一個測試sql,確認數據庫恢復完成,數據庫備份是好的。恢復測試數據庫的參數,比如SGA大小都沒有優化,沒有設置值。如果需要用于其他測試,可能需要適當調整。
主要步驟:
先建立一個init文件,只包含數據庫名,版本號,和一個控制文件位置,用它啟動到nomount, 從autobackup恢復控制文件。 用這個控制文件啟動到mount,進行數據庫恢復。在打開前,修改日志名字到RESTORE_LO指定位置。
1. 檢查當前備份
ls /u02/rmanbackup
得到自動備份文件名如下圖。記錄其自動備份的編號,第一次備份是00.
查看備份日志,得到DBID.
*[oracle@testdb1 rmanbackup]$ grep DBID logs/rman_full_2025-05-29-1547.log
connected to target database: RISK (DBID=3842787173)
[oracle@testdb1 rmanbackup]$*
2. 修改恢復測試腳本
根據上述信息修改恢復腳本。
數據庫ORACLE_SID, DBID,備份日期, 測試時數據庫恢復使用的目錄 /u02/testdata
如果目錄不存在,先手動建立。
mkdir /u02/testdata
對于第一次運行,需要根據實際環境設置,以后基本不需要修改。
3 關閉數據庫并mv當前的spfile(如果不是在同一個服務器恢復,此步可以忽略)
如果在當前服務器恢復(不建議這么做,建議在其他服務器測試)。需要先停止數據庫。并且把當前使用的spfile改名(備份,并防止測試時用它)。
sqlplus / as sysdba
shut immediate;
cd $ORACLE_HOME/dbs
mv spfile${ORACLE_SID}.ora spfile${ORACLE_SID}.ora.def
輸出
SQL> shut immediate
Database closed.
Database dismounted.
ORACLE instance shut down.
SQL> exit
Disconnected from Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production
With the Partitioning, OLAP, Data Mining and Real Application Testing options
[oracle@testdb1 scripts]$ cd $ORACLE_HOME/dbs
[oracle@testdb1 dbs]$ ls -l *risk*
-rw-r----- 1 oracle oinstall 11681792 Apr 11 16:01 cntrlrisk.dbf
-rw-rw---- 1 oracle oinstall 1544 May 29 15:59 hc_risk.dat
-rw-r--r-- 1 oracle oinstall 376 Apr 12 22:00 initrisk.ora
-rw-r--r-- 1 oracle oinstall 83 Apr 12 21:59 initrisk.ora.0412
-rw-r----- 1 oracle oinstall 1536 Apr 22 09:50 orapwrisk
-rw-r----- 1 oracle oinstall 12369920 May 29 15:49 snapcf_risk.f
-rw-r----- 1 oracle oinstall 2560 May 29 05:45 spfilerisk.ora
[oracle@testdb1 dbs]$ mv spfilerisk.ora spfilerisk.ora.def
[oracle@testdb1 dbs]$ ls spfile*
spfilerisk.ora.def
[oracle@testdb1 dbs]$
4. 執行恢復
./auto_rman_test.sh
輸出:
[oracle@testdb1 scripts]$ ps -ef|grep pmon
oracle 4181 2779 0 16:04 pts/0 00:00:00 grep --color=auto pmon
[oracle@testdb1 scripts]$ env|grep SID
ORACLE_SID=risk
[oracle@testdb1 scripts]$ vi auto_rman_test.sh
[oracle@testdb1 scripts]$ clear
[oracle@testdb1 scripts]$ cat auto_rman_test.sh
#!/bin/sh
#########################
# desc: test rman backup to be restored.
# set ORACLE_SID,ORACLE_HOME.
# RMANBAK_LOC: the rman backup is located. it must be same as primary db.
# RESTORE_LOC: the rmanbackup will be restored to.
# TESTSQL: the sql to be tested.
# DBID: the database ID.
# 2020.11.15. v0.1
# update: 2022.12.29 20:00
# update: 2023.01.03. 主庫的在線日志目錄要在備庫上也創建.日志文件將在這個目錄創建.并授予oracle讀寫權限.否則reset logs失敗.
# update: 2023.02.11 增加了until的條件.可以設置時間或者sequence.
# update: 2025.05.17. 如果在原來服務器測試, 需要在測試前,把spfile文件改名。否則會覆蓋以前的控制文件。
############################# 參數設置#################
export ORACLE_HOME=/u01/app/oracle/product/11.2.0/db_1
export ORACLE_SID=risk
export DBID="3842787173"export RMANBAK_LOC="/u02/rmanbackup" #主庫rman備份目錄, 必須和主庫備份目錄相同.
export RESTORE_LOC="/u02/testdata" #恢復到指定目錄
export BACKUP_DATE="20250529" #數據庫備份的時間,用來設置自動備份的控制文件名字,當天是`date +%Y%m%d`
export TESTSQL="select count(*) from dba_tables;"
#export TESTSQL="select * from settlement.t_systemstatus"
#export UNTILTIME="`date +%Y-%m-%d` 03:00:00" #恢復的untiltime, 可以是主庫開始備份的時間,或指定時間,如下
#export UNTILTIME="2023-12-29 17:43:00"
#export UNTILSEQ=851
#export UNTILTIME=""#############局部變量
LOGFILE=/tmp/rman_recovery_`date +%Y%m%d-%H%M`.log
SQLLOG=/tmp/sqlplus_recovery_`date +%Y%m%d-%H%M`.log
MOUNTCMD=/tmp/mount.cmd
RECVCMD=/tmp/recv.cmd
#conrolfile控制文件名字格式, rman設置自動備份控制文件格式為 auto_%T_%F.%d
UPPER_SID=`echo $ORACLE_SID | tr '[:lower:]' '[:upper:]'`
CONTROLFILE=$RMANBAK_LOC/auto_${BACKUP_DATE}_c-$DBID-${BACKUP_DATE}-00.${UPPER_SID}
echo $CONTROLFILEif [ ! -d ${RESTORE_LOC}/${ORACLE_SID} ]; thenmkdir -p ${RESTORE_LOC}/${ORACLE_SID}
ficat >$ORACLE_HOME/dbs/init${ORACLE_SID}.ora<<EOF
db_name=${ORACLE_SID}
control_files=${RESTORE_LOC}/${ORACLE_SID}/contorl01.ctl
compatible='11.2.0.4.0'
EOF#啟動nomount
$ORACLE_HOME/bin/sqlplus -s / as sysdba <<EOF
startup force nomount;
exit;
EOF#創建恢復控制文件腳本
cat > $MOUNTCMD <<EOF
run{
restore controlfile from "$CONTROLFILE";
sql 'alter database mount';
}
EOF#恢復控制文件并mount
$ORACLE_HOME/bin/rman target / cmdfile $MOUNTCMD
if [ $? != 0 ]; thenecho "error mount"exit 1;
fi#設置until條件
echo -e "untiltime is : $UNTILTIME"if [ ${#UNTILTIME} -gt 0 ]; then #沒有指定恢復時間UNTIL_CON="set until time '$UNTILTIME';"
elif [ ${#UNTILSEQ} -gt 0 ]; thenUNTIL_CON="set until sequence $UNTILSEQ;"
elseUNTIL_CON=""
fi
echo $UNTIL_CONcat > $RECVCMD <<EOF
run{
catalog start with "$RMANBAK_LOC" noprompt;
crosscheck backup;
crosscheck archivelog all;
delete noprompt expired backup;
delete noprompt expired archivelog all;
sql "alter session set nls_date_format=''yyyy-mm-dd hh24:mi:ss''";
$UNTIL_CON
set newname for database to '$RESTORE_LOC/$ORACLE_SID/%U';
restore database;
switch datafile all;
switch tempfile all;
recover database;
}
EOFif [ ! -d /tmp/log ]; thenmkdir /tmp/log
fi
mv /tmp/*.log /tmp/log
$ORACLE_HOME/bin/rman target / cmdfile $RECVCMD >$LOGFILE#change online redo log to new location# 新的目標目錄前綴
REDO_LOC="${RESTORE_LOC}/${ORACLE_SID}"# 輸出 SQL 文件
SQL_SCRIPT="rename_redo.sql"
> "$SQL_SCRIPT" # 清空/新建文件# 從數據庫讀取 redo 路徑并生成 SQL
sqlplus -s / as sysdba <<EOF | grep "^/" | while read -r old_path
SET HEADING OFF
SET FEEDBACK OFF
SET PAGESIZE 0
SET TRIMSPOOL ON
SELECT MEMBER FROM V\$LOGFILE;
EXIT
EOF
dofilename=$(basename "$old_path")new_path="${REDO_LOC}/${filename}"echo "ALTER DATABASE RENAME FILE '$old_path' TO '$new_path';" >> "$SQL_SCRIPT"
donesqlplus -s / as sysdba <<EOF
SET HEADING OFF
SET FEEDBACK OFF
SET PAGESIZE 0
SPOOL clear_redo.sqlSELECT 'ALTER DATABASE CLEAR LOGFILE GROUP ' || GROUP# || ';'
FROM V\$LOG
ORDER BY GROUP#;SPOOL OFF
EOF$ORACLE_HOME/bin/sqlplus -s <<EOF >$SQLLOG
conn / as sysdba
start rename_redo.sql
start clear_redo.sql
alter database open resetlogs;
$TESTSQL
exit;
EOFmore $SQLLOG
[oracle@testdb1 scripts]$ ./auto_rman_test.sh
/u02/rmanbackup/auto_20250529_c-3842787173-20250529-00.RISK
ORACLE instance started.Total System Global Area 229683200 bytes
Fixed Size 2251936 bytes
Variable Size 171967328 bytes
Database Buffers 50331648 bytes
Redo Buffers 5132288 bytesRecovery Manager: Release 11.2.0.4.0 - Production on Thu May 29 16:04:57 2025Copyright (c) 1982, 2011, Oracle and/or its affiliates. All rights reserved.connected to target database: RISK (not mounted)RMAN> run{
2> restore controlfile from "/u02/rmanbackup/auto_20250529_c-3842787173-20250529-00.RISK";
3> sql 'alter database mount';
4> }
5>
Starting restore at 29-MAY-25
using target database control file instead of recovery catalog
allocated channel: ORA_DISK_1
channel ORA_DISK_1: SID=170 device type=DISKchannel ORA_DISK_1: restoring control file
channel ORA_DISK_1: restore complete, elapsed time: 00:00:01
output file name=/u02/testdata/risk/contorl01.ctl
Finished restore at 29-MAY-25sql statement: alter database mount
released channel: ORA_DISK_1Recovery Manager complete.
untiltime is :mv: cannot move ‘/tmp/ntpdate.log’ to ‘/tmp/log/ntpdate.log’: Operation not permitted
ALTER DATABASE CLEAR LOGFILE GROUP 1;
ALTER DATABASE CLEAR LOGFILE GROUP 2;
ALTER DATABASE CLEAR LOGFILE GROUP 3;
ALTER DATABASE CLEAR LOGFILE GROUP 4;
ALTER DATABASE CLEAR LOGFILE GROUP 5;
ALTER DATABASE CLEAR LOGFILE GROUP 6;
ALTER DATABASE CLEAR LOGFILE GROUP 7;Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.Database altered.COUNT(*)
----------1322 =》恢復成功,執行了測試sql[oracle@testdb1 scripts]$ cd /u02/testdata/
[oracle@testdb1 testdata]$ ls -l
total 4
drwxr-xr-x 2 oracle oinstall 4096 May 29 16:08 risk
[oracle@testdb1 testdata]$ ls risk
contorl01.ctl data_D-RISK_TS-SYSTEM_FNO-1 data_D-RISK_TS-UNDOTBS1_FNO-3 redo02.log redo05.log
data_D-RISK_TS-SOE_FNO-6 data_D-RISK_TS-TBS_SPLEX_FNO-5 data_D-RISK_TS-USERS_FNO-4 redo03.log redo06.log
data_D-RISK_TS-SYSAUX_FNO-2 data_D-RISK_TS-TEMP_FNO-1 redo01.log redo04.log redo07.log
[oracle@testdb1 testdata]$
5 測試完成后清理
恢復測試完成后,測試sql執行正常,說明備份正常。
可以清理測試庫。
把測試庫關閉,刪除測試庫文件。
如果是在同一個服務器恢復測試,測試完成后,還需要把
spfile文件mv到原來的名字。
cd $ORACLE_HOME/dbs
mv spfile${ORACLE_SID}.ora.def spfile${ORACLE_SID}.ora
6. 排錯
[oracle@testdb1 scripts]$ ./auto_rman_test.sh
/u02/rmanbackup/auto_20250517_c-4230975025-20250517-10.CTP
./auto_rman_test.sh: line 43: /u01/app/oracle/product/11.2.0/dbhome_1/dbs/initctp.ora: No such file or directory
./auto_rman_test.sh: line 50: /u01/app/oracle/product/11.2.0/dbhome_1/bin/sqlplus: No such file or directory
./auto_rman_test.sh: line 64: /u01/app/oracle/product/11.2.0/dbhome_1/bin/rman: No such file or directory
error mount
[oracle@testdb1 scripts]$ env|grep HOMEORACLE_HOME=/u01/app/oracle/product/11.2.0/db_1
[oracle@testdb1 scripts]$
分析: auto_rman_test.sh中$ORACLE_HOME設置錯誤。修正后重新執行正常。
7 部署自動恢復測試
數據庫服務器: crontab設置把rman備份上傳到測試服務器,
測試服務器: crontab設置rman自動恢復。
這樣可以自動測試當天備份,確認數據庫備份是可用的。