Skip to content

Commit

Permalink
Improve the fairings guide.
Browse files Browse the repository at this point in the history
  • Loading branch information
SergioBenitez committed Jul 11, 2017
1 parent f34bb42 commit da08f1e
Show file tree
Hide file tree
Showing 3 changed files with 114 additions and 99 deletions.
2 changes: 1 addition & 1 deletion site/guide/configuration.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# Configuration

Rocket aims to have flexible and usable configuration system. Rocket
Rocket aims to have a flexible and usable configuration system. Rocket
applications can be configured via a configuration file, through environment
variables, or both. Configurations are separated into three environments:
development, staging, and production. The working environment is selected via an
Expand Down
209 changes: 111 additions & 98 deletions site/guide/fairings.md
Original file line number Diff line number Diff line change
@@ -1,24 +1,52 @@
# Fairings

Fairings are Rocket's approach to structured middleware. They allow for
interposition at various points in the application and request/response
lifecycle through callbacks issued by Rocket.
Fairings are Rocket's approach to structured middleware. With fairings, your
application can hook into the request lifecycle to record or rewrite information
about incoming requests and outgoing responses.

## Overview

A _fairing_ is any type that implements the [`Fairing`] trait. The `Fairing`
trait is composed of methods that represent callbacks that Rocket will run at
requested points in a program. Through these methods, fairings can rewrite or
record information about requests and responses as well as perform actions when
a Rocket application launches.
A _fairing_ is a special name for a type that implements the [`Fairing`] trait.
Fairings receive callbacks from Rocket when certain events, like that of an
incoming request, occur. Rocket passes information about the event to the
fairing, and the fairing can do what it wants with the information. This
includes rewriting data when applicable, recording information about the event
or data, or doing nothing at all.

Fairings are a lot like middleware in other frameworks with a few key
distinctions:

* Fairings **cannot** terminate or respond to an incoming request directly.
* Fairings **cannot** inject arbitrary, non-request data into a request.
* Fairings _can_ prevent an application from launching.
* Fairings _can_ inspect and modify the application's configuration.

If you are familiar with middleware from other frameworks, you may find yourself
reaching for fairings instinctively. Before doing so, consider whether Rocket
provides a better solution to your problem: While middleware may be the best
solution to a problem in another framework, it is often a suboptimal solution in
Rocket. Rocket provides richer mechanisms such as [request guards] and [data
guards] that can be used to solve problems in a cleaner, more composable, and
more robust manner.

As a general rule of thumb, only _globally applicable_ actions should be
effected through fairings. You should _not_ use a fairing to implement
authentication or authorization (preferring to use a [request guard] instead)
_unless_ the authentication or authorization applies to the entire application.
On the other hand, you _should_ use a fairing to record timing and/or usage
statistics or global security policies.

[`Fairing`]: https://api.rocket.rs/rocket/fairing/trait.Fairing.html
[request guard]: /guide/requests/#request-guards
[request guards]: /guide/requests/#request-guards
[data guards]: /guide/requests/#body-data

### Attaching

For a fairing to be active, it must first be _attached_ through the the
[`attach`] method on a [`Rocket`] instance. For instance, to attach fairings
named `req_fairing` and `res_fairing` to a new Rocket instance, you might write:
Fairings are registered with Rocket via the [`attach`] method on a [`Rocket`]
instance. Only when a fairing is attached will its callbacks fire. As an
example, the following snippet attached two fairings, `req_fairing` and
`res_fairing`, to a new Rocket instance:

```rust
rocket::ignite()
Expand All @@ -27,111 +55,61 @@ rocket::ignite()
.launch();
```

Once a fairing is attached, Rocket will execute its callbacks at the appropiate
time.

[`attach`]: https://api.rocket.rs/rocket/struct.Rocket.html#method.attach
[`Rocket`]: https://api.rocket.rs/rocket/struct.Rocket.html

Fairings are executed in the order in which they are attached: the first
attached fairing has its callbacks executed before all others. Because fairing
callbacks may not be commutative, the order in which fairings are attached may
be significant.

### Callbacks

A fairing can implement any combination of the following four callbacks:
There are four events for which Rocket issues fairing callbacks. Each of these
events is described below:

* **Attach**
* **Attach (`on_attach`)**

An attach callback is called when a fairing is first attached via the
[`attach`](https://api.rocket.rs/rocket/struct.Rocket.html#method.attach)
method. An attach callback can arbitrarily modify the `Rocket` instance
being constructed and optionally abort launch.
being constructed and optionally abort launch. Attach fairings are commonly
used to parse and validate configuration values, aborting on bad
configurations, and inserting the parsed value into managed state for later
retrieval.

* **Launch**
* **Launch (`on_launch`)**

A launch callback is called immediately before the Rocket application has
launched. A launch callback can inspect the `Rocket` instance being
launched.
launched. A launch callback can be a convenient hook for launching services
related to the Rocket application being launched.

* **Request**
* **Request (`on_request`)**

A request callback is called just after a request is received. A request
callback can modify the request at will and peek into the incoming data. It
may not, however, abort or respond directly to the request; these issues are
better handled via request guards or via response callbacks.

* **Response**
* **Response (`on_response`)**

A response callback is called when a response is ready to be sent to the
client. A response callback can modify the response at will. For example, a
response callback can provide a default response when the user fails to
handle the request by checking for 404 responses.


### Execution Order

Fairings are executed in the order in which they are attached: the first
attached fairing has its callbacks executed before all others. Because fairing
callbacks may not be commutative, the order in which fairings are attached may
be significant.

### Ad-Hoc Fairings

For simple occasions, implementing the `Fairing` trait can be cumbersome. This
is why Rocket provides the [`AdHoc`] type, which creates a fairing from a simple
function or clusure.

Using the `AdHoc` type is easy: simply call the `on_attach`, `on_launch`,
`on_request`, or `on_response` constructors to create an `AdHoc` structure from
a function or closure. Then, attach the structure to a `Rocket` instance. Rocket
takes care of the rest.

As an example, the code below creates a `Rocket` instance with two attached
ad-hoc fairings. The first, a launch fairing, simply prints a message indicating
that the application is about to the launch. The second, a request fairing,
changes the method of all requests to `PUT`.

```rust
use rocket::fairing::AdHoc;
use rocket::http::Method;

rocket::ignite()
.attach(AdHoc::on_launch(|_| {
println!("Rocket is about to launch! Exciting!");
}))
.attach(AdHoc::on_request(|req, _| {
req.set_method(Method::Put);
}));
```

[`AdHoc`]: https://api.rocket.rs/rocket/fairing/enum.AdHoc.html

## Considerations

Fairings are a large hammer that can easily be abused and misused. If you
are considering writing a `Fairing` implementation, first consider if it is
appropriate to do so. While middleware is often the best solution to some
problems in other frameworks, it is often a suboptimal solution in Rocket.
This is because Rocket provides richer mechanisms such as [request guards]
and [data guards] that can be used to accomplish the same objective in a
cleaner, more composable, and more robust manner.

As a general rule of thumb, only _globally applicable actions_ should be
implemented via fairings. For instance, you should _not_ use a fairing to
implement authentication or authorization (preferring to use a [request
guard] instead) _unless_ the authentication or authorization applies to the
entire application. On the other hand, you _should_ use a fairing to record
timing and/or usage statistics or to implement global security policies.

[request guard]: https://api.rocket.rs/rocket/request/trait.FromRequest.html
[request guards]: https://api.rocket.rs/rocket/request/trait.FromRequest.html
[data guards]: https://api.rocket.rs/rocket/data/trait.FromData.html
client. A response callback can modify part or all of the response. As such,
a response fairing can be used to provide a response when the greater
applications fails to by rewriting **404** responses as desired. As another
example, response fairings can also be used to inject headers into all
outgoing responses.

## Implementing

A fairing must implement the [`Fairing`] trait. A `Fairing` implementation has
one required method: [`info`], which returns an [`Info`] structure. This
structure is used by Rocket to assign a name to the `Fairing` and determine
which callbacks to actually issue on the `Fairing`. A `Fairing` can also
implement any of the available callbacks: [`on_attach`], [`on_launch`],
[`on_request`], and [`on_response`].
Recall that a fairing is any type that implements the [`Fairing`] trait. A
`Fairing` implementation has one required method: [`info`], which returns an
[`Info`] structure. This structure is used by Rocket to assign a name to the
fairing and determine the set of callbacks the fairing is registering for. A
`Fairing` can implement any of the available callbacks: [`on_attach`],
[`on_launch`], [`on_request`], and [`on_response`]. Each callback has a default
implementation that does absolutely nothing.

[`Info`]: https://api.rocket.rs/rocket/fairing/struct.Info.html
[`info`]: https://api.rocket.rs/rocket/fairing/trait.Fairing.html#tymethod.info
Expand All @@ -140,13 +118,13 @@ implement any of the available callbacks: [`on_attach`], [`on_launch`],
[`on_request`]: https://api.rocket.rs/rocket/fairing/trait.Fairing.html#method.on_request
[`on_response`]: https://api.rocket.rs/rocket/fairing/trait.Fairing.html#method.on_response

### Restrictions
### Requirements

A `Fairing` must be `Send + Sync + 'static`. This means that the fairing must be
sendable across thread boundaries (`Send`), thread-safe (`Sync`), and have only
`'static` references, if any (`'static`). Note that these bounds _do not_
prohibit a `Fairing` from holding state: the state need simply be thread-safe
and statically available or heap allocated.
A type implementing `Fairing` is required to be `Send + Sync + 'static`. This
means that the fairing must be sendable across thread boundaries (`Send`),
thread-safe (`Sync`), and have only static references, if any (`'static`). Note
that these bounds _do not_ prohibit a `Fairing` from holding state: the state
need simply be thread-safe and statically available or heap allocated.

### Example

Expand All @@ -156,8 +134,11 @@ state, it would require us to annotate every `GET` and `POST` request with
custom types, polluting handler signatures. Instead, we can create a simple
fairing that acts globally.

The `Counter` fairing below records the number of all `GET` and `POST` requests
received. It makes these counts available at a special `'/counts'` path.
The code for a `Counter` fairing below implements exactly this. The fairing
receives a request callback, where it increments a counter on each `GET` and
`POST` request. It also receives a response callback, where it responds to
unrouted requests to the `/counts` path by returning the recorded number of
counts.

```rust
struct Counter {
Expand All @@ -166,13 +147,15 @@ struct Counter {
}

impl Fairing for Counter {
// This is a request and respone fairing named "GET/POST Counter".
fn info(&self) -> Info {
Info {
name: "GET/POST Counter",
kind: Kind::Request | Kind::Response
}
}

// Increment the counter for `GET` and `POST` requests.
fn on_request(&self, request: &mut Request, _: &Data) {
match request.method() {
Method::Get => self.get.fetch_add(1, Ordering::Relaxed),
Expand All @@ -187,6 +170,7 @@ impl Fairing for Counter {
return
}

// Rewrite the response to return the current counts.
if request.method() == Method::Get && request.uri().path() == "/counts" {
let get_count = self.get.load(Ordering::Relaxed);
let post_count = self.post.load(Ordering::Relaxed);
Expand All @@ -203,3 +187,32 @@ impl Fairing for Counter {
For brevity, imports are not shown. The complete example can be found in the
[`Fairing`
documentation](https://api.rocket.rs/rocket/fairing/trait.Fairing.html#example).

## Ad-Hoc Fairings

For simple occasions, implementing the `Fairing` trait can be cumbersome. This
is why Rocket provides the [`AdHoc`] type, which creates a fairing from a simple
function or clusure. Using the `AdHoc` type is easy: simply call the
`on_attach`, `on_launch`, `on_request`, or `on_response` constructors on `AdHoc`
to create an `AdHoc` structure from a function or closure.

As an example, the code below creates a `Rocket` instance with two attached
ad-hoc fairings. The first, a launch fairing, simply prints a message indicating
that the application is about to the launch. The second, a request fairing,
changes the method of all requests to `PUT`.

```rust
use rocket::fairing::AdHoc;
use rocket::http::Method;

rocket::ignite()
.attach(AdHoc::on_launch(|_| {
println!("Rocket is about to launch! Exciting!");
}))
.attach(AdHoc::on_request(|req, _| {
req.set_method(Method::Put);
}));
```

[`AdHoc`]: https://api.rocket.rs/rocket/fairing/enum.AdHoc.html

2 changes: 2 additions & 0 deletions site/guide/requests.md
Original file line number Diff line number Diff line change
Expand Up @@ -488,6 +488,8 @@ looks like this, where `T: FromData`:
fn new(input: T) -> String { ... }
```

Any type that implements [`FromData`] is also known as _data guard_.

[`FromData`]: https://api.rocket.rs/rocket/data/trait.FromData.html

### Forms
Expand Down

0 comments on commit da08f1e

Please sign in to comment.