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

io.fabric8.cdi.Fabric8Extension Maven / Gradle / Ivy

There is a newer version: 3.0.12
Show newest version
/**
 *  Copyright 2005-2016 Red Hat, Inc.
 *
 *  Red Hat 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 io.fabric8.cdi;

import io.fabric8.annotations.Alias;
import io.fabric8.annotations.Configuration;
import io.fabric8.annotations.Endpoint;
import io.fabric8.annotations.External;
import io.fabric8.annotations.Factory;
import io.fabric8.annotations.Path;
import io.fabric8.annotations.PortName;
import io.fabric8.annotations.Protocol;
import io.fabric8.annotations.ServiceName;
import io.fabric8.cdi.bean.ConfigurationBean;
import io.fabric8.cdi.bean.KubernetesClientBean;
import io.fabric8.cdi.bean.ServiceBean;
import io.fabric8.cdi.bean.ServiceUrlBean;
import io.fabric8.cdi.bean.ServiceUrlCollectionBean;
import io.fabric8.cdi.producers.FactoryMethodProducer;
import io.fabric8.cdi.qualifiers.EndpointQualifier;
import io.fabric8.cdi.qualifiers.ExternalQualifier;
import io.fabric8.cdi.qualifiers.PathQualifier;
import io.fabric8.cdi.qualifiers.PortQualifier;
import io.fabric8.cdi.qualifiers.ProtocolQualifier;
import io.fabric8.kubernetes.client.KubernetesClient;

import javax.enterprise.event.Observes;
import javax.enterprise.inject.Instance;
import javax.enterprise.inject.spi.AfterBeanDiscovery;
import javax.enterprise.inject.spi.Annotated;
import javax.enterprise.inject.spi.AnnotatedMethod;
import javax.enterprise.inject.spi.AnnotatedParameter;
import javax.enterprise.inject.spi.AnnotatedType;
import javax.enterprise.inject.spi.BeanManager;
import javax.enterprise.inject.spi.Extension;
import javax.enterprise.inject.spi.InjectionPoint;
import javax.enterprise.inject.spi.ProcessAnnotatedType;
import javax.enterprise.inject.spi.ProcessInjectionPoint;
import javax.enterprise.inject.spi.ProcessManagedBean;
import java.lang.annotation.Annotation;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;

import static io.fabric8.cdi.Utils.getFactoryMethodPort;
import static io.fabric8.cdi.Utils.getFactoryMethodProtocol;
import static io.fabric8.cdi.Utils.getFactoryMethodPath;
import static io.fabric8.cdi.Utils.or;

public class Fabric8Extension implements Extension {

    private static final String INJECTION_POINT_UNKNOWN_TYPE = "Failed to process injection point on bean %s with type: %s. Don't know how to handle type.";
    private static final Set factories = new LinkedHashSet<>();


    public void afterDiscovery(final @Observes AfterBeanDiscovery event, BeanManager beanManager) {

        KubernetesHolder.useBeanManager(beanManager);
        //Only add the bean if no other bean is found.
        if (beanManager.getBeans(KubernetesClient.class).isEmpty()) {
            event.addBean(new KubernetesClientBean());
        }

        //We need to process factories in reverse order so that we make feasible forwarding for service id etc.
        List reverseFactories = new ArrayList<>(FactoryMethodContext.sort(factories));
        Collections.reverse(reverseFactories);

        for (final FactoryMethodContext factoryMethodContext : reverseFactories) {
            ServiceBean.doWith(factoryMethodContext.getReturnType(), new ServiceBean.Callback() {
                @Override
                public ServiceBean apply(ServiceBean bean) {
                    String serviceId = bean.getServiceName();
                    String serviceProtocol = or(bean.getServiceProtocol(), getFactoryMethodProtocol(factoryMethodContext.getFactoryMethod().getJavaMember()));
                    String servicePort = or(bean.getServicePort(), getFactoryMethodPort(factoryMethodContext.getFactoryMethod().getJavaMember()));
                    String servicePath = or(bean.getServicePath(), getFactoryMethodPath(factoryMethodContext.getFactoryMethod().getJavaMember()));
                    Boolean serviceExternal = bean.getServiceExternal();
                    Boolean serviceEndpoint = bean.getServiceEndpoint();

                    //Ensure that there is a factory String -> sourceType before adding producer.
                    if (!String.class.equals(factoryMethodContext.getSourceType())) {
                        ServiceBean.getBean(serviceId, serviceProtocol, servicePort, servicePath, null, serviceEndpoint, serviceExternal, factoryMethodContext.getSourceType());
                    }

                    return bean.withProducer(new FactoryMethodProducer(factoryMethodContext.getBean(), factoryMethodContext.getFactoryMethod(), serviceId, serviceProtocol, servicePort, servicePath));
                }
            });
        }

        for (ServiceUrlBean bean : ServiceUrlBean.getBeans()) {
            event.addBean(bean);
        }

        for (ServiceUrlCollectionBean bean : ServiceUrlCollectionBean.getBeans()) {
            event.addBean(bean);
        }

        for (ServiceBean bean : ServiceBean.getBeans()) {
            if (bean.getProducer() != null) {
                event.addBean(bean);
            }
        }
        for (ConfigurationBean b : ConfigurationBean.getBeans()) {
            event.addBean(b);
        }
    }

    public  void processAnnotatedType(@Observes ProcessAnnotatedType pat,
                                         BeanManager beanManager) {
        AnnotatedType type = pat.getAnnotatedType();
    }


    public  void onInjectionPoint(@Observes ProcessInjectionPoint event, BeanManager beanManager) {
        final InjectionPoint injectionPoint = event.getInjectionPoint();
        if (isServiceInjectionPoint(injectionPoint)) {
            Annotated annotated = injectionPoint.getAnnotated();
            ServiceName name = annotated.getAnnotation(ServiceName.class);
            Protocol protocol = annotated.getAnnotation(Protocol.class);
            PortName port = annotated.getAnnotation(PortName.class);
            Path path = annotated.getAnnotation(Path.class);
            Alias alias = annotated.getAnnotation(Alias.class);
            Endpoint endpoint = annotated.getAnnotation(Endpoint.class);
            External external = annotated.getAnnotation(External.class);

            String serviceName = name.value();
            String serviceProtocol = protocol != null ? protocol.value() : null;
            String servicePort = port != null ? port.value() : null;
            String servicePath = path != null ? path.value() : null;
            String serviceAlias = alias != null ? alias.value() : null;
            Boolean serviceExternal = external != null ? external.value() : false;
            Boolean serviceEndpoint = endpoint != null ? endpoint.value() : false;

            Type type = annotated.getBaseType();
            if (type instanceof ParameterizedType && Instance.class.equals(((ParameterizedType) type).getRawType())) {
                type = ((ParameterizedType) type).getActualTypeArguments()[0];
            }

            if (type.equals(String.class)) {
                ServiceUrlBean.getBean(serviceName, serviceProtocol, servicePort, servicePath, serviceAlias, serviceEndpoint, serviceExternal);
            } else if (isGenericOf(type, List.class, String.class)) {
                ServiceUrlCollectionBean.getBean(serviceName, serviceProtocol, servicePort, servicePath, serviceAlias, serviceEndpoint, serviceExternal, Types.LIST_OF_STRINGS);
            } else if (isGenericOf(type, List.class, null)) {
                //TODO: Integrate with Factories(?)
            } else if (isGenericOf(type, Set.class, String.class)) {
                ServiceUrlCollectionBean.getBean(serviceName, serviceProtocol, servicePort, servicePath, serviceAlias, serviceEndpoint, serviceExternal, Types.SET_OF_STRINGS);
            } else if (isGenericOf(type, Set.class, null)) {
                //TODO: Integrate with Factories(?)
            } else if (type instanceof Class) {
                ServiceBean.getBean(serviceName, serviceProtocol, servicePort, servicePath, serviceAlias, serviceEndpoint, serviceExternal, type);
            } else {
                throw new RuntimeException(String.format(INJECTION_POINT_UNKNOWN_TYPE, injectionPoint.getBean().getBeanClass(), type));
            }

            if (protocol == null) {
                setDefaultProtocol(event);
            }
            if (port == null) {
                setDefaultPort(event);
            }
            if (path == null) {
                setDefaultPath(event);
            }
            if (endpoint == null) {
                setDefaultEndpoint(event);
            }
            if (external == null) {
                setDefaultExternal(event);
            }
        } else if (isConfigurationInjectionPoint(injectionPoint)) {
            Annotated annotated = injectionPoint.getAnnotated();
            Configuration configuration = annotated.getAnnotation(Configuration.class);
            Type type = injectionPoint.getType();
            String configurationId = configuration.value();
            ConfigurationBean.getBean(configurationId, type);
        }
    }

    public  void onManagedBean(final @Observes ProcessManagedBean event) {
        for (final AnnotatedMethod method : event.getAnnotatedBeanClass().getMethods()) {
            final Factory factory = method.getAnnotation(Factory.class);
            if (factory != null) {
                final Type sourceType = getSourceType(method);
                final Type returnType = method.getJavaMember().getReturnType();
                factories.add(new FactoryMethodContext(event.getBean(), sourceType, returnType, method));
            }
        }
    }

    private static  Type getSourceType(AnnotatedMethod method) {
        for (AnnotatedParameter parameter : method.getParameters()) {
            if (parameter.isAnnotationPresent(ServiceName.class)) {
                return parameter.getBaseType();
            }
        }
        return String.class;
    }

    /**
     * Checks if the InjectionPoint is annotated with the @Configuration qualifier.
     *
     * @param injectionPoint The injection point.
     * @return
     */
    private static boolean isConfigurationInjectionPoint(InjectionPoint injectionPoint) {
        Set qualifiers = injectionPoint.getQualifiers();
        for (Annotation annotation : qualifiers) {
            if (annotation.annotationType().isAssignableFrom(Configuration.class)) {
                return true;
            }
        }
        return false;
    }


    private  void setDefaultProtocol(ProcessInjectionPoint event) {
        //if protocol is not specified decorate injection point with "default" protocol.
        event.setInjectionPoint(new DelegatingInjectionPoint(event.getInjectionPoint()) {
            @Override
            public Set getQualifiers() {
                Set qualifiers = new LinkedHashSet<>(super.getQualifiers());
                qualifiers.add(new ProtocolQualifier(""));
                return Collections.unmodifiableSet(qualifiers);
            }
        });
    }

    private  void setDefaultPort(ProcessInjectionPoint event) {
        //if portname is not specified decorate injection point with "default" port.
        event.setInjectionPoint(new DelegatingInjectionPoint(event.getInjectionPoint()) {
            @Override
            public Set getQualifiers() {
                Set qualifiers = new LinkedHashSet<>(super.getQualifiers());
                qualifiers.add(new PortQualifier(""));
                return Collections.unmodifiableSet(qualifiers);
            }
        });
    }

    private  void setDefaultPath(ProcessInjectionPoint event) {
        //if path is not specified decorate injection point with "default" path.
        event.setInjectionPoint(new DelegatingInjectionPoint(event.getInjectionPoint()) {
            @Override
            public Set getQualifiers() {
                Set qualifiers = new LinkedHashSet<>(super.getQualifiers());
                qualifiers.add(new PathQualifier(""));
                return Collections.unmodifiableSet(qualifiers);
            }
        });
    }

    private  void setDefaultExternal(ProcessInjectionPoint event) {
        //if external is not specified decorate injection point with "default" external=false.
        event.setInjectionPoint(new DelegatingInjectionPoint(event.getInjectionPoint()) {
            @Override
            public Set getQualifiers() {
                Set qualifiers = new LinkedHashSet<>(super.getQualifiers());
                qualifiers.add(new ExternalQualifier(false));
                return Collections.unmodifiableSet(qualifiers);
            }
        });
    }

    private  void setDefaultEndpoint(ProcessInjectionPoint event) {
        //if endpoint is not specified decorate injection point with "default" endpoint=false.
        event.setInjectionPoint(new DelegatingInjectionPoint(event.getInjectionPoint()) {
            @Override
            public Set getQualifiers() {
                Set qualifiers = new LinkedHashSet<>(super.getQualifiers());
                qualifiers.add(new EndpointQualifier(false));
                return Collections.unmodifiableSet(qualifiers);
            }
        });
    }

    private static boolean isGenericOf(Type type, Type raw, Type argument) {
        if (type instanceof ParameterizedType) {
            ParameterizedType p = (ParameterizedType) type;
            return p.getRawType() != null
                    && p.getRawType().equals(raw)
                    && p.getActualTypeArguments().length == 1
                    && (p.getActualTypeArguments()[0].equals(argument) || argument == null);
        } else if (type instanceof Class) {
            return argument == null && type.equals(raw);
        } else {
            return false;
        }
    }


    /**
     * Checks if the InjectionPoint is annotated with the @Service qualifier.
     *
     * @param injectionPoint The injection point.
     * @return
     */
    public static boolean isServiceInjectionPoint(InjectionPoint injectionPoint) {
        Set qualifiers = injectionPoint.getQualifiers();
        for (Annotation annotation : qualifiers) {
            if (annotation.annotationType().isAssignableFrom(ServiceName.class)) {
                return true;
            }
        }
        return false;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy