12ef69c31b
- Introduce modules for testing input/output options - Provide macros for generating test cases - Hide details of subprocess execution
204 lines
5.2 KiB
Rust
204 lines
5.2 KiB
Rust
use anyhow::Result;
|
|
use std::fs::OpenOptions;
|
|
use std::io::Write;
|
|
use std::str::from_utf8;
|
|
|
|
mod common;
|
|
|
|
use common::common_args::*;
|
|
use common::input_arg::*;
|
|
use common::test_dir::*;
|
|
use common::*;
|
|
|
|
//------------------------------------------
|
|
|
|
const USAGE: &str = "Usage: thin_dump [options] {device|file}\n\
|
|
Options:\n \
|
|
{-h|--help}\n \
|
|
{-f|--format} {xml|human_readable|custom}\n \
|
|
{-r|--repair}\n \
|
|
{-m|--metadata-snap} [block#]\n \
|
|
{-o <xml file>}\n \
|
|
{--dev-id} <dev-id>\n \
|
|
{--skip-mappings}\n \
|
|
{-V|--version}";
|
|
|
|
//------------------------------------------
|
|
|
|
test_accepts_help!(THIN_DUMP, USAGE);
|
|
test_accepts_version!(THIN_DUMP);
|
|
test_rejects_bad_option!(THIN_DUMP);
|
|
|
|
test_missing_input_arg!(THIN_DUMP);
|
|
test_input_file_not_found!(THIN_DUMP, ARG);
|
|
test_input_cannot_be_a_directory!(THIN_DUMP, ARG);
|
|
test_unreadable_input_file!(THIN_DUMP, ARG);
|
|
|
|
//------------------------------------------
|
|
// test dump & restore cycle
|
|
|
|
#[test]
|
|
fn dump_restore_cycle() -> Result<()> {
|
|
let mut td = TestDir::new()?;
|
|
|
|
let md = mk_valid_md(&mut td)?;
|
|
let md_path = md.to_str().unwrap();
|
|
let output = run_ok_raw(THIN_DUMP, &[md_path])?;
|
|
|
|
let xml = td.mk_path("meta.xml");
|
|
let mut file = OpenOptions::new()
|
|
.read(false)
|
|
.write(true)
|
|
.create(true)
|
|
.open(&xml)?;
|
|
file.write_all(&output.stdout[0..])?;
|
|
drop(file);
|
|
|
|
let md2 = mk_zeroed_md(&mut td)?;
|
|
let md2_path = md2.to_str().unwrap();
|
|
let xml_path = xml.to_str().unwrap();
|
|
run_ok(THIN_RESTORE, &["-i", xml_path, "-o", md2_path])?;
|
|
|
|
let output2 = run_ok_raw(THIN_DUMP, &[md2_path])?;
|
|
assert_eq!(output.stdout, output2.stdout);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
//------------------------------------------
|
|
// test no stderr with a normal dump
|
|
|
|
#[test]
|
|
fn no_stderr() -> Result<()> {
|
|
let mut td = TestDir::new()?;
|
|
|
|
let md = mk_valid_md(&mut td)?;
|
|
let md_path = md.to_str().unwrap();
|
|
let output = run_ok_raw(THIN_DUMP, &[md_path])?;
|
|
|
|
assert_eq!(output.stderr.len(), 0);
|
|
Ok(())
|
|
}
|
|
|
|
//------------------------------------------
|
|
// test superblock overriding & repair
|
|
// TODO: share with thin_repair
|
|
|
|
fn override_something(flag: &str, value: &str, pattern: &str) -> Result<()> {
|
|
let mut td = TestDir::new()?;
|
|
let md = mk_valid_md(&mut td)?;
|
|
let md_path = md.to_str().unwrap();
|
|
let output = run_ok_raw(THIN_DUMP, &[md_path, flag, value])?;
|
|
|
|
assert_eq!(output.stderr.len(), 0);
|
|
assert!(from_utf8(&output.stdout[0..])?.contains(pattern));
|
|
Ok(())
|
|
}
|
|
|
|
#[test]
|
|
fn override_transaction_id() -> Result<()> {
|
|
override_something("--transaction-id", "2345", "transaction=\"2345\"")
|
|
}
|
|
|
|
#[test]
|
|
fn override_data_block_size() -> Result<()> {
|
|
override_something("--data-block-size", "8192", "data_block_size=\"8192\"")
|
|
}
|
|
|
|
#[test]
|
|
fn override_nr_data_blocks() -> Result<()> {
|
|
override_something("--nr-data-blocks", "234500", "nr_data_blocks=\"234500\"")
|
|
}
|
|
|
|
// FIXME: duplicate with superblock_succeeds in thin_repair.rs
|
|
#[test]
|
|
fn repair_superblock() -> Result<()> {
|
|
let mut td = TestDir::new()?;
|
|
let md = mk_valid_md(&mut td)?;
|
|
let before = run_ok_raw(
|
|
THIN_DUMP,
|
|
&[
|
|
"--transaction-id=5",
|
|
"--data-block-size=128",
|
|
"--nr-data-blocks=4096000",
|
|
md.to_str().unwrap(),
|
|
],
|
|
)?;
|
|
damage_superblock(&md)?;
|
|
|
|
let after = run_ok_raw(
|
|
THIN_DUMP,
|
|
&[
|
|
"--repair",
|
|
"--transaction-id=5",
|
|
"--data-block-size=128",
|
|
"--nr-data-blocks=4096000",
|
|
md.to_str().unwrap(),
|
|
],
|
|
)?;
|
|
assert_eq!(after.stderr.len(), 0);
|
|
assert_eq!(before.stdout, after.stdout);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
//------------------------------------------
|
|
// test compatibility between options
|
|
// TODO: share with thin_repair
|
|
|
|
#[test]
|
|
fn missing_transaction_id() -> Result<()> {
|
|
let mut td = TestDir::new()?;
|
|
let md = mk_valid_md(&mut td)?;
|
|
damage_superblock(&md)?;
|
|
let stderr = run_fail(
|
|
THIN_DUMP,
|
|
&[
|
|
"--repair",
|
|
"--data-block-size=128",
|
|
"--nr-data-blocks=4096000",
|
|
md.to_str().unwrap(),
|
|
],
|
|
)?;
|
|
assert!(stderr.contains("transaction id"));
|
|
Ok(())
|
|
}
|
|
|
|
#[test]
|
|
fn missing_data_block_size() -> Result<()> {
|
|
let mut td = TestDir::new()?;
|
|
let md = mk_valid_md(&mut td)?;
|
|
damage_superblock(&md)?;
|
|
let stderr = run_fail(
|
|
THIN_DUMP,
|
|
&[
|
|
"--repair",
|
|
"--transaction-id=5",
|
|
"--nr-data-blocks=4096000",
|
|
md.to_str().unwrap(),
|
|
],
|
|
)?;
|
|
assert!(stderr.contains("data block size"));
|
|
Ok(())
|
|
}
|
|
|
|
#[test]
|
|
fn missing_nr_data_blocks() -> Result<()> {
|
|
let mut td = TestDir::new()?;
|
|
let md = mk_valid_md(&mut td)?;
|
|
damage_superblock(&md)?;
|
|
let stderr = run_fail(
|
|
THIN_DUMP,
|
|
&[
|
|
"--repair",
|
|
"--transaction-id=5",
|
|
"--data-block-size=128",
|
|
md.to_str().unwrap(),
|
|
],
|
|
)?;
|
|
assert!(stderr.contains("nr data blocks"));
|
|
Ok(())
|
|
}
|
|
|
|
//------------------------------------------
|