backend: Inline gix::Repository::commit_as to prepare for signing

Additional bonus is that this allows us to avoid creating keep refs for the
intermediate commits in the data race preventing loop.
This commit is contained in:
Anton Bulakh 2023-11-12 03:40:23 +02:00 committed by Anton Bulakh
parent 811ac5ff23
commit 5ab00e197a

View file

@ -25,6 +25,7 @@ use async_trait::async_trait;
use gix::objs::CommitRefIter; use gix::objs::CommitRefIter;
use itertools::Itertools; use itertools::Itertools;
use prost::Message; use prost::Message;
use smallvec::SmallVec;
use thiserror::Error; use thiserror::Error;
use crate::backend::{ use crate::backend::{
@ -538,13 +539,7 @@ fn deserialize_extras(commit: &mut Commit, bytes: &[u8]) {
} }
} }
/// Creates a random ref in refs/jj/. Used for preventing GC of commits we /// Creates a ref in refs/jj/. Used for preventing GC of commits we create.
/// create.
fn create_no_gc_ref() -> String {
let random_bytes: [u8; 16] = rand::random();
format!("{NO_GC_REF_NAMESPACE}{}", hex::encode(random_bytes))
}
fn prevent_gc(git_repo: &gix::Repository, id: &CommitId) -> Result<(), BackendError> { fn prevent_gc(git_repo: &gix::Repository, id: &CommitId) -> Result<(), BackendError> {
git_repo git_repo
.reference( .reference(
@ -923,7 +918,7 @@ impl Backend for GitBackend {
"Cannot write a commit with no parents".into(), "Cannot write a commit with no parents".into(),
)); ));
} }
let mut parents = vec![]; let mut parents = SmallVec::new();
for parent_id in &contents.parents { for parent_id in &contents.parents {
if *parent_id == self.root_commit_id { if *parent_id == self.root_commit_id {
// Git doesn't have a root commit, so if the parent is the root commit, we don't // Git doesn't have a root commit, so if the parent is the root commit, we don't
@ -942,6 +937,7 @@ impl Backend for GitBackend {
} }
} }
let extras = serialize_extras(&contents); let extras = serialize_extras(&contents);
// If two writers write commits of the same id with different metadata, they // If two writers write commits of the same id with different metadata, they
// will both succeed and the metadata entries will be "merged" later. Since // will both succeed and the metadata entries will be "merged" later. Since
// metadata entry is keyed by the commit id, one of the entries would be lost. // metadata entry is keyed by the commit id, one of the entries would be lost.
@ -950,31 +946,40 @@ impl Backend for GitBackend {
// repository is rsync-ed. // repository is rsync-ed.
let (table, table_lock) = self.read_extra_metadata_table_locked()?; let (table, table_lock) = self.read_extra_metadata_table_locked()?;
let id = loop { let id = loop {
let git_id = locked_repo let commit = gix::objs::Commit {
.commit_as( message: message.to_owned().into(),
committer, tree: git_tree_id,
author, author: author.into(),
create_no_gc_ref(), committer: committer.into(),
message, encoding: None,
git_tree_id, parents: parents.clone(),
parents.iter().copied(), extra_headers: Default::default(),
) };
.map_err(|err| BackendError::WriteObject {
object_type: "commit", // todo sign commits here
source: Box::new(err),
})?; let git_id =
let id = CommitId::from_bytes(git_id.as_bytes()); locked_repo
match table.get_value(id.as_bytes()) { .write_object(&commit)
.map_err(|err| BackendError::WriteObject {
object_type: "commit",
source: Box::new(err),
})?;
match table.get_value(git_id.as_bytes()) {
Some(existing_extras) if existing_extras != extras => { Some(existing_extras) if existing_extras != extras => {
// It's possible a commit already exists with the same commit id but different // It's possible a commit already exists with the same commit id but different
// change id. Adjust the timestamp until this is no longer the case. // change id. Adjust the timestamp until this is no longer the case.
committer.time.seconds -= 1; committer.time.seconds -= 1;
} }
_ => { _ => break CommitId::from_bytes(git_id.as_bytes()),
break id;
}
} }
}; };
// Everything up to this point had no permanent effect on the repo except
// GC-able objects
prevent_gc(&locked_repo, &id)?;
// Update the signature to match the one that was actually written to the object // Update the signature to match the one that was actually written to the object
// store // store
contents.committer.timestamp.timestamp = MillisSinceEpoch(committer.time.seconds * 1000); contents.committer.timestamp.timestamp = MillisSinceEpoch(committer.time.seconds * 1000);