2021-02-13 12:37:00 -07:00
|
|
|
use std::env;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
|
2021-05-08 10:29:46 -06:00
|
|
|
use atuin_common::utils::uuid_v4;
|
|
|
|
use chrono::{TimeZone, Utc};
|
2021-02-14 11:40:51 -07:00
|
|
|
use directories::UserDirs;
|
2021-02-13 12:37:00 -07:00
|
|
|
use eyre::{eyre, Result};
|
|
|
|
use structopt::StructOpt;
|
|
|
|
|
2021-04-20 14:53:07 -06:00
|
|
|
use atuin_client::history::History;
|
2021-04-26 04:50:31 -06:00
|
|
|
use atuin_client::import::{bash::Bash, zsh::Zsh};
|
2021-05-08 10:29:46 -06:00
|
|
|
use atuin_client::{database::Database, import::resh::ReshEntry};
|
2021-02-13 12:37:00 -07:00
|
|
|
use indicatif::ProgressBar;
|
|
|
|
|
|
|
|
#[derive(StructOpt)]
|
2021-02-14 08:15:26 -07:00
|
|
|
pub enum Cmd {
|
2021-02-13 12:37:00 -07:00
|
|
|
#[structopt(
|
|
|
|
about="import history for the current shell",
|
|
|
|
aliases=&["a", "au", "aut"],
|
|
|
|
)]
|
|
|
|
Auto,
|
|
|
|
|
|
|
|
#[structopt(
|
|
|
|
about="import history from the zsh history file",
|
|
|
|
aliases=&["z", "zs"],
|
|
|
|
)]
|
|
|
|
Zsh,
|
2021-04-26 04:50:31 -06:00
|
|
|
|
|
|
|
#[structopt(
|
|
|
|
about="import history from the bash history file",
|
|
|
|
aliases=&["b", "ba", "bas"],
|
|
|
|
)]
|
|
|
|
Bash,
|
2021-05-08 10:29:46 -06:00
|
|
|
|
|
|
|
#[structopt(
|
|
|
|
about="import history from the resh history file",
|
|
|
|
aliases=&["r", "re", "res"],
|
|
|
|
)]
|
|
|
|
Resh,
|
2021-02-13 12:37:00 -07:00
|
|
|
}
|
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
impl Cmd {
|
2021-04-25 11:21:52 -06:00
|
|
|
pub async fn run(&self, db: &mut (impl Database + Send + Sync)) -> Result<()> {
|
2021-04-26 04:50:31 -06:00
|
|
|
println!(" Atuin ");
|
2021-02-15 02:07:49 -07:00
|
|
|
println!("======================");
|
|
|
|
println!(" \u{1f30d} ");
|
|
|
|
println!(" \u{1f418}\u{1f418}\u{1f418}\u{1f418} ");
|
|
|
|
println!(" \u{1f422} ");
|
|
|
|
println!("======================");
|
2021-02-14 08:15:26 -07:00
|
|
|
println!("Importing history...");
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
match self {
|
|
|
|
Self::Auto => {
|
|
|
|
let shell = env::var("SHELL").unwrap_or_else(|_| String::from("NO_SHELL"));
|
|
|
|
|
2021-02-14 11:40:51 -07:00
|
|
|
if shell.ends_with("/zsh") {
|
2021-02-14 08:15:26 -07:00
|
|
|
println!("Detected ZSH");
|
2021-04-25 11:21:52 -06:00
|
|
|
import_zsh(db).await
|
2021-02-14 08:15:26 -07:00
|
|
|
} else {
|
|
|
|
println!("cannot import {} history", shell);
|
|
|
|
Ok(())
|
|
|
|
}
|
2021-02-13 12:37:00 -07:00
|
|
|
}
|
|
|
|
|
2021-04-25 11:21:52 -06:00
|
|
|
Self::Zsh => import_zsh(db).await,
|
2021-04-26 04:50:31 -06:00
|
|
|
Self::Bash => import_bash(db).await,
|
2021-05-08 10:29:46 -06:00
|
|
|
Self::Resh => import_resh(db).await,
|
2021-02-13 12:37:00 -07:00
|
|
|
}
|
2021-02-14 08:15:26 -07:00
|
|
|
}
|
|
|
|
}
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-05-08 10:29:46 -06:00
|
|
|
async fn import_resh(db: &mut (impl Database + Send + Sync)) -> Result<()> {
|
|
|
|
let histpath = std::path::Path::new(std::env::var("HOME")?.as_str()).join(".resh_history.json");
|
|
|
|
|
|
|
|
println!("Parsing .resh_history.json...");
|
|
|
|
#[allow(clippy::filter_map)]
|
|
|
|
let history = std::fs::read_to_string(histpath)?
|
|
|
|
.split('\n')
|
|
|
|
.map(str::trim)
|
|
|
|
.map(|x| serde_json::from_str::<ReshEntry>(x))
|
|
|
|
.filter_map(Result::ok)
|
|
|
|
.map(|x| {
|
|
|
|
#[allow(clippy::cast_possible_truncation)]
|
|
|
|
#[allow(clippy::cast_sign_loss)]
|
|
|
|
let timestamp = {
|
|
|
|
let secs = x.realtime_before.floor() as i64;
|
|
|
|
let nanosecs = (x.realtime_before.fract() * 1_000_000_000_f64).round() as u32;
|
|
|
|
Utc.timestamp(secs, nanosecs)
|
|
|
|
};
|
|
|
|
#[allow(clippy::cast_possible_truncation)]
|
|
|
|
#[allow(clippy::cast_sign_loss)]
|
|
|
|
let duration = {
|
|
|
|
let secs = x.realtime_after.floor() as i64;
|
|
|
|
let nanosecs = (x.realtime_after.fract() * 1_000_000_000_f64).round() as u32;
|
|
|
|
let difference = Utc.timestamp(secs, nanosecs) - timestamp;
|
|
|
|
difference.num_nanoseconds().unwrap_or(0)
|
|
|
|
};
|
|
|
|
|
|
|
|
History {
|
|
|
|
id: uuid_v4(),
|
|
|
|
timestamp,
|
|
|
|
duration,
|
|
|
|
exit: x.exit_code,
|
|
|
|
command: x.cmd_line,
|
|
|
|
cwd: x.pwd,
|
|
|
|
session: uuid_v4(),
|
|
|
|
hostname: x.host,
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
println!("Updating database...");
|
|
|
|
|
|
|
|
let progress = ProgressBar::new(history.len() as u64);
|
|
|
|
|
|
|
|
let buf_size = 100;
|
|
|
|
let mut buf = Vec::<_>::with_capacity(buf_size);
|
|
|
|
|
|
|
|
for i in history {
|
|
|
|
buf.push(i);
|
|
|
|
|
|
|
|
if buf.len() == buf_size {
|
|
|
|
db.save_bulk(&buf).await?;
|
|
|
|
progress.inc(buf.len() as u64);
|
|
|
|
|
|
|
|
buf.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !buf.is_empty() {
|
|
|
|
db.save_bulk(&buf).await?;
|
|
|
|
progress.inc(buf.len() as u64);
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-04-25 11:21:52 -06:00
|
|
|
async fn import_zsh(db: &mut (impl Database + Send + Sync)) -> Result<()> {
|
2021-02-14 08:15:26 -07:00
|
|
|
// oh-my-zsh sets HISTFILE=~/.zhistory
|
|
|
|
// zsh has no default value for this var, but uses ~/.zhistory.
|
|
|
|
// we could maybe be smarter about this in the future :)
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
let histpath = env::var("HISTFILE");
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
let histpath = if let Ok(p) = histpath {
|
2021-02-14 11:40:51 -07:00
|
|
|
let histpath = PathBuf::from(p);
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 11:40:51 -07:00
|
|
|
if !histpath.exists() {
|
|
|
|
return Err(eyre!(
|
2021-02-15 02:07:49 -07:00
|
|
|
"Could not find history file {:?}. try updating $HISTFILE",
|
|
|
|
histpath
|
2021-02-14 11:40:51 -07:00
|
|
|
));
|
|
|
|
}
|
2021-02-14 08:15:26 -07:00
|
|
|
|
2021-02-14 11:40:51 -07:00
|
|
|
histpath
|
|
|
|
} else {
|
|
|
|
let user_dirs = UserDirs::new().unwrap();
|
|
|
|
let home_dir = user_dirs.home_dir();
|
|
|
|
|
|
|
|
let mut candidates = [".zhistory", ".zsh_history"].iter();
|
|
|
|
loop {
|
|
|
|
match candidates.next() {
|
|
|
|
Some(candidate) => {
|
|
|
|
let histpath = home_dir.join(candidate);
|
|
|
|
if histpath.exists() {
|
|
|
|
break histpath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => return Err(eyre!("Could not find history file. try setting $HISTFILE")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2021-02-14 08:15:26 -07:00
|
|
|
|
2021-02-15 02:07:49 -07:00
|
|
|
let zsh = Zsh::new(histpath)?;
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
let progress = ProgressBar::new(zsh.loc);
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
let buf_size = 100;
|
|
|
|
let mut buf = Vec::<History>::with_capacity(buf_size);
|
|
|
|
|
2021-02-14 15:12:35 -07:00
|
|
|
for i in zsh
|
|
|
|
.filter_map(Result::ok)
|
|
|
|
.filter(|x| !x.command.trim().is_empty())
|
|
|
|
{
|
|
|
|
buf.push(i);
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
if buf.len() == buf_size {
|
2021-04-25 11:21:52 -06:00
|
|
|
db.save_bulk(&buf).await?;
|
2021-02-13 12:37:00 -07:00
|
|
|
progress.inc(buf.len() as u64);
|
|
|
|
|
2021-02-15 02:07:49 -07:00
|
|
|
buf.clear();
|
2021-02-14 08:15:26 -07:00
|
|
|
}
|
2021-02-13 12:37:00 -07:00
|
|
|
}
|
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
if !buf.is_empty() {
|
2021-04-25 11:21:52 -06:00
|
|
|
db.save_bulk(&buf).await?;
|
2021-02-14 08:15:26 -07:00
|
|
|
progress.inc(buf.len() as u64);
|
|
|
|
}
|
2021-02-13 13:21:49 -07:00
|
|
|
|
2021-02-15 13:31:58 -07:00
|
|
|
progress.finish();
|
|
|
|
println!("Import complete!");
|
2021-02-13 12:37:00 -07:00
|
|
|
|
2021-02-14 08:15:26 -07:00
|
|
|
Ok(())
|
2021-02-13 12:37:00 -07:00
|
|
|
}
|
2021-04-26 04:50:31 -06:00
|
|
|
|
|
|
|
// TODO: don't just copy paste this lol
|
|
|
|
async fn import_bash(db: &mut (impl Database + Send + Sync)) -> Result<()> {
|
|
|
|
// oh-my-zsh sets HISTFILE=~/.zhistory
|
|
|
|
// zsh has no default value for this var, but uses ~/.zhistory.
|
|
|
|
// we could maybe be smarter about this in the future :)
|
|
|
|
|
|
|
|
let histpath = env::var("HISTFILE");
|
|
|
|
|
|
|
|
let histpath = if let Ok(p) = histpath {
|
|
|
|
let histpath = PathBuf::from(p);
|
|
|
|
|
|
|
|
if !histpath.exists() {
|
|
|
|
return Err(eyre!(
|
|
|
|
"Could not find history file {:?}. try updating $HISTFILE",
|
|
|
|
histpath
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
histpath
|
|
|
|
} else {
|
|
|
|
let user_dirs = UserDirs::new().unwrap();
|
|
|
|
let home_dir = user_dirs.home_dir();
|
|
|
|
|
|
|
|
home_dir.join(".bash_history")
|
|
|
|
};
|
|
|
|
|
|
|
|
let bash = Bash::new(histpath)?;
|
|
|
|
|
|
|
|
let progress = ProgressBar::new(bash.loc);
|
|
|
|
|
|
|
|
let buf_size = 100;
|
|
|
|
let mut buf = Vec::<History>::with_capacity(buf_size);
|
|
|
|
|
|
|
|
for i in bash
|
|
|
|
.filter_map(Result::ok)
|
|
|
|
.filter(|x| !x.command.trim().is_empty())
|
|
|
|
{
|
|
|
|
buf.push(i);
|
|
|
|
|
|
|
|
if buf.len() == buf_size {
|
|
|
|
db.save_bulk(&buf).await?;
|
|
|
|
progress.inc(buf.len() as u64);
|
|
|
|
|
|
|
|
buf.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !buf.is_empty() {
|
|
|
|
db.save_bulk(&buf).await?;
|
|
|
|
progress.inc(buf.len() as u64);
|
|
|
|
}
|
|
|
|
|
|
|
|
progress.finish();
|
|
|
|
println!("Import complete!");
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|