2020-12-09 11:58:43 +00:00
|
|
|
use anyhow::{Error};
|
|
|
|
use serde_json::Value;
|
|
|
|
|
|
|
|
use proxmox::{
|
|
|
|
api::{
|
|
|
|
api,
|
|
|
|
cli::*,
|
|
|
|
RpcEnvironment,
|
|
|
|
ApiHandler,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
use proxmox_backup::{
|
|
|
|
api2::{
|
|
|
|
self,
|
|
|
|
types::{
|
|
|
|
CHANGER_ID_SCHEMA,
|
|
|
|
LINUX_DRIVE_PATH_SCHEMA,
|
|
|
|
ScsiTapeChanger,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
tape::{
|
|
|
|
complete_changer_path,
|
|
|
|
},
|
|
|
|
config::{
|
|
|
|
self,
|
|
|
|
drive::{
|
|
|
|
complete_drive_name,
|
|
|
|
complete_changer_name,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
pub fn changer_commands() -> CommandLineInterface {
|
|
|
|
|
|
|
|
let cmd_def = CliCommandMap::new()
|
2020-12-10 06:58:45 +00:00
|
|
|
.insert("scan", CliCommand::new(&API_METHOD_SCAN_FOR_CHANGERS))
|
2020-12-09 11:58:43 +00:00
|
|
|
.insert("list", CliCommand::new(&API_METHOD_LIST_CHANGERS))
|
|
|
|
.insert("config",
|
|
|
|
CliCommand::new(&API_METHOD_GET_CONFIG)
|
|
|
|
.arg_param(&["name"])
|
|
|
|
.completion_cb("name", complete_changer_name)
|
|
|
|
)
|
|
|
|
.insert(
|
|
|
|
"remove",
|
|
|
|
CliCommand::new(&API_METHOD_DELETE_CHANGER)
|
|
|
|
.arg_param(&["name"])
|
|
|
|
.completion_cb("name", complete_changer_name)
|
|
|
|
)
|
|
|
|
.insert(
|
|
|
|
"create",
|
|
|
|
CliCommand::new(&API_METHOD_CREATE_CHANGER)
|
|
|
|
.arg_param(&["name"])
|
|
|
|
.completion_cb("name", complete_drive_name)
|
|
|
|
.completion_cb("path", complete_changer_path)
|
|
|
|
)
|
|
|
|
.insert(
|
|
|
|
"update",
|
|
|
|
CliCommand::new(&API_METHOD_UPDATE_CHANGER)
|
|
|
|
.arg_param(&["name"])
|
|
|
|
.completion_cb("name", complete_changer_name)
|
|
|
|
.completion_cb("path", complete_changer_path)
|
|
|
|
)
|
|
|
|
.insert("status",
|
|
|
|
CliCommand::new(&API_METHOD_GET_STATUS)
|
|
|
|
.arg_param(&["name"])
|
|
|
|
.completion_cb("name", complete_changer_name)
|
|
|
|
)
|
|
|
|
.insert("transfer",
|
|
|
|
CliCommand::new(&API_METHOD_TRANSFER)
|
|
|
|
.arg_param(&["name"])
|
|
|
|
.completion_cb("name", complete_changer_name)
|
|
|
|
)
|
|
|
|
;
|
|
|
|
|
|
|
|
cmd_def.into()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
name: {
|
|
|
|
schema: CHANGER_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
path: {
|
|
|
|
schema: LINUX_DRIVE_PATH_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Create a new changer device
|
|
|
|
fn create_changer(
|
|
|
|
param: Value,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let info = &api2::config::changer::API_METHOD_CREATE_CHANGER;
|
|
|
|
match info.handler {
|
|
|
|
ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
"output-format": {
|
|
|
|
schema: OUTPUT_FORMAT,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// List changers
|
|
|
|
fn list_changers(
|
|
|
|
param: Value,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let output_format = get_output_format(¶m);
|
|
|
|
let info = &api2::config::changer::API_METHOD_LIST_CHANGERS;
|
|
|
|
let mut data = match info.handler {
|
|
|
|
ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
let options = default_table_format_options()
|
|
|
|
.column(ColumnConfig::new("name"))
|
|
|
|
.column(ColumnConfig::new("path"))
|
|
|
|
.column(ColumnConfig::new("vendor"))
|
|
|
|
.column(ColumnConfig::new("model"))
|
|
|
|
.column(ColumnConfig::new("serial"))
|
|
|
|
;
|
|
|
|
|
|
|
|
format_and_print_result_full(&mut data, info.returns, &output_format, &options);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
"output-format": {
|
|
|
|
schema: OUTPUT_FORMAT,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Scan for SCSI tape changers
|
|
|
|
fn scan_for_changers(
|
|
|
|
param: Value,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let output_format = get_output_format(¶m);
|
|
|
|
let info = &api2::tape::changer::API_METHOD_SCAN_CHANGERS;
|
|
|
|
let mut data = match info.handler {
|
|
|
|
ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
let options = default_table_format_options()
|
|
|
|
.column(ColumnConfig::new("path"))
|
|
|
|
.column(ColumnConfig::new("vendor"))
|
|
|
|
.column(ColumnConfig::new("model"))
|
|
|
|
.column(ColumnConfig::new("serial"))
|
|
|
|
;
|
|
|
|
|
|
|
|
format_and_print_result_full(&mut data, info.returns, &output_format, &options);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
"output-format": {
|
|
|
|
schema: OUTPUT_FORMAT,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
name: {
|
|
|
|
schema: CHANGER_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Get tape changer configuration
|
|
|
|
fn get_config(
|
|
|
|
param: Value,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let output_format = get_output_format(¶m);
|
|
|
|
let info = &api2::config::changer::API_METHOD_GET_CONFIG;
|
|
|
|
let mut data = match info.handler {
|
|
|
|
ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
let options = default_table_format_options()
|
|
|
|
.column(ColumnConfig::new("name"))
|
|
|
|
.column(ColumnConfig::new("path"))
|
|
|
|
;
|
|
|
|
|
|
|
|
format_and_print_result_full(&mut data, info.returns, &output_format, &options);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
name: {
|
|
|
|
schema: CHANGER_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Delete a tape changer configuration
|
|
|
|
fn delete_changer(
|
|
|
|
param: Value,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let info = &api2::config::changer::API_METHOD_DELETE_CHANGER;
|
|
|
|
|
|
|
|
match info.handler {
|
|
|
|
ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
name: {
|
|
|
|
schema: CHANGER_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
path: {
|
|
|
|
schema: LINUX_DRIVE_PATH_SCHEMA,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Update a tape changer configuration
|
|
|
|
fn update_changer(
|
|
|
|
param: Value,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let info = &api2::config::changer::API_METHOD_UPDATE_CHANGER;
|
|
|
|
|
|
|
|
match info.handler {
|
|
|
|
ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
"output-format": {
|
|
|
|
schema: OUTPUT_FORMAT,
|
|
|
|
optional: true,
|
|
|
|
},
|
|
|
|
name: {
|
|
|
|
schema: CHANGER_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Get tape changer status
|
|
|
|
fn get_status(
|
|
|
|
param: Value,
|
|
|
|
rpcenv: &mut dyn RpcEnvironment,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let output_format = get_output_format(¶m);
|
|
|
|
let info = &api2::tape::changer::API_METHOD_GET_STATUS;
|
|
|
|
let mut data = match info.handler {
|
|
|
|
ApiHandler::Sync(handler) => (handler)(param, info, rpcenv)?,
|
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
|
|
|
|
let options = default_table_format_options()
|
|
|
|
.column(ColumnConfig::new("entry-kind"))
|
|
|
|
.column(ColumnConfig::new("entry-id"))
|
|
|
|
.column(ColumnConfig::new("changer-id"))
|
|
|
|
.column(ColumnConfig::new("loaded-slot"))
|
|
|
|
;
|
|
|
|
|
|
|
|
format_and_print_result_full(&mut data, info.returns, &output_format, &options);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[api(
|
|
|
|
input: {
|
|
|
|
properties: {
|
|
|
|
name: {
|
|
|
|
schema: CHANGER_ID_SCHEMA,
|
|
|
|
},
|
|
|
|
from: {
|
|
|
|
description: "Source slot number",
|
|
|
|
minimum: 1,
|
|
|
|
},
|
|
|
|
to: {
|
|
|
|
description: "Destination slot number",
|
|
|
|
minimum: 1,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)]
|
|
|
|
/// Transfers media from one slot to another
|
|
|
|
fn transfer(
|
|
|
|
name: String,
|
|
|
|
from: u64,
|
|
|
|
to: u64,
|
|
|
|
_param: Value,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
|
|
|
|
let (config, _digest) = config::drive::config()?;
|
|
|
|
|
|
|
|
let data: ScsiTapeChanger = config.lookup("changer", &name)?;
|
|
|
|
|
|
|
|
let mut command = std::process::Command::new("mtx");
|
|
|
|
command.args(&["-f", &data.path, "transfer", &from.to_string(), &to.to_string()]);
|
|
|
|
|
|
|
|
proxmox_backup::tools::run_command(command, None)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|