tape: add media api
This commit is contained in:
parent
b5c1296eaa
commit
fba0b77469
168
src/api2/tape/media.rs
Normal file
168
src/api2/tape/media.rs
Normal file
@ -0,0 +1,168 @@
|
|||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
use anyhow::Error;
|
||||||
|
|
||||||
|
use proxmox::api::{api, Router, SubdirMap};
|
||||||
|
use proxmox::list_subdirs_api_method;
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
config::{
|
||||||
|
self,
|
||||||
|
},
|
||||||
|
api2::types::{
|
||||||
|
MEDIA_POOL_NAME_SCHEMA,
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
const SUBDIRS: SubdirMap = &[
|
||||||
|
(
|
||||||
|
"list",
|
||||||
|
&Router::new()
|
||||||
|
.get(&API_METHOD_LIST_MEDIA)
|
||||||
|
),
|
||||||
|
];
|
||||||
|
|
||||||
|
|
||||||
|
pub const ROUTER: Router = Router::new()
|
||||||
|
.get(&list_subdirs_api_method!(SUBDIRS))
|
||||||
|
.subdirs(SUBDIRS);
|
@ -4,10 +4,12 @@ use proxmox::list_subdirs_api_method;
|
|||||||
|
|
||||||
pub mod drive;
|
pub mod drive;
|
||||||
pub mod changer;
|
pub mod changer;
|
||||||
|
pub mod media;
|
||||||
|
|
||||||
pub const SUBDIRS: SubdirMap = &[
|
pub const SUBDIRS: SubdirMap = &[
|
||||||
("changer", &changer::ROUTER),
|
("changer", &changer::ROUTER),
|
||||||
("drive", &drive::ROUTER),
|
("drive", &drive::ROUTER),
|
||||||
|
("media", &media::ROUTER),
|
||||||
];
|
];
|
||||||
|
|
||||||
pub const ROUTER: Router = Router::new()
|
pub const ROUTER: Router = Router::new()
|
||||||
|
@ -455,6 +455,7 @@ fn main() {
|
|||||||
.insert("changer", changer_commands())
|
.insert("changer", changer_commands())
|
||||||
.insert("drive", drive_commands())
|
.insert("drive", drive_commands())
|
||||||
.insert("pool", pool_commands())
|
.insert("pool", pool_commands())
|
||||||
|
.insert("media", media_commands())
|
||||||
.insert(
|
.insert(
|
||||||
"load-media",
|
"load-media",
|
||||||
CliCommand::new(&API_METHOD_LOAD_MEDIA)
|
CliCommand::new(&API_METHOD_LOAD_MEDIA)
|
||||||
|
119
src/bin/proxmox_tape/media.rs
Normal file
119
src/bin/proxmox_tape/media.rs
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
use anyhow::{Error};
|
||||||
|
use serde_json::Value;
|
||||||
|
|
||||||
|
use proxmox::{
|
||||||
|
api::{
|
||||||
|
api,
|
||||||
|
cli::*,
|
||||||
|
RpcEnvironment,
|
||||||
|
ApiHandler,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
use proxmox_backup::{
|
||||||
|
api2::{
|
||||||
|
self,
|
||||||
|
types::{
|
||||||
|
MEDIA_POOL_NAME_SCHEMA,
|
||||||
|
MediaLocationKind,
|
||||||
|
MediaStatus,
|
||||||
|
MediaListEntry,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
config::{
|
||||||
|
media_pool::complete_pool_name,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
pub fn media_commands() -> CommandLineInterface {
|
||||||
|
|
||||||
|
let cmd_def = CliCommandMap::new()
|
||||||
|
.insert(
|
||||||
|
"list",
|
||||||
|
CliCommand::new(&API_METHOD_LIST_MEDIA)
|
||||||
|
.completion_cb("pool", complete_pool_name)
|
||||||
|
)
|
||||||
|
;
|
||||||
|
|
||||||
|
cmd_def.into()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[api(
|
||||||
|
input: {
|
||||||
|
properties: {
|
||||||
|
pool: {
|
||||||
|
schema: MEDIA_POOL_NAME_SCHEMA,
|
||||||
|
optional: true,
|
||||||
|
},
|
||||||
|
"output-format": {
|
||||||
|
schema: OUTPUT_FORMAT,
|
||||||
|
optional: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)]
|
||||||
|
/// List pool media
|
||||||
|
async fn list_media(
|
||||||
|
param: Value,
|
||||||
|
rpcenv: &mut dyn RpcEnvironment,
|
||||||
|
) -> Result<(), Error> {
|
||||||
|
|
||||||
|
let output_format = get_output_format(¶m);
|
||||||
|
let info = &api2::tape::media::API_METHOD_LIST_MEDIA;
|
||||||
|
let mut data = match info.handler {
|
||||||
|
ApiHandler::Async(handler) => (handler)(param, info, rpcenv).await?,
|
||||||
|
_ => unreachable!(),
|
||||||
|
};
|
||||||
|
|
||||||
|
fn render_location(_value: &Value, record: &Value) -> Result<String, Error> {
|
||||||
|
let record: MediaListEntry = serde_json::from_value(record.clone())?;
|
||||||
|
Ok(match record.location {
|
||||||
|
MediaLocationKind::Online => {
|
||||||
|
record.location_hint.unwrap_or(String::from("-"))
|
||||||
|
}
|
||||||
|
MediaLocationKind::Offline => String::from("offline"),
|
||||||
|
MediaLocationKind::Vault => {
|
||||||
|
format!("V({})", record.location_hint.unwrap_or(String::from("-")))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_status(_value: &Value, record: &Value) -> Result<String, Error> {
|
||||||
|
let record: MediaListEntry = serde_json::from_value(record.clone())?;
|
||||||
|
Ok(match record.status {
|
||||||
|
MediaStatus::Damaged | MediaStatus::Retired => {
|
||||||
|
serde_json::to_value(&record.status)?
|
||||||
|
.as_str().unwrap()
|
||||||
|
.to_string()
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
if record.expired {
|
||||||
|
String::from("expired")
|
||||||
|
} else {
|
||||||
|
serde_json::to_value(&record.status)?
|
||||||
|
.as_str().unwrap()
|
||||||
|
.to_string()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
let options = default_table_format_options()
|
||||||
|
.sortby("pool", false)
|
||||||
|
.sortby("media-set-uuid", false)
|
||||||
|
.sortby("seq-nr", false)
|
||||||
|
.sortby("changer-id", false)
|
||||||
|
.column(ColumnConfig::new("changer-id"))
|
||||||
|
.column(ColumnConfig::new("pool"))
|
||||||
|
.column(ColumnConfig::new("media-set-name"))
|
||||||
|
.column(ColumnConfig::new("seq-nr"))
|
||||||
|
.column(ColumnConfig::new("status").renderer(render_status))
|
||||||
|
.column(ColumnConfig::new("location").renderer(render_location))
|
||||||
|
.column(ColumnConfig::new("uuid"))
|
||||||
|
.column(ColumnConfig::new("media-set-uuid"))
|
||||||
|
;
|
||||||
|
|
||||||
|
format_and_print_result_full(&mut data, info.returns, &output_format, &options);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
@ -6,3 +6,6 @@ pub use drive::*;
|
|||||||
|
|
||||||
mod pool;
|
mod pool;
|
||||||
pub use pool::*;
|
pub use pool::*;
|
||||||
|
|
||||||
|
mod media;
|
||||||
|
pub use media::*;
|
||||||
|
Loading…
Reference in New Issue
Block a user