2021-04-13 12:14:07 -06:00
|
|
|
use std::fs::{create_dir_all, File};
|
|
|
|
use std::io::prelude::*;
|
|
|
|
use std::path::{Path, PathBuf};
|
2021-03-10 14:24:08 -07:00
|
|
|
|
2021-04-13 12:14:07 -06:00
|
|
|
use chrono::prelude::*;
|
|
|
|
use chrono::Utc;
|
2021-04-13 15:31:41 -06:00
|
|
|
use config::{Config, Environment, File as ConfigFile};
|
2021-03-10 14:24:08 -07:00
|
|
|
use directories::ProjectDirs;
|
|
|
|
use eyre::{eyre, Result};
|
2021-04-13 12:14:07 -06:00
|
|
|
use parse_duration::parse;
|
2021-03-10 14:24:08 -07:00
|
|
|
|
2021-04-13 12:14:07 -06:00
|
|
|
pub const HISTORY_PAGE_SIZE: i64 = 100;
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, Deserialize)]
|
2021-04-20 14:53:07 -06:00
|
|
|
pub struct Settings {
|
2021-03-10 14:24:08 -07:00
|
|
|
pub dialect: String,
|
2021-04-13 12:14:07 -06:00
|
|
|
pub auto_sync: bool,
|
2021-04-09 05:40:21 -06:00
|
|
|
pub sync_address: String,
|
|
|
|
pub sync_frequency: String,
|
|
|
|
pub db_path: String,
|
2021-04-13 12:14:07 -06:00
|
|
|
pub key_path: String,
|
|
|
|
pub session_path: String,
|
|
|
|
|
|
|
|
// This is automatically loaded when settings is created. Do not set in
|
|
|
|
// config! Keep secrets and settings apart.
|
|
|
|
pub session_token: String,
|
2021-03-10 14:24:08 -07:00
|
|
|
}
|
|
|
|
|
2021-04-20 14:53:07 -06:00
|
|
|
impl Settings {
|
2021-04-13 12:14:07 -06:00
|
|
|
pub fn save_sync_time() -> Result<()> {
|
|
|
|
let sync_time_path = ProjectDirs::from("com", "elliehuxtable", "atuin")
|
|
|
|
.ok_or_else(|| eyre!("could not determine key file location"))?;
|
|
|
|
let sync_time_path = sync_time_path.data_dir().join("last_sync_time");
|
|
|
|
|
|
|
|
std::fs::write(sync_time_path, Utc::now().to_rfc3339())?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn last_sync() -> Result<chrono::DateTime<Utc>> {
|
|
|
|
let sync_time_path = ProjectDirs::from("com", "elliehuxtable", "atuin");
|
|
|
|
|
|
|
|
if sync_time_path.is_none() {
|
|
|
|
debug!("failed to load projectdirs, not syncing");
|
|
|
|
return Err(eyre!("could not load project dirs"));
|
|
|
|
}
|
|
|
|
|
|
|
|
let sync_time_path = sync_time_path.unwrap();
|
|
|
|
let sync_time_path = sync_time_path.data_dir().join("last_sync_time");
|
|
|
|
|
|
|
|
if !sync_time_path.exists() {
|
|
|
|
return Ok(Utc.ymd(1970, 1, 1).and_hms(0, 0, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
let time = std::fs::read_to_string(sync_time_path)?;
|
|
|
|
let time = chrono::DateTime::parse_from_rfc3339(time.as_str())?;
|
|
|
|
|
|
|
|
Ok(time.with_timezone(&Utc))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn should_sync(&self) -> Result<bool> {
|
|
|
|
if !self.auto_sync {
|
|
|
|
return Ok(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
match parse(self.sync_frequency.as_str()) {
|
|
|
|
Ok(d) => {
|
|
|
|
let d = chrono::Duration::from_std(d).unwrap();
|
2021-04-20 14:53:07 -06:00
|
|
|
Ok(Utc::now() - Settings::last_sync()? >= d)
|
2021-04-13 12:14:07 -06:00
|
|
|
}
|
|
|
|
Err(e) => Err(eyre!("failed to check sync: {}", e)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 14:24:08 -07:00
|
|
|
pub fn new() -> Result<Self> {
|
|
|
|
let config_dir = ProjectDirs::from("com", "elliehuxtable", "atuin").unwrap();
|
|
|
|
let config_dir = config_dir.config_dir();
|
|
|
|
|
2021-04-13 12:14:07 -06:00
|
|
|
create_dir_all(config_dir)?;
|
2021-03-10 14:24:08 -07:00
|
|
|
|
2021-04-21 11:13:51 -06:00
|
|
|
let mut config_file = if let Ok(p) = std::env::var("ATUIN_CONFIG_DIR") {
|
2021-04-13 15:31:41 -06:00
|
|
|
PathBuf::from(p)
|
|
|
|
} else {
|
|
|
|
let mut config_file = PathBuf::new();
|
|
|
|
config_file.push(config_dir);
|
|
|
|
config_file
|
|
|
|
};
|
2021-03-10 14:24:08 -07:00
|
|
|
|
2021-04-21 11:13:51 -06:00
|
|
|
config_file.push("config.toml");
|
|
|
|
|
2021-03-10 14:24:08 -07:00
|
|
|
let mut s = Config::new();
|
|
|
|
|
|
|
|
let db_path = ProjectDirs::from("com", "elliehuxtable", "atuin")
|
2021-04-13 12:14:07 -06:00
|
|
|
.ok_or_else(|| eyre!("could not determine db file location"))?
|
2021-03-10 14:24:08 -07:00
|
|
|
.data_dir()
|
|
|
|
.join("history.db");
|
|
|
|
|
2021-04-13 12:14:07 -06:00
|
|
|
let key_path = ProjectDirs::from("com", "elliehuxtable", "atuin")
|
|
|
|
.ok_or_else(|| eyre!("could not determine key file location"))?
|
|
|
|
.data_dir()
|
|
|
|
.join("key");
|
|
|
|
|
|
|
|
let session_path = ProjectDirs::from("com", "elliehuxtable", "atuin")
|
|
|
|
.ok_or_else(|| eyre!("could not determine session file location"))?
|
|
|
|
.data_dir()
|
|
|
|
.join("session");
|
|
|
|
|
2021-04-20 14:53:07 -06:00
|
|
|
s.set_default("db_path", db_path.to_str())?;
|
|
|
|
s.set_default("key_path", key_path.to_str())?;
|
|
|
|
s.set_default("session_path", session_path.to_str())?;
|
|
|
|
s.set_default("dialect", "us")?;
|
|
|
|
s.set_default("auto_sync", true)?;
|
|
|
|
s.set_default("sync_frequency", "5m")?;
|
|
|
|
s.set_default("sync_address", "https://api.atuin.sh")?;
|
2021-03-21 14:04:39 -06:00
|
|
|
|
2021-03-10 14:24:08 -07:00
|
|
|
if config_file.exists() {
|
2021-04-13 12:14:07 -06:00
|
|
|
s.merge(ConfigFile::with_name(config_file.to_str().unwrap()))?;
|
|
|
|
} else {
|
|
|
|
let example_config = include_bytes!("../config.toml");
|
|
|
|
let mut file = File::create(config_file)?;
|
|
|
|
file.write_all(example_config)?;
|
2021-03-10 14:24:08 -07:00
|
|
|
}
|
|
|
|
|
2021-04-13 15:31:41 -06:00
|
|
|
s.merge(Environment::with_prefix("atuin").separator("_"))?;
|
|
|
|
|
2021-03-10 14:24:08 -07:00
|
|
|
// all paths should be expanded
|
2021-04-20 14:53:07 -06:00
|
|
|
let db_path = s.get_str("db_path")?;
|
2021-03-10 14:24:08 -07:00
|
|
|
let db_path = shellexpand::full(db_path.as_str())?;
|
2021-04-20 14:53:07 -06:00
|
|
|
s.set("db_path", db_path.to_string())?;
|
2021-04-13 12:14:07 -06:00
|
|
|
|
2021-04-20 14:53:07 -06:00
|
|
|
let key_path = s.get_str("key_path")?;
|
2021-04-13 12:14:07 -06:00
|
|
|
let key_path = shellexpand::full(key_path.as_str())?;
|
2021-04-20 14:53:07 -06:00
|
|
|
s.set("key_path", key_path.to_string())?;
|
2021-04-13 12:14:07 -06:00
|
|
|
|
2021-04-20 14:53:07 -06:00
|
|
|
let session_path = s.get_str("session_path")?;
|
2021-04-13 12:14:07 -06:00
|
|
|
let session_path = shellexpand::full(session_path.as_str())?;
|
2021-04-20 14:53:07 -06:00
|
|
|
s.set("session_path", session_path.to_string())?;
|
2021-04-13 12:14:07 -06:00
|
|
|
|
|
|
|
// Finally, set the auth token
|
|
|
|
if Path::new(session_path.to_string().as_str()).exists() {
|
|
|
|
let token = std::fs::read_to_string(session_path.to_string())?;
|
2021-04-20 14:53:07 -06:00
|
|
|
s.set("session_token", token.trim())?;
|
2021-04-13 12:14:07 -06:00
|
|
|
} else {
|
2021-04-20 14:53:07 -06:00
|
|
|
s.set("session_token", "not logged in")?;
|
2021-04-13 12:14:07 -06:00
|
|
|
}
|
2021-03-10 14:24:08 -07:00
|
|
|
|
|
|
|
s.try_into()
|
|
|
|
.map_err(|e| eyre!("failed to deserialize: {}", e))
|
|
|
|
}
|
|
|
|
}
|