add protected info of snapshots to api and task logs

adds the info that a snapshot is protected to:
* snapshot list
* manual pruning (also dry-run)
* prune jobs

Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
Signed-off-by: Wolfgang Bumiller <w.bumiller@proxmox.com>
This commit is contained in:
Dominik Csapak 2021-10-27 13:22:28 +02:00 committed by Wolfgang Bumiller
parent db4b469285
commit 02db72678f
5 changed files with 49 additions and 18 deletions

View File

@ -390,6 +390,8 @@ pub struct SnapshotListItem {
/// The owner of the snapshots group /// The owner of the snapshots group
#[serde(skip_serializing_if = "Option::is_none")] #[serde(skip_serializing_if = "Option::is_none")]
pub owner: Option<Authid>, pub owner: Option<Authid>,
/// Protection from prunes
pub protected: bool,
} }
#[api( #[api(

View File

@ -7,7 +7,30 @@ use pbs_api_types::PruneOptions;
use super::BackupInfo; use super::BackupInfo;
enum PruneMark { Protected, Keep, KeepPartial, Remove } #[derive(Clone, Copy, PartialEq, Eq)]
pub enum PruneMark { Protected, Keep, KeepPartial, Remove }
impl PruneMark {
pub fn keep(&self) -> bool {
*self != PruneMark::Remove
}
pub fn protected(&self) -> bool {
*self == PruneMark::Protected
}
}
impl std::fmt::Display for PruneMark {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let txt = match self {
PruneMark::Protected => "protected",
PruneMark::Keep => "keep",
PruneMark::KeepPartial => "keep-partial",
PruneMark::Remove => "remove",
};
write!(f, "{}", txt)
}
}
fn mark_selections<F: Fn(&BackupInfo) -> Result<String, Error>> ( fn mark_selections<F: Fn(&BackupInfo) -> Result<String, Error>> (
mark: &mut HashMap<PathBuf, PruneMark>, mark: &mut HashMap<PathBuf, PruneMark>,
@ -125,7 +148,7 @@ pub fn cli_options_string(options: &PruneOptions) -> String {
pub fn compute_prune_info( pub fn compute_prune_info(
mut list: Vec<BackupInfo>, mut list: Vec<BackupInfo>,
options: &PruneOptions, options: &PruneOptions,
) -> Result<Vec<(BackupInfo, bool)>, Error> { ) -> Result<Vec<(BackupInfo, PruneMark)>, Error> {
let mut mark = HashMap::new(); let mut mark = HashMap::new();
@ -173,15 +196,16 @@ pub fn compute_prune_info(
})?; })?;
} }
let prune_info: Vec<(BackupInfo, bool)> = list.into_iter() let prune_info: Vec<(BackupInfo, PruneMark)> = list.into_iter()
.map(|info| { .map(|info| {
let backup_id = info.backup_dir.relative_path(); let backup_id = info.backup_dir.relative_path();
let keep = match mark.get(&backup_id) { let mark = if info.protected {
Some(PruneMark::Keep) => true, PruneMark::Protected
Some(PruneMark::KeepPartial) => true, } else {
_ => false, *mark.get(&backup_id).unwrap_or(&PruneMark::Remove)
}; };
(info, keep)
(info, mark)
}) })
.collect(); .collect();

View File

@ -441,6 +441,7 @@ pub fn list_snapshots (
let backup_type = group.backup_type().to_string(); let backup_type = group.backup_type().to_string();
let backup_id = group.backup_id().to_string(); let backup_id = group.backup_id().to_string();
let backup_time = info.backup_dir.backup_time(); let backup_time = info.backup_dir.backup_time();
let protected = info.backup_dir.is_protected(base_path.clone());
match get_all_snapshot_files(&datastore, &info) { match get_all_snapshot_files(&datastore, &info) {
Ok((manifest, files)) => { Ok((manifest, files)) => {
@ -479,6 +480,7 @@ pub fn list_snapshots (
files, files,
size, size,
owner, owner,
protected,
} }
}, },
Err(err) => { Err(err) => {
@ -503,6 +505,7 @@ pub fn list_snapshots (
files, files,
size: None, size: None,
owner, owner,
protected,
} }
}, },
} }
@ -846,8 +849,8 @@ pub fn prune(
let keep_all = !pbs_datastore::prune::keeps_something(&prune_options); let keep_all = !pbs_datastore::prune::keeps_something(&prune_options);
if dry_run { if dry_run {
for (info, mut keep) in prune_info { for (info, mark) in prune_info {
if keep_all { keep = true; } let keep = keep_all || mark.keep();
let backup_time = info.backup_dir.backup_time(); let backup_time = info.backup_dir.backup_time();
let group = info.backup_dir.group(); let group = info.backup_dir.group();
@ -857,6 +860,7 @@ pub fn prune(
"backup-id": group.backup_id(), "backup-id": group.backup_id(),
"backup-time": backup_time, "backup-time": backup_time,
"keep": keep, "keep": keep,
"protected": mark.protected(),
})); }));
} }
return Ok(json!(prune_result)); return Ok(json!(prune_result));
@ -874,8 +878,8 @@ pub fn prune(
store, backup_type, backup_id); store, backup_type, backup_id);
} }
for (info, mut keep) in prune_info { for (info, mark) in prune_info {
if keep_all { keep = true; } let keep = keep_all || mark.keep();
let backup_time = info.backup_dir.backup_time(); let backup_time = info.backup_dir.backup_time();
let timestamp = info.backup_dir.backup_time_string(); let timestamp = info.backup_dir.backup_time_string();
@ -887,7 +891,7 @@ pub fn prune(
group.backup_type(), group.backup_type(),
group.backup_id(), group.backup_id(),
timestamp, timestamp,
if keep { "keep" } else { "remove" }, mark,
); );
task_log!(worker, "{}", msg); task_log!(worker, "{}", msg);
@ -897,6 +901,7 @@ pub fn prune(
"backup-id": group.backup_id(), "backup-id": group.backup_id(),
"backup-time": backup_time, "backup-time": backup_time,
"keep": keep, "keep": keep,
"protected": mark.protected(),
})); }));
if !(dry_run || keep) { if !(dry_run || keep) {

View File

@ -63,12 +63,12 @@ pub fn prune_datastore(
group.backup_id() group.backup_id()
); );
for (info, mut keep) in prune_info { for (info, mark) in prune_info {
if keep_all { keep = true; } let keep = keep_all || mark.keep();
task_log!( task_log!(
worker, worker,
"{} {}/{}/{}", "{} {}/{}/{}",
if keep { "keep" } else { "remove" }, mark,
group.backup_type(), group.backup_type(),
group.backup_id(), group.backup_id(),
info.backup_dir.backup_time_string() info.backup_dir.backup_time_string()

View File

@ -19,8 +19,8 @@ fn get_prune_list(
prune_info prune_info
.iter() .iter()
.filter_map(|(info, keep)| { .filter_map(|(info, mark)| {
if *keep != return_kept { if mark.keep() != return_kept {
None None
} else { } else {
Some(info.backup_dir.relative_path()) Some(info.backup_dir.relative_path())