mirror of
https://github.com/dswd/zvault
synced 2025-01-04 19:51:51 +00:00
Better ui
This commit is contained in:
parent
c02a60f78b
commit
ced360b881
@ -1,3 +1,8 @@
|
|||||||
|
https://partner-images.canonical.com/core/xenial/20170331/ubuntu-xenial-core-cloudimg-amd64-root.tar.gz
|
||||||
|
http://sun.aei.polsl.pl/~sdeor/corpus/silesia.zip
|
||||||
|
http://wanos.co/assets/silesia.tar
|
||||||
|
|
||||||
|
|
||||||
Algorithm comparison on file test_data/ubuntu-16.04.1.tar
|
Algorithm comparison on file test_data/ubuntu-16.04.1.tar
|
||||||
|
|
||||||
Reading input file... done. 183869440 bytes
|
Reading input file... done. 183869440 bytes
|
||||||
|
@ -172,6 +172,7 @@ impl BundleDb {
|
|||||||
pub fn update_cache(&mut self, new: &[StoredBundle], gone: &[StoredBundle]) -> Result<(), BundleDbError> {
|
pub fn update_cache(&mut self, new: &[StoredBundle], gone: &[StoredBundle]) -> Result<(), BundleDbError> {
|
||||||
for bundle in new {
|
for bundle in new {
|
||||||
if bundle.info.mode == BundleMode::Meta {
|
if bundle.info.mode == BundleMode::Meta {
|
||||||
|
info!("Copying new meta bundle to local cache: {}", bundle.info.id);
|
||||||
try!(self.copy_remote_bundle_to_cache(bundle));
|
try!(self.copy_remote_bundle_to_cache(bundle));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
322
src/cli/args.rs
322
src/cli/args.rs
@ -1,6 +1,8 @@
|
|||||||
use ::prelude::*;
|
use ::prelude::*;
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
use clap::{App, AppSettings, Arg, SubCommand};
|
use clap::{App, AppSettings, Arg, SubCommand};
|
||||||
|
|
||||||
pub enum Arguments {
|
pub enum Arguments {
|
||||||
@ -127,12 +129,18 @@ pub enum Arguments {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
pub fn parse_repo_path(repo_path: &str, backup_restr: Option<bool>, path_restr: Option<bool>) -> Result<(&str, Option<&str>, Option<&str>), ErrorCode> {
|
fn parse_repo_path(repo_path: &str, existing: bool, backup_restr: Option<bool>, path_restr: Option<bool>) -> Result<(&str, Option<&str>, Option<&str>), String> {
|
||||||
let mut parts = repo_path.splitn(3, "::");
|
let mut parts = repo_path.splitn(3, "::");
|
||||||
let mut repo = parts.next().unwrap_or(&DEFAULT_REPOSITORY);
|
let mut repo = parts.next().unwrap_or(&DEFAULT_REPOSITORY);
|
||||||
if repo.is_empty() {
|
if repo.is_empty() {
|
||||||
repo = &DEFAULT_REPOSITORY;
|
repo = &DEFAULT_REPOSITORY;
|
||||||
}
|
}
|
||||||
|
if existing && !Path::new(repo).join("config.yaml").exists() {
|
||||||
|
return Err("The specified repository does not exist".to_string());
|
||||||
|
}
|
||||||
|
if !existing && Path::new(repo).exists() {
|
||||||
|
return Err("The specified repository already exists".to_string());
|
||||||
|
}
|
||||||
let mut backup = parts.next();
|
let mut backup = parts.next();
|
||||||
if let Some(val) = backup {
|
if let Some(val) = backup {
|
||||||
if val.is_empty() {
|
if val.is_empty() {
|
||||||
@ -147,82 +155,105 @@ pub fn parse_repo_path(repo_path: &str, backup_restr: Option<bool>, path_restr:
|
|||||||
}
|
}
|
||||||
if let Some(restr) = backup_restr {
|
if let Some(restr) = backup_restr {
|
||||||
if !restr && backup.is_some() {
|
if !restr && backup.is_some() {
|
||||||
error!("No backup may be given here");
|
return Err("No backup may be given here".to_string());
|
||||||
return Err(ErrorCode::InvalidArgs);
|
|
||||||
}
|
}
|
||||||
if restr && backup.is_none() {
|
if restr && backup.is_none() {
|
||||||
error!("A backup must be specified");
|
return Err("A backup must be specified".to_string());
|
||||||
return Err(ErrorCode::InvalidArgs);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if let Some(restr) = path_restr {
|
if let Some(restr) = path_restr {
|
||||||
if !restr && path.is_some() {
|
if !restr && path.is_some() {
|
||||||
error!("No subpath may be given here");
|
return Err("No subpath may be given here".to_string());
|
||||||
return Err(ErrorCode::InvalidArgs);
|
|
||||||
}
|
}
|
||||||
if restr && path.is_none() {
|
if restr && path.is_none() {
|
||||||
error!("A subpath must be specified");
|
return Err("A subpath must be specified".to_string());
|
||||||
return Err(ErrorCode::InvalidArgs);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok((repo, backup, path))
|
Ok((repo, backup, path))
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_num(num: &str, name: &str) -> Result<u64, ErrorCode> {
|
#[allow(unknown_lints,needless_pass_by_value)]
|
||||||
|
fn validate_repo_path(repo_path: String, existing: bool, backup_restr: Option<bool>, path_restr: Option<bool>) -> Result<(), String> {
|
||||||
|
parse_repo_path(&repo_path, existing, backup_restr, path_restr).map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_num(num: &str) -> Result<u64, String> {
|
||||||
if let Ok(num) = num.parse::<u64>() {
|
if let Ok(num) = num.parse::<u64>() {
|
||||||
Ok(num)
|
Ok(num)
|
||||||
} else {
|
} else {
|
||||||
error!("{} must be a number, was '{}'", name, num);
|
Err("Must be a number".to_string())
|
||||||
Err(ErrorCode::InvalidArgs)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_chunker(val: &str) -> Result<ChunkerType, ErrorCode> {
|
#[allow(unknown_lints,needless_pass_by_value)]
|
||||||
|
fn validate_num(val: String) -> Result<(), String> {
|
||||||
|
parse_num(&val).map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_chunker(val: &str) -> Result<ChunkerType, String> {
|
||||||
if let Ok(chunker) = ChunkerType::from_string(val) {
|
if let Ok(chunker) = ChunkerType::from_string(val) {
|
||||||
Ok(chunker)
|
Ok(chunker)
|
||||||
} else {
|
} else {
|
||||||
error!("Invalid chunker method/size: {}", val);
|
Err("Invalid chunker method/size".to_string())
|
||||||
Err(ErrorCode::InvalidArgs)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_compression(val: &str) -> Result<Option<Compression>, ErrorCode> {
|
#[allow(unknown_lints,needless_pass_by_value)]
|
||||||
|
fn validate_chunker(val: String) -> Result<(), String> {
|
||||||
|
parse_chunker(&val).map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_compression(val: &str) -> Result<Option<Compression>, String> {
|
||||||
if val == "none" {
|
if val == "none" {
|
||||||
return Ok(None)
|
return Ok(None)
|
||||||
}
|
}
|
||||||
if let Ok(compression) = Compression::from_string(val) {
|
if let Ok(compression) = Compression::from_string(val) {
|
||||||
Ok(Some(compression))
|
Ok(Some(compression))
|
||||||
} else {
|
} else {
|
||||||
error!("Invalid compression method/level: {}", val);
|
Err("Invalid compression method/level".to_string())
|
||||||
Err(ErrorCode::InvalidArgs)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_public_key(val: &str) -> Result<PublicKey, ErrorCode> {
|
#[allow(unknown_lints,needless_pass_by_value)]
|
||||||
|
fn validate_compression(val: String) -> Result<(), String> {
|
||||||
|
parse_compression(&val).map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_public_key(val: &str) -> Result<Option<PublicKey>, String> {
|
||||||
|
if val.to_lowercase() == "none" {
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
let bytes = match parse_hex(val) {
|
let bytes = match parse_hex(val) {
|
||||||
Ok(bytes) => bytes,
|
Ok(bytes) => bytes,
|
||||||
Err(_) => {
|
Err(_) => {
|
||||||
error!("Invalid key: {}", val);
|
return Err("Invalid hexadecimal".to_string());
|
||||||
return Err(ErrorCode::InvalidArgs);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
if let Some(key) = PublicKey::from_slice(&bytes) {
|
if let Some(key) = PublicKey::from_slice(&bytes) {
|
||||||
Ok(key)
|
Ok(Some(key))
|
||||||
} else {
|
} else {
|
||||||
error!("Invalid key: {}", val);
|
return Err("Invalid key".to_string())
|
||||||
Err(ErrorCode::InvalidArgs)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_hash(val: &str) -> Result<HashMethod, ErrorCode> {
|
#[allow(unknown_lints,needless_pass_by_value)]
|
||||||
|
fn validate_public_key(val: String) -> Result<(), String> {
|
||||||
|
parse_public_key(&val).map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_hash(val: &str) -> Result<HashMethod, String> {
|
||||||
if let Ok(hash) = HashMethod::from(val) {
|
if let Ok(hash) = HashMethod::from(val) {
|
||||||
Ok(hash)
|
Ok(hash)
|
||||||
} else {
|
} else {
|
||||||
error!("Invalid hash method: {}", val);
|
Err("Invalid hash method".to_string())
|
||||||
Err(ErrorCode::InvalidArgs)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(unknown_lints,needless_pass_by_value)]
|
||||||
|
fn validate_hash(val: String) -> Result<(), String> {
|
||||||
|
parse_hash(&val).map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
fn parse_bundle_id(val: &str) -> Result<BundleId, ErrorCode> {
|
fn parse_bundle_id(val: &str) -> Result<BundleId, ErrorCode> {
|
||||||
if let Ok(hash) = Hash::from_string(val) {
|
if let Ok(hash) = Hash::from_string(val) {
|
||||||
Ok(BundleId(hash))
|
Ok(BundleId(hash))
|
||||||
@ -232,109 +263,166 @@ fn parse_bundle_id(val: &str) -> Result<BundleId, ErrorCode> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(unknown_lints,needless_pass_by_value)]
|
||||||
|
fn validate_existing_path(val: String) -> Result<(), String> {
|
||||||
|
if !Path::new(&val).exists() {
|
||||||
|
Err("Path does not exist".to_string())
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#[allow(unknown_lints,cyclomatic_complexity)]
|
#[allow(unknown_lints,cyclomatic_complexity)]
|
||||||
pub fn parse() -> Result<Arguments, ErrorCode> {
|
pub fn parse() -> Result<Arguments, ErrorCode> {
|
||||||
let args = App::new("zvault").version(crate_version!()).author(crate_authors!(",\n")).about(crate_description!())
|
let args = App::new("zvault").version(crate_version!()).author(crate_authors!(",\n")).about(crate_description!())
|
||||||
.settings(&[AppSettings::AllowMissingPositional, AppSettings::VersionlessSubcommands, AppSettings::SubcommandRequiredElseHelp])
|
.settings(&[AppSettings::AllowMissingPositional, AppSettings::VersionlessSubcommands, AppSettings::SubcommandRequiredElseHelp])
|
||||||
.global_settings(&[AppSettings::UnifiedHelpMessage, AppSettings::ColoredHelp, AppSettings::ColorAuto])
|
.global_settings(&[AppSettings::UnifiedHelpMessage, AppSettings::ColoredHelp, AppSettings::ColorAuto])
|
||||||
.subcommand(SubCommand::with_name("init").about("Initialize a new repository")
|
.subcommand(SubCommand::with_name("init").about("Initialize a new repository")
|
||||||
.arg(Arg::from_usage("[bundle_size] --bundle-size [SIZE] 'Set the target bundle size in MiB (default: 25)'"))
|
.arg(Arg::from_usage("[bundle_size] --bundle-size [SIZE] 'Set the target bundle size in MiB'")
|
||||||
.arg(Arg::from_usage("--chunker [CHUNKER] 'Set the chunker algorithm and target chunk size (default: fastcdc/16)'"))
|
.default_value(DEFAULT_BUNDLE_SIZE_STR).validator(validate_num))
|
||||||
.arg(Arg::from_usage("-c --compression [COMPRESSION] 'Set the compression method and level (default: brotli/3)'"))
|
.arg(Arg::from_usage("--chunker [CHUNKER] 'Set the chunker algorithm and target chunk size'")
|
||||||
|
.default_value(DEFAULT_CHUNKER).validator(validate_chunker))
|
||||||
|
.arg(Arg::from_usage("-c --compression [COMPRESSION] 'Set the compression method and level'")
|
||||||
|
.default_value(DEFAULT_COMPRESSION).validator(validate_compression))
|
||||||
.arg(Arg::from_usage("-e --encrypt 'Generate a keypair and enable encryption'"))
|
.arg(Arg::from_usage("-e --encrypt 'Generate a keypair and enable encryption'"))
|
||||||
.arg(Arg::from_usage("--hash [HASH] 'Set the hash method (default: blake2)'"))
|
.arg(Arg::from_usage("--hash [HASH] 'Set the hash method'")
|
||||||
.arg(Arg::from_usage("-r --remote <REMOTE> 'Set the path to the mounted remote storage'"))
|
.default_value(DEFAULT_HASH).validator(validate_hash))
|
||||||
.arg(Arg::from_usage("[REPO] 'The path for the new repository'")))
|
.arg(Arg::from_usage("-r --remote <REMOTE> 'Set the path to the mounted remote storage'")
|
||||||
|
.validator(validate_existing_path))
|
||||||
|
.arg(Arg::from_usage("[REPO] 'The path for the new repository'")
|
||||||
|
.default_value("").validator(|val| validate_repo_path(val, false, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("backup").about("Create a new backup")
|
.subcommand(SubCommand::with_name("backup").about("Create a new backup")
|
||||||
.arg(Arg::from_usage("--full 'Create a full backup without using a reference'"))
|
.arg(Arg::from_usage("--full 'Create a full backup without using a reference'"))
|
||||||
.arg(Arg::from_usage("[reference] --ref [REF] 'Base the new backup on this reference'").conflicts_with("full"))
|
.arg(Arg::from_usage("[reference] --ref [REF] 'Base the new backup on this reference'")
|
||||||
|
.conflicts_with("full"))
|
||||||
.arg(Arg::from_usage("[cross_device] -x --xdev 'Allow to cross filesystem boundaries'"))
|
.arg(Arg::from_usage("[cross_device] -x --xdev 'Allow to cross filesystem boundaries'"))
|
||||||
.arg(Arg::from_usage("-e --exclude [PATTERN]... 'Exclude this path or file pattern'"))
|
.arg(Arg::from_usage("-e --exclude [PATTERN]... 'Exclude this path or file pattern'"))
|
||||||
.arg(Arg::from_usage("[excludes_from] --excludes-from [FILE] 'Read the list of excludes from this file'"))
|
.arg(Arg::from_usage("[excludes_from] --excludes-from [FILE] 'Read the list of excludes from this file'"))
|
||||||
.arg(Arg::from_usage("[no_default_excludes] --no-default-excludes 'Do not load the default excludes file'"))
|
.arg(Arg::from_usage("[no_default_excludes] --no-default-excludes 'Do not load the default excludes file'"))
|
||||||
.arg(Arg::from_usage("--tar 'Read the source data from a tar file'").conflicts_with_all(&["reference", "exclude", "excludes_from"]))
|
.arg(Arg::from_usage("--tar 'Read the source data from a tar file'")
|
||||||
.arg(Arg::from_usage("<SRC> 'Source path to backup'"))
|
.conflicts_with_all(&["reference", "exclude", "excludes_from"]))
|
||||||
.arg(Arg::from_usage("<BACKUP> 'Backup path, [repository]::backup'")))
|
.arg(Arg::from_usage("<SRC> 'Source path to backup'")
|
||||||
|
.validator(validate_existing_path))
|
||||||
|
.arg(Arg::from_usage("<BACKUP> 'Backup path, [repository]::backup'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(true), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("restore").about("Restore a backup or subtree")
|
.subcommand(SubCommand::with_name("restore").about("Restore a backup or subtree")
|
||||||
.arg(Arg::from_usage("--tar 'Restore in form of a tar file'"))
|
.arg(Arg::from_usage("--tar 'Restore in form of a tar file'"))
|
||||||
.arg(Arg::from_usage("<BACKUP> 'The backup/subtree path, [repository]::backup[::subtree]'"))
|
.arg(Arg::from_usage("<BACKUP> 'The backup/subtree path, [repository]::backup[::subtree]'")
|
||||||
.arg(Arg::from_usage("<DST> 'Destination path for backup'")))
|
.validator(|val| validate_repo_path(val, true, Some(true), None)))
|
||||||
|
.arg(Arg::from_usage("<DST> 'Destination path for backup'")
|
||||||
|
.validator(validate_existing_path)))
|
||||||
.subcommand(SubCommand::with_name("remove").aliases(&["rm", "delete", "del"]).about("Remove a backup or a subtree")
|
.subcommand(SubCommand::with_name("remove").aliases(&["rm", "delete", "del"]).about("Remove a backup or a subtree")
|
||||||
.arg(Arg::from_usage("<BACKUP> 'The backup/subtree path, [repository]::backup[::subtree]'")))
|
.arg(Arg::from_usage("<BACKUP> 'The backup/subtree path, [repository]::backup[::subtree]'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(true), None))))
|
||||||
.subcommand(SubCommand::with_name("prune").about("Remove backups based on age")
|
.subcommand(SubCommand::with_name("prune").about("Remove backups based on age")
|
||||||
.arg(Arg::from_usage("-p --prefix [PREFIX] 'Only consider backups starting with this prefix'"))
|
.arg(Arg::from_usage("-p --prefix [PREFIX] 'Only consider backups starting with this prefix'"))
|
||||||
.arg(Arg::from_usage("-d --daily [NUM] 'Keep this number of daily backups'"))
|
.arg(Arg::from_usage("-d --daily [NUM] 'Keep this number of daily backups'")
|
||||||
.arg(Arg::from_usage("-w --weekly [NUM] 'Keep this number of weekly backups'"))
|
.default_value("0").validator(validate_num))
|
||||||
.arg(Arg::from_usage("-m --monthly [NUM] 'Keep this number of monthly backups'"))
|
.arg(Arg::from_usage("-w --weekly [NUM] 'Keep this number of weekly backups'")
|
||||||
.arg(Arg::from_usage("-y --yearly [NUM] 'Keep this number of yearly backups'"))
|
.default_value("0").validator(validate_num))
|
||||||
|
.arg(Arg::from_usage("-m --monthly [NUM] 'Keep this number of monthly backups'")
|
||||||
|
.default_value("0").validator(validate_num))
|
||||||
|
.arg(Arg::from_usage("-y --yearly [NUM] 'Keep this number of yearly backups'")
|
||||||
|
.default_value("0").validator(validate_num))
|
||||||
.arg(Arg::from_usage("-f --force 'Actually run the prune instead of simulating it'"))
|
.arg(Arg::from_usage("-f --force 'Actually run the prune instead of simulating it'"))
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'")))
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("vacuum").about("Reclaim space by rewriting bundles")
|
.subcommand(SubCommand::with_name("vacuum").about("Reclaim space by rewriting bundles")
|
||||||
.arg(Arg::from_usage("-r --ratio [NUM] 'Ratio in % of unused space in a bundle to rewrite that bundle'"))
|
.arg(Arg::from_usage("-r --ratio [NUM] 'Ratio in % of unused space in a bundle to rewrite that bundle'")
|
||||||
|
.default_value(DEFAULT_VACUUM_RATIO_STR).validator(validate_num))
|
||||||
.arg(Arg::from_usage("-f --force 'Actually run the vacuum instead of simulating it'"))
|
.arg(Arg::from_usage("-f --force 'Actually run the vacuum instead of simulating it'"))
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'")))
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("check").about("Check the repository, a backup or a backup subtree")
|
.subcommand(SubCommand::with_name("check").about("Check the repository, a backup or a backup subtree")
|
||||||
.arg(Arg::from_usage("--full 'Also check file contents (slow)'"))
|
.arg(Arg::from_usage("--full 'Also check file contents (slow)'"))
|
||||||
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'")))
|
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, None, None))))
|
||||||
.subcommand(SubCommand::with_name("list").alias("ls").about("List backups or backup contents")
|
.subcommand(SubCommand::with_name("list").alias("ls").about("List backups or backup contents")
|
||||||
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'")))
|
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, None, None))))
|
||||||
.subcommand(SubCommand::with_name("mount").about("Mount the repository, a backup or a subtree")
|
.subcommand(SubCommand::with_name("mount").about("Mount the repository, a backup or a subtree")
|
||||||
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'"))
|
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'")
|
||||||
.arg(Arg::from_usage("<MOUNTPOINT> 'Existing mount point'")))
|
.validator(|val| validate_repo_path(val, true, None, None)))
|
||||||
|
.arg(Arg::from_usage("<MOUNTPOINT> 'Existing mount point'")
|
||||||
|
.validator(validate_existing_path)))
|
||||||
.subcommand(SubCommand::with_name("bundlelist").about("List bundles in a repository")
|
.subcommand(SubCommand::with_name("bundlelist").about("List bundles in a repository")
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'")))
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("bundleinfo").about("Display information on a bundle")
|
.subcommand(SubCommand::with_name("bundleinfo").about("Display information on a bundle")
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'"))
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false))))
|
||||||
.arg(Arg::from_usage("<BUNDLE> 'Id of the bundle'")))
|
.arg(Arg::from_usage("<BUNDLE> 'Id of the bundle'")))
|
||||||
.subcommand(SubCommand::with_name("import").about("Reconstruct a repository from the remote storage")
|
.subcommand(SubCommand::with_name("import").about("Reconstruct a repository from the remote storage")
|
||||||
.arg(Arg::from_usage("-k --key [FILE]... 'Key file needed to read the bundles'"))
|
.arg(Arg::from_usage("-k --key [FILE]... 'Key file needed to read the bundles'"))
|
||||||
.arg(Arg::from_usage("<REMOTE> 'Remote repository path'"))
|
.arg(Arg::from_usage("<REMOTE> 'Remote repository path'")
|
||||||
.arg(Arg::from_usage("[REPO] 'The path for the new repository'")))
|
.validator(validate_existing_path))
|
||||||
|
.arg(Arg::from_usage("[REPO] 'The path for the new repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, false, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("info").about("Display information on a repository, a backup or a subtree")
|
.subcommand(SubCommand::with_name("info").about("Display information on a repository, a backup or a subtree")
|
||||||
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'")))
|
.arg(Arg::from_usage("[PATH] 'Path of the repository/backup/subtree, [repository][::backup[::subtree]]'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, None, None))))
|
||||||
.subcommand(SubCommand::with_name("analyze").about("Analyze the used and reclaimable space of bundles")
|
.subcommand(SubCommand::with_name("analyze").about("Analyze the used and reclaimable space of bundles")
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'")))
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("versions").about("Find different versions of a file in all backups")
|
.subcommand(SubCommand::with_name("versions").about("Find different versions of a file in all backups")
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'"))
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false))))
|
||||||
.arg(Arg::from_usage("<PATH> 'Path of the file'")))
|
.arg(Arg::from_usage("<PATH> 'Path of the file'")))
|
||||||
.subcommand(SubCommand::with_name("diff").about("Display differences between two backup versions")
|
.subcommand(SubCommand::with_name("diff").about("Display differences between two backup versions")
|
||||||
.arg(Arg::from_usage("<OLD> 'Old version, [repository]::backup[::subpath]'"))
|
.arg(Arg::from_usage("<OLD> 'Old version, [repository]::backup[::subpath]'")
|
||||||
.arg(Arg::from_usage("<NEW> 'New version, [repository]::backup[::subpath]'")))
|
.validator(|val| validate_repo_path(val, true, Some(true), None)))
|
||||||
|
.arg(Arg::from_usage("<NEW> 'New version, [repository]::backup[::subpath]'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(true), None))))
|
||||||
.subcommand(SubCommand::with_name("config").about("Display or change the configuration")
|
.subcommand(SubCommand::with_name("config").about("Display or change the configuration")
|
||||||
.arg(Arg::from_usage("[bundle_size] --bundle-size [SIZE] 'Set the target bundle size in MiB (default: 25)'"))
|
.arg(Arg::from_usage("[bundle_size] --bundle-size [SIZE] 'Set the target bundle size in MiB'")
|
||||||
.arg(Arg::from_usage("--chunker [CHUNKER] 'Set the chunker algorithm and target chunk size (default: fastcdc/16)'"))
|
.validator(validate_num))
|
||||||
.arg(Arg::from_usage("-c --compression [COMPRESSION] 'Set the compression method and level (default: brotli/3)'"))
|
.arg(Arg::from_usage("--chunker [CHUNKER] 'Set the chunker algorithm and target chunk size'")
|
||||||
.arg(Arg::from_usage("-e --encryption [PUBLIC_KEY] 'The public key to use for encryption'"))
|
.validator(validate_chunker))
|
||||||
.arg(Arg::from_usage("--hash [HASH] 'Set the hash method (default: blake2)'"))
|
.arg(Arg::from_usage("-c --compression [COMPRESSION] 'Set the compression method and level'")
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'")))
|
.validator(validate_compression))
|
||||||
|
.arg(Arg::from_usage("-e --encryption [PUBLIC_KEY] 'The public key to use for encryption'")
|
||||||
|
.validator(validate_public_key))
|
||||||
|
.arg(Arg::from_usage("--hash [HASH] 'Set the hash method'")
|
||||||
|
.validator(validate_hash))
|
||||||
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("genkey").about("Generate a new key pair")
|
.subcommand(SubCommand::with_name("genkey").about("Generate a new key pair")
|
||||||
.arg(Arg::from_usage("[FILE] 'Destination file for the keypair'")))
|
.arg(Arg::from_usage("[FILE] 'Destination file for the keypair'")))
|
||||||
.subcommand(SubCommand::with_name("addkey").about("Add a key pair to the repository")
|
.subcommand(SubCommand::with_name("addkey").about("Add a key pair to the repository")
|
||||||
.arg(Arg::from_usage("-g --generate 'Generate a new key pair'").conflicts_with("FILE"))
|
.arg(Arg::from_usage("-g --generate 'Generate a new key pair'")
|
||||||
|
.conflicts_with("FILE"))
|
||||||
.arg(Arg::from_usage("[set_default] --default -d 'Set the key pair as default'"))
|
.arg(Arg::from_usage("[set_default] --default -d 'Set the key pair as default'"))
|
||||||
.arg(Arg::from_usage("[FILE] 'File containing the keypair'").conflicts_with("generate"))
|
.arg(Arg::from_usage("[FILE] 'File containing the keypair'")
|
||||||
.arg(Arg::from_usage("[REPO] 'Path of the repository'")))
|
.required_unless("generate").validator(validate_existing_path))
|
||||||
|
.arg(Arg::from_usage("[REPO] 'Path of the repository'")
|
||||||
|
.validator(|val| validate_repo_path(val, true, Some(false), Some(false)))))
|
||||||
.subcommand(SubCommand::with_name("algotest").about("Test a specific algorithm combination")
|
.subcommand(SubCommand::with_name("algotest").about("Test a specific algorithm combination")
|
||||||
.arg(Arg::from_usage("[bundle_size] --bundle-size [SIZE] 'Set the target bundle size in MiB (default: 25)'"))
|
.arg(Arg::from_usage("[bundle_size] --bundle-size [SIZE] 'Set the target bundle size in MiB'")
|
||||||
.arg(Arg::from_usage("--chunker [CHUNKER] 'Set the chunker algorithm and target chunk size (default: fastcdc/16)'"))
|
.default_value(DEFAULT_BUNDLE_SIZE_STR).validator(validate_num))
|
||||||
.arg(Arg::from_usage("-c --compression [COMPRESSION] 'Set the compression method and level (default: brotli/3)'"))
|
.arg(Arg::from_usage("--chunker [CHUNKER] 'Set the chunker algorithm and target chunk size'")
|
||||||
.arg(Arg::from_usage("-e --encryption 'Generate a keypair and enable encryption'"))
|
.default_value(DEFAULT_CHUNKER).validator(validate_chunker))
|
||||||
.arg(Arg::from_usage("--hash [HASH] 'Set the hash method (default: blake2)'"))
|
.arg(Arg::from_usage("-c --compression [COMPRESSION] 'Set the compression method and level'")
|
||||||
.arg(Arg::from_usage("<FILE> 'File with test data'"))).get_matches();
|
.default_value(DEFAULT_COMPRESSION).validator(validate_compression))
|
||||||
|
.arg(Arg::from_usage("-e --encrypt 'Generate a keypair and enable encryption'"))
|
||||||
|
.arg(Arg::from_usage("--hash [HASH] 'Set the hash method'")
|
||||||
|
.default_value(DEFAULT_HASH).validator(validate_hash))
|
||||||
|
.arg(Arg::from_usage("<FILE> 'File with test data'")
|
||||||
|
.validator(validate_existing_path))).get_matches();
|
||||||
if let Some(args) = args.subcommand_matches("init") {
|
if let Some(args) = args.subcommand_matches("init") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), false, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Init {
|
return Ok(Arguments::Init {
|
||||||
bundle_size: (try!(parse_num(args.value_of("bundle_size").unwrap_or(&DEFAULT_BUNDLE_SIZE.to_string()), "Bundle size")) * 1024 * 1024) as usize,
|
bundle_size: (parse_num(args.value_of("bundle_size").unwrap()).unwrap() * 1024 * 1024) as usize,
|
||||||
chunker: try!(parse_chunker(args.value_of("chunker").unwrap_or(DEFAULT_CHUNKER))),
|
chunker: parse_chunker(args.value_of("chunker").unwrap()).unwrap(),
|
||||||
compression: try!(parse_compression(args.value_of("compression").unwrap_or(DEFAULT_COMPRESSION))),
|
compression: parse_compression(args.value_of("compression").unwrap()).unwrap(),
|
||||||
encryption: args.is_present("encrypt"),
|
encryption: args.is_present("encrypt"),
|
||||||
hash: try!(parse_hash(args.value_of("hash").unwrap_or(DEFAULT_HASH))),
|
hash: parse_hash(args.value_of("hash").unwrap()).unwrap(),
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
remote_path: args.value_of("remote").unwrap().to_string()
|
remote_path: args.value_of("remote").unwrap().to_string()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("backup") {
|
if let Some(args) = args.subcommand_matches("backup") {
|
||||||
let (repository, backup, _inode) = try!(parse_repo_path(args.value_of("BACKUP").unwrap(), Some(true), Some(false)));
|
let (repository, backup, _inode) = parse_repo_path(args.value_of("BACKUP").unwrap(), true, Some(true), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Backup {
|
return Ok(Arguments::Backup {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
backup_name: backup.unwrap().to_string(),
|
backup_name: backup.unwrap().to_string(),
|
||||||
@ -349,7 +437,7 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("restore") {
|
if let Some(args) = args.subcommand_matches("restore") {
|
||||||
let (repository, backup, inode) = try!(parse_repo_path(args.value_of("BACKUP").unwrap(), Some(true), None));
|
let (repository, backup, inode) = parse_repo_path(args.value_of("BACKUP").unwrap(), true, Some(true), None).unwrap();
|
||||||
return Ok(Arguments::Restore {
|
return Ok(Arguments::Restore {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
backup_name: backup.unwrap().to_string(),
|
backup_name: backup.unwrap().to_string(),
|
||||||
@ -359,7 +447,7 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("remove") {
|
if let Some(args) = args.subcommand_matches("remove") {
|
||||||
let (repository, backup, inode) = try!(parse_repo_path(args.value_of("BACKUP").unwrap(), Some(true), None));
|
let (repository, backup, inode) = parse_repo_path(args.value_of("BACKUP").unwrap(), true, Some(true), None).unwrap();
|
||||||
return Ok(Arguments::Remove {
|
return Ok(Arguments::Remove {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
backup_name: backup.unwrap().to_string(),
|
backup_name: backup.unwrap().to_string(),
|
||||||
@ -367,39 +455,39 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("prune") {
|
if let Some(args) = args.subcommand_matches("prune") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Prune {
|
return Ok(Arguments::Prune {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
prefix: args.value_of("prefix").unwrap_or("").to_string(),
|
prefix: args.value_of("prefix").unwrap_or("").to_string(),
|
||||||
force: args.is_present("force"),
|
force: args.is_present("force"),
|
||||||
daily: match args.value_of("daily") {
|
daily: match args.value_of("daily") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some(try!(parse_num(v, "daily backups")) as usize)
|
Some(v) => Some(parse_num(v).unwrap() as usize)
|
||||||
},
|
},
|
||||||
weekly: match args.value_of("weekly") {
|
weekly: match args.value_of("weekly") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some(try!(parse_num(v, "weekly backups")) as usize)
|
Some(v) => Some(parse_num(v).unwrap() as usize)
|
||||||
},
|
},
|
||||||
monthly: match args.value_of("monthly") {
|
monthly: match args.value_of("monthly") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some(try!(parse_num(v, "monthly backups")) as usize)
|
Some(v) => Some(parse_num(v).unwrap() as usize)
|
||||||
},
|
},
|
||||||
yearly: match args.value_of("yearly") {
|
yearly: match args.value_of("yearly") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some(try!(parse_num(v, "yearly backups")) as usize)
|
Some(v) => Some(parse_num(v).unwrap() as usize)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("vacuum") {
|
if let Some(args) = args.subcommand_matches("vacuum") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Vacuum {
|
return Ok(Arguments::Vacuum {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
force: args.is_present("force"),
|
force: args.is_present("force"),
|
||||||
ratio: try!(parse_num(args.value_of("ratio").unwrap_or(&DEFAULT_VACUUM_RATIO.to_string()), "ratio")) as f32 / 100.0
|
ratio: parse_num(args.value_of("ratio").unwrap()).unwrap() as f32 / 100.0
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("check") {
|
if let Some(args) = args.subcommand_matches("check") {
|
||||||
let (repository, backup, inode) = try!(parse_repo_path(args.value_of("PATH").unwrap_or(""), None, None));
|
let (repository, backup, inode) = parse_repo_path(args.value_of("PATH").unwrap_or(""), true, None, None).unwrap();
|
||||||
return Ok(Arguments::Check {
|
return Ok(Arguments::Check {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
backup_name: backup.map(|v| v.to_string()),
|
backup_name: backup.map(|v| v.to_string()),
|
||||||
@ -408,7 +496,7 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("list") {
|
if let Some(args) = args.subcommand_matches("list") {
|
||||||
let (repository, backup, inode) = try!(parse_repo_path(args.value_of("PATH").unwrap_or(""), None, None));
|
let (repository, backup, inode) = parse_repo_path(args.value_of("PATH").unwrap_or(""), true, None, None).unwrap();
|
||||||
return Ok(Arguments::List {
|
return Ok(Arguments::List {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
backup_name: backup.map(|v| v.to_string()),
|
backup_name: backup.map(|v| v.to_string()),
|
||||||
@ -416,20 +504,20 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("bundlelist") {
|
if let Some(args) = args.subcommand_matches("bundlelist") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::BundleList {
|
return Ok(Arguments::BundleList {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("bundleinfo") {
|
if let Some(args) = args.subcommand_matches("bundleinfo") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::BundleInfo {
|
return Ok(Arguments::BundleInfo {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
bundle_id: try!(parse_bundle_id(args.value_of("BUNDLE").unwrap()))
|
bundle_id: try!(parse_bundle_id(args.value_of("BUNDLE").unwrap()))
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("info") {
|
if let Some(args) = args.subcommand_matches("info") {
|
||||||
let (repository, backup, inode) = try!(parse_repo_path(args.value_of("PATH").unwrap_or(""), None, None));
|
let (repository, backup, inode) = parse_repo_path(args.value_of("PATH").unwrap_or(""), true, None, None).unwrap();
|
||||||
return Ok(Arguments::Info {
|
return Ok(Arguments::Info {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
backup_name: backup.map(|v| v.to_string()),
|
backup_name: backup.map(|v| v.to_string()),
|
||||||
@ -437,7 +525,7 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("mount") {
|
if let Some(args) = args.subcommand_matches("mount") {
|
||||||
let (repository, backup, inode) = try!(parse_repo_path(args.value_of("PATH").unwrap_or(""), None, None));
|
let (repository, backup, inode) = parse_repo_path(args.value_of("PATH").unwrap_or(""), true, None, None).unwrap();
|
||||||
return Ok(Arguments::Mount {
|
return Ok(Arguments::Mount {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
backup_name: backup.map(|v| v.to_string()),
|
backup_name: backup.map(|v| v.to_string()),
|
||||||
@ -446,15 +534,15 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("versions") {
|
if let Some(args) = args.subcommand_matches("versions") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Versions {
|
return Ok(Arguments::Versions {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
path: args.value_of("PATH").unwrap().to_string()
|
path: args.value_of("PATH").unwrap().to_string()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("diff") {
|
if let Some(args) = args.subcommand_matches("diff") {
|
||||||
let (repository_old, backup_old, inode_old) = try!(parse_repo_path(args.value_of("OLD").unwrap(), Some(true), None));
|
let (repository_old, backup_old, inode_old) = parse_repo_path(args.value_of("OLD").unwrap(), true, Some(true), None).unwrap();
|
||||||
let (repository_new, backup_new, inode_new) = try!(parse_repo_path(args.value_of("NEW").unwrap(), Some(true), None));
|
let (repository_new, backup_new, inode_new) = parse_repo_path(args.value_of("NEW").unwrap(), true, Some(true), None).unwrap();
|
||||||
return Ok(Arguments::Diff {
|
return Ok(Arguments::Diff {
|
||||||
repo_path_old: repository_old.to_string(),
|
repo_path_old: repository_old.to_string(),
|
||||||
backup_name_old: backup_old.unwrap().to_string(),
|
backup_name_old: backup_old.unwrap().to_string(),
|
||||||
@ -465,13 +553,13 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("analyze") {
|
if let Some(args) = args.subcommand_matches("analyze") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Analyze {
|
return Ok(Arguments::Analyze {
|
||||||
repo_path: repository.to_string()
|
repo_path: repository.to_string()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("import") {
|
if let Some(args) = args.subcommand_matches("import") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), false, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Import {
|
return Ok(Arguments::Import {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
remote_path: args.value_of("REMOTE").unwrap().to_string(),
|
remote_path: args.value_of("REMOTE").unwrap().to_string(),
|
||||||
@ -479,28 +567,27 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("config") {
|
if let Some(args) = args.subcommand_matches("config") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
return Ok(Arguments::Config {
|
return Ok(Arguments::Config {
|
||||||
bundle_size: match args.value_of("bundle_size") {
|
bundle_size: match args.value_of("bundle_size") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some((try!(parse_num(v, "Bundle size")) * 1024 * 1024) as usize)
|
Some(v) => Some((parse_num(v).unwrap() * 1024 * 1024) as usize)
|
||||||
},
|
},
|
||||||
chunker: match args.value_of("chunker") {
|
chunker: match args.value_of("chunker") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some(try!(parse_chunker(v)))
|
Some(v) => Some(parse_chunker(v).unwrap())
|
||||||
},
|
},
|
||||||
compression: match args.value_of("compression") {
|
compression: match args.value_of("compression") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some(try!(parse_compression(v)))
|
Some(v) => Some(parse_compression(v).unwrap())
|
||||||
},
|
},
|
||||||
encryption: match args.value_of("encryption") {
|
encryption: match args.value_of("encryption") {
|
||||||
None => None,
|
None => None,
|
||||||
Some("none") => Some(None),
|
Some(v) => Some(parse_public_key(v).unwrap())
|
||||||
Some(v) => Some(Some(try!(parse_public_key(v))))
|
|
||||||
},
|
},
|
||||||
hash: match args.value_of("hash") {
|
hash: match args.value_of("hash") {
|
||||||
None => None,
|
None => None,
|
||||||
Some(v) => Some(try!(parse_hash(v)))
|
Some(v) => Some(parse_hash(v).unwrap())
|
||||||
},
|
},
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
})
|
})
|
||||||
@ -511,16 +598,7 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("addkey") {
|
if let Some(args) = args.subcommand_matches("addkey") {
|
||||||
let (repository, _backup, _inode) = try!(parse_repo_path(args.value_of("REPO").unwrap_or(""), Some(false), Some(false)));
|
let (repository, _backup, _inode) = parse_repo_path(args.value_of("REPO").unwrap_or(""), true, Some(false), Some(false)).unwrap();
|
||||||
let generate = args.is_present("generate");
|
|
||||||
if !generate && !args.is_present("FILE") {
|
|
||||||
println!("Without --generate, a file containing the key pair must be given");
|
|
||||||
return Err(ErrorCode::InvalidArgs)
|
|
||||||
}
|
|
||||||
if generate && args.is_present("FILE") {
|
|
||||||
println!("With --generate, no file may be given");
|
|
||||||
return Err(ErrorCode::InvalidArgs)
|
|
||||||
}
|
|
||||||
return Ok(Arguments::AddKey {
|
return Ok(Arguments::AddKey {
|
||||||
repo_path: repository.to_string(),
|
repo_path: repository.to_string(),
|
||||||
set_default: args.is_present("set_default"),
|
set_default: args.is_present("set_default"),
|
||||||
@ -529,11 +607,11 @@ pub fn parse() -> Result<Arguments, ErrorCode> {
|
|||||||
}
|
}
|
||||||
if let Some(args) = args.subcommand_matches("algotest") {
|
if let Some(args) = args.subcommand_matches("algotest") {
|
||||||
return Ok(Arguments::AlgoTest {
|
return Ok(Arguments::AlgoTest {
|
||||||
bundle_size: (try!(parse_num(args.value_of("bundle_size").unwrap_or(&DEFAULT_BUNDLE_SIZE.to_string()), "Bundle size")) * 1024 * 1024) as usize,
|
bundle_size: (parse_num(args.value_of("bundle_size").unwrap()).unwrap() * 1024 * 1024) as usize,
|
||||||
chunker: try!(parse_chunker(args.value_of("chunker").unwrap_or(DEFAULT_CHUNKER))),
|
chunker: parse_chunker(args.value_of("chunker").unwrap()).unwrap(),
|
||||||
compression: try!(parse_compression(args.value_of("compression").unwrap_or(DEFAULT_COMPRESSION))),
|
compression: parse_compression(args.value_of("compression").unwrap()).unwrap(),
|
||||||
encrypt: args.is_present("encrypt"),
|
encrypt: args.is_present("encrypt"),
|
||||||
hash: try!(parse_hash(args.value_of("hash").unwrap_or(DEFAULT_HASH))),
|
hash: parse_hash(args.value_of("hash").unwrap()).unwrap(),
|
||||||
file: args.value_of("FILE").unwrap().to_string(),
|
file: args.value_of("FILE").unwrap().to_string(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -67,8 +67,8 @@ impl ErrorCode {
|
|||||||
pub const DEFAULT_CHUNKER: &'static str = "fastcdc/16";
|
pub const DEFAULT_CHUNKER: &'static str = "fastcdc/16";
|
||||||
pub const DEFAULT_HASH: &'static str = "blake2";
|
pub const DEFAULT_HASH: &'static str = "blake2";
|
||||||
pub const DEFAULT_COMPRESSION: &'static str = "brotli/3";
|
pub const DEFAULT_COMPRESSION: &'static str = "brotli/3";
|
||||||
pub const DEFAULT_BUNDLE_SIZE: usize = 25;
|
pub const DEFAULT_BUNDLE_SIZE_STR: &'static str = "25";
|
||||||
pub const DEFAULT_VACUUM_RATIO: usize = 0;
|
pub const DEFAULT_VACUUM_RATIO_STR: &'static str = "0";
|
||||||
lazy_static! {
|
lazy_static! {
|
||||||
pub static ref DEFAULT_REPOSITORY: String = {
|
pub static ref DEFAULT_REPOSITORY: String = {
|
||||||
env::home_dir().unwrap().join(".zvault").to_string_lossy().to_string()
|
env::home_dir().unwrap().join(".zvault").to_string_lossy().to_string()
|
||||||
@ -271,11 +271,13 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
}, remote_path), "create repository", ErrorCode::CreateRepository);
|
}, remote_path), "create repository", ErrorCode::CreateRepository);
|
||||||
if encryption {
|
if encryption {
|
||||||
let (public, secret) = gen_keypair();
|
let (public, secret) = gen_keypair();
|
||||||
|
info!("Created the following key pair");
|
||||||
println!("public: {}", to_hex(&public[..]));
|
println!("public: {}", to_hex(&public[..]));
|
||||||
println!("secret: {}", to_hex(&secret[..]));
|
println!("secret: {}", to_hex(&secret[..]));
|
||||||
repo.set_encryption(Some(&public));
|
repo.set_encryption(Some(&public));
|
||||||
checked!(repo.register_key(public, secret), "add key", ErrorCode::AddKey);
|
checked!(repo.register_key(public, secret), "add key", ErrorCode::AddKey);
|
||||||
checked!(repo.save_config(), "save config", ErrorCode::SaveConfig);
|
checked!(repo.save_config(), "save config", ErrorCode::SaveConfig);
|
||||||
|
warn!("Please store this key pair in a secure location before using the repository");
|
||||||
println!();
|
println!();
|
||||||
}
|
}
|
||||||
print_config(&repo.config);
|
print_config(&repo.config);
|
||||||
@ -337,6 +339,7 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
} else {
|
} else {
|
||||||
repo.create_backup_recursively(&src_path, reference_backup.as_ref(), &options)
|
repo.create_backup_recursively(&src_path, reference_backup.as_ref(), &options)
|
||||||
};
|
};
|
||||||
|
info!("Backup finished");
|
||||||
let backup = match result {
|
let backup = match result {
|
||||||
Ok(backup) => backup,
|
Ok(backup) => backup,
|
||||||
Err(RepositoryError::Backup(BackupError::FailedPaths(backup, _failed_paths))) => {
|
Err(RepositoryError::Backup(BackupError::FailedPaths(backup, _failed_paths))) => {
|
||||||
@ -364,6 +367,7 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
} else {
|
} else {
|
||||||
checked!(repo.restore_inode_tree(inode, &dst_path), "restore backup", ErrorCode::RestoreRun);
|
checked!(repo.restore_inode_tree(inode, &dst_path), "restore backup", ErrorCode::RestoreRun);
|
||||||
}
|
}
|
||||||
|
info!("Backup finished");
|
||||||
},
|
},
|
||||||
Arguments::Remove{repo_path, backup_name, inode} => {
|
Arguments::Remove{repo_path, backup_name, inode} => {
|
||||||
let mut repo = try!(open_repository(&repo_path));
|
let mut repo = try!(open_repository(&repo_path));
|
||||||
@ -468,6 +472,8 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
} else {
|
} else {
|
||||||
checked!(FuseFilesystem::from_repository(&mut repo), "create fuse filesystem", ErrorCode::FuseMount)
|
checked!(FuseFilesystem::from_repository(&mut repo), "create fuse filesystem", ErrorCode::FuseMount)
|
||||||
};
|
};
|
||||||
|
info!("Mounting the filesystem...");
|
||||||
|
info!("Please unmount the filesystem via 'fusermount -u {}' when done.", mount_point);
|
||||||
checked!(fs.mount(&mount_point), "mount filesystem", ErrorCode::FuseMount);
|
checked!(fs.mount(&mount_point), "mount filesystem", ErrorCode::FuseMount);
|
||||||
},
|
},
|
||||||
Arguments::Analyze{repo_path} => {
|
Arguments::Analyze{repo_path} => {
|
||||||
@ -491,12 +497,18 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
},
|
},
|
||||||
Arguments::Import{repo_path, remote_path, key_files} => {
|
Arguments::Import{repo_path, remote_path, key_files} => {
|
||||||
checked!(Repository::import(repo_path, remote_path, key_files), "import repository", ErrorCode::ImportRun);
|
checked!(Repository::import(repo_path, remote_path, key_files), "import repository", ErrorCode::ImportRun);
|
||||||
|
info!("Import finished");
|
||||||
},
|
},
|
||||||
Arguments::Versions{repo_path, path} => {
|
Arguments::Versions{repo_path, path} => {
|
||||||
let mut repo = try!(open_repository(&repo_path));
|
let mut repo = try!(open_repository(&repo_path));
|
||||||
|
let mut found = false;
|
||||||
for (name, mut inode) in checked!(repo.find_versions(&path), "find versions", ErrorCode::VersionsRun) {
|
for (name, mut inode) in checked!(repo.find_versions(&path), "find versions", ErrorCode::VersionsRun) {
|
||||||
inode.name = format!("{}::{}", name, &path);
|
inode.name = format!("{}::{}", name, &path);
|
||||||
println!("{}", format_inode_one_line(&inode));
|
println!("{}", format_inode_one_line(&inode));
|
||||||
|
found = true;
|
||||||
|
}
|
||||||
|
if !found {
|
||||||
|
info!("No versions of that file were found.");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
Arguments::Diff{repo_path_old, backup_name_old, inode_old, repo_path_new, backup_name_new, inode_new} => {
|
Arguments::Diff{repo_path_old, backup_name_old, inode_old, repo_path_new, backup_name_new, inode_new} => {
|
||||||
@ -510,38 +522,52 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
let inode1 = checked!(repo.get_backup_inode(&backup_old, inode_old.unwrap_or_else(|| "/".to_string())), "load subpath inode", ErrorCode::LoadInode);
|
let inode1 = checked!(repo.get_backup_inode(&backup_old, inode_old.unwrap_or_else(|| "/".to_string())), "load subpath inode", ErrorCode::LoadInode);
|
||||||
let inode2 = checked!(repo.get_backup_inode(&backup_new, inode_new.unwrap_or_else(|| "/".to_string())), "load subpath inode", ErrorCode::LoadInode);
|
let inode2 = checked!(repo.get_backup_inode(&backup_new, inode_new.unwrap_or_else(|| "/".to_string())), "load subpath inode", ErrorCode::LoadInode);
|
||||||
let diffs = checked!(repo.find_differences(&inode1, &inode2), "find differences", ErrorCode::DiffRun);
|
let diffs = checked!(repo.find_differences(&inode1, &inode2), "find differences", ErrorCode::DiffRun);
|
||||||
for diff in diffs {
|
for diff in &diffs {
|
||||||
println!("{} {:?}", match diff.0 {
|
println!("{} {:?}", match diff.0 {
|
||||||
DiffType::Add => "add",
|
DiffType::Add => "add",
|
||||||
DiffType::Mod => "mod",
|
DiffType::Mod => "mod",
|
||||||
DiffType::Del => "del"
|
DiffType::Del => "del"
|
||||||
}, diff.1);
|
}, diff.1);
|
||||||
}
|
}
|
||||||
|
if diffs.is_empty() {
|
||||||
|
info!("No differences found");
|
||||||
|
}
|
||||||
},
|
},
|
||||||
Arguments::Config{repo_path, bundle_size, chunker, compression, encryption, hash} => {
|
Arguments::Config{repo_path, bundle_size, chunker, compression, encryption, hash} => {
|
||||||
let mut repo = try!(open_repository(&repo_path));
|
let mut repo = try!(open_repository(&repo_path));
|
||||||
|
let mut changed = false;
|
||||||
if let Some(bundle_size) = bundle_size {
|
if let Some(bundle_size) = bundle_size {
|
||||||
repo.config.bundle_size = bundle_size
|
repo.config.bundle_size = bundle_size;
|
||||||
|
changed = true;
|
||||||
}
|
}
|
||||||
if let Some(chunker) = chunker {
|
if let Some(chunker) = chunker {
|
||||||
warn!("Changing the chunker makes it impossible to use existing data for deduplication");
|
warn!("Changing the chunker makes it impossible to use existing data for deduplication");
|
||||||
repo.config.chunker = chunker
|
repo.config.chunker = chunker;
|
||||||
|
changed = true;
|
||||||
}
|
}
|
||||||
if let Some(compression) = compression {
|
if let Some(compression) = compression {
|
||||||
repo.config.compression = compression
|
repo.config.compression = compression;
|
||||||
|
changed = true;
|
||||||
}
|
}
|
||||||
if let Some(encryption) = encryption {
|
if let Some(encryption) = encryption {
|
||||||
repo.set_encryption(encryption.as_ref())
|
repo.set_encryption(encryption.as_ref());
|
||||||
|
changed = true;
|
||||||
}
|
}
|
||||||
if let Some(hash) = hash {
|
if let Some(hash) = hash {
|
||||||
warn!("Changing the hash makes it impossible to use existing data for deduplication");
|
warn!("Changing the hash makes it impossible to use existing data for deduplication");
|
||||||
repo.config.hash = hash
|
repo.config.hash = hash;
|
||||||
|
changed = true;
|
||||||
|
}
|
||||||
|
if changed {
|
||||||
|
checked!(repo.save_config(), "save config", ErrorCode::SaveConfig);
|
||||||
|
info!("The configuration has been updated.");
|
||||||
|
} else {
|
||||||
|
print_config(&repo.config);
|
||||||
}
|
}
|
||||||
checked!(repo.save_config(), "save config", ErrorCode::SaveConfig);
|
|
||||||
print_config(&repo.config);
|
|
||||||
},
|
},
|
||||||
Arguments::GenKey{file} => {
|
Arguments::GenKey{file} => {
|
||||||
let (public, secret) = gen_keypair();
|
let (public, secret) = gen_keypair();
|
||||||
|
info!("Created the following key pair");
|
||||||
println!("public: {}", to_hex(&public[..]));
|
println!("public: {}", to_hex(&public[..]));
|
||||||
println!("secret: {}", to_hex(&secret[..]));
|
println!("secret: {}", to_hex(&secret[..]));
|
||||||
if let Some(file) = file {
|
if let Some(file) = file {
|
||||||
@ -553,6 +579,7 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
let (public, secret) = if let Some(file) = file {
|
let (public, secret) = if let Some(file) = file {
|
||||||
checked!(Crypto::load_keypair_from_file(file), "load key pair", ErrorCode::LoadKey)
|
checked!(Crypto::load_keypair_from_file(file), "load key pair", ErrorCode::LoadKey)
|
||||||
} else {
|
} else {
|
||||||
|
info!("Created the following key pair");
|
||||||
let (public, secret) = gen_keypair();
|
let (public, secret) = gen_keypair();
|
||||||
println!("public: {}", to_hex(&public[..]));
|
println!("public: {}", to_hex(&public[..]));
|
||||||
println!("secret: {}", to_hex(&secret[..]));
|
println!("secret: {}", to_hex(&secret[..]));
|
||||||
@ -562,6 +589,7 @@ pub fn run() -> Result<(), ErrorCode> {
|
|||||||
if set_default {
|
if set_default {
|
||||||
repo.set_encryption(Some(&public));
|
repo.set_encryption(Some(&public));
|
||||||
checked!(repo.save_config(), "save config", ErrorCode::SaveConfig);
|
checked!(repo.save_config(), "save config", ErrorCode::SaveConfig);
|
||||||
|
warn!("Please store this key pair in a secure location before using the repository");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
Arguments::AlgoTest{bundle_size, chunker, compression, encrypt, hash, file} => {
|
Arguments::AlgoTest{bundle_size, chunker, compression, encrypt, hash, file} => {
|
||||||
|
@ -112,12 +112,13 @@ impl Repository {
|
|||||||
mark_needed(&backups, &mut keep, max, |d| (d.year(), d.month(), d.day()));
|
mark_needed(&backups, &mut keep, max, |d| (d.year(), d.month(), d.day()));
|
||||||
}
|
}
|
||||||
let mut remove = Vec::new();
|
let mut remove = Vec::new();
|
||||||
|
info!("Removing the following backups");
|
||||||
for (i, backup) in backups.into_iter().enumerate() {
|
for (i, backup) in backups.into_iter().enumerate() {
|
||||||
if !keep.get(i) {
|
if !keep.get(i) {
|
||||||
|
println!(" - {}", backup.0);
|
||||||
remove.push(backup.0);
|
remove.push(backup.0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
info!("Removing the following backups: {:?}", remove);
|
|
||||||
if force {
|
if force {
|
||||||
for name in remove {
|
for name in remove {
|
||||||
try!(self.delete_backup(&name));
|
try!(self.delete_backup(&name));
|
||||||
|
@ -160,7 +160,7 @@ pub struct Config {
|
|||||||
impl Default for Config {
|
impl Default for Config {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
Config {
|
Config {
|
||||||
compression: None,
|
compression: Some(Compression::from_string("brotli/3").unwrap()),
|
||||||
encryption: None,
|
encryption: None,
|
||||||
bundle_size: 25*1024*1024,
|
bundle_size: 25*1024*1024,
|
||||||
chunker: ChunkerType::from_string("fastcdc/16").unwrap(),
|
chunker: ChunkerType::from_string("fastcdc/16").unwrap(),
|
||||||
|
@ -140,11 +140,14 @@ impl Repository {
|
|||||||
try!(repo.crypto.lock().unwrap().register_keyfile(file));
|
try!(repo.crypto.lock().unwrap().register_keyfile(file));
|
||||||
}
|
}
|
||||||
repo = try!(Repository::open(path));
|
repo = try!(Repository::open(path));
|
||||||
let mut backups: Vec<Backup> = try!(repo.get_backups()).into_iter().map(|(_, v)| v).collect();
|
let mut backups: Vec<(String, Backup)> = try!(repo.get_backups()).into_iter().collect();
|
||||||
backups.sort_by_key(|b| b.date);
|
backups.sort_by_key(|&(_, ref b)| b.date);
|
||||||
if let Some(backup) = backups.pop() {
|
if let Some((name, backup)) = backups.pop() {
|
||||||
|
info!("Taking configuration from the last backup '{}'", name);
|
||||||
repo.config = backup.config;
|
repo.config = backup.config;
|
||||||
try!(repo.save_config())
|
try!(repo.save_config())
|
||||||
|
} else {
|
||||||
|
warn!("No backup found in the repository to take configuration from, please set the configuration manually.");
|
||||||
}
|
}
|
||||||
Ok(repo)
|
Ok(repo)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user