io.helidon.http.Status Maven / Gradle / Ivy
Show all versions of helidon-http Show documentation
/*
* Copyright (c) 2024 Oracle and/or its affiliates.
*
* 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 io.helidon.http;
import java.util.Objects;
/**
* Commonly used status codes defined by HTTP, see
* HTTP/1.1 documentation.
* Additional status codes can be added by applications
* by call {@link #create(int)} or {@link #create(int, String)} with unknown status code, or with text
* that differs from the predefined status codes.
*
* Although the constants are instances of this class, they can be compared using instance equality, as the only
* way to obtain an instance is through methods {@link #create(int)} {@link #create(int, String)}, which ensures
* the same instance is returned for known status codes and reason phrases.
*
* A good reference is the IANA list of HTTP Status Codes (we may not cover all of them in this type):
* IANA HTTP Status Codes
*/
public class Status {
/**
* 100 Continue,
* see HTTP/1.1 documentations.
*/
public static final Status CONTINUE_100 = new Status(100, "Continue", true);
/**
* 101 Switching Protocols,
* see HTTP/1.1 documentations.
*/
public static final Status SWITCHING_PROTOCOLS_101 = new Status(101, "Switching Protocols", true);
/**
* 200 OK, see HTTP/1.1 documentation.
*/
public static final Status OK_200 = new Status(200, "OK", true);
/**
* 201 Created, see HTTP/1.1 documentation.
*/
public static final Status CREATED_201 = new Status(201, "Created", true);
/**
* 202 Accepted, see HTTP/1.1 documentation
* .
*/
public static final Status ACCEPTED_202 = new Status(202, "Accepted", true);
/**
* 203 Non-Authoritative Information, see
* HTTP/1.1 documentation.
*
* @since 4.0.6
*/
public static final Status NON_AUTHORITATIVE_INFORMATION_203 = new Status(203, "Non-Authoritative Information", true);
/**
* 204 No Content, see
* HTTP/1.1 documentation.
*/
public static final Status NO_CONTENT_204 = new Status(204, "No Content", true);
/**
* 205 Reset Content, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status RESET_CONTENT_205 = new Status(205, "Reset Content", true);
/**
* 206 Reset Content, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status PARTIAL_CONTENT_206 = new Status(206, "Partial Content", true);
/**
* 207 Multi-Status, see
* RFC 4918 - HTTP Extensions for WebDAV.
*
* @since 4.0.6
*/
public static final Status MULTI_STATUS_207 = new Status(207, "Multi-Status", true);
/**
* 301 Moved Permanently, see
* HTTP/1.1 documentation.
*/
public static final Status MOVED_PERMANENTLY_301 = new Status(301, "Moved Permanently", true);
/**
* 302 Found, see HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status FOUND_302 = new Status(302, "Found", true);
/**
* 303 See Other, see
* HTTP/1.1 documentation.
*/
public static final Status SEE_OTHER_303 = new Status(303, "See Other", true);
/**
* 304 Not Modified, see
* HTTP/1.1 documentation.
*/
public static final Status NOT_MODIFIED_304 = new Status(304, "Not Modified", true);
/**
* 305 Use Proxy, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status USE_PROXY_305 = new Status(305, "Use Proxy", true);
/**
* 307 Temporary Redirect, see
* HTTP/1.1 documentation.
*/
public static final Status TEMPORARY_REDIRECT_307 = new Status(307, "Temporary Redirect", true);
/**
* 308 Permanent Redirect, see
* HTTP Status Code 308 documentation.
*/
public static final Status PERMANENT_REDIRECT_308 = new Status(308, "Permanent Redirect", true);
/**
* 400 Bad Request, see
* HTTP/1.1 documentation.
*/
public static final Status BAD_REQUEST_400 = new Status(400, "Bad Request", true);
/**
* 401 Unauthorized, see
* HTTP/1.1 documentation.
*/
public static final Status UNAUTHORIZED_401 = new Status(401, "Unauthorized", true);
/**
* 402 Payment Required, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status PAYMENT_REQUIRED_402 = new Status(402, "Payment Required", true);
/**
* 403 Forbidden, see
* HTTP/1.1 documentation.
*/
public static final Status FORBIDDEN_403 = new Status(403, "Forbidden", true);
/**
* 404 Not Found, see
* HTTP/1.1 documentation.
*/
public static final Status NOT_FOUND_404 = new Status(404, "Not Found", true);
/**
* 405 Method Not Allowed, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status METHOD_NOT_ALLOWED_405 = new Status(405, "Method Not Allowed", true);
/**
* 406 Not Acceptable, see
* HTTP/1.1 documentation.
*/
public static final Status NOT_ACCEPTABLE_406 = new Status(406, "Not Acceptable", true);
/**
* 407 Proxy Authentication Required, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status PROXY_AUTHENTICATION_REQUIRED_407 = new Status(407, "Proxy Authentication Required", true);
/**
* 408 Request Timeout, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status REQUEST_TIMEOUT_408 = new Status(408, "Request Timeout", true);
/**
* 409 Conflict, see
* HTTP/1.1 documentation.
*/
public static final Status CONFLICT_409 = new Status(409, "Conflict", true);
/**
* 410 Gone, see HTTP/1.1 documentation.
*/
public static final Status GONE_410 = new Status(410, "Gone", true);
/**
* 411 Length Required, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status LENGTH_REQUIRED_411 = new Status(411, "Length Required", true);
/**
* 412 Precondition Failed, see
* HTTP/1.1 documentation.
*/
public static final Status PRECONDITION_FAILED_412 = new Status(412, "Precondition Failed", true);
/**
* 413 Request Entity Too Large, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status REQUEST_ENTITY_TOO_LARGE_413 = new Status(413, "Request Entity Too Large", true);
/**
* 414 Request-URI Too Long, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status REQUEST_URI_TOO_LONG_414 = new Status(414, "Request-URI Too Long", true);
/**
* 415 Unsupported Media Type, see
* HTTP/1.1 documentation.
*/
public static final Status UNSUPPORTED_MEDIA_TYPE_415 = new Status(415, "Unsupported Media Type", true);
/**
* 416 Requested Range Not Satisfiable, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status REQUESTED_RANGE_NOT_SATISFIABLE_416 = new Status(416, "Requested Range Not Satisfiable", true);
/**
* 417 Expectation Failed, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status EXPECTATION_FAILED_417 = new Status(417, "Expectation Failed", true);
/**
* 418 I'm a teapot, see
* Hyper Text Coffee Pot Control Protocol (HTCPCP/1.0).
*/
public static final Status I_AM_A_TEAPOT_418 = new Status(418, "I'm a teapot", true);
/**
* Misdirected request, see
* RFC 9110 - Http Semantics.
*/
public static final Status MISDIRECTED_REQUEST_421 = new Status(421, "Misdirected Request", true);
/**
* Unprocessable content, see
* RFC 9110 - Http Semantics.
*/
public static final Status UNPROCESSABLE_CONTENT_422 = new Status(422, "Unprocessable Content", true);
/**
* Locked, see
* RFC 4918 - HTTP Extensions for WebDAV.
*/
public static final Status LOCKED_423 = new Status(423, "Locked", true);
/**
* Failed dependency, see
* RFC 4918 - HTTP Extensions for WebDAV.
*/
public static final Status FAILED_DEPENDENCY_424 = new Status(424, "Failed Dependency", true);
/**
* Upgrade required, see
* RFC 9110 - Http Semantics.
*/
public static final Status UPGRADE_REQUIRED_426 = new Status(426, "Upgrade Required", true);
/**
* Precondition required, see
* RFC 6585 - Additional HTTP Status Codes.
*/
public static final Status PRECONDITION_REQUIRED_428 = new Status(428, "Precondition Required", true);
/**
* Too many requests, see
* RFC 6585 - Additional HTTP Status Codes.
*/
public static final Status TOO_MANY_REQUESTS_429 = new Status(429, "Too Many Requests", true);
/**
* 500 Internal Server Error, see
* HTTP/1.1 documentation.
*/
public static final Status INTERNAL_SERVER_ERROR_500 = new Status(500, "Internal Server Error", true);
/**
* 501 Not Implemented, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status NOT_IMPLEMENTED_501 = new Status(501, "Not Implemented", true);
/**
* 502 Bad Gateway, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status BAD_GATEWAY_502 = new Status(502, "Bad Gateway", true);
/**
* 503 Service Unavailable, see
* HTTP/1.1 documentation.
*/
public static final Status SERVICE_UNAVAILABLE_503 = new Status(503, "Service Unavailable", true);
/**
* 504 Gateway Timeout, see
* HTTP/1.1 documentation.
*
* @since 2.0
*/
public static final Status GATEWAY_TIMEOUT_504 = new Status(504, "Gateway Timeout", true);
/**
* 505 HTTP Version Not Supported, see
* HTTP/1.1 documentation.
*
* @since 3.0.3
*/
public static final Status HTTP_VERSION_NOT_SUPPORTED_505 = new Status(505, "HTTP Version Not Supported", true);
/**
* 507 Insufficient Storage, see
* WebDAV documentation.
*
* @since 4.0.6
*/
public static final Status INSUFFICIENT_STORAGE_507 = new Status(507, "Insufficient Storage", true);
/**
* 508 Loop Detected, see
* RFC 5842 - Bindings for the Constrained Application Protocol (CoAP).
*
* @since 4.0.6
*/
public static final Status LOOP_DETECTED_508 = new Status(508, "Loop Detected", true);
/**
* 510 Not Extended, see
* RFC 2774 - An HTTP Extension Framework.
*
* @since 4.0.6
*/
public static final Status NOT_EXTENDED_510 = new Status(510, "Not Extended", true);
/**
* 511 Network Authentication Required, see
* RFC 6585 - Additional HTTP Status Codes.
*
* @since 4.0.6
*/
public static final Status NETWORK_AUTHENTICATION_REQUIRED_511 = new Status(511, "Network Authentication Required", true);
static {
// THIS MUST BE AFTER THE LAST CONSTANT
StatusHelper.statusesDone();
}
private final int code;
private final String reason;
private final Family family;
private final String codeText;
private final String stringValue;
private Status(int statusCode, String reasonPhrase, boolean instance) {
this.code = statusCode;
this.reason = reasonPhrase;
this.family = Family.of(statusCode);
this.codeText = String.valueOf(code);
this.stringValue = code + " " + reason;
if (instance) {
StatusHelper.add(this);
}
}
private Status(int statusCode, String reasonPhrase, Family family, String codeText) {
// for custom codes
this.code = statusCode;
this.reason = reasonPhrase;
this.family = family;
this.codeText = codeText;
this.stringValue = code + " " + reason;
}
/**
* Convert a numerical status code into the corresponding Status.
*
* For an unknown code, an ad-hoc {@link Status} is created.
*
* @param statusCode the numerical status code
* @return the matching Status; either a constant from this class, or an ad-hoc {@link Status}
*/
public static Status create(int statusCode) {
Status found = StatusHelper.find(statusCode);
if (found == null) {
return createNew(Family.of(statusCode), statusCode, "", String.valueOf(statusCode));
}
return found;
}
/**
* Convert a numerical status code into the corresponding Status.
*
* It either returns an existing {@link Status} constant if possible.
* For an unknown code, or code/reason phrase combination it creates
* an ad-hoc {@link Status}.
*
* @param statusCode the numerical status code
* @param reasonPhrase the reason phrase; if {@code null} or a known reason phrase, an instance with the default
* phrase is returned; otherwise, a new instance is returned
* @return the matching Status
*/
public static Status create(int statusCode, String reasonPhrase) {
Status found = StatusHelper.find(statusCode);
if (found == null) {
return createNew(Family.of(statusCode), statusCode, reasonPhrase, String.valueOf(statusCode));
}
if (reasonPhrase == null) {
return found;
}
if (found.reasonPhrase().equalsIgnoreCase(reasonPhrase)) {
return found;
}
return createNew(found.family(), statusCode, reasonPhrase, found.codeText());
}
private static Status createNew(Family family, int statusCode, String reasonPhrase, String codeText) {
return new Status(statusCode, reasonPhrase, family, codeText);
}
/**
* Get the associated integer value representing the status code.
*
* @return the integer value representing the status code.
*/
public int code() {
return code;
}
/**
* Get the class of status code.
*
* @return the class of status code.
*/
public Family family() {
return family;
}
/**
* Get the reason phrase.
*
* @return the reason phrase.
*/
public String reasonPhrase() {
return reason;
}
/**
* Text of the {@link #code()}.
*
* @return code string (number as a string)
*/
public String codeText() {
return codeText;
}
/**
* Get the response status as string.
*
* @return the response status string in the form of a partial HTTP response status line,
* i.e. {@code "Status-Code SP Reason-Phrase"}.
*/
public String toString() {
return stringValue;
}
/**
* Text of the status as used in HTTP/1, such as "200 OK".
*
* @return text of this status
*/
public String text() {
return stringValue;
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
Status status = (Status) o;
return code == status.code && reason.equals(status.reason);
}
@Override
public int hashCode() {
return Objects.hash(code, reason);
}
/**
* An enumeration representing the class of status code. Family is used
* here since class is overloaded in Java.
*
* Copied from JAX-RS.
*/
public enum Family {
/**
* {@code 1xx} HTTP status codes.
*/
INFORMATIONAL,
/**
* {@code 2xx} HTTP status codes.
*/
SUCCESSFUL,
/**
* {@code 3xx} HTTP status codes.
*/
REDIRECTION,
/**
* {@code 4xx} HTTP status codes.
*/
CLIENT_ERROR,
/**
* {@code 5xx} HTTP status codes.
*/
SERVER_ERROR,
/**
* Other, unrecognized HTTP status codes.
*/
OTHER;
/**
* Get the family for the response status code.
*
* @param statusCode response status code to get the family for.
* @return family of the response status code.
*/
public static Family of(int statusCode) {
return switch (statusCode / 100) {
case 1 -> Family.INFORMATIONAL;
case 2 -> Family.SUCCESSFUL;
case 3 -> Family.REDIRECTION;
case 4 -> Family.CLIENT_ERROR;
case 5 -> Family.SERVER_ERROR;
default -> Family.OTHER;
};
}
}
}