You've already forked AstralRinth
forked from xxxOFFxxx/AstralRinth
* Update Rust version * Update async-compression 0.4.25 -> 0.4.27 * Update async-tungstenite 0.29.1 -> 0.30.0 * Update bytemuck 1.23.0 -> 1.23.1 * Update clap 4.5.40 -> 4.5.43 * Update deadpool-redis 0.21.1 -> 0.22.0 and redis 0.31.0 -> 0.32.4 * Update enumset 1.1.6 -> 1.1.7 * Update hyper-util 0.1.14 -> 0.1.16 * Update indexmap 2.9.0 -> 2.10.0 * Update indicatif 0.17.11 -> 0.18.0 * Update jemalloc_pprof 0.7.0 -> 0.8.1 * Update lettre 0.11.17 -> 0.11.18 * Update meilisearch-sdk 0.28.0 -> 0.29.1 * Update notify 8.0.0 -> 8.2.0 and notify-debouncer-mini 0.6.0 -> 0.7.0 * Update quick-xml 0.37.5 -> 0.38.1 * Fix theseus lint * Update reqwest 0.12.20 -> 0.12.22 * Cargo fmt in theseus * Update rgb 0.8.50 -> 0.8.52 * Update sentry 0.41.0 -> 0.42.0 and sentry-actix 0.41.0 -> 0.42.0 * Update serde_json 1.0.140 -> 1.0.142 * Update serde_with 3.13.0 -> 3.14.0 * Update spdx 0.10.8 -> 0.10.9 * Update sysinfo 0.35.2 -> 0.36.1 * Update tauri suite * Fix build by updating mappings * Update tokio 1.45.1 -> 1.47.1 and tokio-util 0.7.15 -> 0.7.16 * Update tracing-actix-web 0.7.18 -> 0.7.19 * Update zip 4.2.0 -> 4.3.0 * Misc Cargo.lock updates * Update Dockerfiles
116 lines
3.1 KiB
Rust
116 lines
3.1 KiB
Rust
use super::ids::*;
|
|
use crate::auth::AuthProvider;
|
|
use crate::auth::oauth::uris::OAuthRedirectUris;
|
|
use crate::database::models::DatabaseError;
|
|
use crate::database::redis::RedisPool;
|
|
use crate::models::pats::Scopes;
|
|
use chrono::Duration;
|
|
use rand::Rng;
|
|
use rand::distributions::Alphanumeric;
|
|
use rand_chacha::ChaCha20Rng;
|
|
use rand_chacha::rand_core::SeedableRng;
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
const FLOWS_NAMESPACE: &str = "flows";
|
|
|
|
#[derive(Deserialize, Serialize)]
|
|
#[serde(tag = "type", rename_all = "snake_case")]
|
|
pub enum DBFlow {
|
|
OAuth {
|
|
user_id: Option<DBUserId>,
|
|
url: String,
|
|
provider: AuthProvider,
|
|
},
|
|
Login2FA {
|
|
user_id: DBUserId,
|
|
},
|
|
Initialize2FA {
|
|
user_id: DBUserId,
|
|
secret: String,
|
|
},
|
|
ForgotPassword {
|
|
user_id: DBUserId,
|
|
},
|
|
ConfirmEmail {
|
|
user_id: DBUserId,
|
|
confirm_email: String,
|
|
},
|
|
MinecraftAuth,
|
|
InitOAuthAppApproval {
|
|
user_id: DBUserId,
|
|
client_id: DBOAuthClientId,
|
|
existing_authorization_id: Option<DBOAuthClientAuthorizationId>,
|
|
scopes: Scopes,
|
|
redirect_uris: OAuthRedirectUris,
|
|
state: Option<String>,
|
|
},
|
|
OAuthAuthorizationCodeSupplied {
|
|
user_id: DBUserId,
|
|
client_id: DBOAuthClientId,
|
|
authorization_id: DBOAuthClientAuthorizationId,
|
|
scopes: Scopes,
|
|
original_redirect_uri: Option<String>, // Needed for https://datatracker.ietf.org/doc/html/rfc6749#section-4.1.3
|
|
},
|
|
}
|
|
|
|
impl DBFlow {
|
|
pub async fn insert(
|
|
&self,
|
|
expires: Duration,
|
|
redis: &RedisPool,
|
|
) -> Result<String, DatabaseError> {
|
|
let mut redis = redis.connect().await?;
|
|
|
|
let flow = ChaCha20Rng::from_entropy()
|
|
.sample_iter(&Alphanumeric)
|
|
.take(32)
|
|
.map(char::from)
|
|
.collect::<String>();
|
|
|
|
redis
|
|
.set_serialized_to_json(
|
|
FLOWS_NAMESPACE,
|
|
&flow,
|
|
&self,
|
|
Some(expires.num_seconds()),
|
|
)
|
|
.await?;
|
|
Ok(flow)
|
|
}
|
|
|
|
pub async fn get(
|
|
id: &str,
|
|
redis: &RedisPool,
|
|
) -> Result<Option<DBFlow>, DatabaseError> {
|
|
let mut redis = redis.connect().await?;
|
|
|
|
redis.get_deserialized_from_json(FLOWS_NAMESPACE, id).await
|
|
}
|
|
|
|
/// Gets the flow and removes it from the cache, but only removes if the flow was present and the predicate returned true
|
|
/// The predicate should validate that the flow being removed is the correct one, as a security measure
|
|
pub async fn take_if(
|
|
id: &str,
|
|
predicate: impl FnOnce(&DBFlow) -> bool,
|
|
redis: &RedisPool,
|
|
) -> Result<Option<DBFlow>, DatabaseError> {
|
|
let flow = Self::get(id, redis).await?;
|
|
if let Some(flow) = flow.as_ref()
|
|
&& predicate(flow)
|
|
{
|
|
Self::remove(id, redis).await?;
|
|
}
|
|
Ok(flow)
|
|
}
|
|
|
|
pub async fn remove(
|
|
id: &str,
|
|
redis: &RedisPool,
|
|
) -> Result<Option<()>, DatabaseError> {
|
|
let mut redis = redis.connect().await?;
|
|
|
|
redis.delete(FLOWS_NAMESPACE, id).await?;
|
|
Ok(Some(()))
|
|
}
|
|
}
|