2020-08-06 13:46:01 +00:00
|
|
|
use std::convert::TryFrom;
|
2019-11-21 13:53:15 +00:00
|
|
|
use std::fmt;
|
2019-02-14 10:11:39 +00:00
|
|
|
|
2020-04-17 12:11:25 +00:00
|
|
|
use anyhow::{format_err, Error};
|
2019-02-14 10:11:39 +00:00
|
|
|
|
2019-11-21 13:53:15 +00:00
|
|
|
use proxmox::api::schema::*;
|
2019-02-14 10:11:39 +00:00
|
|
|
|
2020-05-28 19:02:54 +00:00
|
|
|
use crate::api2::types::*;
|
2019-02-14 10:11:39 +00:00
|
|
|
|
2019-11-21 08:36:41 +00:00
|
|
|
/// API schema format definition for repository URLs
|
|
|
|
pub const BACKUP_REPO_URL: ApiStringFormat = ApiStringFormat::Pattern(&BACKUP_REPO_URL_REGEX);
|
|
|
|
|
2019-02-14 10:11:39 +00:00
|
|
|
/// Reference remote backup locations
|
|
|
|
///
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct BackupRepository {
|
|
|
|
/// The user name used for Authentication
|
2020-08-06 13:46:01 +00:00
|
|
|
user: Option<Userid>,
|
2019-02-14 10:11:39 +00:00
|
|
|
/// The host name or IP address
|
2019-03-13 08:47:12 +00:00
|
|
|
host: Option<String>,
|
2020-09-29 14:18:58 +00:00
|
|
|
/// The port
|
|
|
|
port: Option<u16>,
|
2019-02-14 10:11:39 +00:00
|
|
|
/// The name of the datastore
|
2019-03-13 08:47:12 +00:00
|
|
|
store: String,
|
2019-02-14 10:11:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl BackupRepository {
|
|
|
|
|
2020-09-29 14:18:58 +00:00
|
|
|
pub fn new(user: Option<Userid>, host: Option<String>, port: Option<u16>, store: String) -> Self {
|
|
|
|
Self { user, host, port, store }
|
2019-12-19 07:44:38 +00:00
|
|
|
}
|
|
|
|
|
2020-08-06 13:46:01 +00:00
|
|
|
pub fn user(&self) -> &Userid {
|
2019-03-13 08:47:12 +00:00
|
|
|
if let Some(ref user) = self.user {
|
2020-08-06 13:46:01 +00:00
|
|
|
return &user;
|
2019-03-13 08:47:12 +00:00
|
|
|
}
|
2020-08-06 13:46:01 +00:00
|
|
|
Userid::root_userid()
|
2019-03-13 08:47:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn host(&self) -> &str {
|
|
|
|
if let Some(ref host) = self.host {
|
|
|
|
return host;
|
|
|
|
}
|
|
|
|
"localhost"
|
|
|
|
}
|
|
|
|
|
2020-09-29 14:18:58 +00:00
|
|
|
pub fn port(&self) -> u16 {
|
|
|
|
if let Some(port) = self.port {
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
8007
|
|
|
|
}
|
|
|
|
|
2019-03-13 08:47:12 +00:00
|
|
|
pub fn store(&self) -> &str {
|
|
|
|
&self.store
|
|
|
|
}
|
2019-03-13 08:57:36 +00:00
|
|
|
}
|
2019-03-13 08:47:12 +00:00
|
|
|
|
2019-03-13 08:57:36 +00:00
|
|
|
impl fmt::Display for BackupRepository {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2020-09-29 14:18:58 +00:00
|
|
|
match (&self.user, &self.host, self.port) {
|
|
|
|
(Some(user), _, _) => write!(f, "{}@{}:{}:{}", user, self.host(), self.port(), self.store),
|
|
|
|
(None, Some(host), None) => write!(f, "{}:{}", host, self.store),
|
|
|
|
(None, _, Some(port)) => write!(f, "{}:{}:{}", self.host(), port, self.store),
|
|
|
|
(None, None, None) => write!(f, "{}", self.store),
|
|
|
|
}
|
2019-03-13 08:47:12 +00:00
|
|
|
}
|
2019-02-14 10:11:39 +00:00
|
|
|
}
|
2019-03-13 09:09:39 +00:00
|
|
|
|
|
|
|
impl std::str::FromStr for BackupRepository {
|
|
|
|
type Err = Error;
|
|
|
|
|
|
|
|
/// Parse a repository URL.
|
|
|
|
///
|
|
|
|
/// This parses strings like `user@host:datastore`. The `user` and
|
|
|
|
/// `host` parts are optional, where `host` defaults to the local
|
|
|
|
/// host, and `user` defaults to `root@pam`.
|
|
|
|
fn from_str(url: &str) -> Result<Self, Self::Err> {
|
|
|
|
|
2019-11-21 08:36:41 +00:00
|
|
|
let cap = (BACKUP_REPO_URL_REGEX.regex_obj)().captures(url)
|
2019-03-13 09:09:39 +00:00
|
|
|
.ok_or_else(|| format_err!("unable to parse repository url '{}'", url))?;
|
|
|
|
|
|
|
|
Ok(Self {
|
2020-08-06 13:46:01 +00:00
|
|
|
user: cap.get(1).map(|m| Userid::try_from(m.as_str().to_owned())).transpose()?,
|
2019-03-13 09:09:39 +00:00
|
|
|
host: cap.get(2).map(|m| m.as_str().to_owned()),
|
2020-09-29 14:18:58 +00:00
|
|
|
port: cap.get(3).map(|m| m.as_str().parse::<u16>()).transpose()?,
|
|
|
|
store: cap[4].to_owned(),
|
2019-03-13 09:09:39 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|