use clap::Subcommand;
use eyre::{Result, WrapErr};
use atuin_client::{
database::Database,
encryption,
history::store::HistoryStore,
record::{sqlite_store::SqliteStore, store::Store, sync},
settings::Settings,
};
mod status;
use crate::command::client::account;
#[derive(Subcommand, Debug)]
#[command(infer_subcommands = true)]
pub enum Cmd {
Sync {
#[arg(long, short)]
force: bool,
},
Login(account::login::Cmd),
Logout,
Register(account::register::Cmd),
Key {
#[arg(long)]
base64: bool,
},
Status,
}
impl Cmd {
pub async fn run(
self,
settings: Settings,
db: &impl Database,
store: SqliteStore,
) -> Result<()> {
match self {
Self::Sync { force } => run(&settings, force, db, store).await,
Self::Login(l) => l.run(&settings, &store).await,
Self::Logout => account::logout::run().await,
Self::Register(r) => r.run(&settings, &store).await,
Self::Status => status::run(&settings, db).await,
Self::Key { base64 } => {
use atuin_client::encryption::{encode_key, load_key};
let key = load_key(&settings).wrap_err("could not load encryption key")?;
if base64 {
let encode = encode_key(&key).wrap_err("could not encode encryption key")?;
println!("{encode}");
} else {
let mnemonic = bip39::Mnemonic::from_entropy(&key, bip39::Language::English)
.map_err(|_| eyre::eyre!("invalid key"))?;
println!("{mnemonic}");
}
Ok(())
}
}
}
}
async fn run(
settings: &Settings,
force: bool,
db: &impl Database,
store: SqliteStore,
) -> Result<()> {
if settings.sync.records {
let encryption_key: [u8; 32] = encryption::load_key(settings)
.context("could not load encryption key")?
.into();
let host_id = Settings::host_id().await?;
let history_store = HistoryStore::new(store.clone(), host_id, encryption_key);
let (uploaded, downloaded) = sync::sync(settings, &store).await?;
crate::sync::build(settings, &store, db, Some(&downloaded)).await?;
println!("{uploaded}/{} up/down to record store", downloaded.len());
let history_length = db.history_count(true).await?;
let store_history_length = store.len_tag("history").await?;
#[allow(clippy::cast_sign_loss)]
if history_length as u64 > store_history_length {
println!(
"{history_length} in history index, but {store_history_length} in history store"
);
println!("Running automatic history store init...");
history_store.init_store(db).await?;
println!("Re-running sync due to new records locally");
let (uploaded, downloaded) = sync::sync(settings, &store).await?;
crate::sync::build(settings, &store, db, Some(&downloaded)).await?;
println!("{uploaded}/{} up/down to record store", downloaded.len());
}
} else {
atuin_client::sync::sync(settings, force, db).await?;
}
println!(
"Sync complete! {} items in history database, force: {}",
db.history_count(true).await?,
force
);
Ok(())
}