servletcontainer.java

来自「resetful样式的ws样例,一种面向资源的webservices服务」· Java 代码 · 共 549 行 · 第 1/2 页

JAVA
549
字号
/* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. *  * Copyright 1997-2007 Sun Microsystems, Inc. All rights reserved. *  * The contents of this file are subject to the terms of either the GNU * General Public License Version 2 only ("GPL") or the Common Development * and Distribution License("CDDL") (collectively, the "License").  You * may not use this file except in compliance with the License. You can obtain * a copy of the License at https://jersey.dev.java.net/CDDL+GPL.html * or jersey/legal/LICENSE.txt.  See the License for the specific * language governing permissions and limitations under the License. *  * When distributing the software, include this License Header Notice in each * file and include the License file at jersey/legal/LICENSE.txt. * Sun designates this particular file as subject to the "Classpath" exception * as provided by Sun in the GPL Version 2 section of the License file that * accompanied this code.  If applicable, add the following below the License * Header, with the fields enclosed by brackets [] replaced by your own * identifying information: "Portions Copyrighted [year] * [name of copyright owner]" *  * Contributor(s): *  * If you wish your version of this file to be governed by only the CDDL or * only the GPL Version 2, indicate your decision by adding "[Contributor] * elects to include this software in this distribution under the [CDDL or GPL * Version 2] license."  If you don't indicate a single choice of license, a * recipient has the option to distribute your version of this file under * either the CDDL, the GPL Version 2 or to extend the choice of license to * its licensees as provided above.  However, if you add GPL Version 2 code * and therefore, elected the GPL Version 2 license, then the option applies * only if the new code is made subject to such option by the copyright * holder. */package com.sun.jersey.spi.container.servlet;import com.sun.jersey.api.container.ContainerException;import com.sun.jersey.api.core.ApplicationConfigAdapter;import com.sun.jersey.api.core.ResourceConfig;import com.sun.jersey.impl.ThreadLocalInvoker;import com.sun.jersey.api.core.ClasspathResourceConfig;import com.sun.jersey.api.core.PackagesResourceConfig;import com.sun.jersey.api.uri.UriComponent;import com.sun.jersey.impl.container.servlet.JSPTemplateProcessor;import com.sun.jersey.impl.container.servlet.ServletContainerRequest;import com.sun.jersey.spi.container.ContainerListener;import com.sun.jersey.spi.container.ContainerNotifier;import com.sun.jersey.spi.container.ContainerRequest;import com.sun.jersey.spi.container.ContainerResponse;import com.sun.jersey.spi.container.ContainerResponseWriter;import com.sun.jersey.spi.container.InBoundHeaders;import com.sun.jersey.spi.container.WebApplication;import com.sun.jersey.spi.container.WebApplicationFactory;import com.sun.jersey.spi.inject.SingletonTypeInjectableProvider;import java.io.IOException;import java.io.OutputStream;import java.lang.reflect.Constructor;import java.lang.reflect.Proxy;import java.lang.reflect.Type;import java.net.URI;import java.util.ArrayList;import java.util.Enumeration;import java.util.HashMap;import java.util.LinkedList;import java.util.List;import java.util.Map;import javax.servlet.ServletConfig;import javax.servlet.ServletContext;import javax.servlet.ServletException;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse;import javax.ws.rs.core.ApplicationConfig;import javax.ws.rs.core.Context;import javax.ws.rs.core.MultivaluedMap;import javax.ws.rs.core.UriBuilder;/** * A servlet container for deploying root resource classes. * <p> * The web.xml MAY configure the servlet to have an initialization parameter  * "com.sun.jersey.config.property.resourceConfigClass" and whose value is a  * fully qualified name of a class that implements {@link ResourceConfig}.  * If the concrete class has a constructor that takes a single parameter of the  * type Map then the class is instantiated with that constructor and an instance  * of Map that contains all the initialization parameters is passed as the parameter. * Otherwise the default contructor is used to instantate the class. * <p> * If the initialization parameter  * "com.sun.jersey.config.property.resourceConfigClass" is not present a new  * instance of {@link ClasspathResourceConfig} is created. The initialization  * parameter "com.sun.jersey.config.property.classpath" MAY be set to provide  * one or more paths. Each path MUST be separated by ';'. Each path MUST * be a virtual path as specified by the {@link ServletContext#getRealPath} method, * and each path is transformed by that method. The transformed paths are * added as a property value to a Map instance using the property name  * (@link ClasspathResourceConfig.PROPERTY_CLASSPATH}. Any additional  * initialization parameters are then added to the Map instance. Then that Map * instance is passe to the constructor of {@link ClasspathResourceConfig}. * If this parameter is not set then the  * default value is set to the following virtual paths:  * "/WEB-INF/lib;/WEB-INF/classes". * <p> * A new {@link WebApplication} instance will be created and configured such * that the following classes may be injected onto the field of a root  * resource class or a parameter of a method of root resource class that is  * annotated with {@link javax.ws.rs.core.Context}: {@link HttpServletRequest},  * {@link HttpServletResponse}, {@link ServletContext}, and {@link ServletConfig}. */public class ServletContainer extends HttpServlet implements ContainerListener {    public static final String APPLICATION_CONFIG_CLASS =            "javax.ws.rs.ApplicationConfig";        public static final String RESOURCE_CONFIG_CLASS =             "com.sun.jersey.config.property.resourceConfigClass";        private final ThreadLocalInvoker<HttpServletRequest> requestInvoker =            new ThreadLocalInvoker<HttpServletRequest>();        private final ThreadLocalInvoker<HttpServletResponse> responseInvoker =            new ThreadLocalInvoker<HttpServletResponse>();        private ServletConfig config;        private ServletContext context;    private ResourceConfig resourceConfig;        private WebApplication application;        @Override    public final void init(ServletConfig servletConfig) throws ServletException {        super.init(servletConfig);            config = servletConfig;                context = config.getServletContext();                resourceConfig = createResourceConfig(config);        initResourceConfigFeatures(servletConfig, resourceConfig);                load();                Object o = resourceConfig.getProperties().get(                ResourceConfig.PROPERTY_CONTAINER_NOTIFIER);        if (o instanceof ContainerNotifier) {            ContainerNotifier crf = (ContainerNotifier)o;            crf.addListener(this);        }            }        private final static class Writer extends OutputStream implements ContainerResponseWriter {        final HttpServletResponse response;                Writer(HttpServletResponse response) {            this.response = response;        }        public OutputStream writeStatusAndHeaders(long contentLength,                 ContainerResponse cResponse) throws IOException {            response.setStatus(cResponse.getStatus());            if (contentLength != -1 && contentLength < Integer.MAX_VALUE)                 response.setContentLength((int)contentLength);                    MultivaluedMap<String, Object> headers = cResponse.getHttpHeaders();            for (Map.Entry<String, List<Object>> e : headers.entrySet()) {                for (Object v : e.getValue()) {                    response.addHeader(e.getKey(), ContainerResponse.getHeaderValue(v));                }            }                        return this;        }                OutputStream out;                public void write(int b) throws IOException {            initiate();            out.write(b);        }        @Override        public void write(byte b[]) throws IOException {            initiate();            out.write(b);        }        @Override        public void write(byte b[], int off, int len) throws IOException {            initiate();            out.write(b, off, len);        }        @Override        public void flush() throws IOException {            initiate();            out.flush();        }        @Override        public void close() throws IOException {            initiate();            out.close();        }                void initiate() throws IOException {            if (out == null)                out = response.getOutputStream();        }            }        @Override    public void service(HttpServletRequest request, HttpServletResponse response)    throws ServletException, IOException {        /**         * There is an annoying edge case where the service method is         * invoked for the case when the URI is equal to the deployment URL         * minus the '/', for example http://locahost:8080/HelloWorldWebApp         */        if (request.getPathInfo() != null &&                 request.getPathInfo().equals("/") && !request.getRequestURI().endsWith("/")) {            response.setStatus(404);            return;                    }                // Copy the application field to local instance to ensure that the         // currently loaded web application is used to process        // request        final WebApplication _application = application;                        /**         * The HttpServletRequest.getRequestURL() contains the complete URI         * minus the query and fragment components.         */        UriBuilder absoluteUriBuilder = UriBuilder.fromUri(                request.getRequestURL().toString());                /**         * The HttpServletRequest.getPathInfo() and          * HttpServletRequest.getServletPath() are in decoded form.         *         * On some servlet implementations the getPathInfo() removed         * contiguous '/' characters. This is problematic if URIs         * are embedded, for example as the last path segment.         * We need to work around this and not use getPathInfo         * for the decodedPath.         */        final String decodedBasePath = (request.getPathInfo() != null)            ? request.getContextPath() + request.getServletPath() + "/"            : request.getContextPath() + "/";                final String encodedBasePath = UriComponent.encode(decodedBasePath,                 UriComponent.Type.PATH);                if (!decodedBasePath.equals(encodedBasePath)) {            throw new ContainerException("The servlet context path and/or the " +                    "servlet path contain characters that are percent enocded");        }                final URI baseUri = absoluteUriBuilder.encode(false).                replacePath(encodedBasePath).                build();                String queryParameters = request.getQueryString();        if (queryParameters == null) queryParameters = "";                final URI requestUri = absoluteUriBuilder.encode(false).                replacePath(request.getRequestURI()).                replaceQueryParams(queryParameters).

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?