diff --git a/src/etcetera.rs b/src/etcetera.rs new file mode 100644 index 0000000..e5a6fff --- /dev/null +++ b/src/etcetera.rs @@ -0,0 +1,163 @@ +use crate::error::{Error, Result}; + +pub mod base_strategy { + use crate::error::Result; + use std::path::PathBuf; + + pub trait BaseStrategy { + fn cache_dir(&self) -> PathBuf; + } + + macro_rules! create_strategies { + ($base: ty) => { + pub fn choose_base_strategy() -> Result<$base> { + <$base>::new() + } + }; + } + + cfg_if::cfg_if! { + if #[cfg(target_os = "windows")] { + create_strategies!(Windows); + } else if #[cfg(any(target_os = "macos", target_os = "ios"))] { + create_strategies!(Xdg); + } else { + create_strategies!(Xdg); + } + } + + mod windows { + use crate::error::Result; + use std::path::PathBuf; + + pub struct Windows { + home_dir: PathBuf, + } + + impl Windows { + pub fn new() -> Result { + Ok(Self { + home_dir: crate::etcetera::home_dir()?, + }) + } + + fn dir_inner(env: &'static str) -> Option { + std::env::var_os(env) + .filter(|s| !s.is_empty()) + .map(PathBuf::from) + .or_else(|| Self::dir_crt(env)) + } + + // Ref: https://github.com/rust-lang/cargo/blob/home-0.5.11/crates/home/src/windows.rs + // We should keep this code in sync with the above. + #[cfg(all(windows, not(target_vendor = "uwp")))] + fn dir_crt(env: &'static str) -> Option { + use std::ffi::OsString; + use std::os::windows::ffi::OsStringExt; + use std::ptr; + use std::slice; + + use windows_sys::Win32::Foundation::S_OK; + use windows_sys::Win32::System::Com::CoTaskMemFree; + use windows_sys::Win32::UI::Shell::{ + FOLDERID_LocalAppData, FOLDERID_RoamingAppData, SHGetKnownFolderPath, + KF_FLAG_DONT_VERIFY, + }; + + extern "C" { + fn wcslen(buf: *const u16) -> usize; + } + + let folder_id = match env { + "APPDATA" => FOLDERID_RoamingAppData, + "LOCALAPPDATA" => FOLDERID_LocalAppData, + _ => return None, + }; + + unsafe { + let mut path = ptr::null_mut(); + match SHGetKnownFolderPath( + &folder_id, + KF_FLAG_DONT_VERIFY as u32, + std::ptr::null_mut(), + &mut path, + ) { + S_OK => { + let path_slice = slice::from_raw_parts(path, wcslen(path)); + let s = OsString::from_wide(path_slice); + CoTaskMemFree(path.cast()); + Some(PathBuf::from(s)) + } + _ => { + // Free any allocated memory even on failure. A null ptr is a no-op for `CoTaskMemFree`. + CoTaskMemFree(path.cast()); + None + } + } + } + } + + #[cfg(not(all(windows, not(target_vendor = "uwp"))))] + fn dir_crt(_env: &'static str) -> Option { + None + } + } + + impl super::BaseStrategy for Windows { + fn cache_dir(&self) -> PathBuf { + Self::dir_inner("LOCALAPPDATA") + .unwrap_or_else(|| self.home_dir.join("AppData").join("Local")) + } + } + } + + mod xdg { + use crate::error::Result; + use std::path::Path; + use std::path::PathBuf; + + pub struct Xdg { + home_dir: PathBuf, + } + + impl Xdg { + pub fn new() -> Result { + Ok(Self { + home_dir: crate::etcetera::home_dir()?, + }) + } + + fn env_var_or_none(env_var: &str) -> Option { + std::env::var(env_var).ok().and_then(|path| { + let path = PathBuf::from(path); + + // Return None if the path obtained from the environment variable isn’t absolute. + if path.is_absolute() { + Some(path) + } else { + None + } + }) + } + + fn env_var_or_default(&self, env_var: &str, default: impl AsRef) -> PathBuf { + Self::env_var_or_none(env_var).unwrap_or_else(|| self.home_dir.join(default)) + } + } + + impl super::BaseStrategy for Xdg { + fn cache_dir(&self) -> PathBuf { + self.env_var_or_default("XDG_CACHE_HOME", ".cache/") + } + } + } + + pub use windows::Windows; + pub use xdg::Xdg; +} + +pub use base_strategy::{choose_base_strategy, BaseStrategy}; + +pub fn home_dir() -> Result { + home::home_dir().ok_or(Error::HomeDir) +} diff --git a/src/etcetera/base_strategy.rs b/src/etcetera/base_strategy.rs deleted file mode 100644 index f3ed44e..0000000 --- a/src/etcetera/base_strategy.rs +++ /dev/null @@ -1,30 +0,0 @@ -use crate::error::Result; -use std::path::PathBuf; - -pub trait BaseStrategy { - fn cache_dir(&self) -> PathBuf; -} - -macro_rules! create_strategies { - ($base: ty) => { - pub fn choose_base_strategy() -> Result<$base> { - <$base>::new() - } - }; -} - -cfg_if::cfg_if! { - if #[cfg(target_os = "windows")] { - create_strategies!(Windows); - } else if #[cfg(any(target_os = "macos", target_os = "ios"))] { - create_strategies!(Xdg); - } else { - create_strategies!(Xdg); - } -} - -mod windows; -mod xdg; - -pub use windows::Windows; -pub use xdg::Xdg; diff --git a/src/etcetera/base_strategy/windows.rs b/src/etcetera/base_strategy/windows.rs deleted file mode 100644 index e3756b9..0000000 --- a/src/etcetera/base_strategy/windows.rs +++ /dev/null @@ -1,82 +0,0 @@ -use crate::error::Result; -use std::path::PathBuf; - -pub struct Windows { - home_dir: PathBuf, -} - -impl Windows { - pub fn new() -> Result { - Ok(Self { - home_dir: crate::etcetera::home_dir()?, - }) - } - - fn dir_inner(env: &'static str) -> Option { - std::env::var_os(env) - .filter(|s| !s.is_empty()) - .map(PathBuf::from) - .or_else(|| Self::dir_crt(env)) - } - - // Ref: https://github.com/rust-lang/cargo/blob/home-0.5.11/crates/home/src/windows.rs - // We should keep this code in sync with the above. - #[cfg(all(windows, not(target_vendor = "uwp")))] - fn dir_crt(env: &'static str) -> Option { - use std::ffi::OsString; - use std::os::windows::ffi::OsStringExt; - use std::ptr; - use std::slice; - - use windows_sys::Win32::Foundation::S_OK; - use windows_sys::Win32::System::Com::CoTaskMemFree; - use windows_sys::Win32::UI::Shell::{ - FOLDERID_LocalAppData, FOLDERID_RoamingAppData, SHGetKnownFolderPath, - KF_FLAG_DONT_VERIFY, - }; - - extern "C" { - fn wcslen(buf: *const u16) -> usize; - } - - let folder_id = match env { - "APPDATA" => FOLDERID_RoamingAppData, - "LOCALAPPDATA" => FOLDERID_LocalAppData, - _ => return None, - }; - - unsafe { - let mut path = ptr::null_mut(); - match SHGetKnownFolderPath( - &folder_id, - KF_FLAG_DONT_VERIFY as u32, - std::ptr::null_mut(), - &mut path, - ) { - S_OK => { - let path_slice = slice::from_raw_parts(path, wcslen(path)); - let s = OsString::from_wide(path_slice); - CoTaskMemFree(path.cast()); - Some(PathBuf::from(s)) - } - _ => { - // Free any allocated memory even on failure. A null ptr is a no-op for `CoTaskMemFree`. - CoTaskMemFree(path.cast()); - None - } - } - } - } - - #[cfg(not(all(windows, not(target_vendor = "uwp"))))] - fn dir_crt(_env: &'static str) -> Option { - None - } -} - -impl super::BaseStrategy for Windows { - fn cache_dir(&self) -> PathBuf { - Self::dir_inner("LOCALAPPDATA") - .unwrap_or_else(|| self.home_dir.join("AppData").join("Local")) - } -} diff --git a/src/etcetera/base_strategy/xdg.rs b/src/etcetera/base_strategy/xdg.rs deleted file mode 100644 index d22369e..0000000 --- a/src/etcetera/base_strategy/xdg.rs +++ /dev/null @@ -1,38 +0,0 @@ -use crate::error::Result; -use std::path::Path; -use std::path::PathBuf; - -pub struct Xdg { - home_dir: PathBuf, -} - -impl Xdg { - pub fn new() -> Result { - Ok(Self { - home_dir: crate::etcetera::home_dir()?, - }) - } - - fn env_var_or_none(env_var: &str) -> Option { - std::env::var(env_var).ok().and_then(|path| { - let path = PathBuf::from(path); - - // Return None if the path obtained from the environment variable isn’t absolute. - if path.is_absolute() { - Some(path) - } else { - None - } - }) - } - - fn env_var_or_default(&self, env_var: &str, default: impl AsRef) -> PathBuf { - Self::env_var_or_none(env_var).unwrap_or_else(|| self.home_dir.join(default)) - } -} - -impl super::BaseStrategy for Xdg { - fn cache_dir(&self) -> PathBuf { - self.env_var_or_default("XDG_CACHE_HOME", ".cache/") - } -} diff --git a/src/etcetera/mod.rs b/src/etcetera/mod.rs deleted file mode 100644 index 708f6bf..0000000 --- a/src/etcetera/mod.rs +++ /dev/null @@ -1,9 +0,0 @@ -use crate::error::{Error, Result}; - -pub mod base_strategy; - -pub use base_strategy::{choose_base_strategy, BaseStrategy}; - -pub fn home_dir() -> Result { - home::home_dir().ok_or(Error::HomeDir) -}