All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.helidon.http.Status Maven / Gradle / Ivy

The newest version!
/*
 * 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; }; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy