use std::path::Path; use std::sync::Arc; use anyhow::{bail, Error}; use serde_json::Value; use proxmox::{ sortable, identity, list_subdirs_api_method, tools::Uuid, sys::error::SysError, api::{ api, RpcEnvironment, RpcEnvironmentType, Router, SubdirMap, }, }; use crate::{ config::{ self, drive::check_drive_exists, }, api2::{ types::{ UPID_SCHEMA, CHANGER_NAME_SCHEMA, DRIVE_NAME_SCHEMA, MEDIA_LABEL_SCHEMA, MEDIA_POOL_NAME_SCHEMA, Authid, DriveListEntry, LinuxTapeDrive, TapeDeviceInfo, MediaIdFlat, LabelUuidMap, MamAttribute, LinuxDriveAndMediaStatus, }, tape::restore::restore_media, }, server::WorkerTask, tape::{ TAPE_STATUS_DIR, MediaPool, Inventory, MediaCatalog, MediaId, linux_tape_device_list, lookup_drive, file_formats::{ MediaLabel, MediaSetLabel, }, drive::{ TapeDriver, LinuxTapeHandle, Lp17VolumeStatistics, open_linux_tape_device, media_changer, required_media_changer, open_drive, }, changer::update_changer_online_status, }, }; #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, "label-text": { schema: MEDIA_LABEL_SCHEMA, }, }, }, )] /// Load media with specified label /// /// Issue a media load request to the associated changer device. pub async fn load_media(drive: String, label_text: String) -> Result<(), Error> { let (config, _digest) = config::drive::config()?; tokio::task::spawn_blocking(move || { let (mut changer, _) = required_media_changer(&config, &drive)?; changer.load_media(&label_text) }).await? } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, "source-slot": { description: "Source slot number.", minimum: 1, }, }, }, )] /// Load media from the specified slot /// /// Issue a media load request to the associated changer device. pub async fn load_slot(drive: String, source_slot: u64) -> Result<(), Error> { let (config, _digest) = config::drive::config()?; tokio::task::spawn_blocking(move || { let (mut changer, _) = required_media_changer(&config, &drive)?; changer.load_media_from_slot(source_slot) }).await? } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, "label-text": { schema: MEDIA_LABEL_SCHEMA, }, }, }, returns: { description: "The import-export slot number the media was transfered to.", type: u64, minimum: 1, }, )] /// Export media with specified label pub async fn export_media(drive: String, label_text: String) -> Result { let (config, _digest) = config::drive::config()?; tokio::task::spawn_blocking(move || { let (mut changer, changer_name) = required_media_changer(&config, &drive)?; match changer.export_media(&label_text)? { Some(slot) => Ok(slot), None => bail!("media '{}' is not online (via changer '{}')", label_text, changer_name), } }).await? } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, "target-slot": { description: "Target slot number. If omitted, defaults to the slot that the drive was loaded from.", minimum: 1, optional: true, }, }, }, )] /// Unload media via changer pub async fn unload( drive: String, target_slot: Option, _param: Value, ) -> Result<(), Error> { let (config, _digest) = config::drive::config()?; tokio::task::spawn_blocking(move || { let (mut changer, _) = required_media_changer(&config, &drive)?; changer.unload_media(target_slot) }).await? } #[api( input: { properties: {}, }, returns: { description: "The list of autodetected tape drives.", type: Array, items: { type: TapeDeviceInfo, }, }, )] /// Scan tape drives pub fn scan_drives(_param: Value) -> Result, Error> { let list = linux_tape_device_list(); Ok(list) } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, fast: { description: "Use fast erase.", type: bool, optional: true, default: true, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Erase media pub fn erase_media( drive: String, fast: Option, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let (config, _digest) = config::drive::config()?; check_drive_exists(&config, &drive)?; // early check before starting worker let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "erase-media", Some(drive.clone()), auth_id, to_stdout, move |_worker| { let mut drive = open_drive(&config, &drive)?; drive.erase_media(fast.unwrap_or(true))?; Ok(()) } )?; Ok(upid_str.into()) } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Rewind tape pub fn rewind( drive: String, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let (config, _digest) = config::drive::config()?; check_drive_exists(&config, &drive)?; // early check before starting worker let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "rewind-media", Some(drive.clone()), auth_id, to_stdout, move |_worker| { let mut drive = open_drive(&config, &drive)?; drive.rewind()?; Ok(()) } )?; Ok(upid_str.into()) } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Eject/Unload drive media pub fn eject_media( drive: String, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let (config, _digest) = config::drive::config()?; check_drive_exists(&config, &drive)?; // early check before starting worker let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "eject-media", Some(drive.clone()), auth_id, to_stdout, move |_worker| { if let Some((mut changer, _)) = media_changer(&config, &drive)? { changer.unload_media(None)?; } else { let mut drive = open_drive(&config, &drive)?; drive.eject_media()?; } Ok(()) })?; Ok(upid_str.into()) } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, "label-text": { schema: MEDIA_LABEL_SCHEMA, }, pool: { schema: MEDIA_POOL_NAME_SCHEMA, optional: true, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Label media /// /// Write a new media label to the media in 'drive'. The media is /// assigned to the specified 'pool', or else to the free media pool. /// /// Note: The media need to be empty (you may want to erase it first). pub fn label_media( drive: String, pool: Option, label_text: String, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; 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 to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "label-media", Some(drive.clone()), auth_id, to_stdout, move |worker| { let mut drive = open_drive(&config, &drive)?; drive.rewind()?; match drive.read_next_file() { Ok(Some(_file)) => bail!("media is not empty (erase first)"), 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 { bail!("media read error - {}", err); } } } let ctime = proxmox::tools::time::epoch_i64(); let label = MediaLabel { label_text: label_text.to_string(), uuid: Uuid::generate(), ctime, }; write_media_label(worker, &mut drive, label, pool) } )?; Ok(upid_str.into()) } fn write_media_label( worker: Arc, drive: &mut Box, label: MediaLabel, pool: Option, ) -> Result<(), Error> { drive.label_tape(&label)?; let mut media_set_label = None; if let Some(ref pool) = pool { // assign media to pool by writing special media set label worker.log(format!("Label media '{}' for pool '{}'", label.label_text, pool)); let set = MediaSetLabel::with_data(&pool, [0u8; 16].into(), 0, label.ctime, None); drive.write_media_set_label(&set, None)?; media_set_label = Some(set); } else { worker.log(format!("Label media '{}' (no pool assignment)", label.label_text)); } let media_id = MediaId { label, media_set_label }; let status_path = Path::new(TAPE_STATUS_DIR); // Create the media catalog MediaCatalog::overwrite(status_path, &media_id, false)?; let mut inventory = Inventory::load(status_path)?; inventory.store(media_id.clone(), false)?; drive.rewind()?; match drive.read_label() { Ok((Some(info), _)) => { if info.label.uuid != media_id.label.uuid { bail!("verify label failed - got wrong label uuid"); } if let Some(ref pool) = pool { match info.media_set_label { Some(set) => { if set.uuid != [0u8; 16].into() { bail!("verify media set label failed - got wrong set uuid"); } if &set.pool != pool { bail!("verify media set label failed - got wrong pool"); } } None => { bail!("verify media set label failed (missing set label)"); } } } }, Ok((None, _)) => bail!("verify label failed (got empty media)"), Err(err) => bail!("verify label failed - {}", err), }; drive.rewind()?; Ok(()) } #[api( protected: true, input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, password: { description: "Encryption key password.", }, }, }, )] /// Try to restore a tape encryption key pub async fn restore_key( drive: String, password: String, ) -> Result<(), Error> { let (config, _digest) = config::drive::config()?; tokio::task::spawn_blocking(move || { let mut drive = open_drive(&config, &drive)?; let (_media_id, key_config) = drive.read_label()?; if let Some(key_config) = key_config { let password_fn = || { Ok(password.as_bytes().to_vec()) }; let (key, ..) = key_config.decrypt(&password_fn)?; config::tape_encryption_keys::insert_key(key, key_config, true)?; } else { bail!("media does not contain any encryption key configuration"); } Ok(()) }).await? } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, inventorize: { description: "Inventorize media", optional: true, }, }, }, returns: { type: MediaIdFlat, }, )] /// Read media label (optionally inventorize media) pub async fn read_label( drive: String, inventorize: Option, ) -> Result { let (config, _digest) = config::drive::config()?; tokio::task::spawn_blocking(move || { let mut drive = open_drive(&config, &drive)?; let (media_id, _key_config) = drive.read_label()?; let media_id = match media_id { Some(media_id) => { let mut flat = MediaIdFlat { uuid: media_id.label.uuid.clone(), label_text: media_id.label.label_text.clone(), ctime: media_id.label.ctime, media_set_ctime: None, media_set_uuid: None, encryption_key_fingerprint: None, pool: None, seq_nr: None, }; if let Some(ref set) = media_id.media_set_label { flat.pool = Some(set.pool.clone()); flat.seq_nr = Some(set.seq_nr); flat.media_set_uuid = Some(set.uuid.clone()); flat.media_set_ctime = Some(set.ctime); flat.encryption_key_fingerprint = set .encryption_key_fingerprint .as_ref() .map(|fp| crate::tools::format::as_fingerprint(fp.bytes())); let encrypt_fingerprint = set.encryption_key_fingerprint.clone() .map(|fp| (fp, set.uuid.clone())); if let Err(err) = drive.set_encryption(encrypt_fingerprint) { // try, but ignore errors. just log to stderr eprintln!("uable to load encryption key: {}", err); } } if let Some(true) = inventorize { let state_path = Path::new(TAPE_STATUS_DIR); let mut inventory = Inventory::load(state_path)?; inventory.store(media_id, false)?; } flat } None => { bail!("Media is empty (no label)."); } }; Ok(media_id) }).await? } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Clean drive pub fn clean_drive( drive: String, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let (config, _digest) = config::drive::config()?; check_drive_exists(&config, &drive)?; // early check before starting worker let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "clean-drive", Some(drive.clone()), auth_id, to_stdout, move |worker| { let (mut changer, _changer_name) = required_media_changer(&config, &drive)?; worker.log("Starting drive clean"); changer.clean_drive()?; worker.log("Drive cleaned sucessfully"); Ok(()) })?; Ok(upid_str.into()) } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, }, }, returns: { description: "The list of media labels with associated media Uuid (if any).", type: Array, items: { type: LabelUuidMap, }, }, )] /// List known media labels (Changer Inventory) /// /// Note: Only useful for drives with associated changer device. /// /// This method queries the changer to get a list of media labels. /// /// Note: This updates the media online status. pub async fn inventory( drive: String, ) -> Result, Error> { let (config, _digest) = config::drive::config()?; tokio::task::spawn_blocking(move || { let (mut changer, changer_name) = required_media_changer(&config, &drive)?; let label_text_list = changer.online_media_label_texts()?; let state_path = Path::new(TAPE_STATUS_DIR); let mut inventory = Inventory::load(state_path)?; update_changer_online_status( &config, &mut inventory, &changer_name, &label_text_list, )?; let mut list = Vec::new(); for label_text in label_text_list.iter() { if label_text.starts_with("CLN") { // skip cleaning unit continue; } let label_text = label_text.to_string(); if let Some(media_id) = inventory.find_media_by_label_text(&label_text) { list.push(LabelUuidMap { label_text, uuid: Some(media_id.label.uuid.clone()) }); } else { list.push(LabelUuidMap { label_text, uuid: None }); } } Ok(list) }).await? } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, "read-all-labels": { description: "Load all tapes and try read labels (even if already inventoried)", type: bool, optional: true, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Update inventory /// /// Note: Only useful for drives with associated changer device. /// /// This method queries the changer to get a list of media labels. It /// then loads any unknown media into the drive, reads the label, and /// store the result to the media database. /// /// Note: This updates the media online status. pub fn update_inventory( drive: String, read_all_labels: Option, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let (config, _digest) = config::drive::config()?; check_drive_exists(&config, &drive)?; // early check before starting worker let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "inventory-update", Some(drive.clone()), auth_id, to_stdout, move |worker| { let (mut changer, changer_name) = required_media_changer(&config, &drive)?; let label_text_list = changer.online_media_label_texts()?; if label_text_list.is_empty() { worker.log("changer device does not list any media labels".to_string()); } let state_path = Path::new(TAPE_STATUS_DIR); let mut inventory = Inventory::load(state_path)?; update_changer_online_status(&config, &mut inventory, &changer_name, &label_text_list)?; for label_text in label_text_list.iter() { if label_text.starts_with("CLN") { worker.log(format!("skip cleaning unit '{}'", label_text)); continue; } let label_text = label_text.to_string(); if !read_all_labels.unwrap_or(false) && inventory.find_media_by_label_text(&label_text).is_some() { worker.log(format!("media '{}' already inventoried", label_text)); continue; } if let Err(err) = changer.load_media(&label_text) { worker.warn(format!("unable to load media '{}' - {}", label_text, err)); continue; } let mut drive = open_drive(&config, &drive)?; match drive.read_label() { Err(err) => { worker.warn(format!("unable to read label form media '{}' - {}", label_text, err)); } Ok((None, _)) => { worker.log(format!("media '{}' is empty", label_text)); } Ok((Some(media_id), _key_config)) => { if label_text != media_id.label.label_text { worker.warn(format!("label text missmatch ({} != {})", label_text, media_id.label.label_text)); continue; } worker.log(format!("inventorize media '{}' with uuid '{}'", label_text, media_id.label.uuid)); inventory.store(media_id, false)?; } } changer.unload_media(None)?; } Ok(()) } )?; Ok(upid_str.into()) } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, pool: { schema: MEDIA_POOL_NAME_SCHEMA, optional: true, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Label media with barcodes from changer device pub fn barcode_label_media( drive: String, pool: Option, rpcenv: &mut dyn RpcEnvironment, ) -> Result { 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 auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "barcode-label-media", Some(drive.clone()), auth_id, to_stdout, move |worker| { barcode_label_media_worker(worker, drive, pool) } )?; Ok(upid_str.into()) } fn barcode_label_media_worker( worker: Arc, drive: String, pool: Option, ) -> Result<(), Error> { let (config, _digest) = config::drive::config()?; let (mut changer, changer_name) = required_media_changer(&config, &drive)?; let label_text_list = changer.online_media_label_texts()?; let state_path = Path::new(TAPE_STATUS_DIR); let mut inventory = Inventory::load(state_path)?; update_changer_online_status(&config, &mut inventory, &changer_name, &label_text_list)?; if label_text_list.is_empty() { bail!("changer device does not list any media labels"); } for label_text in label_text_list { if label_text.starts_with("CLN") { continue; } inventory.reload()?; if inventory.find_media_by_label_text(&label_text).is_some() { worker.log(format!("media '{}' already inventoried (already labeled)", label_text)); continue; } worker.log(format!("checking/loading media '{}'", label_text)); if let Err(err) = changer.load_media(&label_text) { worker.warn(format!("unable to load media '{}' - {}", label_text, err)); continue; } let mut drive = open_drive(&config, &drive)?; drive.rewind()?; match drive.read_next_file() { Ok(Some(_file)) => { worker.log(format!("media '{}' is not empty (erase first)", label_text)); 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 { worker.warn(format!("media '{}' read error (maybe not empty - erase first)", label_text)); continue; } } } let ctime = proxmox::tools::time::epoch_i64(); let label = MediaLabel { label_text: label_text.to_string(), uuid: Uuid::generate(), ctime, }; write_media_label(worker.clone(), &mut drive, label, pool.clone())? } Ok(()) } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, }, }, returns: { description: "A List of medium auxiliary memory attributes.", type: Array, items: { type: MamAttribute, }, }, )] /// Read Cartridge Memory (Medium auxiliary memory attributes) pub fn cartridge_memory(drive: String) -> Result, Error> { let (config, _digest) = config::drive::config()?; let drive_config: LinuxTapeDrive = config.lookup("linux", &drive)?; let mut handle = drive_config.open()?; handle.cartridge_memory() } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, }, }, returns: { type: Lp17VolumeStatistics, }, )] /// Read Volume Statistics (SCSI log page 17h) pub fn volume_statistics(drive: String) -> Result { let (config, _digest) = config::drive::config()?; let drive_config: LinuxTapeDrive = config.lookup("linux", &drive)?; let mut handle = drive_config.open()?; handle.volume_statistics() } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, }, }, returns: { type: LinuxDriveAndMediaStatus, }, )] /// Get drive/media status pub fn status(drive: String) -> Result { let (config, _digest) = config::drive::config()?; let drive_config: LinuxTapeDrive = config.lookup("linux", &drive)?; // Note: use open_linux_tape_device, because this also works if no medium loaded let file = open_linux_tape_device(&drive_config.path)?; let mut handle = LinuxTapeHandle::new(file); handle.get_drive_and_media_status() } #[api( input: { properties: { drive: { schema: DRIVE_NAME_SCHEMA, }, force: { description: "Force overriding existing index.", type: bool, optional: true, }, verbose: { description: "Verbose mode - log all found chunks.", type: bool, optional: true, }, }, }, returns: { schema: UPID_SCHEMA, }, )] /// Scan media and record content pub fn catalog_media( drive: String, force: Option, verbose: Option, rpcenv: &mut dyn RpcEnvironment, ) -> Result { let verbose = verbose.unwrap_or(false); let force = force.unwrap_or(false); let (config, _digest) = config::drive::config()?; check_drive_exists(&config, &drive)?; // early check before starting worker let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?; let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI; let upid_str = WorkerTask::new_thread( "catalog-media", Some(drive.clone()), auth_id, to_stdout, move |worker| { let mut drive = open_drive(&config, &drive)?; drive.rewind()?; let media_id = match drive.read_label()? { (Some(media_id), key_config) => { worker.log(format!( "found media label: {}", serde_json::to_string_pretty(&serde_json::to_value(&media_id)?)? )); if key_config.is_some() { worker.log(format!( "encryption key config: {}", serde_json::to_string_pretty(&serde_json::to_value(&key_config)?)? )); } media_id }, (None, _) => bail!("media is empty (no media label found)"), }; let status_path = Path::new(TAPE_STATUS_DIR); let mut inventory = Inventory::load(status_path)?; inventory.store(media_id.clone(), false)?; let pool = match media_id.media_set_label { None => { worker.log("media is empty"); MediaCatalog::destroy(status_path, &media_id.label.uuid)?; return Ok(()); } Some(ref set) => { if set.uuid.as_ref() == [0u8;16] { // media is empty worker.log("media is empty"); MediaCatalog::destroy(status_path, &media_id.label.uuid)?; return Ok(()); } let encrypt_fingerprint = set.encryption_key_fingerprint.clone() .map(|fp| (fp, set.uuid.clone())); drive.set_encryption(encrypt_fingerprint)?; set.pool.clone() } }; let _lock = MediaPool::lock(status_path, &pool)?; if MediaCatalog::exists(status_path, &media_id.label.uuid) && !force { bail!("media catalog exists (please use --force to overwrite)"); } restore_media(&worker, &mut drive, &media_id, None, verbose)?; Ok(()) } )?; Ok(upid_str.into()) } #[api( input: { properties: { changer: { schema: CHANGER_NAME_SCHEMA, optional: true, }, }, }, returns: { description: "The list of configured drives with model information.", type: Array, items: { type: DriveListEntry, }, }, )] /// List drives pub fn list_drives( changer: Option, _param: Value, ) -> Result, Error> { let (config, _) = config::drive::config()?; let linux_drives = linux_tape_device_list(); let drive_list: Vec = config.convert_to_typed_array("linux")?; let mut list = Vec::new(); for drive in drive_list { if changer.is_some() && drive.changer != changer { continue; } let mut entry = DriveListEntry { name: drive.name, path: drive.path.clone(), changer: drive.changer, changer_drivenum: drive.changer_drivenum, vendor: None, model: None, serial: None, }; if let Some(info) = lookup_drive(&linux_drives, &drive.path) { entry.vendor = Some(info.vendor.clone()); entry.model = Some(info.model.clone()); entry.serial = Some(info.serial.clone()); } list.push(entry); } Ok(list) } #[sortable] pub const SUBDIRS: SubdirMap = &sorted!([ ( "barcode-label-media", &Router::new() .put(&API_METHOD_BARCODE_LABEL_MEDIA) ), ( "catalog", &Router::new() .put(&API_METHOD_CATALOG_MEDIA) ), ( "clean", &Router::new() .put(&API_METHOD_CLEAN_DRIVE) ), ( "eject-media", &Router::new() .post(&API_METHOD_EJECT_MEDIA) ), ( "erase-media", &Router::new() .post(&API_METHOD_ERASE_MEDIA) ), ( "export-media", &Router::new() .put(&API_METHOD_EXPORT_MEDIA) ), ( "inventory", &Router::new() .get(&API_METHOD_INVENTORY) .put(&API_METHOD_UPDATE_INVENTORY) ), ( "label-media", &Router::new() .post(&API_METHOD_LABEL_MEDIA) ), ( "load-media", &Router::new() .put(&API_METHOD_LOAD_MEDIA) ), ( "load-slot", &Router::new() .put(&API_METHOD_LOAD_SLOT) ), ( "cartridge-memory", &Router::new() .get(&API_METHOD_CARTRIDGE_MEMORY) ), ( "volume-statistics", &Router::new() .get(&API_METHOD_VOLUME_STATISTICS) ), ( "read-label", &Router::new() .get(&API_METHOD_READ_LABEL) ), ( "rewind", &Router::new() .post(&API_METHOD_REWIND) ), ( "status", &Router::new() .get(&API_METHOD_STATUS) ), ( "unload", &Router::new() .put(&API_METHOD_UNLOAD) ), ]); const ITEM_ROUTER: Router = Router::new() .get(&list_subdirs_api_method!(SUBDIRS)) .subdirs(&SUBDIRS); pub const ROUTER: Router = Router::new() .get(&API_METHOD_LIST_DRIVES) .match_all("drive", &ITEM_ROUTER);