2017-03-16 08:42:30 +00:00
|
|
|
use std::io::{self, Write, Read};
|
2017-03-17 10:03:07 +00:00
|
|
|
use std::str::FromStr;
|
2017-03-10 11:43:32 +00:00
|
|
|
|
|
|
|
mod ae;
|
|
|
|
mod rabin;
|
|
|
|
mod fastcdc;
|
|
|
|
|
|
|
|
pub use self::ae::AeChunker;
|
|
|
|
pub use self::rabin::RabinChunker;
|
|
|
|
pub use self::fastcdc::FastCdcChunker;
|
|
|
|
|
|
|
|
// https://moinakg.wordpress.com/2013/06/22/high-performance-content-defined-chunking/
|
|
|
|
|
|
|
|
// Paper: "A Comprehensive Study of the Past, Present, and Future of Data Deduplication"
|
|
|
|
// Paper-URL: http://wxia.hustbackup.cn/IEEE-Survey-final.pdf
|
|
|
|
|
|
|
|
// https://borgbackup.readthedocs.io/en/stable/internals.html#chunks
|
|
|
|
// https://github.com/bup/bup/blob/master/lib/bup/bupsplit.c
|
|
|
|
|
2017-03-16 08:42:30 +00:00
|
|
|
quick_error!{
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum ChunkerError {
|
|
|
|
Read(err: io::Error) {
|
|
|
|
cause(err)
|
2017-03-22 08:19:16 +00:00
|
|
|
description("Failed to read input")
|
|
|
|
display("Chunker error: failed to read input\n\tcaused by: {}", err)
|
2017-03-16 08:42:30 +00:00
|
|
|
}
|
|
|
|
Write(err: io::Error) {
|
|
|
|
cause(err)
|
2017-03-22 08:19:16 +00:00
|
|
|
description("Failed to write to output")
|
|
|
|
display("Chunker error: failed to write to output\n\tcaused by: {}", err)
|
2017-03-16 08:42:30 +00:00
|
|
|
}
|
2017-03-22 08:19:16 +00:00
|
|
|
Custom(reason: &'static str) {
|
|
|
|
from()
|
2017-03-16 08:42:30 +00:00
|
|
|
description("Custom error")
|
2017-03-22 08:19:16 +00:00
|
|
|
display("Chunker error: {}", reason)
|
2017-03-16 08:42:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-10 11:43:32 +00:00
|
|
|
#[derive(Debug, Eq, PartialEq)]
|
|
|
|
pub enum ChunkerStatus {
|
|
|
|
Continue,
|
|
|
|
Finished
|
|
|
|
}
|
|
|
|
|
|
|
|
pub trait IChunker: Sized {
|
|
|
|
fn chunk<R: Read, W: Write>(&mut self, r: &mut R, w: &mut W) -> Result<ChunkerStatus, ChunkerError>;
|
|
|
|
fn get_type(&self) -> ChunkerType;
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum Chunker {
|
|
|
|
Ae(Box<AeChunker>),
|
|
|
|
Rabin(Box<RabinChunker>),
|
|
|
|
FastCdc(Box<FastCdcChunker>)
|
|
|
|
}
|
|
|
|
|
2017-03-16 08:42:30 +00:00
|
|
|
|
2017-03-10 11:43:32 +00:00
|
|
|
impl IChunker for Chunker {
|
|
|
|
fn get_type(&self) -> ChunkerType {
|
|
|
|
match *self {
|
|
|
|
Chunker::Ae(ref c) => c.get_type(),
|
|
|
|
Chunker::Rabin(ref c) => c.get_type(),
|
|
|
|
Chunker::FastCdc(ref c) => c.get_type()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn chunk<R: Read, W: Write>(&mut self, r: &mut R, w: &mut W) -> Result<ChunkerStatus, ChunkerError> {
|
|
|
|
match *self {
|
|
|
|
Chunker::Ae(ref mut c) => c.chunk(r, w),
|
|
|
|
Chunker::Rabin(ref mut c) => c.chunk(r, w),
|
|
|
|
Chunker::FastCdc(ref mut c) => c.chunk(r, w)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-02 16:55:53 +00:00
|
|
|
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
|
2017-03-10 11:43:32 +00:00
|
|
|
pub enum ChunkerType {
|
|
|
|
Ae(usize),
|
|
|
|
Rabin((usize, u32)),
|
|
|
|
FastCdc((usize, u64))
|
|
|
|
}
|
|
|
|
serde_impl!(ChunkerType(u64) {
|
|
|
|
Ae(usize) => 1,
|
|
|
|
Rabin((usize, u32)) => 2,
|
|
|
|
FastCdc((usize, u64)) => 3
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
impl ChunkerType {
|
|
|
|
pub fn from(name: &str, avg_size: usize, seed: u64) -> Result<Self, &'static str> {
|
|
|
|
match name {
|
|
|
|
"ae" => Ok(ChunkerType::Ae(avg_size)),
|
|
|
|
"rabin" => Ok(ChunkerType::Rabin((avg_size, seed as u32))),
|
|
|
|
"fastcdc" => Ok(ChunkerType::FastCdc((avg_size, seed))),
|
|
|
|
_ => Err("Unsupported chunker type")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-17 10:03:07 +00:00
|
|
|
pub fn from_string(name: &str) -> Result<Self, &'static str> {
|
|
|
|
let (name, size) = if let Some(pos) = name.find('/') {
|
|
|
|
let size = try!(usize::from_str(&name[pos+1..]).map_err(|_| "Chunk size must be a number"));
|
|
|
|
let name = &name[..pos];
|
|
|
|
(name, size)
|
|
|
|
} else {
|
|
|
|
(name, 8)
|
|
|
|
};
|
|
|
|
Self::from(name, size * 1024, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-10 11:43:32 +00:00
|
|
|
#[inline]
|
|
|
|
pub fn create(&self) -> Chunker {
|
|
|
|
match *self {
|
|
|
|
ChunkerType::Ae(size) => Chunker::Ae(Box::new(AeChunker::new(size))),
|
|
|
|
ChunkerType::Rabin((size, seed)) => Chunker::Rabin(Box::new(RabinChunker::new(size, seed))),
|
|
|
|
ChunkerType::FastCdc((size, seed)) => Chunker::FastCdc(Box::new(FastCdcChunker::new(size, seed)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn name(&self) -> &'static str {
|
|
|
|
match *self {
|
|
|
|
ChunkerType::Ae(_size) => "ae",
|
|
|
|
ChunkerType::Rabin((_size, _seed)) => "rabin",
|
|
|
|
ChunkerType::FastCdc((_size, _seed)) => "fastcdc"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn avg_size(&self) -> usize {
|
|
|
|
match *self {
|
|
|
|
ChunkerType::Ae(size) => size,
|
|
|
|
ChunkerType::Rabin((size, _seed)) => size,
|
|
|
|
ChunkerType::FastCdc((size, _seed)) => size
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-18 16:22:11 +00:00
|
|
|
pub fn to_string(&self) -> String {
|
|
|
|
format!("{}/{}", self.name(), self.avg_size()/1024)
|
|
|
|
}
|
|
|
|
|
2017-03-10 11:43:32 +00:00
|
|
|
pub fn seed(&self) -> u64 {
|
|
|
|
match *self {
|
|
|
|
ChunkerType::Ae(_size) => 0,
|
|
|
|
ChunkerType::Rabin((_size, seed)) => seed as u64,
|
|
|
|
ChunkerType::FastCdc((_size, seed)) => seed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|