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

com.caucho.config.gen.CandiInvocationContext Maven / Gradle / Ivy

/*
 * Copyright (c) 1998-2018 Caucho Technology -- all rights reserved
 *
 * This file is part of Resin(R) Open Source
 *
 * Each copy or derived work must preserve the copyright notice and this
 * notice unmodified.
 *
 * Resin Open Source is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * Resin Open Source is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE, or any warranty
 * of NON-INFRINGEMENT.  See the GNU General Public License for more
 * details.
 *
 * You should have received a copy of the GNU General Public License
 * along with Resin Open Source; if not, write to the
 *
 *   Free SoftwareFoundation, Inc.
 *   59 Temple Place, Suite 330
 *   Boston, MA 02111-1307  USA
 *
 * @author Scott Ferguson
 */

package com.caucho.config.gen;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;

import javax.enterprise.inject.spi.InterceptionType;
import javax.enterprise.inject.spi.Interceptor;
import javax.interceptor.InvocationContext;

import com.caucho.loader.EnvironmentLocal;
import com.caucho.util.L10N;

public class CandiInvocationContext implements InvocationContext {
  private static final L10N L = new L10N(CandiInvocationContext.class);
  
  private static final EnvironmentLocal> _contextDataLocal
  = new EnvironmentLocal>();
  
  private static final HashMap,Class> _boxMap
    = new HashMap,Class>();
  
  private final Object _target;
  private final Method _apiMethod;
  private final Method _implMethod;

  private final Interceptor []_chainMethods;
  private final Object []_chainObjects;
  private final int []_chainIndex;
  
  private final InterceptionType _type;

  private Object []_param;

  private int _index;
  
  public CandiInvocationContext(InterceptionType type,
                                Object target,
                                Method apiMethod,
                                Method implMethod,
                                Interceptor []chainMethods,
                                Object []chainObjects,
                                int []chainIndex,
                                Object []param)
  {
    _target = target;
    _type = type;
    
    _apiMethod = apiMethod;
    _implMethod = implMethod;
    _chainMethods = chainMethods;
    _chainObjects = chainObjects;
    _chainIndex = chainIndex;
    _param = param;
  }

  @Override
  public Object getTarget()
  {
    return _target;
  }

  @Override
  public Method getMethod()
  {
    return _apiMethod;
  }

  @Override
  public Object getTimer()
  {
    return null;
  }

  @Override
  public Object[] getParameters()
    throws IllegalStateException
  {
    if (_param != null)
      return _param;
    else
      throw new IllegalStateException(L.l("No parameters are allowed in this context"));
  }

  @Override
  public void setParameters(Object[] parameters)
    throws IllegalStateException
  {
    Class []paramType = _apiMethod.getParameterTypes();
    
    if (parameters == null) {
      throw new IllegalArgumentException(L.l("{0}.{1}: interception parameters cannot be null",
                                             _apiMethod.getDeclaringClass().getName(),
                                             _apiMethod.getName()));
    }
    
    if (paramType.length != parameters.length) {
      throw new IllegalArgumentException(L.l("{0}.{1}: interception parameters '{2}' do not match the expected '{3}'",
                                             _apiMethod.getDeclaringClass().getName(),
                                             _apiMethod.getName(),
                                             parameters.length,
                                             paramType.length));
    }
    
    for (int i = paramType.length - 1; i >= 0; i--) {
      Object value = parameters[i];
      
      Class argType = getArgType(paramType[i]);
      
      if (value != null && ! argType.isAssignableFrom(value.getClass())) {
        throw new IllegalArgumentException(L.l("{0}.{1}: interception parameters '{2}' do not match the expected '{3}'",
                                               _apiMethod.getDeclaringClass().getName(),
                                               _apiMethod.getName(),
                                               value,
                                               paramType[i].getName()));
      }
    }
    
    _param = parameters;
  }
  
  public static Map getCurrentContextData()
  {
    Map contextData = _contextDataLocal.get();
    
    if (contextData == null) {
      contextData = new HashMap();
      
      _contextDataLocal.set(contextData);
    }

    return contextData;
  }

  @Override
  public Map getContextData()
  {
    return getCurrentContextData();
  }

  @Override
  public Object proceed()
    throws Exception
  {
    boolean isTop = _index == 0;
    
    try {
      Object result;
      
      // ioc/0c57
      if (_chainObjects != null && _index < _chainIndex.length) {
        int i = _index++;
        
        if (_chainObjects[_chainIndex[i]] == null)
          throw new NullPointerException(i + " index[i]=" + _chainIndex[i] + " " + _type + " " + _chainMethods[i]);

        result = _chainMethods[i].intercept(_type,
                                            _chainObjects[_chainIndex[i]], 
                                            this);
      }
      else {
        result = _implMethod.invoke(_target, _param);
      }
      
      return result;
    } catch (InterceptorException e) {
      Throwable cause = e.getCause();

      if (cause instanceof Exception)
        throw (Exception) cause;
      else
        throw e;
    } catch (InvocationTargetException e) {
      Throwable cause = e.getCause();

      if (cause instanceof Exception)
        throw (Exception) cause;
      else
        throw e;
    } catch (IllegalArgumentException e) {
      throw new IllegalArgumentException(e + "\n  " + _implMethod
                                         + "\n  " + _target,
                                         e);
    } catch (RuntimeException e) {
      throw e;
    } catch (Exception e) {
      throw e;
    } finally {
      if (isTop)
        _contextDataLocal.set(null);
    }
  }
  
  private Class getArgType(Class type)
  {
    Class boxType = _boxMap.get(type);
    
    if (boxType != null)
      return boxType;
    else
      return type;
  }
  
  public String toString()
  {
    return getClass().getSimpleName() + "[" + _implMethod.getName() + "]";
  }
  
  static {
    _boxMap.put(boolean.class, Boolean.class);
    _boxMap.put(char.class, Character.class);
    _boxMap.put(byte.class, Byte.class);
    _boxMap.put(short.class, Short.class);
    _boxMap.put(int.class, Integer.class);
    _boxMap.put(long.class, Long.class);
    _boxMap.put(float.class, Float.class);
    _boxMap.put(double.class, Double.class);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy