Tuesday, December 20, 2011
SCN information from Goldengate trail files
Here is a sample extract parameter to accomplish:
EXTRACT E_SITEA
SETENV (ORACLE_SID=ORCL)
GETENV (ORACLE_SID)
USERID orclgg, PASSWORD oracle
EXTTRAIL D:/GG11/dirdat/ea
TRANLOGOPTIONS EXCLUDEUSER ORCLGG
STATOPTIONS RESETREPORTSTATS
REPORT AT 00:01
REPORTROLLOVER AT 00:01
REPORTCOUNT EVERY 60 SECONDS, RATE
GETUPDATEBEFORES
TABLE sitea.products, tokens (tk-scn=@getenv("ORATRANSACTION","SCN));
Now start the extract and see the trail file generated using LOGDUMP.
OGG_HOME> logdump
Logdump 275 >open D:/GG11/dirdat/ea000004
Current LogTrail is D:\GG11\dirdat\ea000004
Logdump 276 >ghdr on
Logdump 277 >detail data
Logdump 278 >usertoken detail
Logdump 279 >n
2011/12/20 20:24:26.633.000 FileHeader Len 952 RBA 0
Name: *FileHeader*
3000 01b0 3000 0008 4747 0d0a 544c 0a0d 3100 0002 | 0...0...GG..TL..1...
0002 3200 0004 2000 0000 3300 0008 02f1 daba b318 | ..2... ...3.........
a528 3400 002e 002c 7572 693a 564d 5241 4748 4156 | .(4....,uri:xxxxx
2d4c 4150 3a69 6463 3a6f 7261 636c 653a 636f 6d3a | -LAP:idc:oracle:com:
6472 6976 652d 443a 4747 3131 3600 0019 0017 443a | drive-D:GG116.....D:
2f47 4731 312f 6469 7264 6174 2f65 6130 3030 3030 | /GG11/dirdat/ea00000
3437 0000 0101 3800 0004 0000 0004 39ff 0008 0000 | 47....8.......9.....
Logdump 280 >n
___________________________________________________________________
Hdr-Ind : E (x45) Partition : . (x00)
UndoFlag : . (x00) BeforeAfter: A (x41)
RecLength : 0 (x0000) IO Time : 2011/12/20 20:24:26.592.000
IOType : 150 (x96) OrigNode : 0 (x00)
TransInd : . (x03) FormatType : R (x52)
SyskeyLen : 0 (x00) Incomplete : . (x00)
AuditRBA : 0 AuditPos : 0
Continued : N (x00) RecCount : 0 (x00)
2011/12/20 20:17:15.000.000 Insert Len 39 RBA 1022
Name: SITEB.PRODUCTS
After Image: Partition 4 GU s
0000 000a 0000 0000 0000 0000 0002 0001 0007 0000 | ....................
0003 4a4a 4a00 0200 0a00 0000 0000 0000 0000 14 | ..JJJ..............
Column 0 (x0000), Len 10 (x000a)
0000 0000 0000 0000 0002 | ..........
Column 1 (x0001), Len 7 (x0007)
0000 0003 4a4a 4a | ....JJJ
Column 2 (x0002), Len 10 (x000a)
0000 0000 0000 0000 0014 | ..........
User tokens: 15 bytes
tk-scn : 1109315 -->> SCN number
Monday, December 19, 2011
Conflict detection in an Active-Active environment
The below scenarios can occur when the timing of simultaneous changes results in one of the out of sync conditioins:
# A replicated insert attempts to add a row that already exists in the target.
# The before image of a replicated update doesnot match the current row in the target.
# a replicated delete attempts to remove a row that doesnot exist on the target.
An example, when UserA on SiteA updates a row, and UserB on SiteB updates the same row. If userB's transaction occurs before userA's transactions is synchronized to Database B, there will be a confict on the replicated transaction.
In these scenarious, Goldengate doesnot provide default procedures for handling the conflicts and this can be handled with custom conflict resulution scripts that needs to be included in the parameter files.
Here is a small demonstration in an online retail environment:
* An online shopping store is stocked with 20 limited ipods. If the quantity goes to 0, then this item is tagged to be out of stock on the online system.
* Customer A places an order for 15 ipods that came from Site A database:
sql would be:
"update products set quantity_in_stock = quantity_in_stock -15 where product_name = 'ipod' and quantity_in_stock = 20;"
* Customer B simultaneiously places an order for 4 ipods which came from ABC database.
sql would be:
"update products set quantity_in_stock = quantity_in_stock - 4 where products_name = 'ipod' and quantity_in_stock = 20;"
In the above example, the number of ipods in stock according to the SiteA database is 5 and the number of ipods in stock according to SiteB database is 16. When Goldengate attempts to update the SiteA database with the transaction from customer A, the transaction will fail because the before image of quantity_in_stock is 20 which doesnot match the current figure value which is 16. Reversely the transaction replicated to the SiteB database will also fail because the before image of quantity_in_stock 20 doesnot match the current value of 5.
So, unless a conflict resolution script is not written, Goldengate doesnot veirfy target row data before replicating the changes. And this conflict resolution can be handled by writing a sql stored procedure or a query using the SQLEXEC parameter in the replicat parameter file on both the databases.
Demo(Extract pump is not used as both the DB's are in the same server):
Site A
_______
e_sitea --> Extract(Change data capture that reads redo from the SiteA database)
EXTRACT E_SITEA
SETENV (ORACLE_SID=SITEA)
GETENV (ORACLE_SID)
USERID orclgg, PASSWORD oracle
EXTTRAIL D:/GG11/dirdat/ea
-- This parameter prevents the transaction looping as the transactions already exists on the target(SiteB).
TRANLOGOPTIONS EXCLUDEUSER ORCLGG
-- This parameter causes Extract to capture the before image of the record and write it into the trail file.
GETUPDATEBEFORES
TABLE SITEA.PRODUCTS;
r_sitea --> Replicat(Replicat to apply data captured from the SiteB database to the SiteA database)
REPLICAT R_SITEB
SETENV (ORACLE_SID=SITEA)
GETENV (ORACLE_SID)
USERID orclgg, PASSWORD oracle
ASSUMETARGETDEFS
ALLOWDUPTARGETMAP
IGNOREUPDATES
MAP SITEA.*, SITEB.*;
GETUPDATES
IGNOREINSERTS
IGNOREDELETES
MAP SITEB.PRODUCTS, TARGET SITEB.PRODUCTS,
SQLEXEC (id conflict,
query "select qty_in_stock from sitea.products where products_number = :vpnum",
params (vpnum = products_number)),
colmap (products_number = products_number,
products_name = products_name,
qty_in_stock = @if(conflict.qty_in_stock <> BEFORE.qty_in_stock,
@compute(conflict.qty_in_stock - (BEFORE.qty_in_stock - qty_in_stock)),
qty_in_stock)
);
Site B
______
e_siteB --> Extract (Change data capture that reads redo from the SiteB database)
EXTRACT E_SITEB
SETENV (ORACLE_SID=SITEB)
GETENV (ORACLE_SID)
USERID orclgg1, PASSWORD oracle
EXTTRAIL D:/GG1112/dirdat/eb
-- This parameter prevents the transaction looping as the transactions already exists on the target(SiteA).
TRANLOGOPTIONS EXCLUDEUSER ORCLGG1
-- This parameter causes Extract to capture the before image of the record and write it into the trail file.
GETUPDATEBEFORES
TABLE SITEB.PRODUCTS;
r_siteA --> Replicat (Replicat to apply data captured from the SiteA database to the SiteB database)
REPLICAT R_SITEA
SETENV (ORACLE_SID=SITEB)
GETENV (ORACLE_SID)
USERID orclgg1, PASSWORD oracle
ASSUMETARGETDEFS
DISCARDFILE ./dirrpt/rwconf.dsc, purge
ALLOWDUPTARGETMAP
IGNOREUPDATES
MAP SITEB.*, TARGET SITEA.*;
GETUPDATES
IGNOREINSERTS
IGNOREDELETES
MAP SITEA.PRODUCTS, TARGET SITEB.PRODUCTS,
SQLEXEC (id conflict,
query "select qty_in_stock from siteb.products where products_number = :vpnum",
params (vpnum = products_number)),
colmap (products_number = products_number,
products_name = products_name,
qty_in_stock = @if(conflict.qty_in_stock <> BEFORE.qty_in_stock,
@compute(conflict.qty_in_stock - (BEFORE.qty_in_stock - qty_in_stock)),
qty_in_stock)
);
Over here, the replicats on both the sites have been added with "ALLOWDUPTARGETMAP", this is used to perform conflict detection on the update operation. This would turn off processing the update operations and do a wild card mapping of all sitea and siteb tables(only the inserts and deletes will be processed by the wild card mapping).
Next, we turn on processing for updates, and turn off processing for insert and delete operations. This update would be applied using a pl/sql code which is used in by SQLEXEC.
For all update operations performed on the PRODUCTS table, a query is executed against the target table and the data from the qty_in_stock column is returned. The GoldenGate column mapping munction @IF, is used to compare the returned value with the before column image read from the GoldenGate Trail. If the values are not equal, the first statement is executed where the source table value for qty_in_stock is subtracted from the before image qty_in_stock value. The result is then subtracted from the qty_in_stock value returned from the query, and the result is applied to the target table.
Test:
1. Create the below table on both the databases SiteA and SiteB:
create table products (
products_number number(10),
products_name varchar2(10),
qty_in_stock number(5),
primary key (products_number)
using index
);
insert into products values (1,'IPOD', 20);
commit;
2. Get the database to log the keys for the table products using "add trandata" from ggsci prompt. Do this on both the GG instances.
Site A:
ggsci> dblogin userid orclgg, password ggsci> add trandata sourcea.products
Site B:
ggsci> dblogin userid orclgg, password ggsci> add trandata sourceb.products
3. create the Goldengate components on both sides.
Site A:
ggsci> add extract e_sitea, tranlog, begin now
ggsci> add exttrail D:/GG11/dirdat/ea extract e_sitea
ggsci> dblogin userid orclgg, password
ggsci> add replicat r_siteb, exttrail D:/GG1112/dirdat/eb checkpointtable
Site B:
ggsci> add extract e_siteb, tranlog, begin now
ggsci> add exttrail D:/GG1112/dirdat/eb extract e_siteb
ggsci> dblogin userid orclgg1, password
ggsci> add replicat r_sitea, exttrail D:/GG11/dirdat/ea checkpointtable
4. Update the product table on both the databases and commit the transaction.
update sitea.products set qty_in_stock = qty_in_stock-15 where products_name = 'IPOD' and qty_in_stock = 20;
commit;
update siteb.products set qty_in_stock = qty_in_stock-4 where products_name = 'IPOD' and qty_in_stock = 20;
commit;
Now, validate the results from both the databases and see that "qty_in_stock" results with 1 ipod.
Friday, December 16, 2011
How to guage approximate time for the replicat to catchup the long running transactions
Thursday, December 15, 2011
Steps for performing the Various filtering method between two oracle databases through initial Data load method
1. Make sure to create a same table structure on source and Target Machines and manager process must be running.
2. Enable supplemental redo logging on the tables to be replicated
3. Sample Table structure:
Create table sender.University (studcode number(10),studname varchar2(10),country varchar2(10) , constraint uty_key unique(studcode));
Create table receiver.University (studcode number(10),studname varchar2(10),country varchar2(10) , constraint uty_key unique(studcode));
On Source Machine
Step 1 :GGSCI> add extract fillter, sourceistable
Step 2 : GGSCI> edit params fillter
Extract fillter
Userid ogg@goldengate:1521/source,password ****
Rmthost goldengate, mgrport 7810
Rmttask replicat,Group repl2
Table sender.university, FILTER ( @STRFIND ( city, “UAE” > 0));
Step 3 :GGSCI > start extract fillter
On Target Machine
Step 1 :GGSCI> add replicat repl2, specialrun
Step 2: GGSCI> edit params repl2
Replicat repl2
Userid ogg@goldengate:1521/target,password ****
assumetargetdefs
MAP sender. university, TARGET receiver.university;
Step 3: GGSCI> start replicat repl2
The output of the filter condition:
On Source Machine :
An insert sample statements for checking the replication
Insert into sender.university( 100,’SAM’,’UAE’)
Insert into sender.university( 101,’SAM’,’UK’)
commit;
On Target Machine :
If the country value is “UAE” only should replicate to Target Machine from the Source Machine as per the filter condition.
SQL> select * from receiver.university;
Studcode Studname Country
100 SAM UAE
What is FILTER Keyword?
The FILTER clause offers us more functionality than the WHERE clause because you can employ any of GoldenGate’s column conversion functions to filter data, whereas the WHERE clause accepts basic WHERE operators.
For example we can use standard arithmetic operators like ‘+’,'-’,'/’,'*’ or comparison operators like ‘>’,’<', '=' as well as GoldenGate functions like @COMPUTE, @DATE, @STRFIND, @STRNUM etc
For example we can use the STRFIND function as part of the Extract parameter file to only extract records from the table that match a particular string value as shown below.
TABLE sender.university,FILTER (@STRFIND (city, “LONDON”) > 0)
Another few examples for filtering
Filtering Columns
Suppose that instead of extracting all the columns in the Employees table, you’re only interested in the EMPLOYEE_ID,FIRST_NAME and LAST_NAME columns. You can do this by adding the following to your extract TABLE Parameter.
TABLE HR.EMPLOYEES, COLS (employee_id,first_name,last_name)
In the example, all columns are ignored except the three columns listed in the COLS parameter. Keep in mind that if you use column filtering in your data pump extract, you can no longer use passthru mode.
Another example for Column filtering is when you want all the columns in a table to be extracted except for a few columns. Let’s assume you want to extract the entire
EMPLOYEES table except for the EMAIL column. You can do that in the Extract with COLEXCEPT parameter, as shown in the following example:
TABLE HR.EMPLOYEES COLSEXCEPT (email);
Keep in mind that you can’t exclude key columns with COLSEXCEPT.
Filtering Rows
Let’s look at another example using filtering based on the DML operation, such as insert, update, or delete. Suppose you only want to replicate deletes if the employee’s monthly salary is less than 1000.
You can set up a filter as shown next that only executes the filter for delete statements. This example
filters using the Extract TABLE parameter
Table HR.EMPLOYEES,
FILTER (ON DELETE, SALARY / 12 < 1000);
You’ve seen a few examples of the extensive capabilities of Goldengate to filter tables,rows and columns .
Implementing Parallel Extracts and Replicats Using Key Ranges – Performance Tuning
Creating parallel Extracts and replicates is to use the @RANGE function to spilt the incoming table rows into equal buckets using a Golden gate hash algorithm on the key values.
You do this to divide the replication workload into parallel processes by key ranges and make the replication process more data and reduce the replicate lag.
Sample Table structure:
Create table sender.empl (empid number(10),empname varchar2(10),constraint emp_key unique(empid));
Create table receiver.empl (empid number(10),empname varchar2(10),constraint emp_key unique(empid));
On Source database
1. GGSCI> add extract ext_ran, tranlog, Begin now
2. GGSCI> add exttrail /u01/app/oracle/product/GG1/dirdat/lt,extract ext_ran
3. GGSCI> add extract dpump, exttrailsource /u01/app/oracle/product/GG1/dirdat/lt
4. GGSCI> edit params ext_ran
Extract ext_ran
Userid ogg@goldengate:1521/source,password ****
Exttrail /u01/app/oracle/product/GG1/dirdat/lt
Table sender.empl;
5. GGSCI> add rmttrail /u01/app/oracle/product/GG2/dirdat/rt,extract dpump, megabytes 500
6. GGSCI> edit params dpump
Extract dpump
Userid ogg@goldengate:1521/source,password ****
Rmthost goldengate,mgrport 7810
Rmttrail /u01/app/oracle/product/GG2/dirdat/rt
Passthru
Table sender.empl;
On Target database
1. GGSCI> add replicat rep_ran1, exttrail /u01/app/oracle/product/GG2/rt,checkpointtable ogg.chkptab
2. GGSCI> edit params rep_ran1
Replicat rep_ran1
Userid OGG@goldengate:1521/target,password ****
MAP sender.empl, TARGET receiver.empl, EMPLOYEES, COLMAP (USEDEFAULTS), FILTER (@RANGE (1,3));
3. GGSCI> add replicat rep_ran2, exttrail /u01/app/oracle/product/GG2/rt,checkpointtable ogg.chkptab
4. GGSCI> edit params rep_ran2
Replicat rep_ran2
Userid OGG@goldengate:1521/target,password ****
MAP sender.empl, TARGET receiver.empl, EMPLOYEES,COLMAP (USEDEFAULTS),FILTER (@RANGE (2,3));
5. GGSCI> add replicat rep_ran3, exttrail /u01/app/oracle/product/GG2/rt,checkpointtable ogg.chkptab
6. GGSCI> edit params rep_ran3
Replicat rep_ran3
Userid OGG@goldengate:1521/target,password ****
MAP sender.empl, TARGET receiver.empl, COLMAP (USEDEFAULTS),FILTER (@RANGE (3,3));
On Source & Target database
1. GGSCI > start extract exp_ran
2. GGSCI> start replicat rep_ran1
3. GGSCI> start replicat rep_ran2
4. GGSCI> start replicat rep_ran3
0r
1. GGSCI> start replicat rep_ran*
After some changes are applied , you can check the status of the replicats to make sure they’re processing properly.
1. GGSCI> info rep *
Replicat abending with database error 1403
This error can be seen very frequently on a production environment when implementing Goldengate for the first time.
During the instantiation of the target database, if there is any open transaction which is yet to be committed, the extract is started with "begin now" and then the export/rman backup is started. This open transaction could be missed out by either of the processes and when this open transaction is committed at a later point of time and an update/delete is fired against this transaction which is captured by the extract, guess what happens?
Replicat would not be able to find the row for doing the latter transaction.
There is a way to avoid this situation by altering the extract to the oldest long-pending transaction from gv$transaction/v$transaction. Use the below to find:
SQL> select min(START_TIME) from gv$transaction;
MIN(START_TIME)
--------------------
12/15/11 20:53:54
Alter the extract with the begin timestamp to process the oldest open transaction:
# delete the existing extract and pump trails
# create a new trail sequence for extract and point the pump to read the new trail sequence
# ggsci> alter EXTRACT
# ggsci> alter EXTRACT
# alter the replicat to the new trail sequence and start the processes.
Ofcourse remember to use HANDLECOLLISIONS, until you pass by the time your ORIGINAL TRAIL was processed until.
And for NEXT TIME make sure you run your GV$TRANSACTION sql above. And create the extract process to start capturing from the time you get from GV$TRANSACTION.
Wednesday, December 7, 2011
Incompatible record in the trail file
Extract or Replicat abends with error message "Incompatible Record" or ERROR 514 or ERROR 509.
2) Two Extracts over writing the same set of trail files by mistake.
Some ways to avoid:
- Create a new trail sequence (to avoid confusion and keep the old one intact)
- Change the extract checkpoint after you locate the correct rba (make sure you backup the checkpoint file)
- alter the replicat to this new trail sequence
Data integrity issues may exist when using any of these recovery techniques above.
Friday, December 2, 2011
FILTER PARAMETER
Steps for performing filtering the data between two oracle databases through initial load
2. Enable supplemental redo logging on the tables to be replicated
3. Sample Table structure:
Create table sender.University
(studcode number(10),
studname varchar2(10),
country varchar2(10) ,
constraint uty_key unique(studcode));
Create table receiver.University
(studcode number(10),
studname varchar2(10),
country varchar2(10) ,
constraint uty_key unique(studcode));
On Source Machine
Step 1 :GGSCI> add extract fillter, sourceistable
Userid ogg@goldengate:1521/source,password ****
Rmthost goldengate, mgrport 7810
Rmttask replicat,Group repl2
Table sender.university, FILTER ( @STRFIND ( city, “UAE” > 0));
Step 3 :GGSCI > start extract fillter
Step 1 :GGSCI> add replicat repl2, specialrun
userid ogg@goldengate:1521/target,password ****
assumetargetdefs
MAP sender. university, TARGET receiver.university;
Test/Verification:
The output of the filter condition:
Sample insert statements for checking the replication
Insert into sender.university values ( 101,’SAM’,’UK’);
commit;
SQL> select * from receiver.university;
Studcode Studname Country
100 SAM UAE
What is FILTER Keyword?
The FILTER clause offers us more functionality than the WHERE clause because you can employ any of GoldenGate’s column conversion functions to filter data, whereas the WHERE clause accepts basic WHERE operators.
For example we can use standard arithmetic operators like ‘+’,'-’,'/’,'*’ or comparison operators like ‘>’,’<', '=' as well as GoldenGate functions like @COMPUTE, @DATE, @STRFIND, @STRNUM etc
For example we can use the STRFIND function as part of the Extract parameter file to only extract records from the table that match a particular string value as shown below.
TABLE sender.university,FILTER (@STRFIND (city, “LONDON”) > 0)
----------------------------All the best------------------------------------------------
Thursday, December 1, 2011
Steps for enabling DDL Replication with two Oracle databases
Steps for performing/testing Online DDL Replication between two Oracle Databases
Here I am performing Golden gate unidirectional replication between 2 Oracle11g instances.
Assuming that the user ogg are created on both Source and Target databases.
Run the following scripts on source from the directory where the GoldenGate software was installed.
SQL> alter session set recyclebin=OFF;
SQL> @marker_setup
SQL> @ddl_setup
SQL> @role_setup
Enter GoldenGate schema name: OGG
SQL> grant ggs_ggsuser_role to ogg;
SQL> @ddl_enable
SQL> @ddl_pin OGG # Optional
After installing please follow the below steps :
1. Make sure to create a same table structure on source and Target Machines and manager process must be running.
2. Enable supplemental redo logging on the tables to be replicated.
3. Sample Table structure:
Create table sender.globe
(phonecode number(10),
country varchar2(10),
constraint glb_key unique(phonecode));
Create table receiver.globe
(phonecode number(10),
country varchar2(10),
constraint glb_key unique(phonecode));
On Source machine
Userid ogg@goldengate:1521/source,password XXXXXX
Rmthost goldengate, mgrport 7810
Rmttrail /u01/app/oracle/product/OGG/dirdat/rt
DDL INCLUDE MAPPED
Table sender.globe;
On Target machine
Step 1 :GGSCI> add replicat repl_ddl, exttrail /u01/app/oracle/product/GG2/rt,
Step 2: GGSCI> edit params ./GLOBALS
checkpointtable ogg.chkptab
Step 4: GGSCI> edit params repl_ddl
Replicat repl_ddl
Userid ggate@goldengate:1521/target,password XXX
assumetargetdefs
MAP sender.globe, TARGET receiver.globe;
Start Extract and Replicat processes on source & target machines
- GGSCI> start extract extp_ddl
2. GGSCI> start replicat repl_ddl
Once the Lag is 0, perform the following:
On Source Databases
SQL> ALTER TABLE SENDER.GLOBE ADD STATE VARCHAR2(10);
On Target Machine
SQL> DESC SENDER.GLOBE
phonecode number(10)
country varchar2(10)
state varchar2(10)
Look on “DDL INCLUDE MAPPED” on source extract file. This is the parameter that causes DDL replication between the databases.
Short explanation for DDL INCLUDE MAPPED: This parameter enables DDL Support for all tables which have been mapped.