All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.simpleframework.xml.core.ClassScanner Maven / Gradle / Ivy

Go to download

Simple is a high performance XML serialization and configuration framework for Java

The newest version!
/*
 * ClassScanner.java July 2008
 *
 * Copyright (C) 2008, Niall Gallagher 
 *
 * 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.simpleframework.xml.core;

import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
import java.util.List;
import java.util.Map;

import org.simpleframework.xml.DefaultType;
import org.simpleframework.xml.Namespace;
import org.simpleframework.xml.NamespaceList;
import org.simpleframework.xml.Order;
import org.simpleframework.xml.Root;

/**
 * The ClassScanner performs the reflective inspection
 * of a class and extracts all the class level annotations. This will
 * also extract the methods that are annotated. This ensures that the
 * callback methods can be invoked during the deserialization process.
 * Also, this will read the namespace annotations that are used.
 * 
 * @author Niall Gallagher
 * 
 * @see org.simpleframework.xml.core.Scanner
 */ 
class ClassScanner  {
   
   /**
    * This is the namespace decorator associated with this scanner.
    */
   private NamespaceDecorator decorator;
   
   /**
    * This is the scanner that is used to acquire the constructors.
    */
   private ConstructorScanner scanner;
   
   /**
    * This function acts as a pointer to the types commit process.
    */
   private Function commit;
   
   /**
    * This function acts as a pointer to the types validate process.
    */
   private Function validate;

   /**
    * This function acts as a pointer to the types persist process.
    */
   private Function persist;

   /**
    * This function acts as a pointer to the types complete process.
    */
   private Function complete;   
   
   /**
    * This function is used as a pointer to the replacement method.
    */
   private Function replace;
   
   /**
    * This function is used as a pointer to the resolution method.
    */
   private Function resolve;
   
   /**
    * This object contains various support functions for the class.
    */
   private Support support;
   
   /**
    * This is the root annotation that has been scanned from the type.
    */
   private Root root;
   
   /**
    * This is the order annotation that has been scanned from the type.
    */
   private Order order;
   
   /**
    * Constructor for the ClassScanner object. This is 
    * used to scan the provided class for annotations that are used 
    * to build a schema for an XML file to follow. 
    * 
    * @param detail this contains the details for the class scanned
    * @param support this contains various support functions
    */
   public ClassScanner(Detail detail, Support support) throws Exception { 
      this.scanner = new ConstructorScanner(detail, support);
      this.decorator = new NamespaceDecorator();
      this.support = support;
      this.scan(detail);
   }      

   /**
    * This is used to acquire the default signature for the class. 
    * The default signature is the signature for the no argument
    * constructor for the type. If there is no default constructor
    * for the type then this will return null.
    * 
    * @return this returns the default signature if it exists
    */
   public Signature getSignature() {
      return scanner.getSignature();
   }
   
   /**
    * This returns the signatures for the type. All constructors are
    * represented as a signature and returned. More signatures than
    * constructors will be returned if a constructor is annotated 
    * with a union annotation.
    *
    * @return this returns the list of signatures for the type
    */
   public List getSignatures(){
      return scanner.getSignatures();
   }
   
   /**
    * This returns a map of all parameters that exist. This is used
    * to validate all the parameters against the field and method
    * annotations that exist within the class. 
    * 
    * @return this returns a map of all parameters within the type
    */
   public ParameterMap getParameters() {
      return scanner.getParameters();
   }
   
   /**
    * This is used to acquire the Decorator for this.
    * A decorator is an object that adds various details to the
    * node without changing the overall structure of the node. For
    * example comments and namespaces can be added to the node with
    * a decorator as they do not affect the deserialization.
    * 
    * @return this returns the decorator associated with this
    */
   public Decorator getDecorator() {
      return decorator;
   }

   /**
    * This returns the order annotation used to determine the order
    * of serialization of attributes and elements. The order is a
    * class level annotation that can be used only once per class
    * XML schema. If none exists then this will return null.
    *  of the class processed by this scanner.
    * 
    * @return this returns the name of the object being scanned
    */
   public Order getOrder() {
      return order;
   }
   
   /**
    * This returns the root of the class processed by this scanner.
    * The root determines the type of deserialization that is to
    * be performed and also contains the name of the root element. 
    * 
    * @return this returns the name of the object being scanned
    */
   public Root getRoot() {
      return root;
   }

   /**
    * This method is used to retrieve the schema class commit method
    * during the deserialization process. The commit method must be
    * marked with the Commit annotation so that when the
    * object is deserialized the persister has a chance to invoke the
    * method so that the object can build further data structures.
    * 
    * @return this returns the commit method for the schema class
    */
   public Function getCommit() {
      return commit;           
   }

   /**
    * This method is used to retrieve the schema class validation
    * method during the deserialization process. The validation method
    * must be marked with the Validate annotation so that
    * when the object is deserialized the persister has a chance to 
    * invoke that method so that object can validate its field values.
    * 
    * @return this returns the validate method for the schema class
    */   
   public Function getValidate() {
      return validate;       
   }
   
   /**
    * This method is used to retrieve the schema class persistence
    * method. This is invoked during the serialization process to
    * get the object a chance to perform an necessary preparation
    * before the serialization of the object proceeds. The persist
    * method must be marked with the Persist annotation.
    * 
    * @return this returns the persist method for the schema class
    */
   public Function getPersist() {
      return persist;           
   }

   /**
    * This method is used to retrieve the schema class completion
    * method. This is invoked after the serialization process has
    * completed and gives the object a chance to restore its state
    * if the persist method required some alteration or locking.
    * This is marked with the Complete annotation.
    * 
    * @return returns the complete method for the schema class
    */   
   public Function getComplete() {
      return complete;           
   }
   
   /**
    * This method is used to retrieve the schema class replacement
    * method. The replacement method is used to substitute an object
    * that has been deserialized with another object. This allows
    * a seamless delegation mechanism to be implemented. This is
    * marked with the Replace annotation. 
    * 
    * @return returns the replace method for the schema class
    */
   public Function getReplace() {
      return replace;
   }
   
   /**
    * This method is used to retrieve the schema class replacement
    * method. The replacement method is used to substitute an object
    * that has been deserialized with another object. This allows
    * a seamless delegation mechanism to be implemented. This is
    * marked with the Replace annotation. 
    * 
    * @return returns the replace method for the schema class
    */
   public Function getResolve() {
      return resolve;
   }
  
   /**
    * Scan the fields and methods such that the given class is scanned 
    * first then all super classes up to the root Object. 
    * All fields and methods from the most specialized classes override 
    * fields and methods from higher up the inheritance hierarchy. This
    * means that annotated details can be overridden.
    * 
    * @param detail contains the methods and fields to be examined
    */   
   private void scan(Detail detail) throws Exception {
      DefaultType access = detail.getOverride();
      Class type = detail.getType();

      while(type != null) {
         Detail value = support.getDetail(type, access);

         namespace(value);
         method(value);
         definition(value);
         type = value.getSuper();
      }      
      commit(detail); 
   }
   
   /**
    * This method is used to extract the Root annotation
    * and the Order annotation from the detail provided.
    * These annotation are taken from the first definition encountered
    * from the most specialized class up through the base classes.
    * 
    * @param detail this detail object used to acquire the annotations
    */
   private void definition(Detail detail) throws Exception {
      if(root == null) {
         root = detail.getRoot();
      }
      if(order == null) {
         order = detail.getOrder();
      }
   }
   
   /**
    * This is used to acquire the namespace annotations that apply to 
    * the scanned class. Namespace annotations are added only if they
    * have not already been extracted from a more specialized class.
    * When scanned all the namespace definitions are used to qualify
    * the XML that is produced from serializing the class.
    * 
    * @param type this is the type to extract the annotations from
    */
   private void namespace(Detail detail) throws Exception {
      NamespaceList scope = detail.getNamespaceList();
      Namespace namespace = detail.getNamespace();
      
      if(namespace != null) {
         decorator.add(namespace);
      }
      if(scope != null) {
         Namespace[] list = scope.value();
         
         for(Namespace name : list) {
            decorator.add(name);
         }
      }
   }
   
   /**
    * This is used to set the primary namespace for nodes that will
    * be decorated by the namespace decorator. If no namespace is set
    * using this method then this decorator will leave the namespace
    * reference unchanged and only add namespaces for scoping.
    * 
    * @param detail the detail object that contains the namespace
    */
   private void commit(Detail detail) {
      Namespace namespace = detail.getNamespace();
      
      if(namespace != null) {
         decorator.set(namespace);
      }
   }

   /**
    * This is used to scan the specified class for methods so that
    * the persister callback annotations can be collected. These
    * annotations help object implementations to validate the data
    * that is injected into the instance during deserialization.
    * 
    * @param detail this is a detail from within the class hierarchy
    */
   private void method(Detail detail) throws Exception {
      List list = detail.getMethods();

      for(MethodDetail entry : list) {
         method(entry);              
      }     
   }
   
   /**
    * Scans the provided method for a persister callback method. If 
    * the method contains an method annotated as a callback that 
    * method is stored so that it can be invoked by the persister
    * during the serialization and deserialization process.
    * 
    * @param detail the method to scan for callback annotations
    */
   private void method(MethodDetail detail) {
      Annotation[] list = detail.getAnnotations();
      Method method = detail.getMethod();
      
      for(Annotation label : list) {
         if(label instanceof Commit) {           
            commit(method);
         }
         if(label instanceof Validate) {      
            validate(method);
         }
         if(label instanceof Persist) {      
            persist(method);
         }
         if(label instanceof Complete) {      
            complete(method);
         }    
         if(label instanceof Replace) {
            replace(method);              
         }   
         if(label instanceof Resolve) {
            resolve(method);              
         }  
      }
   }
   
   /**
    * This method is used to check the provided method to determine
    * if it contains the Replace annotation. If the
    * method contains the required annotation it is stored so that
    * it can be invoked during the deserialization process.
    *
    * @param method this is the method checked for the annotation
    */ 
   private void replace(Method method) {
      if(replace == null) {
         replace = getFunction(method);
      }
   }
   
   /**
    * This method is used to check the provided method to determine
    * if it contains the Resolve annotation. If the
    * method contains the required annotation it is stored so that
    * it can be invoked during the deserialization process.
    *
    * @param method this is the method checked for the annotation
    */ 
   private void resolve(Method method) {
      if(resolve == null) {
         resolve = getFunction(method);
      }
   }
   
   /**
    * This method is used to check the provided method to determine
    * if it contains the Commit annotation. If the
    * method contains the required annotation it is stored so that
    * it can be invoked during the deserialization process.
    *
    * @param method this is the method checked for the annotation
    */ 
   private void commit(Method method) { 
      if(commit == null) {
         commit = getFunction(method);
      }
   }
   
   /**
    * This method is used to check the provided method to determine
    * if it contains the Validate annotation. If the
    * method contains the required annotation it is stored so that
    * it can be invoked during the deserialization process.
    *
    * @param method this is the method checked for the annotation
    */ 
   private void validate(Method method) {
      if(validate == null) {
         validate = getFunction(method);  
      }
   }
   
   /**
    * This method is used to check the provided method to determine
    * if it contains the Persist annotation. If the
    * method contains the required annotation it is stored so that
    * it can be invoked during the deserialization process.
    *
    * @param method this is the method checked for the annotation
    */    
   private void persist(Method method) {
      if(persist == null) {
         persist = getFunction(method);  
      }
   }

   /**
    * This method is used to check the provided method to determine
    * if it contains the Complete annotation. If the
    * method contains the required annotation it is stored so that
    * it can be invoked during the deserialization process.
    *
    * @param method this is the method checked for the annotation
    */ 
   private void complete(Method method) {
      if(complete == null) {
         complete = getFunction(method);    
      }
   } 
   
   /**
    * This is used to acquire a Function object for the
    * method provided. The function returned will allow the callback
    * method to be invoked when given the context and target object.
    * 
    * @param method this is the method that is to be invoked
    * 
    * @return this returns the function that is to be invoked
    */
   private Function getFunction(Method method) {
      boolean contextual = isContextual(method);
      
      if(!method.isAccessible()) {
         method.setAccessible(true);
      }
      return new Function(method, contextual);
   }
   
   /**
    * This is used to determine whether the annotated method takes a
    * contextual object. If the method takes a Map then
    * this returns true, otherwise it returns false.
    *
    * @param method this is the method to check the parameters of
    *
    * @return this returns true if the method takes a map object
    */ 
   private boolean isContextual(Method method)  {
      Class[] list = method.getParameterTypes();

      if(list.length == 1) {
         return Map.class.equals(list[0]);                 
      }      
      return false;
   }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy