use jsonwebtoken::{Algorithm, Validation, decode, decode_header};
use systemprompt_models::auth::{JwtAudience, JwtClaims};
use crate::error::{AuthError, AuthResult};
use crate::keys::authority;
pub const JWT_LEEWAY_SECONDS: u64 = 30;
#[derive(Debug, Clone, Default)]
pub struct ValidationPolicy<'a> {
pub validate_exp: bool,
pub validate_nbf: bool,
pub leeway_seconds: u64,
pub issuer: Option<&'a str>,
pub audiences: &'a [JwtAudience],
}
impl<'a> ValidationPolicy<'a> {
#[must_use]
pub const fn session_context() -> Self {
Self {
validate_exp: true,
validate_nbf: true,
leeway_seconds: JWT_LEEWAY_SECONDS,
issuer: None,
audiences: &[],
}
}
#[must_use]
pub const fn issuer_scoped(issuer: &'a str, audiences: &'a [JwtAudience]) -> Self {
Self {
validate_exp: true,
validate_nbf: true,
leeway_seconds: JWT_LEEWAY_SECONDS,
issuer: Some(issuer),
audiences,
}
}
}
pub fn decode_rs256_claims(token: &str, policy: &ValidationPolicy<'_>) -> AuthResult<JwtClaims> {
let header = decode_header(token).map_err(AuthError::InvalidToken)?;
if header.alg != Algorithm::RS256 {
return Err(AuthError::UnsupportedAlgorithm {
got: format!("{:?}", header.alg),
});
}
let kid = header.kid.as_deref().ok_or(AuthError::MissingKid)?;
let key = authority::decoding_key_for_kid(kid)
.map_err(|e| AuthError::KeyLookup(e.to_string()))?
.ok_or_else(|| AuthError::UnknownKid(kid.to_owned()))?;
let mut validation = Validation::new(Algorithm::RS256);
validation.validate_exp = policy.validate_exp;
validation.validate_nbf = policy.validate_nbf;
validation.leeway = policy.leeway_seconds;
if let Some(issuer) = policy.issuer {
validation.set_issuer(&[issuer]);
}
if policy.audiences.is_empty() {
validation.validate_aud = false;
} else {
let audience_strs: Vec<&str> = policy.audiences.iter().map(JwtAudience::as_str).collect();
validation.set_audience(&audience_strs);
}
decode::<JwtClaims>(token, key, &validation)
.map(|data| data.claims)
.map_err(AuthError::InvalidToken)
}