create src

This commit is contained in:
awfixer
2026-03-11 02:04:19 -07:00
commit 52f7a22bf2
2595 changed files with 402870 additions and 0 deletions

202
examples/log.rs Normal file
View File

@@ -0,0 +1,202 @@
use std::{
io::{stdout, Write},
path::{Path, PathBuf},
};
/// A toy-version of `git log`.
use clap::Parser;
use src::{
bstr::{BString, ByteSlice},
date::time::format,
revision::walk::Sorting,
};
fn main() {
let args = Args::parse_from(src::env::args_os());
match run(args) {
Ok(()) => {}
Err(e) => eprintln!("error: {e}"),
}
}
#[derive(Debug, clap::Parser)]
#[clap(name = "log", about = "git log example", version = option_env!("src_VERSION"))]
struct Args {
/// Alternative git directory to use
#[clap(name = "dir", long = "git-dir")]
git_dir: Option<PathBuf>,
/// Number of commits to return
#[clap(short, long)]
count: Option<usize>,
/// Number of commits to skip
#[clap(short, long)]
skip: Option<usize>,
/// Commits are sorted as they are mentioned in the commit graph.
#[clap(short, long)]
breadth_first: bool,
/// Commits are sorted by their commit time in descending order.
#[clap(short, long)]
newest_first: bool,
/// Show commits with the specified minimum number of parents
#[clap(long)]
min_parents: Option<usize>,
/// Show commits with the specified maximum number of parents
#[clap(long)]
max_parents: Option<usize>,
/// Show only merge commits (implies --min-parents=2)
#[clap(long)]
merges: bool,
/// Show only non-merge commits (implies --max-parents=1)
#[clap(long)]
no_merges: bool,
/// Reverse the commit sort order (and loads all of them into memory).
#[clap(short, long)]
reverse: bool,
/// The ref-spec for the first commit to log, or HEAD.
#[clap(name = "commit")]
committish: Option<String>,
/// The path interested in log history of
#[clap(name = "path")]
paths: Vec<PathBuf>,
}
fn run(args: Args) -> anyhow::Result<()> {
let repo = src::discover(args.git_dir.as_deref().unwrap_or(Path::new(".")))?;
let commit = repo
.rev_parse_single({
args.committish
.map(|mut c| {
c.push_str("^{commit}");
c
})
.as_deref()
.unwrap_or("HEAD")
})?
.object()?
.try_into_commit()?;
let sorting = if args.breadth_first {
Sorting::BreadthFirst
} else {
// else if args.newest_first {
Sorting::ByCommitTime(Default::default())
};
let mut min_parents = args.min_parents.unwrap_or(0);
let mut max_parents = args.max_parents.unwrap_or(usize::MAX);
if args.merges {
min_parents = 2;
}
if args.no_merges {
max_parents = 1;
}
let mut log_iter: Box<dyn Iterator<Item = Result<LogEntryInfo, _>>> = Box::new(
repo.rev_walk([commit.id])
.sorting(sorting)
.all()?
.filter(|info| {
info.as_ref().map_or(true, |info| {
info.parent_ids.len() <= max_parents &&
info.parent_ids.len() >= min_parents &&
// if the list of paths is empty the filter passes.
// if paths are provided check that any one of them are
// in fact relevant for the current commit.
(args.paths.is_empty() || args.paths.iter().any(|path| {
// TODO: should make use of the `git2::DiffOptions`
// counterpart in src for a set of files and also to
// generate diffs. When ready, also make paths resistant
// to illformed UTF8 by not using ".display()".
// PERFORMANCE WARNING: What follows is a clever implementation
// that is also **very** slow - do not use on bigger sample
// repositories as this needs native support in `src` to
// be fast enough.
match repo.rev_parse_single(
format!("{}:{}", info.id, path.display()).as_str()
) {
// check by parsing the revspec on the path with
// the prefix of the tree of the current commit,
// vs. the same counterpart but using each of
// commit's parents; if any pairs don't match,
// this indicates this path was changed in this
// commit thus should be included in output.
// naturally, root commits have no parents and
// by definition whatever paths in there must
// have been introduced there, so include them.
Ok(oid) => info.parent_ids.is_empty() || info
.parent_ids
.iter()
.any(|id| {
repo.rev_parse_single(
format!("{id}:{}", path.display()).as_str()
).ok() != Some(oid)
}),
// no oid for the path resolved with this commit
// so this commit can be omitted from output.
Err(_) => false,
}
}))
})
})
.map(|info| -> anyhow::Result<_> {
let info = info?;
let commit = info.object()?;
let commit_ref = commit.decode()?;
let author = commit_ref.author()?;
Ok(LogEntryInfo {
commit_id: commit.id().to_hex().to_string(),
parents: info.parent_ids().map(|id| id.shorten_or_id().to_string()).collect(),
author: {
let mut buf = Vec::new();
author.actor().write_to(&mut buf)?;
buf.into()
},
time: author.time()?.format_or_unix(format::DEFAULT),
message: commit_ref.message.to_owned(),
})
}),
);
if args.reverse {
let mut results: Vec<_> = log_iter.collect();
results.reverse();
log_iter = Box::new(results.into_iter());
}
let mut log_iter = log_iter
.skip(args.skip.unwrap_or_default())
.take(args.count.unwrap_or(usize::MAX))
.peekable();
let mut out = stdout().lock();
let mut buf = Vec::new();
while let Some(entry) = log_iter.next() {
buf.clear();
let entry = entry?;
writeln!(buf, "commit {}", entry.commit_id)?;
if entry.parents.len() > 1 {
writeln!(buf, "Merge: {}", entry.parents.join(" "))?;
}
writeln!(buf, "Author: {}", entry.author)?;
writeln!(buf, "Date: {}\n", entry.time)?;
for line in entry.message.lines() {
write!(buf, " ")?;
buf.write_all(line)?;
writeln!(buf)?;
}
// only include newline if more log entries, mimicking `git log`
if log_iter.peek().is_some() {
writeln!(buf)?;
}
out.write_all(&buf)?;
}
Ok(())
}
struct LogEntryInfo {
commit_id: String,
parents: Vec<String>,
author: BString,
time: String,
message: BString,
}

78
examples/ls-tree.rs Normal file
View File

@@ -0,0 +1,78 @@
use std::io::{stdout, Write};
use clap::Parser;
use src::{bstr::BString, objs::tree::EntryMode, traverse::tree::Recorder, ObjectId};
fn main() {
let args = Args::parse_from(src::env::args_os());
match run(args) {
Ok(()) => {}
Err(e) => eprintln!("error: {e}"),
}
}
#[derive(Debug, clap::Parser)]
#[clap(name = "ls-tree", about = "git ls-tree example", version = option_env!("src_VERSION"))]
#[clap(arg_required_else_help = true)]
struct Args {
/// Recurse into subtrees
#[clap(short = 'r')]
recursive: bool,
/// Only show trees
#[clap(short = 'd')]
tree_only: bool,
/// Show trees when recursing
#[clap(short = 't')]
tree_recursing: bool,
/// A revspec pointing to a tree-ish object, e.g. 'HEAD', 'HEAD:src/'
#[clap(name = "tree-ish")]
treeish: String,
}
fn run(args: Args) -> anyhow::Result<()> {
let repo = src::discover(".")?;
let tree = repo.rev_parse_single(&*args.treeish)?.object()?.peel_to_tree()?;
let entries = if args.recursive {
let mut recorder = Recorder::default();
tree.traverse().breadthfirst(&mut recorder)?;
recorder
.records
.into_iter()
.filter(|entry| args.tree_recursing || args.tree_only || entry.mode.is_no_tree())
.filter(|entry| !args.tree_only || (entry.mode.is_tree()))
.map(|entry| Entry::new(entry.mode, entry.oid, entry.filepath))
.collect::<Vec<_>>()
} else {
tree.iter()
.filter_map(|res| res.ok().map(|entry| entry.inner)) // dropping errors silently
.filter(|entry| !args.tree_only || (entry.mode.is_tree()))
.map(|entry| Entry::new(entry.mode, entry.oid.to_owned(), entry.filename.to_owned()))
.collect::<Vec<_>>()
};
let mut out = stdout().lock();
for entry in entries {
writeln!(
out,
"{:>6o} {:4} {} {}",
entry.mode,
entry.mode.as_str(),
entry.hash,
entry.path
)?;
}
Ok(())
}
struct Entry {
mode: EntryMode,
hash: ObjectId,
path: BString,
}
impl Entry {
fn new(mode: EntryMode, hash: ObjectId, path: BString) -> Self {
Self { mode, hash, path }
}
}