2021-02-18 14:40:22 +00:00
|
|
|
use std::panic::UnwindSafe;
|
2020-12-10 11:30:27 +00:00
|
|
|
use std::path::Path;
|
2020-12-11 08:10:22 +00:00
|
|
|
use std::sync::Arc;
|
2021-04-16 11:17:17 +00:00
|
|
|
use std::collections::HashMap;
|
2020-12-11 08:10:22 +00:00
|
|
|
|
2021-02-18 14:40:22 +00:00
|
|
|
use anyhow::{bail, format_err, Error};
|
2020-12-08 14:42:50 +00:00
|
|
|
use serde_json::Value;
|
|
|
|
|
2020-12-10 11:30:27 +00:00
|
|
|
use proxmox::{
|
|
|
|
sortable,
|
|
|
|
identity,
|
|
|
|
list_subdirs_api_method,
|
|
|
|
tools::Uuid,
|
|
|
|
api::{
|
|
|
|
api,
|
2021-02-05 09:50:21 +00:00
|
|
|
section_config::SectionConfigData,
|
2020-12-11 08:10:22 +00:00
|
|
|
RpcEnvironment,
|
2020-12-30 18:01:39 +00:00
|
|
|
RpcEnvironmentType,
|
2021-03-03 11:10:00 +00:00
|
|
|
Permission,
|
2020-12-10 11:30:27 +00:00
|
|
|
Router,
|
|
|
|
SubdirMap,
|
|
|
|
},
|
|
|
|
};
|
2020-12-08 14:42:50 +00:00
|
|
|
|
2021-09-09 08:32:44 +00:00
|
|
|
use pbs_api_types::{
|
|
|
|
UPID_SCHEMA, CHANGER_NAME_SCHEMA, DRIVE_NAME_SCHEMA, MEDIA_LABEL_SCHEMA, MEDIA_POOL_NAME_SCHEMA,
|
|
|
|
Authid, DriveListEntry, LtoTapeDrive, MediaIdFlat, LabelUuidMap, MamAttribute,
|
|
|
|
LtoDriveAndMediaStatus, Lp17VolumeStatistics,
|
|
|
|
};
|
|
|
|
|
2021-07-07 12:37:47 +00:00
|
|
|
use pbs_datastore::task_log;
|
2021-09-09 08:32:44 +00:00
|
|
|
use pbs_api_types::{PRIV_TAPE_AUDIT, PRIV_TAPE_READ, PRIV_TAPE_WRITE};
|
2021-09-10 04:53:53 +00:00
|
|
|
use pbs_config::CachedUserInfo;
|
2021-07-07 12:37:47 +00:00
|
|
|
|
2020-12-08 14:42:50 +00:00
|
|
|
use crate::{
|
2021-09-09 08:32:44 +00:00
|
|
|
api2::tape::restore::{
|
|
|
|
fast_catalog_restore,
|
|
|
|
restore_media,
|
2020-12-08 14:42:50 +00:00
|
|
|
},
|
2020-12-11 08:10:22 +00:00
|
|
|
server::WorkerTask,
|
2020-12-08 14:42:50 +00:00
|
|
|
tape::{
|
2020-12-10 11:30:27 +00:00
|
|
|
TAPE_STATUS_DIR,
|
|
|
|
Inventory,
|
2020-12-29 09:55:20 +00:00
|
|
|
MediaCatalog,
|
2020-12-10 11:30:27 +00:00
|
|
|
MediaId,
|
2021-04-12 09:25:40 +00:00
|
|
|
BlockReadError,
|
2021-03-22 05:32:18 +00:00
|
|
|
lock_media_set,
|
|
|
|
lock_media_pool,
|
|
|
|
lock_unassigned_media_pool,
|
2021-03-30 15:07:59 +00:00
|
|
|
lto_tape_device_list,
|
2021-01-30 08:36:54 +00:00
|
|
|
lookup_device_identification,
|
2021-01-21 16:25:32 +00:00
|
|
|
file_formats::{
|
2020-12-14 16:37:16 +00:00
|
|
|
MediaLabel,
|
2020-12-10 11:30:27 +00:00
|
|
|
MediaSetLabel,
|
|
|
|
},
|
2021-01-21 16:12:01 +00:00
|
|
|
drive::{
|
|
|
|
TapeDriver,
|
2021-03-30 15:07:59 +00:00
|
|
|
LtoTapeHandle,
|
|
|
|
open_lto_tape_device,
|
2021-09-03 07:10:18 +00:00
|
|
|
open_lto_tape_drive,
|
2021-02-05 09:50:21 +00:00
|
|
|
media_changer,
|
2021-01-21 16:12:01 +00:00
|
|
|
required_media_changer,
|
|
|
|
open_drive,
|
2021-02-05 09:50:21 +00:00
|
|
|
lock_tape_device,
|
2021-02-18 14:40:22 +00:00
|
|
|
set_tape_device_state,
|
2021-03-02 11:19:36 +00:00
|
|
|
get_tape_device_state,
|
2021-03-08 07:27:02 +00:00
|
|
|
tape_alert_flags_critical,
|
2021-01-21 16:12:01 +00:00
|
|
|
},
|
|
|
|
changer::update_changer_online_status,
|
2020-12-08 14:42:50 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2021-02-18 14:40:22 +00:00
|
|
|
fn run_drive_worker<F>(
|
|
|
|
rpcenv: &dyn RpcEnvironment,
|
|
|
|
drive: String,
|
|
|
|
worker_type: &str,
|
|
|
|
job_id: Option<String>,
|
|
|
|
f: F,
|
|
|
|
) -> Result<String, Error>
|
|
|
|
where
|
|
|
|
F: Send
|
|
|
|
+ UnwindSafe
|
|
|
|
+ 'static
|
|
|
|
+ FnOnce(Arc<WorkerTask>, SectionConfigData) -> Result<(), Error>,
|
|
|
|
{
|
|
|
|
// early check/lock before starting worker
|
2021-09-03 07:10:18 +00:00
|
|
|
let (config, _digest) = pbs_config::drive::config()?;
|
2021-02-18 14:40:22 +00:00
|
|
|
let lock_guard = lock_tape_device(&config, &drive)?;
|
|
|
|
|
|
|
|
let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?;
|
|
|
|
let to_stdout = rpcenv.env_type() == RpcEnvironmentType::CLI;
|
|
|
|
|
|
|
|
WorkerTask::new_thread(worker_type, job_id, auth_id, to_stdout, move |worker| {
|
|
|
|
let _lock_guard = lock_guard;
|
|
|
|
set_tape_device_state(&drive, &worker.upid().to_string())
|
|
|
|
.map_err(|err| format_err!("could not set tape device state: {}", err))?;
|
|
|
|
|
|
|
|
let result = f(worker, config);
|
|
|
|
set_tape_device_state(&drive, "")
|
|
|
|
.map_err(|err| format_err!("could not unset tape device state: {}", err))?;
|
|
|
|
result
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-02-18 14:40:24 +00:00
|
|
|
async fn run_drive_blocking_task<F, R>(drive: String, state: String, f: F) -> Result<R, Error>
|
|
|
|
where
|
|
|
|
F: Send + 'static + FnOnce(SectionConfigData) -> Result<R, Error>,
|
|
|
|
R: Send + 'static,
|
|
|
|
{
|
|
|
|
// early check/lock before starting worker
|
2021-09-03 07:10:18 +00:00
|
|
|
let (config, _digest) = pbs_config::drive::config()?;
|
2021-02-18 14:40:24 +00:00
|
|
|
let lock_guard = lock_tape_device(&config, &drive)?;
|
|
|
|
tokio::task::spawn_blocking(move || {
|
|
|
|
let _lock_guard = lock_guard;
|
|
|
|
set_tape_device_state(&drive, &state)
|
|
|
|
.map_err(|err| format_err!("could not set tape device state: {}", err))?;
|
|
|
|
let result = f(config);
|
|
|
|
set_tape_device_state(&drive, "")
|
|
|
|
.map_err(|err| format_err!("could not unset tape device state: {}", err))?;
|
|
|
|
result
|
|
|
|
})
|
|
|
|
.await?
|
|
|
|
}
|
|
|
|
|
2020-12-08 14:42:50 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
2020-12-10 07:04:55 +00:00
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-08 14:42:50 +00:00
|
|
|
},
|
2021-01-13 12:26:59 +00:00
|
|
|
"label-text": {
|
2021-01-07 13:26:43 +00:00
|
|
|
schema: MEDIA_LABEL_SCHEMA,
|
2020-12-08 14:42:50 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2021-02-18 08:04:51 +00:00
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-08 14:42:50 +00:00
|
|
|
)]
|
2021-01-07 13:26:43 +00:00
|
|
|
/// Load media with specified label
|
|
|
|
///
|
|
|
|
/// Issue a media load request to the associated changer device.
|
2021-02-18 08:04:51 +00:00
|
|
|
pub fn load_media(
|
|
|
|
drive: String,
|
|
|
|
label_text: String,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
|
|
|
let job_id = format!("{}:{}", drive, label_text);
|
|
|
|
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2021-02-18 08:04:51 +00:00
|
|
|
"load-media",
|
|
|
|
Some(job_id),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| {
|
2021-02-18 08:04:51 +00:00
|
|
|
task_log!(worker, "loading media '{}' into drive '{}'", label_text, drive);
|
|
|
|
let (mut changer, _) = required_media_changer(&config, &drive)?;
|
2021-02-20 09:23:16 +00:00
|
|
|
changer.load_media(&label_text)?;
|
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
2021-02-18 08:04:51 +00:00
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(upid_str.into())
|
2020-12-08 14:42:50 +00:00
|
|
|
}
|
2021-01-10 12:44:44 +00:00
|
|
|
|
2020-12-10 06:52:56 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-10 06:52:56 +00:00
|
|
|
},
|
2021-01-07 13:26:43 +00:00
|
|
|
"source-slot": {
|
|
|
|
description: "Source slot number.",
|
|
|
|
minimum: 1,
|
2020-12-10 06:52:56 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-10 06:52:56 +00:00
|
|
|
)]
|
2021-01-07 13:26:43 +00:00
|
|
|
/// Load media from the specified slot
|
2020-12-10 06:52:56 +00:00
|
|
|
///
|
|
|
|
/// Issue a media load request to the associated changer device.
|
2021-01-07 13:26:43 +00:00
|
|
|
pub async fn load_slot(drive: String, source_slot: u64) -> Result<(), Error> {
|
2021-02-18 14:40:25 +00:00
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
format!("load from slot {}", source_slot),
|
|
|
|
move |config| {
|
|
|
|
let (mut changer, _) = required_media_changer(&config, &drive)?;
|
2021-02-20 09:23:16 +00:00
|
|
|
changer.load_media_from_slot(source_slot)?;
|
|
|
|
Ok(())
|
2021-02-18 14:40:25 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
2020-12-10 06:52:56 +00:00
|
|
|
}
|
|
|
|
|
2021-01-10 12:44:44 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
|
|
|
},
|
2021-01-13 12:26:59 +00:00
|
|
|
"label-text": {
|
2021-01-10 12:44:44 +00:00
|
|
|
schema: MEDIA_LABEL_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
2021-03-10 15:37:09 +00:00
|
|
|
description: "The import-export slot number the media was transferred to.",
|
2021-01-10 12:44:44 +00:00
|
|
|
type: u64,
|
|
|
|
minimum: 1,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2021-01-10 12:44:44 +00:00
|
|
|
)]
|
|
|
|
/// Export media with specified label
|
2021-01-13 12:26:59 +00:00
|
|
|
pub async fn export_media(drive: String, label_text: String) -> Result<u64, Error> {
|
2021-02-18 14:40:25 +00:00
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
format!("export media {}", label_text),
|
|
|
|
move |config| {
|
|
|
|
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
|
|
|
|
),
|
|
|
|
}
|
2021-01-10 12:44:44 +00:00
|
|
|
}
|
2021-02-18 14:40:25 +00:00
|
|
|
)
|
|
|
|
.await
|
2021-01-10 12:44:44 +00:00
|
|
|
}
|
|
|
|
|
2020-12-08 14:42:50 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
2020-12-10 07:35:11 +00:00
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-08 14:42:50 +00:00
|
|
|
},
|
2021-01-07 13:26:43 +00:00
|
|
|
"target-slot": {
|
2020-12-08 14:42:50 +00:00
|
|
|
description: "Target slot number. If omitted, defaults to the slot that the drive was loaded from.",
|
|
|
|
minimum: 1,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2021-02-18 08:04:51 +00:00
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-08 14:42:50 +00:00
|
|
|
)]
|
|
|
|
/// Unload media via changer
|
2021-02-18 08:04:51 +00:00
|
|
|
pub fn unload(
|
2020-12-10 07:35:11 +00:00
|
|
|
drive: String,
|
2021-01-07 13:26:43 +00:00
|
|
|
target_slot: Option<u64>,
|
2021-02-18 08:04:51 +00:00
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2021-02-18 08:04:51 +00:00
|
|
|
"unload-media",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| {
|
2021-02-18 08:04:51 +00:00
|
|
|
task_log!(worker, "unloading media from drive '{}'", drive);
|
|
|
|
|
|
|
|
let (mut changer, _) = required_media_changer(&config, &drive)?;
|
2021-02-20 09:23:16 +00:00
|
|
|
changer.unload_media(target_slot)?;
|
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
2021-02-18 08:04:51 +00:00
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(upid_str.into())
|
2020-12-08 14:42:50 +00:00
|
|
|
}
|
|
|
|
|
2020-12-09 16:35:31 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-09 16:35:31 +00:00
|
|
|
},
|
|
|
|
fast: {
|
|
|
|
description: "Use fast erase.",
|
|
|
|
type: bool,
|
|
|
|
optional: true,
|
|
|
|
default: true,
|
|
|
|
},
|
2021-02-16 08:35:24 +00:00
|
|
|
"label-text": {
|
|
|
|
schema: MEDIA_LABEL_SCHEMA,
|
|
|
|
optional: true,
|
|
|
|
},
|
2020-12-09 16:35:31 +00:00
|
|
|
},
|
|
|
|
},
|
2020-12-11 10:15:58 +00:00
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_WRITE, false),
|
|
|
|
},
|
2020-12-09 16:35:31 +00:00
|
|
|
)]
|
2021-03-31 07:19:19 +00:00
|
|
|
/// Format media. Check for label-text if given (cancels if wrong media).
|
|
|
|
pub fn format_media(
|
2020-12-11 10:15:58 +00:00
|
|
|
drive: String,
|
|
|
|
fast: Option<bool>,
|
2021-02-16 08:35:24 +00:00
|
|
|
label_text: Option<String>,
|
2020-12-11 10:15:58 +00:00
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2021-03-31 07:19:19 +00:00
|
|
|
"format-media",
|
2020-12-11 10:15:58 +00:00
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| {
|
2021-02-16 10:05:26 +00:00
|
|
|
if let Some(ref label) = label_text {
|
|
|
|
task_log!(worker, "try to load media '{}'", label);
|
|
|
|
if let Some((mut changer, _)) = media_changer(&config, &drive)? {
|
|
|
|
changer.load_media(label)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut handle = open_drive(&config, &drive)?;
|
2021-02-12 10:30:34 +00:00
|
|
|
|
2021-02-16 10:05:26 +00:00
|
|
|
match handle.read_label() {
|
2021-02-12 10:30:34 +00:00
|
|
|
Err(err) => {
|
2021-02-16 08:35:24 +00:00
|
|
|
if let Some(label) = label_text {
|
|
|
|
bail!("expected label '{}', found unrelated data", label);
|
|
|
|
}
|
2021-02-12 10:30:34 +00:00
|
|
|
/* assume drive contains no or unrelated data */
|
|
|
|
task_log!(worker, "unable to read media label: {}", err);
|
2021-03-31 07:19:19 +00:00
|
|
|
task_log!(worker, "format anyways");
|
|
|
|
handle.format_media(fast.unwrap_or(true))?;
|
2021-02-12 10:30:34 +00:00
|
|
|
}
|
|
|
|
Ok((None, _)) => {
|
2021-02-16 08:35:24 +00:00
|
|
|
if let Some(label) = label_text {
|
|
|
|
bail!("expected label '{}', found empty tape", label);
|
|
|
|
}
|
2021-03-31 07:19:19 +00:00
|
|
|
task_log!(worker, "found empty media - format anyways");
|
|
|
|
handle.format_media(fast.unwrap_or(true))?;
|
2021-02-12 10:30:34 +00:00
|
|
|
}
|
|
|
|
Ok((Some(media_id), _key_config)) => {
|
2021-02-16 08:35:24 +00:00
|
|
|
if let Some(label_text) = label_text {
|
|
|
|
if media_id.label.label_text != label_text {
|
|
|
|
bail!(
|
|
|
|
"expected label '{}', found '{}', aborting",
|
|
|
|
label_text,
|
|
|
|
media_id.label.label_text
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-12 10:30:34 +00:00
|
|
|
task_log!(
|
|
|
|
worker,
|
|
|
|
"found media '{}' with uuid '{}'",
|
|
|
|
media_id.label.label_text, media_id.label.uuid,
|
|
|
|
);
|
|
|
|
|
|
|
|
let status_path = Path::new(TAPE_STATUS_DIR);
|
2021-03-22 05:32:18 +00:00
|
|
|
let mut inventory = Inventory::new(status_path);
|
|
|
|
|
|
|
|
if let Some(MediaSetLabel { ref pool, ref uuid, ..}) = media_id.media_set_label {
|
|
|
|
let _pool_lock = lock_media_pool(status_path, pool)?;
|
|
|
|
let _media_set_lock = lock_media_set(status_path, uuid, None)?;
|
|
|
|
MediaCatalog::destroy(status_path, &media_id.label.uuid)?;
|
|
|
|
inventory.remove_media(&media_id.label.uuid)?;
|
|
|
|
} else {
|
|
|
|
let _lock = lock_unassigned_media_pool(status_path)?;
|
|
|
|
MediaCatalog::destroy(status_path, &media_id.label.uuid)?;
|
|
|
|
inventory.remove_media(&media_id.label.uuid)?;
|
|
|
|
};
|
2021-02-12 10:30:34 +00:00
|
|
|
|
2021-03-31 07:19:19 +00:00
|
|
|
handle.format_media(fast.unwrap_or(true))?;
|
2021-02-12 10:30:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-11 10:15:58 +00:00
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
2020-12-11 10:15:58 +00:00
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(upid_str.into())
|
2020-12-09 16:35:31 +00:00
|
|
|
}
|
|
|
|
|
2020-12-09 16:43:38 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-09 16:43:38 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2020-12-11 10:15:58 +00:00
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-09 16:43:38 +00:00
|
|
|
)]
|
|
|
|
/// Rewind tape
|
2020-12-11 10:15:58 +00:00
|
|
|
pub fn rewind(
|
|
|
|
drive: String,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2020-12-11 10:15:58 +00:00
|
|
|
"rewind-media",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |_worker, config| {
|
2020-12-11 10:15:58 +00:00
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
drive.rewind()?;
|
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
2020-12-11 10:15:58 +00:00
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(upid_str.into())
|
2020-12-09 16:43:38 +00:00
|
|
|
}
|
|
|
|
|
2020-12-09 16:50:48 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-09 16:50:48 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2021-01-28 15:46:28 +00:00
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-09 16:50:48 +00:00
|
|
|
)]
|
|
|
|
/// Eject/Unload drive media
|
2021-01-28 15:46:28 +00:00
|
|
|
pub fn eject_media(
|
|
|
|
drive: String,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2021-01-28 15:46:28 +00:00
|
|
|
"eject-media",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |_worker, config| {
|
|
|
|
if let Some((mut changer, _)) = media_changer(&config, &drive)? {
|
2021-01-28 15:46:28 +00:00
|
|
|
changer.unload_media(None)?;
|
|
|
|
} else {
|
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
drive.eject_media()?;
|
|
|
|
}
|
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
|
|
|
)?;
|
2021-01-28 15:46:28 +00:00
|
|
|
|
|
|
|
Ok(upid_str.into())
|
2020-12-09 16:50:48 +00:00
|
|
|
}
|
|
|
|
|
2020-12-10 11:30:27 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-10 11:30:27 +00:00
|
|
|
},
|
2021-01-13 12:26:59 +00:00
|
|
|
"label-text": {
|
2020-12-10 11:30:27 +00:00
|
|
|
schema: MEDIA_LABEL_SCHEMA,
|
|
|
|
},
|
|
|
|
pool: {
|
|
|
|
schema: MEDIA_POOL_NAME_SCHEMA,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2020-12-11 08:10:22 +00:00
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_WRITE, false),
|
|
|
|
},
|
2020-12-10 11:30:27 +00:00
|
|
|
)]
|
|
|
|
/// 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.
|
|
|
|
///
|
2021-03-31 07:19:19 +00:00
|
|
|
/// Note: The media need to be empty (you may want to format it first).
|
2020-12-10 11:30:27 +00:00
|
|
|
pub fn label_media(
|
|
|
|
drive: String,
|
|
|
|
pool: Option<String>,
|
2021-01-13 12:26:59 +00:00
|
|
|
label_text: String,
|
2020-12-11 08:10:22 +00:00
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2020-12-10 11:30:27 +00:00
|
|
|
if let Some(ref pool) = pool {
|
2021-09-06 06:56:04 +00:00
|
|
|
let (pool_config, _digest) = pbs_config::media_pool::config()?;
|
2020-12-10 11:30:27 +00:00
|
|
|
|
|
|
|
if pool_config.sections.get(pool).is_none() {
|
|
|
|
bail!("no such pool ('{}')", pool);
|
|
|
|
}
|
|
|
|
}
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2020-12-11 08:10:22 +00:00
|
|
|
"label-media",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| {
|
2020-12-11 08:10:22 +00:00
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
|
|
|
|
drive.rewind()?;
|
|
|
|
|
|
|
|
match drive.read_next_file() {
|
2021-04-12 09:25:40 +00:00
|
|
|
Ok(_reader) => bail!("media is not empty (format it first)"),
|
|
|
|
Err(BlockReadError::EndOfFile) => { /* EOF mark at BOT, assume tape is empty */ },
|
|
|
|
Err(BlockReadError::EndOfStream) => { /* tape is empty */ },
|
2020-12-11 08:10:22 +00:00
|
|
|
Err(err) => {
|
2021-04-12 09:25:40 +00:00
|
|
|
bail!("media read error - {}", err);
|
2020-12-11 08:10:22 +00:00
|
|
|
}
|
|
|
|
}
|
2020-12-10 11:30:27 +00:00
|
|
|
|
2020-12-11 08:10:22 +00:00
|
|
|
let ctime = proxmox::tools::time::epoch_i64();
|
2020-12-14 16:37:16 +00:00
|
|
|
let label = MediaLabel {
|
2021-01-13 12:26:59 +00:00
|
|
|
label_text: label_text.to_string(),
|
2020-12-11 08:10:22 +00:00
|
|
|
uuid: Uuid::generate(),
|
|
|
|
ctime,
|
|
|
|
};
|
2020-12-10 11:30:27 +00:00
|
|
|
|
2020-12-11 08:10:22 +00:00
|
|
|
write_media_label(worker, &mut drive, label, pool)
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
2020-12-11 08:10:22 +00:00
|
|
|
)?;
|
2020-12-10 11:30:27 +00:00
|
|
|
|
2020-12-11 08:10:22 +00:00
|
|
|
Ok(upid_str.into())
|
2020-12-10 11:30:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn write_media_label(
|
2020-12-11 08:10:22 +00:00
|
|
|
worker: Arc<WorkerTask>,
|
2020-12-10 11:30:27 +00:00
|
|
|
drive: &mut Box<dyn TapeDriver>,
|
2020-12-14 16:37:16 +00:00
|
|
|
label: MediaLabel,
|
2020-12-10 11:30:27 +00:00
|
|
|
pool: Option<String>,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
drive.label_tape(&label)?;
|
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
let status_path = Path::new(TAPE_STATUS_DIR);
|
2020-12-10 11:30:27 +00:00
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
let media_id = if let Some(ref pool) = pool {
|
2020-12-10 11:30:27 +00:00
|
|
|
// assign media to pool by writing special media set label
|
2021-01-13 12:26:59 +00:00
|
|
|
worker.log(format!("Label media '{}' for pool '{}'", label.label_text, pool));
|
2021-01-18 12:36:11 +00:00
|
|
|
let set = MediaSetLabel::with_data(&pool, [0u8; 16].into(), 0, label.ctime, None);
|
2020-12-10 11:30:27 +00:00
|
|
|
|
2021-01-19 05:19:18 +00:00
|
|
|
drive.write_media_set_label(&set, None)?;
|
2021-03-22 05:32:18 +00:00
|
|
|
|
|
|
|
let media_id = MediaId { label, media_set_label: Some(set) };
|
|
|
|
|
|
|
|
// Create the media catalog
|
|
|
|
MediaCatalog::overwrite(status_path, &media_id, false)?;
|
|
|
|
|
|
|
|
let mut inventory = Inventory::new(status_path);
|
|
|
|
inventory.store(media_id.clone(), false)?;
|
|
|
|
|
|
|
|
media_id
|
2020-12-10 11:30:27 +00:00
|
|
|
} else {
|
2021-01-13 12:26:59 +00:00
|
|
|
worker.log(format!("Label media '{}' (no pool assignment)", label.label_text));
|
2020-12-10 11:30:27 +00:00
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
let media_id = MediaId { label, media_set_label: None };
|
2020-12-10 11:30:27 +00:00
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
// Create the media catalog
|
|
|
|
MediaCatalog::overwrite(status_path, &media_id, false)?;
|
2020-12-29 09:55:20 +00:00
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
let mut inventory = Inventory::new(status_path);
|
|
|
|
inventory.store(media_id.clone(), false)?;
|
2020-12-29 09:55:20 +00:00
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
media_id
|
|
|
|
};
|
2020-12-10 11:30:27 +00:00
|
|
|
|
|
|
|
drive.rewind()?;
|
|
|
|
|
|
|
|
match drive.read_label() {
|
2021-01-19 05:19:18 +00:00
|
|
|
Ok((Some(info), _)) => {
|
2020-12-10 11:30:27 +00:00
|
|
|
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 {
|
2020-12-16 12:27:53 +00:00
|
|
|
Some(set) => {
|
2020-12-10 11:30:27 +00:00
|
|
|
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)");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2021-01-19 05:19:18 +00:00
|
|
|
Ok((None, _)) => bail!("verify label failed (got empty media)"),
|
2020-12-10 11:30:27 +00:00
|
|
|
Err(err) => bail!("verify label failed - {}", err),
|
|
|
|
};
|
|
|
|
|
|
|
|
drive.rewind()?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-12-10 12:20:39 +00:00
|
|
|
#[api(
|
2021-01-19 05:19:18 +00:00
|
|
|
protected: true,
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
|
|
|
},
|
|
|
|
password: {
|
|
|
|
description: "Encryption key password.",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2021-01-19 05:19:18 +00:00
|
|
|
)]
|
|
|
|
/// Try to restore a tape encryption key
|
|
|
|
pub async fn restore_key(
|
|
|
|
drive: String,
|
|
|
|
password: String,
|
|
|
|
) -> Result<(), Error> {
|
2021-02-18 14:40:25 +00:00
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
"restore key".to_string(),
|
|
|
|
move |config| {
|
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
2021-01-19 05:19:18 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
let (_media_id, key_config) = drive.read_label()?;
|
2021-01-19 05:19:18 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
if let Some(key_config) = key_config {
|
|
|
|
let password_fn = || { Ok(password.as_bytes().to_vec()) };
|
|
|
|
let (key, ..) = key_config.decrypt(&password_fn)?;
|
2021-09-07 08:37:08 +00:00
|
|
|
pbs_config::tape_encryption_keys::insert_key(key, key_config, true)?;
|
2021-02-18 14:40:25 +00:00
|
|
|
} else {
|
|
|
|
bail!("media does not contain any encryption key configuration");
|
|
|
|
}
|
2021-01-19 05:19:18 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
Ok(())
|
2021-01-19 05:19:18 +00:00
|
|
|
}
|
2021-02-18 14:40:25 +00:00
|
|
|
)
|
|
|
|
.await
|
2021-01-19 05:19:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
2020-12-10 12:20:39 +00:00
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-10 12:20:39 +00:00
|
|
|
},
|
2021-01-14 10:51:23 +00:00
|
|
|
inventorize: {
|
|
|
|
description: "Inventorize media",
|
|
|
|
optional: true,
|
|
|
|
},
|
2020-12-10 12:20:39 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
2020-12-16 12:27:53 +00:00
|
|
|
type: MediaIdFlat,
|
2020-12-10 12:20:39 +00:00
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-10 12:20:39 +00:00
|
|
|
)]
|
2021-01-19 05:19:18 +00:00
|
|
|
/// Read media label (optionally inventorize media)
|
2021-01-14 10:51:23 +00:00
|
|
|
pub async fn read_label(
|
|
|
|
drive: String,
|
|
|
|
inventorize: Option<bool>,
|
|
|
|
) -> Result<MediaIdFlat, Error> {
|
2021-02-18 14:40:25 +00:00
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
"reading label".to_string(),
|
|
|
|
move |config| {
|
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
2020-12-10 12:20:39 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
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()
|
2021-07-06 11:26:35 +00:00
|
|
|
.map(|fp| pbs_tools::format::as_fingerprint(fp.bytes()));
|
2021-02-18 14:40:25 +00:00
|
|
|
|
|
|
|
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
|
2021-03-22 05:32:18 +00:00
|
|
|
eprintln!("unable to load encryption key: {}", err);
|
2021-02-18 14:40:25 +00:00
|
|
|
}
|
|
|
|
}
|
2021-02-05 09:50:21 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
if let Some(true) = inventorize {
|
|
|
|
let state_path = Path::new(TAPE_STATUS_DIR);
|
2021-03-22 05:32:18 +00:00
|
|
|
let mut inventory = Inventory::new(state_path);
|
|
|
|
|
|
|
|
if let Some(MediaSetLabel { ref pool, ref uuid, ..}) = media_id.media_set_label {
|
|
|
|
let _pool_lock = lock_media_pool(state_path, pool)?;
|
|
|
|
let _lock = lock_media_set(state_path, uuid, None)?;
|
|
|
|
MediaCatalog::destroy_unrelated_catalog(state_path, &media_id)?;
|
|
|
|
inventory.store(media_id, false)?;
|
|
|
|
} else {
|
|
|
|
let _lock = lock_unassigned_media_pool(state_path)?;
|
|
|
|
MediaCatalog::destroy(state_path, &media_id.label.uuid)?;
|
|
|
|
inventory.store(media_id, false)?;
|
|
|
|
};
|
2021-01-21 09:30:29 +00:00
|
|
|
}
|
2021-01-14 10:51:23 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
flat
|
2021-01-14 10:51:23 +00:00
|
|
|
}
|
2021-02-18 14:40:25 +00:00
|
|
|
None => {
|
|
|
|
bail!("Media is empty (no label).");
|
|
|
|
}
|
|
|
|
};
|
2021-01-14 10:51:23 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
Ok(media_id)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.await
|
2020-12-10 12:20:39 +00:00
|
|
|
}
|
|
|
|
|
2021-01-08 10:32:56 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2021-01-08 10:32:56 +00:00
|
|
|
)]
|
|
|
|
/// Clean drive
|
|
|
|
pub fn clean_drive(
|
|
|
|
drive: String,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2021-01-08 10:32:56 +00:00
|
|
|
"clean-drive",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| {
|
2021-01-08 10:32:56 +00:00
|
|
|
let (mut changer, _changer_name) = required_media_changer(&config, &drive)?;
|
|
|
|
|
|
|
|
worker.log("Starting drive clean");
|
|
|
|
|
|
|
|
changer.clean_drive()?;
|
|
|
|
|
2021-03-30 15:07:59 +00:00
|
|
|
if let Ok(drive_config) = config.lookup::<LtoTapeDrive>("lto", &drive) {
|
2021-03-08 07:27:02 +00:00
|
|
|
// Note: clean_drive unloads the cleaning media, so we cannot use drive_config.open
|
2021-03-30 15:07:59 +00:00
|
|
|
let mut handle = LtoTapeHandle::new(open_lto_tape_device(&drive_config.path)?)?;
|
2021-03-08 07:27:02 +00:00
|
|
|
|
|
|
|
// test for critical tape alert flags
|
|
|
|
if let Ok(alert_flags) = handle.tape_alert_flags() {
|
|
|
|
if !alert_flags.is_empty() {
|
|
|
|
worker.log(format!("TapeAlertFlags: {:?}", alert_flags));
|
|
|
|
if tape_alert_flags_critical(alert_flags) {
|
|
|
|
bail!("found critical tape alert flags: {:?}", alert_flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// test wearout (max. 50 mounts)
|
|
|
|
if let Ok(volume_stats) = handle.volume_statistics() {
|
|
|
|
worker.log(format!("Volume mounts: {}", volume_stats.volume_mounts));
|
|
|
|
let wearout = volume_stats.volume_mounts * 2; // (*100.0/50.0);
|
|
|
|
worker.log(format!("Cleaning tape wearout: {}%", wearout));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-10 15:37:09 +00:00
|
|
|
worker.log("Drive cleaned successfully");
|
2021-01-08 10:32:56 +00:00
|
|
|
|
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
|
|
|
)?;
|
2021-01-08 10:32:56 +00:00
|
|
|
|
|
|
|
Ok(upid_str.into())
|
|
|
|
}
|
|
|
|
|
2020-12-11 06:39:28 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-11 06:39:28 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
description: "The list of media labels with associated media Uuid (if any).",
|
|
|
|
type: Array,
|
|
|
|
items: {
|
|
|
|
type: LabelUuidMap,
|
|
|
|
},
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-11 06:39:28 +00:00
|
|
|
)]
|
2020-12-11 09:42:29 +00:00
|
|
|
/// List known media labels (Changer Inventory)
|
2020-12-11 06:39:28 +00:00
|
|
|
///
|
|
|
|
/// Note: Only useful for drives with associated changer device.
|
|
|
|
///
|
2020-12-11 09:42:29 +00:00
|
|
|
/// This method queries the changer to get a list of media labels.
|
|
|
|
///
|
|
|
|
/// Note: This updates the media online status.
|
2020-12-12 08:20:04 +00:00
|
|
|
pub async fn inventory(
|
2020-12-11 06:39:28 +00:00
|
|
|
drive: String,
|
|
|
|
) -> Result<Vec<LabelUuidMap>, Error> {
|
2021-02-18 14:40:25 +00:00
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
"inventorize".to_string(),
|
|
|
|
move |config| {
|
|
|
|
let (mut changer, changer_name) = required_media_changer(&config, &drive)?;
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
let label_text_list = changer.online_media_label_texts()?;
|
2021-02-05 09:50:21 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
let state_path = Path::new(TAPE_STATUS_DIR);
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
let mut inventory = Inventory::load(state_path)?;
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
update_changer_online_status(
|
|
|
|
&config,
|
|
|
|
&mut inventory,
|
|
|
|
&changer_name,
|
|
|
|
&label_text_list,
|
|
|
|
)?;
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
let mut list = Vec::new();
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
for label_text in label_text_list.iter() {
|
|
|
|
if label_text.starts_with("CLN") {
|
|
|
|
// skip cleaning unit
|
|
|
|
continue;
|
|
|
|
}
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
let label_text = label_text.to_string();
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
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 });
|
|
|
|
}
|
2020-12-12 08:20:04 +00:00
|
|
|
}
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-02-18 14:40:25 +00:00
|
|
|
Ok(list)
|
2020-12-11 06:39:28 +00:00
|
|
|
}
|
2021-02-18 14:40:25 +00:00
|
|
|
)
|
|
|
|
.await
|
2020-12-11 09:42:29 +00:00
|
|
|
}
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2020-12-11 09:42:29 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-11 09:42:29 +00:00
|
|
|
},
|
|
|
|
"read-all-labels": {
|
|
|
|
description: "Load all tapes and try read labels (even if already inventoried)",
|
|
|
|
type: bool,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-11 09:42:29 +00:00
|
|
|
)]
|
|
|
|
/// 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<bool>,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2020-12-11 09:42:29 +00:00
|
|
|
"inventory-update",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| {
|
2020-12-30 16:16:57 +00:00
|
|
|
let (mut changer, changer_name) = required_media_changer(&config, &drive)?;
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
let label_text_list = changer.online_media_label_texts()?;
|
|
|
|
if label_text_list.is_empty() {
|
2021-01-19 09:54:37 +00:00
|
|
|
worker.log("changer device does not list any media labels".to_string());
|
2020-12-11 06:39:28 +00:00
|
|
|
}
|
2020-12-11 09:42:29 +00:00
|
|
|
|
|
|
|
let state_path = Path::new(TAPE_STATUS_DIR);
|
|
|
|
|
|
|
|
let mut inventory = Inventory::load(state_path)?;
|
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
update_changer_online_status(&config, &mut inventory, &changer_name, &label_text_list)?;
|
2020-12-11 09:42:29 +00:00
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
for label_text in label_text_list.iter() {
|
|
|
|
if label_text.starts_with("CLN") {
|
|
|
|
worker.log(format!("skip cleaning unit '{}'", label_text));
|
2020-12-11 09:42:29 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
let label_text = label_text.to_string();
|
2020-12-11 09:42:29 +00:00
|
|
|
|
2021-01-19 09:27:59 +00:00
|
|
|
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;
|
2020-12-11 09:42:29 +00:00
|
|
|
}
|
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
if let Err(err) = changer.load_media(&label_text) {
|
|
|
|
worker.warn(format!("unable to load media '{}' - {}", label_text, err));
|
2020-12-11 06:39:28 +00:00
|
|
|
continue;
|
|
|
|
}
|
2020-12-11 09:42:29 +00:00
|
|
|
|
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
match drive.read_label() {
|
|
|
|
Err(err) => {
|
2021-01-13 12:26:59 +00:00
|
|
|
worker.warn(format!("unable to read label form media '{}' - {}", label_text, err));
|
2020-12-11 09:42:29 +00:00
|
|
|
}
|
2021-01-19 05:19:18 +00:00
|
|
|
Ok((None, _)) => {
|
2021-01-13 12:26:59 +00:00
|
|
|
worker.log(format!("media '{}' is empty", label_text));
|
2020-12-11 09:42:29 +00:00
|
|
|
}
|
2021-01-19 05:19:18 +00:00
|
|
|
Ok((Some(media_id), _key_config)) => {
|
2021-01-13 12:26:59 +00:00
|
|
|
if label_text != media_id.label.label_text {
|
2021-03-10 15:37:09 +00:00
|
|
|
worker.warn(format!("label text mismatch ({} != {})", label_text, media_id.label.label_text));
|
2020-12-11 09:42:29 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-01-13 12:26:59 +00:00
|
|
|
worker.log(format!("inventorize media '{}' with uuid '{}'", label_text, media_id.label.uuid));
|
2021-03-22 05:32:18 +00:00
|
|
|
|
|
|
|
if let Some(MediaSetLabel { ref pool, ref uuid, ..}) = media_id.media_set_label {
|
|
|
|
let _pool_lock = lock_media_pool(state_path, pool)?;
|
|
|
|
let _lock = lock_media_set(state_path, uuid, None)?;
|
|
|
|
MediaCatalog::destroy_unrelated_catalog(state_path, &media_id)?;
|
|
|
|
inventory.store(media_id, false)?;
|
|
|
|
} else {
|
|
|
|
let _lock = lock_unassigned_media_pool(state_path)?;
|
|
|
|
MediaCatalog::destroy(state_path, &media_id.label.uuid)?;
|
|
|
|
inventory.store(media_id, false)?;
|
|
|
|
};
|
2020-12-11 09:42:29 +00:00
|
|
|
}
|
|
|
|
}
|
2021-01-08 10:32:56 +00:00
|
|
|
changer.unload_media(None)?;
|
2020-12-11 06:39:28 +00:00
|
|
|
}
|
2020-12-11 09:42:29 +00:00
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
2020-12-11 09:42:29 +00:00
|
|
|
)?;
|
2020-12-11 06:39:28 +00:00
|
|
|
|
2020-12-11 09:42:29 +00:00
|
|
|
Ok(upid_str.into())
|
2020-12-11 06:39:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-11 06:50:19 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
2020-12-13 08:18:16 +00:00
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
2020-12-11 06:50:19 +00:00
|
|
|
},
|
|
|
|
pool: {
|
|
|
|
schema: MEDIA_POOL_NAME_SCHEMA,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2020-12-11 08:10:22 +00:00
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_WRITE, false),
|
|
|
|
},
|
2020-12-11 06:50:19 +00:00
|
|
|
)]
|
|
|
|
/// Label media with barcodes from changer device
|
|
|
|
pub fn barcode_label_media(
|
|
|
|
drive: String,
|
|
|
|
pool: Option<String>,
|
2020-12-11 08:10:22 +00:00
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
2020-12-11 06:50:19 +00:00
|
|
|
if let Some(ref pool) = pool {
|
2021-09-06 06:56:04 +00:00
|
|
|
let (pool_config, _digest) = pbs_config::media_pool::config()?;
|
2020-12-11 06:50:19 +00:00
|
|
|
|
|
|
|
if pool_config.sections.get(pool).is_none() {
|
|
|
|
bail!("no such pool ('{}')", pool);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2020-12-11 08:10:22 +00:00
|
|
|
"barcode-label-media",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| barcode_label_media_worker(worker, drive, &config, pool),
|
2020-12-11 08:10:22 +00:00
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(upid_str.into())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn barcode_label_media_worker(
|
|
|
|
worker: Arc<WorkerTask>,
|
|
|
|
drive: String,
|
2021-02-05 09:50:21 +00:00
|
|
|
drive_config: &SectionConfigData,
|
2020-12-11 08:10:22 +00:00
|
|
|
pool: Option<String>,
|
|
|
|
) -> Result<(), Error> {
|
2021-02-05 09:50:21 +00:00
|
|
|
let (mut changer, changer_name) = required_media_changer(drive_config, &drive)?;
|
2020-12-11 06:50:19 +00:00
|
|
|
|
2021-03-09 07:33:21 +00:00
|
|
|
let mut label_text_list = changer.online_media_label_texts()?;
|
|
|
|
|
|
|
|
// make sure we label them in the right order
|
|
|
|
label_text_list.sort();
|
2020-12-11 06:50:19 +00:00
|
|
|
|
|
|
|
let state_path = Path::new(TAPE_STATUS_DIR);
|
|
|
|
|
|
|
|
let mut inventory = Inventory::load(state_path)?;
|
|
|
|
|
2021-02-05 09:50:21 +00:00
|
|
|
update_changer_online_status(drive_config, &mut inventory, &changer_name, &label_text_list)?;
|
2020-12-11 06:50:19 +00:00
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
if label_text_list.is_empty() {
|
2020-12-11 06:50:19 +00:00
|
|
|
bail!("changer device does not list any media labels");
|
|
|
|
}
|
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
for label_text in label_text_list {
|
|
|
|
if label_text.starts_with("CLN") { continue; }
|
2020-12-11 06:50:19 +00:00
|
|
|
|
|
|
|
inventory.reload()?;
|
2021-01-13 12:26:59 +00:00
|
|
|
if inventory.find_media_by_label_text(&label_text).is_some() {
|
|
|
|
worker.log(format!("media '{}' already inventoried (already labeled)", label_text));
|
2020-12-11 06:50:19 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
worker.log(format!("checking/loading media '{}'", label_text));
|
2020-12-11 06:50:19 +00:00
|
|
|
|
2021-01-13 12:26:59 +00:00
|
|
|
if let Err(err) = changer.load_media(&label_text) {
|
|
|
|
worker.warn(format!("unable to load media '{}' - {}", label_text, err));
|
2020-12-11 06:50:19 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-02-05 09:50:21 +00:00
|
|
|
let mut drive = open_drive(drive_config, &drive)?;
|
2020-12-11 06:50:19 +00:00
|
|
|
drive.rewind()?;
|
|
|
|
|
|
|
|
match drive.read_next_file() {
|
2021-04-12 09:25:40 +00:00
|
|
|
Ok(_reader) => {
|
2021-03-31 07:19:19 +00:00
|
|
|
worker.log(format!("media '{}' is not empty (format it first)", label_text));
|
2020-12-11 06:50:19 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-04-12 09:25:40 +00:00
|
|
|
Err(BlockReadError::EndOfFile) => { /* EOF mark at BOT, assume tape is empty */ },
|
|
|
|
Err(BlockReadError::EndOfStream) => { /* tape is empty */ },
|
|
|
|
Err(_err) => {
|
|
|
|
worker.warn(format!("media '{}' read error (maybe not empty - format it first)", label_text));
|
|
|
|
continue;
|
2020-12-11 06:50:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let ctime = proxmox::tools::time::epoch_i64();
|
2020-12-14 16:37:16 +00:00
|
|
|
let label = MediaLabel {
|
2021-01-13 12:26:59 +00:00
|
|
|
label_text: label_text.to_string(),
|
2020-12-11 06:50:19 +00:00
|
|
|
uuid: Uuid::generate(),
|
|
|
|
ctime,
|
|
|
|
};
|
|
|
|
|
2020-12-11 08:10:22 +00:00
|
|
|
write_media_label(worker.clone(), &mut drive, label, pool.clone())?
|
2020-12-11 06:50:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-12-21 11:11:52 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
description: "A List of medium auxiliary memory attributes.",
|
|
|
|
type: Array,
|
|
|
|
items: {
|
|
|
|
type: MamAttribute,
|
|
|
|
},
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_AUDIT, false),
|
|
|
|
},
|
2020-12-21 11:11:52 +00:00
|
|
|
)]
|
2020-12-22 08:27:34 +00:00
|
|
|
/// Read Cartridge Memory (Medium auxiliary memory attributes)
|
2021-02-18 14:40:26 +00:00
|
|
|
pub async fn cartridge_memory(drive: String) -> Result<Vec<MamAttribute>, Error> {
|
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
"reading cartridge memory".to_string(),
|
|
|
|
move |config| {
|
2021-03-30 15:07:59 +00:00
|
|
|
let drive_config: LtoTapeDrive = config.lookup("lto", &drive)?;
|
2021-09-03 07:10:18 +00:00
|
|
|
let mut handle = open_lto_tape_drive(&drive_config)?;
|
2020-12-21 11:11:52 +00:00
|
|
|
|
2021-02-18 14:40:26 +00:00
|
|
|
handle.cartridge_memory()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.await
|
2020-12-21 11:11:52 +00:00
|
|
|
}
|
|
|
|
|
2021-01-22 07:45:35 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
type: Lp17VolumeStatistics,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_AUDIT, false),
|
|
|
|
},
|
2021-01-22 07:45:35 +00:00
|
|
|
)]
|
|
|
|
/// Read Volume Statistics (SCSI log page 17h)
|
2021-02-18 14:40:26 +00:00
|
|
|
pub async fn volume_statistics(drive: String) -> Result<Lp17VolumeStatistics, Error> {
|
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
"reading volume statistics".to_string(),
|
|
|
|
move |config| {
|
2021-03-30 15:07:59 +00:00
|
|
|
let drive_config: LtoTapeDrive = config.lookup("lto", &drive)?;
|
2021-09-03 07:10:18 +00:00
|
|
|
let mut handle = open_lto_tape_drive(&drive_config)?;
|
2021-01-22 07:45:35 +00:00
|
|
|
|
2021-02-18 14:40:26 +00:00
|
|
|
handle.volume_statistics()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.await
|
2021-01-22 07:45:35 +00:00
|
|
|
}
|
|
|
|
|
2020-12-22 09:42:22 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
2021-03-30 15:07:59 +00:00
|
|
|
type: LtoDriveAndMediaStatus,
|
2020-12-22 09:42:22 +00:00
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_AUDIT, false),
|
|
|
|
},
|
2020-12-22 09:42:22 +00:00
|
|
|
)]
|
2020-12-23 10:24:34 +00:00
|
|
|
/// Get drive/media status
|
2021-03-30 15:07:59 +00:00
|
|
|
pub async fn status(drive: String) -> Result<LtoDriveAndMediaStatus, Error> {
|
2021-02-18 14:40:26 +00:00
|
|
|
run_drive_blocking_task(
|
|
|
|
drive.clone(),
|
|
|
|
"reading drive status".to_string(),
|
|
|
|
move |config| {
|
2021-03-30 15:07:59 +00:00
|
|
|
let drive_config: LtoTapeDrive = config.lookup("lto", &drive)?;
|
2020-12-22 09:42:22 +00:00
|
|
|
|
2021-03-30 15:07:59 +00:00
|
|
|
// Note: use open_lto_tape_device, because this also works if no medium loaded
|
|
|
|
let file = open_lto_tape_device(&drive_config.path)?;
|
2020-12-22 09:42:22 +00:00
|
|
|
|
2021-03-30 15:07:59 +00:00
|
|
|
let mut handle = LtoTapeHandle::new(file)?;
|
2020-12-23 10:24:34 +00:00
|
|
|
|
2021-02-18 14:40:26 +00:00
|
|
|
handle.get_drive_and_media_status()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.await
|
2020-12-22 09:42:22 +00:00
|
|
|
}
|
|
|
|
|
2020-12-30 08:48:18 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
drive: {
|
|
|
|
schema: DRIVE_NAME_SCHEMA,
|
|
|
|
},
|
|
|
|
force: {
|
|
|
|
description: "Force overriding existing index.",
|
|
|
|
type: bool,
|
|
|
|
optional: true,
|
|
|
|
},
|
2021-03-25 12:08:34 +00:00
|
|
|
scan: {
|
|
|
|
description: "Re-read the whole tape to reconstruct the catalog instead of restoring saved versions.",
|
|
|
|
type: bool,
|
|
|
|
optional: true,
|
|
|
|
},
|
2020-12-30 08:48:18 +00:00
|
|
|
verbose: {
|
|
|
|
description: "Verbose mode - log all found chunks.",
|
|
|
|
type: bool,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
schema: UPID_SCHEMA,
|
|
|
|
},
|
2021-03-05 10:40:52 +00:00
|
|
|
access: {
|
|
|
|
permission: &Permission::Privilege(&["tape", "device", "{drive}"], PRIV_TAPE_READ, false),
|
|
|
|
},
|
2020-12-30 08:48:18 +00:00
|
|
|
)]
|
|
|
|
/// Scan media and record content
|
|
|
|
pub fn catalog_media(
|
|
|
|
drive: String,
|
|
|
|
force: Option<bool>,
|
2021-03-25 12:08:34 +00:00
|
|
|
scan: Option<bool>,
|
2020-12-30 08:48:18 +00:00
|
|
|
verbose: Option<bool>,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<Value, Error> {
|
|
|
|
let verbose = verbose.unwrap_or(false);
|
|
|
|
let force = force.unwrap_or(false);
|
2021-03-25 12:08:34 +00:00
|
|
|
let scan = scan.unwrap_or(false);
|
2020-12-30 08:48:18 +00:00
|
|
|
|
2021-02-18 14:40:23 +00:00
|
|
|
let upid_str = run_drive_worker(
|
|
|
|
rpcenv,
|
|
|
|
drive.clone(),
|
2020-12-30 08:48:18 +00:00
|
|
|
"catalog-media",
|
|
|
|
Some(drive.clone()),
|
2021-02-18 14:40:23 +00:00
|
|
|
move |worker, config| {
|
2020-12-30 08:48:18 +00:00
|
|
|
let mut drive = open_drive(&config, &drive)?;
|
|
|
|
|
|
|
|
drive.rewind()?;
|
|
|
|
|
|
|
|
let media_id = match drive.read_label()? {
|
2021-01-19 05:19:18 +00:00
|
|
|
(Some(media_id), key_config) => {
|
2020-12-30 08:48:18 +00:00
|
|
|
worker.log(format!(
|
|
|
|
"found media label: {}",
|
|
|
|
serde_json::to_string_pretty(&serde_json::to_value(&media_id)?)?
|
|
|
|
));
|
2021-01-19 05:19:18 +00:00
|
|
|
if key_config.is_some() {
|
|
|
|
worker.log(format!(
|
|
|
|
"encryption key config: {}",
|
|
|
|
serde_json::to_string_pretty(&serde_json::to_value(&key_config)?)?
|
|
|
|
));
|
|
|
|
}
|
2020-12-30 08:48:18 +00:00
|
|
|
media_id
|
|
|
|
},
|
2021-01-19 05:19:18 +00:00
|
|
|
(None, _) => bail!("media is empty (no media label found)"),
|
2020-12-30 08:48:18 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
let status_path = Path::new(TAPE_STATUS_DIR);
|
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
let mut inventory = Inventory::new(status_path);
|
2020-12-30 08:48:18 +00:00
|
|
|
|
2021-03-23 12:39:33 +00:00
|
|
|
let (_media_set_lock, media_set_uuid) = match media_id.media_set_label {
|
2020-12-30 08:48:18 +00:00
|
|
|
None => {
|
|
|
|
worker.log("media is empty");
|
2021-03-22 05:32:18 +00:00
|
|
|
let _lock = lock_unassigned_media_pool(status_path)?;
|
2020-12-30 08:48:18 +00:00
|
|
|
MediaCatalog::destroy(status_path, &media_id.label.uuid)?;
|
2021-03-22 05:32:18 +00:00
|
|
|
inventory.store(media_id.clone(), false)?;
|
2020-12-30 08:48:18 +00:00
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
Some(ref set) => {
|
|
|
|
if set.uuid.as_ref() == [0u8;16] { // media is empty
|
|
|
|
worker.log("media is empty");
|
2021-03-22 05:32:18 +00:00
|
|
|
let _lock = lock_unassigned_media_pool(status_path)?;
|
2020-12-30 08:48:18 +00:00
|
|
|
MediaCatalog::destroy(status_path, &media_id.label.uuid)?;
|
2021-03-22 05:32:18 +00:00
|
|
|
inventory.store(media_id.clone(), false)?;
|
2020-12-30 08:48:18 +00:00
|
|
|
return Ok(());
|
|
|
|
}
|
2021-01-20 16:27:01 +00:00
|
|
|
let encrypt_fingerprint = set.encryption_key_fingerprint.clone()
|
|
|
|
.map(|fp| (fp, set.uuid.clone()));
|
|
|
|
|
2021-01-18 12:36:11 +00:00
|
|
|
drive.set_encryption(encrypt_fingerprint)?;
|
|
|
|
|
2021-03-22 05:32:18 +00:00
|
|
|
let _pool_lock = lock_media_pool(status_path, &set.pool)?;
|
|
|
|
let media_set_lock = lock_media_set(status_path, &set.uuid, None)?;
|
|
|
|
|
|
|
|
MediaCatalog::destroy_unrelated_catalog(status_path, &media_id)?;
|
|
|
|
|
|
|
|
inventory.store(media_id.clone(), false)?;
|
|
|
|
|
2021-03-23 12:39:33 +00:00
|
|
|
(media_set_lock, &set.uuid)
|
2020-12-30 08:48:18 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-01-19 10:06:26 +00:00
|
|
|
if MediaCatalog::exists(status_path, &media_id.label.uuid) && !force {
|
|
|
|
bail!("media catalog exists (please use --force to overwrite)");
|
2020-12-30 08:48:18 +00:00
|
|
|
}
|
|
|
|
|
2021-03-25 12:08:34 +00:00
|
|
|
if !scan {
|
|
|
|
let media_set = inventory.compute_media_set_members(media_set_uuid)?;
|
|
|
|
|
|
|
|
if fast_catalog_restore(&worker, &mut drive, &media_set, &media_id.label.uuid)? {
|
|
|
|
return Ok(())
|
|
|
|
}
|
2021-03-23 12:39:33 +00:00
|
|
|
|
2021-03-25 12:08:34 +00:00
|
|
|
task_log!(worker, "no catalog found");
|
2021-03-23 12:39:33 +00:00
|
|
|
}
|
|
|
|
|
2021-03-25 12:08:34 +00:00
|
|
|
task_log!(worker, "scanning entire media to reconstruct catalog");
|
2021-03-23 12:39:33 +00:00
|
|
|
|
2021-03-24 09:09:23 +00:00
|
|
|
drive.rewind()?;
|
|
|
|
drive.read_label()?; // skip over labels - we already read them above
|
|
|
|
|
2021-04-16 11:17:17 +00:00
|
|
|
let mut checked_chunks = HashMap::new();
|
2021-05-04 10:21:47 +00:00
|
|
|
restore_media(worker, &mut drive, &media_id, None, &mut checked_chunks, verbose)?;
|
2020-12-30 08:48:18 +00:00
|
|
|
|
|
|
|
Ok(())
|
2021-02-18 14:40:23 +00:00
|
|
|
},
|
2020-12-30 08:48:18 +00:00
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(upid_str.into())
|
|
|
|
}
|
|
|
|
|
2021-01-27 10:33:49 +00:00
|
|
|
#[api(
|
|
|
|
input: {
|
2021-01-28 11:59:41 +00:00
|
|
|
properties: {
|
|
|
|
changer: {
|
|
|
|
schema: CHANGER_NAME_SCHEMA,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
2021-01-27 10:33:49 +00:00
|
|
|
},
|
|
|
|
returns: {
|
|
|
|
description: "The list of configured drives with model information.",
|
|
|
|
type: Array,
|
|
|
|
items: {
|
|
|
|
type: DriveListEntry,
|
|
|
|
},
|
|
|
|
},
|
2021-03-03 11:10:00 +00:00
|
|
|
access: {
|
|
|
|
description: "List configured tape drives filtered by Tape.Audit privileges",
|
|
|
|
permission: &Permission::Anybody,
|
|
|
|
},
|
2021-01-27 10:33:49 +00:00
|
|
|
)]
|
|
|
|
/// List drives
|
|
|
|
pub fn list_drives(
|
2021-01-28 11:59:41 +00:00
|
|
|
changer: Option<String>,
|
2021-01-27 10:33:49 +00:00
|
|
|
_param: Value,
|
2021-03-03 11:10:00 +00:00
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
2021-01-27 10:33:49 +00:00
|
|
|
) -> Result<Vec<DriveListEntry>, Error> {
|
2021-03-03 11:10:00 +00:00
|
|
|
let auth_id: Authid = rpcenv.get_auth_id().unwrap().parse()?;
|
|
|
|
let user_info = CachedUserInfo::new()?;
|
2021-01-27 10:33:49 +00:00
|
|
|
|
2021-09-03 07:10:18 +00:00
|
|
|
let (config, _) = pbs_config::drive::config()?;
|
2021-01-27 10:33:49 +00:00
|
|
|
|
2021-03-30 15:07:59 +00:00
|
|
|
let lto_drives = lto_tape_device_list();
|
2021-01-27 10:33:49 +00:00
|
|
|
|
2021-03-30 15:07:59 +00:00
|
|
|
let drive_list: Vec<LtoTapeDrive> = config.convert_to_typed_array("lto")?;
|
2021-01-27 10:33:49 +00:00
|
|
|
|
|
|
|
let mut list = Vec::new();
|
|
|
|
|
|
|
|
for drive in drive_list {
|
2021-01-28 11:59:41 +00:00
|
|
|
if changer.is_some() && drive.changer != changer {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-03-03 11:10:00 +00:00
|
|
|
let privs = user_info.lookup_privs(&auth_id, &["tape", "drive", &drive.name]);
|
|
|
|
if (privs & PRIV_TAPE_AUDIT) == 0 {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-03-30 15:07:59 +00:00
|
|
|
let info = lookup_device_identification(<o_drives, &drive.path);
|
2021-03-02 11:19:36 +00:00
|
|
|
let state = get_tape_device_state(&config, &drive.name)?;
|
|
|
|
let entry = DriveListEntry { config: drive, info, state };
|
2021-01-27 10:33:49 +00:00
|
|
|
list.push(entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(list)
|
|
|
|
}
|
|
|
|
|
2020-12-10 09:09:12 +00:00
|
|
|
#[sortable]
|
|
|
|
pub const SUBDIRS: SubdirMap = &sorted!([
|
2020-12-11 06:50:19 +00:00
|
|
|
(
|
|
|
|
"barcode-label-media",
|
|
|
|
&Router::new()
|
2021-01-29 10:49:11 +00:00
|
|
|
.post(&API_METHOD_BARCODE_LABEL_MEDIA)
|
2020-12-11 06:50:19 +00:00
|
|
|
),
|
2020-12-30 08:48:18 +00:00
|
|
|
(
|
|
|
|
"catalog",
|
|
|
|
&Router::new()
|
2021-01-29 10:49:11 +00:00
|
|
|
.post(&API_METHOD_CATALOG_MEDIA)
|
2020-12-30 08:48:18 +00:00
|
|
|
),
|
2021-01-08 10:32:56 +00:00
|
|
|
(
|
|
|
|
"clean",
|
|
|
|
&Router::new()
|
|
|
|
.put(&API_METHOD_CLEAN_DRIVE)
|
|
|
|
),
|
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()
|
2021-01-28 15:46:28 +00:00
|
|
|
.post(&API_METHOD_EJECT_MEDIA)
|
2020-12-09 16:35:31 +00:00
|
|
|
),
|
2020-12-09 16:50:48 +00:00
|
|
|
(
|
2021-03-31 07:19:19 +00:00
|
|
|
"format-media",
|
2020-12-09 16:50:48 +00:00
|
|
|
&Router::new()
|
2021-03-31 07:19:19 +00:00
|
|
|
.post(&API_METHOD_FORMAT_MEDIA)
|
2020-12-09 16:50:48 +00:00
|
|
|
),
|
2021-01-29 09:50:11 +00:00
|
|
|
(
|
|
|
|
"export-media",
|
|
|
|
&Router::new()
|
|
|
|
.put(&API_METHOD_EXPORT_MEDIA)
|
|
|
|
),
|
2020-12-11 06:39:28 +00:00
|
|
|
(
|
|
|
|
"inventory",
|
|
|
|
&Router::new()
|
|
|
|
.get(&API_METHOD_INVENTORY)
|
2020-12-11 09:42:29 +00:00
|
|
|
.put(&API_METHOD_UPDATE_INVENTORY)
|
2020-12-11 06:39:28 +00:00
|
|
|
),
|
2020-12-10 11:30:27 +00:00
|
|
|
(
|
|
|
|
"label-media",
|
|
|
|
&Router::new()
|
2021-01-29 09:50:11 +00:00
|
|
|
.post(&API_METHOD_LABEL_MEDIA)
|
2020-12-10 11:30:27 +00:00
|
|
|
),
|
2021-01-28 11:59:42 +00:00
|
|
|
(
|
|
|
|
"load-media",
|
|
|
|
&Router::new()
|
2021-02-18 08:04:51 +00:00
|
|
|
.post(&API_METHOD_LOAD_MEDIA)
|
2021-01-28 11:59:42 +00:00
|
|
|
),
|
2020-12-08 14:42:50 +00:00
|
|
|
(
|
|
|
|
"load-slot",
|
|
|
|
&Router::new()
|
2021-04-30 08:13:13 +00:00
|
|
|
.post(&API_METHOD_LOAD_SLOT)
|
2020-12-08 14:42:50 +00:00
|
|
|
),
|
2020-12-21 11:11:52 +00:00
|
|
|
(
|
2020-12-22 08:27:34 +00:00
|
|
|
"cartridge-memory",
|
2020-12-21 11:11:52 +00:00
|
|
|
&Router::new()
|
2021-01-28 11:59:43 +00:00
|
|
|
.get(&API_METHOD_CARTRIDGE_MEMORY)
|
2020-12-21 11:11:52 +00:00
|
|
|
),
|
2021-01-22 07:45:35 +00:00
|
|
|
(
|
|
|
|
"volume-statistics",
|
|
|
|
&Router::new()
|
2021-01-28 11:59:43 +00:00
|
|
|
.get(&API_METHOD_VOLUME_STATISTICS)
|
2021-01-22 07:45:35 +00:00
|
|
|
),
|
2020-12-11 06:39:28 +00:00
|
|
|
(
|
|
|
|
"read-label",
|
|
|
|
&Router::new()
|
|
|
|
.get(&API_METHOD_READ_LABEL)
|
|
|
|
),
|
2021-03-01 10:11:38 +00:00
|
|
|
(
|
|
|
|
"restore-key",
|
|
|
|
&Router::new()
|
|
|
|
.post(&API_METHOD_RESTORE_KEY)
|
|
|
|
),
|
2020-12-10 08:09:06 +00:00
|
|
|
(
|
|
|
|
"rewind",
|
|
|
|
&Router::new()
|
2021-01-28 15:36:10 +00:00
|
|
|
.post(&API_METHOD_REWIND)
|
2020-12-10 08:09:06 +00:00
|
|
|
),
|
2020-12-22 09:42:22 +00:00
|
|
|
(
|
|
|
|
"status",
|
|
|
|
&Router::new()
|
|
|
|
.get(&API_METHOD_STATUS)
|
|
|
|
),
|
2020-12-08 14:42:50 +00:00
|
|
|
(
|
|
|
|
"unload",
|
|
|
|
&Router::new()
|
2021-02-18 08:04:51 +00:00
|
|
|
.post(&API_METHOD_UNLOAD)
|
2020-12-08 14:42:50 +00:00
|
|
|
),
|
2020-12-10 09:09:12 +00:00
|
|
|
]);
|
2020-12-08 14:42:50 +00:00
|
|
|
|
2021-01-27 10:33:49 +00:00
|
|
|
const ITEM_ROUTER: Router = Router::new()
|
2020-12-08 14:42:50 +00:00
|
|
|
.get(&list_subdirs_api_method!(SUBDIRS))
|
2021-01-27 10:33:49 +00:00
|
|
|
.subdirs(&SUBDIRS);
|
|
|
|
|
|
|
|
pub const ROUTER: Router = Router::new()
|
|
|
|
.get(&API_METHOD_LIST_DRIVES)
|
|
|
|
.match_all("drive", &ITEM_ROUTER);
|