
io.fabric8.cdi.Fabric8Extension Maven / Gradle / Ivy
/**
* 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