chunk_store: reduce number of directories
Else, scans everything takes too long ...
This commit is contained in:
parent
15e9b4ed60
commit
e95950e40a
|
@ -36,15 +36,14 @@ pub fn digest_to_hex(digest: &[u8]) -> String {
|
||||||
|
|
||||||
fn digest_to_prefix(digest: &[u8]) -> PathBuf {
|
fn digest_to_prefix(digest: &[u8]) -> PathBuf {
|
||||||
|
|
||||||
let mut buf = Vec::<u8>::with_capacity(3+1+2+1);
|
let mut buf = Vec::<u8>::with_capacity(2+1+2+1);
|
||||||
|
|
||||||
buf.push(HEX_CHARS[(digest[0] as usize) >> 4]);
|
buf.push(HEX_CHARS[(digest[0] as usize) >> 4]);
|
||||||
buf.push(HEX_CHARS[(digest[0] as usize) &0xf]);
|
buf.push(HEX_CHARS[(digest[0] as usize) &0xf]);
|
||||||
buf.push(HEX_CHARS[(digest[1] as usize) >> 4]);
|
|
||||||
buf.push('/' as u8);
|
buf.push('/' as u8);
|
||||||
|
|
||||||
|
buf.push(HEX_CHARS[(digest[1] as usize) >> 4]);
|
||||||
buf.push(HEX_CHARS[(digest[1] as usize) & 0xf]);
|
buf.push(HEX_CHARS[(digest[1] as usize) & 0xf]);
|
||||||
buf.push(HEX_CHARS[(digest[2] as usize) >> 4]);
|
|
||||||
buf.push('/' as u8);
|
buf.push('/' as u8);
|
||||||
|
|
||||||
let path = unsafe { String::from_utf8_unchecked(buf)};
|
let path = unsafe { String::from_utf8_unchecked(buf)};
|
||||||
|
@ -76,13 +75,27 @@ impl ChunkStore {
|
||||||
bail!("unable to create chunk store '{}' subdir {:?} - {}", name, chunk_dir, err);
|
bail!("unable to create chunk store '{}' subdir {:?} - {}", name, chunk_dir, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
// create 4096 subdir
|
// create 256*256 subdirs
|
||||||
for i in 0..4096 {
|
let mut last_percentage = 0;
|
||||||
|
|
||||||
|
for i in 0..256 {
|
||||||
let mut l1path = chunk_dir.clone();
|
let mut l1path = chunk_dir.clone();
|
||||||
l1path.push(format!("{:03x}",i));
|
l1path.push(format!("{:02x}",i));
|
||||||
if let Err(err) = std::fs::create_dir(&l1path) {
|
if let Err(err) = std::fs::create_dir(&l1path) {
|
||||||
bail!("unable to create chunk store '{}' subdir {:?} - {}", name, l1path, err);
|
bail!("unable to create chunk store '{}' subdir {:?} - {}", name, l1path, err);
|
||||||
}
|
}
|
||||||
|
for j in 0..256 {
|
||||||
|
let mut l2path = l1path.clone();
|
||||||
|
l2path.push(format!("{:02x}",j));
|
||||||
|
if let Err(err) = std::fs::create_dir(&l2path) {
|
||||||
|
bail!("unable to create chunk store '{}' subdir {:?} - {}", name, l2path, err);
|
||||||
|
}
|
||||||
|
let percentage = ((i*256+j)*100)/(256*256);
|
||||||
|
if percentage != last_percentage {
|
||||||
|
eprintln!("Percentage done: {}", percentage);
|
||||||
|
last_percentage = percentage;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Self::open(name, base)
|
Self::open(name, base)
|
||||||
|
@ -146,7 +159,7 @@ impl ChunkStore {
|
||||||
let filename = entry.file_name();
|
let filename = entry.file_name();
|
||||||
if let Ok(stat) = nix::sys::stat::fstatat(rawfd, filename, nix::fcntl::AtFlags::AT_SYMLINK_NOFOLLOW) {
|
if let Ok(stat) = nix::sys::stat::fstatat(rawfd, filename, nix::fcntl::AtFlags::AT_SYMLINK_NOFOLLOW) {
|
||||||
let age = now - stat.st_atime;
|
let age = now - stat.st_atime;
|
||||||
println!("FOUND {} {:?}", age/(3600*24), filename);
|
//println!("FOUND {} {:?}", age/(3600*24), filename);
|
||||||
if age/(3600*24) >= 2 {
|
if age/(3600*24) >= 2 {
|
||||||
println!("UNLINK {} {:?}", age/(3600*24), filename);
|
println!("UNLINK {} {:?}", age/(3600*24), filename);
|
||||||
let res = unsafe { libc::unlinkat(rawfd, filename.as_ptr(), 0) };
|
let res = unsafe { libc::unlinkat(rawfd, filename.as_ptr(), 0) };
|
||||||
|
@ -175,8 +188,8 @@ impl ChunkStore {
|
||||||
|
|
||||||
let base_fd = base_handle.as_raw_fd();
|
let base_fd = base_handle.as_raw_fd();
|
||||||
|
|
||||||
for i in 0..4096 {
|
for i in 0..256 {
|
||||||
let l1name = PathBuf::from(format!("{:03x}", i));
|
let l1name = PathBuf::from(format!("{:02x}", i));
|
||||||
let mut l1_handle = match nix::dir::Dir::openat(
|
let mut l1_handle = match nix::dir::Dir::openat(
|
||||||
base_fd, &l1name, OFlag::O_RDONLY, Mode::empty()) {
|
base_fd, &l1name, OFlag::O_RDONLY, Mode::empty()) {
|
||||||
Ok(h) => h,
|
Ok(h) => h,
|
||||||
|
@ -186,22 +199,13 @@ impl ChunkStore {
|
||||||
|
|
||||||
let l1_fd = l1_handle.as_raw_fd();
|
let l1_fd = l1_handle.as_raw_fd();
|
||||||
|
|
||||||
for l1_entry in l1_handle.iter() {
|
for j in 0..256 {
|
||||||
let l1_entry = match l1_entry {
|
let l2name = PathBuf::from(format!("{:02x}", j));
|
||||||
Ok(l1_entry) => l1_entry,
|
|
||||||
Err(_) => continue /* ignore errors? */,
|
|
||||||
};
|
|
||||||
let file_type = match l1_entry.file_type() {
|
|
||||||
Some(file_type) => file_type,
|
|
||||||
None => bail!("unsupported file system type on chunk store '{}'", self.name),
|
|
||||||
};
|
|
||||||
if file_type != nix::dir::Type::Directory { continue; }
|
|
||||||
|
|
||||||
let l2name = l1_entry.file_name();
|
println!("SCAN {:?} {:?}", l1name, l2name);
|
||||||
if l2name.to_bytes_with_nul()[0] == b'.' { continue; }
|
|
||||||
|
|
||||||
let mut l2_handle = match Dir::openat(
|
let mut l2_handle = match Dir::openat(
|
||||||
l1_fd, l2name, OFlag::O_RDONLY, Mode::empty()) {
|
l1_fd, &l2name, OFlag::O_RDONLY, Mode::empty()) {
|
||||||
Ok(h) => h,
|
Ok(h) => h,
|
||||||
Err(err) => bail!(
|
Err(err) => bail!(
|
||||||
"unable to open store '{}' dir {:?}/{:?}/{:?} - {}",
|
"unable to open store '{}' dir {:?}/{:?}/{:?} - {}",
|
||||||
|
@ -237,11 +241,6 @@ impl ChunkStore {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut chunk_dir = self.chunk_dir.clone();
|
|
||||||
chunk_dir.push(&prefix);
|
|
||||||
|
|
||||||
if let Err(_) = std::fs::create_dir(&chunk_dir) { /* ignore */ }
|
|
||||||
|
|
||||||
let mut tmp_path = chunk_path.clone();
|
let mut tmp_path = chunk_path.clone();
|
||||||
tmp_path.set_extension("tmp");
|
tmp_path.set_extension("tmp");
|
||||||
let mut f = std::fs::File::create(&tmp_path)?;
|
let mut f = std::fs::File::create(&tmp_path)?;
|
||||||
|
@ -269,7 +268,6 @@ impl ChunkStore {
|
||||||
pub fn base_path(&self) -> PathBuf {
|
pub fn base_path(&self) -> PathBuf {
|
||||||
self.base.clone()
|
self.base.clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -75,8 +75,11 @@ impl DataStore {
|
||||||
|
|
||||||
pub fn garbage_collection(&self) -> Result<(), Error> {
|
pub fn garbage_collection(&self) -> Result<(), Error> {
|
||||||
|
|
||||||
|
println!("Start GC phase1 (mark chunks)");
|
||||||
|
|
||||||
self.mark_used_chunks()?;
|
self.mark_used_chunks()?;
|
||||||
|
|
||||||
|
println!("Start GC phase2 (sweep unused chunks)");
|
||||||
self.chunk_store.sweep_used_chunks()?;
|
self.chunk_store.sweep_used_chunks()?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|
Loading…
Reference in New Issue