You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

72 lines
2.3 KiB

use log::*;
use std::path::Path;
use tempfile::TempDir;
/// A "work-in-progress" directory, which will "teleport" into the path
/// given in `target` only on successful, guarding from invalid state in the file system.
///
/// Underneath, it uses `fs::rename`, so make sure to make the `temp_dir` inside the same
/// mount as `target`. This is why we have the `new_in` constructor.
pub struct DirectoryPortal<P: AsRef<Path>> {
temp_dir: TempDir,
target: P,
}
impl<P: AsRef<Path>> DirectoryPortal<P> {
/// Create a new portal which will keep the temp files in
/// a subdirectory of `parent_dir` until teleporting to `target`.
#[must_use]
pub fn new_in(parent_dir: impl AsRef<Path>, target: P) -> Self {
let temp_dir = TempDir::new_in(parent_dir).expect("Can't generate a temp directory");
debug!("Created a temp directory in {:?}", temp_dir.path());
Self { target, temp_dir }
}
pub fn teleport(self) -> std::io::Result<P> {
debug!(
"Moving directory {:?} into {:?}",
self.temp_dir.path(),
self.target.as_ref()
);
std::fs::rename(&self.temp_dir, &self.target)?;
Ok(self.target)
}
}
impl<P: AsRef<Path>> std::ops::Deref for DirectoryPortal<P> {
type Target = Path;
fn deref(&self) -> &Self::Target {
self.as_ref()
}
}
impl<P: AsRef<Path>> AsRef<Path> for DirectoryPortal<P> {
fn as_ref(&self) -> &Path {
self.temp_dir.as_ref()
}
}
#[cfg(test)]
mod tests {
use super::*;
use pretty_assertions::assert_eq;
use tempfile::tempdir;
#[test_env_log::test]
fn test_portal() {
let tempdir = tempdir().expect("Can't generate a temp directory");
let portal = DirectoryPortal::new_in(std::env::temp_dir(), tempdir.path().join("subdir"));
let new_file_path = portal.to_path_buf().join("README.md");
std::fs::write(&new_file_path, "Hello world!").expect("Can't write file");
let target = portal.teleport().expect("Can't close directory portal");
let file_exists: Vec<_> = target
.read_dir()
.expect("Can't read dir")
.map(|x| x.unwrap().file_name().into_string().unwrap())
.collect();
assert_eq!(file_exists, vec!["README.md"]);
}
}