views:

2484

answers:

6

Hi everyone,

I'm trying to access a data source that is defined within a web container (JBoss) from a fat client outside the container.

I've decided to look up the data source through JNDI. Actually, my persistence framework (Ibatis) does this.

When performing queries I always end up getting this error:

java.lang.IllegalAccessException: Method=public abstract java.sql.Connection java.sql.Statement.getConnection() throws java.sql.SQLException does not return Serializable 

Stacktrace:
org.jboss.resource.adapter.jdbc.remote.WrapperDataSourceService.doStatementMethod(WrapperDataSourceS
ervice.java:411),
org.jboss.resource.adapter.jdbc.remote.WrapperDataSourceService.invoke(WrapperDataSourceService.java
:223),
sun.reflect.GeneratedMethodAccessor106.invoke(Unknown Source),
sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25),
java.lang.reflect.Method.invoke(Method.java:585),
org.jboss.mx.interceptor.ReflectedDispatcher.invoke(ReflectedDispatcher.java:155),
org.jboss.mx.server.Invocation.dispatch(Invocation.java:94),
org.jboss.mx.server.Invocation.invoke(Invocation.java:86),
org.jboss.mx.server.AbstractMBeanInvoker.invoke(AbstractMBeanInvoker.java:264),
org.jboss.mx.server.MBeanServerImpl.invoke(MBeanServerImpl.java:659),

My Datasource:

<?xml version="1.0" encoding="UTF-8"?>
<datasources>
  <local-tx-datasource>
        <jndi-name>jdbc/xxxxxDS</jndi-name>
        <connection-url>jdbc:oracle:thin:@xxxxxxxxx:1521:xxxxxxx</connection-url>
  <use-java-context>false</use-java-context>
        <driver-class>oracle.jdbc.driver.OracleDriver</driver-class>
        <user-name>xxxxxxxx</user-name>
        <password>xxxxxx</password>
        <exception-sorter-class-name>org.jboss.resource.adapter.jdbc.vendor.OracleExceptionSorter</exception-sorter-class-name>
        <min-pool-size>5</min-pool-size>
        <max-pool-size>20</max-pool-size>
    </local-tx-datasource>
</datasources>

Does anyone have a clue where this could come from?

Maybe someone even knows a better way how to achieve this. Any hints are much appreciated!

Cheers,

Michael

A: 

I think the exception indicates that the SQLConnection object you're trying to retrieve doesn't implement the Serializable interface, so it can't be passed to you the way you asked for it.

From the limited work I've done with JDNI, if you're asking for an object via JNDI it must be serializable. As far as I know, there's no way round that - if I think of a better way I'll post it up...

OK, one obvious option is to provide a serializable object local to the datasource that uses it but doesn't have the datasource as part of its serializable object graph. The fat client could then look up that object and query it instead.

Or create a (web?) service through which to access the datasource is governed - again your fat client would hit the service - this would probably be better encapsulated and more reuseable approach if those are concerns for you.

Brabster
+1  A: 

Not sure if this is the same issue?

JBoss DataSource config

DataSource wrappers are not usable outside of the server VM

toolkit
A: 

@toolkit: Well, not exactly. Since I can access the data source over JNDI, it is actually visible and thus usable.

Or am I getting something totally wrong?

@Brabster: I think you're on the right track. Isn't there a way to make the connection serializable? Maybe it's just a configuration issue...

+1  A: 

@Michael Well, java.sql.Connection is an Interface - it might technically be possible for the concrete implementation you're getting from JBoss to be Serializable - but I don't think you're really going to have any options you can use. If it was possible, it would probably be easy :)

I think @toolkit might have said the right words with useable outside the VM - the JDBC drivers will be talking to native driver code running in the underlying OS I guess, so that might explain why you can't just pass a connection over the network elsewhere.

My advice, (if you don't get any better advice!) would be to find a different approach - if you have access to locate the resource on the JBoss directory, maybe implement a proxy object that you can locate and obtain from the directory that allows you to use the connection remotely from your fat client. That's a design pattern called data transfer object I think Wikipedia entry

Brabster
A: 

I've read up on Ibatis now - maybe you can make your implementations of Dao etc. Serializable, post them into your directory and so retrieve them and use them in your fat client? You'd get reuse benefits out of that too.

Here's an example of something looks similar for Wicket

Brabster
A: 

JBoss wraps up all DataSources with it's own ones.

That lets it play tricks with autocommit to get the specified J2EE behaviour out of a JDBC connection. They are mostly serailizable. But you needn't trust them.

I'd look carefully at it's wrappers. I've written a surrogate for JBoss's J2EE wrappers wrapper for JDBC that works with OOCJNDI to get my DAO code unit test-able standalone.

You just wrap java.sql.Driver, point OOCJNDI at your class, and run in JUnit.

The Driver wrapper can just directly create a SQL Driver and delegate to it.

Return a java.sql.Connection wrapper of your own devising on Connect.

A ConnectionWrapper can just wrap the Connection your Oracle driver gives you, and all it does special is set Autocommit true.

Don't forget Eclipse can wrt delgates for you. Add a member you need to delegate to , then select it and right click, source -=>add delgage methods.

This is great when you get paid by the line ;-)

Bada-bing, Bada-boom, JUnit out of the box J2EE testing.

Your problem is probably amenable to the same thing, with JUnit crossed out and FatCLient written in an crayon.

My FatClient uses RMI generated with xdoclet to talk to the J2EE server, so I don't have your problem.

Tim Williscroft