001 /** 002 * Copyright 2005-2013 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.krad.service.impl; 017 018 import com.thoughtworks.xstream.XStream; 019 import com.thoughtworks.xstream.converters.MarshallingContext; 020 import com.thoughtworks.xstream.converters.SingleValueConverter; 021 import com.thoughtworks.xstream.converters.reflection.ObjectAccessException; 022 import com.thoughtworks.xstream.converters.reflection.PureJavaReflectionProvider; 023 import com.thoughtworks.xstream.converters.reflection.ReflectionConverter; 024 import com.thoughtworks.xstream.converters.reflection.ReflectionProvider; 025 import com.thoughtworks.xstream.io.HierarchicalStreamWriter; 026 import com.thoughtworks.xstream.mapper.Mapper; 027 import org.apache.commons.logging.Log; 028 import org.apache.commons.logging.LogFactory; 029 import org.apache.ojb.broker.core.proxy.ListProxyDefaultImpl; 030 import org.joda.time.DateTime; 031 import org.joda.time.format.ISODateTimeFormat; 032 import org.kuali.rice.krad.service.KRADServiceLocator; 033 import org.kuali.rice.krad.service.PersistenceService; 034 import org.kuali.rice.krad.service.XmlObjectSerializerService; 035 import org.kuali.rice.krad.service.util.DateTimeConverter; 036 037 import java.lang.reflect.Field; 038 import java.util.ArrayList; 039 import java.util.Iterator; 040 041 042 /** 043 * Service implementation for the XmlObjectSerializer structure. This is the default implementation that gets 044 * delivered with Kuali. It utilizes the XStream open source libraries and framework. 045 * 046 * @author Kuali Rice Team (rice.collab@kuali.org) 047 */ 048 public class XmlObjectSerializerServiceImpl implements XmlObjectSerializerService { 049 private static final Log LOG = LogFactory.getLog(XmlObjectSerializerServiceImpl.class); 050 051 private PersistenceService persistenceService; 052 053 private XStream xstream; 054 055 public XmlObjectSerializerServiceImpl() { 056 xstream = new XStream(new ProxyAwareJavaReflectionProvider()); 057 058 // See http://xstream.codehaus.org/faq.html#Serialization_CGLIB 059 // To use a newer version of XStream we may need to do something like this: 060 // xstream = new XStream() { 061 // 062 // @Override 063 // public ReflectionProvider getReflectionProvider() { 064 // return new ProxyAwareJavaReflectionProvider(); 065 // } 066 // 067 // protected MapperWrapper wrapMapper(MapperWrapper next) { 068 // return new CGLIBMapper(next); 069 // } 070 // }; 071 // xstream.registerConverter(new CGLIBEnhancedConverter(xstream.getMapper(), xstream.getReflectionProvider())); 072 073 xstream.registerConverter(new ProxyConverter(xstream.getMapper(), xstream.getReflectionProvider() )); 074 xstream.addDefaultImplementation(ArrayList.class, ListProxyDefaultImpl.class); 075 xstream.registerConverter(new DateTimeConverter()); 076 } 077 078 /** 079 * @see org.kuali.rice.krad.service.XmlObjectSerializer#toXml(java.lang.Object) 080 */ 081 public String toXml(Object object) { 082 if ( LOG.isDebugEnabled() ) { 083 LOG.debug( "toXml(" + object + ") : \n" + xstream.toXML(object) ); 084 } 085 return xstream.toXML(object); 086 } 087 088 /** 089 * @see org.kuali.rice.krad.service.XmlObjectSerializer#fromXml(java.lang.String) 090 */ 091 public Object fromXml(String xml) { 092 if ( LOG.isDebugEnabled() ) { 093 LOG.debug( "fromXml() : \n" + xml ); 094 } 095 if ( xml != null ) { 096 xml = xml.replaceAll( "--EnhancerByCGLIB--[0-9a-f]{0,8}", "" ); 097 } 098 return xstream.fromXML(xml); 099 } 100 101 /** 102 * This custom converter only handles proxies for BusinessObjects. List-type proxies are handled by configuring XStream to treat 103 * ListProxyDefaultImpl as ArrayLists (see constructor for this service). 104 */ 105 public class ProxyConverter extends ReflectionConverter { 106 public ProxyConverter(Mapper mapper, ReflectionProvider reflectionProvider) { 107 super(mapper, reflectionProvider); 108 } 109 110 @Override 111 // since the ReflectionConverter supertype defines canConvert without using a parameterized Class type, we must declare 112 // the overridden version the same way 113 @SuppressWarnings("unchecked") 114 public boolean canConvert(Class clazz) { 115 return clazz.getName().contains("CGLIB"); 116 } 117 118 @Override 119 public void marshal(Object obj, HierarchicalStreamWriter writer, MarshallingContext context) { 120 super.marshal(getPersistenceService().resolveProxy(obj), writer, context); 121 } 122 123 // we shouldn't need an unmarshal method because all proxy metadata is taken out of the XML, so we'll reserialize as a base BO. 124 } 125 126 public class ProxyAwareJavaReflectionProvider extends PureJavaReflectionProvider { 127 128 public ProxyAwareJavaReflectionProvider() { 129 super(); 130 } 131 /** 132 * @see com.thoughtworks.xstream.converters.reflection.PureJavaReflectionProvider#visitSerializableFields(java.lang.Object, com.thoughtworks.xstream.converters.reflection.ReflectionProvider.Visitor) 133 */ 134 @Override 135 public void visitSerializableFields(Object object, Visitor visitor) { 136 for (Iterator iterator = fieldDictionary.serializableFieldsFor(object.getClass()); iterator.hasNext();) { 137 Field field = (Field) iterator.next(); 138 if (!fieldModifiersSupported(field)) { 139 continue; 140 } 141 validateFieldAccess(field); 142 Object value = null; 143 try { 144 value = field.get(object); 145 if (value != null && getPersistenceService().isProxied(value)) { 146 value = getPersistenceService().resolveProxy(value); 147 } 148 } catch (IllegalArgumentException e) { 149 throw new ObjectAccessException("Could not get field " + field.getClass() + "." + field.getName(), e); 150 } catch (IllegalAccessException e) { 151 throw new ObjectAccessException("Could not get field " + field.getClass() + "." + field.getName(), e); 152 } 153 visitor.visit(field.getName(), field.getType(), field.getDeclaringClass(), value); 154 } 155 } 156 157 } 158 159 public PersistenceService getPersistenceService() { 160 if ( persistenceService == null ) { 161 persistenceService = KRADServiceLocator.getPersistenceService(); 162 } 163 return persistenceService; 164 } 165 166 }