tape: implement media content list api
This commit is contained in:
		@ -1,6 +1,7 @@
 | 
				
			|||||||
use std::path::Path;
 | 
					use std::path::Path;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use anyhow::{bail, format_err, Error};
 | 
					use anyhow::{bail, format_err, Error};
 | 
				
			||||||
 | 
					use serde::{Serialize, Deserialize};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
use proxmox::{
 | 
					use proxmox::{
 | 
				
			||||||
    api::{api, Router, SubdirMap},
 | 
					    api::{api, Router, SubdirMap},
 | 
				
			||||||
@ -12,17 +13,24 @@ use crate::{
 | 
				
			|||||||
        self,
 | 
					        self,
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    api2::types::{
 | 
					    api2::types::{
 | 
				
			||||||
 | 
					        BACKUP_ID_SCHEMA,
 | 
				
			||||||
 | 
					        BACKUP_TYPE_SCHEMA,
 | 
				
			||||||
        MEDIA_POOL_NAME_SCHEMA,
 | 
					        MEDIA_POOL_NAME_SCHEMA,
 | 
				
			||||||
        MEDIA_LABEL_SCHEMA,
 | 
					        MEDIA_LABEL_SCHEMA,
 | 
				
			||||||
        MediaPoolConfig,
 | 
					        MediaPoolConfig,
 | 
				
			||||||
        MediaListEntry,
 | 
					        MediaListEntry,
 | 
				
			||||||
        MediaStatus,
 | 
					        MediaStatus,
 | 
				
			||||||
 | 
					        MediaContentEntry,
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    backup::{
 | 
				
			||||||
 | 
					        BackupDir,
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    tape::{
 | 
					    tape::{
 | 
				
			||||||
        TAPE_STATUS_DIR,
 | 
					        TAPE_STATUS_DIR,
 | 
				
			||||||
        Inventory,
 | 
					        Inventory,
 | 
				
			||||||
        MediaStateDatabase,
 | 
					        MediaStateDatabase,
 | 
				
			||||||
        MediaPool,
 | 
					        MediaPool,
 | 
				
			||||||
 | 
					        MediaCatalog,
 | 
				
			||||||
        update_online_status,
 | 
					        update_online_status,
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
@ -181,6 +189,133 @@ pub fn destroy_media(changer_id: String, force: Option<bool>,) -> Result<(), Err
 | 
				
			|||||||
    Ok(())
 | 
					    Ok(())
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#[api(
 | 
				
			||||||
 | 
					    properties: {
 | 
				
			||||||
 | 
					        pool: {
 | 
				
			||||||
 | 
					            schema: MEDIA_POOL_NAME_SCHEMA,
 | 
				
			||||||
 | 
					            optional: true,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "changer-id": {
 | 
				
			||||||
 | 
					            schema: MEDIA_LABEL_SCHEMA,
 | 
				
			||||||
 | 
					            optional: true,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "media": {
 | 
				
			||||||
 | 
					            description: "Filter by media UUID.",
 | 
				
			||||||
 | 
					            type: String,
 | 
				
			||||||
 | 
					            optional: true,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "media-set": {
 | 
				
			||||||
 | 
					            description: "Filter by media set UUID.",
 | 
				
			||||||
 | 
					            type: String,
 | 
				
			||||||
 | 
					            optional: true,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "backup-type": {
 | 
				
			||||||
 | 
					            schema: BACKUP_TYPE_SCHEMA,
 | 
				
			||||||
 | 
					            optional: true,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        "backup-id": {
 | 
				
			||||||
 | 
					            schema: BACKUP_ID_SCHEMA,
 | 
				
			||||||
 | 
					            optional: true,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					)]
 | 
				
			||||||
 | 
					#[derive(Serialize,Deserialize)]
 | 
				
			||||||
 | 
					#[serde(rename_all="kebab-case")]
 | 
				
			||||||
 | 
					/// Content list filter parameters
 | 
				
			||||||
 | 
					pub struct MediaContentListFilter {
 | 
				
			||||||
 | 
					    pub pool: Option<String>,
 | 
				
			||||||
 | 
					    pub changer_id: Option<String>,
 | 
				
			||||||
 | 
					    pub media: Option<String>,
 | 
				
			||||||
 | 
					    pub media_set: Option<String>,
 | 
				
			||||||
 | 
					    pub backup_type: Option<String>,
 | 
				
			||||||
 | 
					    pub backup_id: Option<String>,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#[api(
 | 
				
			||||||
 | 
					    input: {
 | 
				
			||||||
 | 
					        properties: {
 | 
				
			||||||
 | 
					            "filter": {
 | 
				
			||||||
 | 
					                type: MediaContentListFilter,
 | 
				
			||||||
 | 
					                flatten: true,
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    returns: {
 | 
				
			||||||
 | 
					        description: "Media content list.",
 | 
				
			||||||
 | 
					        type: Array,
 | 
				
			||||||
 | 
					        items: {
 | 
				
			||||||
 | 
					            type: MediaContentEntry,
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					)]
 | 
				
			||||||
 | 
					/// List media content
 | 
				
			||||||
 | 
					pub fn list_content(
 | 
				
			||||||
 | 
					    filter: MediaContentListFilter,
 | 
				
			||||||
 | 
					) -> Result<Vec<MediaContentEntry>, Error> {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let (config, _digest) = config::media_pool::config()?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let status_path = Path::new(TAPE_STATUS_DIR);
 | 
				
			||||||
 | 
					    let inventory = Inventory::load(status_path)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let media_uuid = filter.media.and_then(|s| s.parse().ok());
 | 
				
			||||||
 | 
					    let media_set_uuid = filter.media_set.and_then(|s| s.parse().ok());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let mut list = Vec::new();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    for media_id in inventory.list_used_media() {
 | 
				
			||||||
 | 
					        let set = media_id.media_set_label.as_ref().unwrap();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if let Some(ref changer_id) = filter.changer_id {
 | 
				
			||||||
 | 
					            if &media_id.label.changer_id != changer_id { continue; }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if let Some(ref pool) = filter.pool {
 | 
				
			||||||
 | 
					            if &set.pool != pool { continue; }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if let Some(ref media_uuid) = media_uuid {
 | 
				
			||||||
 | 
					            if &media_id.label.uuid != media_uuid { continue; }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        if let Some(ref media_set_uuid) = media_set_uuid {
 | 
				
			||||||
 | 
					            if &set.uuid != media_set_uuid { continue; }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        let config: MediaPoolConfig = config.lookup("pool", &set.pool)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        let media_set_name = inventory
 | 
				
			||||||
 | 
					            .generate_media_set_name(&set.uuid, config.template.clone())
 | 
				
			||||||
 | 
					            .unwrap_or_else(|_| set.uuid.to_string());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        let catalog = MediaCatalog::open(status_path, &media_id.label.uuid, false, false)?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        for snapshot in catalog.snapshot_index().keys() {
 | 
				
			||||||
 | 
					            let backup_dir: BackupDir = snapshot.parse()?;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if let Some(ref backup_type) = filter.backup_type {
 | 
				
			||||||
 | 
					                if backup_dir.group().backup_type() != backup_type { continue; }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            if let Some(ref backup_id) = filter.backup_id {
 | 
				
			||||||
 | 
					                if backup_dir.group().backup_id() != backup_id { continue; }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            list.push(MediaContentEntry {
 | 
				
			||||||
 | 
					                uuid: media_id.label.uuid.to_string(),
 | 
				
			||||||
 | 
					                changer_id: media_id.label.changer_id.to_string(),
 | 
				
			||||||
 | 
					                pool: set.pool.clone(),
 | 
				
			||||||
 | 
					                media_set_name: media_set_name.clone(),
 | 
				
			||||||
 | 
					                media_set_uuid: set.uuid.to_string(),
 | 
				
			||||||
 | 
					                seq_nr: set.seq_nr,
 | 
				
			||||||
 | 
					                snapshot: snapshot.to_owned(),
 | 
				
			||||||
 | 
					                backup_time: backup_dir.backup_time(),
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Ok(list)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const SUBDIRS: SubdirMap = &[
 | 
					const SUBDIRS: SubdirMap = &[
 | 
				
			||||||
    (
 | 
					    (
 | 
				
			||||||
        "destroy",
 | 
					        "destroy",
 | 
				
			||||||
@ -192,6 +327,11 @@ const SUBDIRS: SubdirMap = &[
 | 
				
			|||||||
        &Router::new()
 | 
					        &Router::new()
 | 
				
			||||||
            .get(&API_METHOD_LIST_MEDIA)
 | 
					            .get(&API_METHOD_LIST_MEDIA)
 | 
				
			||||||
    ),
 | 
					    ),
 | 
				
			||||||
 | 
					    (
 | 
				
			||||||
 | 
					        "content",
 | 
				
			||||||
 | 
					        &Router::new()
 | 
				
			||||||
 | 
					            .get(&API_METHOD_LIST_CONTENT)
 | 
				
			||||||
 | 
					    ),
 | 
				
			||||||
];
 | 
					];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -79,3 +79,26 @@ pub struct LabelUuidMap {
 | 
				
			|||||||
    /// Associated Uuid (if any)
 | 
					    /// Associated Uuid (if any)
 | 
				
			||||||
    pub uuid: Option<String>,
 | 
					    pub uuid: Option<String>,
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#[api()]
 | 
				
			||||||
 | 
					#[derive(Serialize,Deserialize)]
 | 
				
			||||||
 | 
					#[serde(rename_all = "kebab-case")]
 | 
				
			||||||
 | 
					/// Media content list entry
 | 
				
			||||||
 | 
					pub struct MediaContentEntry {
 | 
				
			||||||
 | 
					    /// Media changer ID
 | 
				
			||||||
 | 
					    pub changer_id: String,
 | 
				
			||||||
 | 
					    /// Media Uuid
 | 
				
			||||||
 | 
					    pub uuid: String,
 | 
				
			||||||
 | 
					    /// Media set name
 | 
				
			||||||
 | 
					    pub media_set_name: String,
 | 
				
			||||||
 | 
					    /// Media set uuid
 | 
				
			||||||
 | 
					    pub media_set_uuid: String,
 | 
				
			||||||
 | 
					    /// Media set seq_nr
 | 
				
			||||||
 | 
					    pub seq_nr: u64,
 | 
				
			||||||
 | 
					    /// Media Pool
 | 
				
			||||||
 | 
					    pub pool: String,
 | 
				
			||||||
 | 
					    /// Backup snapshot
 | 
				
			||||||
 | 
					    pub snapshot: String,
 | 
				
			||||||
 | 
					    /// Snapshot creation time (epoch)
 | 
				
			||||||
 | 
					    pub backup_time: i64,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -18,8 +18,13 @@ use proxmox_backup::{
 | 
				
			|||||||
            MediaStatus,
 | 
					            MediaStatus,
 | 
				
			||||||
            MediaListEntry,
 | 
					            MediaListEntry,
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
 | 
					        tape::media::MediaContentListFilter,
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					    tape::{
 | 
				
			||||||
 | 
					        complete_media_changer_id,
 | 
				
			||||||
 | 
					        complete_media_uuid,
 | 
				
			||||||
 | 
					        complete_media_set_uuid,
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
    tape::complete_media_changer_id,
 | 
					 | 
				
			||||||
    config::{
 | 
					    config::{
 | 
				
			||||||
        media_pool::complete_pool_name,
 | 
					        media_pool::complete_pool_name,
 | 
				
			||||||
    },
 | 
					    },
 | 
				
			||||||
@ -39,6 +44,14 @@ pub fn media_commands() -> CommandLineInterface {
 | 
				
			|||||||
                .arg_param(&["changer-id"])
 | 
					                .arg_param(&["changer-id"])
 | 
				
			||||||
                .completion_cb("changer-id", complete_media_changer_id)
 | 
					                .completion_cb("changer-id", complete_media_changer_id)
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
 | 
					        .insert(
 | 
				
			||||||
 | 
					            "content",
 | 
				
			||||||
 | 
					            CliCommand::new(&API_METHOD_LIST_CONTENT)
 | 
				
			||||||
 | 
					                .completion_cb("pool", complete_pool_name)
 | 
				
			||||||
 | 
					                .completion_cb("changer-id", complete_media_changer_id)
 | 
				
			||||||
 | 
					                .completion_cb("media", complete_media_uuid)
 | 
				
			||||||
 | 
					                .completion_cb("media-set", complete_media_set_uuid)
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
        ;
 | 
					        ;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    cmd_def.into()
 | 
					    cmd_def.into()
 | 
				
			||||||
@ -110,3 +123,49 @@ async fn list_media(
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    Ok(())
 | 
					    Ok(())
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#[api(
 | 
				
			||||||
 | 
					    input: {
 | 
				
			||||||
 | 
					        properties: {
 | 
				
			||||||
 | 
					            "filter": {
 | 
				
			||||||
 | 
					                type: MediaContentListFilter,
 | 
				
			||||||
 | 
					                flatten: true,
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					            "output-format": {
 | 
				
			||||||
 | 
					                schema: OUTPUT_FORMAT,
 | 
				
			||||||
 | 
					                optional: true,
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					)]
 | 
				
			||||||
 | 
					/// List media content
 | 
				
			||||||
 | 
					fn list_content(
 | 
				
			||||||
 | 
					    param: Value,
 | 
				
			||||||
 | 
					    rpcenv: &mut dyn RpcEnvironment,
 | 
				
			||||||
 | 
					) -> Result<(), Error> {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let output_format = get_output_format(¶m);
 | 
				
			||||||
 | 
					    let info = &api2::tape::media::API_METHOD_LIST_CONTENT;
 | 
				
			||||||
 | 
					    let mut data = match info.handler {
 | 
				
			||||||
 | 
					        ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
 | 
				
			||||||
 | 
					        _ => unreachable!(),
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let options = default_table_format_options()
 | 
				
			||||||
 | 
					        .sortby("media-set-uuid", false)
 | 
				
			||||||
 | 
					        .sortby("seq-nr", false)
 | 
				
			||||||
 | 
					        .sortby("snapshot", false)
 | 
				
			||||||
 | 
					        .sortby("backup-time", false)
 | 
				
			||||||
 | 
					        .column(ColumnConfig::new("changer-id"))
 | 
				
			||||||
 | 
					        .column(ColumnConfig::new("pool"))
 | 
				
			||||||
 | 
					        .column(ColumnConfig::new("media-set-name"))
 | 
				
			||||||
 | 
					        .column(ColumnConfig::new("seq-nr"))
 | 
				
			||||||
 | 
					        .column(ColumnConfig::new("snapshot"))
 | 
				
			||||||
 | 
					        .column(ColumnConfig::new("media-set-uuid"))
 | 
				
			||||||
 | 
					        ;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    format_and_print_result_full(&mut data, &info.returns, &output_format, &options);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Ok(())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user