10 releases

0.3.1 Jun 3, 2023
0.3.0 Apr 13, 2021
0.2.4 Apr 5, 2021
0.2.3 Feb 5, 2021
0.1.0 Aug 17, 2020

#628 in HTTP server

MIT license

80KB
2.5K SLoC

Eve-rs

Eve-rs is a middleware-based rust http framework which is inspired from ExpressJS framework.

Example

Let's create a demo application using eve-rs, using an intuative approach to get a better understanding of the framework.

Create a cargo project, here we have named it demo, and in the Cargo.toml file, add the following dependencies under the depedency section.

[dependencies]
eve-rs = '0.2.0'
log = '0.4.8'
tokio = {version = '0.2.22', features = ['full']}

after updating toml file, we then enter the following code in the main.rs file

Import eve-rs crate

First step is to bring eve_rs to the scope of the project. So we add the following line

use eve_rs::{App as DemoApp, Context, DefaultContext, NextHandler, Error, listen, Request, DefaultMiddleware};

Middleware

Just like in express framework, a middleware is basically a function that has access to request and response objects. In eve-rs, we use the same terms, and use middlewares for routing purposes.

For our demo application, we will create a single middleware called plaintext, which will print out a plain text on a get request.

A middleware takes in two parameters, Context and NextHandler.

Context gives you APIs which gives you more information related to the request. Such as getting request body, request status, content type and much more. Since it is a trait (Similar to Interfaces in other programming languages), we get the freedom to define our own context for the application, and on the other hand, the user can use DefaultContext provide by eve.

NextHandler is the next middleware in the sequence. That is, if the current middleware called finishes processing and does not throw an error, and next middleware is passed, then after executing the current middleware, the next middleware is called.

In our example code, we do not have a next middleware to be executed, so we give _next as the parameter.

async fn plaintext(
	mut context: DefaultContext,
	_next: NextHandler<DefaultContext>
) -> Result<DefaultContext, Error<DefaultContext>> {

	let val = "Hello, World!";
	context.body(val);
	
	/**
	Use this if your code uses a next handler.
	let response = next(context).await;
	return response;
	*/
	Ok(context)
}

Create App

Our next step is to create an eve app. The App struct in eve-rs gives us a create() function, to create an App. The function takes in two parameters; context_generator and state.

context_generator is a function that is responsible to create a context for our middlewares. It takes in two parameters Request and state. State, here could be any configuration we need our app to have. Let's assume that we need our app to have some state, so we will pass the state in the following way:

pub struct State {
	pub database_name : String;
}

fn context_generator(request : Request, state : &State) -> YourAppContext {
	let state = state.clone();
	YourAppContext::new(request, state)
}

Since, in our example we are using DefaultContext, we can create a context without passing in the state.

fn default_context_generator(request: Request, _ : &()) -> DefaultContext {     
	// default context has no state as an argument.
	DefaultContext::new(request)
}

Once we have a context generator, we can go ahead and create an app.

pub fn create_app() -> DemoApp<DefaultContext, DefaultMiddleware<()>, ()>  {
	DemoApp::<DefaultContext, DefaultMiddleware<()>, ()>::create(default_context_generator, ())
}

in the above code, the second parameter is the state that's stored across the DemoApp and since we do not have a state, we pass in () to the create function.

Once our app is created, we add the middlewares in the scope of the app by using app.use_middleware() function.

app.use_middleware("/plaintext", &[DefaultMiddleware::new(|context, next| {
	Box::pin(async move {
		plaintext(context, next).await 
	})
})])

Below is how our code will look by combining all the above explained methods.


// Bring eve-rs modules to the program scope.
use eve_rs::{App as DemoApp, Context, DefaultContext, NextHandler, Error, listen, Request, DefaultMiddleware};

// middleware
async fn plaintext(
	mut context: DefaultContext,
	 _: NextHandler<DefaultContext>
) -> Result<DefaultContext, Error<DefaultContext>> {

	let val = "Hello, World!";
	context.body(val);
	
	Ok(context)
}

// function to create an eve-rs app.
pub fn create_app() -> DemoApp<DefaultContext, DefaultMiddleware<()>, ()>  {
	DemoApp::<DefaultContext, DefaultMiddleware<()>, ()>::create(default_context_generator, ())
}


// context generator.
fn default_context_generator(request: Request, _: &()) -> DefaultContext {     
	// default context has no state as an argument.
	DefaultContext::new(request)
}


#[async_std::main]
async fn main() {
	println!("Starting server...");

	// call function to create default app.
	let mut app = create_app();
	
	// add middleware to the stack.
	// Can also be one of:
	// get
	// post
	// put
	// delete
	// head
	// options
	// connect
	// patch
	// trace
	app.use_middleware("/plaintext", &[DefaultMiddleware::new(|context, next| {
		Box::pin(async move {
			plaintext(context, next).await 
		})
	})]);
	

	// assign port number.
	let port = 8080;

	log::info!("Listening for connections on 127.0.0.1:{}", port);
	listen(app, ([127, 0, 0, 1], port), None).await;
}

Build the project using cargo build command.
Run the project using cargo run command.

In the above example we have used DefaultContext as the Context . This means that eve-rs gives the freedom to implement the Context in your own way. For the sake of simplicity we have used DefaultContext.

Same goes with DefaultMiddleware. Here we have used a default implementation. Feel free to implement your own.

Credits

This project is heavily inspired by Thruster. This framework is very similar to Thruster, save for a few small design decisions. While this framework is currently in use in production at our company, if you're looking for something that's a little more mature, you should definitely pick Thruster.

Huge thanks to @trezm for helping me out through the development of this.

Dependencies

~13–29MB
~443K SLoC