blob: 75cbf4aa3e3c67b4c8f8f9099785ffa63bb35372 [file] [log] [blame]
/*
* 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.bval.jsr303.xml;
import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.security.AccessController;
import java.security.PrivilegedAction;
import java.util.HashMap;
import java.util.Map;
import javax.validation.Payload;
import javax.validation.ValidationException;
import org.apache.bval.jsr303.ConstraintAnnotationAttributes;
import org.apache.bval.jsr303.util.SecureActions;
/**
* Description: Holds the information and creates an annotation proxy during xml
* parsing of validation mapping constraints. <br/>
*/
// TODO move this guy up to org.apache.bval.jsr303 or
// org.apache.bval.jsr303.model
final public class AnnotationProxyBuilder<A extends Annotation> {
private final Class<A> type;
private final Map<String, Object> elements = new HashMap<String, Object>();
/**
* Create a new AnnotationProxyBuilder instance.
*
* @param annotationType
*/
public AnnotationProxyBuilder(Class<A> annotationType) {
this.type = annotationType;
}
/**
* Create a new AnnotationProxyBuilder instance.
*
* @param annotationType
* @param elements
*/
public AnnotationProxyBuilder(Class<A> annotationType, Map<String, Object> elements) {
this(annotationType);
for (Map.Entry<String, Object> entry : elements.entrySet()) {
this.elements.put(entry.getKey(), entry.getValue());
}
}
/**
* Create a builder initially configured to create an annotation equivalent
* to <code>annot</code>.
*
* @param annot Annotation to be replicated.
*/
@SuppressWarnings("unchecked")
public AnnotationProxyBuilder(A annot) {
this((Class<A>) annot.annotationType());
// Obtain the "elements" of the annotation
final Method[] methods = doPrivileged(SecureActions.getDeclaredMethods(annot.annotationType()));
for (Method m : methods) {
if (!m.isAccessible()) {
m.setAccessible(true);
}
try {
Object value = m.invoke(annot);
this.elements.put(m.getName(), value);
} catch (IllegalArgumentException e) {
// No args, so should not happen
throw new ValidationException("Cannot access annotation " + annot + " element: " + m.getName());
} catch (IllegalAccessException e) {
throw new ValidationException("Cannot access annotation " + annot + " element: " + m.getName());
} catch (InvocationTargetException e) {
throw new ValidationException("Cannot access annotation " + annot + " element: " + m.getName());
}
}
}
/**
* Add an element to the configuration.
*
* @param elementName
* @param value
*/
public void putValue(String elementName, Object value) {
elements.put(elementName, value);
}
/**
* Get the specified element value from the current configuration.
*
* @param elementName
* @return Object value
*/
public Object getValue(String elementName) {
return elements.get(elementName);
}
/**
* Learn whether a given element has been configured.
*
* @param elementName
* @return <code>true</code> if an <code>elementName</code> element is found
* on this annotation
*/
public boolean contains(String elementName) {
return elements.containsKey(elementName);
}
/**
* Get the number of configured elements.
*
* @return int
*/
public int size() {
return elements.size();
}
/**
* Get the configured Annotation type.
*
* @return Class<A>
*/
public Class<A> getType() {
return type;
}
/**
* Configure the well-known JSR303 "message" element.
*
* @param message
*/
public void setMessage(String message) {
ConstraintAnnotationAttributes.MESSAGE.put(elements, message);
}
/**
* Configure the well-known JSR303 "groups" element.
*
* @param groups
*/
public void setGroups(Class<?>[] groups) {
ConstraintAnnotationAttributes.GROUPS.put(elements, groups);
}
/**
* Configure the well-known JSR303 "payload" element.
*
* @param payload
*/
public void setPayload(Class<? extends Payload>[] payload) {
ConstraintAnnotationAttributes.PAYLOAD.put(elements, payload);
}
/**
* Create the annotation represented by this builder.
*
* @return {@link Annotation}
*/
public A createAnnotation() {
ClassLoader classLoader = SecureActions.getClassLoader(getType());
@SuppressWarnings("unchecked")
final Class<A> proxyClass = (Class<A>) Proxy.getProxyClass(classLoader, getType());
final InvocationHandler handler = new AnnotationProxy(this);
return doPrivileged(new PrivilegedAction<A>() {
public A run() {
try {
Constructor<A> constructor = proxyClass.getConstructor(InvocationHandler.class);
return constructor.newInstance(handler);
} catch (Exception e) {
throw new ValidationException("Unable to create annotation for configured constraint", e);
}
}
});
}
private static <T> T doPrivileged(final PrivilegedAction<T> action) {
if (System.getSecurityManager() != null) {
return AccessController.doPrivileged(action);
} else {
return action.run();
}
}
}