Using WebLogic jDriver for Oracle/XA in Distributed Transactions
The following sections describe how to integrate transactions with EJB and RMI applications that use the WebLogic jDriver for Oracle/XA and run under BEA WebLogic Server.
- Differences Using the WebLogic jDriver for Oracle in XA versus Non-XA Mode
- Configuring JDBC XA and Non-XA Resources
- Limitations of the WebLogic jDriver for Oracle XA
- Implementing Distributed Transactions
Differences Using the WebLogic jDriver for Oracle in XA versus Non-XA Mode
WebLogic jDriver for Oracle fully supports the JDBC 2.0 Optional Package API for distributed transactions. Applications using the driver in distributed transaction (XA) mode can use all JDBC 2.0 Core API the same way as in local transaction (non-XA) mode, with the exception of the following:
- Connections have to be obtained via the JDBC 2.0 javax.sql.DataSource API, but not through the deprecated java.sql.DriverManager or java.sql.Driver API.
- When used in WebLogic Server, configure a TxDataSource in order to use it. Refer to "Configuring JDBC DataSources in the Administration Console Online Help for instructions about configuring TxDataSource and Connection Pools.
- Auto commit is false by default. Attempting to enable autocommit mode by calling the java.sql.Connection.setAutoCommit method on the Connection will throw a SQLException.
- Attempting to complete the distributed transaction by calling java.sql.Connection.commit or java.sql.Connection.rollback methods will throw a SQLException.
The reason for the last two differences is because when the WebLogic jDriver for Oracle/XA participates in a distributed transaction, it is the external Transaction Manager that is demarcating and coordinating the distributed transaction.
For more information, refer to the JDBC 2.0 Standard Extension API spec [version 1.0, dated 98/12/7 Section 7.1 last 2 paragraphs].
Configuring JDBC XA and Non-XA Resources
Use the Administration Console to configure your JDBC resources, as described in the following sections.
JDBC/XA Resources
To allow XA JDBC drivers to participate in distributed transactions, configure the JDBC connection pool as follows:
- Specify the DriverName property as the name of the class supporting the javax.sql.XADataSource interface. That is, use weblogic.jdbc.oci.xa.XADataSource as the DriverName property (Driver Classname in the Administration Console).
- Ensure that the database properties are specified. For more information on data source properties for the WebLogic jDriver for Oracle, see Configuring XA JDBC Drivers for Distributed Transactions in the Administration Console Online Help.
See the Administration Console Online Help for the JDBC Connection Pools panel for procedures and attribute definitions.
Non-XA JDBC Resources
To support non-XA JDBC resources, select the enableTwoPhaseCommit database property (Emulate Two-Phase Commit for non-XA Driver in the Administration Console) when configuring a JDBC Tx Data Source (a data source with Honor Global Transactions selected). For more information on this property, see Configuring Non-XA JDBC Drivers for Distributed Transactions in the Administration Console Online Help.
Limitations of the WebLogic jDriver for Oracle XA
WebLogic jDriver for Oracle in XA mode does not support the following:
- Mixing local and global transactions. This throws a SQLException if an SQL operation is attempted with no global transaction.
- Performing DDL operations (e.g. create/drop table, stored procedures, and so forth). If you want to perform DDL operations, you need to define two different connection pools as follows:
- One non-XA connection pool that can be used for DDL operations.
- One XA connection pool that can be used for DML operations in distributed transactions.
- Setting the transaction isolation level for a transaction. Transactions use the transaction isolation level set on the connection or the default transaction isolation level for the database.
- Enabling support for local transactions. You cannot set supportsLocalTransaction to true for connection pools that use the WebLogic jDriver for Oracle in XA mode. If you attempt to commit a local transaction on a connection from the connection pool, the following exception is thrown:
java.sql.SQLException:Does not support SQL execution with no global transaction
Implementing Distributed Transactions
This topic includes the following sections:
Importing Packages
Listing 4-1 shows the packages that the application imports. In particular, note that:
- The java.sql.* and javax.sql.* packages are required for database operations.
- The javax.naming.* package is required for performing a JNDI lookup on the pool name, which is passed in as a command-line parameter upon server startup. The pool name must be registered on that server group.
Listing 4-1 Importing Required Packages
import java.sql.*;
import javax.sql.*;
import javax.naming.*;
Finding the Data Source via JNDI
Listing 4-2 shows how to find the data source via JNDI.
Listing 4-2 Finding the Data Source via JNDI
static DataSource pool;...public void get_connpool(String pool_name) throws Exception { try { javax.naming.Context ctx = new InitialContext(); pool = (DataSource)ctx.lookup("jdbc/" + pool_name); } catch (javax.naming.NamingException ex){ TP.userlog("Couldn't obtain JDBC connection pool: " + pool_name); throw ex; } }
}
Performing a Distributed Transaction
Listing 4-3 shows a distributed transaction involving two database connections and implemented as a business method within a session bean.
Listing 4-3 Performing a Distributed Transaction
public class myEJB implements SessionBean { EJBContext ejbContext;
public void myMethod(...) { javax,transaction.UserTransaction usertx; javax.sql.DataSource data1; javax.sql.DataSource data2; java.sql.Connection conn1; java.sql.Connection conn2; java.sql.Statement stat1; java.sql.Statement stat2;
InitialContext initCtx = new InitialContext(); // // Initialize a user transaction object. // usertx = ejbContext.getUserTransaction(); //Start a new user transaction. usertx.begin(); // Establish a connection with the first database // and prepare it for handling a transaction. data1 = (javax.sql.DataSource) initCtx.lookup("java:comp/env/jdbc/DataBase1"); conn1 = data1.getConnection(); stat1 = conn1.getStatement(); // Establish a connection with the second database // and prepare it for handling a transaction. data2 = (javax.sql.DataSource) initCtx.lookup("java:comp/env/jdbc/DataBase2"); conn2 = data1.getConnection(); stat2 = conn2.getStatement(); //Update both conn1 and conn2. The EJB Container //automatically enlists the participating resources. stat1.executeQuery(...); stat1.executeUpdate(...); stat2.executeQuery(...); stat2.executeUpdate(...); stat1.executeUpdate(...); stat2.executeUpdate(...); //Commit the transaction (apply the changes to the //participating databases). usertx.commit(); //Release all connections and statements. stat1.close(); stat2.close(); conn1.close(); conn2.close(); } ...
}