From 28c9de1d09b9d8da903dd80edda4a5c4277a5b88 Mon Sep 17 00:00:00 2001 From: Milad Khajavi Date: Tue, 27 Feb 2024 19:27:57 +0330 Subject: [PATCH] Response Documentation (#2696) --- docs/dsl/response.md | 369 +++++++++++++++++++++++++++++++++++++++++-- docs/dsl/status.md | 80 ++++++++++ docs/sidebars.js | 1 + 3 files changed, 434 insertions(+), 16 deletions(-) create mode 100644 docs/dsl/status.md diff --git a/docs/dsl/response.md b/docs/dsl/response.md index 63ce2142d..beb5fe028 100644 --- a/docs/dsl/response.md +++ b/docs/dsl/response.md @@ -6,58 +6,202 @@ title: Response **ZIO HTTP** `Response` is designed to encode HTTP Response. It supports all HTTP status codes and headers along with custom methods and headers (as defined in [RFC2616](https://datatracker.ietf.org/doc/html/rfc2616) ) +## Response Usage + +In ZIO HTTP, a `Response` is used in two contexts, server-side and client-side. + +### Server Side + +In the server-side context, a `Response` is created and returned by a `Handler`: + +```scala mdoc:compile-only +import zio._ +import zio.http._ + +object HelloWorldExample extends ZIOAppDefault { + val app: HttpApp[Any] = + Routes( + Method.GET / "text" -> + handler { + Response.text("Hello World!") + }, + ).toHttpApp + + override val run = Server.serve(app).provide(Server.default) +} +``` + +### Client Side + +In the client-side context, a `Response` is received from the client after making a request to a server: + +```scala mdoc:compile-only +import zio._ +import zio.http.Header.ContentType.render +import zio.http._ + +object ClientExample extends ZIOAppDefault { + val program = for { + res <- Client.request(Request.get("https://zio.dev/")) + contentType <- ZIO.from(res.header(Header.ContentType)) + _ <- Console.printLine("------Content Type------") + _ <- Console.printLine(render(contentType)) + data <- res.body.asString + _ <- Console.printLine("----------Body----------") + _ <- Console.printLine(data) + } yield () + + override val run = program.provide(Client.default, Scope.default) +} +``` + ## Creating a Response -`Response` can be created with `status`, `headers` and `data`. +A `Response` can be created with `status`, `headers`, and `data` using the default constructor: + +```scala +case class Response( + status: Status = Status.Ok, + headers: Headers = Headers.empty, + body: Body = Body.empty, +) +``` + +The below snippet creates a response with default params, `status` as `Status.OK`, `headers` as `Headers.empty`, and `data` as `Body.Empty`: -The below snippet creates a response with default params, `status` as `Status.OK`, `headers` as `Headers.empty` and `data` as `Body.Empty`. ```scala mdoc import zio.http._ import zio._ Response() ``` -### Empty Response -`ok` creates an empty response with status code 200 +### Status Codes + +ZIO HTTP has several constructors for the most common status codes: + +| Method | Description | Status Code | +|------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------|-------------------------------------| +| `Response.ok` | Successful request | 200 OK | +| `Response.badRequest` | The server cannot or will not process the request due to an apparent client error | 400 Bad Request | +| `Response.unauthorized` | Similar to 403 Forbidden, but specifically for use when authentication is required and has failed or has not yet been provided | 401 Unauthorized | +| `Response.forbidden` | The client does not have access rights to the content; that is, it is unauthorized | 403 Forbidden | +| `Response.notFound` | The requested resource could not be found but may be available in the future | 404 Not Found | +| `Response.internalServerError` | A generic error message, given when an unexpected condition was encountered and no more specific message is suitable | 500 Internal Server Error | +| `Response.serviceUnavailable` | The server cannot handle the request (because it is overloaded or down for maintenance) | 503 Service Unavailable | +| `Response.redirect` | Used to inform the client that the resource they're requesting is located at a different URI | 302 Found (Moved Temporarily) | +| `Response.seeOther` | Tells the client to look at a different URL for the requested resource | 303 See Other | +| `Response.gatewayTimeout` | The server was acting as a gateway or proxy and did not receive a timely response from the upstream server | 504 Gateway Timeout | +| `Response.httpVersionNotSupported` | The server does not support the HTTP protocol version that was used in the request | 505 HTTP Version Not Supported | +| `Response.networkAuthenticationRequired` | The client needs to authenticate to gain network access | 511 Network Authentication Required | +| `Response.notExtended` | Further extensions to the request are required for the server to fulfill it | 510 Not Extended | +| `Response.notImplemented` | The server either does not recognize the request method, or it lacks the ability to fulfill the request | 501 Not Implemented | + +For example, to create a response with status code 200, we can use `Response.ok`: ```scala mdoc Response.ok ``` -`status` creates an empty response with provided status code. +And also to create a response with status code 404, we can use `Response.badRequest`: ```scala mdoc +Response.notFound + +Response.notFound("The requested resource could not be found!") +``` + +If we want to create a response with a more specific status code, we can use the `status` method. It takes a `Status` as a parameter and returns a new `Response` with the corresponding status code: + +```scala mdoc:compile-only Response.status(Status.Continue) ``` -### Specialized Response Constructors +To learn more about status codes, see [Status](status.md) page. -`text` creates a response with data as text, content-type header set to text/plain and status code 200 +### From Plain Text, JSON, and HTML + +**`Response.text`** creates a response with data as text, content-type header set to text/plain, and status code 200: ```scala mdoc Response.text("hey") ``` -`json` creates a response with data as json, content-type header set to application/json and status code 200 +**`Response.json`** creates a response with data as JSON, content-type header set to `application/json`, and status code 200: ```scala mdoc Response.json("""{"greetings": "Hello World!"}""") ``` -`html` creates a response with data as html, content-type header set to text/html and status code 200 +**`Response.html`** creates a response with data as HTML, content-type header set to `text/html`, and status code 200. + ```scala mdoc import zio.http.template._ Response.html(Html.fromString("html text")) ``` +### Converting Failures to Responses + +The `Response` companion object provides some constructors to convert exceptions into responses. These constructors are useful for error handling by converting failures into appropriate HTTP responses: + +**`Response.fromThrowable`** Creates a new HTTP response based on the type of throwable provided. This method facilitates the conversion of various types of exceptions into appropriate HTTP responses, making error handling more streamlined: + +```scala +object Response { + def fromThrowable(throwable: Throwable): Response = ??? +} +``` + +Here is the table of exceptions and their corresponding status code: + +| Throwable Type | Status Class | Status Code | Description | +|-----------------------------------|-----------------------|-------------|----------------------------------------------| +| `AccessDeniedException` | `Forbidden` | 403 | Access to a resource is denied. | +| `IllegalAccessException` | `Forbidden` | 403 | Illegal access to a resource is attempted. | +| `IllegalAccessError` | `Forbidden` | 403 | Illegal access to a resource occurs. | +| `NotDirectoryException` | `BadRequest` | 400 | The specified path is not a directory. | +| `IllegalArgumentException` | `BadRequest` | 400 | An invalid argument is provided. | +| `java.io.FileNotFoundException` | `NotFound` | 404 | The specified file or resource is not found. | +| `java.net.ConnectException` | `ServiceUnavailable` | 503 | Unable to connect to a service. | +| `java.net.SocketTimeoutException` | `GatewayTimeout` | 504 | Connection or read operation timed out. | +| Others (unrecognized throwable) | `InternalServerError` | 500 | An unexpected error occurred. | + +Another low-level method for error handling is `Response.fromCause` which creates a response from a `Cause`: + +```scala +object Response { + def fromCause(cause: Cause[Any]): Response = ??? +} +``` + +This constructor is similar to `Response.fromThrowable`, but it also captures the interruption of the fiber. If the provided `Cause` is a failure due to interruption, the status code of the response will be `Status.RequestTimeout`. + +We can use `Response.fromCause` in combination with the `Handler#mapErrorCause`, `Route#handleErrorCause`, and `Routes#handleErrorCause` methods. These methods take a function that maps the `Cause[Err] => Err` and return a `Handler`, `Route` or a `Routes` with the error handling logic applied: + +```scala mdoc +import zio.http._ +import java.io.IOException + +val failedHandler = Handler.fail(new IOException()) + +failedHandler.mapErrorCause(Response.fromCause) +``` + +:::note +In many cases, it is more convenient to use the `sandbox` method to automatically convert all failures into a corresponding `Response`. But in some cases, to have more granular control over the error handling, we may want to use `Response.fromCause` and `Response.fromThrowable` directly. +::: + +:::info +The `Cause` is a data structure that represents the result of a failed computation in ZIO. To learn more about `Cause`, see the [Cause](https://zio.dev/reference/core/cause) page on the ZIO core documentation. +::: + ### Specialized Response Operators `status` to update the `status` of `Response` -```scal mdoca -Response.text("Hello World!").status(Status.NOT_FOUND) +```scala mdoc +Response.text("Hello World!").status(Status.NotFound) ``` `updateHeaders` to update the `headers` of `Response` @@ -66,7 +210,7 @@ Response.text("Hello World!").status(Status.NOT_FOUND) Response.ok.updateHeaders(_ => Headers("key", "value")) ``` -### Response from Http Errors +### Response from HTTP Errors `error` creates a response with a provided status code and message. @@ -74,10 +218,203 @@ Response.ok.updateHeaders(_ => Headers("key", "value")) Response.error(Status.BadRequest, "It's not good!") ``` -## Adding Cookie to Response +### Creating a Response from Server-Sent Events + +The `Response.fromServerSentEvents` method creates a response with a stream of server-sent events: + +```scala +object Response { + def fromServerSentEvents(stream: ZStream[Any, Nothing, ServerSentEvent]): Response = ??? +} +``` + +Let's try a complete example: + +```scala mdoc:compile-only +import zio._ +import zio.http._ +import zio.stream._ +import java.time.LocalDateTime +import java.time.format.DateTimeFormatter.ISO_LOCAL_TIME + +object ServerSentExample extends ZIOAppDefault { + + val stream: ZStream[Any, Nothing, ServerSentEvent] = + ZStream.repeatWithSchedule( + ServerSentEvent(ISO_LOCAL_TIME.format(LocalDateTime.now)), + Schedule.spaced(1.second), + ) + + val app = + Routes( + Method.GET / "events" -> handler { + Response.fromServerSentEvents(stream) + }, + ).toHttpApp + def run = Server.serve(app).provide(Server.default) +} +``` + +After running the above example, we can open the browser and navigate to `http://localhost:8080/events` to see the server-sent events in action. The browser will display the time every second. + +Also, we can use the `curl` command to see the server-sent events: + +```bash +curl -N http://localhost:8080/events +``` + +This will display the time every second in the terminal: + +```bash +data: 13:51:31.036249 + +data: 13:51:32.037663 + +data: 13:51:33.039565 + +data: 13:51:34.041464 + +... +``` + +### Creating a Response from a WebSocketApp + +The `Response.fromWebSocketApp` constructor takes a `WebSocketApp` and creates a `Response` with a WebSocket connection: + +```scala +object Response { + def fromWebSocketApp[R](app: WebSocketApp[R]): ZIO[R, Nothing, Response] = ??? +} +``` + +Let's try an echo server which sends back the received messages: + +```scala mdoc:compile-only +import zio._ +import zio.http._ + +object WebsocketExample extends ZIOAppDefault { + + val app: HttpApp[Any] = { + Routes( + Method.GET / "echo" -> handler { + Response.fromSocketApp( + WebSocketApp( + handler { (channel: WebSocketChannel) => + channel.receiveAll { + case ChannelEvent.Read(message) => + channel.send(ChannelEvent.read(message)) + case other => + ZIO.debug(other) + } + }, + ), + ) + }, + ).toHttpApp + } + + def run = + Server.serve(app).provide(Server.default) +} +``` + +To test this example, we can use the `websocat` command-line tool: + +```bash +> websocat ws://localhost:8080/echo +hello +hello +bye +bye +``` + +## Operations + +### Adding Cookies and Flashes to Response + +**`addCookie`** adds cookies in the headers of the response: + +```scala mdoc:compile-only +import zio.http._ -`addCookie` adds cookies in the headers of the response. -```scala mdoc val cookie = Cookie.Response("key", "value") Response.ok.addCookie(cookie) -``` \ No newline at end of file +``` + +**`addFlash`** adds flash messages in the headers of the response: + +```scala mdoc:compile-only +import zio.http._ + +val flash = Flash.setValue("key1", "value1") +Response.ok.addFlash(flash) +``` + +### Getting Headers from Response + +To get headers from a response, we can use the `header` method: + +```scala mdoc:invisible +val response = Response.ok.addHeader("Content-Type", "application/json; charset=utf-8") +``` + +```scala mdoc +response.header(Header.ContentType) +``` + +List of methods available to get headers from a `Response`: + +| Method | Description | Return Type | +|----------------------------------------|-----------------------------------------------------------------------------------------------|--------------------------------------------------| +| `header(headerType: HeaderType)` | Gets a header or returns `None` if not present or unparsable. | `Option[headerType.HeaderValue]` | +| `headers(headerType: HeaderType)` | Gets multiple headers of the specified type. | `Chunk[headerType.HeaderValue]` | +| `headerOrFail(headerType: HeaderType)` | Gets a header, returning `None` if absent, or an `Either` with parsing error or parsed value. | `Option[Either[String, headerType.HeaderValue]]` | +| `headers` | Returns all headers. | `Headers` | +| `rawHeader(name: CharSequence)` | Gets the raw unparsed value of a header by name. | `Option[String]` | +| `rawHeader(headerType: HeaderType)` | Gets the raw unparsed value of a header by type. | `Option[String]` | + +### Modifying Headers in Response + +To add headers to a response, we can use the `Response#addHeader` method: + +```scala mdoc +Response.ok.addHeader("Content-Type", "application/json; charset=utf-8") + +Response.ok.addHeader(Header.ContentType(MediaType.application.json)) +``` + +Here are the methods available to modify headers in a `Response`: + +| Method | Description | +|-----------------|----------------------------------------------------------------------------------------------------| +| `addHeader` | Adds a header or a header with the given name and value, returning a new `Response` instance. | +| `addHeaders` | Adds multiple headers, returning a new `Response` instance. | +| `removeHeader` | Removes a specified header, returning a new `Response` instance. | +| `removeHeaders` | Removes multiple specified headers, returning a new `Response` instance. | +| `setHeaders` | Sets the headers to the provided headers, returning a new `Response` instance. | +| `updateHeaders` | Updates the current headers using a provided update function, returning a new `Response` instance. | + +### Checking for Headers in Response + +The `Response` class provides methods that allow us to check if the headers meet certain conditions: + +```scala mdoc +response.hasContentType(MediaType.application.json.fullType) +``` + +There are several such methods available in the `Response` class: + +| `Method` | Description | +|---------------------------------------|----------------------------------------------------------------| +| `hasContentType(value: CharSequence)` | Checks if the headers have the given content type. | +| `hasFormUrlencodedContentType` | Checks if the headers have a form-urlencoded content type. | +| `hasFormMultipartContentType` | Checks if the headers have a multipart/form-data content type. | +| `hasHeader(name: CharSequence)` | Checks if the headers contain a header with the given name. | +| `hasHeader(headerType: HeaderType)` | Checks if the headers contain a header of the given type. | +| `hasHeader(header: Header)` | Checks if the headers contain a specific header. | +| `hasJsonContentType` | Checks if the headers have a JSON content type. | +| `hasMediaType(mediaType: MediaType)` | Checks if the headers have the specified media type. | +| `hasTextPlainContentType` | Checks if the headers have a text/plain content type. | +| `hasXhtmlXmlContentType` | Checks if the headers have an XHTML/XML content type. | +| `hasXmlContentType` | Checks if the headers have an XML content type. | diff --git a/docs/dsl/status.md b/docs/dsl/status.md new file mode 100644 index 000000000..be81a9c16 --- /dev/null +++ b/docs/dsl/status.md @@ -0,0 +1,80 @@ +--- +id: status +title: Status +--- + +HTTP status codes are standard response codes given by web services on the Internet. The codes help identify the cause of the problem when a web page or other resource does not load properly. + +In ZIO HTTP, `Status` is a sealed trait that represents the status code of an HTTP response. It is designed to encode HTTP status codes and headers along with custom methods and headers (as defined in [RFC2615](https://datatracker.ietf.org/doc/html/rfc2616)). + +The ZIO HTTP has a predefined set of constructors for `Response` for the most common status codes. +To create a `Response` with a status code, we can use the predefined constructors for the most common status codes, like `Response.ok`, `Response.notFound`, `Response.forbidden`, etc. + +But, if we need a specific status code, we can pass one of the following status codes to the `Response.status` constructor: + +```scala mdoc +import zio.http._ + +Response.status(Status.PermanentRedirect) +``` + +Here is a list of status codes and their descriptions: + +| Category | Status Class Name | Description | Code Range | +|---------------|------------------------|------------------------------------------------------------------|-------------| +| Informational | `Continue` | The server has received the request headers, and the client should proceed to send the request body. | 100 | +| | `SwitchingProtocols` | The server is switching protocols according to the client's request. | 101 | +| | `Processing` | The server is processing the request but has not completed it yet. | 102 | +| Success | `Ok` | The request has succeeded. | 200 | +| | `Created` | The request has been fulfilled and resulted in a new resource being created. | 201 | +| | `Accepted` | The request has been accepted for processing but has not been completed. | 202 | +| | `NonAuthoritativeInformation` | The returned meta-information in the entity-header is not the definitive set available from the origin server. | 203 | +| | `NoContent` | The server successfully processed the request and is not returning any content. | 204 | +| | `ResetContent` | The server successfully processed the request but is not returning any content. | 205 | +| | `PartialContent` | The server successfully processed only part of the request. | 206 | +| | `MultiStatus` | The server has multiple status codes for different independent operations. | 207 | +| Redirection | `MultipleChoices` | The request has multiple possible responses and the user or user agent can choose the correct one. | 300 | +| | `MovedPermanently` | The requested page has been permanently moved to a new location. | 301 | +| | `Found` | The requested page has been found but is temporarily located at another URI. | 302 | +| | `SeeOther` | The response to the request can be found under another URI using a GET method. | 303 | +| | `NotModified` | The resource has not been modified since the last request. | 304 | +| | `UseProxy` | The requested resource is available only through a proxy, whose address is provided in the response. | 305 | +| | `TemporaryRedirect` | The requested resource has been temporarily moved to a different URI. | 307 | +| | `PermanentRedirect` | The requested resource has been permanently moved to a different URI. | 308 | +| Client Error | `BadRequest` | The request cannot be fulfilled due to bad syntax. | 400 | +| | `Unauthorized` | The request requires user authentication. | 401 | +| | `PaymentRequired` | Reserved for future use. | 402 | +| | `Forbidden` | The server understood the request, but is refusing to fulfill it. | 403 | +| | `NotFound` | The requested resource could not be found but may be available again in the future. | 404 | +| | `MethodNotAllowed` | The method specified in the request is not allowed for the resource identified by the request URI. | 405 | +| | `NotAcceptable` | The resource identified by the request is only capable of generating response entities that have content characteristics not acceptable according to the accept headers sent in the request. | 406 | +| | `ProxyAuthenticationRequired` | Similar to 401 but authentication is needed for accessing the proxy. | 407 | +| | `RequestTimeout` | The server timed out waiting for the request. | 408 | +| | `Conflict` | Indicates that the request could not be processed because of conflict in the request. | 409 | +| | `Gone` | The requested resource is no longer available at the server and no forwarding address is known. | 410 | +| | `LengthRequired` | The server refuses to accept the request without a defined Content-Length. | 411 | +| | `PreconditionFailed` | The precondition given in one or more of the request-header fields evaluated to false when it was tested on the server. | 412 | +| | `RequestEntityTooLarge` | The server is refusing to service the request because the request entity is larger than the server is willing or able to process. | 413 | +| | `RequestUriTooLong` | The server is refusing to interpret the request because the Request-URI is longer than the server is willing to interpret. | 414 | +| | `UnsupportedMediaType` | The request entity has a media type that the server or resource does not support. | 415 | +| | `RequestedRangeNotSatisfiable` | The client has asked for a portion of the file, but the server cannot supply that portion. | 416 | +| | `ExpectationFailed` | The server cannot meet the requirements of the Expect request-header field. | 417 | +| | `MisdirectedRequest` | The request was directed at a server that is not able to produce a response. | 421 | +| | `UnprocessableEntity` | The request was well-formed but was unable to be followed due to semantic errors. | 422 | +| | `Locked` | The resource that is being accessed is locked. | 423 | +| | `FailedDependency` | The method could not be performed on the resource because the server is unable to store the representation needed to successfully complete the request. | 424 | +| | `UnorderedCollection` | The server is not able to produce a response representing the current state of the target resource due to some internal error. | 425 | +| | `UpgradeRequired` | The server refuses to perform the request using the current protocol but might be willing to do so after the client upgrades to a different protocol. | 426 | +| | `PreconditionRequired` | The origin server requires the request to be conditional. | 428 | +| | `TooManyRequests` | The user has sent too many requests in a given amount of time ("rate limiting"). | 429 | +| | `RequestHeaderFieldsTooLarge` | The server is unwilling to process the request because either an individual header field or all the header fields collectively are too large. | 431 | +| Server Error | `InternalServerError` | A generic error message, given when an unexpected condition was encountered and no more specific message is suitable. | 500 | +| | `NotImplemented` | The server either does not recognize the request method, or it lacks the ability to fulfill the request. | 501 | +| | `BadGateway` | The server was acting as a gateway or proxy and received an invalid response from the upstream server. | 502 | +| | `ServiceUnavailable` | The server is currently unable to handle the request due to a temporary overload or maintenance of the server. | 503 | +| | `GatewayTimeout` | The server, while acting as a gateway or proxy, did not receive a timely response from the upstream server specified by the URI. | 504 | +| | `HttpVersionNotSupported` | The server does not support the HTTP protocol version that was used in the request. | 505 | +| | `VariantAlsoNegotiates` | Transparent content negotiation for the request results in a circular reference. | 506 | +| | `InsufficientStorage` | The server is unable to store the representation needed to complete the request. | 507 | +| | `NotExtended` | Further extensions to the request are required for the server to fulfill it. | 510 | +| | `NetworkAuthenticationRequired` | The client needs to authenticate to gain network access. | 511 | diff --git a/docs/sidebars.js b/docs/sidebars.js index 6b9f726fc..2fd6e1479 100644 --- a/docs/sidebars.js +++ b/docs/sidebars.js @@ -22,6 +22,7 @@ const sidebars = { "dsl/headers", "dsl/cookies", "dsl/middleware", + "dsl/status", { type: "category", label: "DSL",