Feather is a lightweight, DX-first web framework for Rust. Inspired by the simplicity of Express.js, but designed for Rust’s performance and safety.
-
Middleware-First Architecture
Everything is a middleware—even if it's not, it produces a middleware in the end.
The newmiddleware!
macro makes writing route and middleware closures concise and ergonomic. -
Easy State Management Using Context
The Context API makes it very easy to manage state without the use of Extractors/Macros. -
All in One
Feather is a complete web framework that includes routing, middleware, logging, JWT authentication, and more, all in one package. -
Feel of Async Without Async
Feather is multithreaded by default, running on Feather-Runtime.
Every request is given a thread from the server's threadpool and that thread is responsible for returning the response to that request.
So you can run long running tasks on another thread in the middlewares, but the response can only be returned from the middleware the request is accepted on.
If you want to go deeper take a look at Feather-Runtime
Add Feather to your Cargo.toml
:
[dependencies]
feather = "~0.4"
use feather::middlewares::builtins;
use feather::{App, next, middleware};
fn main() {
let mut app = App::new();
app.get("/", middleware!(|_req, res, _ctx| {
res.send_text("Hello, world!");
next!()
}));
app.use_middleware(builtins::Logger);
app.listen("127.0.0.1:5050");
}
That’s all — no async.
Middleware is the heart of Feather. You may write it as a closure (using the middleware!
macro), a struct, or chain them together:
use feather::{App, next, middleware};
fn main() {
let mut app = App::new();
app.use_middleware(middleware!(|_req, _res, _ctx| {
println!("Custom global middleware!");
next!()
}));
app.get("/", middleware!(|_req, res, _ctx| {
res.send_text("Hello, world!");
next!()
}));
app.listen("127.0.0.1:5050");
}
Or as a struct:
use feather::{middlewares::Middleware, next};
struct Custom;
impl Middleware for Custom {
fn handle(&self, req: &mut feather::Request, res: &mut feather::Response, ctx: &mut feather::AppContext) -> feather::Outcome {
println!("Custom struct middleware!");
next!()
}
}
Feather's Context API allows you to manage application-wide state without extractors or macros.
use feather::{next, App, middleware};
#[derive(Debug)]
struct Counter { pub count: i32 }
fn main() {
let mut app = App::new();
app.context().set_state(Counter { count: 0 });
app.get("/", middleware!(|_req, res, ctx| {
let counter = ctx.get_mut_state::<Counter>().unwrap();
counter.count += 1;
res.send_text(format!("Counted! {}", counter.count));
next!()
}));
app.listen("127.0.0.1:5050");
}
Context is especially useful when needing to access databases and files.
Feather has a native JWT module activated using a cargo feature jwt
:
[dependencies]
feather = { version = "*", features = ["jwt"] }
use feather::jwt::{generate_jwt, with_jwt_auth};
use feather::{App, next};
fn main() {
let mut app = App::new();
app.get("/auth", with_jwt_auth("secretcode", |_req, res, _ctx, claim| {
println!("Claim: {:?}", claim);
res.send_text("Hello, JWT!");
next!()
}));
app.listen("127.0.0.1:8080")
}
- Be the simplest Rust web framework to get started with
- Be modular and easy to extend
- Focus on DX without sacrificing Rust's safety and performance
PRs are welcome!
If you have ideas or bugs, please open an issue or submit a pull request.
# Getting started with dev
git clone https://github.com/BersisSe/feather.git
cd feather
cargo run --example app
Feather is MIT licensed. See LICENSE.
Feather is inspired by Express.js and exists to bring that same productivity to Rust.
Huge thanks to the Rust community for their support and contributions!
Special thanks to the contributors who have helped make Feather better!
If you like Feather:
- ⭐ Star it on GitHub,
- Share it on Reddit, HN, or Discord
- Build something and show up!