mod alias; mod delete_token; mod hash; mod metrics; mod migrate; use crate::{ config, details::Details, error_code::{ErrorCode, OwnedErrorCode}, stream::LocalBoxStream, }; use base64::Engine; use futures_core::Stream; use std::{fmt::Debug, sync::Arc}; use url::Url; use uuid::Uuid; pub(crate) mod postgres; pub(crate) mod sled; pub(crate) use alias::Alias; pub(crate) use delete_token::DeleteToken; pub(crate) use hash::Hash; pub(crate) use migrate::{migrate_04, migrate_repo}; pub(crate) type ArcRepo = Arc; #[derive(Clone, Debug)] pub(crate) enum Repo { Sled(self::sled::SledRepo), Postgres(self::postgres::PostgresRepo), } #[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] enum MaybeUuid { Uuid(Uuid), Name(String), } #[derive(Debug)] pub(crate) struct HashAlreadyExists; #[derive(Debug)] pub(crate) struct AliasAlreadyExists; #[derive(Debug)] pub(crate) struct VariantAlreadyExists; #[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] pub(crate) struct UploadId { id: Uuid, } #[derive(Debug)] pub(crate) enum UploadResult { Success { alias: Alias, token: DeleteToken, }, Failure { message: String, code: OwnedErrorCode, }, } #[derive(Debug, thiserror::Error)] pub(crate) enum RepoError { #[error("Error in sled")] SledError(#[from] crate::repo::sled::SledError), #[error("Error in postgres")] PostgresError(#[from] crate::repo::postgres::PostgresError), #[error("Upload was already claimed")] AlreadyClaimed, #[error("Panic in blocking operation")] Canceled, } impl RepoError { pub(crate) const fn error_code(&self) -> ErrorCode { match self { Self::SledError(e) => e.error_code(), Self::PostgresError(e) => e.error_code(), Self::AlreadyClaimed => ErrorCode::ALREADY_CLAIMED, Self::Canceled => ErrorCode::PANIC, } } pub(crate) const fn is_disconnected(&self) -> bool { match self { Self::PostgresError(e) => e.is_disconnected(), _ => false, } } } #[async_trait::async_trait(?Send)] pub(crate) trait FullRepo: UploadRepo + SettingsRepo + DetailsRepo + AliasRepo + QueueRepo + HashRepo + VariantRepo + StoreMigrationRepo + AliasAccessRepo + VariantAccessRepo + ProxyRepo + Send + Sync + Debug { async fn health_check(&self) -> Result<(), RepoError>; #[tracing::instrument(skip(self))] async fn identifier_from_alias(&self, alias: &Alias) -> Result>, RepoError> { let Some(hash) = self.hash(alias).await? else { return Ok(None); }; self.identifier(hash).await } #[tracing::instrument(skip(self))] async fn aliases_from_alias(&self, alias: &Alias) -> Result, RepoError> { let Some(hash) = self.hash(alias).await? else { return Ok(vec![]); }; self.aliases_for_hash(hash).await } } #[async_trait::async_trait(?Send)] impl FullRepo for Arc where T: FullRepo, { async fn health_check(&self) -> Result<(), RepoError> { T::health_check(self).await } } pub(crate) trait BaseRepo {} impl BaseRepo for Arc where T: BaseRepo {} #[async_trait::async_trait(?Send)] pub(crate) trait ProxyRepo: BaseRepo { async fn relate_url(&self, url: Url, alias: Alias) -> Result<(), RepoError>; async fn related(&self, url: Url) -> Result, RepoError>; async fn remove_relation(&self, alias: Alias) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl ProxyRepo for Arc where T: ProxyRepo, { async fn relate_url(&self, url: Url, alias: Alias) -> Result<(), RepoError> { T::relate_url(self, url, alias).await } async fn related(&self, url: Url) -> Result, RepoError> { T::related(self, url).await } async fn remove_relation(&self, alias: Alias) -> Result<(), RepoError> { T::remove_relation(self, alias).await } } #[async_trait::async_trait(?Send)] pub(crate) trait AliasAccessRepo: BaseRepo { async fn accessed_alias(&self, alias: Alias) -> Result<(), RepoError> { self.set_accessed_alias(alias, time::OffsetDateTime::now_utc()) .await } async fn set_accessed_alias( &self, alias: Alias, accessed: time::OffsetDateTime, ) -> Result<(), RepoError>; async fn alias_accessed_at( &self, alias: Alias, ) -> Result, RepoError>; async fn older_aliases( &self, timestamp: time::OffsetDateTime, ) -> Result>, RepoError>; async fn remove_alias_access(&self, alias: Alias) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl AliasAccessRepo for Arc where T: AliasAccessRepo, { async fn set_accessed_alias( &self, alias: Alias, accessed: time::OffsetDateTime, ) -> Result<(), RepoError> { T::set_accessed_alias(self, alias, accessed).await } async fn alias_accessed_at( &self, alias: Alias, ) -> Result, RepoError> { T::alias_accessed_at(self, alias).await } async fn older_aliases( &self, timestamp: time::OffsetDateTime, ) -> Result>, RepoError> { T::older_aliases(self, timestamp).await } async fn remove_alias_access(&self, alias: Alias) -> Result<(), RepoError> { T::remove_alias_access(self, alias).await } } #[async_trait::async_trait(?Send)] pub(crate) trait VariantAccessRepo: BaseRepo { async fn accessed_variant(&self, hash: Hash, variant: String) -> Result<(), RepoError> { self.set_accessed_variant(hash, variant, time::OffsetDateTime::now_utc()) .await } async fn set_accessed_variant( &self, hash: Hash, variant: String, accessed: time::OffsetDateTime, ) -> Result<(), RepoError>; async fn variant_accessed_at( &self, hash: Hash, variant: String, ) -> Result, RepoError>; async fn older_variants( &self, timestamp: time::OffsetDateTime, ) -> Result>, RepoError>; async fn remove_variant_access(&self, hash: Hash, variant: String) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl VariantAccessRepo for Arc where T: VariantAccessRepo, { async fn set_accessed_variant( &self, hash: Hash, variant: String, accessed: time::OffsetDateTime, ) -> Result<(), RepoError> { T::set_accessed_variant(self, hash, variant, accessed).await } async fn variant_accessed_at( &self, hash: Hash, variant: String, ) -> Result, RepoError> { T::variant_accessed_at(self, hash, variant).await } async fn older_variants( &self, timestamp: time::OffsetDateTime, ) -> Result>, RepoError> { T::older_variants(self, timestamp).await } async fn remove_variant_access(&self, hash: Hash, variant: String) -> Result<(), RepoError> { T::remove_variant_access(self, hash, variant).await } } #[async_trait::async_trait(?Send)] pub(crate) trait UploadRepo: BaseRepo { async fn create_upload(&self) -> Result; async fn wait(&self, upload_id: UploadId) -> Result; async fn claim(&self, upload_id: UploadId) -> Result<(), RepoError>; async fn complete_upload( &self, upload_id: UploadId, result: UploadResult, ) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl UploadRepo for Arc where T: UploadRepo, { async fn create_upload(&self) -> Result { T::create_upload(self).await } async fn wait(&self, upload_id: UploadId) -> Result { T::wait(self, upload_id).await } async fn claim(&self, upload_id: UploadId) -> Result<(), RepoError> { T::claim(self, upload_id).await } async fn complete_upload( &self, upload_id: UploadId, result: UploadResult, ) -> Result<(), RepoError> { T::complete_upload(self, upload_id, result).await } } #[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] pub(crate) struct JobId(Uuid); #[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] pub(crate) enum JobResult { Success, Failure, Aborted, } impl JobId { pub(crate) fn gen() -> Self { Self(Uuid::now_v7()) } pub(crate) const fn as_bytes(&self) -> &[u8; 16] { self.0.as_bytes() } pub(crate) const fn from_bytes(bytes: [u8; 16]) -> Self { Self(Uuid::from_bytes(bytes)) } } #[async_trait::async_trait(?Send)] pub(crate) trait QueueRepo: BaseRepo { async fn queue_length(&self) -> Result; async fn push( &self, queue: &'static str, job: serde_json::Value, unique_key: Option<&'static str>, ) -> Result, RepoError>; async fn pop( &self, queue: &'static str, worker_id: Uuid, ) -> Result<(JobId, serde_json::Value), RepoError>; async fn heartbeat( &self, queue: &'static str, worker_id: Uuid, job_id: JobId, ) -> Result<(), RepoError>; async fn complete_job( &self, queue: &'static str, worker_id: Uuid, job_id: JobId, job_status: JobResult, ) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl QueueRepo for Arc where T: QueueRepo, { async fn queue_length(&self) -> Result { T::queue_length(self).await } async fn push( &self, queue: &'static str, job: serde_json::Value, unique_key: Option<&'static str>, ) -> Result, RepoError> { T::push(self, queue, job, unique_key).await } async fn pop( &self, queue: &'static str, worker_id: Uuid, ) -> Result<(JobId, serde_json::Value), RepoError> { T::pop(self, queue, worker_id).await } async fn heartbeat( &self, queue: &'static str, worker_id: Uuid, job_id: JobId, ) -> Result<(), RepoError> { T::heartbeat(self, queue, worker_id, job_id).await } async fn complete_job( &self, queue: &'static str, worker_id: Uuid, job_id: JobId, job_status: JobResult, ) -> Result<(), RepoError> { T::complete_job(self, queue, worker_id, job_id, job_status).await } } #[async_trait::async_trait(?Send)] pub(crate) trait SettingsRepo: BaseRepo { async fn set(&self, key: &'static str, value: Arc<[u8]>) -> Result<(), RepoError>; async fn get(&self, key: &'static str) -> Result>, RepoError>; async fn remove(&self, key: &'static str) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl SettingsRepo for Arc where T: SettingsRepo, { async fn set(&self, key: &'static str, value: Arc<[u8]>) -> Result<(), RepoError> { T::set(self, key, value).await } async fn get(&self, key: &'static str) -> Result>, RepoError> { T::get(self, key).await } async fn remove(&self, key: &'static str) -> Result<(), RepoError> { T::remove(self, key).await } } #[async_trait::async_trait(?Send)] pub(crate) trait DetailsRepo: BaseRepo { async fn relate_details( &self, identifier: &Arc, details: &Details, ) -> Result<(), RepoError>; async fn details(&self, identifier: &Arc) -> Result, RepoError>; async fn cleanup_details(&self, identifier: &Arc) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl DetailsRepo for Arc where T: DetailsRepo, { async fn relate_details( &self, identifier: &Arc, details: &Details, ) -> Result<(), RepoError> { T::relate_details(self, identifier, details).await } async fn details(&self, identifier: &Arc) -> Result, RepoError> { T::details(self, identifier).await } async fn cleanup_details(&self, identifier: &Arc) -> Result<(), RepoError> { T::cleanup_details(self, identifier).await } } #[async_trait::async_trait(?Send)] pub(crate) trait StoreMigrationRepo: BaseRepo { async fn is_continuing_migration(&self) -> Result; async fn mark_migrated( &self, old_identifier: &Arc, new_identifier: &Arc, ) -> Result<(), RepoError>; async fn is_migrated(&self, identifier: &Arc) -> Result; async fn clear(&self) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl StoreMigrationRepo for Arc where T: StoreMigrationRepo, { async fn is_continuing_migration(&self) -> Result { T::is_continuing_migration(self).await } async fn mark_migrated( &self, old_identifier: &Arc, new_identifier: &Arc, ) -> Result<(), RepoError> { T::mark_migrated(self, old_identifier, new_identifier).await } async fn is_migrated(&self, identifier: &Arc) -> Result { T::is_migrated(self, identifier).await } async fn clear(&self) -> Result<(), RepoError> { T::clear(self).await } } #[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)] pub(crate) struct OrderedHash { timestamp: time::OffsetDateTime, hash: Hash, } pub(crate) struct HashPage { pub(crate) limit: usize, prev: Option, next: Option, pub(crate) hashes: Vec, } fn hash_to_slug(hash: &Hash) -> String { base64::prelude::BASE64_URL_SAFE.encode(hash.to_bytes()) } fn hash_from_slug(s: &str) -> Option { let bytes = base64::prelude::BASE64_URL_SAFE.decode(s).ok()?; let hash = Hash::from_bytes(&bytes)?; Some(hash) } impl HashPage { pub(crate) fn current(&self) -> Option { self.hashes.first().map(hash_to_slug) } pub(crate) fn next(&self) -> Option { self.next.as_ref().map(hash_to_slug) } pub(crate) fn prev(&self) -> Option { self.prev.as_ref().map(hash_to_slug) } } impl dyn FullRepo { pub(crate) fn hashes(self: &Arc) -> impl Stream> { let repo = self.clone(); streem::try_from_fn(|yielder| async move { let mut slug = None; loop { tracing::trace!("hashes_stream: looping"); let page = repo.hash_page(slug, 100).await?; slug = page.next(); for hash in page.hashes { yielder.yield_ok(hash).await; } if slug.is_none() { break; } } Ok(()) }) } } #[async_trait::async_trait(?Send)] pub(crate) trait HashRepo: BaseRepo { async fn size(&self) -> Result; async fn hash_page(&self, slug: Option, limit: usize) -> Result { let hash = slug.as_deref().and_then(hash_from_slug); let bound = if let Some(hash) = hash { self.bound(hash).await? } else { None }; self.hashes_ordered(bound, limit).await } async fn hash_page_by_date( &self, date: time::OffsetDateTime, limit: usize, ) -> Result; async fn bound(&self, hash: Hash) -> Result, RepoError>; async fn hashes_ordered( &self, bound: Option, limit: usize, ) -> Result; async fn create_hash( &self, hash: Hash, identifier: &Arc, ) -> Result, RepoError> { self.create_hash_with_timestamp(hash, identifier, time::OffsetDateTime::now_utc()) .await } async fn create_hash_with_timestamp( &self, hash: Hash, identifier: &Arc, timestamp: time::OffsetDateTime, ) -> Result, RepoError>; async fn update_identifier(&self, hash: Hash, identifier: &Arc) -> Result<(), RepoError>; async fn identifier(&self, hash: Hash) -> Result>, RepoError>; async fn relate_blurhash(&self, hash: Hash, blurhash: Arc) -> Result<(), RepoError>; async fn blurhash(&self, hash: Hash) -> Result>, RepoError>; async fn relate_motion_identifier( &self, hash: Hash, identifier: &Arc, ) -> Result<(), RepoError>; async fn motion_identifier(&self, hash: Hash) -> Result>, RepoError>; async fn cleanup_hash(&self, hash: Hash) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl HashRepo for Arc where T: HashRepo, { async fn size(&self) -> Result { T::size(self).await } async fn bound(&self, hash: Hash) -> Result, RepoError> { T::bound(self, hash).await } async fn hashes_ordered( &self, bound: Option, limit: usize, ) -> Result { T::hashes_ordered(self, bound, limit).await } async fn hash_page_by_date( &self, date: time::OffsetDateTime, limit: usize, ) -> Result { T::hash_page_by_date(self, date, limit).await } async fn create_hash_with_timestamp( &self, hash: Hash, identifier: &Arc, timestamp: time::OffsetDateTime, ) -> Result, RepoError> { T::create_hash_with_timestamp(self, hash, identifier, timestamp).await } async fn update_identifier(&self, hash: Hash, identifier: &Arc) -> Result<(), RepoError> { T::update_identifier(self, hash, identifier).await } async fn identifier(&self, hash: Hash) -> Result>, RepoError> { T::identifier(self, hash).await } async fn relate_blurhash(&self, hash: Hash, blurhash: Arc) -> Result<(), RepoError> { T::relate_blurhash(self, hash, blurhash).await } async fn blurhash(&self, hash: Hash) -> Result>, RepoError> { T::blurhash(self, hash).await } async fn relate_motion_identifier( &self, hash: Hash, identifier: &Arc, ) -> Result<(), RepoError> { T::relate_motion_identifier(self, hash, identifier).await } async fn motion_identifier(&self, hash: Hash) -> Result>, RepoError> { T::motion_identifier(self, hash).await } async fn cleanup_hash(&self, hash: Hash) -> Result<(), RepoError> { T::cleanup_hash(self, hash).await } } #[async_trait::async_trait(?Send)] pub(crate) trait VariantRepo: BaseRepo { async fn relate_variant_identifier( &self, hash: Hash, variant: String, identifier: &Arc, ) -> Result, RepoError>; async fn variant_identifier( &self, hash: Hash, variant: String, ) -> Result>, RepoError>; async fn variants(&self, hash: Hash) -> Result)>, RepoError>; async fn remove_variant(&self, hash: Hash, variant: String) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl VariantRepo for Arc where T: VariantRepo, { async fn relate_variant_identifier( &self, hash: Hash, variant: String, identifier: &Arc, ) -> Result, RepoError> { T::relate_variant_identifier(self, hash, variant, identifier).await } async fn variant_identifier( &self, hash: Hash, variant: String, ) -> Result>, RepoError> { T::variant_identifier(self, hash, variant).await } async fn variants(&self, hash: Hash) -> Result)>, RepoError> { T::variants(self, hash).await } async fn remove_variant(&self, hash: Hash, variant: String) -> Result<(), RepoError> { T::remove_variant(self, hash, variant).await } } #[async_trait::async_trait(?Send)] pub(crate) trait AliasRepo: BaseRepo { async fn create_alias( &self, alias: &Alias, delete_token: &DeleteToken, hash: Hash, ) -> Result, RepoError>; async fn delete_token(&self, alias: &Alias) -> Result, RepoError>; async fn hash(&self, alias: &Alias) -> Result, RepoError>; async fn aliases_for_hash(&self, hash: Hash) -> Result, RepoError>; async fn cleanup_alias(&self, alias: &Alias) -> Result<(), RepoError>; } #[async_trait::async_trait(?Send)] impl AliasRepo for Arc where T: AliasRepo, { async fn create_alias( &self, alias: &Alias, delete_token: &DeleteToken, hash: Hash, ) -> Result, RepoError> { T::create_alias(self, alias, delete_token, hash).await } async fn delete_token(&self, alias: &Alias) -> Result, RepoError> { T::delete_token(self, alias).await } async fn hash(&self, alias: &Alias) -> Result, RepoError> { T::hash(self, alias).await } async fn aliases_for_hash(&self, hash: Hash) -> Result, RepoError> { T::aliases_for_hash(self, hash).await } async fn cleanup_alias(&self, alias: &Alias) -> Result<(), RepoError> { T::cleanup_alias(self, alias).await } } impl Repo { #[tracing::instrument(skip(config))] pub(crate) async fn open(config: config::Repo) -> color_eyre::Result { match config { config::Repo::Sled(config::Sled { path, cache_capacity, export_path, }) => { let repo = self::sled::SledRepo::build(path, cache_capacity, export_path)?; Ok(Self::Sled(repo)) } config::Repo::Postgres(config::Postgres { url, use_tls, certificate_file, }) => { let repo = self::postgres::PostgresRepo::connect(url, use_tls, certificate_file).await?; Ok(Self::Postgres(repo)) } } } pub(crate) fn to_arc(&self) -> ArcRepo { match self { Self::Sled(sled_repo) => Arc::new(sled_repo.clone()), Self::Postgres(postgres_repo) => Arc::new(postgres_repo.clone()), } } } impl MaybeUuid { fn from_str(s: &str) -> Self { if let Ok(uuid) = Uuid::parse_str(s) { MaybeUuid::Uuid(uuid) } else { MaybeUuid::Name(s.into()) } } fn as_bytes(&self) -> &[u8] { match self { Self::Uuid(uuid) => &uuid.as_bytes()[..], Self::Name(name) => name.as_bytes(), } } } impl UploadId { pub(crate) fn generate() -> Self { Self { id: Uuid::new_v4() } } pub(crate) fn as_bytes(&self) -> &[u8] { &self.id.as_bytes()[..] } } impl std::str::FromStr for UploadId { type Err = ::Err; fn from_str(s: &str) -> Result { Ok(UploadId { id: s.parse()? }) } } impl std::fmt::Display for UploadId { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { std::fmt::Display::fmt(&self.id, f) } } impl std::fmt::Display for MaybeUuid { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { Self::Uuid(id) => write!(f, "{id}"), Self::Name(name) => write!(f, "{name}"), } } }