]> git.proxmox.com Git - proxmox-backup.git/blame - src/api2/admin/sync.rs
api/{pull, sync}: refactor to do_sync_job
[proxmox-backup.git] / src / api2 / admin / sync.rs
CommitLineData
664d8a27 1use anyhow::{format_err, Error};
d43f86f3 2use serde_json::Value;
d43f86f3
DC
3
4use proxmox::api::{api, ApiMethod, Router, RpcEnvironment};
5use proxmox::api::router::SubdirMap;
6use proxmox::{list_subdirs_api_method, sortable};
7
8use crate::api2::types::*;
42b68f72 9use crate::api2::pull::do_sync_job;
d43f86f3 10use crate::config::sync::{self, SyncJobStatus, SyncJobConfig};
664d8a27
DC
11use crate::server::UPID;
12use crate::config::jobstate::JobState;
d43f86f3
DC
13use crate::tools::systemd::time::{
14 parse_calendar_event, compute_next_event};
15
16#[api(
17 input: {
18 properties: {},
19 },
20 returns: {
21 description: "List configured jobs and their status.",
22 type: Array,
23 items: { type: sync::SyncJobStatus },
24 },
25)]
26/// List all sync jobs
27pub fn list_sync_jobs(
28 _param: Value,
29 mut rpcenv: &mut dyn RpcEnvironment,
30) -> Result<Vec<SyncJobStatus>, Error> {
31
32 let (config, digest) = sync::config()?;
33
34 let mut list: Vec<SyncJobStatus> = config.convert_to_typed_array("sync")?;
35
664d8a27
DC
36 for job in &mut list {
37 let last_state = JobState::load("syncjob", &job.id)
38 .map_err(|err| format_err!("could not open statefile for {}: {}", &job.id, err))?;
39 let (upid, endtime, state, starttime) = match last_state {
40 JobState::Created { time } => (None, None, None, time),
41 JobState::Started { upid } => {
42 let parsed_upid: UPID = upid.parse()?;
43 (Some(upid), None, None, parsed_upid.starttime)
44 },
45 JobState::Finished { upid, endtime, state } => {
46 let parsed_upid: UPID = upid.parse()?;
47 (Some(upid), Some(endtime), Some(state.to_string()), parsed_upid.starttime)
48 },
d43f86f3 49 };
d43f86f3 50
664d8a27
DC
51 job.last_run_upid = upid;
52 job.last_run_state = state;
53 job.last_run_endtime = endtime;
54
55 let last = job.last_run_endtime.unwrap_or_else(|| starttime);
8d785899
DC
56
57 job.next_run = (|| -> Option<i64> {
58 let schedule = job.schedule.as_ref()?;
59 let event = parse_calendar_event(&schedule).ok()?;
60 compute_next_event(&event, last, false).ok()
61 })();
d43f86f3
DC
62 }
63
64 rpcenv["digest"] = proxmox::tools::digest_to_hex(&digest).into();
65
66 Ok(list)
67}
68
69#[api(
70 input: {
71 properties: {
72 id: {
73 schema: JOB_ID_SCHEMA,
74 }
75 }
76 }
77)]
78/// Runs the sync jobs manually.
42b68f72 79fn run_sync_job(
d43f86f3
DC
80 id: String,
81 _info: &ApiMethod,
82 rpcenv: &mut dyn RpcEnvironment,
83) -> Result<String, Error> {
84
85 let (config, _digest) = sync::config()?;
86 let sync_job: SyncJobConfig = config.lookup("sync", &id)?;
87
e7cb4dc5 88 let userid: Userid = rpcenv.get_user().unwrap().parse()?;
d43f86f3 89
42b68f72 90 let upid_str = do_sync_job(&id, sync_job, &userid)?;
d43f86f3
DC
91
92 Ok(upid_str)
93}
94
95#[sortable]
96const SYNC_INFO_SUBDIRS: SubdirMap = &[
97 (
98 "run",
99 &Router::new()
100 .post(&API_METHOD_RUN_SYNC_JOB)
101 ),
102];
103
104const SYNC_INFO_ROUTER: Router = Router::new()
105 .get(&list_subdirs_api_method!(SYNC_INFO_SUBDIRS))
106 .subdirs(SYNC_INFO_SUBDIRS);
107
108
109pub const ROUTER: Router = Router::new()
110 .get(&API_METHOD_LIST_SYNC_JOBS)
111 .match_all("id", &SYNC_INFO_ROUTER);