/*
 * Copyright 2012 The Netty Project
 *
 * The Netty Project licenses this file to you 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 io.netty.handler.codec.http;

import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufUtil;
import io.netty.util.AsciiString;
import io.netty.util.CharsetUtil;

import static io.netty.handler.codec.http.HttpConstants.SP;
import static io.netty.util.ByteProcessor.FIND_ASCII_SPACE;
import static java.lang.Integer.parseInt;

The response code and its description of HTTP or its derived protocols, such as RTSP and ICAP.
/** * The response code and its description of HTTP or its derived protocols, such as * <a href="http://en.wikipedia.org/wiki/Real_Time_Streaming_Protocol">RTSP</a> and * <a href="http://en.wikipedia.org/wiki/Internet_Content_Adaptation_Protocol">ICAP</a>. */
public class HttpResponseStatus implements Comparable<HttpResponseStatus> {
100 Continue
/** * 100 Continue */
public static final HttpResponseStatus CONTINUE = newStatus(100, "Continue");
101 Switching Protocols
/** * 101 Switching Protocols */
public static final HttpResponseStatus SWITCHING_PROTOCOLS = newStatus(101, "Switching Protocols");
102 Processing (WebDAV, RFC2518)
/** * 102 Processing (WebDAV, RFC2518) */
public static final HttpResponseStatus PROCESSING = newStatus(102, "Processing");
200 OK
/** * 200 OK */
public static final HttpResponseStatus OK = newStatus(200, "OK");
201 Created
/** * 201 Created */
public static final HttpResponseStatus CREATED = newStatus(201, "Created");
202 Accepted
/** * 202 Accepted */
public static final HttpResponseStatus ACCEPTED = newStatus(202, "Accepted");
203 Non-Authoritative Information (since HTTP/1.1)
/** * 203 Non-Authoritative Information (since HTTP/1.1) */
public static final HttpResponseStatus NON_AUTHORITATIVE_INFORMATION = newStatus(203, "Non-Authoritative Information");
204 No Content
/** * 204 No Content */
public static final HttpResponseStatus NO_CONTENT = newStatus(204, "No Content");
205 Reset Content
/** * 205 Reset Content */
public static final HttpResponseStatus RESET_CONTENT = newStatus(205, "Reset Content");
206 Partial Content
/** * 206 Partial Content */
public static final HttpResponseStatus PARTIAL_CONTENT = newStatus(206, "Partial Content");
207 Multi-Status (WebDAV, RFC2518)
/** * 207 Multi-Status (WebDAV, RFC2518) */
public static final HttpResponseStatus MULTI_STATUS = newStatus(207, "Multi-Status");
300 Multiple Choices
/** * 300 Multiple Choices */
public static final HttpResponseStatus MULTIPLE_CHOICES = newStatus(300, "Multiple Choices");
301 Moved Permanently
/** * 301 Moved Permanently */
public static final HttpResponseStatus MOVED_PERMANENTLY = newStatus(301, "Moved Permanently");
302 Found
/** * 302 Found */
public static final HttpResponseStatus FOUND = newStatus(302, "Found");
303 See Other (since HTTP/1.1)
/** * 303 See Other (since HTTP/1.1) */
public static final HttpResponseStatus SEE_OTHER = newStatus(303, "See Other");
304 Not Modified
/** * 304 Not Modified */
public static final HttpResponseStatus NOT_MODIFIED = newStatus(304, "Not Modified");
305 Use Proxy (since HTTP/1.1)
/** * 305 Use Proxy (since HTTP/1.1) */
public static final HttpResponseStatus USE_PROXY = newStatus(305, "Use Proxy");
307 Temporary Redirect (since HTTP/1.1)
/** * 307 Temporary Redirect (since HTTP/1.1) */
public static final HttpResponseStatus TEMPORARY_REDIRECT = newStatus(307, "Temporary Redirect");
308 Permanent Redirect (RFC7538)
/** * 308 Permanent Redirect (RFC7538) */
public static final HttpResponseStatus PERMANENT_REDIRECT = newStatus(308, "Permanent Redirect");
400 Bad Request
/** * 400 Bad Request */
public static final HttpResponseStatus BAD_REQUEST = newStatus(400, "Bad Request");
401 Unauthorized
/** * 401 Unauthorized */
public static final HttpResponseStatus UNAUTHORIZED = newStatus(401, "Unauthorized");
402 Payment Required
/** * 402 Payment Required */
public static final HttpResponseStatus PAYMENT_REQUIRED = newStatus(402, "Payment Required");
403 Forbidden
/** * 403 Forbidden */
public static final HttpResponseStatus FORBIDDEN = newStatus(403, "Forbidden");
404 Not Found
/** * 404 Not Found */
public static final HttpResponseStatus NOT_FOUND = newStatus(404, "Not Found");
405 Method Not Allowed
/** * 405 Method Not Allowed */
public static final HttpResponseStatus METHOD_NOT_ALLOWED = newStatus(405, "Method Not Allowed");
406 Not Acceptable
/** * 406 Not Acceptable */
public static final HttpResponseStatus NOT_ACCEPTABLE = newStatus(406, "Not Acceptable");
407 Proxy Authentication Required
/** * 407 Proxy Authentication Required */
public static final HttpResponseStatus PROXY_AUTHENTICATION_REQUIRED = newStatus(407, "Proxy Authentication Required");
408 Request Timeout
/** * 408 Request Timeout */
public static final HttpResponseStatus REQUEST_TIMEOUT = newStatus(408, "Request Timeout");
409 Conflict
/** * 409 Conflict */
public static final HttpResponseStatus CONFLICT = newStatus(409, "Conflict");
410 Gone
/** * 410 Gone */
public static final HttpResponseStatus GONE = newStatus(410, "Gone");
411 Length Required
/** * 411 Length Required */
public static final HttpResponseStatus LENGTH_REQUIRED = newStatus(411, "Length Required");
412 Precondition Failed
/** * 412 Precondition Failed */
public static final HttpResponseStatus PRECONDITION_FAILED = newStatus(412, "Precondition Failed");
413 Request Entity Too Large
/** * 413 Request Entity Too Large */
public static final HttpResponseStatus REQUEST_ENTITY_TOO_LARGE = newStatus(413, "Request Entity Too Large");
414 Request-URI Too Long
/** * 414 Request-URI Too Long */
public static final HttpResponseStatus REQUEST_URI_TOO_LONG = newStatus(414, "Request-URI Too Long");
415 Unsupported Media Type
/** * 415 Unsupported Media Type */
public static final HttpResponseStatus UNSUPPORTED_MEDIA_TYPE = newStatus(415, "Unsupported Media Type");
416 Requested Range Not Satisfiable
/** * 416 Requested Range Not Satisfiable */
public static final HttpResponseStatus REQUESTED_RANGE_NOT_SATISFIABLE = newStatus(416, "Requested Range Not Satisfiable");
417 Expectation Failed
/** * 417 Expectation Failed */
public static final HttpResponseStatus EXPECTATION_FAILED = newStatus(417, "Expectation Failed");
421 Misdirected Request 421 Status Code
/** * 421 Misdirected Request * * <a href="https://tools.ietf.org/html/draft-ietf-httpbis-http2-15#section-9.1.2">421 Status Code</a> */
public static final HttpResponseStatus MISDIRECTED_REQUEST = newStatus(421, "Misdirected Request");
422 Unprocessable Entity (WebDAV, RFC4918)
/** * 422 Unprocessable Entity (WebDAV, RFC4918) */
public static final HttpResponseStatus UNPROCESSABLE_ENTITY = newStatus(422, "Unprocessable Entity");
423 Locked (WebDAV, RFC4918)
/** * 423 Locked (WebDAV, RFC4918) */
public static final HttpResponseStatus LOCKED = newStatus(423, "Locked");
424 Failed Dependency (WebDAV, RFC4918)
/** * 424 Failed Dependency (WebDAV, RFC4918) */
public static final HttpResponseStatus FAILED_DEPENDENCY = newStatus(424, "Failed Dependency");
425 Unordered Collection (WebDAV, RFC3648)
/** * 425 Unordered Collection (WebDAV, RFC3648) */
public static final HttpResponseStatus UNORDERED_COLLECTION = newStatus(425, "Unordered Collection");
426 Upgrade Required (RFC2817)
/** * 426 Upgrade Required (RFC2817) */
public static final HttpResponseStatus UPGRADE_REQUIRED = newStatus(426, "Upgrade Required");
428 Precondition Required (RFC6585)
/** * 428 Precondition Required (RFC6585) */
public static final HttpResponseStatus PRECONDITION_REQUIRED = newStatus(428, "Precondition Required");
429 Too Many Requests (RFC6585)
/** * 429 Too Many Requests (RFC6585) */
public static final HttpResponseStatus TOO_MANY_REQUESTS = newStatus(429, "Too Many Requests");
431 Request Header Fields Too Large (RFC6585)
/** * 431 Request Header Fields Too Large (RFC6585) */
public static final HttpResponseStatus REQUEST_HEADER_FIELDS_TOO_LARGE = newStatus(431, "Request Header Fields Too Large");
500 Internal Server Error
/** * 500 Internal Server Error */
public static final HttpResponseStatus INTERNAL_SERVER_ERROR = newStatus(500, "Internal Server Error");
501 Not Implemented
/** * 501 Not Implemented */
public static final HttpResponseStatus NOT_IMPLEMENTED = newStatus(501, "Not Implemented");
502 Bad Gateway
/** * 502 Bad Gateway */
public static final HttpResponseStatus BAD_GATEWAY = newStatus(502, "Bad Gateway");
503 Service Unavailable
/** * 503 Service Unavailable */
public static final HttpResponseStatus SERVICE_UNAVAILABLE = newStatus(503, "Service Unavailable");
504 Gateway Timeout
/** * 504 Gateway Timeout */
public static final HttpResponseStatus GATEWAY_TIMEOUT = newStatus(504, "Gateway Timeout");
505 HTTP Version Not Supported
/** * 505 HTTP Version Not Supported */
public static final HttpResponseStatus HTTP_VERSION_NOT_SUPPORTED = newStatus(505, "HTTP Version Not Supported");
506 Variant Also Negotiates (RFC2295)
/** * 506 Variant Also Negotiates (RFC2295) */
public static final HttpResponseStatus VARIANT_ALSO_NEGOTIATES = newStatus(506, "Variant Also Negotiates");
507 Insufficient Storage (WebDAV, RFC4918)
/** * 507 Insufficient Storage (WebDAV, RFC4918) */
public static final HttpResponseStatus INSUFFICIENT_STORAGE = newStatus(507, "Insufficient Storage");
510 Not Extended (RFC2774)
/** * 510 Not Extended (RFC2774) */
public static final HttpResponseStatus NOT_EXTENDED = newStatus(510, "Not Extended");
511 Network Authentication Required (RFC6585)
/** * 511 Network Authentication Required (RFC6585) */
public static final HttpResponseStatus NETWORK_AUTHENTICATION_REQUIRED = newStatus(511, "Network Authentication Required"); private static HttpResponseStatus newStatus(int statusCode, String reasonPhrase) { return new HttpResponseStatus(statusCode, reasonPhrase, true); }
Returns the HttpResponseStatus represented by the specified code. If the specified code is a standard HTTP status code, a cached instance will be returned. Otherwise, a new instance will be returned.
/** * Returns the {@link HttpResponseStatus} represented by the specified code. * If the specified code is a standard HTTP status code, a cached instance * will be returned. Otherwise, a new instance will be returned. */
public static HttpResponseStatus valueOf(int code) { HttpResponseStatus status = valueOf0(code); return status != null ? status : new HttpResponseStatus(code); } private static HttpResponseStatus valueOf0(int code) { switch (code) { case 100: return CONTINUE; case 101: return SWITCHING_PROTOCOLS; case 102: return PROCESSING; case 200: return OK; case 201: return CREATED; case 202: return ACCEPTED; case 203: return NON_AUTHORITATIVE_INFORMATION; case 204: return NO_CONTENT; case 205: return RESET_CONTENT; case 206: return PARTIAL_CONTENT; case 207: return MULTI_STATUS; case 300: return MULTIPLE_CHOICES; case 301: return MOVED_PERMANENTLY; case 302: return FOUND; case 303: return SEE_OTHER; case 304: return NOT_MODIFIED; case 305: return USE_PROXY; case 307: return TEMPORARY_REDIRECT; case 308: return PERMANENT_REDIRECT; case 400: return BAD_REQUEST; case 401: return UNAUTHORIZED; case 402: return PAYMENT_REQUIRED; case 403: return FORBIDDEN; case 404: return NOT_FOUND; case 405: return METHOD_NOT_ALLOWED; case 406: return NOT_ACCEPTABLE; case 407: return PROXY_AUTHENTICATION_REQUIRED; case 408: return REQUEST_TIMEOUT; case 409: return CONFLICT; case 410: return GONE; case 411: return LENGTH_REQUIRED; case 412: return PRECONDITION_FAILED; case 413: return REQUEST_ENTITY_TOO_LARGE; case 414: return REQUEST_URI_TOO_LONG; case 415: return UNSUPPORTED_MEDIA_TYPE; case 416: return REQUESTED_RANGE_NOT_SATISFIABLE; case 417: return EXPECTATION_FAILED; case 421: return MISDIRECTED_REQUEST; case 422: return UNPROCESSABLE_ENTITY; case 423: return LOCKED; case 424: return FAILED_DEPENDENCY; case 425: return UNORDERED_COLLECTION; case 426: return UPGRADE_REQUIRED; case 428: return PRECONDITION_REQUIRED; case 429: return TOO_MANY_REQUESTS; case 431: return REQUEST_HEADER_FIELDS_TOO_LARGE; case 500: return INTERNAL_SERVER_ERROR; case 501: return NOT_IMPLEMENTED; case 502: return BAD_GATEWAY; case 503: return SERVICE_UNAVAILABLE; case 504: return GATEWAY_TIMEOUT; case 505: return HTTP_VERSION_NOT_SUPPORTED; case 506: return VARIANT_ALSO_NEGOTIATES; case 507: return INSUFFICIENT_STORAGE; case 510: return NOT_EXTENDED; case 511: return NETWORK_AUTHENTICATION_REQUIRED; } return null; }
Returns the HttpResponseStatus represented by the specified code and reasonPhrase. If the specified code is a standard HTTP status code and reasonPhrase, a cached instance will be returned. Otherwise, a new instance will be returned.
Params:
  • code – The response code value.
  • reasonPhrase – The response code reason phrase.
Returns:the HttpResponseStatus represented by the specified code and reasonPhrase.
/** * Returns the {@link HttpResponseStatus} represented by the specified {@code code} and {@code reasonPhrase}. * If the specified code is a standard HTTP status {@code code} and {@code reasonPhrase}, a cached instance * will be returned. Otherwise, a new instance will be returned. * @param code The response code value. * @param reasonPhrase The response code reason phrase. * @return the {@link HttpResponseStatus} represented by the specified {@code code} and {@code reasonPhrase}. */
public static HttpResponseStatus valueOf(int code, String reasonPhrase) { HttpResponseStatus responseStatus = valueOf0(code); return responseStatus != null && responseStatus.reasonPhrase().contentEquals(reasonPhrase) ? responseStatus : new HttpResponseStatus(code, reasonPhrase); }
Parses the specified HTTP status line into a HttpResponseStatus. The expected formats of the line are:
  • statusCode (e.g. 200)
  • statusCode reasonPhrase (e.g. 404 Not Found)
Throws:
/** * Parses the specified HTTP status line into a {@link HttpResponseStatus}. The expected formats of the line are: * <ul> * <li>{@code statusCode} (e.g. 200)</li> * <li>{@code statusCode} {@code reasonPhrase} (e.g. 404 Not Found)</li> * </ul> * * @throws IllegalArgumentException if the specified status line is malformed */
public static HttpResponseStatus parseLine(CharSequence line) { return (line instanceof AsciiString) ? parseLine((AsciiString) line) : parseLine(line.toString()); }
Parses the specified HTTP status line into a HttpResponseStatus. The expected formats of the line are:
  • statusCode (e.g. 200)
  • statusCode reasonPhrase (e.g. 404 Not Found)
Throws:
/** * Parses the specified HTTP status line into a {@link HttpResponseStatus}. The expected formats of the line are: * <ul> * <li>{@code statusCode} (e.g. 200)</li> * <li>{@code statusCode} {@code reasonPhrase} (e.g. 404 Not Found)</li> * </ul> * * @throws IllegalArgumentException if the specified status line is malformed */
public static HttpResponseStatus parseLine(String line) { try { int space = line.indexOf(' '); return space == -1 ? valueOf(parseInt(line)) : valueOf(parseInt(line.substring(0, space)), line.substring(space + 1)); } catch (Exception e) { throw new IllegalArgumentException("malformed status line: " + line, e); } }
Parses the specified HTTP status line into a HttpResponseStatus. The expected formats of the line are:
  • statusCode (e.g. 200)
  • statusCode reasonPhrase (e.g. 404 Not Found)
Throws:
/** * Parses the specified HTTP status line into a {@link HttpResponseStatus}. The expected formats of the line are: * <ul> * <li>{@code statusCode} (e.g. 200)</li> * <li>{@code statusCode} {@code reasonPhrase} (e.g. 404 Not Found)</li> * </ul> * * @throws IllegalArgumentException if the specified status line is malformed */
public static HttpResponseStatus parseLine(AsciiString line) { try { int space = line.forEachByte(FIND_ASCII_SPACE); return space == -1 ? valueOf(line.parseInt()) : valueOf(line.parseInt(0, space), line.toString(space + 1)); } catch (Exception e) { throw new IllegalArgumentException("malformed status line: " + line, e); } } private final int code; private final AsciiString codeAsText; private HttpStatusClass codeClass; private final String reasonPhrase; private final byte[] bytes;
Creates a new instance with the specified code and the auto-generated default reason phrase.
/** * Creates a new instance with the specified {@code code} and the auto-generated default reason phrase. */
private HttpResponseStatus(int code) { this(code, HttpStatusClass.valueOf(code).defaultReasonPhrase() + " (" + code + ')', false); }
Creates a new instance with the specified code and its reasonPhrase.
/** * Creates a new instance with the specified {@code code} and its {@code reasonPhrase}. */
public HttpResponseStatus(int code, String reasonPhrase) { this(code, reasonPhrase, false); } private HttpResponseStatus(int code, String reasonPhrase, boolean bytes) { if (code < 0) { throw new IllegalArgumentException( "code: " + code + " (expected: 0+)"); } if (reasonPhrase == null) { throw new NullPointerException("reasonPhrase"); } for (int i = 0; i < reasonPhrase.length(); i ++) { char c = reasonPhrase.charAt(i); // Check prohibited characters. switch (c) { case '\n': case '\r': throw new IllegalArgumentException( "reasonPhrase contains one of the following prohibited characters: " + "\\r\\n: " + reasonPhrase); } } this.code = code; String codeString = Integer.toString(code); codeAsText = new AsciiString(codeString); this.reasonPhrase = reasonPhrase; if (bytes) { this.bytes = (codeString + ' ' + reasonPhrase).getBytes(CharsetUtil.US_ASCII); } else { this.bytes = null; } }
Returns the code of this HttpResponseStatus.
/** * Returns the code of this {@link HttpResponseStatus}. */
public int code() { return code; }
Returns the status code as AsciiString.
/** * Returns the status code as {@link AsciiString}. */
public AsciiString codeAsText() { return codeAsText; }
Returns the reason phrase of this HttpResponseStatus.
/** * Returns the reason phrase of this {@link HttpResponseStatus}. */
public String reasonPhrase() { return reasonPhrase; }
Returns the class of this HttpResponseStatus
/** * Returns the class of this {@link HttpResponseStatus} */
public HttpStatusClass codeClass() { HttpStatusClass type = this.codeClass; if (type == null) { this.codeClass = type = HttpStatusClass.valueOf(code); } return type; } @Override public int hashCode() { return code(); }
Equality of HttpResponseStatus only depends on code(). The reason phrase is not considered for equality.
/** * Equality of {@link HttpResponseStatus} only depends on {@link #code()}. The * reason phrase is not considered for equality. */
@Override public boolean equals(Object o) { if (!(o instanceof HttpResponseStatus)) { return false; } return code() == ((HttpResponseStatus) o).code(); }
Equality of HttpResponseStatus only depends on code(). The reason phrase is not considered for equality.
/** * Equality of {@link HttpResponseStatus} only depends on {@link #code()}. The * reason phrase is not considered for equality. */
@Override public int compareTo(HttpResponseStatus o) { return code() - o.code(); } @Override public String toString() { return new StringBuilder(reasonPhrase.length() + 4) .append(codeAsText) .append(' ') .append(reasonPhrase) .toString(); } void encode(ByteBuf buf) { if (bytes == null) { ByteBufUtil.copy(codeAsText, buf); buf.writeByte(SP); buf.writeCharSequence(reasonPhrase, CharsetUtil.US_ASCII); } else { buf.writeBytes(bytes); } } }