org.bremersee.spring.autoconfigure.JerseyAutoConfiguration.java Source code

Java tutorial

Introduction

Here is the source code for org.bremersee.spring.autoconfigure.JerseyAutoConfiguration.java

Source

/*
 * Copyright 2012-2014 the original author or authors.
 *
 * Licensed 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.bremersee.spring.autoconfigure;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;

import javax.annotation.PostConstruct;
import javax.servlet.DispatcherType;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.ws.rs.Consumes;
import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.Response.Status;
import javax.ws.rs.ext.ContextResolver;
import javax.ws.rs.ext.Provider;
import javax.xml.bind.JAXBContext;

import org.glassfish.jersey.CommonProperties;
import org.glassfish.jersey.server.ResourceConfig;
import org.glassfish.jersey.servlet.ServletContainer;
import org.glassfish.jersey.servlet.ServletProperties;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication;
import org.springframework.boot.autoconfigure.web.DispatcherServletAutoConfiguration;
import org.springframework.boot.context.embedded.FilterRegistrationBean;
import org.springframework.boot.context.embedded.RegistrationBean;
import org.springframework.boot.context.embedded.ServletRegistrationBean;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.Ordered;
import org.springframework.core.annotation.Order;
import org.springframework.oxm.jaxb.Jaxb2Marshaller;
import org.springframework.util.StringUtils;
import org.springframework.web.WebApplicationInitializer;
import org.springframework.web.filter.RequestContextFilter;

/**
 * {@link EnableAutoConfiguration Auto-configuration} for Jersey.
 *
 * @author Dave Syer
 * @author Andy Wilkinson
 * @author Christian Bremer
 */
@Configuration
@ConditionalOnClass(name = { "org.glassfish.jersey.server.spring.SpringComponentProvider",
        "javax.servlet.ServletRegistration" })
@ConditionalOnWebApplication
@Order(Ordered.HIGHEST_PRECEDENCE)
@AutoConfigureBefore(DispatcherServletAutoConfiguration.class)
@EnableConfigurationProperties(JerseyProperties.class)
public class JerseyAutoConfiguration implements WebApplicationInitializer {

    protected final Logger log = LoggerFactory.getLogger(getClass());

    @Autowired
    private JerseyProperties jersey;

    @Autowired(required = false)
    private List<JerseyComponentsProvider> componentProviders = new ArrayList<JerseyComponentsProvider>();

    @Autowired(required = false)
    @Qualifier("payloadMarshaller")
    protected Jaxb2Marshaller jaxb2Marshaller;

    private String path;

    private ResourceConfig config;

    @PostConstruct
    public void path() {
        //this.path = findPath(AnnotationUtils.findAnnotation(this.config.getClass(), ApplicationPath.class));
        this.path = findPath(jersey.getPath());
        this.config = new ResourceConfig();

        if (jaxb2Marshaller != null) {
            this.config.register(new JAXBContextProvider(jaxb2Marshaller));
        }

        final Map<Class<? extends RuntimeException>, Integer> exceptionStatusCodeMap = new HashMap<>();
        final Set<Object> componentsSet = new LinkedHashSet<Object>();
        if (this.componentProviders != null) {
            for (JerseyComponentsProvider provider : this.componentProviders) {
                Object[] components = provider.getJerseyComponents();
                if (components != null) {
                    for (Object component : components) {
                        if (component != null) {
                            componentsSet.add(component);
                        }
                    }
                }
                Map<Class<? extends RuntimeException>, Integer> map = provider.getExceptionStatusCodeMap();
                if (map != null) {
                    exceptionStatusCodeMap.putAll(map);
                }
            }
            for (Object component : componentsSet) {
                if (component != null) {
                    if (component instanceof String) {
                        try {
                            config.register(Class.forName(component.toString()));
                        } catch (Exception e) {
                            // log?
                        }
                    } else if (component instanceof Class<?>) {
                        config.register((Class<?>) component);
                    } else {
                        config.register(component);
                    }
                }
            }
        }

        this.config.register(new DefaultExceptionMapper(exceptionStatusCodeMap));

        // @formatter:off
        log.info("\n" + "**********************************************************************\n"
                + "*  Jersey Auto Configuration                                         *\n"
                + "**********************************************************************\n" + "properties = "
                + jersey + "\n" + "path (initialized) = " + path + "\n" + "components = " + componentsSet + "\n"
                + "**********************************************************************");
        // @formatter:on
    }

    @Bean
    @ConditionalOnMissingBean
    public FilterRegistrationBean requestContextFilter() {
        FilterRegistrationBean registration = new FilterRegistrationBean();
        registration.setFilter(new RequestContextFilter());
        registration.setOrder(this.jersey.getFilter().getOrder() - 1);
        registration.setName("requestContextFilter");
        return registration;
    }

    @Bean
    @ConditionalOnMissingBean(name = "jerseyFilterRegistration")
    @ConditionalOnProperty(prefix = "spring.jersey", name = "type", havingValue = "filter")
    public FilterRegistrationBean jerseyFilterRegistration() {
        FilterRegistrationBean registration = new FilterRegistrationBean();
        registration.setFilter(new ServletContainer(this.config));
        registration.setUrlPatterns(Arrays.asList(this.path));
        registration.setOrder(this.jersey.getFilter().getOrder());
        registration.addInitParameter(ServletProperties.FILTER_CONTEXT_PATH, stripPattern(this.path));
        addInitParameters(registration);
        registration.setName("jerseyFilter");
        registration.setDispatcherTypes(EnumSet.allOf(DispatcherType.class));
        return registration;
    }

    private String stripPattern(String path) {
        if (path.endsWith("/*")) {
            path = path.substring(0, path.lastIndexOf("/*"));
        }
        return path;
    }

    @Bean
    @ConditionalOnMissingBean(name = "jerseyServletRegistration")
    @ConditionalOnProperty(prefix = "spring.jersey", name = "type", havingValue = "servlet", matchIfMissing = true)
    public ServletRegistrationBean jerseyServletRegistration() {
        ServletRegistrationBean registration = new ServletRegistrationBean(new ServletContainer(this.config),
                this.path);
        addInitParameters(registration);
        registration.setName("jerseyServlet");
        return registration;
    }

    private void addInitParameters(RegistrationBean registration) {
        registration.addInitParameter(CommonProperties.METAINF_SERVICES_LOOKUP_DISABLE, "true");
        for (Entry<String, String> entry : this.jersey.getInit().entrySet()) {
            registration.addInitParameter(entry.getKey(), entry.getValue());
        }
    }

    @Override
    public void onStartup(ServletContext servletContext) throws ServletException {
        // We need to switch *off* the Jersey WebApplicationInitializer because it
        // will try and register a ContextLoaderListener which we don't need
        servletContext.setInitParameter("contextConfigLocation", "<NONE>");
    }

    private String findPath(String path) {
        if (!StringUtils.hasText(path)) {
            return "/*";
        }
        if (!path.startsWith("/")) {
            path = "/" + path;
        }
        while (path.endsWith("/*")) {
            path = path.substring(0, path.length() - 2);
        }
        return path.equals("/") ? "/*" : path + "/*";
    }

    @Provider
    @Produces({ MediaType.APPLICATION_XML, "text/xml", MediaType.WILDCARD })
    @Consumes({ MediaType.APPLICATION_XML, "text/xml" })
    public static class JAXBContextProvider implements ContextResolver<JAXBContext> {

        protected final Jaxb2Marshaller jaxb2Marshaller;

        public JAXBContextProvider(Jaxb2Marshaller jaxb2Marshaller) {
            this.jaxb2Marshaller = jaxb2Marshaller;
        }

        @Override
        public JAXBContext getContext(Class<?> type) {
            if (jaxb2Marshaller != null && type != null && jaxb2Marshaller.supports(type)) {
                return jaxb2Marshaller.getJaxbContext();
            }
            return null;
        }

    }

    @Provider
    public static class DefaultExceptionMapper
            implements org.glassfish.jersey.spi.ExtendedExceptionMapper<RuntimeException> {

        protected final Logger log = LoggerFactory.getLogger(getClass());

        private final Map<Class<? extends RuntimeException>, Integer> exceptionStatusCodeMap;

        public DefaultExceptionMapper() {
            this(null);
        }

        public DefaultExceptionMapper(Map<Class<? extends RuntimeException>, Integer> exceptionStatusCodeMap) {
            this.exceptionStatusCodeMap = exceptionStatusCodeMap == null
                    ? new HashMap<Class<? extends RuntimeException>, Integer>()
                    : exceptionStatusCodeMap;
            initExceptionStatusCodeMap();
        }

        protected void initExceptionStatusCodeMap() {
            addRuntimeExceptionClassToMap(RuntimeException.class, Status.INTERNAL_SERVER_ERROR.getStatusCode());
            addRuntimeExceptionClassToMap(IllegalArgumentException.class, Status.BAD_REQUEST.getStatusCode());
            addRuntimeExceptionClassToMap("org.springframework.security.access.AccessDeniedException",
                    Status.FORBIDDEN.getStatusCode());
            addRuntimeExceptionClassToMap("javax.persistence.EntityNotFoundException",
                    Status.NOT_FOUND.getStatusCode());
        }

        protected void addRuntimeExceptionClassToMap(Class<? extends RuntimeException> cls, int statusCode) {
            if (cls != null) {
                if (exceptionStatusCodeMap.get(cls) == null) {
                    exceptionStatusCodeMap.put(cls, statusCode);
                }
            }
        }

        @SuppressWarnings("unchecked")
        protected void addRuntimeExceptionClassToMap(String clsName, int statusCode) {
            if (!StringUtils.isEmpty(clsName)) {
                try {
                    Class<? extends RuntimeException> cls = (Class<? extends RuntimeException>) Class
                            .forName(clsName);
                    addRuntimeExceptionClassToMap(cls, statusCode);
                } catch (Throwable t) {
                    log.info("RuntimeException [" + clsName
                            + "] was not found. Status code mapping for this exception is not available.");
                }
            }
        }

        @Override
        public Response toResponse(RuntimeException exception) {
            if (exception != null) {
                Integer statusCode = exceptionStatusCodeMap.get(exception);
                // TODO
                int a = 0;
                //                if (statusCode == null && exception instanceof StatusCodeAwareException) {
                //                    statusCode = ((StatusCodeAwareException)exception).getStatusCode();
                //                }
                if (statusCode == null) {
                    statusCode = Status.INTERNAL_SERVER_ERROR.getStatusCode();
                }
                final String msg;
                if (StringUtils.isEmpty(exception.getMessage())) {
                    Status status = Status.fromStatusCode(statusCode);
                    msg = status != null ? status.getReasonPhrase() : "No reason";
                } else {
                    msg = exception.getMessage();
                }
                return Response.status(statusCode).entity(msg).type("text/plain").build();
            } else {
                return Response.serverError().build();
            }
        }

        @Override
        public boolean isMappable(RuntimeException exception) {
            if (exception != null) {
                // TODO
                int a = 0;
                //                if (exception instanceof StatusCodeAwareException) {
                //                    return true;
                //                }
                return exceptionStatusCodeMap.get(exception) != null;
            }
            return false;
        }

    }
}