use std::{
io::Read,
path::{Path, PathBuf},
};
use git_features::fs::walkdir::DirEntryIter;
use git_object::bstr::ByteSlice;
use os_str_bytes::OsStrBytes;
use crate::{
store::file::{self, loose::Reference},
FullName,
};
pub(in crate::store::file) struct SortedLoosePaths {
pub(crate) base: PathBuf,
file_walk: DirEntryIter,
}
impl SortedLoosePaths {
pub fn at_root_with_names(path: impl AsRef<Path>, base: impl Into<PathBuf>) -> Self {
let file_walk = git_features::fs::walkdir_sorted_new(path).into_iter();
SortedLoosePaths {
base: base.into(),
file_walk,
}
}
}
impl Iterator for SortedLoosePaths {
type Item = std::io::Result<(PathBuf, FullName)>;
fn next(&mut self) -> Option<Self::Item> {
for entry in self.file_walk.by_ref() {
match entry {
Ok(entry) => {
if !entry.file_type().is_file() {
continue;
}
let full_path = entry.path().to_owned();
let full_name = full_path
.strip_prefix(&self.base)
.expect("prefix-stripping cannot fail as prefix is our root")
.to_raw_bytes();
#[cfg(windows)]
let full_name: Vec<u8> = full_name.into_owned().replace(b"\\", b"/");
if git_validate::reference::name_partial(full_name.as_bstr()).is_ok() {
#[cfg(not(windows))]
let name = FullName(full_name.into_owned().into());
#[cfg(windows)]
let name = FullName(full_name.into());
return Some(Ok((full_path, name)));
} else {
continue;
}
}
Err(err) => return Some(Err(err.into_io_error().expect("no symlink related errors"))),
}
}
None
}
}
pub struct Loose {
ref_paths: SortedLoosePaths,
buf: Vec<u8>,
}
impl Loose {
pub fn at_root(root: impl AsRef<Path>, base: impl Into<PathBuf>) -> Self {
Loose {
ref_paths: SortedLoosePaths::at_root_with_names(root, base),
buf: Vec::new(),
}
}
}
impl Iterator for Loose {
type Item = Result<Reference, loose::Error>;
fn next(&mut self) -> Option<Self::Item> {
self.ref_paths.next().map(|res| {
res.map_err(loose::Error::Traversal).and_then(|(validated_path, name)| {
std::fs::File::open(&validated_path)
.and_then(|mut f| {
self.buf.clear();
f.read_to_end(&mut self.buf)
})
.map_err(loose::Error::ReadFileContents)
.and_then(|_| {
let relative_path = validated_path
.strip_prefix(&self.ref_paths.base)
.expect("root contains path");
Reference::try_from_path(name, &self.buf).map_err(|err| loose::Error::ReferenceCreation {
err,
relative_path: relative_path.into(),
})
})
})
})
}
}
impl file::Store {
pub fn loose_iter(&self) -> std::io::Result<Loose> {
let refs = self.refs_dir();
if !refs.is_dir() {
return Err(std::io::ErrorKind::NotFound.into());
}
Ok(Loose::at_root(refs, self.base.clone()))
}
pub fn loose_iter_prefixed(&self, prefix: impl AsRef<Path>) -> std::io::Result<Loose> {
let prefix = self.validate_prefix(prefix.as_ref())?;
Ok(Loose::at_root(self.base.join(prefix), self.base.clone()))
}
pub(in crate::store::file) fn refs_dir(&self) -> PathBuf {
self.base.join("refs")
}
pub(in crate::store::file) fn validate_prefix<'a>(&self, prefix: &'a Path) -> std::io::Result<&'a Path> {
if prefix.is_absolute() {
return Err(std::io::Error::new(
std::io::ErrorKind::InvalidInput,
"prefix must be a relative path, like 'refs/heads'",
));
}
use std::path::Component::*;
if prefix.components().any(|c| matches!(c, CurDir | ParentDir)) {
return Err(std::io::Error::new(
std::io::ErrorKind::InvalidInput,
"Refusing to handle prefixes with relative path components",
));
}
Ok(prefix)
}
}
pub mod loose {
mod error {
use std::{io, path::PathBuf};
use quick_error::quick_error;
use crate::file;
quick_error! {
#[derive(Debug)]
#[allow(missing_docs)]
pub enum Error {
Traversal(err: io::Error) {
display("The file system could not be traversed")
source(err)
}
ReadFileContents(err: io::Error) {
display("The ref file could not be read in full")
source(err)
}
ReferenceCreation{ err: file::loose::reference::decode::Error, relative_path: PathBuf } {
display("The reference at '{}' could not be instantiated", relative_path.display())
source(err)
}
}
}
}
pub use error::Error;
}