Note: The stable
branch contains some bugfixes against the last release. The master
branch currently tracks gotham's master branch and the next release will use gotham 0.5.0 and be
compatible with the new future / async stuff.
This crate is an extension to the popular gotham web framework for Rust. It allows you to create resources with assigned methods that aim to be a more convenient way of creating handlers for requests.
This is an opinionated framework on top of [gotham]. Unless your web server handles mostly JSON as request/response bodies and does that in a RESTful way, this framework is probably a bad fit for your application. The ultimate goal of gotham-restful is to provide a way to write a RESTful web server in Rust as convenient as possible with the least amount of boilerplate neccessary.
Assuming you assign /foobar
to your resource, you can implement the following methods:
| Method Name | Required Arguments | HTTP Verb | HTTP Path | | ----------- | ------------------ | --------- | ----------- | | readall | | GET | /foobar | | read | id | GET | /foobar/:id | | search | query | GET | /foobar/search | | create | body | POST | /foobar | | changeall | body | PUT | /foobar | | change | id, body | PUT | /foobar/:id | | remove_all | | DELETE | /foobar | | remove | id | DELETE | /foobar/:id |
Each of those methods has a macro that creates the neccessary boilerplate for the Resource. A simple example could look like this:
```rust /// Our RESTful resource.
struct FooResource;
/// The return type of the foo read method.
struct Foo { id: u64 }
/// The foo read method handler.
fn read(id: u64) -> Success
Some methods require arguments. Those should be
* id Should be a deserializable json-primitive like i64
or String
.
* body Should be any deserializable object, or any type implementing [RequestBody
].
* query Should be any deserializable object whose variables are json-primitives. It will
however not be parsed from json, but from HTTP GET parameters like in search?id=1
. The
type needs to implement [QueryStringExtractor
].
Additionally, non-async handlers may take a reference to gotham's [State
]. If you need to
have an async handler (that is, the function that the method macro is invoked on is declared
as async fn
), consider returning the boxed future instead. Since [State
] does not implement
Sync
there is unfortunately no more convenient way.
By default, every request body is parsed from json, and every respone is converted to json using [serde_json]. However, you may also use raw bodies. This is an example where the request body is simply returned as the response again, no json parsing involved:
```rust
struct ImageResource;
struct RawImage {
content: Vec
fn create(body : RawImage) -> Raw
To make life easier for common use-cases, this create offers a few features that might be helpful
when you implement your web server. The complete feature list is
- auth
Advanced JWT middleware
- chrono
openapi support for chrono types
- cors
CORS handling for all method handlers
- database
diesel middleware support
- errorlog
log errors returned from method handlers
- openapi
router additions to generate an openapi spec
- uuid
openapi support for uuid
In order to enable authentication support, enable the auth
feature gate. This allows you to
register a middleware that can automatically check for the existence of an JWT authentication
token. Besides being supported by the method macros, it supports to lookup the required JWT secret
with the JWT data, hence you can use several JWT secrets and decide on the fly which secret to use.
None of this is currently supported by gotham's own JWT middleware.
A simple example that uses only a single secret could look like this:
```rust
struct SecretResource;
struct Secret { id: u64, intended_for: String }
struct AuthData { sub: String, exp: u64 }
fn read(auth: AuthStatus
fn main() {
let auth: AuthMiddleware
The cors feature allows an easy usage of this web server from other origins. By default, only
the Access-Control-Allow-Methods
header is touched. To change the behaviour, add your desired
configuration as a middleware.
A simple example that allows authentication from every origin (note that *
always disallows
authentication), and every content type, could look like this:
```rust
struct FooResource;
fn read_all() { // your handler }
fn main() {
let cors = CorsConfig {
origin: Origin::Copy,
headers: vec![CONTENTTYPE],
maxage: 0,
credentials: true
};
let (chain, pipelines) = singlepipeline(newpipeline().add(cors).build());
gotham::start("127.0.0.1:8080", build_router(chain, pipelines, |route| {
route.resource::
The cors feature can also be used for non-resource handlers. Take a look at [CorsRoute
]
for an example.
The database feature allows an easy integration of [diesel] into your handler functions. Please
note however that due to the way gotham's diesel middleware implementation, it is not possible
to run async code while holding a database connection. If you need to combine async and database,
you'll need to borrow the connection from the [State
] yourself and return a boxed future.
A simple non-async example could look like this:
```rust
struct FooResource;
struct Foo { id: i64, value: String }
fn read_all(conn: &PgConnection) -> QueryResult
type Repo = gothammiddlewarediesel::Repo
fn main() { let repo = Repo::new(&env::var("DATABASE_URL").unwrap()); let diesel = DieselMiddleware::new(repo);
let (chain, pipelines) = single_pipeline(new_pipeline().add(diesel).build());
gotham::start("127.0.0.1:8080", build_router(chain, pipelines, |route| {
route.resource::<FooResource>("foo");
}));
} ```
The OpenAPI feature is probably the most powerful one of this crate. Definitely read this section carefully both as a binary as well as a library author to avoid unwanted suprises.
In order to automatically create an openapi specification, gotham-restful needs knowledge over
all routes and the types returned. serde
does a great job at serialization but doesn't give
enough type information, so all types used in the router need to implement OpenapiType
. This
can be derived for almoust any type and there should be no need to implement it manually. A simple
example could look like this:
```rust
struct FooResource;
struct Foo { bar: String }
fn readall() -> Success
fn main() {
gotham::start("127.0.0.1:8080", buildsimplerouter(|route| {
let info = OpenapiInfo {
title: "My Foo API".toowned(),
version: "0.1.0".toowned(),
urls: vec!["https://example.org/foo/api/v1".toowned()]
};
route.withopenapi(info, |mut route| {
route.resource::
Above example adds the resource as before, but adds another endpoint that we specified as /openapi
that will return the generated openapi specification. This allows you to easily write clients
in different languages without worying to exactly replicate your api in each of those languages.
However, as of right now there is one caveat. If you wrote code before enabling the openapi feature,
it is likely to break. This is because of the new requirement of OpenapiType
for all types used
with resources, even outside of the with_openapi
scope. This issue will eventually be resolved.
If you are writing a library that uses gotham-restful, make sure that you expose an openapi feature.
In other words, put
toml
[features]
openapi = ["gotham-restful/openapi"]
into your libraries Cargo.toml
and use the following for all types used with handlers:
```rust
struct Foo; ```
There is a lack of good examples, but there is currently a collection of code in the [example] directory, that might help you. Any help writing more examples is highly appreciated.
Licensed under your option of: - Apache License Version 2.0 - Eclipse Public License Version 2.0