Initial work

This commit is contained in:
Joshua Coles 2024-07-15 13:12:43 +01:00
commit d8206cd99b
4 changed files with 1944 additions and 0 deletions

2
.gitignore vendored Normal file
View File

@ -0,0 +1,2 @@
/target
/.idea

1809
Cargo.lock generated Normal file

File diff suppressed because it is too large Load Diff

13
Cargo.toml Normal file
View File

@ -0,0 +1,13 @@
[package]
name = "toggl-2"
version = "0.1.0"
edition = "2021"
[dependencies]
axum = "0.7.5"
chrono = "0.4.38"
governor = "0.6.3"
reqwest = "0.12.5"
reqwest-ratelimit = "0.2.0"
thiserror = "1.0.62"
tokio = { version = "1.38.0", features = ["full"] }

120
src/main.rs Normal file
View File

@ -0,0 +1,120 @@
use std::cmp::max;
use std::num::NonZero;
use chrono::Utc;
use governor::clock::DefaultClock;
use governor::state::{InMemoryState, NotKeyed};
use reqwest::{Client, Error, Method, Request, RequestBuilder, Response, StatusCode};
use tokio::sync::oneshot;
#[derive(Debug, thiserror::Error)]
enum TogglApiError {
#[error("Reqwest error: {0}")]
ReqwestError(#[from] reqwest::Error),
#[error("Retries exceeded")]
RetriesExceeded,
}
struct TogglApiRequest(
Request,
oneshot::Sender<Result<Response, TogglApiError>>,
);
struct TogglApiWorker {
rate_limiter: governor::RateLimiter<NotKeyed, InMemoryState, DefaultClock>,
rx: tokio::sync::mpsc::Receiver<TogglApiRequest>,
client: Client,
max_retries: Option<u32>,
default_delay: f64,
}
impl TogglApiWorker {
fn new() -> (Self, TogglApi) {
let (tx, rx) = tokio::sync::mpsc::channel(100);
let client = Client::new();
let rate_limiter = governor::RateLimiter::direct(
governor::Quota::per_second(NonZero::new(1u32).unwrap())
);
(Self { rate_limiter, client, rx, max_retries: Some(3), default_delay: 3.14 }, TogglApi { tx })
}
pub async fn start(&mut self) {
loop {
// We limit ourselves to the recommended rate of 1 req/s
self.rate_limiter.until_ready().await;
let TogglApiRequest(request, tx) = self.rx.recv().await.unwrap();
let response = self.make_request(request).await;
tx.send(response).unwrap();
}
}
async fn make_request(&self, request: RequestBuilder) -> Result<Response, TogglApiError> {
let max_retries = self.max_retries.unwrap_or(1);
for _ in 0..max_retries {
let response = self.client.execute(request.clone()).await?;
if response.status().is_server_error() || response.status() == StatusCode::TOO_MANY_REQUESTS {
let delay = self.parse_retry_after_header(&response)
.unwrap_or(self.default_delay);
tokio::time::sleep(tokio::time::Duration::from_secs_f64(delay)).await;
} else {
return Ok(response);
}
}
Err(TogglApiError::RetriesExceeded)
}
fn parse_retry_after_header(&self, response: &Response) -> Option<f64> {
match response.headers().get("Retry-After") {
Some(retry_after) => {
let retry_after = retry_after.to_str()
.unwrap();
let a = retry_after.parse::<f64>()
.ok()
.or_else(|_| {
let date = chrono::NaiveDateTime::parse_from_str(
retry_after,
"%a, %d %b %Y %H:%M:%S GMT",
);
date.map(|date| date.and_utc())
.map(|date| Utc::now().signed_duration_since(date))
.map(|time_delta| time_delta.num_seconds() as f64)
.ok()
});
a
}
None => Some(self.default_delay),
}
}
}
struct TogglApi {
tx: tokio::sync::mpsc::Sender<TogglApiRequest>,
}
impl TogglApi {
pub async fn request(&self, request: Request) -> Result<Response, TogglApiError> {
let (tx, rx) = oneshot::channel();
self.tx.send(TogglApiRequest(request, tx)).await.expect("send request");
rx.await.unwrap()
}
}
#[tokio::main]
async fn main() {
let (mut worker, api_client) = TogglApiWorker::new();
tokio::spawn(async move { worker.start().await; });
dbg!(api_client.request(Request::new(Method::GET, "https://www.google.com".parse().unwrap())).await.unwrap());
}