2020-11-24 18:58:18 +00:00
|
|
|
use std::time::Duration;
|
|
|
|
|
2020-11-21 22:20:32 +00:00
|
|
|
use anyhow::*;
|
|
|
|
use log::LevelFilter;
|
2020-12-01 07:31:08 +00:00
|
|
|
use kochab::{Server, Request, Response, Document};
|
2020-11-21 22:20:32 +00:00
|
|
|
|
|
|
|
#[tokio::main]
|
2020-12-16 16:38:29 +00:00
|
|
|
/// An ultra-simple ratelimiting example
|
|
|
|
///
|
|
|
|
/// We set up two pages:
|
|
|
|
/// * `/*` which can be accessed as much as the user wants
|
|
|
|
/// * /limit/*` which can only be accessed twice every 10 seconds
|
|
|
|
///
|
|
|
|
/// Once we tell it what needs to be ratelimited, kochab will automatically handle keeping
|
|
|
|
/// track of what users have and have not visited that page and how often. A very small
|
|
|
|
/// concurrent background task is in charge of cleaning the in-memory database every so
|
|
|
|
/// often so that a memory leak doesn't form.
|
2020-11-21 22:20:32 +00:00
|
|
|
async fn main() -> Result<()> {
|
2020-12-16 16:38:29 +00:00
|
|
|
// We set up logging so we can see what's happening. This isn't technically required,
|
|
|
|
// and you can use a simpler solution (like env_logger::init()) during production
|
2020-11-21 22:20:32 +00:00
|
|
|
env_logger::builder()
|
2020-11-25 05:42:09 +00:00
|
|
|
.filter_module("kochab", LevelFilter::Debug)
|
2020-11-21 22:20:32 +00:00
|
|
|
.init();
|
|
|
|
|
2020-12-16 16:38:29 +00:00
|
|
|
Server::new() // Create a server
|
|
|
|
.add_route("/", handle_request) // Create a page, content doesn't matter
|
|
|
|
.ratelimit("/limit", 2, Duration::from_secs(10)) // Set the ratelimit to 2 / 10s
|
|
|
|
.serve_ip("localhost:1965") // Start the server
|
2020-11-21 22:20:32 +00:00
|
|
|
.await
|
|
|
|
}
|
|
|
|
|
2020-12-16 16:38:29 +00:00
|
|
|
/// Render a simple page based on the current URL
|
|
|
|
///
|
|
|
|
/// The actual content of the response, and really anything in this section, doesn't
|
|
|
|
/// actually affect the ratelimit, but it's nice to have a usable demo, so we set up a
|
|
|
|
/// couple nice pages
|
2020-12-01 22:13:54 +00:00
|
|
|
async fn handle_request(request: Request) -> Result<Response> {
|
|
|
|
let mut document = Document::new();
|
2020-11-21 22:20:32 +00:00
|
|
|
|
2020-12-01 22:13:54 +00:00
|
|
|
if let Some("limit") = request.trailing_segments().get(0).map(String::as_str) {
|
|
|
|
document.add_text("You're on a rate limited page!")
|
2020-12-16 15:17:27 +00:00
|
|
|
.add_text("You can only access this page twice every 10 seconds");
|
2020-12-01 22:13:54 +00:00
|
|
|
} else {
|
|
|
|
document.add_text("You're on a normal page!")
|
|
|
|
.add_text("You can access this page as much as you like.");
|
2020-11-21 22:20:32 +00:00
|
|
|
}
|
2020-12-01 22:13:54 +00:00
|
|
|
document.add_blank_line()
|
|
|
|
.add_link("/limit", "Go to rate limited page")
|
|
|
|
.add_link("/", "Go to a page that's not rate limited");
|
|
|
|
Ok(document.into())
|
2020-11-21 22:20:32 +00:00
|
|
|
}
|