org.directwebremoting.hibernate.H3BeanConverter Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of dwr Show documentation
Show all versions of dwr Show documentation
DWR is easy Ajax for Java. It makes it simple to call Java code directly from Javascript.
It gets rid of almost all the boiler plate code between the web browser and your Java code.
The newest version!
/*
* Copyright 2005 Joe Walker
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.directwebremoting.hibernate;
import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.directwebremoting.ConversionException;
import org.directwebremoting.convert.BeanConverter;
import org.directwebremoting.extend.PlainProperty;
import org.directwebremoting.extend.Property;
import org.hibernate.Hibernate;
import org.hibernate.engine.SessionImplementor;
import org.hibernate.proxy.HibernateProxy;
import org.hibernate.proxy.LazyInitializer;
/**
* BeanConverter that works with Hibernate to get BeanInfo.
* @author Joe Walker [joe at getahead dot ltd dot uk]
*/
public class H3BeanConverter extends BeanConverter
{
/* (non-Javadoc)
* @see org.directwebremoting.convert.BeanConverter#getPropertyMapFromObject(java.lang.Object, boolean, boolean)
*/
@Override
public Map getPropertyMapFromObject(Object example, boolean readRequired, boolean writeRequired) throws ConversionException
{
Class> clazz = getClass(example);
try
{
BeanInfo info = Introspector.getBeanInfo(clazz);
PropertyDescriptor[] descriptors = info.getPropertyDescriptors();
Map properties = new HashMap();
for (PropertyDescriptor descriptor : descriptors)
{
String name = descriptor.getName();
// We don't marshall getClass()
if ("class".equals(name))
{
continue;
}
// And this is something added by hibernate
if ("hibernateLazyInitializer".equals(name))
{
continue;
}
// Access rules mean we might not want to do this one
if (!isAllowedByIncludeExcludeRules(name))
{
continue;
}
if (readRequired && descriptor.getReadMethod() == null)
{
continue;
}
if (writeRequired && descriptor.getWriteMethod() == null)
{
continue;
}
if (!assumeSession)
{
// We don't marshall un-initialized properties for
// Hibernate3
Method method = findGetter(example, name);
if (method == null)
{
log.warn("Failed to find property: " + name);
properties.put(name, new PlainProperty(name, null));
continue;
}
if (!Hibernate.isPropertyInitialized(example, name))
{
properties.put(name, new PlainProperty(name, null));
continue;
}
// This might be a lazy-collection so we need to double check
Object retval = method.invoke(example);
if (!Hibernate.isInitialized(retval))
{
properties.put(name, new PlainProperty(name, null));
continue;
}
}
properties.put(name, new H3PropertyDescriptorProperty(descriptor));
}
return properties;
}
catch (Exception ex)
{
throw new ConversionException(clazz, ex);
}
}
/**
* Hibernate makes {@link Class#getClass()} difficult ...
* @param example The class that we want to call {@link Class#getClass()} on
* @return The type of the given object
*/
public Class> getClass(Object example)
{
if (example instanceof HibernateProxy)
{
HibernateProxy proxy = (HibernateProxy) example;
LazyInitializer initializer = proxy.getHibernateLazyInitializer();
SessionImplementor implementor = initializer.getSession();
if (initializer.isUninitialized())
{
try
{
// getImplementation is going to want to talk to a session
if (implementor.isClosed())
{
// Give up and return example.getClass();
return example.getClass();
}
}
catch (NoSuchMethodError ex)
{
// We must be using Hibernate 3.0/3.1 which doesn't have
// this method
}
}
return initializer.getImplementation().getClass();
}
else
{
return example.getClass();
}
}
/**
* Cache the method if possible, using the classname and property name to
* allow for similar named methods.
* @param data The bean to introspect
* @param property The property to get the accessor for
* @return The getter method
* @throws IntrospectionException If Introspector.getBeanInfo() fails
*/
protected Method findGetter(Object data, String property) throws IntrospectionException
{
Class> clazz = getClass(data);
String key = clazz.getName() + ":" + property;
Method method = methods.get(key);
if (method == null)
{
Method newMethod = null;
PropertyDescriptor[] props = Introspector.getBeanInfo(clazz).getPropertyDescriptors();
for (PropertyDescriptor prop : props)
{
if (prop.getName().equalsIgnoreCase(property))
{
newMethod = prop.getReadMethod();
}
}
method = methods.putIfAbsent(key, newMethod);
if (method == null) {
// put succeeded, use new value
method = newMethod;
}
}
return method;
}
/**
* @param assumeSession the assumeSession to set
*/
public void setAssumeSession(boolean assumeSession)
{
this.assumeSession = assumeSession;
}
/**
* Do we assume there is an open session and read properties?
*/
protected boolean assumeSession = false;
/**
* The cache of method lookups that we've already done
*/
private final ConcurrentMap methods = new ConcurrentHashMap();
/**
* The log stream
*/
private static final Log log = LogFactory.getLog(H3BeanConverter.class);
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy