View Javadoc

1   // ========================================================================
2   // $Id$
3   // Copyright (c) 2004-2009 Mort Bay Consulting Pty. Ltd.
4   // ------------------------------------------------------------------------
5   // All rights reserved. This program and the accompanying materials
6   // are made available under the terms of the Eclipse Public License v1.0
7   // and Apache License v2.0 which accompanies this distribution.
8   // The Eclipse Public License is available at
9   // http://www.eclipse.org/legal/epl-v10.html
10  // The Apache License v2.0 is available at
11  // http://www.opensource.org/licenses/apache2.0.php
12  // You may elect to redistribute this code under either of these licenses.
13  // ========================================================================
14  package org.eclipse.jetty.rewrite.handler;
15  
16  import java.io.IOException;
17  
18  import javax.servlet.ServletException;
19  import javax.servlet.http.HttpServletRequest;
20  import javax.servlet.http.HttpServletResponse;
21  
22  import org.eclipse.jetty.server.Request;
23  import org.eclipse.jetty.server.handler.HandlerWrapper;
24  
25  /* ------------------------------------------------------------ */
26  /**
27   *<p> Rewrite handler is responsible for managing the rules. Its capabilities
28   * is not only limited for URL rewrites such as RewritePatternRule or RewriteRegexRule.
29   * There is also handling for cookies, headers, redirection, setting status or error codes
30   * whenever the rule finds a match.
31   *
32   * <p> The rules can be matched by the either: pattern matching of PathMap
33   * (eg {@link PatternRule}), regular expressions (eg {@link RegexRule}) or certain conditions set
34   * (eg {@link MsieSslRule} - the requests must be in SSL mode).
35   *
36   * <p> The rules can be grouped into rule containers (class {@link RuleContainer}), and will only
37   * be applied if the request matches the conditions for their container
38   * (e.g., by virtual host name)
39   *
40   * <p>The list of predefined rules is:
41   * <ul>
42   * <li> {@link CookiePatternRule} - adds a new cookie in response. </li>
43   * <li> {@link HeaderPatternRule} - adds/modifies the HTTP headers in response. </li>
44   * <li> {@link RedirectPatternRule} - sets the redirect location. </li>
45   * <li> {@link ResponsePatternRule} - sets the status/error codes. </li>
46   * <li> {@link RewritePatternRule} - rewrites the requested URI. </li>
47   * <li> {@link RewriteRegexRule} - rewrites the requested URI using regular expression for pattern matching. </li>
48   * <li> {@link ProxyRule} - proxies the requested URI to the host defined in proxyTo. </li>
49   * <li> {@link MsieSslRule} - disables the keep alive on SSL for IE5 and IE6. </li>
50   * <li> {@link LegacyRule} - the old version of rewrite. </li>
51   * <li> {@link ForwardedSchemeHeaderRule} - set the scheme according to the headers present. </li>
52   * <li> {@link VirtualHostRuleContainer} - checks whether the request matches one of a set of virtual host names.</li>
53   * </ul>
54   *
55   *
56   * Here is a typical jetty.xml configuration would be: <pre>
57   *
58   *   &lt;Set name="handler"&gt;
59   *     &lt;New id="Handlers" class="org.eclipse.jetty.rewrite.handler.RewriteHandler"&gt;
60   *       &lt;Set name="rules"&gt;
61   *         &lt;Array type="org.eclipse.jetty.rewrite.handler.Rule"&gt;
62   *
63   *           &lt;Item&gt;
64   *             &lt;New id="rewrite" class="org.eclipse.jetty.rewrite.handler.RewritePatternRule"&gt;
65   *               &lt;Set name="pattern"&gt;/*&lt;/Set&gt;
66   *               &lt;Set name="replacement"&gt;/test&lt;/Set&gt;
67   *             &lt;/New&gt;
68   *           &lt;/Item&gt;
69   *
70   *           &lt;Item&gt;
71   *             &lt;New id="rewrite" class="org.eclipse.jetty.rewrite.handler.ProxyRule"&gt;
72   *               &lt;Set name="pattern"&gt;/*&lt;/Set&gt;
73   *               &lt;Set name="proxyTo"&gt;http://webtide.com:8080&lt;/Set&gt;
74   *             &lt;/New&gt;
75   *           &lt;/Item&gt;
76   *
77   *           &lt;Item&gt;
78   *             &lt;New id="response" class="org.eclipse.jetty.rewrite.handler.ResponsePatternRule"&gt;
79   *               &lt;Set name="pattern"&gt;/session/&lt;/Set&gt;
80   *               &lt;Set name="code"&gt;400&lt;/Set&gt;
81   *               &lt;Set name="reason"&gt;Setting error code 400&lt;/Set&gt;
82   *             &lt;/New&gt;
83   *           &lt;/Item&gt;
84   *
85   *           &lt;Item&gt;
86   *             &lt;New id="header" class="org.eclipse.jetty.rewrite.handler.HeaderPatternRule"&gt;
87   *               &lt;Set name="pattern"&gt;*.jsp&lt;/Set&gt;
88   *               &lt;Set name="name"&gt;server&lt;/Set&gt;
89   *               &lt;Set name="value"&gt;dexter webserver&lt;/Set&gt;
90   *             &lt;/New&gt;
91   *           &lt;/Item&gt;
92   *
93   *           &lt;Item&gt;
94   *             &lt;New id="header" class="org.eclipse.jetty.rewrite.handler.HeaderPatternRule"&gt;
95   *               &lt;Set name="pattern"&gt;*.jsp&lt;/Set&gt;
96   *               &lt;Set name="name"&gt;title&lt;/Set&gt;
97   *               &lt;Set name="value"&gt;driven header purpose&lt;/Set&gt;
98   *             &lt;/New&gt;
99   *           &lt;/Item&gt;
100  *
101  *           &lt;Item&gt;
102  *             &lt;New id="redirect" class="org.eclipse.jetty.rewrite.handler.RedirectPatternRule"&gt;
103  *               &lt;Set name="pattern"&gt;/test/dispatch&lt;/Set&gt;
104  *               &lt;Set name="location"&gt;http://jetty.eclipse.org&lt;/Set&gt;
105  *             &lt;/New&gt;
106  *           &lt;/Item&gt;
107  *
108  *           &lt;Item&gt;
109  *             &lt;New id="regexRewrite" class="org.eclipse.jetty.rewrite.handler.RewriteRegexRule"&gt;
110  *               &lt;Set name="regex"&gt;/test-jaas/$&lt;/Set&gt;
111  *               &lt;Set name="replacement"&gt;/demo&lt;/Set&gt;
112  *             &lt;/New&gt;
113  *           &lt;/Item&gt;
114  *
115  *           &lt;Item&gt;
116  *             &lt;New id="forwardedHttps" class="org.eclipse.jetty.rewrite.handler.ForwardedSchemeHeaderRule"&gt;
117  *               &lt;Set name="header"&gt;X-Forwarded-Scheme&lt;/Set&gt;
118  *               &lt;Set name="headerValue"&gt;https&lt;/Set&gt;
119  *               &lt;Set name="scheme"&gt;https&lt;/Set&gt;
120  *             &lt;/New&gt;
121  *           &lt;/Item&gt;
122  *
123  *           &lt;Item&gt;
124  *             &lt;New id="virtualHost" class="org.eclipse.jetty.rewrite.handler.VirtualHostRuleContainer"&gt;
125  *
126  *               &lt;Set name="virtualHosts"&gt;
127  *                 &lt;Array type="java.lang.String"&gt;
128  *                   &lt;Item&gt;eclipse.com&lt;/Item&gt;
129  *                   &lt;Item&gt;www.eclipse.com&lt;/Item&gt;
130  *                   &lt;Item&gt;eclipse.org&lt;/Item&gt;
131  *                   &lt;Item&gt;www.eclipse.org&lt;/Item&gt;
132  *                 &lt;/Array&gt;
133  *               &lt;/Set&gt;
134  *
135  *               &lt;Call name="addRule"&gt;
136  *                 &lt;Arg&gt;
137  *                   &lt;New class="org.eclipse.jetty.rewrite.handler.CookiePatternRule"&gt;
138  *                     &lt;Set name="pattern"&gt;/*&lt;/Set&gt;
139  *                     &lt;Set name="name"&gt;CookiePatternRule&lt;/Set&gt;
140  *                     &lt;Set name="value"&gt;1&lt;/Set&gt;
141  *                   &lt;/New&gt;
142  *                 &lt;/Arg&gt;
143  *               &lt;/Call&gt;
144  *
145  *             &lt;/New&gt;
146  *           &lt;/      Item&gt;
147  *
148  *         &lt;/Array&gt;
149  *       &lt;/Set&gt;
150  *
151  *       &lt;Set name="handler"&gt;
152  *         &lt;New id="Handlers" class="org.eclipse.jetty.server.handler.HandlerCollection"&gt;
153  *           &lt;Set name="handlers"&gt;
154  *            &lt;Array type="org.eclipse.jetty.server.Handler"&gt;
155  *              &lt;Item&gt;
156  *                &lt;New id="Contexts" class="org.eclipse.jetty.server.handler.ContextHandlerCollection"/&gt;
157  *              &lt;/Item&gt;
158  *              &lt;Item&gt;
159  *                &lt;New id="DefaultHandler" class="org.eclipse.jetty.server.handler.DefaultHandler"/&gt;
160  *              &lt;/Item&gt;
161  *              &lt;Item&gt;
162  *                &lt;New id="RequestLog" class="org.eclipse.jetty.server.handler.RequestLogHandler"/&gt;
163  *              &lt;/Item&gt;
164  *            &lt;/Array&gt;
165  *           &lt;/Set&gt;
166  *         &lt;/New&gt;
167  *       &lt;/Set&gt;
168  *
169  *     &lt;/New&gt;
170  *   &lt;/Set&gt;
171  * </pre>
172  *
173  */
174 public class RewriteHandler extends HandlerWrapper
175 {
176 
177     private RuleContainer _rules;
178 
179     /* ------------------------------------------------------------ */
180     public RewriteHandler()
181     {
182         _rules = new RuleContainer();
183     }
184 
185     /* ------------------------------------------------------------ */
186     /**
187      * To enable configuration from jetty.xml on rewriteRequestURI, rewritePathInfo and
188      * originalPathAttribute
189      *
190      * @param legacyRule old style rewrite rule
191      */
192     @Deprecated
193     public void setLegacyRule(LegacyRule legacyRule)
194     {
195         _rules.setLegacyRule(legacyRule);
196     }
197 
198     /* ------------------------------------------------------------ */
199     /**
200      * Returns the list of rules.
201      * @return an array of {@link Rule}.
202      */
203     public Rule[] getRules()
204     {
205         return _rules.getRules();
206     }
207 
208     /* ------------------------------------------------------------ */
209     /**
210      * Assigns the rules to process.
211      * @param rules an array of {@link Rule}.
212      */
213     public void setRules(Rule[] rules)
214     {
215         _rules.setRules(rules);
216     }
217 
218     /*------------------------------------------------------------ */
219     /**
220      * Assigns the rules to process.
221      * @param rules a {@link RuleContainer} containing other rules to process
222      */
223     public void setRules(RuleContainer rules)
224     {
225         _rules = rules;
226     }
227 
228     /* ------------------------------------------------------------ */
229     /**
230      * Add a Rule
231      * @param rule The rule to add to the end of the rules array
232      */
233     public void addRule(Rule rule)
234     {
235         _rules.addRule(rule);
236     }
237 
238     /* ------------------------------------------------------------ */
239     /**
240      * @return the rewriteRequestURI If true, this handler will rewrite the value
241      * returned by {@link HttpServletRequest#getRequestURI()}.
242      */
243     public boolean isRewriteRequestURI()
244     {
245         return _rules.isRewriteRequestURI();
246     }
247 
248     /* ------------------------------------------------------------ */
249     /**
250      * @param rewriteRequestURI true if this handler will rewrite the value
251      * returned by {@link HttpServletRequest#getRequestURI()}.
252      */
253     public void setRewriteRequestURI(boolean rewriteRequestURI)
254     {
255         _rules.setRewriteRequestURI(rewriteRequestURI);
256     }
257 
258     /* ------------------------------------------------------------ */
259     /**
260      * @return true if this handler will rewrite the value
261      * returned by {@link HttpServletRequest#getPathInfo()}.
262      */
263     public boolean isRewritePathInfo()
264     {
265         return _rules.isRewritePathInfo();
266     }
267 
268     /* ------------------------------------------------------------ */
269     /**
270      * @param rewritePathInfo true if this handler will rewrite the value
271      * returned by {@link HttpServletRequest#getPathInfo()}.
272      */
273     public void setRewritePathInfo(boolean rewritePathInfo)
274     {
275         _rules.setRewritePathInfo(rewritePathInfo);
276     }
277 
278     /* ------------------------------------------------------------ */
279     /**
280      * @return the originalPathAttribte. If non null, this string will be used
281      * as the attribute name to store the original request path.
282      */
283     public String getOriginalPathAttribute()
284     {
285         return _rules.getOriginalPathAttribute();
286     }
287 
288     /* ------------------------------------------------------------ */
289     /**
290      * @param originalPathAttribute If non null, this string will be used
291      * as the attribute name to store the original request path.
292      */
293     public void setOriginalPathAttribute(String originalPathAttribute)
294     {
295         _rules.setOriginalPathAttribute(originalPathAttribute);
296     }
297 
298 
299     /* ------------------------------------------------------------ */
300     /* (non-Javadoc)
301      * @see org.eclipse.jetty.server.handler.HandlerWrapper#handle(java.lang.String, javax.servlet.http.HttpServletRequest, javax.servlet.http.HttpServletResponse, int)
302      */
303     @Override
304     public void handle(String target, Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException
305     {
306         if (isStarted())
307         {
308             String returned = _rules.matchAndApply(target, request, response);
309             target = (returned == null) ? target : returned;
310 
311             if (!baseRequest.isHandled())
312                 super.handle(target, baseRequest, request, response);
313         }
314     }
315 
316 }