use anyhow::bail;
use gix::{commit::describe::SelectRef, prelude::ObjectIdExt, Repository, Submodule};
use crate::OutputFormat;
pub fn list(repo: Repository, mut out: impl std::io::Write, format: OutputFormat) -> anyhow::Result<()> {
if format != OutputFormat::Human {
bail!("Only human output is supported for now")
}
let Some(submodules) = repo.submodules()? else {
return Ok(());
};
for sm in submodules {
print_sm(sm, &mut out)?;
}
Ok(())
}
fn print_sm(sm: Submodule<'_>, out: &mut impl std::io::Write) -> anyhow::Result<()> {
let _span = gix::trace::coarse!("print_sm", path = ?sm.path());
let state = sm.state()?;
let mut sm_repo = sm.open()?;
if let Some(repo) = sm_repo.as_mut() {
repo.object_cache_size_if_unset(4 * 1024 * 1024);
};
writeln!(
out,
" {is_active} {path} {config} head:{head_id} index:{index_id} ({worktree}) [{url}]",
is_active = if !sm.is_active()? || !state.repository_exists {
"ⅹ"
} else {
"✓"
},
path = sm.path()?,
config = if state.superproject_configuration {
"config:yes"
} else {
"config:no"
},
head_id = submodule_short_hash(sm.head_id()?, sm_repo.as_ref()),
index_id = submodule_short_hash(sm.index_id()?, sm_repo.as_ref()),
worktree = match sm_repo {
Some(repo) => {
repo.head_commit()?
.describe()
.names(SelectRef::AllRefs)
.format()?
.to_string()
}
None => {
"no worktree".to_string()
}
},
url = sm.url()?.to_bstring()
)?;
Ok(())
}
fn submodule_short_hash(id: Option<gix::ObjectId>, repo: Option<&Repository>) -> String {
id.map_or_else(
|| "none".to_string(),
|id| repo.map_or_else(|| id.to_string(), |repo| id.attach(repo).shorten_or_id().to_string()),
)
}