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

com.google.web.bindery.autobean.vm.impl.FactoryHandler Maven / Gradle / Ivy

/*
 * 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.web.bindery.autobean.vm.impl;

import com.google.web.bindery.autobean.shared.AutoBean.PropertyName;
import com.google.web.bindery.autobean.shared.AutoBeanFactory;
import com.google.web.bindery.autobean.shared.AutoBeanUtils;
import com.google.web.bindery.autobean.vm.Configuration;

import java.lang.reflect.Field;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Proxy;

/**
 * Handles dispatches on AutoBeanFactory interfaces.
 */
public class FactoryHandler implements InvocationHandler {
  private final Configuration configuration;

  /**
   * Constructor.
   * 
   * @param categories the classes specified by a Category annotation
   */
  public FactoryHandler(Configuration configuration) {
    this.configuration = configuration;
  }

  /**
   * Handles both declared factory methods as well as the dynamic create
   * methods.
   */
  public Object invoke(Object proxy, Method method, Object[] args)
      throws Throwable {

    Class beanType;
    Object toWrap = null;
    String name = method.getName();
    if (BeanMethod.OBJECT.matches(method)) {
      // Redirect all methods of Object early on
      return method.invoke(this, args);
    }

    if (name.equals("create")) {
      // Dynamic create. Guaranteed to have at least one argument
      // create(clazz); or create(clazz, toWrap);
      beanType = (Class) args[0];
      if (args.length == 2) {
        toWrap = args[1];
      }
    } else if (name.equals("getEnum")) {
      Class clazz = (Class) args[0];
      String token = (String) args[1];
      return getEnum(clazz, token);
    } else if (name.equals("getToken")) {
      Enum e = (Enum) args[0];
      return getToken(e);
    } else {
      // Declared factory method, use the parameterization
      // AutoBean foo(); or Autobean foo(Foo toWrap);
      ParameterizedType returnType = (ParameterizedType) method.getGenericReturnType();
      beanType = (Class) returnType.getActualTypeArguments()[0];

      if (args != null && args.length == 1) {
        toWrap = args[0];
      }
    }

    // Return any existing wrapper
    ProxyAutoBean toReturn = (ProxyAutoBean) AutoBeanUtils.getAutoBean(toWrap);
    if (toReturn == null) {
      // Create the implementation bean
      if (toWrap == null) {
        toReturn = new ProxyAutoBean((AutoBeanFactory) proxy, beanType,
            configuration);
      } else {
        toReturn = new ProxyAutoBean((AutoBeanFactory) proxy, beanType,
            configuration, toWrap);
      }
    }

    return toReturn;
  }

  @Override
  public boolean equals(Object other) {
    if (other != null && Proxy.isProxyClass(other.getClass())) {
      other = Proxy.getInvocationHandler(other);
    }
    return super.equals(other);
  }

  @Override
  public final int hashCode() {
    return super.hashCode();
  }

  /**
   * EnumMap support.
   */
  private Object getEnum(Class clazz, String token)
      throws IllegalAccessException {
    for (Field f : clazz.getFields()) {
      String fieldName;
      PropertyName annotation = f.getAnnotation(PropertyName.class);
      if (annotation != null) {
        fieldName = annotation.value();
      } else {
        fieldName = f.getName();
      }
      if (token.equals(fieldName)) {
        f.setAccessible(true);
        return f.get(null);
      }
    }
    throw new IllegalArgumentException("Cannot find enum " + token
        + " in type " + clazz.getCanonicalName());
  }

  /**
   * EnumMap support.
   */
  private Object getToken(Enum e) throws NoSuchFieldException {
    // Remember enum constants are fields
    PropertyName annotation = e.getDeclaringClass().getField(e.name()).getAnnotation(
        PropertyName.class);
    if (annotation != null) {
      return annotation.value();
    } else {
      return e.name();
    }
  }
}