mirror of
https://git.asonix.dog/asonix/pict-rs
synced 2025-01-05 17:21:28 +00:00
Keep track of image variants rather than walking fs on delete
This commit is contained in:
parent
f9f77c3c57
commit
1fbb5005ef
4 changed files with 109 additions and 30 deletions
|
@ -1,7 +1,7 @@
|
||||||
[package]
|
[package]
|
||||||
name = "pict-rs"
|
name = "pict-rs"
|
||||||
description = "A simple image hosting service"
|
description = "A simple image hosting service"
|
||||||
version = "0.1.1"
|
version = "0.1.2"
|
||||||
authors = ["asonix <asonix@asonix.dog>"]
|
authors = ["asonix <asonix@asonix.dog>"]
|
||||||
license = "AGPL-3.0"
|
license = "AGPL-3.0"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
|
|
|
@ -58,6 +58,9 @@ pub enum UploadError {
|
||||||
|
|
||||||
#[error("No filename provided in request")]
|
#[error("No filename provided in request")]
|
||||||
MissingFilename,
|
MissingFilename,
|
||||||
|
|
||||||
|
#[error("Error converting Path to String")]
|
||||||
|
Path,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<actix_web::client::SendRequestError> for UploadError {
|
impl From<actix_web::client::SendRequestError> for UploadError {
|
||||||
|
|
|
@ -213,6 +213,11 @@ async fn serve(
|
||||||
|
|
||||||
// Save the file in another task, we want to return the thumbnail now
|
// Save the file in another task, we want to return the thumbnail now
|
||||||
actix_rt::spawn(async move {
|
actix_rt::spawn(async move {
|
||||||
|
if let Err(e) = manager.store_variant(path2.clone()).await {
|
||||||
|
error!("Error storing variant, {}", e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if let Err(e) = safe_save_file(path2, img_bytes2).await {
|
if let Err(e) = safe_save_file(path2, img_bytes2).await {
|
||||||
error!("Error saving file, {}", e);
|
error!("Error saving file, {}", e);
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,7 @@ struct UploadManagerInner {
|
||||||
hasher: sha2::Sha256,
|
hasher: sha2::Sha256,
|
||||||
image_dir: PathBuf,
|
image_dir: PathBuf,
|
||||||
alias_tree: sled::Tree,
|
alias_tree: sled::Tree,
|
||||||
|
filename_tree: sled::Tree,
|
||||||
db: sled::Db,
|
db: sled::Db,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -61,11 +62,32 @@ impl UploadManager {
|
||||||
hasher: sha2::Sha256::new(),
|
hasher: sha2::Sha256::new(),
|
||||||
image_dir: root_dir,
|
image_dir: root_dir,
|
||||||
alias_tree: db.open_tree("alias")?,
|
alias_tree: db.open_tree("alias")?,
|
||||||
|
filename_tree: db.open_tree("filename")?,
|
||||||
db,
|
db,
|
||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn store_variant(&self, path: PathBuf) -> Result<(), UploadError> {
|
||||||
|
let filename = path
|
||||||
|
.file_name()
|
||||||
|
.and_then(|f| f.to_str())
|
||||||
|
.map(|s| s.to_string())
|
||||||
|
.ok_or(UploadError::Path)?;
|
||||||
|
let path_string = path.to_str().ok_or(UploadError::Path)?.to_string();
|
||||||
|
|
||||||
|
let fname_tree = self.inner.filename_tree.clone();
|
||||||
|
let hash: sled::IVec = web::block(move || fname_tree.get(filename.as_bytes()))
|
||||||
|
.await?
|
||||||
|
.ok_or(UploadError::MissingFilename)?;
|
||||||
|
|
||||||
|
let key = variant_key(&hash, &path_string);
|
||||||
|
let db = self.inner.db.clone();
|
||||||
|
web::block(move || db.insert(key, path_string.as_bytes())).await?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) async fn delete(&self, alias: String, token: String) -> Result<(), UploadError> {
|
pub(crate) async fn delete(&self, alias: String, token: String) -> Result<(), UploadError> {
|
||||||
use sled::Transactional;
|
use sled::Transactional;
|
||||||
let db = self.inner.db.clone();
|
let db = self.inner.db.clone();
|
||||||
|
@ -121,22 +143,15 @@ impl UploadManager {
|
||||||
|
|
||||||
// -- DELETE HASH ENTRY --
|
// -- DELETE HASH ENTRY --
|
||||||
let db = self.inner.db.clone();
|
let db = self.inner.db.clone();
|
||||||
let real_filename = web::block(move || {
|
let hash2 = hash.clone();
|
||||||
let real_filename = db.remove(&hash)?.ok_or(UploadError::MissingFile)?;
|
let filename = web::block(move || db.remove(&hash2))
|
||||||
|
.await?
|
||||||
Ok(real_filename) as Result<sled::IVec, UploadError>
|
.ok_or(UploadError::MissingFile)?;
|
||||||
})
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
let real_filename = String::from_utf8(real_filename.to_vec())?;
|
|
||||||
|
|
||||||
let image_dir = self.image_dir();
|
|
||||||
|
|
||||||
// -- DELETE FILES --
|
// -- DELETE FILES --
|
||||||
|
let this = self.clone();
|
||||||
actix_rt::spawn(async move {
|
actix_rt::spawn(async move {
|
||||||
if let Err(e) =
|
if let Err(e) = this.cleanup_files(filename).await {
|
||||||
web::block(move || blocking_delete_all_by_filename(image_dir, &real_filename)).await
|
|
||||||
{
|
|
||||||
error!("Error removing files from fs, {}", e);
|
error!("Error removing files from fs, {}", e);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -144,6 +159,48 @@ impl UploadManager {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn cleanup_files(&self, filename: sled::IVec) -> Result<(), UploadError> {
|
||||||
|
let mut path = self.image_dir();
|
||||||
|
let fname = String::from_utf8(filename.to_vec())?;
|
||||||
|
path.push(fname);
|
||||||
|
|
||||||
|
let mut errors = Vec::new();
|
||||||
|
if let Err(e) = actix_fs::remove_file(path).await {
|
||||||
|
errors.push(e.into());
|
||||||
|
}
|
||||||
|
|
||||||
|
let fname_tree = self.inner.filename_tree.clone();
|
||||||
|
let hash = web::block(move || fname_tree.remove(filename))
|
||||||
|
.await?
|
||||||
|
.ok_or(UploadError::MissingFile)?;
|
||||||
|
|
||||||
|
let (start, end) = variant_key_bounds(&hash);
|
||||||
|
let db = self.inner.db.clone();
|
||||||
|
let keys = web::block(move || {
|
||||||
|
let mut keys = Vec::new();
|
||||||
|
for key in db.range(start..end).keys() {
|
||||||
|
keys.push(key?.to_owned());
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(keys) as Result<Vec<sled::IVec>, UploadError>
|
||||||
|
})
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
for key in keys {
|
||||||
|
let db = self.inner.db.clone();
|
||||||
|
if let Some(path) = web::block(move || db.remove(key)).await? {
|
||||||
|
if let Err(e) = remove_path(path).await {
|
||||||
|
errors.push(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for error in errors {
|
||||||
|
error!("Error deleting files, {}", error);
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
/// Generate a delete token for an alias
|
/// Generate a delete token for an alias
|
||||||
pub(crate) async fn delete_token(&self, alias: String) -> Result<String, UploadError> {
|
pub(crate) async fn delete_token(&self, alias: String) -> Result<String, UploadError> {
|
||||||
use rand::distributions::{Alphanumeric, Distribution};
|
use rand::distributions::{Alphanumeric, Distribution};
|
||||||
|
@ -279,8 +336,13 @@ impl UploadManager {
|
||||||
|
|
||||||
let filename = self.next_file(content_type).await?;
|
let filename = self.next_file(content_type).await?;
|
||||||
let filename2 = filename.clone();
|
let filename2 = filename.clone();
|
||||||
|
let hash2 = hash.clone();
|
||||||
let res = web::block(move || {
|
let res = web::block(move || {
|
||||||
db.compare_and_swap(hash, None as Option<sled::IVec>, Some(filename2.as_bytes()))
|
db.compare_and_swap(
|
||||||
|
hash2,
|
||||||
|
None as Option<sled::IVec>,
|
||||||
|
Some(filename2.as_bytes()),
|
||||||
|
)
|
||||||
})
|
})
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
|
@ -293,6 +355,10 @@ impl UploadManager {
|
||||||
return Ok((Dup::Exists, name));
|
return Ok((Dup::Exists, name));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let fname_tree = self.inner.filename_tree.clone();
|
||||||
|
let filename2 = filename.clone();
|
||||||
|
web::block(move || fname_tree.insert(filename2, hash)).await?;
|
||||||
|
|
||||||
Ok((Dup::New, filename))
|
Ok((Dup::New, filename))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -386,21 +452,9 @@ impl UploadManager {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn blocking_delete_all_by_filename(mut dir: PathBuf, filename: &str) -> Result<(), UploadError> {
|
async fn remove_path(path: sled::IVec) -> Result<(), UploadError> {
|
||||||
for res in std::fs::read_dir(dir.clone())? {
|
let path_string = String::from_utf8(path.to_vec())?;
|
||||||
let entry = res?;
|
actix_fs::remove_file(path_string).await?;
|
||||||
|
|
||||||
if entry.path().is_dir() {
|
|
||||||
blocking_delete_all_by_filename(entry.path(), filename)?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dir.push(filename);
|
|
||||||
|
|
||||||
if dir.is_file() {
|
|
||||||
std::fs::remove_file(dir)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -439,6 +493,23 @@ fn delete_key(alias: &str) -> String {
|
||||||
format!("{}/delete", alias)
|
format!("{}/delete", alias)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn variant_key(hash: &[u8], path: &str) -> Vec<u8> {
|
||||||
|
let mut key = hash.to_vec();
|
||||||
|
key.extend(&[2]);
|
||||||
|
key.extend(path.as_bytes());
|
||||||
|
key
|
||||||
|
}
|
||||||
|
|
||||||
|
fn variant_key_bounds(hash: &[u8]) -> (Vec<u8>, Vec<u8>) {
|
||||||
|
let mut start = hash.to_vec();
|
||||||
|
start.extend(&[2]);
|
||||||
|
|
||||||
|
let mut end = hash.to_vec();
|
||||||
|
end.extend(&[3]);
|
||||||
|
|
||||||
|
(start, end)
|
||||||
|
}
|
||||||
|
|
||||||
fn valid_format(format: image::ImageFormat) -> Result<mime::Mime, UploadError> {
|
fn valid_format(format: image::ImageFormat) -> Result<mime::Mime, UploadError> {
|
||||||
match format {
|
match format {
|
||||||
image::ImageFormat::Jpeg => Ok(mime::IMAGE_JPEG),
|
image::ImageFormat::Jpeg => Ok(mime::IMAGE_JPEG),
|
||||||
|
|
Loading…
Reference in a new issue