use crate::config::{Config, OidcClient, OidcClients};
use axum::{extract, Json};
use chacha20poly1305::{aead::Aead, ChaCha20Poly1305, KeyInit, Nonce as ChaCha20Poly1305Nonce};
use chartered_db::{users::User, ConnectionPool};
use oauth2::{
basic::BasicErrorResponseType, AuthorizationCode, CsrfToken, RequestTokenError, Scope,
StandardErrorResponse, TokenResponse,
};
use openid::{Options, Token, Userinfo};
use serde::{Deserialize, Serialize};
use std::sync::Arc;
use thiserror::Error;
use url::Url;
pub type Nonce = [u8; 16];
#[allow(clippy::unused_async)]
pub async fn list_providers(
extract::Extension(config): extract::Extension<Arc<Config>>,
extract::Extension(oidc_clients): extract::Extension<Arc<OidcClients>>,
) -> Json<ListProvidersResponse> {
Json(ListProvidersResponse {
password: config.auth.password.enabled,
providers: oidc_clients
.keys()
.into_iter()
.map(std::string::ToString::to_string)
.collect(),
})
}
#[allow(clippy::unused_async)]
pub async fn begin_oidc(
extract::Path(provider): extract::Path<String>,
extract::Extension(config): extract::Extension<Arc<Config>>,
extract::Extension(oidc_clients): extract::Extension<Arc<OidcClients>>,
) -> Result<Json<BeginResponse>, Error> {
let client = oidc_clients
.get(&provider)
.ok_or(Error::UnknownOauthProvider)?;
let nonce = rand::random::<Nonce>();
let state = serde_json::to_vec(&State { provider, nonce })?;
let state = encrypt_url_safe(&state, &config)?;
let redirect_url = match client {
OidcClient::Discovered(client) => client.auth_url(&Options {
scope: Some("openid email profile".into()),
nonce: Some(base64::encode_config(&nonce, base64::URL_SAFE_NO_PAD)),
state: Some(state),
..Options::default()
}),
OidcClient::GitHub(client) => {
client
.authorize_url(move || CsrfToken::new(state))
.add_scope(Scope::new("read:user".to_string()))
.add_scope(Scope::new("user:email".to_string()))
.url()
.0
}
};
Ok(Json(BeginResponse {
redirect_url: redirect_url.to_string(),
}))
}
pub async fn complete_oidc(
extract::Query(params): extract::Query<CompleteOidcParams>,
extract::Extension(config): extract::Extension<Arc<Config>>,
extract::Extension(oidc_clients): extract::Extension<Arc<OidcClients>>,
extract::Extension(db): extract::Extension<ConnectionPool>,
extract::Extension(http_client): extract::Extension<reqwest::Client>,
user_agent: Option<extract::TypedHeader<headers::UserAgent>>,
addr: extract::ConnectInfo<std::net::SocketAddr>,
) -> Result<Json<super::LoginResponse>, Error> {
let state: State = serde_json::from_slice(&decrypt_url_safe(¶ms.state, &config)?)?;
let client = oidc_clients
.get(&state.provider)
.ok_or(Error::UnknownOauthProvider)?;
let user = match client {
OidcClient::Discovered(client) => {
let mut token: Token = client.request_token(¶ms.code).await?.into();
if let Some(id_token) = token.id_token.as_mut() {
client.decode_token(id_token)?;
let nonce = base64::encode_config(state.nonce, base64::URL_SAFE_NO_PAD);
client.validate_token(id_token, Some(nonce.as_str()), None)?;
} else {
return Err(Error::MissingToken);
}
UserIr::from(client.request_userinfo(&token).await?)
}
OidcClient::GitHub(client) => {
let token_result = client
.exchange_code(AuthorizationCode::new(params.code))
.request_async(oauth2::reqwest::async_http_client)
.await?;
eprintln!("{}", token_result.access_token().secret());
let res: GitHubUserResponse = http_client
.get("https://api.github.com/user")
.bearer_auth(token_result.access_token().secret())
.header("Accept", "application/vnd.github+json")
.send()
.await?
.json()
.await?;
UserIr::from(res)
}
};
let user = User::find_or_create(
db.clone(),
format!("{}:{}", state.provider, user.id),
user.name,
user.nick,
user.email,
user.profile_url,
user.avatar_url,
)
.await?;
Ok(Json(super::login(db, user, user_agent, addr).await?))
}
pub struct UserIr {
id: String,
name: Option<String>,
nick: Option<String>,
email: Option<String>,
profile_url: Option<Url>,
avatar_url: Option<Url>,
}
impl From<GitHubUserResponse> for UserIr {
fn from(v: GitHubUserResponse) -> Self {
UserIr {
id: v.id.to_string(),
name: Some(v.name.unwrap_or_else(|| v.login.to_string())),
nick: Some(v.login),
email: Some(v.email),
profile_url: v.html_url,
avatar_url: Some(v.avatar_url),
}
}
}
impl From<Userinfo> for UserIr {
fn from(v: Userinfo) -> Self {
UserIr {
id: v.sub.unwrap(),
name: v.name,
nick: v.nickname,
email: v.email,
profile_url: v.profile,
avatar_url: v.picture,
}
}
}
#[derive(Deserialize)]
pub struct GitHubUserResponse {
id: u64,
login: String,
avatar_url: Url,
html_url: Option<Url>,
name: Option<String>,
email: String,
}
const NONCE_LEN: usize = 12;
fn encrypt_url_safe(input: &[u8], config: &Config) -> Result<String, Error> {
let cipher = ChaCha20Poly1305::new(&config.encryption_key);
let nonce = rand::random::<[u8; NONCE_LEN]>();
let nonce = ChaCha20Poly1305Nonce::from_slice(&nonce);
let mut ciphertext = cipher.encrypt(nonce, input)?;
ciphertext.extend_from_slice(nonce);
Ok(base64::encode_config(&ciphertext, base64::URL_SAFE_NO_PAD))
}
fn decrypt_url_safe(input: &str, config: &Config) -> Result<Vec<u8>, Error> {
let cipher = ChaCha20Poly1305::new(&config.encryption_key);
let mut ciphertext = base64::decode_config(input, base64::URL_SAFE_NO_PAD)?;
let ciphertext_nonce = ciphertext.split_off(ciphertext.len() - NONCE_LEN);
let ciphertext_nonce = ChaCha20Poly1305Nonce::from_slice(&ciphertext_nonce);
cipher
.decrypt(ciphertext_nonce, ciphertext.as_ref())
.map_err(Error::from)
}
#[derive(Serialize)]
pub struct ListProvidersResponse {
password: bool,
providers: Vec<String>,
}
#[derive(Serialize)]
pub struct BeginResponse {
redirect_url: String,
}
#[allow(dead_code)]
#[derive(Deserialize)]
pub struct CompleteOidcParams {
state: String,
code: String,
scope: Option<String>,
prompt: Option<String>,
}
#[derive(Serialize, Deserialize, Debug)]
pub struct State {
provider: String,
nonce: Nonce,
}
#[derive(Error, Debug)]
pub enum Error {
#[error("{0}")]
Database(#[from] chartered_db::Error),
#[error("Error serialising/deserialsing state: {0}")]
Serde(#[from] serde_json::Error),
#[error("Unknown OAuth provider given")]
UnknownOauthProvider,
#[error("{0}")]
OAuth(#[from] openid::error::Error),
#[error("{0}")]
OAuthClient(#[from] openid::error::ClientError),
#[error("Error during encryption/decryption")]
Cipher(#[from] chacha20poly1305::aead::Error),
#[error("Base64 error")]
Base64(#[from] base64::DecodeError),
#[error("Missing id_token")]
MissingToken,
#[error("Failed to request profile from OAuth provider")]
FetchProfile(#[from] reqwest::Error),
#[error("Failed to request token from OAuth provider")]
RequestOAuthToken(
#[from]
RequestTokenError<
oauth2::reqwest::Error<reqwest::Error>,
StandardErrorResponse<BasicErrorResponseType>,
>,
),
}
impl Error {
fn status_code(&self) -> axum::http::StatusCode {
use axum::http::StatusCode;
match self {
Self::Database(e) => e.status_code(),
Self::FetchProfile(_) | Self::RequestOAuthToken(_) => StatusCode::BAD_GATEWAY,
_ => StatusCode::INTERNAL_SERVER_ERROR,
}
}
}
define_error_response!(Error);