2020-12-08 14:42:50 +00:00
|
|
|
use anyhow::{bail, Error};
|
|
|
|
use serde_json::Value;
|
|
|
|
|
|
|
|
use proxmox::api::{api, Router, SubdirMap};
|
2020-12-10 09:09:12 +00:00
|
|
|
use proxmox::{sortable, identity, list_subdirs_api_method};
|
2020-12-08 14:42:50 +00:00
|
|
|
|
|
|
|
use crate::{
|
|
|
|
config,
|
|
|
|
api2::types::{
|
|
|
|
DRIVE_ID_SCHEMA,
|
2020-12-10 06:52:56 +00:00
|
|
|
MEDIA_LABEL_SCHEMA,
|
2020-12-08 14:42:50 +00:00
|
|
|
LinuxTapeDrive,
|
|
|
|
ScsiTapeChanger,
|
|
|
|
TapeDeviceInfo,
|
|
|
|
},
|
|
|
|
tape::{
|
|
|
|
MediaChange,
|
|
|
|
mtx_load,
|
|
|
|
mtx_unload,
|
|
|
|
linux_tape_device_list,
|
2020-12-09 16:35:31 +00:00
|
|
|
open_drive,
|
2020-12-09 16:50:48 +00:00
|
|
|
media_changer,
|
2020-12-08 14:42:50 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
2020-12-10 07:04:55 +00:00
|
|
|
drive: {
|
2020-12-08 14:42:50 +00:00
|
|
|
schema: DRIVE_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
slot: {
|
|
|
|
description: "Source slot number",
|
|
|
|
minimum: 1,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Load media via changer from slot
|
|
|
|
pub fn load_slot(
|
2020-12-10 07:04:55 +00:00
|
|
|
drive: String,
|
2020-12-08 14:42:50 +00:00
|
|
|
slot: u64,
|
|
|
|
_param: Value,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let (config, _digest) = config::drive::config()?;
|
|
|
|
|
2020-12-10 07:04:55 +00:00
|
|
|
let drive_config: LinuxTapeDrive = config.lookup("linux", &drive)?;
|
2020-12-08 14:42:50 +00:00
|
|
|
|
2020-12-10 07:04:55 +00:00
|
|
|
let changer: ScsiTapeChanger = match drive_config.changer {
|
2020-12-08 14:42:50 +00:00
|
|
|
Some(ref changer) => config.lookup("changer", changer)?,
|
2020-12-10 07:04:55 +00:00
|
|
|
None => bail!("drive '{}' has no associated changer", drive),
|
2020-12-08 14:42:50 +00:00
|
|
|
};
|
|
|
|
|
2020-12-10 08:09:06 +00:00
|
|
|
let drivenum = drive_config.changer_drive_id.unwrap_or(0);
|
2020-12-08 14:42:50 +00:00
|
|
|
|
|
|
|
mtx_load(&changer.path, slot, drivenum)
|
|
|
|
}
|
|
|
|
|
2020-12-10 06:52:56 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
"changer-id": {
|
|
|
|
schema: MEDIA_LABEL_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Load media with specified label
|
|
|
|
///
|
|
|
|
/// Issue a media load request to the associated changer device.
|
|
|
|
pub fn load_media(drive: String, changer_id: String) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let (config, _digest) = config::drive::config()?;
|
|
|
|
|
|
|
|
let (mut changer, _) = media_changer(&config, &drive, false)?;
|
|
|
|
|
|
|
|
changer.load_media(&changer_id)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-12-08 14:42:50 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
2020-12-10 07:35:11 +00:00
|
|
|
drive: {
|
2020-12-08 14:42:50 +00:00
|
|
|
schema: DRIVE_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
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 fn unload(
|
2020-12-10 07:35:11 +00:00
|
|
|
drive: String,
|
2020-12-08 14:42:50 +00:00
|
|
|
slot: Option<u64>,
|
|
|
|
_param: Value,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let (config, _digest) = config::drive::config()?;
|
|
|
|
|
2020-12-10 07:35:11 +00:00
|
|
|
let mut drive_config: LinuxTapeDrive = config.lookup("linux", &drive)?;
|
2020-12-08 14:42:50 +00:00
|
|
|
|
2020-12-10 07:35:11 +00:00
|
|
|
let changer: ScsiTapeChanger = match drive_config.changer {
|
2020-12-08 14:42:50 +00:00
|
|
|
Some(ref changer) => config.lookup("changer", changer)?,
|
2020-12-10 07:35:11 +00:00
|
|
|
None => bail!("drive '{}' has no associated changer", drive),
|
2020-12-08 14:42:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
let drivenum: u64 = 0;
|
|
|
|
|
|
|
|
if let Some(slot) = slot {
|
|
|
|
mtx_unload(&changer.path, slot, drivenum)
|
|
|
|
} else {
|
2020-12-10 07:35:11 +00:00
|
|
|
drive_config.unload_media()
|
2020-12-08 14:42:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[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<Vec<TapeDeviceInfo>, Error> {
|
|
|
|
|
|
|
|
let list = linux_tape_device_list();
|
|
|
|
|
|
|
|
Ok(list)
|
|
|
|
}
|
|
|
|
|
2020-12-09 16:35:31 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
fast: {
|
|
|
|
description: "Use fast erase.",
|
|
|
|
type: bool,
|
|
|
|
optional: true,
|
|
|
|
default: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Erase media
|
|
|
|
pub fn erase_media(drive: String, fast: Option<bool>) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let (config, _digest) = config::drive::config()?;
|
|
|
|
|
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
|
|
|
|
drive.erase_media(fast.unwrap_or(true))?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-12-09 16:43:38 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Rewind tape
|
|
|
|
pub fn rewind(drive: String) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let (config, _digest) = config::drive::config()?;
|
|
|
|
|
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
|
|
|
|
drive.rewind()?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-12-09 16:50:48 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Eject/Unload drive media
|
|
|
|
pub fn eject_media(drive: String) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let (config, _digest) = config::drive::config()?;
|
|
|
|
|
|
|
|
let (mut changer, _) = media_changer(&config, &drive, false)?;
|
|
|
|
|
|
|
|
if !changer.eject_on_unload() {
|
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
drive.eject_media()?;
|
|
|
|
}
|
|
|
|
|
|
|
|
changer.unload_media()?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-12-10 09:09:12 +00:00
|
|
|
#[sortable]
|
|
|
|
pub const SUBDIRS: SubdirMap = &sorted!([
|
2020-12-09 16:35:31 +00:00
|
|
|
(
|
2020-12-10 09:09:12 +00:00
|
|
|
"eject-media",
|
2020-12-09 16:35:31 +00:00
|
|
|
&Router::new()
|
2020-12-10 09:09:12 +00:00
|
|
|
.put(&API_METHOD_EJECT_MEDIA)
|
2020-12-09 16:35:31 +00:00
|
|
|
),
|
2020-12-09 16:50:48 +00:00
|
|
|
(
|
2020-12-10 09:09:12 +00:00
|
|
|
"erase-media",
|
2020-12-09 16:50:48 +00:00
|
|
|
&Router::new()
|
2020-12-10 09:09:12 +00:00
|
|
|
.put(&API_METHOD_ERASE_MEDIA)
|
2020-12-09 16:50:48 +00:00
|
|
|
),
|
2020-12-08 14:42:50 +00:00
|
|
|
(
|
|
|
|
"load-slot",
|
|
|
|
&Router::new()
|
|
|
|
.put(&API_METHOD_LOAD_SLOT)
|
|
|
|
),
|
2020-12-10 08:09:06 +00:00
|
|
|
(
|
|
|
|
"rewind",
|
|
|
|
&Router::new()
|
|
|
|
.put(&API_METHOD_REWIND)
|
|
|
|
),
|
2020-12-08 14:42:50 +00:00
|
|
|
(
|
|
|
|
"scan",
|
|
|
|
&Router::new()
|
|
|
|
.get(&API_METHOD_SCAN_DRIVES)
|
|
|
|
),
|
|
|
|
(
|
|
|
|
"unload",
|
|
|
|
&Router::new()
|
|
|
|
.put(&API_METHOD_UNLOAD)
|
|
|
|
),
|
2020-12-10 09:09:12 +00:00
|
|
|
]);
|
2020-12-08 14:42:50 +00:00
|
|
|
|
|
|
|
pub const ROUTER: Router = Router::new()
|
|
|
|
.get(&list_subdirs_api_method!(SUBDIRS))
|
|
|
|
.subdirs(SUBDIRS);
|