tape: add drive status api

This commit is contained in:
Dietmar Maurer
2020-12-22 10:42:22 +01:00
parent ee01737e87
commit cb80d900b3
5 changed files with 173 additions and 40 deletions

View File

@ -1,7 +1,7 @@
use std::path::Path;
use std::sync::Arc;
use anyhow::{bail, Error};
use anyhow::{bail, format_err, Error};
use serde_json::Value;
use proxmox::{
@ -35,6 +35,7 @@ use crate::{
MediaIdFlat,
LabelUuidMap,
MamAttribute,
LinuxDriveStatusFlat,
},
server::WorkerTask,
tape::{
@ -51,7 +52,7 @@ use crate::{
open_drive,
media_changer,
update_changer_online_status,
file_formats::{
file_formats::{
MediaLabel,
MediaSetLabel,
},
@ -796,6 +797,33 @@ pub fn cartridge_memory(drive: String) -> Result<Vec<MamAttribute>, Error> {
read_mam_attributes(&drive_config.path)
}
#[api(
input: {
properties: {
drive: {
schema: DRIVE_NAME_SCHEMA,
},
},
},
returns: {
type: LinuxDriveStatusFlat,
},
)]
/// Get drive status
pub fn status(drive: String) -> Result<LinuxDriveStatusFlat, Error> {
let (config, _digest) = config::drive::config()?;
let drive_config: LinuxTapeDrive = config.lookup("linux", &drive)?;
let handle = drive_config.open()
.map_err(|err| format_err!("open drive '{}' ({}) failed - {}", drive, drive_config.path, err))?;
let drive_status = handle.get_drive_status()?;
Ok(drive_status.into())
}
#[sortable]
pub const SUBDIRS: SubdirMap = &sorted!([
(
@ -849,6 +877,11 @@ pub const SUBDIRS: SubdirMap = &sorted!([
&Router::new()
.get(&API_METHOD_SCAN_DRIVES)
),
(
"status",
&Router::new()
.get(&API_METHOD_STATUS)
),
(
"unload",
&Router::new()

View File

@ -1,5 +1,7 @@
//! Types for tape drive API
use std::convert::TryFrom;
use anyhow::{bail, Error};
use serde::{Deserialize, Serialize};
use proxmox::api::{
@ -111,3 +113,69 @@ pub struct MamAttribute {
/// Attribute value
pub value: String,
}
#[api()]
#[derive(Serialize,Deserialize,Debug)]
pub enum TapeDensity {
/// No tape loaded
None,
/// LTO2
LTO2,
/// LTO3
LTO3,
/// LTO4
LTO4,
/// LTO5
LTO5,
/// LTO6
LTO6,
/// LTO7
LTO7,
/// LTO7M8
LTO7M8,
/// LTO8
LTO8,
}
impl TryFrom<u8> for TapeDensity {
type Error = Error;
fn try_from(value: u8) -> Result<Self, Self::Error> {
let density = match value {
0x00 => TapeDensity::None,
0x42 => TapeDensity::LTO2,
0x44 => TapeDensity::LTO3,
0x46 => TapeDensity::LTO4,
0x58 => TapeDensity::LTO5,
0x5a => TapeDensity::LTO6,
0x5c => TapeDensity::LTO7,
0x5d => TapeDensity::LTO7M8,
0x5e => TapeDensity::LTO8,
_ => bail!("unknown tape density code 0x{:02x}", value),
};
Ok(density)
}
}
#[api(
properties: {
density: {
type: TapeDensity,
},
},
)]
#[derive(Serialize,Deserialize)]
#[serde(rename_all = "kebab-case")]
/// Drive status for Linux SCSI drives.
pub struct LinuxDriveStatusFlat {
/// Block size (0 is variable size)
pub blocksize: u32,
/// Tape density
pub density: TapeDensity,
/// Status flags
pub status: String,
/// Current file number
pub file_number: i32,
/// Current block number
pub block_number: i32,
}