blob: 9445f5446dafaebf88dae37e0683170d2b251b26 [file] [log] [blame]
/*
* Copyright 2010 Google Inc.
*
* 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 com.google.gwt.requestfactory.server;
import com.google.gwt.autobean.server.impl.BeanMethod;
import com.google.gwt.autobean.server.impl.TypeUtils;
import com.google.gwt.autobean.shared.ValueCodex;
import com.google.gwt.requestfactory.shared.BaseProxy;
import com.google.gwt.requestfactory.shared.InstanceRequest;
import com.google.gwt.requestfactory.shared.Request;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.validation.ConstraintViolation;
import javax.validation.Validation;
import javax.validation.ValidationException;
import javax.validation.Validator;
import javax.validation.ValidatorFactory;
/**
* Implements all methods that interact with domain objects.
*/
final class ReflectiveServiceLayer extends ServiceLayerDecorator {
/*
* NB: All calls that ReflectiveServiceLayer makes to public APIs inherited
* from ServiceLayer should be made to use the instance returned from
* getTop().
*/
private static final Validator jsr303Validator;
private static final Logger log = Logger.getLogger(ServiceLayer.class.getName());
static {
Validator found;
try {
ValidatorFactory validatorFactory = Validation.buildDefaultValidatorFactory();
found = validatorFactory.getValidator();
} catch (ValidationException e) {
log.log(Level.INFO, "Unable to initialize a JSR 303 Bean Validator", e);
found = null;
}
jsr303Validator = found;
}
/**
* Linear search, but we want to handle getFoo, isFoo, and hasFoo. The result
* of this method will be cached by the ServiceLayerCache.
*/
private static Method getBeanMethod(BeanMethod methodType,
Class<?> domainType, String property) {
for (Method m : domainType.getMethods()) {
if (methodType.matches(m) && property.equals(methodType.inferName(m))) {
m.setAccessible(true);
return m;
}
}
return null;
}
@Override
public <T> T createDomainObject(Class<T> clazz) {
Throwable ex;
try {
Constructor<T> c = clazz.getConstructor();
c.setAccessible(true);
return c.newInstance();
} catch (InstantiationException e) {
return this.<T> report("Could not create a new instance of the requested type");
} catch (NoSuchMethodException e) {
return this.<T> report("The requested type is not default-instantiable");
} catch (InvocationTargetException e) {
return this.<T> report(e);
} catch (IllegalAccessException e) {
ex = e;
} catch (SecurityException e) {
ex = e;
} catch (IllegalArgumentException e) {
ex = e;
}
return this.<T> die(ex,
"Could not create a new instance of domain type %s",
clazz.getCanonicalName());
}
@Override
public Method getGetter(Class<?> domainType, String property) {
return getBeanMethod(BeanMethod.GET, domainType, property);
}
@Override
public Object getId(Object domainObject) {
return getTop().getProperty(domainObject, "id");
}
@Override
public Class<?> getIdType(Class<?> domainType) {
return getFind(domainType).getParameterTypes()[0];
}
@Override
public Object getProperty(Object domainObject, String property) {
try {
Method getter = getTop().getGetter(domainObject.getClass(), property);
if (getter == null) {
die(null, "Could not determine getter for property %s on type %s",
property, domainObject.getClass().getCanonicalName());
}
Object value = getter.invoke(domainObject);
return value;
} catch (IllegalAccessException e) {
return die(e, "Could not retrieve property %s", property);
} catch (InvocationTargetException e) {
return report(e);
}
}
@Override
public Type getRequestReturnType(Method contextMethod) {
Class<?> returnClass = contextMethod.getReturnType();
if (InstanceRequest.class.isAssignableFrom(returnClass)) {
Type[] params = TypeUtils.getParameterization(InstanceRequest.class,
contextMethod.getGenericReturnType());
assert params.length == 2;
return params[1];
} else if (Request.class.isAssignableFrom(returnClass)) {
Type param = TypeUtils.getSingleParameterization(Request.class,
contextMethod.getGenericReturnType());
return param;
} else {
return die(null, "Unknown RequestContext return type %s",
returnClass.getCanonicalName());
}
}
@Override
public Method getSetter(Class<?> domainType, String property) {
Method setter = getBeanMethod(BeanMethod.SET, domainType, property);
if (setter == null) {
setter = getBeanMethod(BeanMethod.SET_BUILDER, domainType, property);
}
return setter;
}
@Override
public Object getVersion(Object domainObject) {
return getTop().getProperty(domainObject, "version");
}
@Override
public Object invoke(Method domainMethod, Object... args) {
Throwable ex;
try {
domainMethod.setAccessible(true);
if (Modifier.isStatic(domainMethod.getModifiers())) {
return domainMethod.invoke(null, args);
} else {
Object[] realArgs = new Object[args.length - 1];
System.arraycopy(args, 1, realArgs, 0, realArgs.length);
return domainMethod.invoke(args[0], realArgs);
}
} catch (IllegalArgumentException e) {
ex = e;
} catch (IllegalAccessException e) {
ex = e;
} catch (InvocationTargetException e) {
return report(e);
}
return die(ex, "Could not invoke method %s", domainMethod.getName());
}
/**
* This implementation attempts to re-load the object from the backing store.
*/
@Override
public boolean isLive(Object domainObject) {
Object id = getTop().getId(domainObject);
return getTop().invoke(getFind(domainObject.getClass()), id) != null;
}
@Override
public <T> T loadDomainObject(Class<T> clazz, Object id) {
if (id == null) {
die(null, "Cannot invoke find method with a null id");
}
return clazz.cast(getTop().invoke(getFind(clazz), id));
}
@Override
public List<Object> loadDomainObjects(List<Class<?>> classes,
List<Object> domainIds) {
if (classes.size() != domainIds.size()) {
die(null,
"Size mismatch in paramaters. classes.size() = %d domainIds.size=%d",
classes.size(), domainIds.size());
}
List<Object> toReturn = new ArrayList<Object>(classes.size());
Iterator<Class<?>> classIt = classes.iterator();
Iterator<Object> idIt = domainIds.iterator();
while (classIt.hasNext()) {
toReturn.add(getTop().loadDomainObject(classIt.next(), idIt.next()));
}
return toReturn;
}
@Override
public void setProperty(Object domainObject, String property,
Class<?> expectedType, Object value) {
try {
Method setter = getTop().getSetter(domainObject.getClass(), property);
if (setter == null) {
die(null, "Could not locate setter for property %s in type %s",
property, domainObject.getClass().getCanonicalName());
}
setter.invoke(domainObject, value);
return;
} catch (IllegalAccessException e) {
die(e, "Could not set property %s", property);
} catch (InvocationTargetException e) {
report(e);
}
}
@Override
public <T> Set<ConstraintViolation<T>> validate(T domainObject) {
if (jsr303Validator != null) {
return jsr303Validator.validate(domainObject);
}
return Collections.emptySet();
}
private Method getFind(Class<?> clazz) {
if (clazz == null) {
return die(null, "Could not find static method with a single"
+ " parameter of a key type");
}
String searchFor = "find" + clazz.getSimpleName();
for (Method method : clazz.getMethods()) {
if (!Modifier.isStatic(method.getModifiers())) {
continue;
}
if (!searchFor.equals(method.getName())) {
continue;
}
if (method.getParameterTypes().length != 1) {
continue;
}
if (!isKeyType(method.getParameterTypes()[0])) {
continue;
}
return method;
}
return getFind(clazz.getSuperclass());
}
/**
* Returns <code>true</code> if the given class can be used as an id or
* version key.
*/
private boolean isKeyType(Class<?> domainClass) {
if (ValueCodex.canDecode(domainClass)) {
return true;
}
return BaseProxy.class.isAssignableFrom(getTop().resolveClientType(
domainClass, BaseProxy.class, true));
}
}