/extensions/servlet/src/com/google/inject/servlet/ServletModule.java
Java | 357 lines | 86 code | 24 blank | 247 comment | 4 complexity | 164cf4c437a29d1fd4c3975f01a52a2d MD5 | raw file
- /**
- * Copyright (C) 2006 Google Inc.
- *
- * 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 com.google.inject.servlet;
- import static com.google.common.base.Preconditions.checkState;
- import com.google.common.collect.ImmutableList;
- import com.google.inject.AbstractModule;
- import com.google.inject.Key;
- import java.util.Map;
- import javax.servlet.Filter;
- import javax.servlet.ServletContext;
- import javax.servlet.http.HttpServlet;
- /**
- * Configures the servlet scopes and creates bindings for the servlet API
- * objects so you can inject the request, response, session, etc.
- *
- * <p>
- * You should subclass this module to register servlets and
- * filters in the {@link #configureServlets()} method.
- *
- * @author crazybob@google.com (Bob Lee)
- * @author dhanji@gmail.com (Dhanji R. Prasanna)
- */
- public class ServletModule extends AbstractModule {
- @Override
- protected final void configure() {
- checkState(filtersModuleBuilder == null, "Re-entry is not allowed.");
- checkState(servletsModuleBuilder == null, "Re-entry is not allowed.");
- filtersModuleBuilder = new FiltersModuleBuilder(binder());
- servletsModuleBuilder = new ServletsModuleBuilder(binder());
- try {
- // Install common bindings (skipped if already installed).
- install(new InternalServletModule());
-
- // Install local filter and servlet bindings.
- configureServlets();
- } finally {
- filtersModuleBuilder = null;
- servletsModuleBuilder = null;
- }
- }
- /**
- * <h3>Servlet Mapping EDSL</h3>
- *
- * <p> Part of the EDSL builder language for configuring servlets
- * and filters with guice-servlet. Think of this as an in-code replacement for web.xml.
- * Filters and servlets are configured here using simple java method calls. Here is a typical
- * example of registering a filter when creating your Guice injector:
- *
- * <pre>
- * Guice.createInjector(..., new ServletModule() {
- *
- * {@literal @}Override
- * protected void configureServlets() {
- * <b>serve("*.html").with(MyServlet.class)</b>
- * }
- * }
- * </pre>
- *
- * This registers a servlet (subclass of {@code HttpServlet}) called {@code MyServlet} to service
- * any web pages ending in {@code .html}. You can also use a path-style syntax to register
- * servlets:
- *
- * <pre>
- * <b>serve("/my/*").with(MyServlet.class)</b>
- * </pre>
- *
- * Every servlet (or filter) is required to be a singleton. If you cannot annotate the class
- * directly, you should add a separate {@code bind(..).in(Singleton.class)} rule elsewhere in
- * your module. Mapping a servlet that is bound under any other scope is an error.
- *
- * <p>
- * <h4>Dispatch Order</h4>
- * You are free to register as many servlets and filters as you like this way. They will
- * be compared and dispatched in the order in which the filter methods are called:
- *
- * <pre>
- *
- * Guice.createInjector(..., new ServletModule() {
- *
- * {@literal @}Override
- * protected void configureServlets() {
- * filter("/*").through(MyFilter.class);
- * filter("*.css").through(MyCssFilter.class);
- * filter("*.jpg").through(new MyJpgFilter());
- * // etc..
- *
- * serve("*.html").with(MyServlet.class);
- * serve("/my/*").with(MyServlet.class);
- * serve("*.jpg").with(new MyServlet());
- * // etc..
- * }
- * }
- * </pre>
- * This will traverse down the list of rules in lexical order. For example, a url
- * "{@code /my/file.js}" (after it runs through the matching filters) will first
- * be compared against the servlet mapping:
- *
- * <pre>
- * serve("*.html").with(MyServlet.class);
- * </pre>
- * And failing that, it will descend to the next servlet mapping:
- *
- * <pre>
- * serve("/my/*").with(MyServlet.class);
- * </pre>
- *
- * Since this rule matches, Guice Servlet will dispatch to {@code MyServlet}. These
- * two mapping rules can also be written in more compact form using varargs syntax:
- *
- * <pre>
- * serve(<b>"*.html", "/my/*"</b>).with(MyServlet.class);
- * </pre>
- *
- * This way you can map several URI patterns to the same servlet. A similar syntax is
- * also available for filter mappings.
- *
- * <p>
- * <h4>Regular Expressions</h4>
- * You can also map servlets (or filters) to URIs using regular expressions:
- * <pre>
- * <b>serveRegex("(.)*ajax(.)*").with(MyAjaxServlet.class)</b>
- * </pre>
- *
- * This will map any URI containing the text "ajax" in it to {@code MyAjaxServlet}. Such as:
- * <ul>
- * <li>http://www.google.com/ajax.html</li>
- * <li>http://www.google.com/content/ajax/index</li>
- * <li>http://www.google.com/it/is_totally_ajaxian</li>
- * </ul>
- *
- *
- * <h3>Initialization Parameters</h3>
- *
- * Servlets (and filters) allow you to pass in init params
- * using the {@code <init-param>} tag in web.xml. You can similarly pass in parameters to
- * Servlets and filters registered in Guice-servlet using a {@link java.util.Map} of parameter
- * name/value pairs. For example, to initialize {@code MyServlet} with two parameters
- * ({@code name="Dhanji", site="google.com"}) you could write:
- *
- * <pre>
- * Map<String, String> params = new HashMap<String, String>();
- * params.put("name", "Dhanji");
- * params.put("site", "google.com");
- *
- * ...
- * serve("/*").with(MyServlet.class, <b>params</b>)
- * </pre>
- *
- * <p>
- * <h3>Binding Keys</h3>
- *
- * You can also bind keys rather than classes. This lets you hide
- * implementations with package-local visbility and expose them using
- * only a Guice module and an annotation:
- *
- * <pre>
- * ...
- * filter("/*").through(<b>Key.get(Filter.class, Fave.class)</b>);
- * </pre>
- *
- * Where {@code Filter.class} refers to the Servlet API interface and {@code Fave.class} is a
- * custom binding annotation. Elsewhere (in one of your own modules) you can bind this
- * filter's implementation:
- *
- * <pre>
- * bind(Filter.class)<b>.annotatedWith(Fave.class)</b>.to(MyFilterImpl.class);
- * </pre>
- *
- * See {@link com.google.inject.Binder} for more information on binding syntax.
- *
- * <p>
- * <h3>Multiple Modules</h3>
- *
- * It is sometimes useful to capture servlet and filter mappings from multiple different
- * modules. This is essential if you want to package and offer drop-in Guice plugins that
- * provide servlet functionality.
- *
- * <p>
- * Guice Servlet allows you to register several instances of {@code ServletModule} to your
- * injector. The order in which these modules are installed determines the dispatch order
- * of filters and the precedence order of servlets. For example, if you had two servlet modules,
- * {@code RpcModule} and {@code WebServiceModule} and they each contained a filter that mapped
- * to the same URI pattern, {@code "/*"}:
- *
- * <p>
- * In {@code RpcModule}:
- * <pre>
- * filter("/*").through(RpcFilter.class);
- * </pre>
- *
- * In {@code WebServiceModule}:
- * <pre>
- * filter("/*").through(WebServiceFilter.class);
- * </pre>
- *
- * Then the order in which these filters are dispatched is determined by the order in which
- * the modules are installed:
- *
- * <pre>
- * <b>install(new WebServiceModule());</b>
- * install(new RpcModule());
- * </pre>
- *
- * In the case shown above {@code WebServiceFilter} will run first.
- *
- * @since 2.0
- */
- protected void configureServlets() {
- }
- private FiltersModuleBuilder filtersModuleBuilder;
- private ServletsModuleBuilder servletsModuleBuilder;
- private FiltersModuleBuilder getFiltersModuleBuilder() {
- checkState(filtersModuleBuilder != null,
- "This method can only be used inside configureServlets()");
- return filtersModuleBuilder;
- }
- private ServletsModuleBuilder getServletModuleBuilder() {
- checkState(servletsModuleBuilder != null,
- "This method can only be used inside configureServlets()");
- return servletsModuleBuilder;
- }
- /**
- * @param urlPattern Any Servlet-style pattern. examples: /*, /html/*, *.html, etc.
- * @since 2.0
- */
- protected final FilterKeyBindingBuilder filter(String urlPattern, String... morePatterns) {
- return getFiltersModuleBuilder()
- .filter(ImmutableList.<String>builder().add(urlPattern).add(morePatterns).build());
- }
- /**
- * @param urlPatterns Any Servlet-style patterns. examples: /*, /html/*, *.html, etc.
- * @since 5.0
- */
- protected final FilterKeyBindingBuilder filter(Iterable<String> urlPatterns) {
- return getFiltersModuleBuilder().filter(ImmutableList.copyOf(urlPatterns));
- }
- /**
- * @param regex Any Java-style regular expression.
- * @since 2.0
- */
- protected final FilterKeyBindingBuilder filterRegex(String regex, String... regexes) {
- return getFiltersModuleBuilder()
- .filterRegex(ImmutableList.<String>builder().add(regex).add(regexes).build());
- }
- /**
- * @param regexes Any Java-style regular expressions.
- * @since 5.0
- */
- protected final FilterKeyBindingBuilder filterRegex(Iterable<String> regexes) {
- return getFiltersModuleBuilder().filterRegex(ImmutableList.copyOf(regexes));
- }
- /**
- * @param urlPattern Any Servlet-style pattern. examples: /*, /html/*, *.html, etc.
- * @since 2.0
- */
- protected final ServletKeyBindingBuilder serve(String urlPattern, String... morePatterns) {
- return getServletModuleBuilder()
- .serve(ImmutableList.<String>builder().add(urlPattern).add(morePatterns).build());
- }
- /**
- * @param urlPatterns Any Servlet-style patterns. examples: /*, /html/*, *.html, etc.
- * @since 5.0
- */
- protected final ServletKeyBindingBuilder serve(Iterable<String> urlPatterns) {
- return getServletModuleBuilder().serve(ImmutableList.copyOf(urlPatterns));
- }
- /**
- * @param regex Any Java-style regular expression.
- * @since 2.0
- */
- protected final ServletKeyBindingBuilder serveRegex(String regex, String... regexes) {
- return getServletModuleBuilder()
- .serveRegex(ImmutableList.<String>builder().add(regex).add(regexes).build());
- }
- /**
- * @param regexes Any Java-style regular expressions.
- * @since 5.0
- */
- protected final ServletKeyBindingBuilder serveRegex(Iterable<String> regexes) {
- return getServletModuleBuilder().serveRegex(ImmutableList.copyOf(regexes));
- }
- /**
- * This method only works if you are using the {@linkplain GuiceServletContextListener} to
- * create your injector. Otherwise, it returns null.
- * @return The current servlet context.
- * @since 3.0
- */
- protected final ServletContext getServletContext() {
- return GuiceFilter.getServletContext();
- }
- /**
- * See the EDSL examples at {@link ServletModule#configureServlets()}
- *
- * @since 2.0
- */
- public static interface FilterKeyBindingBuilder {
- void through(Class<? extends Filter> filterKey);
- void through(Key<? extends Filter> filterKey);
- /** @since 3.0 */
- void through(Filter filter);
- void through(Class<? extends Filter> filterKey, Map<String, String> initParams);
- void through(Key<? extends Filter> filterKey, Map<String, String> initParams);
- /** @since 3.0 */
- void through(Filter filter, Map<String, String> initParams);
- }
- /**
- * See the EDSL examples at {@link ServletModule#configureServlets()}
- *
- * @since 2.0
- */
- public static interface ServletKeyBindingBuilder {
- void with(Class<? extends HttpServlet> servletKey);
- void with(Key<? extends HttpServlet> servletKey);
- /** @since 3.0 */
- void with(HttpServlet servlet);
- void with(Class<? extends HttpServlet> servletKey, Map<String, String> initParams);
- void with(Key<? extends HttpServlet> servletKey, Map<String, String> initParams);
- /** @since 3.0 */
- void with(HttpServlet servlet, Map<String, String> initParams);
- }
- }