use filetime::{set_file_mtime, FileTime}; use std::fs::{copy, create_dir_all, metadata, read_dir, File}; use std::io::prelude::*; use std::path::{Path, PathBuf}; use std::time::SystemTime; use walkdir::WalkDir; use errors::{Error, Result}; pub fn is_path_in_directory(parent: &Path, path: &Path) -> Result { let canonical_path = path .canonicalize() .map_err(|e| format!("Failed to canonicalize {}: {}", path.display(), e))?; let canonical_parent = parent .canonicalize() .map_err(|e| format!("Failed to canonicalize {}: {}", parent.display(), e))?; Ok(canonical_path.starts_with(canonical_parent)) } /// Create a file with the content given pub fn create_file(path: &Path, content: &str) -> Result<()> { let mut file = File::create(&path).map_err(|e| Error::chain(format!("Failed to create {:?}", path), e))?; file.write_all(content.as_bytes())?; Ok(()) } /// Create a directory at the given path if it doesn't exist already pub fn ensure_directory_exists(path: &Path) -> Result<()> { if !path.exists() { create_directory(path)?; } Ok(()) } /// Very similar to `create_dir` from the std except it checks if the folder /// exists before creating it pub fn create_directory(path: &Path) -> Result<()> { if !path.exists() { create_dir_all(path).map_err(|e| { Error::chain(format!("Was not able to create folder {}", path.display()), e) })?; } Ok(()) } /// Return the content of a file, with error handling added pub fn read_file(path: &Path) -> Result { let mut content = String::new(); File::open(path) .map_err(|e| Error::chain(format!("Failed to open '{:?}'", path.display()), e))? .read_to_string(&mut content)?; // Remove utf-8 BOM if any. if content.starts_with("\u{feff}") { content.drain(..3); } Ok(content) } /// Return the content of a file, with error handling added. /// The default error message is overwritten by the message given. /// That means it is allocation 2 strings, oh well pub fn read_file_with_error(path: &Path, message: &str) -> Result { let res = read_file(&path); if res.is_ok() { return res; } let mut err = Error::msg(message); err.source = res.unwrap_err().source; Err(err) } /// Looks into the current folder for the path and see if there's anything that is not a .md /// file. Those will be copied next to the rendered .html file pub fn find_related_assets(path: &Path) -> Vec { let mut assets = vec![]; for entry in read_dir(path).unwrap().filter_map(std::result::Result::ok) { let entry_path = entry.path(); if entry_path.is_file() { match entry_path.extension() { Some(e) => match e.to_str() { Some("md") => continue, _ => assets.push(entry_path.to_path_buf()), }, None => continue, } } } assets } /// Copy a file but takes into account where to start the copy as /// there might be folders we need to create on the way. /// No copy occurs if all of the following conditions are satisfied: /// 1. A file with the same name already exists in the dest path. /// 2. Its modification timestamp is identical to that of the src file. /// 3. Its filesize is identical to that of the src file. pub fn copy_file(src: &Path, dest: &PathBuf, base_path: &PathBuf, hard_link: bool) -> Result<()> { let relative_path = src.strip_prefix(base_path).unwrap(); let target_path = dest.join(relative_path); if let Some(parent_directory) = target_path.parent() { create_dir_all(parent_directory)?; } if hard_link { std::fs::hard_link(src, target_path)? } else { let src_metadata = metadata(src)?; let src_mtime = FileTime::from_last_modification_time(&src_metadata); if Path::new(&target_path).is_file() { let target_metadata = metadata(&target_path)?; let target_mtime = FileTime::from_last_modification_time(&target_metadata); if !(src_mtime == target_mtime && src_metadata.len() == target_metadata.len()) { copy(src, &target_path)?; set_file_mtime(&target_path, src_mtime)?; } } else { copy(src, &target_path)?; set_file_mtime(&target_path, src_mtime)?; } } Ok(()) } pub fn copy_directory(src: &PathBuf, dest: &PathBuf, hard_link: bool) -> Result<()> { for entry in WalkDir::new(src).into_iter().filter_map(std::result::Result::ok) { let relative_path = entry.path().strip_prefix(src).unwrap(); let target_path = dest.join(relative_path); if entry.path().is_dir() { if !target_path.exists() { create_directory(&target_path)?; } } else { copy_file(entry.path(), dest, src, hard_link)?; } } Ok(()) } pub fn get_file_time(path: &Path) -> Option { path.metadata().ok().and_then(|meta| { Some(match (meta.created().ok(), meta.modified().ok()) { (Some(tc), Some(tm)) => tc.max(tm), (Some(tc), None) => tc, (None, Some(tm)) => tm, (None, None) => return None, }) }) } /// Compares source and target files' timestamps and returns true if the source file /// has been created _or_ updated after the target file has pub fn file_stale(p_source: PS, p_target: PT) -> bool where PS: AsRef, PT: AsRef, { let p_source = p_source.as_ref(); let p_target = p_target.as_ref(); if !p_target.exists() { return true; } let time_source = get_file_time(p_source); let time_target = get_file_time(p_target); time_source.and_then(|ts| time_target.map(|tt| ts > tt)).unwrap_or(true) } #[cfg(test)] mod tests { use std::fs::{metadata, read_to_string, File}; use std::io::Write; use std::path::PathBuf; use std::str::FromStr; use tempfile::{tempdir, tempdir_in}; use super::{copy_file, find_related_assets}; #[test] fn can_find_related_assets() { let tmp_dir = tempdir().expect("create temp dir"); File::create(tmp_dir.path().join("index.md")).unwrap(); File::create(tmp_dir.path().join("example.js")).unwrap(); File::create(tmp_dir.path().join("graph.jpg")).unwrap(); File::create(tmp_dir.path().join("fail.png")).unwrap(); let assets = find_related_assets(tmp_dir.path()); assert_eq!(assets.len(), 3); assert_eq!(assets.iter().filter(|p| p.extension().unwrap() != "md").count(), 3); assert_eq!(assets.iter().filter(|p| p.file_name().unwrap() == "example.js").count(), 1); assert_eq!(assets.iter().filter(|p| p.file_name().unwrap() == "graph.jpg").count(), 1); assert_eq!(assets.iter().filter(|p| p.file_name().unwrap() == "fail.png").count(), 1); } #[test] fn test_copy_file_timestamp_preserved() { let base_path = PathBuf::from_str(env!("CARGO_MANIFEST_DIR")).unwrap(); let src_dir = tempdir_in(&base_path).expect("failed to create a temporary source directory."); let dest_dir = tempdir_in(&base_path).expect("failed to create a temporary destination directory."); let src_file_path = src_dir.path().join("test.txt"); let dest_file_path = dest_dir.path().join(src_file_path.strip_prefix(&base_path).unwrap()); File::create(&src_file_path).unwrap(); copy_file(&src_file_path, &dest_dir.path().to_path_buf(), &base_path, false).unwrap(); assert_eq!( metadata(&src_file_path).and_then(|m| m.modified()).unwrap(), metadata(&dest_file_path).and_then(|m| m.modified()).unwrap() ); } #[test] fn test_copy_file_already_exists() { let base_path = PathBuf::from_str(env!("CARGO_MANIFEST_DIR")).unwrap(); let src_dir = tempdir_in(&base_path).expect("failed to create a temporary source directory."); let dest_dir = tempdir_in(&base_path).expect("failed to create a temporary destination directory."); let src_file_path = src_dir.path().join("test.txt"); let dest_file_path = dest_dir.path().join(src_file_path.strip_prefix(&base_path).unwrap()); { let mut src_file = File::create(&src_file_path).unwrap(); src_file.write_all(b"file1").unwrap(); } copy_file(&src_file_path, &dest_dir.path().to_path_buf(), &base_path, false).unwrap(); { let mut dest_file = File::create(&dest_file_path).unwrap(); dest_file.write_all(b"file2").unwrap(); } // Check copy does not occur when moditication timestamps and filesizes are same. filetime::set_file_mtime(&src_file_path, filetime::FileTime::from_unix_time(0, 0)).unwrap(); filetime::set_file_mtime(&dest_file_path, filetime::FileTime::from_unix_time(0, 0)) .unwrap(); copy_file(&src_file_path, &dest_dir.path().to_path_buf(), &base_path, false).unwrap(); assert_eq!(read_to_string(&src_file_path).unwrap(), "file1"); assert_eq!(read_to_string(&dest_file_path).unwrap(), "file2"); // Copy occurs if the timestamps are different while the filesizes are same. filetime::set_file_mtime(&dest_file_path, filetime::FileTime::from_unix_time(42, 42)) .unwrap(); copy_file(&src_file_path, &dest_dir.path().to_path_buf(), &base_path, false).unwrap(); assert_eq!(read_to_string(&src_file_path).unwrap(), "file1"); assert_eq!(read_to_string(&dest_file_path).unwrap(), "file1"); // Copy occurs if the timestamps are same while the filesizes are different. { let mut dest_file = File::create(&dest_file_path).unwrap(); dest_file.write_all(b"This file has different file size to the source file!").unwrap(); } filetime::set_file_mtime(&dest_file_path, filetime::FileTime::from_unix_time(0, 0)) .unwrap(); copy_file(&src_file_path, &dest_dir.path().to_path_buf(), &base_path, false).unwrap(); assert_eq!(read_to_string(&src_file_path).unwrap(), "file1"); assert_eq!(read_to_string(&dest_file_path).unwrap(), "file1"); } }