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

com.paypal.springboot.resteasy.ResteasyEmbeddedServletInitializer_extra Maven / Gradle / Ivy

package com.paypal.springboot.resteasy;

import org.apache.commons.io.FilenameUtils;
import org.jboss.resteasy.plugins.servlet.ResteasyServletInitializer;
import org.reflections.Reflections;
import org.reflections.scanners.SubTypesScanner;
import org.reflections.util.ClasspathHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.beans.factory.config.ConstructorArgumentValues;
import org.springframework.beans.factory.support.BeanDefinitionRegistry;
import org.springframework.beans.factory.support.GenericBeanDefinition;
import org.springframework.core.annotation.AnnotationUtils;

import javax.ws.rs.ApplicationPath;
import javax.ws.rs.Path;
import javax.ws.rs.core.Application;
import javax.ws.rs.ext.Provider;
import java.io.File;
import java.net.URL;
import java.util.Collection;
import java.util.HashSet;
import java.util.Set;

/**
 * This is a Spring version of {@link ResteasyServletInitializer}
 *
 * @author Fabio Carvalho ([email protected] or [email protected])
 */
public class ResteasyEmbeddedServletInitializer implements BeanFactoryPostProcessor {

    private Set> applications;
    private Set> allResources;
    private Set> providers;

    private static final Logger logger = LoggerFactory.getLogger(ResteasyEmbeddedServletInitializer.class);

    /**
     * Copy all entries that are a JAR file or a directory
     */
    private void copyValidClasspathEntries(Collection source, Set destination) {
        String fileName;
        boolean isJarFile;
        boolean isDirectory;

        for (URL url : source) {
            if(destination.contains(url)) {
                continue;
            }

            fileName = url.getFile();
            isJarFile = FilenameUtils.isExtension(fileName, "jar");
            isDirectory = new File(fileName).isDirectory();

            if (isJarFile || isDirectory) {
                destination.add(url);
            } else if (logger.isDebugEnabled()) {
                logger.debug("Ignored classpath entry: " + fileName);
            }
        }
    }

    /**
     * Scan the Classpath searching for classes of type {@link Application}
     */
    private void findJaxrsApplicationClasses() {
        logger.debug("Finding JAX-RS Application classes");

        final Collection systemPropertyURLs = ClasspathHelper.forJavaClassPath();
        final Collection classLoaderURLs = ClasspathHelper.forClassLoader();

        Set classpathURLs = new HashSet();

        copyValidClasspathEntries(systemPropertyURLs, classpathURLs);
        copyValidClasspathEntries(classLoaderURLs, classpathURLs);

        logger.debug("Classpath URLs to be scanned: " + classpathURLs);

        Reflections reflections = new Reflections(classpathURLs, new SubTypesScanner());

        applications = reflections.getSubTypesOf(Application.class);

        if(logger.isDebugEnabled()) {
            for (Object appClass : applications.toArray()) {
                logger.debug("JAX-RS Application class found: {}", ((Class) appClass).getName());
            }
        }
    }

    /**
     * Search for JAX-RS resource and provider Spring beans,
     * which are the ones whose classes are annotated with
     * {@link Path} or {@link Provider} respectively
     *
     * @param beanFactory
     */
    private void findJaxrsResourcesAndProviderClasses(ConfigurableListableBeanFactory beanFactory) {
        logger.debug("Finding JAX-RS resources and providers Spring bean classes");

        String[] resourceBeans = beanFactory.getBeanNamesForAnnotation(Path.class);
        String[] providerBeans = beanFactory.getBeanNamesForAnnotation(Provider.class);

        allResources = new HashSet>();
        providers = new HashSet>();

        for(String resourceBean : resourceBeans) {
            allResources.add(beanFactory.getType(resourceBean));
        }
        for(String providerBean : providerBeans) {
            providers.add(beanFactory.getType(providerBean));
        }

        if(logger.isDebugEnabled()) {
            for (Object resourceClass : allResources.toArray()) {
                logger.debug("JAX-RS resource class found: {}", ((Class) resourceClass).getName());
            }
        }
        if(logger.isDebugEnabled()) {
            for (Object providerClass: providers.toArray()) {
                logger.debug("JAX-RS provider class found: {}", ((Class) providerClass).getName());
            }
        }
    }

    /**
     * Check if any JAX-RS related class was found
     *
     * @return true only if there is at least one JAX-RS class
     */
    private boolean noClasses() {
        if (applications != null && applications.size() != 0) {
            return false;
        }
        if (allResources != null && allResources.size() != 0) {
            return false;
        }
        if (providers != null && providers.size() != 0) {
            return false;
        }

        return true;
    }

    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {
        logger.debug("Post process bean factory has been called");

        findJaxrsApplicationClasses();
        findJaxrsResourcesAndProviderClasses(beanFactory);

        if (noClasses()) {
            logger.warn("No JAX-RS classes have been found");
            return;
        }

        BeanDefinitionRegistry registry = (BeanDefinitionRegistry) beanFactory;

        for (Class applicationClass : applications) {
            ApplicationPath path = applicationClass.getAnnotation(ApplicationPath.class);
            if (path == null) {
                continue;
            }

            logger.debug("registering JAX-RS application class " + applicationClass.getName());

            GenericBeanDefinition applicationServletBean = createApplicationServlet(applicationClass, path.value());
            registry.registerBeanDefinition(applicationClass.getName(), applicationServletBean);
        }

    }

    /**
     * Creates a Servlet bean definition for the given JAX-RS application
     *
     * @param applicationClass
     * @param path
     * @return a Servlet bean definition for the given JAX-RS application
     */
    private GenericBeanDefinition createApplicationServlet(Class applicationClass, String path) {
        GenericBeanDefinition applicationServletBean = new GenericBeanDefinition();
        applicationServletBean.setFactoryBeanName(ResteasyApplicationBuilder.BEAN_NAME);
        applicationServletBean.setFactoryMethodName("build");

        Set> resources = getApplicationResourceClasses(applicationClass);

        ConstructorArgumentValues values = new ConstructorArgumentValues();
        values.addIndexedArgumentValue(0, applicationClass.getName());
        values.addIndexedArgumentValue(1, path);
        values.addIndexedArgumentValue(2, resources);
        values.addIndexedArgumentValue(3, providers);
        applicationServletBean.setConstructorArgumentValues(values);

        applicationServletBean.setAutowireCandidate(false);
        applicationServletBean.setScope("singleton");

        return applicationServletBean;
    }

    private Set> getApplicationResourceClasses(Class applicationClass) {
        Application application;

        try {
            application = applicationClass.newInstance();
        } catch (Exception e) {
            logger.warn("Not able to identify specific application class (" + applicationClass.getSimpleName() + ") resources, returning all instead", e);
            return allResources;
        }

        Set> classes = application.getClasses();
        Set singletons = application.getSingletons();

        if ((classes == null || classes.size() == 0) && (singletons == null || singletons.size() == 0)) {
            // This should be the case for most applications, which relies on scanning,
            // instead of manually registering the resources
            return allResources;
        }

        Set> resources = new HashSet>();

        if(classes != null) {
            for(Class clazz : classes) {
                if(AnnotationUtils.findAnnotation(clazz, Path.class) != null) {
                    resources.add(clazz);
                }
            }
        }

        if(singletons != null) {
            for(Object singleton : singletons) {
                if(AnnotationUtils.findAnnotation(singleton.getClass(), Path.class) != null) {
                    resources.add(singleton.getClass());
                }
            }
        }

        logger.debug("Application class " + applicationClass.getSimpleName() + " has " + resources.size() + " specific resources");

        return resources;
    }

}