Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Oracle JDBC intermittent Connection Issue

Tags:

java

oracle

jdbc

I am experiencing a very strange problem This is a very simple use of JDBC connecting to an Oracle database

OS: Ubuntu Java Version:  1.5.0_16-b02                1.6.0_17-b04 Database: Oracle 11g Release 11.1.0.6.0 

When I make use of the jar file OJDBC14.jar it connects to the database everytime When I make use of the jar file OJDBC5.jar it connects some times and other times it throws an error ( shown below) If I recompile with Java 6 and use OJDBC6.jar I get the same results as OJDBC5.jar

I need specific features in JODB5.jar that are not available in OJDBC14.jar

Any ideas

Error

> Connecting to oracle     java.sql.SQLException: Io exception: Connection reset     at oracle.jdbc.driver.SQLStateMapping.newSQLException(SQLStateMapping.java:74)     at oracle.jdbc.driver.DatabaseError.newSQLException(DatabaseError.java:110)     at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:171)     at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:227)     at oracle.jdbc.driver.DatabaseError.throwSqlException(DatabaseError.java:494)     at oracle.jdbc.driver.T4CConnection.logon(T4CConnection.java:411)     at oracle.jdbc.driver.PhysicalConnection.<init>(PhysicalConnection.java:490)     at oracle.jdbc.driver.T4CConnection.<init>(T4CConnection.java:202)     at oracle.jdbc.driver.T4CDriverExtension.getConnection(T4CDriverExtension.java:33)     at oracle.jdbc.driver.OracleDriver.connect(OracleDriver.java:474)     at java.sql.DriverManager.getConnection(DriverManager.java:525)     at java.sql.DriverManager.getConnection(DriverManager.java:171)     at TestConnect.main(TestConnect.java:13) 

Code

Below is the code I am using

import java.io.*; import java.sql.*; public class TestConnect {     public static void main(String[] args) {         try {             System.out.println("Connecting to oracle");              Connection con=null;             Class.forName("oracle.jdbc.driver.OracleDriver");             con=DriverManager.getConnection(                "jdbc:oracle:thin:@172.16.48.100:1535:sample",                "JOHN",                "90009000");             System.out.println("Connected to oracle");              con.close();             System.out.println("Goodbye");         } catch(Exception e) { e.printStackTrace(); }     } } 
like image 254
Lipska Avatar asked Feb 24 '10 15:02

Lipska


People also ask

Why is JDBC not working?

Connection errors are typically caused by incorrect or invalid connection string parameters, such as an invalid host name or database name. Verify that you are using correct and valid connection string parameters for the Gate. Jdbc. Url property.

How do you fix status failure failed IO error the network adapter could not establish the connection?

Restart it with the "lsnrctl start" command or on a Windows OS by starting the listener service. Ensure the correct hostname is specified in listner. ora. Add the hostname and IP address in the hosts file located under C:\Windows\System32\drivers\etc folder.

What is fast connection failover?

The Fast Connection Failover (FCF) feature is a Fast Application Notification (FAN) client implemented through the connection pool. The feature requires the use of an Oracle JDBC driver and high availability (HA) database configurations like Oracle RAC, Single Restart, or Active Data Guard.


2 Answers

There is a solution provided to this problem in some of the OTN forums (https://kr.forums.oracle.com/forums/thread.jspa?messageID=3699989). But, the root cause of the problem is not explained. Following is my attempt to explain the root cause of the problem.

The Oracle JDBC drivers communicate with the Oracle server in a secure way. The drivers use the java.security.SecureRandom class to gather entropy for securing the communication. This class relies on the native platform support for gathering the entropy.

Entropy is the randomness collected/generated by an operating system or application for use in cryptography or other uses that require random data. This randomness is often collected from hardware sources, either from the hardware noises, audio data, mouse movements or specially provided randomness generators. The kernel gathers the entropy and stores it is an entropy pool and makes the random character data available to the operating system processes or applications through the special files /dev/random and /dev/urandom.

Reading from /dev/random drains the entropy pool with requested amount of bits/bytes, providing a high degree of randomness often desired in cryptographic operations. In case, if the entropy pool is completely drained and sufficient entropy is not available, the read operation on /dev/random blocks until additional entropy is gathered. Due to this, applications reading from /dev/random may block for some random period of time.

In contrast to the above, reading from the /dev/urandom does not block. Reading from /dev/urandom, too, drains the entropy pool but when short of sufficient entropy, it does not block but reuses the bits from the partially read random data. This is said to be susceptible to cryptanalytical attacks. This is a theorotical possibility and hence it is discouraged to read from /dev/urandom to gather randomness in cryptographic operations.

The java.security.SecureRandom class, by default, reads from the /dev/random file and hence sometimes blocks for random period of time. Now, if the read operation does not return for a required amount of time, the Oracle server times out the client (the jdbc drivers, in this case) and drops the communication by closing the socket from its end. The client when tries to resume the communication after returning from the blocking call encounters the IO exception. This problem may occur randomly on any platform, especially, where the entropy is gathered from hardware noises.

As suggested in the OTN forum, the solution to this problem is to override the default behaviour of java.security.SecureRandom class to use the non-blocking read from /dev/urandom instead of the blocking read from /dev/random. This can be done by adding the following system property -Djava.security.egd=file:///dev/urandom to the JVM. Though this is a good solution for the applications like the JDBC drivers, it is discouraged for applications that perform core cryptographic operations like crytographic key generation.

Other solutions could be to use different random seeder implementations available for the platform that do not rely on hardware noises for gathering entropy. With this, you may still require to override the default behaviour of java.security.SecureRandom.

Increasing the socket timeout on the Oracle server side can also be a solution but the side effects should be assessed from the server point of view before attempting this.

like image 120
Drona Avatar answered Sep 28 '22 08:09

Drona


I was facing exactly the same problem. With Windows Vista I could not reproduce the problem but on Ubuntu I reproduced the 'connection reset'-Error constantly.

I found http://forums.oracle.com/forums/thread.jspa?threadID=941911&tstart=0&messageID=3793101

According to a user on that forum:

I opened a ticket with Oracle and this is what they told me.

java.security.SecureRandom is a standard API provided by sun. Among various methods offered by this class void nextBytes(byte[]) is one. This method is used for generating random bytes. Oracle 11g JDBC drivers use this API to generate random number during login. Users using Linux have been encountering SQLException("Io exception: Connection reset").

The problem is two fold

  1. The JVM tries to list all the files in the /tmp (or alternate tmp directory set by -Djava.io.tmpdir) when SecureRandom.nextBytes(byte[]) is invoked. If the number of files is large the method takes a long time to respond and hence cause the server to timeout

  2. The method void nextBytes(byte[]) uses /dev/random on Linux and on some machines which lack the random number generating hardware the operation slows down to the extent of bringing the whole login process to a halt. Ultimately the the user encounters SQLException("Io exception: Connection reset")

Users upgrading to 11g can encounter this issue if the underlying OS is Linux which is running on a faulty hardware.

Cause The cause of this has not yet been determined exactly. It could either be a problem in your hardware or the fact that for some reason the software cannot read from dev/random

Solution Change the setup for your application, so you add the next parameter to the java command:

-Djava.security.egd=file:/dev/../dev/urandom

We made this change in our java.security file and it has gotten rid of the error.

which solved my problem.

like image 37
Camilla Avatar answered Sep 28 '22 08:09

Camilla