Skip to content

Commit 9ab6f96

Browse files
committedMar 2, 2020
Auto merge of #508 - Zeegomo:last-crates-update, r=pietroalbini
Last crates update Add a cron job to update the crates lists every 24h and a related metric to get the time of last update.
2 parents 29add6e + d3c6d84 commit 9ab6f96

File tree

4 files changed

+104
-3
lines changed

4 files changed

+104
-3
lines changed
 

‎src/server/cronjobs.rs

+53
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
use crate::actions::{Action, ActionsCtx, UpdateLists};
2+
use crate::prelude::*;
3+
use crate::server::Data;
4+
use crate::utils;
5+
use std::sync::Arc;
6+
use std::thread;
7+
use std::time::Duration;
8+
9+
const DAY: Duration = Duration::from_secs(60 * 60 * 24);
10+
struct JobDescription {
11+
name: &'static str,
12+
interval: Duration,
13+
exec: fn(Arc<Data>) -> Fallible<()>,
14+
}
15+
16+
static JOBS: &[JobDescription] = &[JobDescription {
17+
name: "crates lists update",
18+
interval: DAY,
19+
exec: update_crates as fn(Arc<Data>) -> Fallible<()>,
20+
}];
21+
22+
pub fn spawn(data: Data) {
23+
let data = Arc::new(data);
24+
for job in JOBS {
25+
// needed to make the borrowck happy
26+
let data = Arc::clone(&data);
27+
28+
thread::spawn(move || loop {
29+
let result = (job.exec)(Arc::clone(&data));
30+
if let Err(e) = result {
31+
utils::report_failure(&e);
32+
}
33+
34+
info!(
35+
"the {} thread will be respawned in {}s",
36+
job.name,
37+
job.interval.as_secs()
38+
);
39+
thread::sleep(job.interval);
40+
});
41+
}
42+
}
43+
44+
fn update_crates(data: Arc<Data>) -> Fallible<()> {
45+
let ctx = ActionsCtx::new(&data.db, &data.config);
46+
47+
UpdateLists {
48+
github: true,
49+
registry: true,
50+
local: false,
51+
}
52+
.apply(&ctx)
53+
}

‎src/server/metrics.rs

+46-3
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,23 @@
1-
use crate::db::Database;
1+
use crate::db::{Database, QueryUtils};
22
use crate::experiments::{Assignee, Experiment};
33
use crate::prelude::*;
44
use crate::server::agents::Agent;
5+
use chrono::{DateTime, Utc};
56
use prometheus::proto::{Metric, MetricFamily};
6-
use prometheus::{IntCounterVec, IntGaugeVec, __register_counter_vec, __register_gauge_vec};
7+
use prometheus::{
8+
IntCounterVec, IntGauge, IntGaugeVec, __register_counter_vec, __register_gauge,
9+
__register_gauge_vec,
10+
};
11+
712
const JOBS_METRIC: &str = "crater_completed_jobs_total";
813
const AGENT_WORK_METRIC: &str = "crater_agent_supposed_to_work";
14+
const LAST_CRATES_UPDATE_METRIC: &str = "crater_last_crates_update";
915

1016
#[derive(Clone)]
1117
pub struct Metrics {
1218
crater_completed_jobs_total: IntCounterVec,
1319
crater_work_status: IntGaugeVec,
20+
crater_last_crates_update: IntGauge,
1421
}
1522

1623
impl Metrics {
@@ -20,10 +27,14 @@ impl Metrics {
2027
prometheus::register_int_counter_vec!(jobs_opts, &["agent", "experiment"])?;
2128
let agent_opts = prometheus::opts!(AGENT_WORK_METRIC, "is agent supposed to work");
2229
let crater_work_status = prometheus::register_int_gauge_vec!(agent_opts, &["agent"])?;
30+
let crates_update_opts =
31+
prometheus::opts!(LAST_CRATES_UPDATE_METRIC, "last update of crates lists");
32+
let crater_last_crates_update = prometheus::register_int_gauge!(crates_update_opts)?;
2333

2434
Ok(Metrics {
2535
crater_completed_jobs_total,
2636
crater_work_status,
37+
crater_last_crates_update,
2738
})
2839
}
2940

@@ -79,20 +90,36 @@ impl Metrics {
7990
Ok(())
8091
}
8192

93+
pub fn update_crates_lists(&self, db: &Database) -> Fallible<()> {
94+
//&[] as &[u32] is just a hint to make the compiler happy
95+
let datetime: Option<DateTime<Utc>> =
96+
db.get_row("SELECT MAX(loaded_at) FROM crates;", &[] as &[u32], |r| {
97+
r.get(0)
98+
})?;
99+
100+
if let Some(datetime) = datetime {
101+
self.crater_last_crates_update.set(datetime.timestamp());
102+
Ok(())
103+
} else {
104+
bail!("no crates loaded");
105+
}
106+
}
107+
82108
pub fn on_complete_experiment(&self, experiment: &str) -> Fallible<()> {
83109
self.remove_experiment_jobs(experiment)
84110
}
85111
}
86112

87113
#[cfg(test)]
88114
mod tests {
89-
use super::{Metrics, AGENT_WORK_METRIC, JOBS_METRIC};
115+
use super::{Metrics, AGENT_WORK_METRIC, JOBS_METRIC, LAST_CRATES_UPDATE_METRIC};
90116
use crate::actions::{Action, ActionsCtx, CreateExperiment, EditExperiment};
91117
use crate::config::Config;
92118
use crate::db::Database;
93119
use crate::experiments::{Assignee, Experiment};
94120
use crate::server::agents::{Agent, Agents};
95121
use crate::server::tokens::Tokens;
122+
use chrono::Utc;
96123
use lazy_static::lazy_static;
97124
use prometheus::proto::MetricFamily;
98125

@@ -202,4 +229,20 @@ mod tests {
202229
assert!(supposed_to_work(&status, Some(agent1)));
203230
assert!(!supposed_to_work(&status, Some(agent2)));
204231
}
232+
233+
#[test]
234+
fn test_crates_list_update() {
235+
let db = Database::temp().unwrap();
236+
let config = Config::default();
237+
238+
let now = Utc::now();
239+
crate::crates::lists::setup_test_lists(&db, &config).unwrap();
240+
METRICS.update_crates_lists(&db).unwrap();
241+
let last_update = Metrics::get_metric_by_name(LAST_CRATES_UPDATE_METRIC)
242+
.unwrap()
243+
.get_metric()[0]
244+
.get_gauge()
245+
.get_value() as i64;
246+
assert!(last_update >= now.timestamp());
247+
}
205248
}

‎src/server/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
pub mod agents;
22
pub mod api_types;
33
mod auth;
4+
mod cronjobs;
45
mod github;
56
mod messages;
67
mod metrics;
@@ -74,6 +75,7 @@ pub fn run(config: Config) -> Fallible<()> {
7475
let mutex = Arc::new(Mutex::new(data.clone()));
7576

7677
data.reports_worker.spawn(data.clone());
78+
cronjobs::spawn(data.clone());
7779

7880
info!("running server...");
7981

‎src/server/routes/metrics.rs

+3
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,9 @@ fn endpoint_metrics(data: Arc<Data>) -> Fallible<Response<Body>> {
3333
&data.db,
3434
&data.agents.all()?.iter().collect::<Vec<&Agent>>(),
3535
)?;
36+
37+
data.metrics.update_crates_lists(&data.db)?;
38+
3639
let mut buffer = Vec::new();
3740
let families = prometheus::gather();
3841
TextEncoder::new().encode(&families, &mut buffer)?;

0 commit comments

Comments
 (0)