Thursday, December 23, 2010

Post 25 of series,Configure discard file,exception table and stored procedure in goldengate replication with Tutorial

For the Gurmukhs, the Way is obvious. At the Lord's Door, they face no obstructions. They praise the Lord's Name, they keep the Naam in their minds, and they remain attached to the Love of the Naam. The Unstruck Celestial Music vibrates for them at the Lord's Door, and they are honored at the True Door.


Index to all the posts of Gurpartap Singh's Blog


Lets configure discardfile and then convert to Exception table and then will use a stored procedure from within goldengate replication. It depends what you want to use but I like exception table as it gives output in a oracle tabular form and so easy to isolate particular set
of sqls by just using simple sql commands.


Lets create simple tables and data to do this type of replication.

create table employee (
emp_id number,
firstname varchar2(20),
lastname varchar2(20)
);

alter table employee add constraint PK_employee primary key (emp_id);

create table location (
location_id number,
province varchar2(20),
city varchar2(20)
);

alter table location add constraint pk_location primary key (location_id);

Add to to goldengate replication use following:

DBLOGIN USERID goldengate, PASSWORD goldengate
ADD TRANDATA goldengate.employee
ADD TRANDATA goldengate.location


ADD EXTRACT etest, TRANLOG, BEGIN NOW
ADD EXTTRAIL /u01/app/oracle/product/goldengate/dirdat/test/t1, EXTRACT etest


EDIT PARAMS etest


EXTRACT etest
USERID goldengate@simarsv1, PASSWORD AACAAAAAAAAAAAKAPATACEHBIGQGCFZCCDIGAEMCQFFBZHVC, ENCRYPTKEY default
RMTHOST goldengate, MGRPORT 7909
EXTTRAIL /u01/app/oracle/product/goldengate/dirdat/test/t1
TRANLOGOPTIONS ASMUSER sys@RAC1ASM, ASMPASSWORD AACAAAAAAAAAAAJASAOBIHRGNFGAYCKBHJPAGIXFGGBJUAXI, ENCRYPTKEY default
TABLE gurpartap.employee;
TABLE gurpartap.location;



ptest
ADD EXTRACT ptest, EXTTRAILSOURCE /u01/app/oracle/product/goldengate/dirdat/test/p1

ADD RMTTRAIL /u01/app/oracle/product/goldengate/dirdat/ts, EXTRACT ptest

EDIT PARAMS ptest

EXTRACT ptest
RMTHOST goldengate, MGRPORT 7909
RMTTRAIL /u01/app/oracle/product/goldengate/dirdat/ts
PASSTHRU
TABLE gurpartap.employee;
TABLE gurpartap.location;




In logs its:


SQL> conn goldengate/goldengate
Connected.
SQL> create table employee (
emp_id number,
firstname varchar2(20),
lastname varchar2(20)
);
2 3 4 5
Table created.

SQL> alter table employee add constraint PK_employee primary key (emp_id);

Table altered.

SQL> create table location (
location_id number,
province varchar2(20),
city varchar2(20)
); 2 3 4 5

Table created.

SQL> alter table location add constraint pk_location primary key (location_id);

Table altered.

SQL>




GGSCI (rac1.rac.meditate.com) 7> DBLOGIN USERID goldengate, PASSWORD goldengate
Successfully logged into database.


GGSCI (rac1.rac.meditate.com) 10> ADD TRANDATA goldengate.employee

Logging of supplemental redo data enabled for table GOLDENGATE.EMPLOYEE.

GGSCI (rac1.rac.meditate.com) 11> ADD TRANDATA GOLDENGATE.location

Logging of supplemental redo data enabled for table GOLDENGATE.LOCATION.

GGSCI (rac1.rac.meditate.com) 12>



--------

GGSCI (rac1.rac.meditate.com) 1> ADD EXTRACT etest, TRANLOG, BEGIN NOW
EXTRACT added.


GGSCI (rac1.rac.meditate.com) 2> ADD EXTTRAIL /u01/app/oracle/product/goldengate/dirdat/pl, EXTRACT etest
EXTTRAIL added.


GGSCI (rac1.rac.meditate.com) 3> EDIT PARAMS etest

EXTRACT etest
SETENV (ORACLE_SID=SIMAR1)
USERID goldengate@simarsv1, PASSWORD AACAAAAAAAAAAAKAPATACEHBIGQGCFZCCDIGAEMCQFFBZHVC, ENCRYPTKEY default
TRANLOGOPTIONS ASMUSER sys@RAC1ASM, ASMPASSWORD AACAAAAAAAAAAAJASAOBIHRGNFGAYCKBHJPAGIXFGGBJUAXI, ENCRYPTKEY default
RMTHOST goldengate, MGRPORT 7909
EXTTRAIL /u01/app/oracle/product/goldengate/dirdat/test/t1
TABLE goldengate.employee;
TABLE goldengate.location;


GGSCI (rac1.rac.meditate.com) 4> ADD EXTRACT ptest, EXTTRAILSOURCE /u01/app/oracle/product/goldengate/dirdat/test/t1
EXTRACT added.


GGSCI (rac1.rac.meditate.com) 5> ADD RMTTRAIL /u01/app/oracle/product/goldengate/dirdat/ts, EXTRACT ptest
RMTTRAIL added.


GGSCI (rac1.rac.meditate.com) 6> EDIT PARAMS ptest



EXTRACT ptest
RMTHOST goldengate, MGRPORT 7909
RMTTRAIL /u01/app/oracle/product/goldengate/dirdat/ts
PASSTHRU
TABLE goldengate.employee;
TABLE goldengate.location;



Now lets start the extracts as:


GGSCI (rac1.rac.meditate.com) 4> start extract etest

Sending START request to MANAGER ...
EXTRACT ETEST starting


GGSCI (rac1.rac.meditate.com) 5> start extract ptest

Sending START request to MANAGER ...
EXTRACT PTEST starting


GGSCI (rac1.rac.meditate.com) 6>



Now lets create the definition file as:


cd /u01/app/oracle/product/goldengate/dirprm
vi testplace.prm

DEFSFILE /u01/app/oracle/product/goldengate/dirdat/test
USERID goldengate@simarsv1, PASSWORD AACAAAAAAAAAAALAJAVCJBLGSBEABDQIAAVAUCVJUHDGSFAJ, ENCRYPTKEY default
TABLE goldengate.employee;
TABLE goldengate.location;

cd /u01/app/oracle/product/goldengate
./defgen paramfile dirprm/testplace.prm


Output is as:

oracle : rac1.rac.meditate.com : @simar1 : /u01/app/oracle/product/goldengate
$ cd /u01/app/oracle/product/goldengate/dirprm

oracle : rac1.rac.meditate.com : @simar1 : /u01/app/oracle/product/goldengate/dirprm
$ vi testplace.prm

oracle : rac1.rac.meditate.com : @simar1 : /u01/app/oracle/product/goldengate/dirprm
$ cd /u01/app/oracle/product/goldengate

oracle : rac1.rac.meditate.com : @simar1 : /u01/app/oracle/product/goldengate
$ ./defgen paramfile dirprm/testplace.prm

***********************************************************************
Oracle GoldenGate Table Definition Generator for Oracle
Version 10.4.0.19 Build 002
Linux, x64, 64bit (optimized), Oracle 11 on Sep 18 2009 00:09:13

Copyright (C) 1995, 2009, Oracle and/or its affiliates. All rights reserved.


Starting at 2010-12-13 23:54:16
***********************************************************************

Operating System Version:
Linux
Version #1 SMP Fri Oct 29 14:21:16 EDT 2010, Release 2.6.18-194.26.1.el5
Node: rac1.rac.meditate.com
Machine: x86_64
soft limit hard limit
Address Space Size : unlimited unlimited
Heap Size : unlimited unlimited
File Size : unlimited unlimited
CPU Time : unlimited unlimited

Process id: 9422

***********************************************************************
** Running with the following parameters **
***********************************************************************
DEFSFILE /u01/app/oracle/product/goldengate/dirdat/test
USERID goldengate@simarsv1, PASSWORD ************************************************* ENCRYPTKEY default
TABLE goldengate.employee;
Retrieving definition for GOLDENGATE.EMPLOYEE
TABLE goldengate.location;
Retrieving definition for GOLDENGATE.LOCATION



Definitions generated for 2 tables in /u01/app/oracle/product/goldengate/dirdat/dirdat/test


oracle : rac1.rac.meditate.com : @simar1 : /u01/app/oracle/product/goldengate
$

Now scp it to the target as:

scp /u01/app/oracle/product/goldengate/dirdat/dirdat/test goldengate:/u01/app/oracle/product/goldengate/dirdat/dirdat/test


Output is as:

oracle : rac1.rac.meditate.com : @simar1 : /u01/app/oracle/product/goldengate
$ scp /u01/app/oracle/product/goldengate/dirdat/dirdat/test goldengate:/u01/app/oracle/product/goldengate/dirdat/dirdat/test

Password:
test 100% 1237 1.2KB/s 00:00

oracle : rac1.rac.meditate.com : @simar1 : /u01/app/oracle/product/goldengate
$

---------------------------------------------------



Now in target lets create the data as:


create table remployee (
emp_id number,
firstname varchar2(20),
lastname varchar2(20)
);

alter table remployee add constraint PK_remployee primary key (emp_id);

create table rlocation (
location_id number,
province varchar2(20),
city varchar2(20)
);

alter table rlocation add constraint pk_rlocation primary key (location_id);


Output is as:

SQL> conn goldengate/goldengate
Connected.
SQL> create table remployee (
emp_id number,
firstname varchar2(20),
lastname varchar2(20)
);
2 3 4 5
Table created.

SQL> alter table remployee add constraint PK_remployee primary key (emp_id);

Table altered.

SQL> create table rlocation (
location_id number,
province varchar2(20),
city varchar2(20)
); 2 3 4 5

Table created.

SQL> alter table rlocation add constraint pk_rlocation primary key (location_id);


Table altered.

SQL> SQL>





Now Goldengate stuff is as:


ADD REPLICAT rtest, EXTTRAIL /u01/app/oracle/product/goldengate/dirdat/ts




EDIT PARAMS rtest

REPLICAT rtest
SOURCEDEFS /u01/app/oracle/product/goldengate/dirdat/test
DISCARDFILE /u01/app/oracle/product/goldengate/dirdat/test, purge
DISCARDROLLOVER ON sunday
USERID goldengate, PASSWORD AACAAAAAAAAAAAKAPATACEHBIGQGCFZCCDIGAEMCQFFBZHVC, ENCRYPTKEY default
REPERROR (DEFAULT, DISCARD)
GETINSERTS
GETUPDATES
GETDELETES
MAP goldengate.employee, TARGET goldengate.remployee;
MAP goldengate.location, TARGET goldengate.rlocation;



Output is as:
GGSCI (goldengate) 1> ADD REPLICAT rtest, EXTTRAIL /u01/app/oracle/product/goldengate/dirdat/ts
REPLICAT added.


GGSCI (sedm1610) 2>
GGSCI (sedm1610) 4> start replicat RTEST

Sending START request to MANAGER ...
REPLICAT RTEST starting


GGSCI (sedm1610) 5>




TEST 1:
On source:
select * from employee;
select * from location;
insert into employee values (1,'GURPARTAP','SINGH');
insert into location values (1,'BRITISH COLUMBIA','WHISTLER');
commit;
select * from employee;
select * from location;

On target:
These will be replicated as:
select * from remployee;
select * from rlocation;


Results as expected:

On source:

SQL> select * from employee;
select * from location;
no rows selected

SQL>

no rows selected

SQL>
SQL> insert into employee values (1,'GURPARTAP','SINGH');

1 row created.

SQL> insert into location values (1,'BRITISH COLUMBIA','WHISTLER');

1 row created.

SQL> commit:
2
SQL> commit;

Commit complete.

SQL> select * from employee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH

1 row selected.

SQL> select * from location;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER

1 row selected.

SQL>


On target:

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH

1 row selected.

SQL> select * from rlocation;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER

1 row selected.

SQL>




TEST2:
On target:
select * from remployee;
insert into remployee values (2,'SIMAR','SINGH');
commit;
select * from remployee;

On source:
select * from employee;
select * from location;
insert into employee values (2,'SIMAR','SINGH');
insert into location values (2,'BRITISH COLUMBIA','VANCOUVER');
commit;
select * from employee;
select * from location;


On target:
These will be replicated as:
select * from remployee;
select * from rlocation;


Results are not as expected:

on target:

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH

1 row selected.

SQL> insert into remployee values (2,'SIMAR','SINGH');

1 row created.

SQL> commit;

Commit complete.

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
2 SIMAR SINGH
1 GURPARTAP SINGH

2 rows selected.

SQL>

On source:

SQL> insert into employee values (2,'SIMAR','SINGH');

1 row created.

SQL> insert into location values (2,'BRITISH COLUMBIA','VANCOUVER');

1 row created.

SQL> commit;

Commit complete.

SQL> select * from employee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH
2 SIMAR SINGH

2 rows selected.

SQL> select * from location;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER
2 BRITISH COLUMBIA VANCOUVER

2 rows selected.

SQL>


On target:

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
2 SIMAR SINGH
1 GURPARTAP SINGH

2 rows selected.

SQL> select * from rlocation;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER
2 BRITISH COLUMBIA VANCOUVER

2 rows selected.

SQL>


The discard file shows:

oracle : goldengate : @goldengate : /u01/app/oracle/product/goldengate
$ cat /u01/app/oracle/product/goldengate/dirdat/distest
Oracle GoldenGate Delivery for Oracle process started, group RTEST discard file opened: 2010-12-14 20:19:22

Current time: 2010-12-14 20:28:47

OCI Error ORA-00001: unique constraint (GOLDENGATE.PK_REMPLOYEE) violated (status = 1), SQL
Operation failed at seqno 2 rba 971
Discarding record on action DISCARD on error 1
Problem replicating GOLDENGATE.EMPLOYEE to GOLDENGATE.REMPLOYEE
Error (1) occurred with insert record (target format)...
*
EMP_ID = 2
FIRSTNAME = SIMAR
LASTNAME = SINGH
*

oracle : goldengate : @goldengate : /u01/app/oracle/product/goldengate
$



Note that even though we inserted 2 rows in two different tables in source in one single transaction but on target the transaction doesn't fail even though we have one of the row already existing, Goldengate inserts the other row in the other table and writes the existing row
to the discard file.





------------------------------------------------------

Now lets use the exception table instead of discard file as:

-- handle exception
MAP goldengate.employee, TARGET goldengate.employee_exception, EXCEPTIONSONLY, &
INSERTALLRECORDS , &
COLMAP (USEDEFAULTS, &
SOURCE_TABLE = @GETENV ("GGHEADER", "TABLENAME"), &
TARGET_TABLE = "GOLDENGATE.REMPLOYEE", &
DML_DATE = @GETENV ("GGHEADER", "COMMITTIMESTAMP"), &
OPTYPE = @GETENV ("GGHEADER", "OPTYPE"), &
DBERRNUM = @GETENV("LASTERR", "DBERRNUM"), &
DBERRMSG = @GETENV("LASTERR", "DBERRMSG"));

-- handle exception
MAP GOLDENGATE.location, TARGET GOLDENGATE.location_exception, EXCEPTIONSONLY, &
INSERTALLRECORDS , &
COLMAP (USEDEFAULTS, &
SOURCE_TABLE = @GETENV ("GGHEADER", "TABLENAME"), &
TARGET_TABLE = "GOLDENGATE.RLOCATION", &
DML_DATE = @GETENV ("GGHEADER", "COMMITTIMESTAMP"), &
OPTYPE = @GETENV ("GGHEADER", "OPTYPE"), &
DBERRNUM = @GETENV("LASTERR", "DBERRNUM"), &
DBERRMSG = @GETENV("LASTERR", "DBERRMSG"));


So, new parameter file looks like:

oracle : goldengate : @goldengate : /u01/app/oracle/product/goldengate/dirprm
$ cat rtest.prm
REPLICAT rtest
SOURCEDEFS /u01/app/oracle/product/goldengate/dirdat/test
DISCARDFILE /u01/app/oracle/product/goldengate/dirdat/distest, purge
DISCARDROLLOVER ON sunday
--- Please note here goldengate is the name of the target machine, target database and taget database user_id.
USERID goldengate@goldengate, PASSWORD AACAAAAAAAAAAAKAPATACEHBIGQGCFZCCDIGAEMCQFFBZHVC, ENCRYPTKEY default
REPERROR (DEFAULT, EXCEPTION)
GETINSERTS
GETUPDATES
GETDELETES
MAP gurpartap.employee, TARGET gurpartap.remployee;
-- handle exception
MAP gurpartap.employee, TARGET gurpartap.employee_exception, EXCEPTIONSONLY, &
INSERTALLRECORDS , &
COLMAP (USEDEFAULTS, &
SOURCE_TABLE = @GETENV ("GGHEADER", "TABLENAME"), &
TARGET_TABLE = "GURPARTAP.REMPLOYEE", &
DML_DATE = @GETENV ("GGHEADER", "COMMITTIMESTAMP"), &
OPTYPE = @GETENV ("GGHEADER", "OPTYPE"), &
DBERRNUM = @GETENV("LASTERR", "DBERRNUM"), &
DBERRMSG = @GETENV("LASTERR", "DBERRMSG"));
-- new
REPERROR (DEFAULT, EXCEPTION)
GETINSERTS
GETUPDATES
GETDELETES
MAP gurpartap.location, TARGET gurpartap.rlocation;
-- handle exception
MAP gurpartap.location, TARGET gurpartap.location_exception, EXCEPTIONSONLY, &
INSERTALLRECORDS , &
COLMAP (USEDEFAULTS, &
SOURCE_TABLE = @GETENV ("GGHEADER", "TABLENAME"), &
TARGET_TABLE = "GURPARTAP.RLOCATION", &
DML_DATE = @GETENV ("GGHEADER", "COMMITTIMESTAMP"), &
OPTYPE = @GETENV ("GGHEADER", "OPTYPE"), &
DBERRNUM = @GETENV("LASTERR", "DBERRNUM"), &
DBERRMSG = @GETENV("LASTERR", "DBERRMSG"));

and we need to create the exception tables as:

create table GURPARTAP.LOCATION_EXCEPTION (
LOCATION_ID number,
PROVINCE varchar2(20),
CITY varchar2(20),
SOURCE_TABLE varchar2(100),
TARGET_TABLE varchar2(100),
DML_DATE date,
OPTYPE varchar2(20),
DBERRNUM number,
DBERRMSG VARCHAR2(4000)
);



create table GURPARTAP.EMPLOYEE_EXCEPTION (
emp_id number,
firstname varchar2(20),
lastname varchar2(20),
SOURCE_TABLE varchar2(100),
TARGET_TABLE varchar2(100),
DML_DATE date,
OPTYPE varchar2(20),
DBERRNUM number,
DBERRMSG VARCHAR2(4000)
);

Output is as:

SQL> show user
USER is "GURPARTAP"
SQL> create table GURPARTAP.EMPLOYEE_EXCEPTION (
2 emp_id number,
3 firstname varchar2(20),
4 lastname varchar2(20),
5 SOURCE_TABLE varchar2(100),
6 TARGET_TABLE varchar2(100),
7 DML_DATE date,
8 OPTYPE varchar2(20),
9 DBERRNUM number,
10 DBERRMSG VARCHAR2(4000)
11 );

Table created.

SQL> desc GURPARTAP.rlocation
Name Null? Type
----------------------------------------------------------------- -------- --------------------------------------------
LOCATION_ID NOT NULL NUMBER
PROVINCE VARCHAR2(20)
CITY VARCHAR2(20)

SQL> create table GURPARTAP.LOCATION_EXCEPTION (
2 LOCATION_ID number,
3 PROVINCE varchar2(20),
4 CITY varchar2(20),
5 SOURCE_TABLE varchar2(100),
6 TARGET_TABLE varchar2(100),
7 DML_DATE date,
8 OPTYPE varchar2(20),
9 DBERRNUM number,
10 DBERRMSG VARCHAR2(4000)
11 );

Table created.

SQL> select * from rlocation;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER

1 row selected.

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH

1 row selected.

SQL>






Lets repeat test 2 again:

Before that I stopped both the extract and the replicat and truncated all the 4 tables i.e
gurpartap.employee,gurpartap.location in source and gurpartap.rlocation,gurpartap.remployee in target and restarted the extracts and
replicates with begin now option as:
alter extract etest,begin now
alter extract ptest,begin now
start extract etest
start extract ptest
alter replicat rtest,begin now
start replicat rtest





TEST2:
On target:
select * from remployee;
insert into remployee values (2,'SIMAR','SINGH');
commit;
select * from remployee;

On source:
select * from employee;
select * from location;
insert into employee values (2,'SIMAR','SINGH');
insert into location values (2,'BRITISH COLUMBIA','VANCOUVER');
commit;
select * from employee;
select * from location;


On target:
These will be replicated as:
select * from remployee;
select * from rlocation;

Results are same as earlier:
Even though one row failed and written to the exception table,
it replicated the other row. Both the rows were inserted in the same
transaction on source.



Here is the output:

On target:
SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH

1 row selected.

SQL> insert into remployee values (2,'SIMAR','SINGH');

1 row created.

SQL> commit;

Commit complete.

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH
2 SIMAR SINGH

2 rows selected.

SQL>


On source:
SQL> insert into employee values (2,'SIMAR','SINGH');

1 row created.

SQL> insert into location values (2,'BRITISH COLUMBIA','VANCOUVER');

1 row created.

SQL> commit;

Commit complete.

SQL> select * from employee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH
2 SIMAR SINGH

2 rows selected.

SQL> select * from location;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER
2 BRITISH COLUMBIA VANCOUVER

2 rows selected.

SQL>


On target:


SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH
2 SIMAR SINGH

2 rows selected.

SQL> select * from rlocation;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER
2 BRITISH COLUMBIA VANCOUVER

2 rows selected.


And it wrote that failed entry in the exceptions table as:



SQL> select * from GURPARTAP.EMPLOYEE_EXCEPTION ;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
SOURCE_TABLE
----------------------------------------------------------------------------------------------------
TARGET_TABLE DML_DATE
---------------------------------------------------------------------------------------------------- ---------
OPTYPE DBERRNUM
-------------------- ----------
DBERRMSG
------------------------------------------------------------------------------------------------------------------------
2 SIMAR SINGH
GURPARTAP.EMPLOYEE
GURPARTAP.REMPLOYEE 15-DEC-10
INSERT 1
OCI Error ORA-00001: unique constraint (GURPARTAP.PK_REMPLOYEE) violated (status = 1), SQL


1 row selected.

SQL>




Now we will create a stored procedure to let it insert in another table i.e. the third table in the tagert.

I will create a third table in the target now as:

create table start_date (
emp_id number,
start_date date
);


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

CREATE OR REPLACE PROCEDURE ckeck_insert (emp_id NUMBER) AS
BEGIN
insert into start_date values (emp_id,sysdate);
commit;
END ckeck_insert;
/


Then to test if its working use:

CALL ckeck_insert (3);



Output is as:

SQL> create table start_date (
emp_id number,
start_date date
);

2 3 4
Table created.

SQL> SQL> show user
USER is "GURPARTAP"
SQL> CREATE OR REPLACE PROCEDURE ckeck_insert (emp_id NUMBER) AS
BEGIN
insert into start_date values (emp_id,sysdate);
commit;
END ckeck_insert;
/
2 3 4 5 6
Procedure created.

SQL> CALL ckeck_insert (3);

Call completed.

SQL> select * from start_date;

EMP_ID START_DAT
---------- ---------
3 15-DEC-10

1 row selected.

SQL>


Now, I will truncate this table.

SQL> truncate table start_date;

Table truncated.

SQL> select * from start_date;

no rows selected

SQL>


Please note the following in below parameter file:

sqlexec (spname gurpartap.ckeck_insert , id ckeck_insert1, &
params(EMP_ID=@IF (@COLTEST(emp_id, MISSING), @COLSTAT(NULL), emp_id)), ERROR RAISE, TRACE ALL);

spname -> is the name of the procedure you want to call.
id ckeck_insert1 -> is required of you are calling this procedure more than one time andin that case id should be differrent in every case like ckeck_insert2 for second.
params -> here we are passing paramter to the procedure and name of teh parameter is emp_id. Please note the name of teh parameter being passed here should be same as the variable name defined in the procedure. Other option is COLTEST and i.e. if the emp_id value is missing then take the null value otherwise take the value of emp_id.


Now I will update the replicat parameter file as:

REPLICAT rtest
SOURCEDEFS /u01/app/oracle/product/goldengate/dirdat/test
DISCARDFILE /u01/app/oracle/product/goldengate/dirdat/distest, purge
DISCARDROLLOVER ON sunday
--- Please note here goldengate is teh name of the target machine, target database and taget database user_id.
USERID goldengate@goldengate, PASSWORD AACAAAAAAAAAAAKAPATACEHBIGQGCFZCCDIGAEMCQFFBZHVC, ENCRYPTKEY default
REPERROR (DEFAULT, EXCEPTION)
GETINSERTS
GETUPDATES
GETDELETES
MAP gurpartap.employee, TARGET gurpartap.remployee, &
sqlexec (spname gurpartap.ckeck_insert , id ckeck_insert1, &
params(EMP_ID=@IF (@COLTEST(emp_id, MISSING), @COLSTAT(NULL), emp_id)), ERROR RAISE, TRACE ALL);
-- handle exception
MAP gurpartap.employee, TARGET gurpartap.employee_exception, EXCEPTIONSONLY, &
INSERTALLRECORDS , &
COLMAP (USEDEFAULTS, &
SOURCE_TABLE = @GETENV ("GGHEADER", "TABLENAME"), &
TARGET_TABLE = "GURPARTAP.REMPLOYEE", &
DML_DATE = @GETENV ("GGHEADER", "COMMITTIMESTAMP"), &
OPTYPE = @GETENV ("GGHEADER", "OPTYPE"), &
DBERRNUM = @GETENV("LASTERR", "DBERRNUM"), &
DBERRMSG = @GETENV("LASTERR", "DBERRMSG"));
-- new
REPERROR (DEFAULT, EXCEPTION)
GETINSERTS
GETUPDATES
GETDELETES
MAP gurpartap.location, TARGET gurpartap.rlocation;
-- handle exception
MAP gurpartap.location, TARGET gurpartap.location_exception, EXCEPTIONSONLY, &
INSERTALLRECORDS , &
COLMAP (USEDEFAULTS, &
SOURCE_TABLE = @GETENV ("GGHEADER", "TABLENAME"), &
TARGET_TABLE = "GURPARTAP.RLOCATION", &
DML_DATE = @GETENV ("GGHEADER", "COMMITTIMESTAMP"), &
OPTYPE = @GETENV ("GGHEADER", "OPTYPE"), &
DBERRNUM = @GETENV("LASTERR", "DBERRNUM"), &
DBERRMSG = @GETENV("LASTERR", "DBERRMSG"));




TEST1:




Lets repeat test 2 again:

Before that I stopped both the extract and the replicat and truncated all the 4 tables i.e
gurpartap.employee,gurpartap.location in source and gurpartap.rlocation,gurpartap.remployee in taeget and restarted the extracts and
replicats with begin now option as:
alter extract etest,begin now
alter extract ptest,begin now
start extract etest
start extract ptest
alter replicat rtest,begin now
start replicat rtest





TEST2:
On target:
select * from remployee;
insert into remployee values (2,'GURPARTAP','SINGH');
commit;
select * from remployee;

On source:
select * from employee;
select * from location;
insert into employee values (2,'SIMAR','SINGH');
insert into location values (2,'BRITISH COLUMBIA','VANCOUVER');
commit;
select * from employee;
select * from location;


On target:
These will be replicated as:
select * from remployee;
select * from rlocation;




Results are :
Even though one row failed and written to the exception table,
it replicated the other row. Both the rows were inserted in the same
transaction on source.



Here is the output:

On target :

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH

1 row selected.

SQL> insert into remployee values (2,'SIMAR','SINGH');

1 row created.

SQL> commit;

Commit complete.

SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH
2 SIMAR SINGH

2 rows selected.

SQL>


On source:

SQL> insert into employee values (2,'SIMAR','SINGH');

1 row created.

SQL> insert into location values (2,'BRITISH COLUMBIA','VANCOUVER');

1 row created.

SQL> commit;

Commit complete.

SQL> select * from employee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH
2 SIMAR SINGH

2 rows selected.

SQL> select * from location;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER
2 BRITISH COLUMBIA VANCOUVER

2 rows selected.

SQL>


On target:


SQL> select * from remployee;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
1 GURPARTAP SINGH
2 SIMAR SINGH

2 rows selected.

SQL> select * from rlocation;

LOCATION_ID PROVINCE CITY
----------- -------------------- --------------------
1 BRITISH COLUMBIA WHISTLER
2 BRITISH COLUMBIA VANCOUVER

2 rows selected.


And it wrote that failed entry in the exceptions table as:



SQL> select * from GURPARTAP.EMPLOYEE_EXCEPTION ;

EMP_ID FIRSTNAME LASTNAME
---------- -------------------- --------------------
SOURCE_TABLE
----------------------------------------------------------------------------------------------------
TARGET_TABLE DML_DATE
---------------------------------------------------------------------------------------------------- ---------
OPTYPE DBERRNUM
-------------------- ----------
DBERRMSG
------------------------------------------------------------------------------------------------------------------------
2 SIMAR SINGH
GURPARTAP.EMPLOYEE
GURPARTAP.REMPLOYEE 15-DEC-10
INSERT 1
OCI Error ORA-00001: unique constraint (GURPARTAP.PK_REMPLOYEE) violated (status = 1), SQL


1 row selected.

SQL>





SQL> select * from START_DATE;

EMP_ID START_DAT
---------- ---------
1 15-DEC-10
2 15-DEC-10

2 rows selected.

SQL>

Next we will create the primary key index on the table where procedure is inserting i.e. START_DATE and see how the inserts and updates will behave.

O Siblings of Destiny, one who becomes Gurmukh and understands this is very rare. To act without understanding is to lose the treasure of this human life. 1 Pause Those who have tasted it, enjoy its flavor; without tasting it, they wander in doubt, lost and deceived. The True
Name is the Ambrosial Nectar; no one can describe it. Drinking it in, one becomes honorable, absorbed in the Perfect Word of the Shabad.




Saturday, December 4, 2010

Post 24 of series, Goldengate initial load using Cross platform transportable tablespaces Solaris to Linux

Upon that cosmic plate of the sky, the sun and the moon are the lamps. The stars and their orbs are the studded pearls. The
fragrance of sandalwood in the air is the temple incense, and the wind is the fan. All the plants of the world are the altar flowers in offering to You, O Luminous Lord. 1 What a beautiful Aartee, lamp-lit worship service this is! O Destroyer of Fear, this is Your Ceremony of Light. The Unstruck Sound-current of the Shabad is the vibration of the temple drums.



Index of all the posts of Gurpartap Singh's Blog


In this post we will do initial load using cross platform transportable tablespaces. Our source is Solaris 64 bit with big endian format and
target is Linux x86 64bit with little endian format. We will use package "DBMS_FILE_TRANSFER" to transfer files from ASM to ASM from
source to target. Incase it would have been a same eindian format you don't need extra space to place datafiles before transfer. In this case we will
need to do conversion from big endian format to little endian format so we will need extra space to place converted files. Database versions on source and target are 10.2.0.4. On 11g you can use 'cp' command to transfer files from ASM to ASM from source to target.

When you convert files it will not effect the orignal files and will create new files after conversion.
I am using note: 371556.1

Lets do it now.

Create a single instance databae on source

Server:goldengate
Database name:DFSRDV

Create a test rac instance on target
Server:rac1/rac2/tac2
Database name: DFTGDV
Instance Names: DFTGDV1 (on rac1)
Instance Names: DFTGDV2 (on rac2)
Instance Names: DFTGDV3 (on rac3)


Following is the tnsentry for the target database:

DFTGDV =
(DESCRIPTION =
(ADDRESS_LIST =
(ADDRESS = (PROTOCOL = TCP)(HOST = rac1-vip)(PORT = 41521))
(ADDRESS = (PROTOCOL = TCP)(HOST = rac2-vip)(PORT = 41521))
(ADDRESS = (PROTOCOL = TCP)(HOST = rac3-vip)(PORT = 41521))
)
(CONNECT_DATA =
(SERVICE_NAME = DFTGDV)
)
)


Following is the tnsentry for the source database:

DFSRDV =
(DESCRIPTION =
(ADDRESS_LIST =
(ADDRESS = (PROTOCOL = TCP)(HOST = goldengate)(PORT = 41521))
)
(CONNECT_DATA =
(SID = DFSRDV)
(SERVER = DEDICATED)
)
)



Create a new tablespace in the source database for testing and then create a schema to use this tablespace as
a primay tablspace and add anoter 2 datafiles to it and then add some tables to this user. I will just give dba to this user
instead of doing grants one by one.


Create tablespace:
CREATE TABLESPACE migration DATAFILE '+ORADATA1_DG' SIZE 200M;

Create user:
create user migrate identified by migrate default tablespace migration temporary tablespace temp;
grant connect,resource,dba to migrate;

Alter tablespace:
alter tablespace migration add datafile '+ORADATA1_DG' SIZE 200M;
alter tablespace migration add datafile '+ORADATA1_DG' SIZE 200M;



Output is as:


oracle : goldengate : @DFSRDV : /home/oracle
$ sqlplus / as sysdba

SQL*Plus: Release 10.2.0.4.0 - Production on Fri Dec 3 18:59:54 2010

Copyright (c) 1982, 2007, Oracle. All Rights Reserved.


Connected to:
Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

SQL> CREATE TABLESPACE migration DATAFILE '+ORADATA1_DG' SIZE 200M;

Tablespace created.

SQL> create user migrate identified by migrate default tablespace migration temporary tablespace temp;

User created.

SQL> grant connect,resource,dba to migrate;

Grant succeeded.

SQL> alter tablespace migration add datafile '+ORADATA1_DG' SIZE 200M;

Tablespace altered.

SQL> alter tablespace migration add datafile '+ORADATA1_DG' SIZE 200M;

Tablespace altered.

SQL>






Now we should have at least 3 datafiles in this tablespace and this being the deault tablespace of the user:

col FILE_NAME format a90
col TABLESPACE_NAME format a20
select FILE_NAME,TABLESPACE_NAME from dba_data_files where TABLESPACE_NAME='MIGRATION';



Output is as:

SQL> col FILE_NAME format a90
col TABLESPACE_NAME format a20
select FILE_NAME,TABLESPACE_NAME from dba_data_files where TABLESPACE_NAME='MIGRATION';
SQL> SQL>
FILE_NAME TABLESPACE_NAME
------------------------------------------------------------------------------------- --------------------
+ORADATA1_DG/dfsrdv/datafile/migration.309.736802539 MIGRATION
+ORADATA1_DG/dfsrdv/datafile/migration.310.736802721 MIGRATION
+ORADATA1_DG/dfsrdv/datafile/migration.311.736802729 MIGRATION

SQL>




Now lets login as user migration and create some tables and data in there before we migrate:

conn migrate/migrate
create table transfer as select * from dba_tables;
select count(*) from transfer;

create table places as select * from dba_objects;
select count(*) from places;

Then lets verify that the tables are in this tablespce:

select table_name,tablespace_name from user_tables;

Here is the output:


SQL> conn migrate/migrate
Connected.
SQL> show user
USER is "MIGRATE"
SQL> create table transfer as select * from dba_tables;

Table created.

SQL> select count(*) from transfer;

COUNT(*)
----------
1589

SQL> create table places as select * from dba_objects;

Table created.

SQL> select count(*) from places;

COUNT(*)
----------
50602

SQL> commit;

Commit complete.

SQL> select table_name,tablespace_name from user_tables;

TABLE_NAME TABLESPACE_NAME
------------------------------ --------------------
PLACES MIGRATION
TRANSFER MIGRATION

SQL>


Now we are ready for the migration:
We will use ASM to ASM direct transfer using package DBMS_FILE_TRANSFER and since we are going to do cross platform migration with different
endian format i.e. [Solaris[tm] OE (64-bit)] to Linux [Linux x86 64-bit], we will have to use RMAN convert to to convert the endian format
from big to little as [Solaris[tm] OE (64-bit)] is big endian format and [Linux x86 64-bit] is small endian format.
Remeber that you need to read the document to see the limitations of using the transportable tablespaces, here I will discuss only few.

So what to use, exp/imp or expdp or impdp


Pre-steps:


From documentation we have:

Beginning with Oracle Database 10g Release 2, you can transport tablespaces
that contain XMLTypes, but you must use the IMP and EXP utilities,
not Data Pump. When using EXP, ensure that the CONSTRAINTS and TRIGGERS
parameters are set to Y (the default).

The following query returns a list of tablespaces that contain XMLTypes:

select distinct p.tablespace_name
from dba_tablespaces p, dba_xml_tables x, dba_users u, all_all_tables t
where t.table_name=x.table_name and
t.tablespace_name=p.tablespace_name and
x.owner=u.username;

Output is as:

SQL> select distinct p.tablespace_name
from dba_tablespaces p, dba_xml_tables x, dba_users u, all_all_tables t
where t.table_name=x.table_name and
t.tablespace_name=p.tablespace_name and
x.owner=u.username; 2 3 4 5

TABLESPACE_NAME
--------------------
USERS
SYSAUX

SQL>


We are going to migrate tablespace "MIGRATION" and that is not in the list so we can use datapump here.

Also, you cannot transport the SYSTEM tablespace or objects owned by the user SYS.


Remeber:
Opaque Types Types(such as RAW, BFILE, and the AnyTypes) can be transported, but
they are not converted as part of the cross-platform transport operation.
Their actual structure is known only to the application, so the application
must address any endianness issues after these types are moved to the new
platform.


Also:
Floating-Point Numbers BINARY_FLOAT and BINARY_DOUBLE types are transportable
using Data Pump but not the original export utility, EXP.


So, we are good to use expdb/impdp

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

So, lets start with the process:

---------Check that the tablespace will be self contained:

conn / as sysdba
execute sys.dbms_tts.transport_set_check('MIGRATION', true);
--- Check for violations : select * from sys.transport_set_violations;


Output is as:

SQL> conn / as sysdba
Connected.
SQL> execute sys.dbms_tts.transport_set_check('MIGRATION', true);

PL/SQL procedure successfully completed.

SQL> select * from sys.transport_set_violations;

no rows selected

SQL>


If you have any rows returned then these violations must be resolved before the tablespaces
can be transported

But we are good to go.


----------The tablespaces need to be in READ ONLY mode in order to successfully run a transport tablespace export.

ALTER TABLESPACE MIGRATION READ ONLY;



Output is as:

SQL> ALTER TABLESPACE MIGRATION READ ONLY;

Tablespace altered.

SQL>




Now we will create the directory object to export the metadata of the tablespace as:

CREATE OR REPLACE DIRECTORY dpump_dir AS '/ora/fs0000/work/dpump_dir' ;
GRANT READ,WRITE ON DIRECTORY dpump_dir TO system;

expdp system/system DUMPFILE=exp_tablespace_migration.dmp DIRECTORY=dpump_dir TRANSPORT_TABLESPACES = MIGRATION

If you want to perform a transport tablespace operation with a strict containment check, use the TRANSPORT_FULL_CHECK parameter:

Syntax would be following but we are not doing that as we already checked that everythingis self contained:
expdp system/password DUMPFILE=expdat.dmp DIRECTORY = dpump_dir TRANSPORT_TABLESPACES= TBS1,TBS2 TRANSPORT_FULL_CHECK=Y




Output is as:


oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$ sqlplus / as sysdba

SQL*Plus: Release 10.2.0.4.0 - Production on Fri Dec 3 20:08:12 2010

Copyright (c) 1982, 2007, Oracle. All Rights Reserved.


Connected to:
Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

SQL> CREATE OR REPLACE DIRECTORY dpump_dir AS '/ora/fs0000/work/dpump_dir' ;

Directory created.

SQL> GRANT READ,WRITE ON DIRECTORY dpump_dir TO system;

Grant succeeded.

SQL> exit
Disconnected from Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$ expdp system/system DUMPFILE=exp_tablespace_migration.dmp DIRECTORY=dpump_dir TRANSPORT_TABLESPACES = MIGRATION

Export: Release 10.2.0.4.0 - 64bit Production on Friday, 03 December, 2010 20:09:01

Copyright (c) 2003, 2007, Oracle. All rights reserved.

Connected to: Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options
Starting "SYSTEM"."SYS_EXPORT_TRANSPORTABLE_01": system/******** DUMPFILE=exp_tablespace_migration.dmp DIRECTORY=dpump_dir TRANSPORT_TABLESPACES = MIGRATION
Processing object type TRANSPORTABLE_EXPORT/PLUGTS_BLK
Processing object type TRANSPORTABLE_EXPORT/TABLE
Processing object type TRANSPORTABLE_EXPORT/POST_INSTANCE/PLUGTS_BLK
Master table "SYSTEM"."SYS_EXPORT_TRANSPORTABLE_01" successfully loaded/unloaded
******************************************************************************
Dump file set for SYSTEM.SYS_EXPORT_TRANSPORTABLE_01 is:
/ora/fs0000/work/dpump_dir/exp_tablespace_migration.dmp
Job "SYSTEM"."SYS_EXPORT_TRANSPORTABLE_01" successfully completed at 20:10:31


oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$ ls -lart
total 243
drwxr-xr-x 12 oracle dba 14 Dec 3 20:06 ..
drwxr-xr-x 2 oracle oinstall 4 Dec 3 20:09 .
-rw-r----- 1 oracle oinstall 118784 Dec 3 20:10 exp_tablespace_migration.dmp
-rw-r----- 1 oracle oinstall 1033 Dec 3 20:10 export.log

oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$ cat export.log
;;;
Export: Release 10.2.0.4.0 - 64bit Production on Friday, 03 December, 2010 20:09:01

Copyright (c) 2003, 2007, Oracle. All rights reserved.
;;;
Connected to: Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options
Starting "SYSTEM"."SYS_EXPORT_TRANSPORTABLE_01": system/******** DUMPFILE=exp_tablespace_migration.dmp DIRECTORY=dpump_dir TRANSPORT_TABLESPACES = MIGRATION
Processing object type TRANSPORTABLE_EXPORT/PLUGTS_BLK
Processing object type TRANSPORTABLE_EXPORT/TABLE
Processing object type TRANSPORTABLE_EXPORT/POST_INSTANCE/PLUGTS_BLK
Master table "SYSTEM"."SYS_EXPORT_TRANSPORTABLE_01" successfully loaded/unloaded
******************************************************************************
Dump file set for SYSTEM.SYS_EXPORT_TRANSPORTABLE_01 is:
/ora/fs0000/work/dpump_dir/exp_tablespace_migration.dmp
Job "SYSTEM"."SYS_EXPORT_TRANSPORTABLE_01" successfully completed at 20:10:31

oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$





Execute the following command to see which platform are we on source and target and here it is:

On source we have:

set lien 300
SELECT tp.platform_id,substr(d.PLATFORM_NAME,2,30), ENDIAN_FORMAT
FROM V$TRANSPORTABLE_PLATFORM tp, V$DATABASE d
WHERE tp.PLATFORM_NAME = d.PLATFORM_NAME;




Output is as:


SQL> SELECT tp.platform_id,substr(d.PLATFORM_NAME,2,30), ENDIAN_FORMAT
FROM V$TRANSPORTABLE_PLATFORM tp, V$DATABASE d
WHERE tp.PLATFORM_NAME = d.PLATFORM_NAME;

2 3
PLATFORM_ID SUBSTR(D.PLATFORM_NAME,2,30) ENDIAN_FORMAT
----------- ------------------------------------------------------------------------------------------------------------------------ --------------
2 olaris[tm] OE (64-bit) Big

SQL> SQL>




On target we have:

SQL> set line 300
SELECT tp.platform_id,substr(d.PLATFORM_NAME,2,30), ENDIAN_FORMAT
FROM V$TRANSPORTABLE_PLATFORM tp, V$DATABASE d
WHERE tp.PLATFORM_NAME = d.PLATFORM_NAME;
SQL> 2 3
PLATFORM_ID SUBSTR(D.PLATFORM_NAME,2,30) ENDIAN_FORMAT
----------- ------------------------------------------------------------------------------------------------------------------------ --------------
13 inux x86 64-bit Little

SQL>


So we are going from big to little.




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

Now we will transfer files using DBMS_FILE_TRANSFER

DBMS_FILE_TRANSFER.PUT_FILE(
source_directory_object IN VARCHAR2,
source_file_name IN VARCHAR2,
destination_directory_object IN VARCHAR2,
destination_file_name IN VARCHAR2,
destination_database IN VARCHAR2);


1) Create a directory at target database host, and give permissions to
local user.
This is the directory object into which the file is placed at the
destination site, it must exist in the remote file system.

CREATE OR REPLACE DIRECTORY target_dir AS '+DEST_ORADATA1_DG' ;
GRANT WRITE ON DIRECTORY target_dir TO system;





Output is as:

oracle : rac1 : @DFTGDV1 : /home/oracle
$ sqlplus / as sysdba

SQL*Plus: Release 10.2.0.4.0 - Production on Fri Dec 3 20:29:55 2010

Copyright (c) 1982, 2007, Oracle. All Rights Reserved.


Connected to:
Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

SQL> CREATE OR REPLACE DIRECTORY target_dir AS '+DEST_ORADATA1_DG' ;

Directory created.

SQL> GRANT WRITE ON DIRECTORY target_dir TO system;

Grant succeeded.

SQL> exit
Disconnected from Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

oracle : rac1 : @DFTGDV1 : /home/oracle
$



Now,

2) Create a directory at source database host. The directory object from which
the file is copied at the local source site. This directory object must
exist at the source site.

CREATE OR REPLACE DIRECTORY source_dir AS '+ORADATA1_DG/dfsrdv' ;
GRANT READ,WRITE ON DIRECTORY source_dir TO system;
CREATE OR REPLACE DIRECTORY source_dir_1 AS 'ORADATA1_DG/dfsrdv/datafile' ;
GRANT READ,WRITE ON DIRECTORY source_dir TO system;




Output is as:

oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$ sqlplus / as sysdba

SQL*Plus: Release 10.2.0.4.0 - Production on Fri Dec 3 20:34:12 2010

Copyright (c) 1982, 2007, Oracle. All Rights Reserved.


Connected to:
Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

SQL> CREATE OR REPLACE DIRECTORY source_dir AS '+ORADATA1_DG/dfsrdv' ;

Directory created.

SQL> GRANT READ,WRITE ON DIRECTORY source_dir TO system;

Grant succeeded.

SQL> CREATE OR REPLACE DIRECTORY source_dir_1 AS '+ORADATA1_DG/dfsrdv/datafile' ;

Directory created.

SQL> GRANT READ,WRITE ON DIRECTORY source_dir TO system;

Grant succeeded.

SQL> exit
Disconnected from Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$





Now,

Create a dblink to connect to target database host:


CREATE DATABASE LINK DFTGDV CONNECT TO system IDENTIFIED BY system USING 'DFTGDV';

Output is as:

$ sqlplus / as sysdba

SQL*Plus: Release 10.2.0.4.0 - Production on Fri Dec 3 20:36:38 2010

Copyright (c) 1982, 2007, Oracle. All Rights Reserved.


Connected to:
Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

SQL> conn system/system
Connected.
SQL> CREATE DATABASE LINK DFTGDV CONNECT TO system IDENTIFIED BY system USING 'DFTGDV';

Database link created.

SQL>

Test the dblink as:
select * from user_users@DFTGDV


Here is the output:

SQL> select * from user_users@DFTGDV
2 ;

USERNAME USER_ID ACCOUNT_STATUS
------------------------------ ---------- --------------------------------
LOCK_DATE EXPIRY_DATE DEFAULT_TABLESPACE
------------------ ------------------ ------------------------------
TEMPORARY_TABLESPACE CREATED INITIAL_RSRC_CONSUMER_GROUP
------------------------------ ------------------ ------------------------------
EXTERNAL_NAME
--------------------------------------------------------------------------------
SYSTEM 5 OPEN
SYSTEM
TEMP 12-MAR-08 SYS_GROUP



SQL>





Now,

4) Connect to source instance:

DFSRDV => Connect string to source database
DFTGDV => dblink to target database
a1.dat => migration.309.736802539
a4.dat => migrated_from_source.dbf

FILE_NAME TABLESPACE_NAME
------------------------------------------------------------------------------------- --------------------
+ORADATA1_DG/dfsrdv/datafile/migration.309.736802539 MIGRATION
+ORADATA1_DG/dfsrdv/datafile/migration.310.736802721 MIGRATION
+ORADATA1_DG/dfsrdv/datafile/migration.311.736802729 MIGRATION



CONNECT system/system@DFSRDV

BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_1' , 'migration.309.736802539' , 'target_dir' , 'migrated_from_source.dbf' , 'DFTGDV' ) ;
END ;


Due to unpublished bug 8412695, I am getting error ORA-19563 (at the time of writing this note oracle was still working
on this bug and there is was no solution to it), so I have to convert the file at source.


SQL> BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_1' , 'migration.309.736802539' , 'target_dir' , 'migrated_from_source.dbf' , 'DFTGDV' ) ;
END ; 2 3
4 /
BEGIN
*
ERROR at line 1:
ORA-19563: File transfer: data file conversion header validation failed for
file +DEST_ORADATA1_DG/migrated_from_source.dbf
ORA-02063: preceding line from DFTGDV
ORA-06512: at "SYS.DBMS_FILE_TRANSFER", line 60
ORA-06512: at "SYS.DBMS_FILE_TRANSFER", line 168
ORA-06512: at line 2


SQL>


So, I will convert the file at source as:


RMAN> convert tablespace migration to platform="Linux x86 64-bit" FORMAT '+DFDV_ORADATA1_DG';



Output is as:

oracle : goldengate : @DFSRDV : /home/oracle
$ rman

Recovery Manager: Release 10.2.0.4.0 - Production on Fri Dec 3 21:06:23 2010

Copyright (c) 1982, 2007, Oracle. All rights reserved.

RMAN> connect target /

connected to target database: DFSRDV (DBID=1616533891)

RMAN> convert tablespace migration to platform="Linux x86 64-bit" FORMAT '+DFDV_ORADATA1_DG';

Starting backup at 03-DEC-10
using target database control file instead of recovery catalog
allocated channel: ORA_DISK_1
channel ORA_DISK_1: sid=33 devtype=DISK
channel ORA_DISK_1: starting datafile conversion
input datafile fno=00006 name=+ORADATA1_DG/dfsrdv/datafile/migration.309.736802539
converted datafile=+ORADATA1_DG/dfsrdv/xtransport/migration.312.736808821
channel ORA_DISK_1: datafile conversion complete, elapsed time: 00:00:07
channel ORA_DISK_1: starting datafile conversion
input datafile fno=00007 name=+ORADATA1_DG/dfsrdv/datafile/migration.310.736802721
converted datafile=+ORADATA1_DG/dfsrdv/xtransport/migration.313.736808829
channel ORA_DISK_1: datafile conversion complete, elapsed time: 00:00:03
channel ORA_DISK_1: starting datafile conversion
input datafile fno=00008 name=+ORADATA1_DG/dfsrdv/datafile/migration.311.736802729
converted datafile=+ORADATA1_DG/dfsrdv/xtransport/migration.314.736808831
channel ORA_DISK_1: datafile conversion complete, elapsed time: 00:00:03
Finished backup at 03-DEC-10

RMAN>



and files are here in ASM as:

ASMCMD> pwd
+ORADATA1_DG/DFSRDV/XTRANSPORT
ASMCMD> ls -lart
WARNING:option 'r' is deprecated for 'ls'
please use 'reverse'

Type Redund Striped Time Sys Name
XTRANSPORT UNPROT COARSE DEC 03 21:00:00 Y none => MIGRATION.312.736808821
XTRANSPORT UNPROT COARSE DEC 03 21:00:00 Y none => MIGRATION.313.736808829
XTRANSPORT UNPROT COARSE DEC 03 21:00:00 Y none => MIGRATION.314.736808831
ASMCMD>


Now, I will recreate the directory object to this dorectroy and then will move to target as:

CREATE OR REPLACE DIRECTORY source_dir_2 AS '+ORADATA1_DG/DFSRDV/XTRANSPORT' ;
GRANT READ,WRITE ON DIRECTORY source_dir_2 TO system;


Output is as:

oracle : goldengate : @DFSRDV : /home/oracle
$ sqlplus / as sysdba

SQL*Plus: Release 10.2.0.4.0 - Production on Fri Dec 3 21:10:28 2010

Copyright (c) 1982, 2007, Oracle. All Rights Reserved.


Connected to:
Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

SQL> CREATE OR REPLACE DIRECTORY source_dir_2 AS '+ORADATA1_DG/DFSRDV/XTRANSPORT' ;
GRANT READ,WRITE ON DIRECTORY source_dir_2 TO system;

Directory created.

SQL>
Grant succeeded.

SQL>


Now lets transfer files as:



File names to transfer are;

MIGRATION.312.736808821
MIGRATION.313.736808829
MIGRATION.314.736808831


On source system do following:

CONNECT system/system@DFSRDV

BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_2' , 'MIGRATION.312.736808821' , 'target_dir' , 'migrated_from_source.dbf' , 'DFTGDV' ) ;
END ;
/


BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_2' , 'MIGRATION.313.736808829' , 'target_dir' , 'migrated_from_source_1.dbf' , 'DFTGDV' ) ;
END ;
/


BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_2' , 'MIGRATION.314.736808831' , 'target_dir' , 'migrated_from_source_2.dbf' , 'DFTGDV' ) ;
END ;
/



Output is as:

SQL> CONNECT system/system@DFSRDV
Connected.
SQL> BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_2' , 'MIGRATION.312.736808821' , 'target_dir' , 'migrated_from_source.dbf' , 'DFTGDV' ) ;
END ; 2 3
4 /

PL/SQL procedure successfully completed.

SQL> BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_2' , 'MIGRATION.313.736808829' , 'target_dir' , 'migrated_from_source_1.dbf' , 'DFTGDV' ) ;
END ; 2 3
4 /

PL/SQL procedure successfully completed.

SQL> BEGIN
DBMS_FILE_TRANSFER.PUT_FILE ( 'source_dir_2' , 'MIGRATION.314.736808831' , 'target_dir' , 'migrated_from_source_2.dbf' , 'DFTGDV' ) ;
END ;
/
2 3 4

PL/SQL procedure successfully completed.

SQL> SQL>






Now, lets find our transfered datafiles in target as:


Here they are:

ASMCMD> cd +DEST_ORADATA1_DG/DFTGDV/XTRANSPORT
ASMCMD> ls -lrt
WARNING:option 'r' is deprecated for 'ls'
please use 'reverse'

Type Redund Striped Time Sys Name
XTRANSPORT UNPROT COARSE DEC 03 21:00:00 Y FILE_TRANSFER.1552.736809213
XTRANSPORT UNPROT COARSE DEC 03 21:00:00 Y FILE_TRANSFER.1553.736809313
XTRANSPORT UNPROT COARSE DEC 03 21:00:00 Y FILE_TRANSFER.1554.736809333
ASMCMD>



Now, lets plug these files in and here it is:


Now lets create new directory object in target to place the expdp dumpfile thatwe had extracted from source as:

CREATE OR REPLACE DIRECTORY dpump_dir AS '/ora/fs0000/work/oraexport/DFTGDV' ;
GRANT READ,WRITE ON DIRECTORY dpump_dir TO system;

Output is as:

connect as sysdba and do the following:

SQL> CREATE OR REPLACE DIRECTORY dpump_dir AS '/ora/fs0000/work/oraexport/DFTGDV' ;

Directory created.

SQL> GRANT READ,WRITE ON DIRECTORY dpump_dir TO system;

Grant succeeded.

SQL>



Now, scp the file exp_tablespace_migration.dmp from source to target as:

oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$ scp exp_tablespace_migration.dmp rac1:/ora/fs0000/work/oraexport/DFTGDV/.
oracle@rac1's password:
exp_tablespace_migra 100% ********************************************************************************************************** 116 KB 00:00

oracle : goldengate : @DFSRDV : /ora/fs0000/work/dpump_dir
$


So now on source we have that dump:

oracle : rac1 : @DFTGDV1 : /ora/fs0000/work/oraexport/DFTGDV
$ ls -lart
total 128
drwxrwxr-x 51 4001 dba 4096 Dec 3 21:47 ..
drwxr-xr-x 2 oracle oinstall 4096 Dec 3 21:53 .
-rw-r----- 1 oracle oinstall 118784 Dec 3 21:53 exp_tablespace_migration.dmp

oracle : rac1 : @DFTGDV1 : /ora/fs0000/work/oraexport/DFTGDV
$



Now, we need to create the use "MIGRATE" in the taget database to plugin the datafiles. You can also use the remap schema option to do that.

But I will create a user and will not use the remap schema option.

create user migrate identified by migrate default tablespace users temporary tablespace temp;
grant connect,resource,dba to migrate;

I am giving same grants to this user as we have in soruce.


Here is the output:


SQL> create user migrate identified by migrate default tablespace users temporary tablespace temp;

User created.

SQL> grant connect,resource,dba to migrate;

Grant succeeded.

SQL>


Now lets import:

impdp system/system DUMPFILE=exp_tablespace_migration.dmp DIRECTORY=dpump_dir TRANSPORT_DATAFILES='+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1552.736809213','+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1553.736809313','+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1554.736809333'


Here is the output:

oracle : rac1 : @DFTGDV1 : /ora/fs0000/work/oraexport/DFTGDV
$ 213','+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1553.736809313','+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1554.736809333' < dumpfile="exp_tablespace_migration.dmp" directory="dpump_dir" transport_datafiles="+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1552.736809213,+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1553.736809313,+DEST_ORADATA1_DG/DFTGDV/XTRANSPORT/FILE_TRANSFER.1554.736809333"> ALTER TABLESPACE MIGRATION READ WRITE;

Tablespace altered.

SQL>



Now lets test if we have transfered the data as:

SQL> conn migrate/migrate
Connected.
SQL> select table_name from user_tables;

TABLE_NAME
------------------------------
TRANSFER
PLACES

SQL> select count(*) from TRANSFER;

COUNT(*)
----------
1589

SQL> select count(*) from PLACES;

COUNT(*)
----------
50602

SQL> exit
Disconnected from Oracle Database 10g Enterprise Edition Release 10.2.0.4.0 - 64bit Production
With the Partitioning, Real Application Clusters, OLAP, Data Mining
and Real Application Testing options

oracle : rac1 : @DFTGDV1 : /ora/fs0000/work/oraexport/DFTGDV
$


and its a success !!!!!!!

Now, if you want you can the default tablespace of user migrate from users to migration as:


SQL> conn / as sysdba
Connected.
SQL> alter user migrate default tablespace migration;

User altered.

SQL>


Thats it !!!!!!!!!!!


You have thousands of eyes, and yet You have no eyes. You have thousands of forms, and yet You do not have even one. You have thousands of Lotus Feet, and yet You do not have even one foot. You have no nose, but you have thousands of noses. This Play of Yours entrances me. 2 Amongst all is the Light?You are that Light. By this
Illumination, that Light is radiant within all.