mirror of
git://git.proxmox.com/git/proxmox-backup.git
synced 2025-01-07 17:18:03 +03:00
new catar binary
currently used for debugging
This commit is contained in:
parent
e75eac73ca
commit
c60d34bdbf
3
debian/catar.bash-completion
vendored
Normal file
3
debian/catar.bash-completion
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
# catar bash completion
|
||||
|
||||
complete -C 'catar bashcomplete' catar
|
1
debian/install
vendored
1
debian/install
vendored
@ -1,6 +1,7 @@
|
||||
target/release/proxmox-backup-api /usr/sbin
|
||||
target/release/pbs /usr/sbin
|
||||
target/release/backup-client /usr/sbin
|
||||
target/release/catar /usr/sbin
|
||||
www/images/logo-128.png /usr/share/javascript/proxmox-backup/images/
|
||||
www/images/proxmox_logo.png /usr/share/javascript/proxmox-backup/images/
|
||||
www/proxmox-backup-gui.js /usr/share/javascript/proxmox-backup/js/
|
||||
|
1
debian/proxmox-backup.bash-completion
vendored
1
debian/proxmox-backup.bash-completion
vendored
@ -1,2 +1,3 @@
|
||||
debian/pbs.bash-completion pbs
|
||||
debian/backup-client.bash-completion backup-client
|
||||
debian/catar.bash-completion catar
|
||||
|
139
src/bin/catar.rs
Normal file
139
src/bin/catar.rs
Normal file
@ -0,0 +1,139 @@
|
||||
extern crate proxmox_backup;
|
||||
|
||||
use failure::*;
|
||||
|
||||
use proxmox_backup::tools;
|
||||
use proxmox_backup::cli::command::*;
|
||||
use proxmox_backup::api::schema::*;
|
||||
use proxmox_backup::api::router::*;
|
||||
|
||||
use serde_json::{Value};
|
||||
|
||||
use std::io::Read;
|
||||
use std::fs::File;
|
||||
use std::os::unix::ffi::OsStrExt;
|
||||
|
||||
use nix::sys::stat::SFlag;
|
||||
|
||||
use proxmox_backup::catar::format_definition::*;
|
||||
use proxmox_backup::tools::*;
|
||||
|
||||
fn required_string_param<'a>(param: &'a Value, name: &str) -> &'a str {
|
||||
param[name].as_str().expect(&format!("missing parameter '{}'", name))
|
||||
}
|
||||
|
||||
fn print_goodby_entries(buffer: &[u8]) -> Result<(), Error> {
|
||||
println!("GOODBY START: {}", buffer.len());
|
||||
|
||||
let entry_size = std::mem::size_of::<CaFormatGoodbyeItem>();
|
||||
if (buffer.len() % entry_size) != 0 {
|
||||
bail!("unexpected goodby item size ({})", entry_size);
|
||||
}
|
||||
|
||||
let mut pos = 0;
|
||||
|
||||
while (pos < buffer.len()) {
|
||||
|
||||
let item = map_struct::<CaFormatGoodbyeItem>(&buffer[pos..pos+entry_size])?;
|
||||
|
||||
if item.hash == CA_FORMAT_GOODBYE_TAIL_MARKER {
|
||||
println!(" Entry Offset: {}", item.offset);
|
||||
if item.size != (buffer.len() + 16) as u64 {
|
||||
bail!("gut unexpected goodby entry size (tail marker)");
|
||||
}
|
||||
} else {
|
||||
println!(" Offset: {}", item.offset);
|
||||
println!(" Size: {}", item.size);
|
||||
println!(" Hash: {:016x}", item.hash);
|
||||
}
|
||||
|
||||
pos += entry_size;
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn dump_archive(param: Value, _info: &ApiMethod) -> Result<Value, Error> {
|
||||
|
||||
let archive = required_string_param(¶m, "archive");
|
||||
let mut file = std::fs::File::open(archive)?;
|
||||
|
||||
println!("CATAR {}", archive);
|
||||
|
||||
let mut buffer = [0u8; 16];
|
||||
|
||||
let mut nesting = 0;
|
||||
|
||||
loop {
|
||||
file.read_exact(&mut buffer)?;
|
||||
|
||||
let header = map_struct::<CaFormatHeader>(&mut buffer)?;
|
||||
|
||||
println!("Type: {:016x}", header.htype);
|
||||
println!("Size: {}", header.size);
|
||||
|
||||
let mut rest = vec![0u8; (header.size as usize) - 16];
|
||||
file.read_exact(&mut rest)?;
|
||||
|
||||
if header.htype == CA_FORMAT_FILENAME {
|
||||
let name = read_os_string(&rest);
|
||||
let hash = compute_goodbye_hash(&rest[..rest.len()-1]);
|
||||
println!("Name: {:?} {:016x}", name, hash);
|
||||
}
|
||||
|
||||
if header.htype == CA_FORMAT_ENTRY {
|
||||
let entry = map_struct::<CaFormatEntry>(&mut rest)?;
|
||||
println!("Mode: {:08x} {:08x}", entry.mode, (entry.mode as u32) & libc::S_IFDIR);
|
||||
if ((entry.mode as u32) & libc::S_IFMT) == libc::S_IFDIR {
|
||||
nesting += 1;
|
||||
}
|
||||
}
|
||||
if header.htype == CA_FORMAT_GOODBYE {
|
||||
nesting -= 1;
|
||||
print_goodby_entries(&rest)?;
|
||||
if nesting == 0 { break; }
|
||||
}
|
||||
}
|
||||
|
||||
Ok(Value::Null)
|
||||
}
|
||||
|
||||
fn create_backup(param: Value, _info: &ApiMethod) -> Result<Value, Error> {
|
||||
|
||||
let _archive = required_string_param(¶m, "archive");
|
||||
let _source = required_string_param(¶m, "source");
|
||||
|
||||
|
||||
Ok(Value::Null)
|
||||
}
|
||||
|
||||
fn main() {
|
||||
|
||||
let cmd_def = CliCommandMap::new()
|
||||
.insert("create", CliCommand::new(
|
||||
ApiMethod::new(
|
||||
create_backup,
|
||||
ObjectSchema::new("Create new catar archive.")
|
||||
.required("archive", StringSchema::new("Archive name"))
|
||||
.required("source", StringSchema::new("Source directory."))
|
||||
))
|
||||
.arg_param(vec!["archive", "source"])
|
||||
.into()
|
||||
)
|
||||
.insert("dump", CliCommand::new(
|
||||
ApiMethod::new(
|
||||
dump_archive,
|
||||
ObjectSchema::new("Textual dump of archive contents (debug toolkit).")
|
||||
.required("archive", StringSchema::new("Archive name."))
|
||||
))
|
||||
.arg_param(vec!["archive"])
|
||||
.into()
|
||||
);
|
||||
|
||||
if let Err(err) = run_cli_command(&cmd_def.into()) {
|
||||
eprintln!("Error: {}", err);
|
||||
print_cli_usage();
|
||||
std::process::exit(-1);
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue
Block a user