gitoxide_core/repository/
submodule.rsuse 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,
dirty_suffix: Option<String>,
) -> 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, dirty_suffix.as_deref(), &mut out)?;
}
Ok(())
}
fn print_sm(sm: Submodule<'_>, dirty_suffix: Option<&str>, 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)
.id_as_fallback(true)
.try_resolve()?
.expect("resolution present if ID can be used as fallback")
.format_with_dirty_suffix(dirty_suffix.map(ToOwned::to_owned))?
.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()),
)
}