001// Copyright 2008-2013 The Apache Software Foundation 002// 003// Licensed under the Apache License, Version 2.0 (the "License"); 004// you may not use this file except in compliance with the License. 005// You may obtain a copy of the License at 006// 007// http://www.apache.org/licenses/LICENSE-2.0 008// 009// Unless required by applicable law or agreed to in writing, software 010// distributed under the License is distributed on an "AS IS" BASIS, 011// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 012// See the License for the specific language governing permissions and 013// limitations under the License. 014 015package org.apache.tapestry5.internal.hibernate; 016 017import org.apache.tapestry5.ValueEncoder; 018import org.apache.tapestry5.ioc.internal.util.InternalUtils; 019import org.apache.tapestry5.ioc.services.PropertyAccess; 020import org.apache.tapestry5.ioc.services.PropertyAdapter; 021import org.apache.tapestry5.ioc.services.TypeCoercer; 022import org.apache.tapestry5.ioc.util.ExceptionUtils; 023import org.hibernate.Session; 024import org.hibernate.mapping.Property; 025import org.slf4j.Logger; 026 027import java.io.Serializable; 028 029public final class HibernateEntityValueEncoder<E> implements ValueEncoder<E> 030{ 031 private final Class<E> entityClass; 032 033 private final Session session; 034 035 private final TypeCoercer typeCoercer; 036 037 private final PropertyAdapter propertyAdapter; 038 039 private final Logger logger; 040 041 public HibernateEntityValueEncoder(Class<E> entityClass, String identifierPropertyName, Session session, 042 PropertyAccess propertyAccess, TypeCoercer typeCoercer, Logger logger) 043 { 044 this.entityClass = entityClass; 045 this.session = session; 046 this.typeCoercer = typeCoercer; 047 this.logger = logger; 048 049 propertyAdapter = propertyAccess.getAdapter(this.entityClass).getPropertyAdapter(identifierPropertyName); 050 } 051 052 @Override 053 public String toClient(E value) 054 { 055 if (value == null) 056 return null; 057 058 Object id = propertyAdapter.get(value); 059 060 if (id == null) 061 { 062 return null; 063 } 064 065 return typeCoercer.coerce(id, String.class); 066 } 067 068 @Override 069 @SuppressWarnings("unchecked") 070 public E toValue(String clientValue) 071 { 072 if (InternalUtils.isBlank(clientValue)) 073 return null; 074 075 Object id = null; 076 077 try 078 { 079 080 id = typeCoercer.coerce(clientValue, propertyAdapter.getType()); 081 } catch (Exception ex) 082 { 083 throw new RuntimeException(String.format( 084 "Exception converting '%s' to instance of %s (id type for entity %s): %s", clientValue, 085 propertyAdapter.getType().getName(), entityClass.getName(), ExceptionUtils.toMessage(ex)), ex); 086 } 087 088 Serializable ser = (Serializable) id; 089 090 E result = (E) session.get(entityClass, ser); 091 092 if (result == null) 093 { 094 // We don't identify the entity type in the message because the logger is based on the 095 // entity type. 096 logger.error(String.format("Unable to convert client value '%s' into an entity instance.", clientValue)); 097 } 098 099 return result; 100 } 101 102}