2023-03-14 17:15:39 -06:00
|
|
|
use std::env;
|
2021-04-25 11:21:52 -06:00
|
|
|
use std::path::PathBuf;
|
|
|
|
|
2023-04-11 09:26:16 -06:00
|
|
|
use rand::RngCore;
|
2021-04-20 14:53:07 -06:00
|
|
|
use uuid::Uuid;
|
2021-04-13 12:14:07 -06:00
|
|
|
|
2023-04-11 09:26:16 -06:00
|
|
|
pub fn random_bytes<const N: usize>() -> [u8; N] {
|
|
|
|
let mut ret = [0u8; N];
|
|
|
|
|
|
|
|
rand::thread_rng().fill_bytes(&mut ret);
|
|
|
|
|
|
|
|
ret
|
|
|
|
}
|
|
|
|
|
|
|
|
// basically just ripped from the uuid crate. they have it as unstable, but we can use it fine.
|
|
|
|
const fn encode_unix_timestamp_millis(millis: u64, random_bytes: &[u8; 10]) -> Uuid {
|
|
|
|
let millis_high = ((millis >> 16) & 0xFFFF_FFFF) as u32;
|
|
|
|
let millis_low = (millis & 0xFFFF) as u16;
|
|
|
|
|
|
|
|
let random_and_version =
|
|
|
|
(random_bytes[0] as u16 | ((random_bytes[1] as u16) << 8) & 0x0FFF) | (0x7 << 12);
|
|
|
|
|
|
|
|
let mut d4 = [0; 8];
|
|
|
|
|
|
|
|
d4[0] = (random_bytes[2] & 0x3F) | 0x80;
|
|
|
|
d4[1] = random_bytes[3];
|
|
|
|
d4[2] = random_bytes[4];
|
|
|
|
d4[3] = random_bytes[5];
|
|
|
|
d4[4] = random_bytes[6];
|
|
|
|
d4[5] = random_bytes[7];
|
|
|
|
d4[6] = random_bytes[8];
|
|
|
|
d4[7] = random_bytes[9];
|
|
|
|
|
|
|
|
Uuid::from_fields(millis_high, millis_low, random_and_version, &d4)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn uuid_v7() -> Uuid {
|
|
|
|
let bytes = random_bytes();
|
2023-09-11 02:26:05 -06:00
|
|
|
let now: u64 = u64::try_from(
|
|
|
|
time::OffsetDateTime::now_utc().unix_timestamp_nanos() / 1_000_000,
|
|
|
|
)
|
|
|
|
.expect("Either you're in the past (1970) - or your in the far future (2554). Good for you");
|
2023-04-11 09:26:16 -06:00
|
|
|
|
|
|
|
encode_unix_timestamp_millis(now, &bytes)
|
|
|
|
}
|
|
|
|
|
2021-04-20 14:53:07 -06:00
|
|
|
pub fn uuid_v4() -> String {
|
2022-04-22 14:38:37 -06:00
|
|
|
Uuid::new_v4().as_simple().to_string()
|
2021-04-20 14:53:07 -06:00
|
|
|
}
|
2021-04-25 11:21:52 -06:00
|
|
|
|
2023-07-14 13:58:20 -06:00
|
|
|
pub fn has_git_dir(path: &str) -> bool {
|
|
|
|
let mut gitdir = PathBuf::from(path);
|
|
|
|
gitdir.push(".git");
|
|
|
|
|
|
|
|
gitdir.exists()
|
|
|
|
}
|
|
|
|
|
|
|
|
// detect if any parent dir has a git repo in it
|
|
|
|
// I really don't want to bring in libgit for something simple like this
|
|
|
|
// If we start to do anything more advanced, then perhaps
|
|
|
|
pub fn in_git_repo(path: &str) -> Option<PathBuf> {
|
|
|
|
let mut gitdir = PathBuf::from(path);
|
|
|
|
|
|
|
|
while gitdir.parent().is_some() && !has_git_dir(gitdir.to_str().unwrap()) {
|
|
|
|
gitdir.pop();
|
|
|
|
}
|
|
|
|
|
|
|
|
// No parent? then we hit root, finding no git
|
|
|
|
if gitdir.parent().is_some() {
|
|
|
|
return Some(gitdir);
|
|
|
|
}
|
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
2021-05-09 12:12:41 -06:00
|
|
|
// TODO: more reliable, more tested
|
|
|
|
// I don't want to use ProjectDirs, it puts config in awkward places on
|
|
|
|
// mac. Data too. Seems to be more intended for GUI apps.
|
2023-03-05 12:00:59 -07:00
|
|
|
|
|
|
|
#[cfg(not(target_os = "windows"))]
|
2021-05-09 12:12:41 -06:00
|
|
|
pub fn home_dir() -> PathBuf {
|
2021-04-25 11:21:52 -06:00
|
|
|
let home = std::env::var("HOME").expect("$HOME not found");
|
2021-05-09 12:12:41 -06:00
|
|
|
PathBuf::from(home)
|
2023-03-05 12:00:59 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "windows")]
|
|
|
|
pub fn home_dir() -> PathBuf {
|
|
|
|
let home = std::env::var("USERPROFILE").expect("%userprofile% not found");
|
|
|
|
PathBuf::from(home)
|
2021-05-09 12:12:41 -06:00
|
|
|
}
|
2021-04-25 11:21:52 -06:00
|
|
|
|
2021-05-09 12:12:41 -06:00
|
|
|
pub fn config_dir() -> PathBuf {
|
|
|
|
let config_dir =
|
|
|
|
std::env::var("XDG_CONFIG_HOME").map_or_else(|_| home_dir().join(".config"), PathBuf::from);
|
|
|
|
config_dir.join("atuin")
|
2021-04-25 11:21:52 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn data_dir() -> PathBuf {
|
2021-05-09 12:12:41 -06:00
|
|
|
let data_dir = std::env::var("XDG_DATA_HOME")
|
|
|
|
.map_or_else(|_| home_dir().join(".local").join("share"), PathBuf::from);
|
2021-04-25 11:21:52 -06:00
|
|
|
|
2021-05-09 12:12:41 -06:00
|
|
|
data_dir.join("atuin")
|
|
|
|
}
|
|
|
|
|
2023-03-14 17:15:39 -06:00
|
|
|
pub fn get_current_dir() -> String {
|
|
|
|
// Prefer PWD environment variable over cwd if available to better support symbolic links
|
|
|
|
match env::var("PWD") {
|
|
|
|
Ok(v) => v,
|
|
|
|
Err(_) => match env::current_dir() {
|
|
|
|
Ok(dir) => dir.display().to_string(),
|
|
|
|
Err(_) => String::from(""),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-09 12:12:41 -06:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2023-09-11 02:26:05 -06:00
|
|
|
use time::Month;
|
|
|
|
|
2021-05-09 12:12:41 -06:00
|
|
|
use super::*;
|
|
|
|
use std::env;
|
|
|
|
|
2023-04-11 09:26:16 -06:00
|
|
|
use std::collections::HashSet;
|
|
|
|
|
2021-05-09 12:12:41 -06:00
|
|
|
#[test]
|
2023-03-02 02:39:15 -07:00
|
|
|
fn test_dirs() {
|
|
|
|
// these tests need to be run sequentially to prevent race condition
|
|
|
|
test_config_dir_xdg();
|
|
|
|
test_config_dir();
|
|
|
|
test_data_dir_xdg();
|
|
|
|
test_data_dir();
|
|
|
|
}
|
|
|
|
|
2021-05-09 12:12:41 -06:00
|
|
|
fn test_config_dir_xdg() {
|
|
|
|
env::remove_var("HOME");
|
|
|
|
env::set_var("XDG_CONFIG_HOME", "/home/user/custom_config");
|
|
|
|
assert_eq!(
|
|
|
|
config_dir(),
|
|
|
|
PathBuf::from("/home/user/custom_config/atuin")
|
|
|
|
);
|
2021-05-10 15:16:07 -06:00
|
|
|
env::remove_var("XDG_CONFIG_HOME");
|
2021-05-09 12:12:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fn test_config_dir() {
|
|
|
|
env::set_var("HOME", "/home/user");
|
|
|
|
env::remove_var("XDG_CONFIG_HOME");
|
|
|
|
assert_eq!(config_dir(), PathBuf::from("/home/user/.config/atuin"));
|
2021-05-10 15:16:07 -06:00
|
|
|
env::remove_var("HOME");
|
2021-05-09 12:12:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fn test_data_dir_xdg() {
|
|
|
|
env::remove_var("HOME");
|
|
|
|
env::set_var("XDG_DATA_HOME", "/home/user/custom_data");
|
|
|
|
assert_eq!(data_dir(), PathBuf::from("/home/user/custom_data/atuin"));
|
2021-05-10 15:16:07 -06:00
|
|
|
env::remove_var("XDG_DATA_HOME");
|
2021-05-09 12:12:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
fn test_data_dir() {
|
|
|
|
env::set_var("HOME", "/home/user");
|
|
|
|
env::remove_var("XDG_DATA_HOME");
|
|
|
|
assert_eq!(data_dir(), PathBuf::from("/home/user/.local/share/atuin"));
|
2021-05-10 15:16:07 -06:00
|
|
|
env::remove_var("HOME");
|
2021-05-09 12:12:41 -06:00
|
|
|
}
|
2023-03-26 08:47:38 -06:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn days_from_month() {
|
2023-09-11 02:26:05 -06:00
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::January), 31);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::February), 28);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::March), 31);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::April), 30);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::May), 31);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::June), 30);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::July), 31);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::August), 31);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::September), 30);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::October), 31);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::November), 30);
|
|
|
|
assert_eq!(time::util::days_in_year_month(2023, Month::December), 31);
|
2023-03-26 08:47:38 -06:00
|
|
|
|
|
|
|
// leap years
|
2023-09-11 02:26:05 -06:00
|
|
|
assert_eq!(time::util::days_in_year_month(2024, Month::February), 29);
|
2023-03-26 08:47:38 -06:00
|
|
|
}
|
2023-04-11 09:26:16 -06:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn uuid_is_unique() {
|
|
|
|
let how_many: usize = 1000000;
|
|
|
|
|
|
|
|
// for peace of mind
|
|
|
|
let mut uuids: HashSet<Uuid> = HashSet::with_capacity(how_many);
|
|
|
|
|
|
|
|
// there will be many in the same millisecond
|
|
|
|
for _ in 0..how_many {
|
|
|
|
let uuid = uuid_v7();
|
|
|
|
uuids.insert(uuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert_eq!(uuids.len(), how_many);
|
|
|
|
}
|
2021-04-25 11:21:52 -06:00
|
|
|
}
|