proxmox-backup/src/api2/tape/media.rs

222 lines
5.9 KiB
Rust
Raw Normal View History

2020-12-14 06:55:57 +00:00
use std::path::Path;
2020-12-14 07:58:40 +00:00
use anyhow::{bail, format_err, Error};
2020-12-14 06:55:57 +00:00
2020-12-14 07:58:40 +00:00
use proxmox::{
api::{api, Router, SubdirMap},
list_subdirs_api_method,
};
2020-12-14 06:55:57 +00:00
use crate::{
config::{
self,
},
api2::types::{
MEDIA_POOL_NAME_SCHEMA,
2020-12-14 07:58:40 +00:00
MEDIA_LABEL_SCHEMA,
2020-12-14 06:55:57 +00:00
MediaPoolConfig,
MediaListEntry,
MediaStatus,
MediaLocationKind,
},
tape::{
TAPE_STATUS_DIR,
Inventory,
MediaStateDatabase,
MediaLocation,
MediaPool,
update_online_status,
},
};
fn split_location(location: &MediaLocation) -> (MediaLocationKind, Option<String>) {
match location {
MediaLocation::Online(changer_name) => {
(MediaLocationKind::Online, Some(changer_name.to_string()))
}
MediaLocation::Offline => {
(MediaLocationKind::Offline, None)
}
MediaLocation::Vault(vault) => {
(MediaLocationKind::Vault, Some(vault.to_string()))
}
}
}
#[api(
input: {
properties: {
pool: {
schema: MEDIA_POOL_NAME_SCHEMA,
optional: true,
},
},
},
returns: {
description: "List of registered backup media.",
type: Array,
items: {
type: MediaListEntry,
},
},
)]
/// List pool media
pub async fn list_media(pool: Option<String>) -> Result<Vec<MediaListEntry>, Error> {
let (config, _digest) = config::media_pool::config()?;
let status_path = Path::new(TAPE_STATUS_DIR);
tokio::task::spawn_blocking(move || {
if let Err(err) = update_online_status(status_path) {
eprintln!("{}", err);
eprintln!("update online media status failed - using old state");
}
}).await?;
let mut list = Vec::new();
for (_section_type, data) in config.sections.values() {
let pool_name = match data["name"].as_str() {
None => continue,
Some(name) => name,
};
if let Some(ref name) = pool {
if name != pool_name {
continue;
}
}
let config: MediaPoolConfig = config.lookup("pool", pool_name)?;
let pool = MediaPool::with_config(pool_name, status_path, &config)?;
let current_time = proxmox::tools::time::epoch_i64();
for media in pool.list_media() {
let (location, location_hint) = split_location(&media.location());
let expired = pool.media_is_expired(&media, current_time);
let media_set_uuid = media.media_set_label().as_ref()
.map(|set| set.uuid.to_string());
let seq_nr = media.media_set_label().as_ref()
.map(|set| set.seq_nr);
let media_set_name = media.media_set_label().as_ref()
.map(|set| {
pool.generate_media_set_name(&set.uuid, config.template.clone())
.unwrap_or_else(|_| set.uuid.to_string())
});
list.push(MediaListEntry {
uuid: media.uuid().to_string(),
changer_id: media.changer_id().to_string(),
pool: Some(pool_name.to_string()),
location,
location_hint,
status: *media.status(),
expired,
media_set_uuid,
media_set_name,
seq_nr,
});
}
}
if pool.is_none() {
let inventory = Inventory::load(status_path)?;
let state_db = MediaStateDatabase::load(status_path)?;
for media_id in inventory.list_unassigned_media() {
let (mut status, location) = state_db.status_and_location(&media_id.label.uuid);
let (location, location_hint) = split_location(&location);
if status == MediaStatus::Unknown {
status = MediaStatus::Writable;
}
list.push(MediaListEntry {
uuid: media_id.label.uuid.to_string(),
changer_id: media_id.label.changer_id.to_string(),
location,
location_hint,
status,
expired: false,
media_set_uuid: None,
media_set_name: None,
seq_nr: None,
pool: None,
});
}
}
Ok(list)
}
2020-12-14 07:58:40 +00:00
#[api(
input: {
properties: {
"changer-id": {
schema: MEDIA_LABEL_SCHEMA,
},
force: {
description: "Force removal (even if media is used in a media set).",
type: bool,
optional: true,
},
},
},
)]
/// Destroy media (completely remove from database)
pub fn destroy_media(changer_id: String, force: Option<bool>,) -> Result<(), Error> {
let force = force.unwrap_or(false);
let status_path = Path::new(TAPE_STATUS_DIR);
let mut inventory = Inventory::load(status_path)?;
let media_id = inventory.find_media_by_changer_id(&changer_id)
.ok_or_else(|| format_err!("no such media '{}'", changer_id))?;
if !force {
if let Some(ref set) = media_id.media_set_label {
let is_empty = set.uuid.as_ref() == [0u8;16];
if !is_empty {
bail!("media '{}' contains data (please use 'force' flag to remove.", changer_id);
}
}
}
let uuid = media_id.label.uuid.clone();
drop(media_id);
inventory.remove_media(&uuid)?;
let mut state_db = MediaStateDatabase::load(status_path)?;
state_db.remove_media(&uuid)?;
Ok(())
}
2020-12-14 06:55:57 +00:00
const SUBDIRS: SubdirMap = &[
2020-12-14 07:58:40 +00:00
(
"destroy",
&Router::new()
.get(&API_METHOD_DESTROY_MEDIA)
),
2020-12-14 06:55:57 +00:00
(
"list",
&Router::new()
.get(&API_METHOD_LIST_MEDIA)
),
];
pub const ROUTER: Router = Router::new()
.get(&list_subdirs_api_method!(SUBDIRS))
.subdirs(SUBDIRS);