2023-10-07 16:32:36 +00:00
|
|
|
use std::{path::PathBuf, sync::Arc};
|
2021-10-23 19:14:12 +00:00
|
|
|
use tokio::io::AsyncRead;
|
|
|
|
use uuid::Uuid;
|
|
|
|
|
2023-10-07 00:42:24 +00:00
|
|
|
pub(crate) type ArcTmpDir = Arc<TmpDir>;
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub(crate) struct TmpDir {
|
2023-10-07 16:36:49 +00:00
|
|
|
path: Option<PathBuf>,
|
2023-10-07 00:42:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TmpDir {
|
|
|
|
pub(crate) async fn init() -> std::io::Result<Arc<Self>> {
|
|
|
|
let path = std::env::temp_dir().join(Uuid::new_v4().to_string());
|
|
|
|
tokio::fs::create_dir(&path).await?;
|
2023-10-07 16:36:49 +00:00
|
|
|
Ok(Arc::new(TmpDir { path: Some(path) }))
|
2023-10-07 00:42:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn tmp_file(&self, ext: Option<&str>) -> PathBuf {
|
|
|
|
if let Some(ext) = ext {
|
2023-10-07 16:36:49 +00:00
|
|
|
self.path
|
|
|
|
.as_ref()
|
|
|
|
.expect("tmp path exists")
|
|
|
|
.join(format!("{}{}", Uuid::new_v4(), ext))
|
2023-10-07 00:42:24 +00:00
|
|
|
} else {
|
2023-10-07 16:36:49 +00:00
|
|
|
self.path
|
|
|
|
.as_ref()
|
|
|
|
.expect("tmp path exists")
|
|
|
|
.join(Uuid::new_v4().to_string())
|
2023-10-07 00:42:24 +00:00
|
|
|
}
|
|
|
|
}
|
2023-10-07 16:36:49 +00:00
|
|
|
|
|
|
|
pub(crate) async fn cleanup(self: Arc<Self>) -> std::io::Result<()> {
|
|
|
|
if let Some(path) = Arc::into_inner(self).and_then(|mut this| this.path.take()) {
|
|
|
|
tokio::fs::remove_dir_all(path).await?;
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
2023-10-07 00:42:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for TmpDir {
|
|
|
|
fn drop(&mut self) {
|
2023-10-07 16:36:49 +00:00
|
|
|
if let Some(path) = self.path.as_ref() {
|
|
|
|
std::fs::remove_dir_all(path).expect("Removed directory");
|
|
|
|
}
|
2023-10-07 00:42:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-23 19:14:12 +00:00
|
|
|
struct TmpFile(PathBuf);
|
|
|
|
|
|
|
|
impl Drop for TmpFile {
|
|
|
|
fn drop(&mut self) {
|
2023-10-21 00:08:11 +00:00
|
|
|
crate::sync::spawn("remove-tmpfile", tokio::fs::remove_file(self.0.clone()));
|
2021-10-23 19:14:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pin_project_lite::pin_project! {
|
|
|
|
pub(crate) struct TmpFileCleanup<R> {
|
|
|
|
#[pin]
|
|
|
|
inner: R,
|
|
|
|
|
|
|
|
file: TmpFile,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn cleanup_tmpfile<R: AsyncRead>(reader: R, file: PathBuf) -> TmpFileCleanup<R> {
|
|
|
|
TmpFileCleanup {
|
|
|
|
inner: reader,
|
|
|
|
file: TmpFile(file),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<R: AsyncRead> AsyncRead for TmpFileCleanup<R> {
|
|
|
|
fn poll_read(
|
|
|
|
mut self: std::pin::Pin<&mut Self>,
|
|
|
|
cx: &mut std::task::Context<'_>,
|
|
|
|
buf: &mut tokio::io::ReadBuf<'_>,
|
|
|
|
) -> std::task::Poll<std::io::Result<()>> {
|
|
|
|
let this = self.as_mut().project();
|
|
|
|
|
|
|
|
this.inner.poll_read(cx, buf)
|
|
|
|
}
|
|
|
|
}
|