/* * Hibernate, Relational Persistence for Idiomatic Java * * License: GNU Lesser General Public License (LGPL), version 2.1 or later. * See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>. */ package org.hibernate.type.descriptor.converter; import java.sql.CallableStatement; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; import javax.persistence.AttributeConverter; import javax.persistence.PersistenceException; import org.hibernate.type.descriptor.ValueBinder; import org.hibernate.type.descriptor.ValueExtractor; import org.hibernate.type.descriptor.WrapperOptions; import org.hibernate.type.descriptor.java.JavaTypeDescriptor; import org.hibernate.type.descriptor.sql.SqlTypeDescriptor; import org.jboss.logging.Logger; /** * Adapter for incorporating JPA {@link AttributeConverter} handling into the SqlTypeDescriptor contract. * <p/> * Essentially this is responsible for mapping to/from the intermediate database type representation. Continuing the * {@code AttributeConverter<Integer,String>} example from * {@link org.hibernate.mapping.SimpleValue#buildAttributeConverterTypeAdapter()}, the "intermediate database type * representation" would be the String representation. So on binding, we convert the incoming Integer to String; * on extraction we extract the value as String and convert to Integer. * * @author Steve Ebersole */ public class AttributeConverterSqlTypeDescriptorAdapter implements SqlTypeDescriptor { private static final Logger log = Logger.getLogger( AttributeConverterSqlTypeDescriptorAdapter.class ); private final AttributeConverter converter; private final SqlTypeDescriptor delegate; private final JavaTypeDescriptor intermediateJavaTypeDescriptor; public AttributeConverterSqlTypeDescriptorAdapter( AttributeConverter converter, SqlTypeDescriptor delegate, JavaTypeDescriptor intermediateJavaTypeDescriptor) { this.converter = converter; this.delegate = delegate; this.intermediateJavaTypeDescriptor = intermediateJavaTypeDescriptor; } @Override public int getSqlType() { return delegate.getSqlType(); } @Override public boolean canBeRemapped() { // any remapping of the underlying SqlTypeDescriptor should have // happened prior to it being passed to us. return false; } // Binding ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ @Override @SuppressWarnings("unchecked") public <X> ValueBinder<X> getBinder(JavaTypeDescriptor<X> javaTypeDescriptor) { // Get the binder for the intermediate type representation final ValueBinder realBinder = delegate.getBinder( intermediateJavaTypeDescriptor ); return new ValueBinder<X>() { @Override public void bind(PreparedStatement st, X value, int index, WrapperOptions options) throws SQLException { final Object convertedValue; try { convertedValue = converter.convertToDatabaseColumn( value ); } catch (PersistenceException pe) { throw pe; } catch (RuntimeException re) { throw new PersistenceException( "Error attempting to apply AttributeConverter", re ); } log.debugf( "Converted value on binding : %s -> %s", value, convertedValue ); realBinder.bind( st, convertedValue, index, options ); } @Override public void bind(CallableStatement st, X value, String name, WrapperOptions options) throws SQLException { final Object convertedValue; try { convertedValue = converter.convertToDatabaseColumn( value ); } catch (PersistenceException pe) { throw pe; } catch (RuntimeException re) { throw new PersistenceException( "Error attempting to apply AttributeConverter", re ); } log.debugf( "Converted value on binding : %s -> %s", value, convertedValue ); realBinder.bind( st, convertedValue, name, options ); } }; } // Extraction ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ @Override public <X> ValueExtractor<X> getExtractor(JavaTypeDescriptor<X> javaTypeDescriptor) { // Get the extractor for the intermediate type representation final ValueExtractor realExtractor = delegate.getExtractor( intermediateJavaTypeDescriptor ); return new ValueExtractor<X>() { @Override public X extract(ResultSet rs, String name, WrapperOptions options) throws SQLException { return doConversion( realExtractor.extract( rs, name, options ) ); } @Override public X extract(CallableStatement statement, int index, WrapperOptions options) throws SQLException { return doConversion( realExtractor.extract( statement, index, options ) ); } @Override public X extract(CallableStatement statement, String[] paramNames, WrapperOptions options) throws SQLException { if ( paramNames.length > 1 ) { throw new IllegalArgumentException( "Basic value extraction cannot handle multiple output parameters" ); } return doConversion( realExtractor.extract( statement, paramNames, options ) ); } @SuppressWarnings("unchecked") private X doConversion(Object extractedValue) { try { X convertedValue = (X) converter.convertToEntityAttribute( extractedValue ); log.debugf( "Converted value on extraction: %s -> %s", extractedValue, convertedValue ); return convertedValue; } catch (PersistenceException pe) { throw pe; } catch (RuntimeException re) { throw new PersistenceException( "Error attempting to apply AttributeConverter", re ); } } }; } }