![JAR search and dependency download from the Maven repository](/logo.png)
org.apache.juneau.swap.SurrogateSwap Maven / Gradle / Ivy
// ***************************************************************************************************************************
// * Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file *
// * distributed with this work for additional information regarding copyright ownership. The ASF licenses this file *
// * to you 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.apache.juneau.swap;
import static org.apache.juneau.internal.ClassUtils.*;
import java.lang.reflect.*;
import java.util.*;
import org.apache.juneau.*;
import org.apache.juneau.annotation.*;
import org.apache.juneau.parser.*;
import org.apache.juneau.reflect.*;
import org.apache.juneau.serializer.*;
/**
* Specialized {@link ObjectSwap} for {@link Surrogate} classes.
*
* See Also:
* - Swaps
*
*
* @param The class type that this transform applies to.
* @param The transformed class type.
*/
public class SurrogateSwap extends ObjectSwap {
private Constructor constructor; // public F(T t);
private Method unswapMethod; // public T build();
/**
* Constructor.
*
* @param forClass The normal class.
* @param constructor The constructor on the surrogate class that takes the normal class as a parameter.
* @param unswapMethod The static method that converts surrogate objects into normal objects.
*/
protected SurrogateSwap(Class forClass, Constructor constructor, Method unswapMethod) {
super(forClass, constructor.getDeclaringClass());
this.constructor = constructor;
this.unswapMethod = unswapMethod;
}
/**
* Given the specified surrogate class, return the list of object swaps.
*
*
* A transform is returned for each public 1-arg constructor found.
* Returns an empty list if no public 1-arg constructors are found.
*
* @param c The surrogate class.
* @param bc The bean context to use for looking up annotations.
* @return The list of object swaps that apply to this class.
*/
@SuppressWarnings({"unchecked", "rawtypes"})
public static List> findObjectSwaps(Class> c, BeanContext bc) {
List> l = new LinkedList<>();
ClassInfo ci = ClassInfo.of(c);
ci.forEachPublicConstructor(
x -> x.hasNoAnnotation(bc, BeanIgnore.class) && x.hasNumParams(1) && x.isPublic(),
x -> {
Class> pt = x.getRawParamType(0);
if (! pt.equals(c.getDeclaringClass())) {
// Find the unswap method if there is one.
MethodInfo mi = ci.getPublicMethod(y -> y.hasReturnType(pt));
Method unswapMethod = mi != null ? mi.inner() : null;
l.add(new SurrogateSwap(pt, x.inner(), unswapMethod));
}
}
);
return l;
}
@Override /* ObjectSwap */
public F swap(BeanSession session, T o) throws SerializeException {
try {
return constructor.newInstance(o);
} catch (Exception e) {
throw new SerializeException(e);
}
}
@Override /* ObjectSwap */
@SuppressWarnings("unchecked")
public T unswap(BeanSession session, F f, ClassMeta> hint) throws ParseException {
if (unswapMethod == null)
throw new ParseException("unswap() method not implement on surrogate class ''{1}'': {0}", className(f), getNormalClass().getFullName());
try {
return (T)unswapMethod.invoke(f);
} catch (Exception e) {
throw new ParseException(e);
}
}
}