use std::collections::HashSet;
use std::path::PathBuf;
use canonical_path::CanonicalPathBuf;
use clap::{Parser, ValueHint};
use libpijul::changestore::ChangeStore;
use libpijul::vertex_buffer::{change_message, VertexBuffer};
use libpijul::*;
use log::debug;
use crate::commands::common_opts::RepoAndChannel;
use crate::commands::load_channel;
use pijul_repository::Repository;
#[derive(Parser, Debug)]
pub struct Credit {
#[clap(flatten)]
base: RepoAndChannel,
#[clap(value_hint = ValueHint::FilePath)]
file: PathBuf,
}
impl Credit {
pub fn run(self) -> Result<(), anyhow::Error> {
let has_repo_path = self.base.repo_path().is_some();
let repo = Repository::find_root(self.base.repo_path())?;
let txn_ = repo.pristine.arc_txn_begin()?;
let txn = txn_.read();
let (channel, _) = load_channel(self.base.channel(), &*txn)?;
let repo_path = CanonicalPathBuf::canonicalize(&repo.path)?;
let (pos, _ambiguous) = if has_repo_path {
let root = std::fs::canonicalize(repo.path.join(&self.file))?;
let path = root.strip_prefix(&repo_path.as_path())?.to_str().unwrap();
txn.follow_oldest_path(&repo.changes, &channel, &path)?
} else {
let mut root = std::env::current_dir()?;
root.push(&self.file);
let root = std::fs::canonicalize(&root)?;
let path = root.strip_prefix(&repo_path.as_path())?.to_str().unwrap();
txn.follow_oldest_path(&repo.changes, &channel, &path)?
};
std::mem::drop(txn);
super::pager(repo.config.pager.as_ref());
match libpijul::output::output_file(
&repo.changes,
&txn_,
&channel,
pos,
&mut Creditor::new(std::io::stdout(), txn_.clone(), channel.clone()),
) {
Ok(_) => {}
Err(libpijul::output::FileError::Io(io)) => {
if let std::io::ErrorKind::BrokenPipe = io.kind() {
} else {
return Err(io.into());
}
}
Err(e) => return Err(e.into()),
}
Ok(())
}
}
pub struct Creditor<W: std::io::Write, T: ChannelTxnT> {
w: W,
buf: Vec<u8>,
new_line: bool,
changes: HashSet<Hash>,
txn: ArcTxn<T>,
channel: ChannelRef<T>,
}
impl<W: std::io::Write, T: ChannelTxnT> Creditor<W, T> {
pub fn new(w: W, txn: ArcTxn<T>, channel: ChannelRef<T>) -> Self {
Creditor {
w,
new_line: true,
buf: Vec::new(),
txn,
channel,
changes: HashSet::new(),
}
}
}
impl<W: std::io::Write, T: TxnTExt> VertexBuffer for Creditor<W, T> {
fn output_line<E, C: FnOnce(&mut [u8]) -> Result<(), E>>(
&mut self,
v: Vertex<ChangeId>,
c: C,
) -> Result<(), E>
where
E: From<std::io::Error>,
{
debug!("outputting vertex {:?}", v);
self.buf.resize(v.end - v.start, 0);
c(&mut self.buf)?;
if !v.change.is_root() {
self.changes.clear();
let txn = self.txn.read();
let channel = self.channel.read();
for e in txn
.iter_adjacent(&channel, v, EdgeFlags::PARENT, EdgeFlags::all())
.unwrap()
{
let e = e.unwrap();
if e.introduced_by().is_root() {
continue;
}
if let Ok(Some(intro)) = txn.get_external(&e.introduced_by()) {
self.changes.insert(intro.into());
}
}
if !self.new_line {
writeln!(self.w)?;
}
writeln!(self.w)?;
let mut is_first = true;
for c in self.changes.drain() {
let c = c.to_base32();
write!(
self.w,
"{}{}",
if is_first { "" } else { ", " },
c.split_at(12).0,
)?;
is_first = false;
}
writeln!(self.w, "\n")?;
}
let ends_with_newline = self.buf.ends_with(b"\n");
if let Ok(s) = std::str::from_utf8(&self.buf[..]) {
for l in s.lines() {
self.w.write_all(b"> ")?;
self.w.write_all(l.as_bytes())?;
self.w.write_all(b"\n")?;
}
}
if !self.buf.is_empty() {
self.new_line = ends_with_newline;
}
Ok(())
}
fn output_conflict_marker<C: ChangeStore>(
&mut self,
marker: &str,
id: usize,
sides: Option<(&C, &[&Hash])>,
) -> Result<(), std::io::Error> {
if !self.new_line {
self.w.write_all(b"\n")?;
}
write!(self.w, "{} {}", marker, id)?;
match sides {
Some((changes, sides)) => {
for side in sides.into_iter() {
let h = side.to_base32();
write!(
self.w,
" [{} {}]",
h.split_at(8).0,
change_message(changes, side)
)?;
}
}
None => (),
};
self.w.write_all(b"\n")?;
Ok(())
}
}