From db7c54f92f386a94db8af7a12626d2657b4dd640 Mon Sep 17 00:00:00 2001 From: edef Date: Sat, 14 Aug 2021 21:28:14 +0000 Subject: ripple/fossil: a basic content-addressable store Fossil stores content-addressed blobs of file contents and Protobuf-encoded directory listings, backed by Sled. Change-Id: I8b49de6342218ca00755cec980b1d0cfb18878a7 --- ripple/fossil/src/bin/add.rs | 31 ++++++++++++++++++++++ ripple/fossil/src/bin/extract.rs | 57 ++++++++++++++++++++++++++++++++++++++++ 2 files changed, 88 insertions(+) create mode 100644 ripple/fossil/src/bin/add.rs create mode 100644 ripple/fossil/src/bin/extract.rs (limited to 'ripple/fossil/src/bin') diff --git a/ripple/fossil/src/bin/add.rs b/ripple/fossil/src/bin/add.rs new file mode 100644 index 0000000..114f893 --- /dev/null +++ b/ripple/fossil/src/bin/add.rs @@ -0,0 +1,31 @@ +// SPDX-FileCopyrightText: edef +// SPDX-License-Identifier: OSL-3.0 + +use { + fossil::Directory, + prost::Message, + std::{ + env, + io::{self, Write}, + path::Path, + }, +}; + +fn main() { + let store = fossil::Store::open("fossil.db").unwrap(); + let mut root = Directory::new(); + + for name in env::args().skip(1) { + let path = Path::new(&name); + let name = path + .file_name() + .and_then(|s| s.to_str()) + .expect("invalid path") + .to_owned(); + + root.children.insert(name, store.add_path(path)); + } + + let mut stdout = io::stdout(); + stdout.write_all(&root.into_pb().encode_to_vec()).unwrap(); +} diff --git a/ripple/fossil/src/bin/extract.rs b/ripple/fossil/src/bin/extract.rs new file mode 100644 index 0000000..f83ce0e --- /dev/null +++ b/ripple/fossil/src/bin/extract.rs @@ -0,0 +1,57 @@ +// SPDX-FileCopyrightText: edef +// SPDX-License-Identifier: OSL-3.0 + +use { + fossil::{store, Directory}, + prost::Message, + std::{ + fs, + io::{self, Read, Write}, + os::unix::{fs::symlink, prelude::OpenOptionsExt}, + path::Path, + }, +}; + +fn main() { + let store = fossil::Store::open("fossil.db").unwrap(); + let root = { + let mut stdin = io::stdin(); + + let mut bytes = Vec::new(); + stdin.read_to_end(&mut bytes).unwrap(); + + let pb = store::Directory::decode(&*bytes).unwrap(); + Directory::from_pb(pb) + }; + + let root_path = Path::new("."); + extract(&store, root_path, &root); +} + +fn extract(store: &fossil::Store, path: &Path, dir: &Directory) { + for (name, node) in &dir.children { + let path = path.join(name); + match node.clone() { + fossil::Node::Directory { r#ref } => { + let blob = store.read_blob(r#ref); + let pb = store::Directory::decode(&*blob).unwrap(); + fs::create_dir(&path).unwrap(); + extract(store, &path, &Directory::from_pb(pb)); + } + fossil::Node::File { r#ref, executable } => { + let mode = if executable { 0o755 } else { 0o644 }; + let mut f = fs::OpenOptions::new() + .write(true) + .create_new(true) + .mode(mode) + .open(path) + .unwrap(); + let blob = store.read_blob(r#ref); + f.write_all(&blob).unwrap(); + } + fossil::Node::Link { target } => { + symlink(target, path).unwrap(); + } + } + } +} -- cgit 1.4.1