You've already forked AstralRinth
forked from didirus/AstralRinth
* Add redis caching to getting a user's project ids * Run `cargo sqlx prepare` to update the sqlx-data.json * Add redis caching for getting user notifications * Fix new clippy warnings * Remove log that shouldn't have been committed * Batch insert of notifications (untested) * sqlx prepare... * Fix merge conflict things and use new redis struct * Fix bug with calling delete_many without any elements (caught by tests) * cargo sqlx prepare * Add tests around cache invalidation (and fix bug they caught!) * Some test reorg based on code review suggestions
66 lines
1.6 KiB
Rust
66 lines
1.6 KiB
Rust
use super::ids::*;
|
|
use crate::auth::flows::AuthProvider;
|
|
use crate::database::models::DatabaseError;
|
|
use crate::database::redis::RedisPool;
|
|
use chrono::Duration;
|
|
use rand::distributions::Alphanumeric;
|
|
use rand::Rng;
|
|
use rand_chacha::rand_core::SeedableRng;
|
|
use rand_chacha::ChaCha20Rng;
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
const FLOWS_NAMESPACE: &str = "flows";
|
|
|
|
#[derive(Deserialize, Serialize)]
|
|
#[serde(tag = "type", rename_all = "snake_case")]
|
|
pub enum Flow {
|
|
OAuth {
|
|
user_id: Option<UserId>,
|
|
url: Option<String>,
|
|
provider: AuthProvider,
|
|
},
|
|
Login2FA {
|
|
user_id: UserId,
|
|
},
|
|
Initialize2FA {
|
|
user_id: UserId,
|
|
secret: String,
|
|
},
|
|
ForgotPassword {
|
|
user_id: UserId,
|
|
},
|
|
ConfirmEmail {
|
|
user_id: UserId,
|
|
confirm_email: String,
|
|
},
|
|
MinecraftAuth,
|
|
}
|
|
|
|
impl Flow {
|
|
pub async fn insert(
|
|
&self,
|
|
expires: Duration,
|
|
redis: &RedisPool,
|
|
) -> Result<String, DatabaseError> {
|
|
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<Flow>, DatabaseError> {
|
|
redis.get_deserialized_from_json(FLOWS_NAMESPACE, id).await
|
|
}
|
|
|
|
pub async fn remove(id: &str, redis: &RedisPool) -> Result<Option<()>, DatabaseError> {
|
|
redis.delete(FLOWS_NAMESPACE, id).await?;
|
|
Ok(Some(()))
|
|
}
|
|
}
|