From bff7e3f3e4ddeeeaf06bc8a0197f4c62c1c19fce Mon Sep 17 00:00:00 2001 From: Dietmar Maurer Date: Fri, 11 Dec 2020 07:50:19 +0100 Subject: [PATCH] tape: implement barcode-label-mdedia --- src/api2/tape/drive.rs | 97 +++++++++++++++++++++++++++++++++++++++++ src/bin/proxmox-tape.rs | 40 +++++++++++++++++ 2 files changed, 137 insertions(+) diff --git a/src/api2/tape/drive.rs b/src/api2/tape/drive.rs index 1f40a63d..784b24a6 100644 --- a/src/api2/tape/drive.rs +++ b/src/api2/tape/drive.rs @@ -522,8 +522,105 @@ pub fn inventory( } +#[api( + input: { + properties: { + drive: { + schema: DRIVE_ID_SCHEMA, + }, + pool: { + schema: MEDIA_POOL_NAME_SCHEMA, + optional: true, + }, + }, + }, +)] +/// Label media with barcodes from changer device +pub fn barcode_label_media( + drive: String, + pool: Option, +) -> Result<(), Error> { + + if let Some(ref pool) = pool { + let (pool_config, _digest) = config::media_pool::config()?; + + if pool_config.sections.get(pool).is_none() { + bail!("no such pool ('{}')", pool); + } + } + + let (config, _digest) = config::drive::config()?; + + let (mut changer, changer_name) = media_changer(&config, &drive, false)?; + + let changer_id_list = changer.list_media_changer_ids()?; + + let state_path = Path::new(TAPE_STATUS_DIR); + + let mut inventory = Inventory::load(state_path)?; + let mut state_db = MediaStateDatabase::load(state_path)?; + + update_changer_online_status(&config, &mut inventory, &mut state_db, &changer_name, &changer_id_list)?; + + if changer_id_list.is_empty() { + bail!("changer device does not list any media labels"); + } + + for changer_id in changer_id_list { + if changer_id.starts_with("CLN") { continue; } + + inventory.reload()?; + if inventory.find_media_by_changer_id(&changer_id).is_some() { + println!("media '{}' already inventoried (already labeled)", changer_id); + continue; + } + + println!("checking/loading media '{}'", changer_id); + + if let Err(err) = changer.load_media(&changer_id) { + eprintln!("unable to load media '{}' - {}", changer_id, err); + continue; + } + + let mut drive = open_drive(&config, &drive)?; + drive.rewind()?; + + match drive.read_next_file() { + Ok(Some(_file)) => { + println!("media '{}' is not empty (erase first)", changer_id); + continue; + } + Ok(None) => { /* EOF mark at BOT, assume tape is empty */ }, + Err(err) => { + if err.is_errno(nix::errno::Errno::ENOSPC) || err.is_errno(nix::errno::Errno::EIO) { + /* assume tape is empty */ + } else { + println!("media '{}' read error (maybe not empty - erase first)", changer_id); + continue; + } + } + } + + let ctime = proxmox::tools::time::epoch_i64(); + let label = DriveLabel { + changer_id: changer_id.to_string(), + uuid: Uuid::generate(), + ctime, + }; + + write_media_label(&mut drive, label, pool.clone())? + } + + Ok(()) +} + #[sortable] pub const SUBDIRS: SubdirMap = &sorted!([ + ( + "barcode-label-media", + &Router::new() + .put(&API_METHOD_BARCODE_LABEL_MEDIA) + ), ( "eject-media", &Router::new() diff --git a/src/bin/proxmox-tape.rs b/src/bin/proxmox-tape.rs index 1b1f3ddc..c97fed53 100644 --- a/src/bin/proxmox-tape.rs +++ b/src/bin/proxmox-tape.rs @@ -326,9 +326,49 @@ fn inventory( Ok(()) } +#[api( + input: { + properties: { + pool: { + schema: MEDIA_POOL_NAME_SCHEMA, + optional: true, + }, + drive: { + schema: DRIVE_ID_SCHEMA, + optional: true, + }, + }, + }, +)] +/// Label media with barcodes from changer device +fn barcode_label_media( + mut param: Value, + rpcenv: &mut dyn RpcEnvironment, +) -> Result<(), Error> { + + let (config, _digest) = config::drive::config()?; + + param["drive"] = lookup_drive_name(¶m, &config)?.into(); + + let info = &api2::tape::drive::API_METHOD_BARCODE_LABEL_MEDIA; + + match info.handler { + ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?, + _ => unreachable!(), + }; + + Ok(()) +} + fn main() { let cmd_def = CliCommandMap::new() + .insert( + "barcode-label", + CliCommand::new(&API_METHOD_BARCODE_LABEL_MEDIA) + .completion_cb("drive", complete_drive_name) + .completion_cb("pool", complete_pool_name) + ) .insert( "rewind", CliCommand::new(&API_METHOD_REWIND)