raxx v0.14.9 Raxx.Server behaviour
Interface to handle server side communication in an HTTP message exchange.
Using Raxx.Server
allows an application to be run on multiple adapters.
For example Ace
has several adapters for different versions of the HTTP protocol, HTTP/1.x and HTTP/2
Getting Started
Send complete response after receiving complete request.
defmodule EchoServer do
use Raxx.Server
def handle_request(%Raxx.Request{method: :POST, path: [], body: body}, _state) do
response(:ok)
|> set_header("content-type", "text/plain")
|> set_body(body)
end
end
Send complete response as soon as request headers are received.
defmodule SimpleServer do
use Raxx.Server
def handle_head(%Raxx.Request{method: :GET, path: []}, _state) do
response(:ok)
|> set_header("content-type", "text/plain")
|> set_body("Hello, World!")
end
end
Store data as it is available from a clients request
defmodule StreamingRequest do
use Raxx.Server
def handle_head(%Raxx.Request{method: :PUT, body: true}, _state) do
{:ok, io_device} = File.open("my/path")
{[], {:file, device}}
end
def handle_body(body, state = {:file, device}) do
IO.write(device, body)
{[], state}
end
def handle_tail(_trailers, state) do
response(:see_other)
|> set_header("location", "/")
end
end
Subscribe server to event source and forward notifications to client.
defmodule SubscribeToMessages do
use Raxx.Server
def handle_head(_request, _state) do
{:ok, _} = ChatRoom.join()
response(:ok)
|> set_header("content-type", "text/event-stream")
|> set_body(true)
end
def handle_info({ChatRoom, data}, state) do
{[body(data)], state}
end
end
Notes
handle_head/2
will always be called with a request that has body as a boolean. For small requests where buffering the whole request is acceptable a simple middleware can be used.- Acceptable return values are the same for all callbacks;
either a
Raxx.Response
, which must be complete or a list of message parts and a new state.
Streaming
Raxx.Server
defines an interface to stream the body of request and responses.
This has several advantages:
- Large payloads do not need to be help in memory
- Server can push information as it becomes available, using Server Sent Events.
- If a request has invalid headers then a reply can be set without handling the body.
- Content can be generated as requested using HTTP/2 flow control
The body of a Raxx message (Raxx.Request or Raxx.Response
) may be one of three types:
io_list
- This is the complete body for the message.:false
- There is no body, for example:GET
requests never have a body.:true
- There is a body, it can be processed as it is received
Server Isolation
To start an exchange a client sends a request. The server, upon receiving this message, sends a reply. A logical HTTP exchange consists of a single request and response.
Methods such as pipelining and multiplexing combine multiple logical exchanges onto a single connection. This is done to improve performance and is a detail not exposed a server.
A Raxx server handles a single HTTP exchange. Therefore a single connection my have multiple servers each isolated in their own process.
Termination
An exchange can be stopped early by terminating the server process. Support for early termination is not consistent between versions of HTTP.
- HTTP/2: server exit with reason
:normal
, stream reset with errorCANCEL
. - HTTP/2: server exit any other reason, stream reset with error
INTERNAL_ERROR
. - HTTP/1.x: server exit with any reason, connection is closed.
Raxx.Server
does not provide a terminate callback.
Any cleanup that needs to be done from an aborted exchange should be handled by monitoring the server process.
Link to this section Summary
Types
Possible return values instructing server to send client data and update state if appropriate
Set of all components that make up a message to or from server
alias for Raxx.Request type
State of application server
Callbacks
Called every time data from the request body is received
Called once when a client starts a stream,
Called for all other messages the server may recieve
Called with a complete request once all the data parts of a body are received
Called once when a request finishes
Link to this section Types
Possible return values instructing server to send client data and update state if appropriate.
part() :: Raxx.Request.t() | Raxx.Response.t() | Raxx.Data.t() | Raxx.Tail.t()
Set of all components that make up a message to or from server.
alias for Raxx.Request type.
State of application server.
Original value is the configuration given when starting the raxx application.
Link to this section Callbacks
Called every time data from the request body is received
Called once when a client starts a stream,
Passed a Raxx.Request
and server configuration.
Note the value of the request body will be a boolean.
This callback can be relied upon to execute before any other callbacks
Called for all other messages the server may recieve
Called with a complete request once all the data parts of a body are received.
Passed a Raxx.Request
and server configuration.
Note the value of the request body will be a string.
This callback will never be called if handle_head/handle_body/handle_tail are overwritten.
Called once when a request finishes.
This will be called with an empty list of headers is request is completed without trailers.