1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
use git_hash::ObjectId;
use git_object::bstr::BString;
use crate::{
store_impl::{file, file::Transaction},
transaction::RefEdit,
};
/// A function receiving an object id to resolve, returning its decompressed bytes,
/// used to obtain the peeled object ids for storage in packed-refs files.
///
/// Resolution means to follow tag objects until the end of the chain.
pub type FindObjectFn =
dyn FnMut(
git_hash::ObjectId,
&mut Vec<u8>,
) -> Result<Option<git_object::Kind>, Box<dyn std::error::Error + Send + Sync + 'static>>;
/// How to handle packed refs during a transaction
pub enum PackedRefs {
/// Only propagate deletions of references. This is the default
DeletionsOnly,
/// Propagate deletions as well as updates to references which are peeled, that is contain an object id
DeletionsAndNonSymbolicUpdates(Box<FindObjectFn>),
/// Propagate deletions as well as updates to references which are peeled, that is contain an object id. Furthermore delete the
/// reference which is originally updated if it exists. If it doesn't, the new value will be written into the packed ref right away.
DeletionsAndNonSymbolicUpdatesRemoveLooseSourceReference(Box<FindObjectFn>),
}
impl Default for PackedRefs {
fn default() -> Self {
PackedRefs::DeletionsOnly
}
}
#[derive(Debug)]
pub(in crate::store_impl::file) struct Edit {
update: RefEdit,
lock: Option<git_lock::Marker>,
/// Set if this update is coming from a symbolic reference and used to make it appear like it is the one that is handled,
/// instead of the referent reference.
parent_index: Option<usize>,
/// For symbolic refs, this is the previous OID to put into the reflog instead of our own previous value. It's the
/// peeled value of the leaf referent.
leaf_referent_previous_oid: Option<ObjectId>,
}
impl Edit {
fn name(&self) -> BString {
self.update.name.0.clone()
}
}
impl std::borrow::Borrow<RefEdit> for Edit {
fn borrow(&self) -> &RefEdit {
&self.update
}
}
impl std::borrow::BorrowMut<RefEdit> for Edit {
fn borrow_mut(&mut self) -> &mut RefEdit {
&mut self.update
}
}
/// Edits
impl file::Store {
/// Open a transaction with the given `edits`, and determine how to fail if a `lock` cannot be obtained.
/// A snapshot of packed references will be obtained automatically if needed to fulfill this transaction
/// and will be provided as result of a successful transaction. Note that upon transaction failure, packed-refs
/// will never have been altered.
///
/// The transaction inherits the parent namespace.
pub fn transaction(&self) -> Transaction<'_> {
Transaction {
store: self,
packed_transaction: None,
updates: None,
packed_refs: PackedRefs::default(),
}
}
}
impl<'s> Transaction<'s> {
/// Configure the way packed refs are handled during the transaction
pub fn packed_refs(mut self, packed_refs: PackedRefs) -> Self {
self.packed_refs = packed_refs;
self
}
}
///
pub mod prepare;
///
pub mod commit;