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

org.apache.webbeans.inject.impl.InjectionPointImpl Maven / Gradle / Ivy

/*
 * 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.webbeans.inject.impl;

import java.io.IOException;
import java.io.NotSerializableException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
import java.lang.annotation.Annotation;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Member;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

import javax.decorator.Delegate;
import javax.enterprise.inject.spi.Annotated;
import javax.enterprise.inject.spi.AnnotatedConstructor;
import javax.enterprise.inject.spi.AnnotatedField;
import javax.enterprise.inject.spi.AnnotatedMember;
import javax.enterprise.inject.spi.AnnotatedMethod;
import javax.enterprise.inject.spi.AnnotatedParameter;
import javax.enterprise.inject.spi.AnnotatedType;
import javax.enterprise.inject.spi.Bean;
import javax.enterprise.inject.spi.InjectionPoint;

import org.apache.webbeans.config.WebBeansContext;
import org.apache.webbeans.container.BeanManagerImpl;
import org.apache.webbeans.event.EventUtil;
import org.apache.webbeans.portable.AnnotatedElementFactory;
import org.apache.webbeans.util.Asserts;
import org.apache.webbeans.util.OwbCustomObjectInputStream;
import org.apache.webbeans.util.WebBeansUtil;

public class InjectionPointImpl implements InjectionPoint, Serializable
{
    private static final long serialVersionUID = 1047233127758068484L;

    private Set qualifierAnnotations;
    
    private Bean ownerBean;
    
    private Member injectionMember;
    
    private Type injectionType;
    
    private Annotated annotated;
    
    private boolean transientt;
    
    private boolean delegate;

    InjectionPointImpl(Bean ownerBean, Collection qualifiers, AnnotatedField annotatedField)
    {
        this(ownerBean, annotatedField.getBaseType(), qualifiers, annotatedField,
                annotatedField.getJavaMember(), annotatedField.isAnnotationPresent(Delegate.class),
                annotatedField.getJavaMember() == null? false : Modifier.isTransient(annotatedField.getJavaMember().getModifiers()));
    }
    
    InjectionPointImpl(Bean ownerBean, Collection qualifiers, AnnotatedParameter parameter)
    {
        this(ownerBean, parameter.getBaseType(), qualifiers, parameter, parameter.getDeclaringCallable().getJavaMember(), parameter.isAnnotationPresent(Delegate.class), false);
    }

    /**
     * This constructor is used to create a 'virtual' InjectionPoint.
     * This is needed if an InjectionPoint was needed during a programmatic lookup or 3rd party beans.
     */
    InjectionPointImpl(Bean bean)
    {
        Asserts.assertNotNull(bean, "bean");
        this.ownerBean = bean;
        this.injectionType = bean.getBeanClass();
        this.qualifierAnnotations = bean.getQualifiers() == null ?
                Collections.emptySet() :
                Collections.unmodifiableSet(new HashSet<>(bean.getQualifiers()));
        this.annotated = null;
        this.injectionMember = null;
        this.delegate = false;
        this.transientt = false;
    }
    
    public InjectionPointImpl(Bean ownerBean, Type type, Collection qualifiers, Annotated annotated, Member member, boolean delegate, boolean isTransient)
    {
        if (type == null)
        {
            throw new IllegalArgumentException("type is null");
        }
        if (member == null)
        {
            throw new IllegalArgumentException("member is null");
        }
        Asserts.assertNotNull(qualifiers, "qualifiers");
        this.ownerBean = ownerBean;
        injectionType = type;
        qualifierAnnotations = Collections.unmodifiableSet(new HashSet<>(qualifiers));
        this.annotated = annotated;
        injectionMember = member;
        this.delegate = delegate;
        transientt = isTransient;
        if(!WebBeansUtil.checkObtainsInjectionPointConditions(this))
        {
            EventUtil.checkObservableInjectionPointConditions(this);
        }
    }
    
    @Override
    public Bean getBean()
    {
        return ownerBean;
    }

    @Override
    public Set getQualifiers()
    {
        
        return qualifierAnnotations;
    }

    @Override
    public Member getMember()
    {
        return injectionMember;
    }

    @Override
    public Type getType()
    {
        
        return injectionType;
    }

    
    @Override
    public Annotated getAnnotated()
    {
        return annotated;
    }

    @Override
    public boolean isDelegate()
    {
        return delegate;
    }

    @Override
    public boolean isTransient()
    {
        return transientt;
    }
    
    private void writeObject(java.io.ObjectOutputStream op) throws IOException
    {
        ObjectOutputStream out = new ObjectOutputStream(op);


        if (ownerBean != null)
        {
            //Write the owning bean class
            out.writeObject(ownerBean.getBeanClass());

            // and it's Qualifiers
            Set qualifiers = ownerBean.getQualifiers();
            writeQualifiers(out, qualifiers);
        }
        else
        {
            if (annotated instanceof AnnotatedMember)
            {
                Class beanClass = ((AnnotatedMember) annotated).getDeclaringType().getJavaClass();
                out.writeObject(beanClass);
                out.writeObject('~');
            }
            else
            {
                throw new NotSerializableException("Cannot detect bean class of InjetionPoint");
            }
        }


        if(injectionMember instanceof Field)
        {
            out.writeByte(0);
            out.writeUTF(injectionMember.getName());
            writeQualifiers(out, qualifierAnnotations);
        }
        
        if(injectionMember instanceof Method)
        {
            out.writeByte(1);
            out.writeUTF(injectionMember.getName());
            Method method = (Method) injectionMember;
            Class[] parameters = method.getParameterTypes();
            out.writeObject(parameters);
            
            AnnotatedParameter ap = (AnnotatedParameter) annotated;
            out.writeByte(ap.getPosition());
        }
        
        if(injectionMember instanceof Constructor)
        {
            out.writeByte(2);
            Constructor constr = (Constructor) injectionMember;
            Class[] parameters = constr.getParameterTypes();
            out.writeObject(parameters);
            
            AnnotatedParameter ap = (AnnotatedParameter) annotated;
            out.writeByte(ap.getPosition());
        }
        
        out.writeBoolean(delegate);
        out.writeBoolean(transientt);
        out.flush();
        
    }

    @SuppressWarnings("unchecked")
    private void readObject(java.io.ObjectInputStream inp) throws IOException, ClassNotFoundException
    {

        ObjectInputStream in = new OwbCustomObjectInputStream(inp, WebBeansUtil.getCurrentClassLoader());

        Class ownerBeanClass = (Class)in.readObject();
        Set ownerQualifiers = readQualifiers(in);

        WebBeansContext webBeansContext = WebBeansContext.currentInstance();
        AnnotatedElementFactory annotatedElementFactory = webBeansContext.getAnnotatedElementFactory();

        //process annotations
        BeanManagerImpl beanManager = webBeansContext.getBeanManagerImpl();
        Set> beans = beanManager.getBeans(ownerBeanClass,
                                                  ownerQualifiers.toArray(new Annotation[ownerQualifiers.size()]));
        ownerBean = beanManager.resolve(beans);

        // determine type of injection point member (0=field, 1=method, 2=constructor) and read...
        int c = in.readByte();
        if(c == 0)
        {
            String fieldName = in.readUTF();
            Field field = webBeansContext.getSecurityService().doPrivilegedGetDeclaredField(ownerBeanClass, fieldName);

            injectionMember = field;
            
            AnnotatedType annotatedType = annotatedElementFactory.newAnnotatedType(ownerBeanClass);
            annotated = annotatedElementFactory.newAnnotatedField(field, annotatedType);
            injectionType = field.getGenericType();

            qualifierAnnotations = readQualifiers(in);
        }
        else if(c == 1)
        {
            String methodName = in.readUTF();
            Class[] parameters = (Class[])in.readObject();
            
            Method method = webBeansContext.getSecurityService().doPrivilegedGetDeclaredMethod(ownerBeanClass, methodName, parameters);
            injectionMember = method;
            
            AnnotatedType annotatedType = annotatedElementFactory.newAnnotatedType(ownerBeanClass);
            AnnotatedMethod am =  (AnnotatedMethod)annotatedElementFactory.
                                    newAnnotatedMethod((Method) injectionMember,annotatedType);
            List> annParameters = am.getParameters();

            annotated = annParameters.get(in.readByte());
            injectionType = annotated.getBaseType();
        }
        else if(c == 2)
        {
            Class[] parameters = (Class[])in.readObject();            
            try
            {
                injectionMember = ownerBeanClass.getConstructor(parameters);

            }
            catch(NoSuchMethodException e)
            {
                injectionMember = null;
            }

            AnnotatedType annotatedType = (AnnotatedType)annotatedElementFactory.newAnnotatedType(ownerBeanClass);
            AnnotatedConstructor am =  annotatedElementFactory
                                            .newAnnotatedConstructor((Constructor) injectionMember,annotatedType);
            List> annParameters = am.getParameters();

            annotated = annParameters.get(in.readByte());
            injectionType = annotated.getBaseType();
        }

        delegate = in.readBoolean();
        transientt = in.readBoolean();
    }

    private void writeQualifiers(ObjectOutputStream out, Set qualifiers) throws IOException
    {
        for (Annotation qualifier : qualifiers)
        {
            out.writeObject('-'); // throw-away delimiter so alternating annotations don't get swallowed in the read.
            out.writeObject(qualifier);
        }

        // terminate character
        out.writeObject('~');
    }

    private Set readQualifiers(ObjectInputStream in) throws IOException, ClassNotFoundException
    {
        Set qualifiers = new HashSet<>();
        while(!in.readObject().equals('~'))   // read throw-away '-' or '~' terminal delimiter.
        {
            Annotation ann = (Annotation) in.readObject();  // now read the annotation.
            qualifiers.add(ann);
        }
        return qualifiers;
    }

    public String toString()
    {
        StringBuilder buffer = new StringBuilder();
        if(injectionMember instanceof Constructor)
        {
            Constructor constructor = (Constructor) injectionMember;
            buffer.append("Constructor Injection Point, constructor name :  ").append(constructor.getName()).append(", Bean Owner : [").append(ownerBean).append("]");
        }
        else if(injectionMember instanceof Method)
        {
            Method method = (Method) injectionMember;
            buffer.append("Method Injection Point, method name :  ").append(method.getName()).append(", Bean Owner : [").append(ownerBean).append("]");
        }
        else if(injectionMember instanceof Field)
        {
            Field field = (Field) injectionMember;
            buffer.append("Field Injection Point, field name :  ").append(field.getName()).append(", Bean Owner : [").append(ownerBean).append("]");            
        }
        
        return buffer.toString();
    }
}