/*
* Copyright 2013 Chris Pheby
*
* 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.jadira.reflection.cloning.implementor;
import java.lang.reflect.Field;
import java.util.IdentityHashMap;
import org.jadira.reflection.access.model.ClassModel;
import org.jadira.reflection.access.model.FieldModel;
import org.jadira.reflection.access.portable.PortableClassAccess;
import org.jadira.reflection.cloning.api.CloneDriver;
import org.jadira.reflection.cloning.api.CloneStrategy;
import org.objenesis.Objenesis;
import org.objenesis.ObjenesisStd;
/**
* A CloneStrategy that uses reflection
*/
public class PortableCloneStrategy extends AbstractCloneStrategy implements CloneStrategy {
private static final char CHAR_NULL = '\u0000';
private static final Objenesis objenesis = new ObjenesisStd();
@SuppressWarnings("unchecked")
@Override
public <T> T newInstance(Class<T> c) {
return (T) objenesis.newInstance(c);
}
private static PortableCloneStrategy instance = new PortableCloneStrategy();
/**
* Creates a new instance
*/
public PortableCloneStrategy() {
}
/**
* Returns a shared instance of PortableCloneStrategy
* @return The instance
*/
public static PortableCloneStrategy getInstance() {
return instance;
}
@Override
protected <T> void handleTransientField(T copy, FieldModel<T> f) {
Class<?> clazz = f.getField().getType();
try {
if (clazz.isPrimitive()) {
if (java.lang.Boolean.TYPE == clazz) {
f.getField().setBoolean(copy, false);
} else if (java.lang.Byte.TYPE == clazz) {
f.getField().setByte(copy, (byte) 0);
} else if (java.lang.Character.TYPE == clazz) {
f.getField().setChar(copy, CHAR_NULL);
} else if (java.lang.Short.TYPE == clazz) {
f.getField().setShort(copy, (short) 0);
} else if (java.lang.Integer.TYPE == clazz) {
f.getField().setInt(copy, 0);
} else if (java.lang.Long.TYPE == clazz) {
f.getField().setLong(copy, 0L);
} else if (java.lang.Float.TYPE == clazz) {
f.getField().setFloat(copy, 0.0F);
} else if (java.lang.Double.TYPE == clazz) {
f.getField().setDouble(copy, 0.0D);
}
} else {
f.getField().set(copy, null);
}
} catch (IllegalArgumentException e) {
throw new IllegalStateException("Problem performing clone for field {" + f.getField().getName() + "} of object {" + System.identityHashCode(copy) + "}: " + e.getMessage(), e);
} catch (IllegalAccessException e) {
throw new IllegalStateException("Problem performing clone for field {" + f.getField().getName() + "} of object {" + System.identityHashCode(copy) + "}: " + e.getMessage(), e);
}
}
@Override
protected <W> ClassModel<W> getClassModel(Class<W> clazz) {
return PortableClassAccess.get(clazz).getClassModel();
}
@Override
protected <T> void handleClonePrimitiveField(T obj, T copy, CloneDriver driver, FieldModel<T> f, IdentityHashMap<Object, Object> referencesToReuse) {
Field field = f.getField();
try {
Class<?> clazz = f.getFieldClass();
if (java.lang.Boolean.TYPE == clazz) {
field.setBoolean(copy, field.getBoolean(obj));
} else if (java.lang.Byte.TYPE == clazz) {
field.setByte(copy, field.getByte(obj));
} else if (java.lang.Character.TYPE == clazz) {
field.setChar(copy, field.getChar(obj));
} else if (java.lang.Short.TYPE == clazz) {
field.setShort(copy, field.getShort(obj));
} else if (java.lang.Integer.TYPE == clazz) {
field.setInt(copy, field.getInt(obj));
} else if (java.lang.Long.TYPE == clazz) {
field.setLong(copy, field.getLong(obj));
} else if (java.lang.Float.TYPE == clazz) {
field.setFloat(copy, field.getFloat(obj));
} else if (java.lang.Double.TYPE == clazz) {
field.setDouble(copy, field.getDouble(obj));
} else {
throw new IllegalStateException("Expected primitive but was :" + clazz.getName());
}
} catch (IllegalArgumentException e) {
throw new IllegalStateException("Problem performing clone for field {" + field.getName() + "} of object {" + System.identityHashCode(obj) + "}: " + e.getMessage(), e);
} catch (IllegalAccessException e) {
throw new IllegalStateException("Problem performing clone for field {" + field.getName() + "} of object {" + System.identityHashCode(obj) + "}: " + e.getMessage(), e);
}
}
@Override
protected <T> Object getFieldValue(T obj, FieldModel<T> f) {
final Field field = f.getField();
final Object fieldObject;
try {
fieldObject = field.get(obj);
} catch (IllegalArgumentException e) {
throw new IllegalStateException("Problem performing clone for field {" + field.getName() + "} of object {" + System.identityHashCode(obj) + "}: " + e.getMessage(), e);
} catch (IllegalAccessException e) {
throw new IllegalStateException("Problem performing clone for field {" + field.getName() + "} of object {" + System.identityHashCode(obj) + "}: " + e.getMessage(), e);
}
return fieldObject;
}
@Override
protected <T> void putFieldValue(T obj, FieldModel<T> f, Object value) {
final Field field = f.getField();
try {
field.set(obj, value);
} catch (IllegalArgumentException e) {
throw new IllegalStateException("Problem performing clone for field {" + field.getName() + "} of object {" + System.identityHashCode(obj) + "}: " + e.getMessage(), e);
} catch (IllegalAccessException e) {
throw new IllegalStateException("Problem performing clone for field {" + field.getName() + "} of object {" + System.identityHashCode(obj) + "}: " + e.getMessage(), e);
}
}
}