src/api2/types.rs: define struct BackupContent, and use it with list_snapshot_files

This commit is contained in:
Dietmar Maurer 2020-01-23 11:16:57 +01:00
parent 51534c8de9
commit 09b1f7b202
2 changed files with 72 additions and 39 deletions

View File

@ -1,4 +1,5 @@
use std::collections::{HashSet, HashMap}; use std::collections::{HashSet, HashMap};
use std::convert::TryFrom;
use chrono::{TimeZone, Local}; use chrono::{TimeZone, Local};
use failure::*; use failure::*;
@ -21,27 +22,30 @@ use crate::config::datastore;
use crate::server::WorkerTask; use crate::server::WorkerTask;
use crate::tools; use crate::tools;
fn read_backup_index(store: &DataStore, backup_dir: &BackupDir) -> Result<Value, Error> { fn read_backup_index(store: &DataStore, backup_dir: &BackupDir) -> Result<Vec<BackupContent>, Error> {
let mut path = store.base_path(); let mut path = store.base_path();
path.push(backup_dir.relative_path()); path.push(backup_dir.relative_path());
path.push("index.json.blob"); path.push("index.json.blob");
let raw_data = file_get_contents(&path)?; let raw_data = file_get_contents(&path)?;
let data = DataBlob::from_raw(raw_data)?.decode(None)?; let index_size = raw_data.len() as u64;
let index_size = data.len(); let blob = DataBlob::from_raw(raw_data)?;
let mut result: Value = serde_json::from_reader(&mut &data[..])?;
let mut result = result["files"].take(); let manifest = BackupManifest::try_from(blob)?;
if result == Value::Null { let mut result = Vec::new();
bail!("missing 'files' property in backup index {:?}", path); for item in manifest.files() {
result.push(BackupContent {
filename: item.filename.clone(),
size: Some(item.size),
});
} }
result.as_array_mut().unwrap().push(json!({ result.push(BackupContent {
"filename": "index.json.blob", filename: "index.json.blob".to_string(),
"size": index_size, size: Some(index_size),
})); });
Ok(result) Ok(result)
} }
@ -108,32 +112,56 @@ fn list_groups(
Ok(groups) Ok(groups)
} }
#[api(
input: {
properties: {
store: {
schema: DATASTORE_SCHEMA,
},
"backup-type": {
schema: BACKUP_TYPE_SCHEMA,
},
"backup-id": {
schema: BACKUP_ID_SCHEMA,
},
"backup-time": {
schema: BACKUP_TIME_SCHEMA,
},
},
},
returns: {
type: Array,
description: "Returns the list of archive files inside a backup snapshots.",
items: {
type: BackupContent,
}
},
)]
/// List snapshot files.
fn list_snapshot_files( fn list_snapshot_files(
param: Value, store: String,
backup_type: String,
backup_id: String,
backup_time: i64,
_info: &ApiMethod, _info: &ApiMethod,
_rpcenv: &mut dyn RpcEnvironment, _rpcenv: &mut dyn RpcEnvironment,
) -> Result<Value, Error> { ) -> Result<Vec<BackupContent>, Error> {
let store = tools::required_string_param(&param, "store")?; let datastore = DataStore::lookup_datastore(&store)?;
let backup_type = tools::required_string_param(&param, "backup-type")?;
let backup_id = tools::required_string_param(&param, "backup-id")?;
let backup_time = tools::required_integer_param(&param, "backup-time")?;
let datastore = DataStore::lookup_datastore(store)?;
let snapshot = BackupDir::new(backup_type, backup_id, backup_time); let snapshot = BackupDir::new(backup_type, backup_id, backup_time);
let mut files = read_backup_index(&datastore, &snapshot)?; let mut files = read_backup_index(&datastore, &snapshot)?;
let info = BackupInfo::new(&datastore.base_path(), snapshot)?; let info = BackupInfo::new(&datastore.base_path(), snapshot)?;
let file_set = files.as_array().unwrap().iter().fold(HashSet::new(), |mut acc, item| { let file_set = files.iter().fold(HashSet::new(), |mut acc, item| {
acc.insert(item["filename"].as_str().unwrap().to_owned()); acc.insert(item.filename.clone());
acc acc
}); });
for file in info.files { for file in info.files {
if file_set.contains(&file) { continue; } if file_set.contains(&file) { continue; }
files.as_array_mut().unwrap().push(json!({ "filename": file })); files.push(BackupContent { filename: file, size: None });
} }
Ok(files) Ok(files)
@ -238,8 +266,8 @@ fn list_snapshots (
if let Ok(index) = read_backup_index(&datastore, &info.backup_dir) { if let Ok(index) = read_backup_index(&datastore, &info.backup_dir) {
let mut backup_size = 0; let mut backup_size = 0;
for item in index.as_array().unwrap().iter() { for item in index.iter() {
if let Some(item_size) = item["size"].as_u64() { if let Some(item_size) = item.size {
backup_size += item_size; backup_size += item_size;
} }
} }
@ -658,20 +686,7 @@ const DATASTORE_INFO_SUBDIRS: SubdirMap = &[
( (
"files", "files",
&Router::new() &Router::new()
.get( .get(&API_METHOD_LIST_SNAPSHOT_FILES)
&ApiMethod::new(
&ApiHandler::Sync(&list_snapshot_files),
&ObjectSchema::new(
"List snapshot files.",
&sorted!([
("store", false, &DATASTORE_SCHEMA),
("backup-type", false, &BACKUP_TYPE_SCHEMA),
("backup-id", false, &BACKUP_ID_SCHEMA),
("backup-time", false, &BACKUP_TIME_SCHEMA),
]),
)
)
)
), ),
( (
"gc", "gc",

View File

@ -259,6 +259,24 @@ pub struct SnapshotListItem {
pub size: Option<u64>, pub size: Option<u64>,
} }
#[api(
properties: {
"filename": {
schema: BACKUP_ARCHIVE_NAME_SCHEMA,
},
},
)]
#[derive(Serialize, Deserialize)]
#[serde(rename_all="kebab-case")]
/// Basic information about archive files inside a backup snapshot.
pub struct BackupContent {
pub filename: String,
/// Archive size (from backup manifest).
#[serde(skip_serializing_if="Option::is_none")]
pub size: Option<u64>,
}
// Regression tests // Regression tests