2021-03-31 10:21:50 +00:00
|
|
|
//! Provides authentication primitives for the HTTP server
|
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
use std::sync::Arc;
|
2021-10-01 05:29:11 +00:00
|
|
|
use std::future::Future;
|
|
|
|
use std::pin::Pin;
|
|
|
|
|
|
|
|
use anyhow::format_err;
|
2021-03-31 10:21:51 +00:00
|
|
|
|
2021-09-21 05:58:48 +00:00
|
|
|
use proxmox::api::UserInformation;
|
|
|
|
|
2021-07-12 09:07:52 +00:00
|
|
|
use pbs_tools::ticket::{self, Ticket};
|
2021-09-10 10:25:32 +00:00
|
|
|
use pbs_config::{token_shadow, CachedUserInfo};
|
|
|
|
use pbs_api_types::{Authid, Userid};
|
2021-09-21 05:58:45 +00:00
|
|
|
use proxmox_rest_server::{ApiAuth, AuthError, extract_cookie};
|
2021-07-12 09:07:52 +00:00
|
|
|
|
2021-03-31 10:21:50 +00:00
|
|
|
use crate::auth_helpers::*;
|
|
|
|
|
|
|
|
use hyper::header;
|
|
|
|
use percent_encoding::percent_decode_str;
|
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
struct UserAuthData {
|
2021-03-31 10:21:50 +00:00
|
|
|
ticket: String,
|
|
|
|
csrf_token: Option<String>,
|
|
|
|
}
|
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
enum AuthData {
|
2021-03-31 10:21:50 +00:00
|
|
|
User(UserAuthData),
|
|
|
|
ApiToken(String),
|
|
|
|
}
|
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
pub struct UserApiAuth {}
|
|
|
|
pub fn default_api_auth() -> Arc<UserApiAuth> {
|
|
|
|
Arc::new(UserApiAuth {})
|
|
|
|
}
|
|
|
|
|
|
|
|
impl UserApiAuth {
|
|
|
|
fn extract_auth_data(headers: &http::HeaderMap) -> Option<AuthData> {
|
|
|
|
if let Some(raw_cookie) = headers.get(header::COOKIE) {
|
|
|
|
if let Ok(cookie) = raw_cookie.to_str() {
|
2021-09-21 05:58:45 +00:00
|
|
|
if let Some(ticket) = extract_cookie(cookie, "PBSAuthCookie") {
|
2021-03-31 10:21:51 +00:00
|
|
|
let csrf_token = match headers.get("CSRFPreventionToken").map(|v| v.to_str()) {
|
|
|
|
Some(Ok(v)) => Some(v.to_owned()),
|
|
|
|
_ => None,
|
|
|
|
};
|
|
|
|
return Some(AuthData::User(UserAuthData { ticket, csrf_token }));
|
|
|
|
}
|
2021-03-31 10:21:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
match headers.get(header::AUTHORIZATION).map(|v| v.to_str()) {
|
|
|
|
Some(Ok(v)) => {
|
|
|
|
if v.starts_with("PBSAPIToken ") || v.starts_with("PBSAPIToken=") {
|
|
|
|
Some(AuthData::ApiToken(v["PBSAPIToken ".len()..].to_owned()))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2021-03-31 10:21:50 +00:00
|
|
|
}
|
2021-03-31 10:21:51 +00:00
|
|
|
_ => None,
|
|
|
|
}
|
2021-03-31 10:21:50 +00:00
|
|
|
}
|
2021-09-21 05:58:48 +00:00
|
|
|
|
2021-10-01 05:29:11 +00:00
|
|
|
async fn check_auth_async(
|
2021-03-31 10:21:51 +00:00
|
|
|
&self,
|
|
|
|
headers: &http::HeaderMap,
|
|
|
|
method: &hyper::Method,
|
2021-09-21 05:58:48 +00:00
|
|
|
) -> Result<(String, Box<dyn UserInformation + Sync + Send>), AuthError> {
|
2021-09-21 05:58:40 +00:00
|
|
|
|
2021-10-01 05:29:11 +00:00
|
|
|
// fixme: make all IO async
|
|
|
|
|
2021-09-21 05:58:40 +00:00
|
|
|
let user_info = CachedUserInfo::new()?;
|
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
let auth_data = Self::extract_auth_data(headers);
|
|
|
|
match auth_data {
|
|
|
|
Some(AuthData::User(user_auth_data)) => {
|
|
|
|
let ticket = user_auth_data.ticket.clone();
|
2021-07-12 09:07:52 +00:00
|
|
|
let ticket_lifetime = ticket::TICKET_LIFETIME;
|
2021-03-31 10:21:50 +00:00
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
let userid: Userid = Ticket::<super::ticket::ApiTicket>::parse(&ticket)?
|
|
|
|
.verify_with_time_frame(public_auth_key(), "PBS", None, -300..ticket_lifetime)?
|
|
|
|
.require_full()?;
|
|
|
|
|
|
|
|
let auth_id = Authid::from(userid.clone());
|
|
|
|
if !user_info.is_active_auth_id(&auth_id) {
|
|
|
|
return Err(format_err!("user account disabled or expired.").into());
|
2021-03-31 10:21:50 +00:00
|
|
|
}
|
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
if method != hyper::Method::GET {
|
|
|
|
if let Some(csrf_token) = &user_auth_data.csrf_token {
|
|
|
|
verify_csrf_prevention_token(
|
|
|
|
csrf_secret(),
|
|
|
|
&userid,
|
|
|
|
&csrf_token,
|
|
|
|
-300,
|
|
|
|
ticket_lifetime,
|
|
|
|
)?;
|
|
|
|
} else {
|
|
|
|
return Err(format_err!("missing CSRF prevention token").into());
|
|
|
|
}
|
|
|
|
}
|
2021-03-31 10:21:50 +00:00
|
|
|
|
2021-09-21 05:58:48 +00:00
|
|
|
Ok((auth_id.to_string(), Box::new(user_info)))
|
2021-03-31 10:21:50 +00:00
|
|
|
}
|
2021-03-31 10:21:51 +00:00
|
|
|
Some(AuthData::ApiToken(api_token)) => {
|
|
|
|
let mut parts = api_token.splitn(2, ':');
|
|
|
|
let tokenid = parts
|
|
|
|
.next()
|
|
|
|
.ok_or_else(|| format_err!("failed to split API token header"))?;
|
|
|
|
let tokenid: Authid = tokenid.parse()?;
|
2021-03-31 10:21:50 +00:00
|
|
|
|
2021-03-31 10:21:51 +00:00
|
|
|
if !user_info.is_active_auth_id(&tokenid) {
|
|
|
|
return Err(format_err!("user account or token disabled or expired.").into());
|
|
|
|
}
|
|
|
|
|
|
|
|
let tokensecret = parts
|
|
|
|
.next()
|
|
|
|
.ok_or_else(|| format_err!("failed to split API token header"))?;
|
|
|
|
let tokensecret = percent_decode_str(tokensecret)
|
|
|
|
.decode_utf8()
|
|
|
|
.map_err(|_| format_err!("failed to decode API token header"))?;
|
2021-03-31 10:21:50 +00:00
|
|
|
|
2021-09-08 12:00:14 +00:00
|
|
|
token_shadow::verify_secret(&tokenid, &tokensecret)?;
|
2021-03-31 10:21:50 +00:00
|
|
|
|
2021-09-21 05:58:48 +00:00
|
|
|
Ok((tokenid.to_string(), Box::new(user_info)))
|
2021-03-31 10:21:51 +00:00
|
|
|
}
|
|
|
|
None => Err(AuthError::NoData),
|
2021-03-31 10:21:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-10-01 05:29:11 +00:00
|
|
|
|
|
|
|
impl ApiAuth for UserApiAuth {
|
|
|
|
fn check_auth<'a>(
|
|
|
|
&'a self,
|
|
|
|
headers: &'a http::HeaderMap,
|
|
|
|
method: &'a hyper::Method,
|
|
|
|
) -> Pin<Box<dyn Future<Output = Result<(String, Box<dyn UserInformation + Sync + Send>), AuthError>> + Send + 'a>> {
|
|
|
|
Box::pin(self.check_auth_async(headers, method))
|
|
|
|
}
|
|
|
|
}
|