Sunday, 31 March 2013

How to recover after mistakenly deleting a database file on Linux?

This can be a life saver; let's say that by mistake, one of the DBAs ( not you, of course), is deleting a datafile. You are under pressure to have the tablespace which contained that datafile back as soon as possible.
Luckily on Linux, you can recover a deleted file, as long as there is still some process attached to the deleted file. And in our case, the DBW is still attached to the file. The process is quite simple, but first of all don't panic and keep the DB up, otherwise the file is gone.

Below is the real test which I've done in one of our databases, together with some embedded comments:



Creating a tablespace and a table on it, dedicated to the test:

SQL> create tablespace test1 datafile '/MYDB/oradata/ora_data00/MYDB_test1_01.dbf' size 200M;

Tablespace created.

SQL> l
  1* create table my_test tablespace test1 as select * from all_objects
        where rownum < 101;

Table created.

SQL> select count(*) from my_test;

  COUNT(*)
----------
       100



SQL> l
  1  select FILE_NAME from dba_data_files
  2* where TABLESPACE_NAME='TEST1'
SQL> /

Datafile Name
----------------------------------------
/MYDB/oradata/ora_data00/MYDB_test1_01
.dbf

Deleting the data file:

rm -f  /MYDB/oradata/ora_data00/MYDB_test1_01.dbf

Identifying the PID for the DBW processes and finding the fd (file descriptor) pointing to our deleted file:


[host1@BMLUS]/u01/app/oracle >ps -fu oracle|grep dbw

oracle   22343     1  0 17:28 ?        00:00:00 ora_dbw0_MYDB
oracle   22347     1  0 17:28 ?        00:00:00 ora_dbw1_MYDB
oracle   22351     1  0 17:28 ?        00:00:00 ora_dbw2_MYDB
oracle   22355     1  0 17:28 ?        00:00:00 ora_dbw3_MYDB



[host1@MYDB]/proc >ll 10139/fd|grep del
lrwx------ 1 oracle dba 64 Mar  4 01:30 9 -> /u01/app/oracle/product/11.1.0/dbs/lkinstMYDB (deleted)
l-wx------ 1 oracle dba 64 Mar  4 01:30 12 -> /u01/app/oracle/admin/MYDB/diag/rdbms/MYDB/MYDB/trace/MYDB_ora_10104.trm (deleted)
l-wx------ 1 oracle dba 64 Mar  4 01:30 11 -> /u01/app/oracle/admin/MYDB/diag/rdbms/MYDB/MYDB/trace/MYDB_ora_10104.trc (deleted)
lrwx------ 1 oracle dba 64 Mar 31 17:16 32 -> /MYDB/oradata/ora_data00/MYDB_test1_01.dbf (deleted)

Now trying to select from the table my_test:


SQL> select count(*) from my_test;

  COUNT(*)
----------
       100


SQL> alter system flush buffer_cache;

System altered.

SQL>  select count(*) from my_test;
 select count(*) from my_test
                      *
ERROR at line 1:
ORA-01116: error in opening database file 9
ORA-01110: data file 9: '/MYDB/oradata/ora_data00/MYDB_test1_01.dbf'
ORA-27041: unable to open file
Linux-x86_64 Error: 2: No such file or directory
Additional information: 3

The 1st magic to do is to create a symbolic link as below, pointing to the file descriptor.


[host1@MYDB]/proc >ln -s /proc/10139/fd/32   /MYDB/oradata/ora_data00/MYDB_test1_01.dbf

SQL> select count(*) from my_test;

  COUNT(*)
----------
       100

Make the tablespace read only:

SQL> alter tablespace test1 read only;

Tablespace altered.

Copy the file to a different name:

cp /MYDB/oradata/ora_data00/MYDB_test1_01.dbf /MYDB/oradata/ora_data00/MYDB_test1_01_recovered.dbf

Shutdown abort:

SQL> shutdown abort
ORACLE instance shut down.
SQL> exit
Disconnected from Oracle Database 11g Enterprise Edition Release 11.1.0.7.0 - 64bit Production
With the Partitioning, OLAP, Data Mining and Real Application Testing options

Delete the symbolic link that we've created and rename the data file to its original name:

[host1@MYDB]/proc >ll /MYDB/oradata/ora_data00/MYDB_test1_01.dbf
lrwxrwxrwx 1 oracle dba 17 Mar 31 17:19 /MYDB/oradata/ora_data00/MYDB_test1_01.dbf -> /proc/10139/fd/32
[host1@MYDB]/proc >rm /MYDB/oradata/ora_data00/MYDB_test1_01.dbf
[host1@MYDB]/proc >mv /MYDB/oradata/ora_data00/MYDB_test1_01_recovered.dbf /MYDB/oradata/ora_data00/MYDB_test1_01.dbf

Startup the DB and make the tablespace read write.

SQL> startup
ORACLE instance started.

Total System Global Area 2689212416 bytes
Fixed Size                  2162640 bytes
Variable Size             570425392 bytes
Database Buffers         2097152000 bytes
Redo Buffers               19472384 bytes
Database mounted.
Database opened.
SQL> alter tablespace test1 read write;

Tablespace altered.


SQL> conn /
Connected.
SQL> select count(*) from my_test;

  COUNT(*)
----------
       100

We are done, back to life, in one piece :-)

Thursday, 28 March 2013

What was the password used to create db link?

You have a database link in one of your Oracle 11g database accounts and you want to create the exact db link in a different account. The only issue is that the user password is blank in user_db_links, due to security reasons. So how to find out the password and to "duplicate" this db link?
In reality, you don't even need to know the password, you can query sys.link$ and use "identified by values" when creating the db link:


SQL> conn / as sysdba
Connected.
SQL> select NAME,HOST,userid,PASSWORDX
from sys.link$ where name='TEST1';


SQL> /

NAME                 HOST       USERID                         PASSWORDX
-------------------- ---------- ------------------------------ --------------------------------------------------------------------------------
TEST1                MYDB     FLORIN                        05177093C45ABC2976294C0780B2B5873E3CE6D80F94D9A314

So all is left is to connect and create the db link:


SQL> conn florin2/florin2
Connected.
SQL> create database link test1 connect to florin identified by values '05177093C45ABC2976294C0780B2B5873E3CE6D80F94D9A314' using 'MYDB';


Database link created.


SQL> select * from dual@test1;

D
-
X





Removing '\par' from text file, on Linux

You've just transferred a text file from Windows to Linux or HP, just to realize it has some strange characters inside. Most of the time, "dos2ux" will do the job, but not in this case. The '\par' is originating from an enriched text file on Windows.
In this case, sed Unix utility is coming to the rescue:



sed 's#\\par$##' My_Windows_file.txt

For more details, you may visit the link below: