blob: de34779c7b26c55d5f14ad12a744081e658bda11 (
plain) (
blame)
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
|
use eyre::Result;
use rand::Rng;
use tokio::time::{self, MissedTickBehavior};
use atuin_client::{
record::{sqlite_store::SqliteStore, sync},
settings::Settings,
};
pub async fn worker(settings: Settings, store: SqliteStore) -> Result<()> {
tracing::info!("booting sync worker");
let mut ticker = time::interval(time::Duration::from_secs(settings.daemon.sync_frequency));
// IMPORTANT: without this, if we miss ticks because a sync takes ages or is otherwise delayed,
// we may end up running a lot of syncs in a hot loop. No bueno!
ticker.set_missed_tick_behavior(MissedTickBehavior::Skip);
loop {
ticker.tick().await;
tracing::info!("sync worker tick");
let res = sync::sync(&settings, &store).await;
if let Err(e) = res {
tracing::error!("sync tick failed with {e}");
let mut rng = rand::thread_rng();
let new_interval = ticker.period().as_secs_f64() * rng.gen_range(2.0..2.2);
// Don't backoff by more than 30 mins
if new_interval > 60.0 * 30.0 {
continue;
}
ticker = time::interval(time::Duration::from_secs(new_interval as u64));
ticker.reset_after(time::Duration::from_secs(new_interval as u64));
tracing::error!("backing off, next sync tick in {new_interval}");
} else {
let (uploaded, downloaded) = res.unwrap();
tracing::info!(
uploaded = ?uploaded,
downloaded = ?downloaded,
"sync complete"
);
// Reset backoff on success
if ticker.period().as_secs() != settings.daemon.sync_frequency {
ticker = time::interval(time::Duration::from_secs(settings.daemon.sync_frequency));
}
}
}
}
|