<div align="center">
<h1>ohkami</h1>
</div>
ohkami *- [狼] means wolf in Japanese -* is **simple** and **macro free** web framework for Rust.
<br/>
## Features
- *simple*: Less things to learn / Less code to write / Less time to hesitate.
- *macro free*: No need for using macros.
- async handlers
- easy error handling
<br/>
## Quick start
1. Add dependencies:
```toml
[dependencies]
ohkami = "0.3"
```
2. Write your first code with ohkami:
```rust
use ohkami::prelude::*;
fn main() -> Result<()> {
Server::setup()
.GET("/", || async {
Response::OK("Hello, world!")
})
.serve_on(":3000")
}
```
3. If you're interested in ohkami, learn more by [examples](https://github.com/kana-rus/ohkami/tree/main/examples) and [documentation](https://docs.rs/ohkami/latest/ohkami/)!
<br/>
## Snippets
### handle query params
```rust
let name = ctx.query::<&str>("name")?;
// if presumable, `::<&str>` isn't needed
```
```rust
let count = ctx.query::<usize>("count")?;
// if presumable, `::<usize>` isn't needed
```
### handle request body
```rust
let body = ctx.body::<D>()?;
// if presumable, `::<D>` isn't needed
// `D` has to be `serde::Deserialize`
```
### handle path params
```rust
fn main() -> Result<()> {
Server::setup()
.GET("/sleepy/:time/:name", sleepy_hello)
.serve_on(":3000")
}
async fn sleepy_hello(time: u64, name: String) -> Result<Response> {
(time < 30)
._else(|| Response::BadRequest("sleeping time (sec) must be less than 30."))?;
std::thread::sleep(
std::time::Duration::from_secs(time)
);
Response::OK(format!("Hello {name}, I'm extremely sleepy..."))
}
```
### return OK response with `text/plain`
```rust
Response::OK("Hello, world!")
```
### return OK response with `application/json`
```rust
Response::OK(JSON("Hello, world!"))
```
```rust
Response::OK(json!("ok": true))
```
```rust
Response::OK(json(user)?)
// serialize Rust value into JSON
// value's type has to be `serde::Serialize`
```
### handle errors
```rust
let user = ctx.body::<User>()?;
// or, you can add an error context message:
let user = ctx.body::<User>()
._else(|e| e.error_context("failed to get user data"))?;
// or discard original error:
let user = ctx.body::<User>()
._else(|_| Response::InternalServerError("can't get user"))?;
// or
._else(|_| Response::InternalServerError(None))?;
```
### handle Option values
```rust
fn server() -> Server {
Server::setup()
.GET("/", || async {Response::OK("Hello!")})
}
fn main() -> Result<()> {
server().serve_on(":3000")
}
```
2. write tests using `assert_to_res` , `assert_not_to_res`:
```rust
#[cfg(test)]
mod test {
use ohkami::{server::Server, response::Response, test_system::{Test, Request, Method}};
use once_cell::sync::Lazy;
static SERVER: Lazy<Server> = Lazy::new(|| super::server());
#[test]
fn test_hello() {
let request = Request::new(Method::GET, "/");
(*SERVER).assert_to_res(&request, Response::OK("Hello!"));
(*SERVER).assert_not_to_res(&request, Response::BadRequest(None));
}
}
```
<br/>
## Development
ohkami is on early stage now and not for producntion use.\
Please give me your feedback! → [GetHub issue](https://github.com/kana-rus/ohkami/issues)
<br/>
## License
This project is under MIT LICENSE ([LICENSE-MIT](https://github.com/kana-rus/ohkami/blob/main/LICENSE-MIT) or [https://opensource.org/licenses/MIT](https://opensource.org/licenses/MIT)).