Package org.hibernate.validation.util.annotationfactory

Source Code of org.hibernate.validation.util.annotationfactory.AnnotationProxy

// $Id: AnnotationProxy.java 17263 2009-08-11 18:00:25Z epbernard $
/*
* JBoss, Home of Professional Open Source
* Copyright 2008, Red Hat Middleware LLC, and individual contributors
* by the @authors tag. See the copyright.txt in the distribution for a
* full listing of individual contributors.
*
* 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.hibernate.validation.util.annotationfactory;

import java.lang.annotation.Annotation;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.util.Comparator;
import java.util.HashMap;
import java.util.Map;
import java.util.SortedSet;
import java.util.TreeSet;
import java.security.AccessController;

import org.hibernate.validation.util.GetDeclaredMethods;

/**
* A concrete implementation of <code>Annotation</code> that pretends it is a
* "real" source code annotation. It's also an <code>InvocationHandler</code>.
* <p/>
* When you create an <code>AnnotationProxy</code>, you must initialize it
* with an <code>AnnotationDescriptor</code>.
* The adapter checks that the provided elements are the same elements defined
* in the annotation interface. However, it does <i>not</i> check that their
* values are the right type. If you omit an element, the adapter will use the
* default value for that element from the annotation interface, if it exists.
* If no default exists, it will throw an exception.
* <p/>
* Warning: this class does not implement <code>hashCode()</code> and
* <code>equals()</code> - it just uses the ones it inherits from <code>Object</code>.
* This means that an <code>AnnotationProxy</code> does <i>not</i> follow the
* recommendations of the <code>Annotation</code> javadoc about these two
* methods. That's why you should never mix <code>AnnotationProxies</code>
* with "real" annotations. For example, don't put them into the same
* <code>Collection</code>.
*
* @author Paolo Perrotta
* @author Davide Marchignoli
* @see java.lang.annotation.Annotation
*/
public class AnnotationProxy implements Annotation, InvocationHandler {

  private final Class<? extends Annotation> annotationType;
  //FIXME it's probably better to use String as a key rather than Method
  //      to speed up and avoid any fancy permsize/GC issue
  //      I'd better check the litterature on the subject
  private final Map<Method, Object> values;

  public AnnotationProxy(AnnotationDescriptor descriptor) {
    this.annotationType = descriptor.type();
    values = getAnnotationValues( descriptor );
  }

  private Map<Method, Object> getAnnotationValues(AnnotationDescriptor descriptor) {
    Map<Method, Object> result = new HashMap<Method, Object>();
    int processedValuesFromDescriptor = 0;
    GetDeclaredMethods action = GetDeclaredMethods.action( annotationType );
    final Method[] declaredMethods;
    if ( System.getSecurityManager() != null ) {
      declaredMethods = AccessController.doPrivileged( action );
    }
    else {
      declaredMethods = action.run();
    }
    for ( Method m : declaredMethods ) {
      if ( descriptor.containsElement( m.getName() ) ) {
        result.put( m, descriptor.valueOf( m.getName() ) );
        processedValuesFromDescriptor++;
      }
      else if ( m.getDefaultValue() != null ) {
        result.put( m, m.getDefaultValue() );
      }
      else {
        throw new IllegalArgumentException( "No value provided for " + m.getName() );
      }
    }
    if ( processedValuesFromDescriptor != descriptor.numberOfElements() ) {
      throw new RuntimeException( "Trying to instanciate " + annotationType + " with unknown paramters." );
    }
    return result;
  }

  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    if ( values.containsKey( method ) ) {
      return values.get( method );
    }
    return method.invoke( this, args );
  }

  public Class<? extends Annotation> annotationType() {
    return annotationType;
  }

  public String toString() {
    StringBuilder result = new StringBuilder();
    result.append( '@' ).append( annotationType().getName() ).append( '(' );
    for ( Method m : getRegisteredMethodsInAlphabeticalOrder() ) {
      result.append( m.getName() ).append( '=' ).append( values.get( m ) ).append( ", " );
    }
    // remove last separator:
    if ( values.size() > 0 ) {
      result.delete( result.length() - 2, result.length() );
      result.append( ")" );
    }
    else {
      result.delete( result.length() - 1, result.length() );
    }

    return result.toString();
  }

  private SortedSet<Method> getRegisteredMethodsInAlphabeticalOrder() {
    SortedSet<Method> result = new TreeSet<Method>(
        new Comparator<Method>() {
          public int compare(Method o1, Method o2) {
            return o1.getName().compareTo( o2.getName() );
          }
        }
    );
    //List<Method> result = new LinkedList<Method>();
    result.addAll( values.keySet() );
    return result;
  }
}
TOP

Related Classes of org.hibernate.validation.util.annotationfactory.AnnotationProxy

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.