/** * The MIT License * * Copyright (c) 2010-2011 Sonatype, Inc. All rights reserved. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package org.hudsonci.utils.marshal.xref; import org.hudsonci.utils.marshal.xref.XReference.InstanceHolder; import com.thoughtworks.xstream.converters.ConversionException; import com.thoughtworks.xstream.converters.MarshallingContext; import com.thoughtworks.xstream.converters.UnmarshallingContext; import com.thoughtworks.xstream.converters.reflection.AbstractReflectionConverter; import com.thoughtworks.xstream.converters.reflection.ReflectionProvider; import com.thoughtworks.xstream.io.HierarchicalStreamReader; import com.thoughtworks.xstream.io.HierarchicalStreamWriter; import com.thoughtworks.xstream.mapper.Mapper; import java.io.IOException; import java.lang.ref.SoftReference; import static com.google.common.base.Preconditions.checkNotNull; /** * Converter for {@link XReference} types. * * @author <a href="mailto:jason@planet57.com">Jason Dillon</a> * @since 2.1.0 */ public abstract class XReferenceConverter extends AbstractReflectionConverter { protected HolderType holderType = HolderType.HARD; public XReferenceConverter(final Mapper mapper, final ReflectionProvider reflection) { super(mapper, reflection); } public HolderType getHolderType() { return holderType; } public void setHolderType(final HolderType type) { this.holderType = checkNotNull(type); } public boolean canConvert(final Class type) { return XReference.class.isAssignableFrom(type); } @Override protected void doMarshal(final Object source, final HierarchicalStreamWriter writer, final MarshallingContext context) { // Do the default marshalling for the reference container super.doMarshal(source, writer, context); // Then delegate the storage of the reference target XReference ref = (XReference)source; Object target = ref.get(); if (target != null) { try { store(ref); ref.holder = createStoredHolder(ref, target); } catch (Exception e) { throw new ConversionException("Failed to marshal reference: " + ref, e); } } } @Override public Object doUnmarshal(final Object result, final HierarchicalStreamReader reader, final UnmarshallingContext context) { // Do the default unmarshalling for the reference container XReference ref = (XReference) super.doUnmarshal(result, reader, context); ref.holder = createUnmarshalHolder(ref); return ref; } /** * Provides reference storing behavior. */ protected abstract void store(final XReference ref) throws IOException; /** * Provides reference loading behavior. */ protected abstract Object load(final XReference ref) throws IOException; /** * Create the holder to be used after the reference has been stored. */ @SuppressWarnings({"unchecked"}) protected XReference.Holder createStoredHolder(final XReference ref, final Object target) { switch (holderType) { case HARD: return new InstanceHolder(target); case SOFT: return new SoftUnmarshalHolder(ref, target); default: throw new Error(); } } /** * Create the holder to be used after the reference has been unmarshalled. */ protected XReference.Holder createUnmarshalHolder(final XReference ref) { switch (holderType) { case HARD: return new UnmarshalHolder(ref); case SOFT: return new SoftUnmarshalHolder(ref); default: throw new Error(); } } /** * Support for {@link XReference.Holder} implementations. */ protected abstract class HolderSupport implements XReference.Holder { protected final XReference ref; protected HolderSupport(final XReference ref) { this.ref = checkNotNull(ref); } protected Object doLoad() { try { return load(ref); } catch (Exception e) { throw new ConversionException("Failed to unmarshal reference: " + ref, e); } } } /** * Default holder types. */ public static enum HolderType { /** * Use hard references. */ HARD, /** * Use soft references. */ SOFT } /** * Unmarshalling holder with a hard reference. */ protected class UnmarshalHolder extends HolderSupport { protected Object instance; protected UnmarshalHolder(final XReference ref) { super(ref); } public Object get() { if (instance == null) { instance = doLoad(); } return instance; } @Override public String toString() { return "UnmarshalHolder{" + "instance=" + instance + '}'; } } /** * Unmarshalling holder with a soft reference. */ @SuppressWarnings({"unchecked"}) protected class SoftUnmarshalHolder extends HolderSupport { protected SoftReference instance; protected SoftUnmarshalHolder(final XReference ref) { super(ref); } protected SoftUnmarshalHolder(final XReference ref, final Object target) { super(ref); checkNotNull(target); this.instance = new SoftReference(target); } public Object get() { Object target; if (instance == null || (target = instance.get()) == null) { target = doLoad(); instance = new SoftReference(target); } return target; } @Override public String toString() { return "SoftUnmarshalHolder{" + "instance=" + instance + '}'; } } }