use ahash::AHasher; use base64::{engine::general_purpose::URL_SAFE_NO_PAD, Engine as _}; use jsonwebtoken::{decode, Algorithm, DecodingKey, Validation}; use moka::sync::Cache; use moka::Expiry; use serde::{Deserialize, Serialize}; use std::hash::{Hash, Hasher}; use std::sync::LazyLock; use std::time::{Duration, Instant, SystemTime}; #[derive(Debug, Serialize, Deserialize)] pub struct Claims { pub master_key: String, pub owner: String, pub exp: u64, pub random: Option, } #[derive(Debug, Deserialize)] struct Expired { exp: Option, } static JWT_VALIDATION: LazyLock = LazyLock::new(|| Validation::new(Algorithm::HS256)); static JWT_CACHE: LazyLock> = LazyLock::new(|| Cache::builder().max_capacity(100_000).expire_after(JwtExpiry).build()); struct JwtExpiry; impl Expiry for JwtExpiry { fn expire_after_create(&self, _key: &u64, value: &u64, _current_time: Instant) -> Option { let now = SystemTime::now().duration_since(std::time::UNIX_EPOCH).unwrap_or_default().as_secs(); if *value > now { Some(Duration::from_secs(value - now)) } else { Some(Duration::ZERO) } } } pub fn check_jwt(token: &str, secret: &str) -> bool { let key = hash_token(token, secret); let now = SystemTime::now().duration_since(std::time::UNIX_EPOCH).unwrap_or_default().as_secs(); if let Some(exp) = JWT_CACHE.get(&key) { if exp < now { return false; } return true; } match is_expired(token, now) { Ok(true) => return false, Ok(false) => {} Err(_) => return false, } match decode::(token, &DecodingKey::from_secret(secret.as_ref()), &JWT_VALIDATION) { Ok(data) => { let now = SystemTime::now().duration_since(std::time::UNIX_EPOCH).unwrap_or_default().as_secs(); if data.claims.exp > now { JWT_CACHE.insert(key, data.claims.exp); true } else { false } } Err(_) => false, } } fn is_expired(token: &str, now: u64) -> Result> { let parts: Vec<&str> = token.split('.').collect(); if parts.len() != 3 { return Err("Invalid JWT format".into()); } let decoded = URL_SAFE_NO_PAD.decode(parts[1])?; let claims: Expired = serde_json::from_slice(&decoded)?; if let Some(exp) = claims.exp { Ok(exp < now) } else { Ok(true) } } fn hash_token(token: &str, secret: &str) -> u64 { let mut hasher = AHasher::default(); token.hash(&mut hasher); secret.hash(&mut hasher); hasher.finish() }