1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
use clap::Args;
use eyre::Result;
use atuin_client::{
database::Database,
record::store::Store,
record::sync::Operation,
record::{sqlite_store::SqliteStore, sync},
settings::Settings,
};
#[derive(Args, Debug)]
pub struct Pull {
/// The tag to push (eg, 'history'). Defaults to all tags
#[arg(long, short)]
pub tag: Option<String>,
/// Force push records
/// This will first wipe the local store, and then download all records from the remote
#[arg(long, default_value = "false")]
pub force: bool,
/// Page Size
/// How many records to download at once. Defaults to 100
#[arg(long, default_value = "100")]
pub page: u64,
}
impl Pull {
pub async fn run(
&self,
settings: &Settings,
store: SqliteStore,
db: &dyn Database,
) -> Result<()> {
if self.force {
println!("Forcing local overwrite!");
println!("Clearing local store");
store.delete_all().await?;
}
// We can actually just use the existing diff/etc to push
// 1. Diff
// 2. Get operations
// 3. Filter operations by
// a) are they a download op?
// b) are they for the host/tag we are pushing here?
let (diff, _) = sync::diff(settings, &store).await?;
let operations = sync::operations(diff, &store).await?;
let operations = operations
.into_iter()
.filter(|op| match op {
// No noops or downloads thx
Operation::Noop { .. } | Operation::Upload { .. } => false,
// pull, so yes plz to downloads!
Operation::Download { tag, .. } => {
if self.force {
return true;
}
if let Some(t) = self.tag.clone()
&& t != *tag
{
return false;
}
true
}
})
.collect();
let (_, downloaded) = sync::sync_remote(operations, &store, settings, self.page).await?;
println!("Downloaded {} records", downloaded.len());
crate::sync::build(settings, &store, db, Some(&downloaded)).await?;
Ok(())
}
}