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::*;
|
2020-01-21 11:28:01 +00:00
|
|
|
use proxmox::const_regex;
|
2019-02-14 10:11:39 +00:00
|
|
|
|
2019-11-21 08:36:41 +00:00
|
|
|
const_regex! {
|
2019-02-14 10:17:08 +00:00
|
|
|
/// Regular expression to parse repository URLs
|
2019-11-21 08:36:41 +00:00
|
|
|
pub BACKUP_REPO_URL_REGEX = r"^(?:(?:([\w@]+)@)?([\w\-_.]+):)?(\w+)$";
|
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
|
2019-03-13 08:47:12 +00:00
|
|
|
user: Option<String>,
|
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>,
|
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 {
|
|
|
|
|
2019-12-19 07:44:38 +00:00
|
|
|
pub fn new(user: Option<String>, host: Option<String>, store: String) -> Self {
|
|
|
|
Self { user, host, store }
|
|
|
|
}
|
|
|
|
|
2019-03-13 08:47:12 +00:00
|
|
|
pub fn user(&self) -> &str {
|
|
|
|
if let Some(ref user) = self.user {
|
|
|
|
return user;
|
|
|
|
}
|
|
|
|
"root@pam"
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn host(&self) -> &str {
|
|
|
|
if let Some(ref host) = self.host {
|
|
|
|
return host;
|
|
|
|
}
|
|
|
|
"localhost"
|
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
|
|
|
if let Some(ref user) = self.user {
|
|
|
|
write!(f, "{}@{}:{}", user, self.host(), self.store)
|
|
|
|
} else if let Some(ref host) = self.host {
|
|
|
|
write!(f, "{}:{}", host, self.store)
|
|
|
|
} else {
|
|
|
|
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 {
|
|
|
|
user: cap.get(1).map(|m| m.as_str().to_owned()),
|
|
|
|
host: cap.get(2).map(|m| m.as_str().to_owned()),
|
|
|
|
store: cap[3].to_owned(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|