Mercurial Hosting > luan
comparison src/org/eclipse/jetty/http/HttpParser.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.http; | |
| 20 | |
| 21 import java.io.IOException; | |
| 22 | |
| 23 import org.eclipse.jetty.io.Buffer; | |
| 24 import org.eclipse.jetty.io.BufferCache.CachedBuffer; | |
| 25 import org.eclipse.jetty.io.BufferUtil; | |
| 26 import org.eclipse.jetty.io.Buffers; | |
| 27 import org.eclipse.jetty.io.ByteArrayBuffer; | |
| 28 import org.eclipse.jetty.io.EndPoint; | |
| 29 import org.eclipse.jetty.io.EofException; | |
| 30 import org.eclipse.jetty.io.View; | |
| 31 import org.eclipse.jetty.io.bio.StreamEndPoint; | |
| 32 import org.eclipse.jetty.util.StringUtil; | |
| 33 import org.eclipse.jetty.util.log.Log; | |
| 34 import org.eclipse.jetty.util.log.Logger; | |
| 35 | |
| 36 public class HttpParser implements Parser | |
| 37 { | |
| 38 private static final Logger LOG = Log.getLogger(HttpParser.class); | |
| 39 | |
| 40 // States | |
| 41 public static final int STATE_START=-14; | |
| 42 public static final int STATE_FIELD0=-13; | |
| 43 public static final int STATE_SPACE1=-12; | |
| 44 public static final int STATE_STATUS=-11; | |
| 45 public static final int STATE_URI=-10; | |
| 46 public static final int STATE_SPACE2=-9; | |
| 47 public static final int STATE_END0=-8; | |
| 48 public static final int STATE_END1=-7; | |
| 49 public static final int STATE_FIELD2=-6; | |
| 50 public static final int STATE_HEADER=-5; | |
| 51 public static final int STATE_HEADER_NAME=-4; | |
| 52 public static final int STATE_HEADER_IN_NAME=-3; | |
| 53 public static final int STATE_HEADER_VALUE=-2; | |
| 54 public static final int STATE_HEADER_IN_VALUE=-1; | |
| 55 public static final int STATE_END=0; | |
| 56 public static final int STATE_EOF_CONTENT=1; | |
| 57 public static final int STATE_CONTENT=2; | |
| 58 public static final int STATE_CHUNKED_CONTENT=3; | |
| 59 public static final int STATE_CHUNK_SIZE=4; | |
| 60 public static final int STATE_CHUNK_PARAMS=5; | |
| 61 public static final int STATE_CHUNK=6; | |
| 62 public static final int STATE_SEEKING_EOF=7; | |
| 63 | |
| 64 private final EventHandler _handler; | |
| 65 private final Buffers _buffers; // source of buffers | |
| 66 private final EndPoint _endp; | |
| 67 private Buffer _header; // Buffer for header data (and small _content) | |
| 68 private Buffer _body; // Buffer for large content | |
| 69 private Buffer _buffer; // The current buffer in use (either _header or _content) | |
| 70 private CachedBuffer _cached; | |
| 71 private final View.CaseInsensitive _tok0; // Saved token: header name, request method or response version | |
| 72 private final View.CaseInsensitive _tok1; // Saved token: header value, request URI or response code | |
| 73 private String _multiLineValue; | |
| 74 private int _responseStatus; // If >0 then we are parsing a response | |
| 75 private boolean _forceContentBuffer; | |
| 76 private boolean _persistent; | |
| 77 | |
| 78 /* ------------------------------------------------------------------------------- */ | |
| 79 protected final View _contentView=new View(); // View of the content in the buffer for {@link Input} | |
| 80 protected int _state=STATE_START; | |
| 81 protected byte _eol; | |
| 82 protected int _length; | |
| 83 protected long _contentLength; | |
| 84 protected long _contentPosition; | |
| 85 protected int _chunkLength; | |
| 86 protected int _chunkPosition; | |
| 87 private boolean _headResponse; | |
| 88 | |
| 89 /* ------------------------------------------------------------------------------- */ | |
| 90 /** | |
| 91 * Constructor. | |
| 92 */ | |
| 93 public HttpParser(Buffer buffer, EventHandler handler) | |
| 94 { | |
| 95 _endp=null; | |
| 96 _buffers=null; | |
| 97 _header=buffer; | |
| 98 _buffer=buffer; | |
| 99 _handler=handler; | |
| 100 | |
| 101 _tok0=new View.CaseInsensitive(_header); | |
| 102 _tok1=new View.CaseInsensitive(_header); | |
| 103 } | |
| 104 | |
| 105 /* ------------------------------------------------------------------------------- */ | |
| 106 /** | |
| 107 * Constructor. | |
| 108 * @param buffers the buffers to use | |
| 109 * @param endp the endpoint | |
| 110 * @param handler the even handler | |
| 111 */ | |
| 112 public HttpParser(Buffers buffers, EndPoint endp, EventHandler handler) | |
| 113 { | |
| 114 _buffers=buffers; | |
| 115 _endp=endp; | |
| 116 _handler=handler; | |
| 117 _tok0=new View.CaseInsensitive(); | |
| 118 _tok1=new View.CaseInsensitive(); | |
| 119 } | |
| 120 | |
| 121 /* ------------------------------------------------------------------------------- */ | |
| 122 public long getContentLength() | |
| 123 { | |
| 124 return _contentLength; | |
| 125 } | |
| 126 | |
| 127 /* ------------------------------------------------------------ */ | |
| 128 public long getContentRead() | |
| 129 { | |
| 130 return _contentPosition; | |
| 131 } | |
| 132 | |
| 133 /* ------------------------------------------------------------ */ | |
| 134 /** Set if a HEAD response is expected | |
| 135 * @param head | |
| 136 */ | |
| 137 public void setHeadResponse(boolean head) | |
| 138 { | |
| 139 _headResponse=head; | |
| 140 } | |
| 141 | |
| 142 /* ------------------------------------------------------------------------------- */ | |
| 143 public int getState() | |
| 144 { | |
| 145 return _state; | |
| 146 } | |
| 147 | |
| 148 /* ------------------------------------------------------------------------------- */ | |
| 149 public boolean inContentState() | |
| 150 { | |
| 151 return _state > 0; | |
| 152 } | |
| 153 | |
| 154 /* ------------------------------------------------------------------------------- */ | |
| 155 public boolean inHeaderState() | |
| 156 { | |
| 157 return _state < 0; | |
| 158 } | |
| 159 | |
| 160 /* ------------------------------------------------------------------------------- */ | |
| 161 public boolean isChunking() | |
| 162 { | |
| 163 return _contentLength==HttpTokens.CHUNKED_CONTENT; | |
| 164 } | |
| 165 | |
| 166 /* ------------------------------------------------------------ */ | |
| 167 public boolean isIdle() | |
| 168 { | |
| 169 return isState(STATE_START); | |
| 170 } | |
| 171 | |
| 172 /* ------------------------------------------------------------ */ | |
| 173 public boolean isComplete() | |
| 174 { | |
| 175 return isState(STATE_END); | |
| 176 } | |
| 177 | |
| 178 /* ------------------------------------------------------------ */ | |
| 179 public boolean isMoreInBuffer() | |
| 180 throws IOException | |
| 181 { | |
| 182 return ( _header!=null && _header.hasContent() || | |
| 183 _body!=null && _body.hasContent()); | |
| 184 } | |
| 185 | |
| 186 /* ------------------------------------------------------------------------------- */ | |
| 187 public boolean isState(int state) | |
| 188 { | |
| 189 return _state == state; | |
| 190 } | |
| 191 | |
| 192 /* ------------------------------------------------------------------------------- */ | |
| 193 public boolean isPersistent() | |
| 194 { | |
| 195 return _persistent; | |
| 196 } | |
| 197 | |
| 198 /* ------------------------------------------------------------------------------- */ | |
| 199 public void setPersistent(boolean persistent) | |
| 200 { | |
| 201 _persistent = persistent; | |
| 202 if (!_persistent &&(_state==STATE_END || _state==STATE_START)) | |
| 203 _state=STATE_SEEKING_EOF; | |
| 204 } | |
| 205 | |
| 206 /* ------------------------------------------------------------------------------- */ | |
| 207 /** | |
| 208 * Parse until {@link #STATE_END END} state. | |
| 209 * If the parser is already in the END state, then it is {@link #reset reset} and re-parsed. | |
| 210 * @throws IllegalStateException If the buffers have already been partially parsed. | |
| 211 */ | |
| 212 public void parse() throws IOException | |
| 213 { | |
| 214 if (_state==STATE_END) | |
| 215 reset(); | |
| 216 if (_state!=STATE_START) | |
| 217 throw new IllegalStateException("!START"); | |
| 218 | |
| 219 // continue parsing | |
| 220 while (_state != STATE_END) | |
| 221 if (parseNext()<0) | |
| 222 return; | |
| 223 } | |
| 224 | |
| 225 /* ------------------------------------------------------------------------------- */ | |
| 226 /** | |
| 227 * Parse until END state. | |
| 228 * This method will parse any remaining content in the current buffer as long as there is | |
| 229 * no unconsumed content. It does not care about the {@link #getState current state} of the parser. | |
| 230 * @see #parse | |
| 231 * @see #parseNext | |
| 232 */ | |
| 233 public boolean parseAvailable() throws IOException | |
| 234 { | |
| 235 boolean progress=parseNext()>0; | |
| 236 | |
| 237 // continue parsing | |
| 238 while (!isComplete() && _buffer!=null && _buffer.length()>0 && !_contentView.hasContent()) | |
| 239 { | |
| 240 progress |= parseNext()>0; | |
| 241 } | |
| 242 return progress; | |
| 243 } | |
| 244 | |
| 245 | |
| 246 /* ------------------------------------------------------------------------------- */ | |
| 247 /** | |
| 248 * Parse until next Event. | |
| 249 * @return an indication of progress <0 EOF, 0 no progress, >0 progress. | |
| 250 */ | |
| 251 public int parseNext() throws IOException | |
| 252 { | |
| 253 try | |
| 254 { | |
| 255 int progress=0; | |
| 256 | |
| 257 if (_state == STATE_END) | |
| 258 return 0; | |
| 259 | |
| 260 if (_buffer==null) | |
| 261 _buffer=getHeaderBuffer(); | |
| 262 | |
| 263 | |
| 264 if (_state == STATE_CONTENT && _contentPosition == _contentLength) | |
| 265 { | |
| 266 _state=STATE_END; | |
| 267 _handler.messageComplete(_contentPosition); | |
| 268 return 1; | |
| 269 } | |
| 270 | |
| 271 int length=_buffer.length(); | |
| 272 | |
| 273 // Fill buffer if we can | |
| 274 if (length == 0) | |
| 275 { | |
| 276 int filled=-1; | |
| 277 IOException ex=null; | |
| 278 try | |
| 279 { | |
| 280 filled=fill(); | |
| 281 LOG.debug("filled {}/{}",filled,_buffer.length()); | |
| 282 } | |
| 283 catch(IOException e) | |
| 284 { | |
| 285 LOG.debug(this.toString(),e); | |
| 286 ex=e; | |
| 287 } | |
| 288 | |
| 289 if (filled > 0 ) | |
| 290 progress++; | |
| 291 else if (filled < 0 ) | |
| 292 { | |
| 293 _persistent=false; | |
| 294 | |
| 295 // do we have content to deliver? | |
| 296 if (_state>STATE_END) | |
| 297 { | |
| 298 if (_buffer.length()>0 && !_headResponse) | |
| 299 { | |
| 300 Buffer chunk=_buffer.get(_buffer.length()); | |
| 301 _contentPosition += chunk.length(); | |
| 302 _contentView.update(chunk); | |
| 303 _handler.content(chunk); // May recurse here | |
| 304 } | |
| 305 } | |
| 306 | |
| 307 // was this unexpected? | |
| 308 switch(_state) | |
| 309 { | |
| 310 case STATE_END: | |
| 311 case STATE_SEEKING_EOF: | |
| 312 _state=STATE_END; | |
| 313 break; | |
| 314 | |
| 315 case STATE_EOF_CONTENT: | |
| 316 _state=STATE_END; | |
| 317 _handler.messageComplete(_contentPosition); | |
| 318 break; | |
| 319 | |
| 320 default: | |
| 321 _state=STATE_END; | |
| 322 if (!_headResponse) | |
| 323 _handler.earlyEOF(); | |
| 324 _handler.messageComplete(_contentPosition); | |
| 325 } | |
| 326 | |
| 327 if (ex!=null) | |
| 328 throw ex; | |
| 329 | |
| 330 if (!isComplete() && !isIdle()) | |
| 331 throw new EofException(); | |
| 332 | |
| 333 return -1; | |
| 334 } | |
| 335 length=_buffer.length(); | |
| 336 } | |
| 337 | |
| 338 | |
| 339 // Handle header states | |
| 340 byte ch; | |
| 341 byte[] array=_buffer.array(); | |
| 342 int last=_state; | |
| 343 while (_state<STATE_END && length-->0) | |
| 344 { | |
| 345 if (last!=_state) | |
| 346 { | |
| 347 progress++; | |
| 348 last=_state; | |
| 349 } | |
| 350 | |
| 351 ch=_buffer.get(); | |
| 352 | |
| 353 if (_eol == HttpTokens.CARRIAGE_RETURN) | |
| 354 { | |
| 355 if (ch == HttpTokens.LINE_FEED) | |
| 356 { | |
| 357 _eol=HttpTokens.LINE_FEED; | |
| 358 continue; | |
| 359 } | |
| 360 throw new HttpException(HttpStatus.BAD_REQUEST_400); | |
| 361 } | |
| 362 _eol=0; | |
| 363 | |
| 364 switch (_state) | |
| 365 { | |
| 366 case STATE_START: | |
| 367 _contentLength=HttpTokens.UNKNOWN_CONTENT; | |
| 368 _cached=null; | |
| 369 if (ch > HttpTokens.SPACE || ch<0) | |
| 370 { | |
| 371 _buffer.mark(); | |
| 372 _state=STATE_FIELD0; | |
| 373 } | |
| 374 break; | |
| 375 | |
| 376 case STATE_FIELD0: | |
| 377 if (ch == HttpTokens.SPACE) | |
| 378 { | |
| 379 _tok0.update(_buffer.markIndex(), _buffer.getIndex() - 1); | |
| 380 _responseStatus=HttpVersions.CACHE.get(_tok0)==null?-1:0; | |
| 381 _state=STATE_SPACE1; | |
| 382 continue; | |
| 383 } | |
| 384 else if (ch < HttpTokens.SPACE && ch>=0) | |
| 385 { | |
| 386 throw new HttpException(HttpStatus.BAD_REQUEST_400); | |
| 387 } | |
| 388 break; | |
| 389 | |
| 390 case STATE_SPACE1: | |
| 391 if (ch > HttpTokens.SPACE || ch<0) | |
| 392 { | |
| 393 _buffer.mark(); | |
| 394 if (_responseStatus>=0) | |
| 395 { | |
| 396 _state=STATE_STATUS; | |
| 397 _responseStatus=ch-'0'; | |
| 398 } | |
| 399 else | |
| 400 _state=STATE_URI; | |
| 401 } | |
| 402 else if (ch < HttpTokens.SPACE) | |
| 403 { | |
| 404 throw new HttpException(HttpStatus.BAD_REQUEST_400); | |
| 405 } | |
| 406 break; | |
| 407 | |
| 408 case STATE_STATUS: | |
| 409 if (ch == HttpTokens.SPACE) | |
| 410 { | |
| 411 _tok1.update(_buffer.markIndex(), _buffer.getIndex() - 1); | |
| 412 _state=STATE_SPACE2; | |
| 413 continue; | |
| 414 } | |
| 415 else if (ch>='0' && ch<='9') | |
| 416 { | |
| 417 _responseStatus=_responseStatus*10+(ch-'0'); | |
| 418 continue; | |
| 419 } | |
| 420 else if (ch < HttpTokens.SPACE && ch>=0) | |
| 421 { | |
| 422 _handler.startResponse(HttpMethods.CACHE.lookup(_tok0), _responseStatus, null); | |
| 423 _eol=ch; | |
| 424 _state=STATE_HEADER; | |
| 425 _tok0.setPutIndex(_tok0.getIndex()); | |
| 426 _tok1.setPutIndex(_tok1.getIndex()); | |
| 427 _multiLineValue=null; | |
| 428 continue; | |
| 429 } | |
| 430 // not a digit, so must be a URI | |
| 431 _state=STATE_URI; | |
| 432 _responseStatus=-1; | |
| 433 break; | |
| 434 | |
| 435 case STATE_URI: | |
| 436 if (ch == HttpTokens.SPACE) | |
| 437 { | |
| 438 _tok1.update(_buffer.markIndex(), _buffer.getIndex() - 1); | |
| 439 _state=STATE_SPACE2; | |
| 440 continue; | |
| 441 } | |
| 442 else if (ch < HttpTokens.SPACE && ch>=0) | |
| 443 { | |
| 444 // HTTP/0.9 | |
| 445 _handler.startRequest(HttpMethods.CACHE.lookup(_tok0), _buffer.sliceFromMark(), null); | |
| 446 _persistent=false; | |
| 447 _state=STATE_SEEKING_EOF; | |
| 448 _handler.headerComplete(); | |
| 449 _handler.messageComplete(_contentPosition); | |
| 450 return 1; | |
| 451 } | |
| 452 break; | |
| 453 | |
| 454 case STATE_SPACE2: | |
| 455 if (ch > HttpTokens.SPACE || ch<0) | |
| 456 { | |
| 457 _buffer.mark(); | |
| 458 _state=STATE_FIELD2; | |
| 459 } | |
| 460 else if (ch < HttpTokens.SPACE) | |
| 461 { | |
| 462 if (_responseStatus>0) | |
| 463 { | |
| 464 _handler.startResponse(HttpMethods.CACHE.lookup(_tok0), _responseStatus, null); | |
| 465 _eol=ch; | |
| 466 _state=STATE_HEADER; | |
| 467 _tok0.setPutIndex(_tok0.getIndex()); | |
| 468 _tok1.setPutIndex(_tok1.getIndex()); | |
| 469 _multiLineValue=null; | |
| 470 } | |
| 471 else | |
| 472 { | |
| 473 // HTTP/0.9 | |
| 474 _handler.startRequest(HttpMethods.CACHE.lookup(_tok0), _tok1, null); | |
| 475 _persistent=false; | |
| 476 _state=STATE_SEEKING_EOF; | |
| 477 _handler.headerComplete(); | |
| 478 _handler.messageComplete(_contentPosition); | |
| 479 return 1; | |
| 480 } | |
| 481 } | |
| 482 break; | |
| 483 | |
| 484 case STATE_FIELD2: | |
| 485 if (ch == HttpTokens.CARRIAGE_RETURN || ch == HttpTokens.LINE_FEED) | |
| 486 { | |
| 487 Buffer version; | |
| 488 if (_responseStatus>0) | |
| 489 _handler.startResponse(version=HttpVersions.CACHE.lookup(_tok0), _responseStatus,_buffer.sliceFromMark()); | |
| 490 else | |
| 491 _handler.startRequest(HttpMethods.CACHE.lookup(_tok0), _tok1, version=HttpVersions.CACHE.lookup(_buffer.sliceFromMark())); | |
| 492 _eol=ch; | |
| 493 _persistent=HttpVersions.CACHE.getOrdinal(version)>=HttpVersions.HTTP_1_1_ORDINAL; | |
| 494 _state=STATE_HEADER; | |
| 495 _tok0.setPutIndex(_tok0.getIndex()); | |
| 496 _tok1.setPutIndex(_tok1.getIndex()); | |
| 497 _multiLineValue=null; | |
| 498 continue; | |
| 499 } | |
| 500 break; | |
| 501 | |
| 502 case STATE_HEADER: | |
| 503 switch(ch) | |
| 504 { | |
| 505 case HttpTokens.COLON: | |
| 506 case HttpTokens.SPACE: | |
| 507 case HttpTokens.TAB: | |
| 508 { | |
| 509 // header value without name - continuation? | |
| 510 _length=-1; | |
| 511 _state=STATE_HEADER_VALUE; | |
| 512 break; | |
| 513 } | |
| 514 | |
| 515 default: | |
| 516 { | |
| 517 // handler last header if any | |
| 518 if (_cached!=null || _tok0.length() > 0 || _tok1.length() > 0 || _multiLineValue != null) | |
| 519 { | |
| 520 Buffer header=_cached!=null?_cached:HttpHeaders.CACHE.lookup(_tok0); | |
| 521 _cached=null; | |
| 522 Buffer value=_multiLineValue == null ? _tok1 : new ByteArrayBuffer(_multiLineValue); | |
| 523 | |
| 524 int ho=HttpHeaders.CACHE.getOrdinal(header); | |
| 525 if (ho >= 0) | |
| 526 { | |
| 527 int vo; | |
| 528 | |
| 529 switch (ho) | |
| 530 { | |
| 531 case HttpHeaders.CONTENT_LENGTH_ORDINAL: | |
| 532 if (_contentLength != HttpTokens.CHUNKED_CONTENT ) | |
| 533 { | |
| 534 try | |
| 535 { | |
| 536 _contentLength=BufferUtil.toLong(value); | |
| 537 } | |
| 538 catch(NumberFormatException e) | |
| 539 { | |
| 540 LOG.ignore(e); | |
| 541 throw new HttpException(HttpStatus.BAD_REQUEST_400); | |
| 542 } | |
| 543 if (_contentLength <= 0) | |
| 544 _contentLength=HttpTokens.NO_CONTENT; | |
| 545 } | |
| 546 break; | |
| 547 | |
| 548 case HttpHeaders.TRANSFER_ENCODING_ORDINAL: | |
| 549 value=HttpHeaderValues.CACHE.lookup(value); | |
| 550 vo=HttpHeaderValues.CACHE.getOrdinal(value); | |
| 551 if (HttpHeaderValues.CHUNKED_ORDINAL == vo) | |
| 552 _contentLength=HttpTokens.CHUNKED_CONTENT; | |
| 553 else | |
| 554 { | |
| 555 String c=value.toString(StringUtil.__ISO_8859_1); | |
| 556 if (c.endsWith(HttpHeaderValues.CHUNKED)) | |
| 557 _contentLength=HttpTokens.CHUNKED_CONTENT; | |
| 558 | |
| 559 else if (c.indexOf(HttpHeaderValues.CHUNKED) >= 0) | |
| 560 throw new HttpException(400,null); | |
| 561 } | |
| 562 break; | |
| 563 | |
| 564 case HttpHeaders.CONNECTION_ORDINAL: | |
| 565 switch(HttpHeaderValues.CACHE.getOrdinal(value)) | |
| 566 { | |
| 567 case HttpHeaderValues.CLOSE_ORDINAL: | |
| 568 _persistent=false; | |
| 569 break; | |
| 570 | |
| 571 case HttpHeaderValues.KEEP_ALIVE_ORDINAL: | |
| 572 _persistent=true; | |
| 573 break; | |
| 574 | |
| 575 case -1: // No match, may be multi valued | |
| 576 { | |
| 577 for (String v : value.toString().split(",")) | |
| 578 { | |
| 579 switch(HttpHeaderValues.CACHE.getOrdinal(v.trim())) | |
| 580 { | |
| 581 case HttpHeaderValues.CLOSE_ORDINAL: | |
| 582 _persistent=false; | |
| 583 break; | |
| 584 | |
| 585 case HttpHeaderValues.KEEP_ALIVE_ORDINAL: | |
| 586 _persistent=true; | |
| 587 break; | |
| 588 } | |
| 589 } | |
| 590 break; | |
| 591 } | |
| 592 } | |
| 593 } | |
| 594 } | |
| 595 | |
| 596 _handler.parsedHeader(header, value); | |
| 597 _tok0.setPutIndex(_tok0.getIndex()); | |
| 598 _tok1.setPutIndex(_tok1.getIndex()); | |
| 599 _multiLineValue=null; | |
| 600 } | |
| 601 _buffer.setMarkIndex(-1); | |
| 602 | |
| 603 // now handle ch | |
| 604 if (ch == HttpTokens.CARRIAGE_RETURN || ch == HttpTokens.LINE_FEED) | |
| 605 { | |
| 606 // is it a response that cannot have a body? | |
| 607 if (_responseStatus > 0 && // response | |
| 608 (_responseStatus == 304 || // not-modified response | |
| 609 _responseStatus == 204 || // no-content response | |
| 610 _responseStatus < 200)) // 1xx response | |
| 611 _contentLength=HttpTokens.NO_CONTENT; // ignore any other headers set | |
| 612 // else if we don't know framing | |
| 613 else if (_contentLength == HttpTokens.UNKNOWN_CONTENT) | |
| 614 { | |
| 615 if (_responseStatus == 0 // request | |
| 616 || _responseStatus == 304 // not-modified response | |
| 617 || _responseStatus == 204 // no-content response | |
| 618 || _responseStatus < 200) // 1xx response | |
| 619 _contentLength=HttpTokens.NO_CONTENT; | |
| 620 else | |
| 621 _contentLength=HttpTokens.EOF_CONTENT; | |
| 622 } | |
| 623 | |
| 624 _contentPosition=0; | |
| 625 _eol=ch; | |
| 626 if (_eol==HttpTokens.CARRIAGE_RETURN && _buffer.hasContent() && _buffer.peek()==HttpTokens.LINE_FEED) | |
| 627 _eol=_buffer.get(); | |
| 628 | |
| 629 // We convert _contentLength to an int for this switch statement because | |
| 630 // we don't care about the amount of data available just whether there is some. | |
| 631 switch (_contentLength > Integer.MAX_VALUE ? Integer.MAX_VALUE : (int) _contentLength) | |
| 632 { | |
| 633 case HttpTokens.EOF_CONTENT: | |
| 634 _state=STATE_EOF_CONTENT; | |
| 635 _handler.headerComplete(); // May recurse here ! | |
| 636 break; | |
| 637 | |
| 638 case HttpTokens.CHUNKED_CONTENT: | |
| 639 _state=STATE_CHUNKED_CONTENT; | |
| 640 _handler.headerComplete(); // May recurse here ! | |
| 641 break; | |
| 642 | |
| 643 case HttpTokens.NO_CONTENT: | |
| 644 _handler.headerComplete(); | |
| 645 _state=_persistent||(_responseStatus>=100&&_responseStatus<200)?STATE_END:STATE_SEEKING_EOF; | |
| 646 _handler.messageComplete(_contentPosition); | |
| 647 return 1; | |
| 648 | |
| 649 default: | |
| 650 _state=STATE_CONTENT; | |
| 651 _handler.headerComplete(); // May recurse here ! | |
| 652 break; | |
| 653 } | |
| 654 return 1; | |
| 655 } | |
| 656 else | |
| 657 { | |
| 658 // New header | |
| 659 _length=1; | |
| 660 _buffer.mark(); | |
| 661 _state=STATE_HEADER_NAME; | |
| 662 | |
| 663 // try cached name! | |
| 664 if (array!=null) | |
| 665 { | |
| 666 _cached=HttpHeaders.CACHE.getBest(array, _buffer.markIndex(), length+1); | |
| 667 | |
| 668 if (_cached!=null) | |
| 669 { | |
| 670 _length=_cached.length(); | |
| 671 _buffer.setGetIndex(_buffer.markIndex()+_length); | |
| 672 length=_buffer.length(); | |
| 673 } | |
| 674 } | |
| 675 } | |
| 676 } | |
| 677 } | |
| 678 | |
| 679 break; | |
| 680 | |
| 681 case STATE_HEADER_NAME: | |
| 682 switch(ch) | |
| 683 { | |
| 684 case HttpTokens.CARRIAGE_RETURN: | |
| 685 case HttpTokens.LINE_FEED: | |
| 686 if (_length > 0) | |
| 687 _tok0.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 688 _eol=ch; | |
| 689 _state=STATE_HEADER; | |
| 690 break; | |
| 691 case HttpTokens.COLON: | |
| 692 if (_length > 0 && _cached==null) | |
| 693 _tok0.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 694 _length=-1; | |
| 695 _state=STATE_HEADER_VALUE; | |
| 696 break; | |
| 697 case HttpTokens.SPACE: | |
| 698 case HttpTokens.TAB: | |
| 699 break; | |
| 700 default: | |
| 701 { | |
| 702 _cached=null; | |
| 703 if (_length == -1) | |
| 704 _buffer.mark(); | |
| 705 _length=_buffer.getIndex() - _buffer.markIndex(); | |
| 706 _state=STATE_HEADER_IN_NAME; | |
| 707 } | |
| 708 } | |
| 709 | |
| 710 break; | |
| 711 | |
| 712 case STATE_HEADER_IN_NAME: | |
| 713 switch(ch) | |
| 714 { | |
| 715 case HttpTokens.CARRIAGE_RETURN: | |
| 716 case HttpTokens.LINE_FEED: | |
| 717 if (_length > 0) | |
| 718 _tok0.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 719 _eol=ch; | |
| 720 _state=STATE_HEADER; | |
| 721 break; | |
| 722 case HttpTokens.COLON: | |
| 723 if (_length > 0 && _cached==null) | |
| 724 _tok0.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 725 _length=-1; | |
| 726 _state=STATE_HEADER_VALUE; | |
| 727 break; | |
| 728 case HttpTokens.SPACE: | |
| 729 case HttpTokens.TAB: | |
| 730 _state=STATE_HEADER_NAME; | |
| 731 break; | |
| 732 default: | |
| 733 { | |
| 734 _cached=null; | |
| 735 _length++; | |
| 736 } | |
| 737 } | |
| 738 break; | |
| 739 | |
| 740 case STATE_HEADER_VALUE: | |
| 741 switch(ch) | |
| 742 { | |
| 743 case HttpTokens.CARRIAGE_RETURN: | |
| 744 case HttpTokens.LINE_FEED: | |
| 745 if (_length > 0) | |
| 746 { | |
| 747 if (_tok1.length() == 0) | |
| 748 _tok1.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 749 else | |
| 750 { | |
| 751 // Continuation line! | |
| 752 if (_multiLineValue == null) _multiLineValue=_tok1.toString(StringUtil.__ISO_8859_1); | |
| 753 _tok1.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 754 _multiLineValue += " " + _tok1.toString(StringUtil.__ISO_8859_1); | |
| 755 } | |
| 756 } | |
| 757 _eol=ch; | |
| 758 _state=STATE_HEADER; | |
| 759 break; | |
| 760 case HttpTokens.SPACE: | |
| 761 case HttpTokens.TAB: | |
| 762 break; | |
| 763 default: | |
| 764 { | |
| 765 if (_length == -1) | |
| 766 _buffer.mark(); | |
| 767 _length=_buffer.getIndex() - _buffer.markIndex(); | |
| 768 _state=STATE_HEADER_IN_VALUE; | |
| 769 } | |
| 770 } | |
| 771 break; | |
| 772 | |
| 773 case STATE_HEADER_IN_VALUE: | |
| 774 switch(ch) | |
| 775 { | |
| 776 case HttpTokens.CARRIAGE_RETURN: | |
| 777 case HttpTokens.LINE_FEED: | |
| 778 if (_length > 0) | |
| 779 { | |
| 780 if (_tok1.length() == 0) | |
| 781 _tok1.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 782 else | |
| 783 { | |
| 784 // Continuation line! | |
| 785 if (_multiLineValue == null) _multiLineValue=_tok1.toString(StringUtil.__ISO_8859_1); | |
| 786 _tok1.update(_buffer.markIndex(), _buffer.markIndex() + _length); | |
| 787 _multiLineValue += " " + _tok1.toString(StringUtil.__ISO_8859_1); | |
| 788 } | |
| 789 } | |
| 790 _eol=ch; | |
| 791 _state=STATE_HEADER; | |
| 792 break; | |
| 793 case HttpTokens.SPACE: | |
| 794 case HttpTokens.TAB: | |
| 795 _state=STATE_HEADER_VALUE; | |
| 796 break; | |
| 797 default: | |
| 798 _length++; | |
| 799 } | |
| 800 break; | |
| 801 } | |
| 802 } // end of HEADER states loop | |
| 803 | |
| 804 // ========================== | |
| 805 | |
| 806 // Handle HEAD response | |
| 807 if (_responseStatus>0 && _headResponse) | |
| 808 { | |
| 809 _state=_persistent||(_responseStatus>=100&&_responseStatus<200)?STATE_END:STATE_SEEKING_EOF; | |
| 810 _handler.messageComplete(_contentLength); | |
| 811 } | |
| 812 | |
| 813 | |
| 814 // ========================== | |
| 815 | |
| 816 // Handle _content | |
| 817 length=_buffer.length(); | |
| 818 Buffer chunk; | |
| 819 last=_state; | |
| 820 while (_state > STATE_END && length > 0) | |
| 821 { | |
| 822 if (last!=_state) | |
| 823 { | |
| 824 progress++; | |
| 825 last=_state; | |
| 826 } | |
| 827 | |
| 828 if (_eol == HttpTokens.CARRIAGE_RETURN && _buffer.peek() == HttpTokens.LINE_FEED) | |
| 829 { | |
| 830 _eol=_buffer.get(); | |
| 831 length=_buffer.length(); | |
| 832 continue; | |
| 833 } | |
| 834 _eol=0; | |
| 835 switch (_state) | |
| 836 { | |
| 837 case STATE_EOF_CONTENT: | |
| 838 chunk=_buffer.get(_buffer.length()); | |
| 839 _contentPosition += chunk.length(); | |
| 840 _contentView.update(chunk); | |
| 841 _handler.content(chunk); // May recurse here | |
| 842 // TODO adjust the _buffer to keep unconsumed content | |
| 843 return 1; | |
| 844 | |
| 845 case STATE_CONTENT: | |
| 846 { | |
| 847 long remaining=_contentLength - _contentPosition; | |
| 848 if (remaining == 0) | |
| 849 { | |
| 850 _state=_persistent?STATE_END:STATE_SEEKING_EOF; | |
| 851 _handler.messageComplete(_contentPosition); | |
| 852 return 1; | |
| 853 } | |
| 854 | |
| 855 if (length > remaining) | |
| 856 { | |
| 857 // We can cast reamining to an int as we know that it is smaller than | |
| 858 // or equal to length which is already an int. | |
| 859 length=(int)remaining; | |
| 860 } | |
| 861 | |
| 862 chunk=_buffer.get(length); | |
| 863 _contentPosition += chunk.length(); | |
| 864 _contentView.update(chunk); | |
| 865 _handler.content(chunk); // May recurse here | |
| 866 | |
| 867 if(_contentPosition == _contentLength) | |
| 868 { | |
| 869 _state=_persistent?STATE_END:STATE_SEEKING_EOF; | |
| 870 _handler.messageComplete(_contentPosition); | |
| 871 } | |
| 872 // TODO adjust the _buffer to keep unconsumed content | |
| 873 return 1; | |
| 874 } | |
| 875 | |
| 876 case STATE_CHUNKED_CONTENT: | |
| 877 { | |
| 878 ch=_buffer.peek(); | |
| 879 if (ch == HttpTokens.CARRIAGE_RETURN || ch == HttpTokens.LINE_FEED) | |
| 880 _eol=_buffer.get(); | |
| 881 else if (ch <= HttpTokens.SPACE) | |
| 882 _buffer.get(); | |
| 883 else | |
| 884 { | |
| 885 _chunkLength=0; | |
| 886 _chunkPosition=0; | |
| 887 _state=STATE_CHUNK_SIZE; | |
| 888 } | |
| 889 break; | |
| 890 } | |
| 891 | |
| 892 case STATE_CHUNK_SIZE: | |
| 893 { | |
| 894 ch=_buffer.get(); | |
| 895 if (ch == HttpTokens.CARRIAGE_RETURN || ch == HttpTokens.LINE_FEED) | |
| 896 { | |
| 897 _eol=ch; | |
| 898 | |
| 899 if (_chunkLength == 0) | |
| 900 { | |
| 901 if (_eol==HttpTokens.CARRIAGE_RETURN && _buffer.hasContent() && _buffer.peek()==HttpTokens.LINE_FEED) | |
| 902 _eol=_buffer.get(); | |
| 903 _state=_persistent?STATE_END:STATE_SEEKING_EOF; | |
| 904 _handler.messageComplete(_contentPosition); | |
| 905 return 1; | |
| 906 } | |
| 907 else | |
| 908 _state=STATE_CHUNK; | |
| 909 } | |
| 910 else if (ch <= HttpTokens.SPACE || ch == HttpTokens.SEMI_COLON) | |
| 911 _state=STATE_CHUNK_PARAMS; | |
| 912 else if (ch >= '0' && ch <= '9') | |
| 913 _chunkLength=_chunkLength * 16 + (ch - '0'); | |
| 914 else if (ch >= 'a' && ch <= 'f') | |
| 915 _chunkLength=_chunkLength * 16 + (10 + ch - 'a'); | |
| 916 else if (ch >= 'A' && ch <= 'F') | |
| 917 _chunkLength=_chunkLength * 16 + (10 + ch - 'A'); | |
| 918 else | |
| 919 throw new IOException("bad chunk char: " + ch); | |
| 920 break; | |
| 921 } | |
| 922 | |
| 923 case STATE_CHUNK_PARAMS: | |
| 924 { | |
| 925 ch=_buffer.get(); | |
| 926 if (ch == HttpTokens.CARRIAGE_RETURN || ch == HttpTokens.LINE_FEED) | |
| 927 { | |
| 928 _eol=ch; | |
| 929 if (_chunkLength == 0) | |
| 930 { | |
| 931 if (_eol==HttpTokens.CARRIAGE_RETURN && _buffer.hasContent() && _buffer.peek()==HttpTokens.LINE_FEED) | |
| 932 _eol=_buffer.get(); | |
| 933 _state=_persistent?STATE_END:STATE_SEEKING_EOF; | |
| 934 _handler.messageComplete(_contentPosition); | |
| 935 return 1; | |
| 936 } | |
| 937 else | |
| 938 _state=STATE_CHUNK; | |
| 939 } | |
| 940 break; | |
| 941 } | |
| 942 | |
| 943 case STATE_CHUNK: | |
| 944 { | |
| 945 int remaining=_chunkLength - _chunkPosition; | |
| 946 if (remaining == 0) | |
| 947 { | |
| 948 _state=STATE_CHUNKED_CONTENT; | |
| 949 break; | |
| 950 } | |
| 951 else if (length > remaining) | |
| 952 length=remaining; | |
| 953 chunk=_buffer.get(length); | |
| 954 _contentPosition += chunk.length(); | |
| 955 _chunkPosition += chunk.length(); | |
| 956 _contentView.update(chunk); | |
| 957 _handler.content(chunk); // May recurse here | |
| 958 // TODO adjust the _buffer to keep unconsumed content | |
| 959 return 1; | |
| 960 } | |
| 961 | |
| 962 case STATE_SEEKING_EOF: | |
| 963 { | |
| 964 // Close if there is more data than CRLF | |
| 965 if (_buffer.length()>2) | |
| 966 { | |
| 967 _state=STATE_END; | |
| 968 _endp.close(); | |
| 969 } | |
| 970 else | |
| 971 { | |
| 972 // or if the data is not white space | |
| 973 while (_buffer.length()>0) | |
| 974 if (!Character.isWhitespace(_buffer.get())) | |
| 975 { | |
| 976 _state=STATE_END; | |
| 977 _endp.close(); | |
| 978 _buffer.clear(); | |
| 979 } | |
| 980 } | |
| 981 | |
| 982 _buffer.clear(); | |
| 983 break; | |
| 984 } | |
| 985 } | |
| 986 | |
| 987 length=_buffer.length(); | |
| 988 } | |
| 989 | |
| 990 return progress; | |
| 991 } | |
| 992 catch(HttpException e) | |
| 993 { | |
| 994 _persistent=false; | |
| 995 _state=STATE_SEEKING_EOF; | |
| 996 throw e; | |
| 997 } | |
| 998 } | |
| 999 | |
| 1000 /* ------------------------------------------------------------------------------- */ | |
| 1001 /** fill the buffers from the endpoint | |
| 1002 * | |
| 1003 */ | |
| 1004 protected int fill() throws IOException | |
| 1005 { | |
| 1006 // Do we have a buffer? | |
| 1007 if (_buffer==null) | |
| 1008 _buffer=getHeaderBuffer(); | |
| 1009 | |
| 1010 // Is there unconsumed content in body buffer | |
| 1011 if (_state>STATE_END && _buffer==_header && _header!=null && !_header.hasContent() && _body!=null && _body.hasContent()) | |
| 1012 { | |
| 1013 _buffer=_body; | |
| 1014 return _buffer.length(); | |
| 1015 } | |
| 1016 | |
| 1017 // Shall we switch to a body buffer? | |
| 1018 if (_buffer==_header && _state>STATE_END && _header.length()==0 && (_forceContentBuffer || (_contentLength-_contentPosition)>_header.capacity()) && (_body!=null||_buffers!=null)) | |
| 1019 { | |
| 1020 if (_body==null) | |
| 1021 _body=_buffers.getBuffer(); | |
| 1022 _buffer=_body; | |
| 1023 } | |
| 1024 | |
| 1025 // Do we have somewhere to fill from? | |
| 1026 if (_endp != null ) | |
| 1027 { | |
| 1028 // Shall we compact the body? | |
| 1029 if (_buffer==_body || _state>STATE_END) | |
| 1030 { | |
| 1031 _buffer.compact(); | |
| 1032 } | |
| 1033 | |
| 1034 // Are we full? | |
| 1035 if (_buffer.space() == 0) | |
| 1036 { | |
| 1037 LOG.warn("HttpParser Full for {} ",_endp); | |
| 1038 _buffer.clear(); | |
| 1039 throw new HttpException(HttpStatus.REQUEST_ENTITY_TOO_LARGE_413, "Request Entity Too Large: "+(_buffer==_body?"body":"head")); | |
| 1040 } | |
| 1041 | |
| 1042 try | |
| 1043 { | |
| 1044 int filled = _endp.fill(_buffer); | |
| 1045 return filled; | |
| 1046 } | |
| 1047 catch(IOException e) | |
| 1048 { | |
| 1049 LOG.debug(e); | |
| 1050 throw (e instanceof EofException) ? e:new EofException(e); | |
| 1051 } | |
| 1052 } | |
| 1053 | |
| 1054 return -1; | |
| 1055 } | |
| 1056 | |
| 1057 /* ------------------------------------------------------------------------------- */ | |
| 1058 public void reset() | |
| 1059 { | |
| 1060 // reset state | |
| 1061 _contentView.setGetIndex(_contentView.putIndex()); | |
| 1062 _state=_persistent?STATE_START:(_endp.isInputShutdown()?STATE_END:STATE_SEEKING_EOF); | |
| 1063 _contentLength=HttpTokens.UNKNOWN_CONTENT; | |
| 1064 _contentPosition=0; | |
| 1065 _length=0; | |
| 1066 _responseStatus=0; | |
| 1067 | |
| 1068 // Consume LF if CRLF | |
| 1069 if (_eol == HttpTokens.CARRIAGE_RETURN && _buffer!=null && _buffer.hasContent() && _buffer.peek() == HttpTokens.LINE_FEED) | |
| 1070 _eol=_buffer.get(); | |
| 1071 | |
| 1072 if (_body!=null && _body.hasContent()) | |
| 1073 { | |
| 1074 // There is content in the body after the end of the request. | |
| 1075 // This is probably a pipelined header of the next request, so we need to | |
| 1076 // copy it to the header buffer. | |
| 1077 if (_header==null) | |
| 1078 getHeaderBuffer(); | |
| 1079 else | |
| 1080 { | |
| 1081 _header.setMarkIndex(-1); | |
| 1082 _header.compact(); | |
| 1083 } | |
| 1084 int take=_header.space(); | |
| 1085 if (take>_body.length()) | |
| 1086 take=_body.length(); | |
| 1087 _body.peek(_body.getIndex(),take); | |
| 1088 _body.skip(_header.put(_body.peek(_body.getIndex(),take))); | |
| 1089 } | |
| 1090 | |
| 1091 if (_header!=null) | |
| 1092 { | |
| 1093 _header.setMarkIndex(-1); | |
| 1094 _header.compact(); | |
| 1095 } | |
| 1096 if (_body!=null) | |
| 1097 _body.setMarkIndex(-1); | |
| 1098 | |
| 1099 _buffer=_header; | |
| 1100 returnBuffers(); | |
| 1101 } | |
| 1102 | |
| 1103 | |
| 1104 /* ------------------------------------------------------------------------------- */ | |
| 1105 public void returnBuffers() | |
| 1106 { | |
| 1107 if (_body!=null && !_body.hasContent() && _body.markIndex()==-1 && _buffers!=null) | |
| 1108 { | |
| 1109 if (_buffer==_body) | |
| 1110 _buffer=_header; | |
| 1111 if (_buffers!=null) | |
| 1112 _buffers.returnBuffer(_body); | |
| 1113 _body=null; | |
| 1114 } | |
| 1115 | |
| 1116 if (_header!=null && !_header.hasContent() && _header.markIndex()==-1 && _buffers!=null) | |
| 1117 { | |
| 1118 if (_buffer==_header) | |
| 1119 _buffer=null; | |
| 1120 _buffers.returnBuffer(_header); | |
| 1121 _header=null; | |
| 1122 } | |
| 1123 } | |
| 1124 | |
| 1125 /* ------------------------------------------------------------------------------- */ | |
| 1126 public void setState(int state) | |
| 1127 { | |
| 1128 this._state=state; | |
| 1129 _contentLength=HttpTokens.UNKNOWN_CONTENT; | |
| 1130 } | |
| 1131 | |
| 1132 /* ------------------------------------------------------------------------------- */ | |
| 1133 public String toString(Buffer buf) | |
| 1134 { | |
| 1135 return "state=" + _state + " length=" + _length + " buf=" + buf.hashCode(); | |
| 1136 } | |
| 1137 | |
| 1138 /* ------------------------------------------------------------------------------- */ | |
| 1139 @Override | |
| 1140 public String toString() | |
| 1141 { | |
| 1142 return String.format("%s{s=%d,l=%d,c=%d}", | |
| 1143 getClass().getSimpleName(), | |
| 1144 _state, | |
| 1145 _length, | |
| 1146 _contentLength); | |
| 1147 } | |
| 1148 | |
| 1149 /* ------------------------------------------------------------ */ | |
| 1150 public Buffer getHeaderBuffer() | |
| 1151 { | |
| 1152 if (_header == null) | |
| 1153 { | |
| 1154 _header=_buffers.getHeader(); | |
| 1155 _tok0.update(_header); | |
| 1156 _tok1.update(_header); | |
| 1157 } | |
| 1158 return _header; | |
| 1159 } | |
| 1160 | |
| 1161 /* ------------------------------------------------------------ */ | |
| 1162 public Buffer getBodyBuffer() | |
| 1163 { | |
| 1164 return _body; | |
| 1165 } | |
| 1166 | |
| 1167 /* ------------------------------------------------------------ */ | |
| 1168 /** | |
| 1169 * @param force True if a new buffer will be forced to be used for content and the header buffer will not be used. | |
| 1170 */ | |
| 1171 public void setForceContentBuffer(boolean force) | |
| 1172 { | |
| 1173 _forceContentBuffer=force; | |
| 1174 } | |
| 1175 | |
| 1176 /* ------------------------------------------------------------ */ | |
| 1177 public Buffer blockForContent(long maxIdleTime) throws IOException | |
| 1178 { | |
| 1179 if (_contentView.length()>0) | |
| 1180 return _contentView; | |
| 1181 | |
| 1182 if (getState() <= STATE_END || isState(STATE_SEEKING_EOF)) | |
| 1183 return null; | |
| 1184 | |
| 1185 try | |
| 1186 { | |
| 1187 parseNext(); | |
| 1188 | |
| 1189 // parse until some progress is made (or IOException thrown for timeout) | |
| 1190 while(_contentView.length() == 0 && !(isState(HttpParser.STATE_END)||isState(HttpParser.STATE_SEEKING_EOF)) && _endp!=null && _endp.isOpen()) | |
| 1191 { | |
| 1192 if (!_endp.isBlocking()) | |
| 1193 { | |
| 1194 if (parseNext()>0) | |
| 1195 continue; | |
| 1196 | |
| 1197 if (!_endp.blockReadable(maxIdleTime)) | |
| 1198 { | |
| 1199 _endp.close(); | |
| 1200 throw new EofException("timeout"); | |
| 1201 } | |
| 1202 } | |
| 1203 | |
| 1204 parseNext(); | |
| 1205 } | |
| 1206 } | |
| 1207 catch(IOException e) | |
| 1208 { | |
| 1209 // TODO is this needed? | |
| 1210 _endp.close(); | |
| 1211 throw e; | |
| 1212 } | |
| 1213 | |
| 1214 return _contentView.length()>0?_contentView:null; | |
| 1215 } | |
| 1216 | |
| 1217 /* ------------------------------------------------------------ */ | |
| 1218 /* (non-Javadoc) | |
| 1219 * @see java.io.InputStream#available() | |
| 1220 */ | |
| 1221 public int available() throws IOException | |
| 1222 { | |
| 1223 if (_contentView!=null && _contentView.length()>0) | |
| 1224 return _contentView.length(); | |
| 1225 | |
| 1226 if (_endp.isBlocking()) | |
| 1227 { | |
| 1228 if (_state>0 && _endp instanceof StreamEndPoint) | |
| 1229 return ((StreamEndPoint)_endp).getInputStream().available()>0?1:0; | |
| 1230 | |
| 1231 return 0; | |
| 1232 } | |
| 1233 | |
| 1234 parseNext(); | |
| 1235 return _contentView==null?0:_contentView.length(); | |
| 1236 } | |
| 1237 | |
| 1238 /* ------------------------------------------------------------ */ | |
| 1239 /* ------------------------------------------------------------ */ | |
| 1240 /* ------------------------------------------------------------ */ | |
| 1241 public static abstract class EventHandler | |
| 1242 { | |
| 1243 public abstract void content(Buffer ref) throws IOException; | |
| 1244 | |
| 1245 public void headerComplete() throws IOException | |
| 1246 { | |
| 1247 } | |
| 1248 | |
| 1249 public void messageComplete(long contentLength) throws IOException | |
| 1250 { | |
| 1251 } | |
| 1252 | |
| 1253 /** | |
| 1254 * This is the method called by parser when a HTTP Header name and value is found | |
| 1255 */ | |
| 1256 public void parsedHeader(Buffer name, Buffer value) throws IOException | |
| 1257 { | |
| 1258 } | |
| 1259 | |
| 1260 /** | |
| 1261 * This is the method called by parser when the HTTP request line is parsed | |
| 1262 */ | |
| 1263 public abstract void startRequest(Buffer method, Buffer url, Buffer version) | |
| 1264 throws IOException; | |
| 1265 | |
| 1266 /** | |
| 1267 * This is the method called by parser when the HTTP request line is parsed | |
| 1268 */ | |
| 1269 public abstract void startResponse(Buffer version, int status, Buffer reason) | |
| 1270 throws IOException; | |
| 1271 | |
| 1272 public void earlyEOF() | |
| 1273 {} | |
| 1274 } | |
| 1275 | |
| 1276 | |
| 1277 | |
| 1278 | |
| 1279 } |
