2
0
Fork 0
mirror of https://git.asonix.dog/asonix/pict-rs synced 2025-01-08 18:51:24 +00:00

Keep track of image variants rather than walking fs on delete

This commit is contained in:
asonix 2020-06-09 18:39:49 -05:00
parent f9f77c3c57
commit 1fbb5005ef
4 changed files with 109 additions and 30 deletions

View file

@ -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"

View file

@ -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 {

View file

@ -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);
} }

View file

@ -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),