mirror of
https://git.asonix.dog/asonix/pict-rs
synced 2024-11-10 06:25:00 +00:00
Remove is_missing error, make Identifier and DeleteToken return Option on fetch
This commit is contained in:
parent
d16e81be20
commit
9e7376d411
8 changed files with 105 additions and 79 deletions
10
src/error.rs
10
src/error.rs
|
@ -81,6 +81,9 @@ pub(crate) enum UploadError {
|
||||||
#[error("Requested a file that doesn't exist")]
|
#[error("Requested a file that doesn't exist")]
|
||||||
MissingAlias,
|
MissingAlias,
|
||||||
|
|
||||||
|
#[error("Requested a file that pict-rs lost track of")]
|
||||||
|
MissingIdentifier,
|
||||||
|
|
||||||
#[error("Provided token did not match expected token")]
|
#[error("Provided token did not match expected token")]
|
||||||
InvalidToken,
|
InvalidToken,
|
||||||
|
|
||||||
|
@ -169,12 +172,7 @@ impl ResponseError for Error {
|
||||||
| UploadError::Repo(crate::repo::RepoError::AlreadyClaimed)
|
| UploadError::Repo(crate::repo::RepoError::AlreadyClaimed)
|
||||||
| UploadError::SilentVideoDisabled,
|
| UploadError::SilentVideoDisabled,
|
||||||
) => StatusCode::BAD_REQUEST,
|
) => StatusCode::BAD_REQUEST,
|
||||||
Some(
|
Some(UploadError::MissingAlias) => StatusCode::NOT_FOUND,
|
||||||
UploadError::Repo(crate::repo::RepoError::SledError(
|
|
||||||
crate::repo::sled::SledError::Missing(_),
|
|
||||||
))
|
|
||||||
| UploadError::MissingAlias,
|
|
||||||
) => StatusCode::NOT_FOUND,
|
|
||||||
Some(UploadError::InvalidToken) => StatusCode::FORBIDDEN,
|
Some(UploadError::InvalidToken) => StatusCode::FORBIDDEN,
|
||||||
Some(UploadError::Range) => StatusCode::RANGE_NOT_SATISFIABLE,
|
Some(UploadError::Range) => StatusCode::RANGE_NOT_SATISFIABLE,
|
||||||
_ => StatusCode::INTERNAL_SERVER_ERROR,
|
_ => StatusCode::INTERNAL_SERVER_ERROR,
|
||||||
|
|
|
@ -2,7 +2,7 @@ use crate::{
|
||||||
concurrent_processor::CancelSafeProcessor,
|
concurrent_processor::CancelSafeProcessor,
|
||||||
config::ImageFormat,
|
config::ImageFormat,
|
||||||
details::Details,
|
details::Details,
|
||||||
error::Error,
|
error::{Error, UploadError},
|
||||||
ffmpeg::{ThumbnailFormat, VideoFormat},
|
ffmpeg::{ThumbnailFormat, VideoFormat},
|
||||||
repo::{Alias, FullRepo},
|
repo::{Alias, FullRepo},
|
||||||
store::Store,
|
store::Store,
|
||||||
|
@ -64,7 +64,10 @@ async fn process<R: FullRepo, S: Store + 'static>(
|
||||||
{
|
{
|
||||||
identifier
|
identifier
|
||||||
} else {
|
} else {
|
||||||
let identifier = repo.identifier(hash.clone()).await?;
|
let Some(identifier) = repo.identifier(hash.clone()).await? else {
|
||||||
|
return Err(UploadError::MissingIdentifier.into());
|
||||||
|
};
|
||||||
|
|
||||||
let reader = crate::ffmpeg::thumbnail(
|
let reader = crate::ffmpeg::thumbnail(
|
||||||
store.clone(),
|
store.clone(),
|
||||||
identifier,
|
identifier,
|
||||||
|
|
|
@ -3,7 +3,7 @@ use crate::{
|
||||||
either::Either,
|
either::Either,
|
||||||
error::{Error, UploadError},
|
error::{Error, UploadError},
|
||||||
magick::ValidInputType,
|
magick::ValidInputType,
|
||||||
repo::{Alias, AliasRepo, DeleteToken, FullRepo, HashRepo},
|
repo::{Alias, AliasRepo, AlreadyExists, DeleteToken, FullRepo, HashRepo},
|
||||||
store::Store,
|
store::Store,
|
||||||
CONFIG,
|
CONFIG,
|
||||||
};
|
};
|
||||||
|
@ -156,8 +156,7 @@ where
|
||||||
tracing::trace!("Saving delete token");
|
tracing::trace!("Saving delete token");
|
||||||
let res = self.repo.relate_delete_token(&alias, &delete_token).await?;
|
let res = self.repo.relate_delete_token(&alias, &delete_token).await?;
|
||||||
|
|
||||||
if res.is_err() {
|
if let Err(AlreadyExists(delete_token)) = res {
|
||||||
let delete_token = self.repo.delete_token(&alias).await?;
|
|
||||||
tracing::trace!("Returning existing delete token, {:?}", delete_token);
|
tracing::trace!("Returning existing delete token, {:?}", delete_token);
|
||||||
return Ok(delete_token);
|
return Ok(delete_token);
|
||||||
}
|
}
|
||||||
|
@ -232,7 +231,7 @@ where
|
||||||
tracing::trace_span!(parent: None, "Spawn task").in_scope(|| {
|
tracing::trace_span!(parent: None, "Spawn task").in_scope(|| {
|
||||||
actix_rt::spawn(
|
actix_rt::spawn(
|
||||||
async move {
|
async move {
|
||||||
if let Ok(token) = repo.delete_token(&alias).await {
|
if let Ok(Some(token)) = repo.delete_token(&alias).await {
|
||||||
let _ = crate::queue::cleanup_alias(&repo, alias, token).await;
|
let _ = crate::queue::cleanup_alias(&repo, alias, token).await;
|
||||||
} else {
|
} else {
|
||||||
let token = DeleteToken::generate();
|
let token = DeleteToken::generate();
|
||||||
|
|
13
src/lib.rs
13
src/lib.rs
|
@ -822,7 +822,14 @@ async fn serve<R: FullRepo, S: Store + 'static>(
|
||||||
(hash, alias, true)
|
(hash, alias, true)
|
||||||
};
|
};
|
||||||
|
|
||||||
let identifier = repo.identifier(hash).await?;
|
let Some(identifier) = repo.identifier(hash.clone()).await? else {
|
||||||
|
tracing::warn!(
|
||||||
|
"Original File identifier for hash {} is missing, queue cleanup task",
|
||||||
|
hex::encode(&hash)
|
||||||
|
);
|
||||||
|
crate::queue::cleanup_hash(&repo, hash).await?;
|
||||||
|
return Ok(HttpResponse::NotFound().finish());
|
||||||
|
};
|
||||||
|
|
||||||
let details = ensure_details(&repo, &store, &alias).await?;
|
let details = ensure_details(&repo, &store, &alias).await?;
|
||||||
|
|
||||||
|
@ -1548,8 +1555,8 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
let original_identifier = match repo.identifier(hash.as_ref().to_vec().into()).await {
|
let original_identifier = match repo.identifier(hash.as_ref().to_vec().into()).await {
|
||||||
Ok(identifier) => identifier,
|
Ok(Some(identifier)) => identifier,
|
||||||
Err(e) if e.is_missing() => {
|
Ok(None) => {
|
||||||
tracing::warn!(
|
tracing::warn!(
|
||||||
"Original File identifier for hash {} is missing, queue cleanup task",
|
"Original File identifier for hash {} is missing, queue cleanup task",
|
||||||
hex::encode(&hash)
|
hex::encode(&hash)
|
||||||
|
|
|
@ -92,8 +92,10 @@ where
|
||||||
|
|
||||||
if !aliases.is_empty() {
|
if !aliases.is_empty() {
|
||||||
for alias in aliases {
|
for alias in aliases {
|
||||||
let token = repo.delete_token(&alias).await?;
|
// TODO: decide if it is okay to skip aliases without tokens
|
||||||
super::cleanup_alias(repo, alias, token).await?;
|
if let Some(token) = repo.delete_token(&alias).await? {
|
||||||
|
super::cleanup_alias(repo, alias, token).await?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Return after queueing cleanup alias, since we will be requeued when the last alias is cleaned
|
// Return after queueing cleanup alias, since we will be requeued when the last alias is cleaned
|
||||||
return Ok(());
|
return Ok(());
|
||||||
|
@ -105,7 +107,7 @@ where
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|(_, v)| v)
|
.map(|(_, v)| v)
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
idents.push(repo.identifier(hash.clone()).await?);
|
idents.extend(repo.identifier(hash.clone()).await?);
|
||||||
idents.extend(repo.motion_identifier(hash.clone()).await?);
|
idents.extend(repo.motion_identifier(hash.clone()).await?);
|
||||||
|
|
||||||
for identifier in idents {
|
for identifier in idents {
|
||||||
|
@ -123,7 +125,8 @@ where
|
||||||
R: FullRepo,
|
R: FullRepo,
|
||||||
{
|
{
|
||||||
let saved_delete_token = repo.delete_token(&alias).await?;
|
let saved_delete_token = repo.delete_token(&alias).await?;
|
||||||
if saved_delete_token != token {
|
|
||||||
|
if saved_delete_token.is_some() && saved_delete_token != Some(token) {
|
||||||
return Err(UploadError::InvalidToken.into());
|
return Err(UploadError::InvalidToken.into());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
47
src/repo.rs
47
src/repo.rs
|
@ -34,7 +34,10 @@ pub(crate) struct DeleteToken {
|
||||||
id: MaybeUuid,
|
id: MaybeUuid,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) struct AlreadyExists;
|
pub(crate) struct HashAlreadyExists;
|
||||||
|
pub(crate) struct AliasAlreadyExists;
|
||||||
|
|
||||||
|
pub(crate) struct AlreadyExists(pub(super) DeleteToken);
|
||||||
|
|
||||||
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
||||||
pub(crate) struct UploadId {
|
pub(crate) struct UploadId {
|
||||||
|
@ -56,15 +59,9 @@ pub(crate) enum RepoError {
|
||||||
|
|
||||||
#[error("Panic in blocking operation")]
|
#[error("Panic in blocking operation")]
|
||||||
Canceled,
|
Canceled,
|
||||||
}
|
|
||||||
|
|
||||||
impl RepoError {
|
#[error("Required field is missing {0}")]
|
||||||
pub(crate) const fn is_missing(&self) -> bool {
|
Missing(&'static str),
|
||||||
match self {
|
|
||||||
Self::SledError(e) => e.is_missing(),
|
|
||||||
_ => false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
|
@ -91,7 +88,7 @@ pub(crate) trait FullRepo:
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
};
|
};
|
||||||
|
|
||||||
self.identifier(hash).await.map(Some)
|
self.identifier(hash).await
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(skip(self))]
|
#[tracing::instrument(skip(self))]
|
||||||
|
@ -112,7 +109,9 @@ pub(crate) trait FullRepo:
|
||||||
return Ok(None);
|
return Ok(None);
|
||||||
};
|
};
|
||||||
|
|
||||||
let identifier = self.identifier::<I>(hash.clone()).await?;
|
let Some(identifier) = self.identifier::<I>(hash.clone()).await? else {
|
||||||
|
return Ok(None);
|
||||||
|
};
|
||||||
|
|
||||||
match self.details(&identifier).await? {
|
match self.details(&identifier).await? {
|
||||||
Some(details) if details.is_motion() => self.motion_identifier::<I>(hash).await,
|
Some(details) if details.is_motion() => self.motion_identifier::<I>(hash).await,
|
||||||
|
@ -270,7 +269,7 @@ pub(crate) trait HashRepo: BaseRepo {
|
||||||
|
|
||||||
async fn hashes(&self) -> Self::Stream;
|
async fn hashes(&self) -> Self::Stream;
|
||||||
|
|
||||||
async fn create(&self, hash: Self::Bytes) -> Result<Result<(), AlreadyExists>, RepoError>;
|
async fn create(&self, hash: Self::Bytes) -> Result<Result<(), HashAlreadyExists>, RepoError>;
|
||||||
|
|
||||||
async fn relate_alias(&self, hash: Self::Bytes, alias: &Alias) -> Result<(), RepoError>;
|
async fn relate_alias(&self, hash: Self::Bytes, alias: &Alias) -> Result<(), RepoError>;
|
||||||
async fn remove_alias(&self, hash: Self::Bytes, alias: &Alias) -> Result<(), RepoError>;
|
async fn remove_alias(&self, hash: Self::Bytes, alias: &Alias) -> Result<(), RepoError>;
|
||||||
|
@ -281,8 +280,10 @@ pub(crate) trait HashRepo: BaseRepo {
|
||||||
hash: Self::Bytes,
|
hash: Self::Bytes,
|
||||||
identifier: &I,
|
identifier: &I,
|
||||||
) -> Result<(), StoreError>;
|
) -> Result<(), StoreError>;
|
||||||
async fn identifier<I: Identifier + 'static>(&self, hash: Self::Bytes)
|
async fn identifier<I: Identifier + 'static>(
|
||||||
-> Result<I, StoreError>;
|
&self,
|
||||||
|
hash: Self::Bytes,
|
||||||
|
) -> Result<Option<I>, StoreError>;
|
||||||
|
|
||||||
async fn relate_variant_identifier<I: Identifier>(
|
async fn relate_variant_identifier<I: Identifier>(
|
||||||
&self,
|
&self,
|
||||||
|
@ -329,7 +330,7 @@ where
|
||||||
T::hashes(self).await
|
T::hashes(self).await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn create(&self, hash: Self::Bytes) -> Result<Result<(), AlreadyExists>, RepoError> {
|
async fn create(&self, hash: Self::Bytes) -> Result<Result<(), HashAlreadyExists>, RepoError> {
|
||||||
T::create(self, hash).await
|
T::create(self, hash).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -356,7 +357,7 @@ where
|
||||||
async fn identifier<I: Identifier + 'static>(
|
async fn identifier<I: Identifier + 'static>(
|
||||||
&self,
|
&self,
|
||||||
hash: Self::Bytes,
|
hash: Self::Bytes,
|
||||||
) -> Result<I, StoreError> {
|
) -> Result<Option<I>, StoreError> {
|
||||||
T::identifier(self, hash).await
|
T::identifier(self, hash).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -410,14 +411,14 @@ where
|
||||||
|
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
pub(crate) trait AliasRepo: BaseRepo {
|
pub(crate) trait AliasRepo: BaseRepo {
|
||||||
async fn create(&self, alias: &Alias) -> Result<Result<(), AlreadyExists>, RepoError>;
|
async fn create(&self, alias: &Alias) -> Result<Result<(), AliasAlreadyExists>, RepoError>;
|
||||||
|
|
||||||
async fn relate_delete_token(
|
async fn relate_delete_token(
|
||||||
&self,
|
&self,
|
||||||
alias: &Alias,
|
alias: &Alias,
|
||||||
delete_token: &DeleteToken,
|
delete_token: &DeleteToken,
|
||||||
) -> Result<Result<(), AlreadyExists>, RepoError>;
|
) -> Result<Result<(), AlreadyExists>, RepoError>;
|
||||||
async fn delete_token(&self, alias: &Alias) -> Result<DeleteToken, RepoError>;
|
async fn delete_token(&self, alias: &Alias) -> Result<Option<DeleteToken>, RepoError>;
|
||||||
|
|
||||||
async fn relate_hash(&self, alias: &Alias, hash: Self::Bytes) -> Result<(), RepoError>;
|
async fn relate_hash(&self, alias: &Alias, hash: Self::Bytes) -> Result<(), RepoError>;
|
||||||
async fn hash(&self, alias: &Alias) -> Result<Option<Self::Bytes>, RepoError>;
|
async fn hash(&self, alias: &Alias) -> Result<Option<Self::Bytes>, RepoError>;
|
||||||
|
@ -430,7 +431,7 @@ impl<T> AliasRepo for actix_web::web::Data<T>
|
||||||
where
|
where
|
||||||
T: AliasRepo,
|
T: AliasRepo,
|
||||||
{
|
{
|
||||||
async fn create(&self, alias: &Alias) -> Result<Result<(), AlreadyExists>, RepoError> {
|
async fn create(&self, alias: &Alias) -> Result<Result<(), AliasAlreadyExists>, RepoError> {
|
||||||
T::create(self, alias).await
|
T::create(self, alias).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -442,7 +443,7 @@ where
|
||||||
T::relate_delete_token(self, alias, delete_token).await
|
T::relate_delete_token(self, alias, delete_token).await
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn delete_token(&self, alias: &Alias) -> Result<DeleteToken, RepoError> {
|
async fn delete_token(&self, alias: &Alias) -> Result<Option<DeleteToken>, RepoError> {
|
||||||
T::delete_token(self, alias).await
|
T::delete_token(self, alias).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -612,7 +613,11 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let main_identifier = repo.identifier::<FileId>(hash.clone()).await?;
|
let Some(main_identifier) = repo.identifier::<FileId>(hash.clone()).await? else {
|
||||||
|
tracing::warn!("Missing identifier for hash {}, queueing cleanup", hex::encode(&hash));
|
||||||
|
crate::queue::cleanup_hash(repo, hash.clone()).await?;
|
||||||
|
return Err(RepoError::Missing("hash -> identifier").into());
|
||||||
|
};
|
||||||
|
|
||||||
if let Some(new_main_identifier) = main_identifier.normalize_for_migration() {
|
if let Some(new_main_identifier) = main_identifier.normalize_for_migration() {
|
||||||
migrate_identifier_details(repo, &main_identifier, &new_main_identifier).await?;
|
migrate_identifier_details(repo, &main_identifier, &new_main_identifier).await?;
|
||||||
|
|
|
@ -1,14 +1,15 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
repo::{
|
repo::{
|
||||||
Alias, AliasRepo, AlreadyExists, BaseRepo, DeleteToken, Details, FullRepo, HashRepo,
|
Alias, AliasAlreadyExists, AliasRepo, AlreadyExists, BaseRepo, DeleteToken, Details,
|
||||||
Identifier, IdentifierRepo, QueueRepo, SettingsRepo, UploadId, UploadRepo, UploadResult,
|
FullRepo, HashAlreadyExists, HashRepo, Identifier, IdentifierRepo, QueueRepo, SettingsRepo,
|
||||||
|
UploadId, UploadRepo, UploadResult,
|
||||||
},
|
},
|
||||||
serde_str::Serde,
|
serde_str::Serde,
|
||||||
store::StoreError,
|
store::StoreError,
|
||||||
stream::from_iterator,
|
stream::from_iterator,
|
||||||
};
|
};
|
||||||
use futures_util::Stream;
|
use futures_util::Stream;
|
||||||
use sled::{Db, IVec, Tree};
|
use sled::{CompareAndSwapError, Db, IVec, Tree};
|
||||||
use std::{
|
use std::{
|
||||||
collections::HashMap,
|
collections::HashMap,
|
||||||
pin::Pin,
|
pin::Pin,
|
||||||
|
@ -44,19 +45,10 @@ pub(crate) enum SledError {
|
||||||
#[error("Invalid details json")]
|
#[error("Invalid details json")]
|
||||||
Details(#[from] serde_json::Error),
|
Details(#[from] serde_json::Error),
|
||||||
|
|
||||||
#[error("Required field was not present: {0}")]
|
|
||||||
Missing(&'static str),
|
|
||||||
|
|
||||||
#[error("Operation panicked")]
|
#[error("Operation panicked")]
|
||||||
Panic,
|
Panic,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SledError {
|
|
||||||
pub(super) const fn is_missing(&self) -> bool {
|
|
||||||
matches!(self, Self::Missing(_))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
pub(crate) struct SledRepo {
|
pub(crate) struct SledRepo {
|
||||||
healthz_count: Arc<AtomicU64>,
|
healthz_count: Arc<AtomicU64>,
|
||||||
|
@ -456,13 +448,13 @@ impl HashRepo for SledRepo {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(level = "trace", skip(self, hash), fields(hash = hex::encode(&hash)))]
|
#[tracing::instrument(level = "trace", skip(self, hash), fields(hash = hex::encode(&hash)))]
|
||||||
async fn create(&self, hash: Self::Bytes) -> Result<Result<(), AlreadyExists>, RepoError> {
|
async fn create(&self, hash: Self::Bytes) -> Result<Result<(), HashAlreadyExists>, RepoError> {
|
||||||
let res = b!(self.hashes, {
|
let res = b!(self.hashes, {
|
||||||
let hash2 = hash.clone();
|
let hash2 = hash.clone();
|
||||||
hashes.compare_and_swap(hash, None as Option<Self::Bytes>, Some(hash2))
|
hashes.compare_and_swap(hash, None as Option<Self::Bytes>, Some(hash2))
|
||||||
});
|
});
|
||||||
|
|
||||||
Ok(res.map_err(|_| AlreadyExists))
|
Ok(res.map_err(|_| HashAlreadyExists))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(level = "trace", skip(self, hash), fields(hash = hex::encode(&hash)))]
|
#[tracing::instrument(level = "trace", skip(self, hash), fields(hash = hex::encode(&hash)))]
|
||||||
|
@ -515,13 +507,12 @@ impl HashRepo for SledRepo {
|
||||||
async fn identifier<I: Identifier + 'static>(
|
async fn identifier<I: Identifier + 'static>(
|
||||||
&self,
|
&self,
|
||||||
hash: Self::Bytes,
|
hash: Self::Bytes,
|
||||||
) -> Result<I, StoreError> {
|
) -> Result<Option<I>, StoreError> {
|
||||||
let opt = b!(self.hash_identifiers, hash_identifiers.get(hash));
|
let Some(ivec) = b!(self.hash_identifiers, hash_identifiers.get(hash)) else {
|
||||||
|
return Ok(None);
|
||||||
|
};
|
||||||
|
|
||||||
opt.ok_or(SledError::Missing("hash -> identifier"))
|
Ok(Some(I::from_bytes(ivec.to_vec())?))
|
||||||
.map_err(RepoError::from)
|
|
||||||
.map_err(StoreError::from)
|
|
||||||
.and_then(|ivec| I::from_bytes(ivec.to_vec()))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(level = "trace", skip(self, hash, identifier), fields(hash = hex::encode(&hash), identifier = identifier.string_repr()))]
|
#[tracing::instrument(level = "trace", skip(self, hash, identifier), fields(hash = hex::encode(&hash), identifier = identifier.string_repr()))]
|
||||||
|
@ -679,7 +670,7 @@ impl HashRepo for SledRepo {
|
||||||
#[async_trait::async_trait(?Send)]
|
#[async_trait::async_trait(?Send)]
|
||||||
impl AliasRepo for SledRepo {
|
impl AliasRepo for SledRepo {
|
||||||
#[tracing::instrument(level = "trace", skip(self))]
|
#[tracing::instrument(level = "trace", skip(self))]
|
||||||
async fn create(&self, alias: &Alias) -> Result<Result<(), AlreadyExists>, RepoError> {
|
async fn create(&self, alias: &Alias) -> Result<Result<(), AliasAlreadyExists>, RepoError> {
|
||||||
let bytes = alias.to_bytes();
|
let bytes = alias.to_bytes();
|
||||||
let bytes2 = bytes.clone();
|
let bytes2 = bytes.clone();
|
||||||
|
|
||||||
|
@ -688,7 +679,7 @@ impl AliasRepo for SledRepo {
|
||||||
aliases.compare_and_swap(bytes, None as Option<Self::Bytes>, Some(bytes2))
|
aliases.compare_and_swap(bytes, None as Option<Self::Bytes>, Some(bytes2))
|
||||||
);
|
);
|
||||||
|
|
||||||
Ok(res.map_err(|_| AlreadyExists))
|
Ok(res.map_err(|_| AliasAlreadyExists))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(level = "trace", skip(self))]
|
#[tracing::instrument(level = "trace", skip(self))]
|
||||||
|
@ -700,23 +691,50 @@ impl AliasRepo for SledRepo {
|
||||||
let key = alias.to_bytes();
|
let key = alias.to_bytes();
|
||||||
let token = delete_token.to_bytes();
|
let token = delete_token.to_bytes();
|
||||||
|
|
||||||
let res = b!(
|
let res = b!(self.alias_delete_tokens, {
|
||||||
self.alias_delete_tokens,
|
let mut prev: Option<Self::Bytes> = None;
|
||||||
alias_delete_tokens.compare_and_swap(key, None as Option<Self::Bytes>, Some(token))
|
|
||||||
);
|
|
||||||
|
|
||||||
Ok(res.map_err(|_| AlreadyExists))
|
loop {
|
||||||
|
let key = key.clone();
|
||||||
|
let token = token.clone();
|
||||||
|
|
||||||
|
let res = alias_delete_tokens.compare_and_swap(key, prev, Some(token))?;
|
||||||
|
|
||||||
|
match res {
|
||||||
|
Ok(()) => return Ok(Ok(())) as Result<_, SledError>,
|
||||||
|
Err(CompareAndSwapError {
|
||||||
|
current: Some(token),
|
||||||
|
..
|
||||||
|
}) => {
|
||||||
|
if let Some(token) = DeleteToken::from_slice(&token) {
|
||||||
|
return Ok(Err(AlreadyExists(token)));
|
||||||
|
} else {
|
||||||
|
prev = Some(token);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Err(CompareAndSwapError { current: None, .. }) => {
|
||||||
|
prev = None;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(res)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(level = "trace", skip(self))]
|
#[tracing::instrument(level = "trace", skip(self))]
|
||||||
async fn delete_token(&self, alias: &Alias) -> Result<DeleteToken, RepoError> {
|
async fn delete_token(&self, alias: &Alias) -> Result<Option<DeleteToken>, RepoError> {
|
||||||
let key = alias.to_bytes();
|
let key = alias.to_bytes();
|
||||||
|
|
||||||
let opt = b!(self.alias_delete_tokens, alias_delete_tokens.get(key));
|
let Some(ivec) = b!(self.alias_delete_tokens, alias_delete_tokens.get(key)) else {
|
||||||
|
return Ok(None);
|
||||||
|
};
|
||||||
|
|
||||||
opt.and_then(|ivec| DeleteToken::from_slice(&ivec))
|
let Some(token) = DeleteToken::from_slice(&ivec) else {
|
||||||
.ok_or(SledError::Missing("alias -> delete-token"))
|
return Ok(None);
|
||||||
.map_err(RepoError::from)
|
};
|
||||||
|
|
||||||
|
Ok(Some(token))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(level = "trace", skip(self, hash), fields(hash = hex::encode(&hash)))]
|
#[tracing::instrument(level = "trace", skip(self, hash), fields(hash = hex::encode(&hash)))]
|
||||||
|
|
|
@ -28,13 +28,6 @@ impl StoreError {
|
||||||
pub(crate) const fn is_not_found(&self) -> bool {
|
pub(crate) const fn is_not_found(&self) -> bool {
|
||||||
matches!(self, Self::FileNotFound(_)) || matches!(self, Self::ObjectNotFound(_))
|
matches!(self, Self::FileNotFound(_)) || matches!(self, Self::ObjectNotFound(_))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) const fn is_missing(&self) -> bool {
|
|
||||||
match self {
|
|
||||||
Self::Repo(e) => e.is_missing(),
|
|
||||||
_ => false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<crate::store::file_store::FileError> for StoreError {
|
impl From<crate::store::file_store::FileError> for StoreError {
|
||||||
|
|
Loading…
Reference in a new issue