Mercurial Hosting > luan
comparison src/org/eclipse/jetty/server/handler/GzipHandler.java @ 802:3428c60d7cfc
replace jetty jars with source
| author | Franklin Schmidt <fschmidt@gmail.com> |
|---|---|
| date | Wed, 07 Sep 2016 21:15:48 -0600 |
| parents | |
| children | 8e9db0bbf4f9 |
comparison
equal
deleted
inserted
replaced
| 801:6a21393191c1 | 802:3428c60d7cfc |
|---|---|
| 1 // | |
| 2 // ======================================================================== | |
| 3 // Copyright (c) 1995-2014 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 // | |
| 9 // The Eclipse Public License is available at | |
| 10 // http://www.eclipse.org/legal/epl-v10.html | |
| 11 // | |
| 12 // The Apache License v2.0 is available at | |
| 13 // http://www.opensource.org/licenses/apache2.0.php | |
| 14 // | |
| 15 // You may elect to redistribute this code under either of these licenses. | |
| 16 // ======================================================================== | |
| 17 // | |
| 18 | |
| 19 package org.eclipse.jetty.server.handler; | |
| 20 | |
| 21 import java.io.IOException; | |
| 22 import java.io.OutputStream; | |
| 23 import java.io.OutputStreamWriter; | |
| 24 import java.io.PrintWriter; | |
| 25 import java.io.UnsupportedEncodingException; | |
| 26 import java.util.HashSet; | |
| 27 import java.util.Set; | |
| 28 import java.util.StringTokenizer; | |
| 29 import java.util.zip.DeflaterOutputStream; | |
| 30 import java.util.zip.GZIPOutputStream; | |
| 31 | |
| 32 import javax.servlet.ServletException; | |
| 33 import javax.servlet.http.HttpServletRequest; | |
| 34 import javax.servlet.http.HttpServletResponse; | |
| 35 | |
| 36 import org.eclipse.jetty.continuation.Continuation; | |
| 37 import org.eclipse.jetty.continuation.ContinuationListener; | |
| 38 import org.eclipse.jetty.continuation.ContinuationSupport; | |
| 39 import org.eclipse.jetty.http.HttpMethods; | |
| 40 import org.eclipse.jetty.http.gzip.CompressedResponseWrapper; | |
| 41 import org.eclipse.jetty.http.gzip.AbstractCompressedStream; | |
| 42 import org.eclipse.jetty.server.Request; | |
| 43 import org.eclipse.jetty.util.log.Log; | |
| 44 import org.eclipse.jetty.util.log.Logger; | |
| 45 | |
| 46 /* ------------------------------------------------------------ */ | |
| 47 /** | |
| 48 * GZIP Handler This handler will gzip the content of a response if: | |
| 49 * <ul> | |
| 50 * <li>The filter is mapped to a matching path</li> | |
| 51 * <li>The response status code is >=200 and <300 | |
| 52 * <li>The content length is unknown or more than the <code>minGzipSize</code> initParameter or the minGzipSize is 0(default)</li> | |
| 53 * <li>The content-type is in the comma separated list of mimeTypes set in the <code>mimeTypes</code> initParameter or if no mimeTypes are defined the | |
| 54 * content-type is not "application/gzip"</li> | |
| 55 * <li>No content-encoding is specified by the resource</li> | |
| 56 * </ul> | |
| 57 * | |
| 58 * <p> | |
| 59 * Compressing the content can greatly improve the network bandwidth usage, but at a cost of memory and CPU cycles. If this handler is used for static content, | |
| 60 * then use of efficient direct NIO may be prevented, thus use of the gzip mechanism of the <code>org.eclipse.jetty.servlet.DefaultServlet</code> is advised instead. | |
| 61 * </p> | |
| 62 */ | |
| 63 public class GzipHandler extends HandlerWrapper | |
| 64 { | |
| 65 private static final Logger LOG = Log.getLogger(GzipHandler.class); | |
| 66 | |
| 67 protected Set<String> _mimeTypes; | |
| 68 protected Set<String> _excluded; | |
| 69 protected int _bufferSize = 8192; | |
| 70 protected int _minGzipSize = 256; | |
| 71 protected String _vary = "Accept-Encoding, User-Agent"; | |
| 72 | |
| 73 /* ------------------------------------------------------------ */ | |
| 74 /** | |
| 75 * Instantiates a new gzip handler. | |
| 76 */ | |
| 77 public GzipHandler() | |
| 78 { | |
| 79 } | |
| 80 | |
| 81 /* ------------------------------------------------------------ */ | |
| 82 /** | |
| 83 * Get the mime types. | |
| 84 * | |
| 85 * @return mime types to set | |
| 86 */ | |
| 87 public Set<String> getMimeTypes() | |
| 88 { | |
| 89 return _mimeTypes; | |
| 90 } | |
| 91 | |
| 92 /* ------------------------------------------------------------ */ | |
| 93 /** | |
| 94 * Set the mime types. | |
| 95 * | |
| 96 * @param mimeTypes | |
| 97 * the mime types to set | |
| 98 */ | |
| 99 public void setMimeTypes(Set<String> mimeTypes) | |
| 100 { | |
| 101 _mimeTypes = mimeTypes; | |
| 102 } | |
| 103 | |
| 104 /* ------------------------------------------------------------ */ | |
| 105 /** | |
| 106 * Set the mime types. | |
| 107 * | |
| 108 * @param mimeTypes | |
| 109 * the mime types to set | |
| 110 */ | |
| 111 public void setMimeTypes(String mimeTypes) | |
| 112 { | |
| 113 if (mimeTypes != null) | |
| 114 { | |
| 115 _mimeTypes = new HashSet<String>(); | |
| 116 StringTokenizer tok = new StringTokenizer(mimeTypes,",",false); | |
| 117 while (tok.hasMoreTokens()) | |
| 118 { | |
| 119 _mimeTypes.add(tok.nextToken()); | |
| 120 } | |
| 121 } | |
| 122 } | |
| 123 | |
| 124 /* ------------------------------------------------------------ */ | |
| 125 /** | |
| 126 * Get the excluded user agents. | |
| 127 * | |
| 128 * @return excluded user agents | |
| 129 */ | |
| 130 public Set<String> getExcluded() | |
| 131 { | |
| 132 return _excluded; | |
| 133 } | |
| 134 | |
| 135 /* ------------------------------------------------------------ */ | |
| 136 /** | |
| 137 * Set the excluded user agents. | |
| 138 * | |
| 139 * @param excluded | |
| 140 * excluded user agents to set | |
| 141 */ | |
| 142 public void setExcluded(Set<String> excluded) | |
| 143 { | |
| 144 _excluded = excluded; | |
| 145 } | |
| 146 | |
| 147 /* ------------------------------------------------------------ */ | |
| 148 /** | |
| 149 * Set the excluded user agents. | |
| 150 * | |
| 151 * @param excluded | |
| 152 * excluded user agents to set | |
| 153 */ | |
| 154 public void setExcluded(String excluded) | |
| 155 { | |
| 156 if (excluded != null) | |
| 157 { | |
| 158 _excluded = new HashSet<String>(); | |
| 159 StringTokenizer tok = new StringTokenizer(excluded,",",false); | |
| 160 while (tok.hasMoreTokens()) | |
| 161 _excluded.add(tok.nextToken()); | |
| 162 } | |
| 163 } | |
| 164 | |
| 165 /* ------------------------------------------------------------ */ | |
| 166 /** | |
| 167 * @return The value of the Vary header set if a response can be compressed. | |
| 168 */ | |
| 169 public String getVary() | |
| 170 { | |
| 171 return _vary; | |
| 172 } | |
| 173 | |
| 174 /* ------------------------------------------------------------ */ | |
| 175 /** | |
| 176 * Set the value of the Vary header sent with responses that could be compressed. | |
| 177 * <p> | |
| 178 * By default it is set to 'Accept-Encoding, User-Agent' since IE6 is excluded by | |
| 179 * default from the excludedAgents. If user-agents are not to be excluded, then | |
| 180 * this can be set to 'Accept-Encoding'. Note also that shared caches may cache | |
| 181 * many copies of a resource that is varied by User-Agent - one per variation of the | |
| 182 * User-Agent, unless the cache does some normalization of the UA string. | |
| 183 * @param vary The value of the Vary header set if a response can be compressed. | |
| 184 */ | |
| 185 public void setVary(String vary) | |
| 186 { | |
| 187 _vary = vary; | |
| 188 } | |
| 189 | |
| 190 /* ------------------------------------------------------------ */ | |
| 191 /** | |
| 192 * Get the buffer size. | |
| 193 * | |
| 194 * @return the buffer size | |
| 195 */ | |
| 196 public int getBufferSize() | |
| 197 { | |
| 198 return _bufferSize; | |
| 199 } | |
| 200 | |
| 201 /* ------------------------------------------------------------ */ | |
| 202 /** | |
| 203 * Set the buffer size. | |
| 204 * | |
| 205 * @param bufferSize | |
| 206 * buffer size to set | |
| 207 */ | |
| 208 public void setBufferSize(int bufferSize) | |
| 209 { | |
| 210 _bufferSize = bufferSize; | |
| 211 } | |
| 212 | |
| 213 /* ------------------------------------------------------------ */ | |
| 214 /** | |
| 215 * Get the minimum reponse size. | |
| 216 * | |
| 217 * @return minimum reponse size | |
| 218 */ | |
| 219 public int getMinGzipSize() | |
| 220 { | |
| 221 return _minGzipSize; | |
| 222 } | |
| 223 | |
| 224 /* ------------------------------------------------------------ */ | |
| 225 /** | |
| 226 * Set the minimum reponse size. | |
| 227 * | |
| 228 * @param minGzipSize | |
| 229 * minimum reponse size | |
| 230 */ | |
| 231 public void setMinGzipSize(int minGzipSize) | |
| 232 { | |
| 233 _minGzipSize = minGzipSize; | |
| 234 } | |
| 235 | |
| 236 /* ------------------------------------------------------------ */ | |
| 237 /** | |
| 238 * @see org.eclipse.jetty.server.handler.HandlerWrapper#handle(java.lang.String, org.eclipse.jetty.server.Request, javax.servlet.http.HttpServletRequest, javax.servlet.http.HttpServletResponse) | |
| 239 */ | |
| 240 @Override | |
| 241 public void handle(String target, Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException | |
| 242 { | |
| 243 if (_handler!=null && isStarted()) | |
| 244 { | |
| 245 String ae = request.getHeader("accept-encoding"); | |
| 246 if (ae != null && ae.indexOf("gzip")>=0 && !response.containsHeader("Content-Encoding") | |
| 247 && !HttpMethods.HEAD.equalsIgnoreCase(request.getMethod())) | |
| 248 { | |
| 249 if (_excluded!=null) | |
| 250 { | |
| 251 String ua = request.getHeader("User-Agent"); | |
| 252 if (_excluded.contains(ua)) | |
| 253 { | |
| 254 _handler.handle(target,baseRequest, request, response); | |
| 255 return; | |
| 256 } | |
| 257 } | |
| 258 | |
| 259 final CompressedResponseWrapper wrappedResponse = newGzipResponseWrapper(request,response); | |
| 260 | |
| 261 boolean exceptional=true; | |
| 262 try | |
| 263 { | |
| 264 _handler.handle(target, baseRequest, request, wrappedResponse); | |
| 265 exceptional=false; | |
| 266 } | |
| 267 finally | |
| 268 { | |
| 269 Continuation continuation = ContinuationSupport.getContinuation(request); | |
| 270 if (continuation.isSuspended() && continuation.isResponseWrapped()) | |
| 271 { | |
| 272 continuation.addContinuationListener(new ContinuationListener() | |
| 273 { | |
| 274 public void onComplete(Continuation continuation) | |
| 275 { | |
| 276 try | |
| 277 { | |
| 278 wrappedResponse.finish(); | |
| 279 } | |
| 280 catch(IOException e) | |
| 281 { | |
| 282 LOG.warn(e); | |
| 283 } | |
| 284 } | |
| 285 | |
| 286 public void onTimeout(Continuation continuation) | |
| 287 {} | |
| 288 }); | |
| 289 } | |
| 290 else if (exceptional && !response.isCommitted()) | |
| 291 { | |
| 292 wrappedResponse.resetBuffer(); | |
| 293 wrappedResponse.noCompression(); | |
| 294 } | |
| 295 else | |
| 296 wrappedResponse.finish(); | |
| 297 } | |
| 298 } | |
| 299 else | |
| 300 { | |
| 301 _handler.handle(target,baseRequest, request, response); | |
| 302 } | |
| 303 } | |
| 304 } | |
| 305 | |
| 306 /** | |
| 307 * Allows derived implementations to replace ResponseWrapper implementation. | |
| 308 * | |
| 309 * @param request the request | |
| 310 * @param response the response | |
| 311 * @return the gzip response wrapper | |
| 312 */ | |
| 313 protected CompressedResponseWrapper newGzipResponseWrapper(HttpServletRequest request, HttpServletResponse response) | |
| 314 { | |
| 315 return new CompressedResponseWrapper(request,response) | |
| 316 { | |
| 317 { | |
| 318 super.setMimeTypes(GzipHandler.this._mimeTypes); | |
| 319 super.setBufferSize(GzipHandler.this._bufferSize); | |
| 320 super.setMinCompressSize(GzipHandler.this._minGzipSize); | |
| 321 } | |
| 322 | |
| 323 @Override | |
| 324 protected AbstractCompressedStream newCompressedStream(HttpServletRequest request,HttpServletResponse response) throws IOException | |
| 325 { | |
| 326 return new AbstractCompressedStream("gzip",request,this,_vary) | |
| 327 { | |
| 328 @Override | |
| 329 protected DeflaterOutputStream createStream() throws IOException | |
| 330 { | |
| 331 return new GZIPOutputStream(_response.getOutputStream(),_bufferSize); | |
| 332 } | |
| 333 }; | |
| 334 } | |
| 335 | |
| 336 @Override | |
| 337 protected PrintWriter newWriter(OutputStream out,String encoding) throws UnsupportedEncodingException | |
| 338 { | |
| 339 return GzipHandler.this.newWriter(out,encoding); | |
| 340 } | |
| 341 }; | |
| 342 } | |
| 343 | |
| 344 /** | |
| 345 * Allows derived implementations to replace PrintWriter implementation. | |
| 346 * | |
| 347 * @param out the out | |
| 348 * @param encoding the encoding | |
| 349 * @return the prints the writer | |
| 350 * @throws UnsupportedEncodingException | |
| 351 */ | |
| 352 protected PrintWriter newWriter(OutputStream out,String encoding) throws UnsupportedEncodingException | |
| 353 { | |
| 354 return encoding==null?new PrintWriter(out):new PrintWriter(new OutputStreamWriter(out,encoding)); | |
| 355 } | |
| 356 } |
