mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-17 17:55:29 +00:00
index: extract a trait for the index
Even though we don't know the details yet, we know that we want to make the index pluggable like the commit and opstore backends. Defining a trait for it should be a good step. We can refine the trait later.
This commit is contained in:
parent
7a985ed122
commit
b955e3de03
13 changed files with 127 additions and 90 deletions
|
@ -23,6 +23,7 @@ use thiserror::Error;
|
|||
use crate::backend::{CommitId, ObjectId};
|
||||
use crate::commit::Commit;
|
||||
use crate::git_backend::NO_GC_REF_NAMESPACE;
|
||||
use crate::index::Index;
|
||||
use crate::op_store::RefTarget;
|
||||
use crate::repo::MutableRepo;
|
||||
use crate::settings::GitSettings;
|
||||
|
|
199
lib/src/index.rs
199
lib/src/index.rs
|
@ -45,6 +45,34 @@ impl IndexPosition {
|
|||
pub const MAX: Self = IndexPosition(u32::MAX);
|
||||
}
|
||||
|
||||
pub trait Index {
|
||||
fn num_commits(&self) -> u32;
|
||||
|
||||
fn stats(&self) -> IndexStats;
|
||||
|
||||
fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition>;
|
||||
|
||||
fn shortest_unique_commit_id_prefix_len(&self, commit_id: &CommitId) -> usize;
|
||||
|
||||
fn resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId>;
|
||||
|
||||
fn entry_by_id(&self, commit_id: &CommitId) -> Option<IndexEntry>;
|
||||
|
||||
fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry;
|
||||
|
||||
fn has_id(&self, commit_id: &CommitId) -> bool;
|
||||
|
||||
fn is_ancestor(&self, ancestor_id: &CommitId, descendant_id: &CommitId) -> bool;
|
||||
|
||||
fn common_ancestors(&self, set1: &[CommitId], set2: &[CommitId]) -> Vec<CommitId>;
|
||||
|
||||
fn walk_revs(&self, wanted: &[CommitId], unwanted: &[CommitId]) -> RevWalk;
|
||||
|
||||
fn heads(&self, candidates: &mut dyn Iterator<Item = &CommitId>) -> Vec<CommitId>;
|
||||
|
||||
fn topo_order(&self, input: &mut dyn Iterator<Item = &CommitId>) -> Vec<IndexEntry>;
|
||||
}
|
||||
|
||||
#[derive(Clone, Copy)]
|
||||
pub enum IndexRef<'a> {
|
||||
Readonly(&'a ReadonlyIndex),
|
||||
|
@ -629,56 +657,58 @@ impl MutableIndex {
|
|||
IndexLoadError::IoError(err) => err,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
pub fn num_commits(&self) -> u32 {
|
||||
impl Index for MutableIndex {
|
||||
fn num_commits(&self) -> u32 {
|
||||
CompositeIndex(self).num_commits()
|
||||
}
|
||||
|
||||
pub fn stats(&self) -> IndexStats {
|
||||
fn stats(&self) -> IndexStats {
|
||||
CompositeIndex(self).stats()
|
||||
}
|
||||
|
||||
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||
fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||
CompositeIndex(self).commit_id_to_pos(commit_id)
|
||||
}
|
||||
|
||||
pub fn shortest_unique_commit_id_prefix_len(&self, commit_id: &CommitId) -> usize {
|
||||
fn shortest_unique_commit_id_prefix_len(&self, commit_id: &CommitId) -> usize {
|
||||
CompositeIndex(self).shortest_unique_commit_id_prefix_len(commit_id)
|
||||
}
|
||||
|
||||
pub fn resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> {
|
||||
fn resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> {
|
||||
CompositeIndex(self).resolve_prefix(prefix)
|
||||
}
|
||||
|
||||
pub fn entry_by_id(&self, commit_id: &CommitId) -> Option<IndexEntry> {
|
||||
fn entry_by_id(&self, commit_id: &CommitId) -> Option<IndexEntry> {
|
||||
CompositeIndex(self).entry_by_id(commit_id)
|
||||
}
|
||||
|
||||
pub fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry {
|
||||
fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry {
|
||||
CompositeIndex(self).entry_by_pos(pos)
|
||||
}
|
||||
|
||||
pub fn has_id(&self, commit_id: &CommitId) -> bool {
|
||||
fn has_id(&self, commit_id: &CommitId) -> bool {
|
||||
CompositeIndex(self).has_id(commit_id)
|
||||
}
|
||||
|
||||
pub fn is_ancestor(&self, ancestor_id: &CommitId, descendant_id: &CommitId) -> bool {
|
||||
fn is_ancestor(&self, ancestor_id: &CommitId, descendant_id: &CommitId) -> bool {
|
||||
CompositeIndex(self).is_ancestor(ancestor_id, descendant_id)
|
||||
}
|
||||
|
||||
pub fn common_ancestors(&self, set1: &[CommitId], set2: &[CommitId]) -> Vec<CommitId> {
|
||||
fn common_ancestors(&self, set1: &[CommitId], set2: &[CommitId]) -> Vec<CommitId> {
|
||||
CompositeIndex(self).common_ancestors(set1, set2)
|
||||
}
|
||||
|
||||
pub fn walk_revs(&self, wanted: &[CommitId], unwanted: &[CommitId]) -> RevWalk {
|
||||
fn walk_revs(&self, wanted: &[CommitId], unwanted: &[CommitId]) -> RevWalk {
|
||||
CompositeIndex(self).walk_revs(wanted, unwanted)
|
||||
}
|
||||
|
||||
pub fn heads(&self, candidates: &mut dyn Iterator<Item = &CommitId>) -> Vec<CommitId> {
|
||||
fn heads(&self, candidates: &mut dyn Iterator<Item = &CommitId>) -> Vec<CommitId> {
|
||||
CompositeIndex(self).heads(candidates)
|
||||
}
|
||||
|
||||
pub fn topo_order(&self, input: &mut dyn Iterator<Item = &CommitId>) -> Vec<IndexEntry> {
|
||||
fn topo_order(&self, input: &mut dyn Iterator<Item = &CommitId>) -> Vec<IndexEntry> {
|
||||
CompositeIndex(self).topo_order(input)
|
||||
}
|
||||
}
|
||||
|
@ -1598,62 +1628,10 @@ impl ReadonlyIndex {
|
|||
IndexRef::Readonly(self)
|
||||
}
|
||||
|
||||
pub fn num_commits(&self) -> u32 {
|
||||
CompositeIndex(self).num_commits()
|
||||
}
|
||||
|
||||
pub fn name(&self) -> &str {
|
||||
&self.name
|
||||
}
|
||||
|
||||
pub fn stats(&self) -> IndexStats {
|
||||
CompositeIndex(self).stats()
|
||||
}
|
||||
|
||||
pub fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||
CompositeIndex(self).commit_id_to_pos(commit_id)
|
||||
}
|
||||
|
||||
pub fn shortest_unique_commit_id_prefix_len(&self, commit_id: &CommitId) -> usize {
|
||||
CompositeIndex(self).shortest_unique_commit_id_prefix_len(commit_id)
|
||||
}
|
||||
|
||||
pub fn resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> {
|
||||
CompositeIndex(self).resolve_prefix(prefix)
|
||||
}
|
||||
|
||||
pub fn entry_by_id(&self, commit_id: &CommitId) -> Option<IndexEntry> {
|
||||
CompositeIndex(self).entry_by_id(commit_id)
|
||||
}
|
||||
|
||||
pub fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry {
|
||||
CompositeIndex(self).entry_by_pos(pos)
|
||||
}
|
||||
|
||||
pub fn has_id(&self, commit_id: &CommitId) -> bool {
|
||||
CompositeIndex(self).has_id(commit_id)
|
||||
}
|
||||
|
||||
pub fn is_ancestor(&self, ancestor_id: &CommitId, descendant_id: &CommitId) -> bool {
|
||||
CompositeIndex(self).is_ancestor(ancestor_id, descendant_id)
|
||||
}
|
||||
|
||||
pub fn common_ancestors(&self, set1: &[CommitId], set2: &[CommitId]) -> Vec<CommitId> {
|
||||
CompositeIndex(self).common_ancestors(set1, set2)
|
||||
}
|
||||
|
||||
pub fn walk_revs(&self, wanted: &[CommitId], unwanted: &[CommitId]) -> RevWalk {
|
||||
CompositeIndex(self).walk_revs(wanted, unwanted)
|
||||
}
|
||||
|
||||
pub fn heads(&self, candidates: &mut dyn Iterator<Item = &CommitId>) -> Vec<CommitId> {
|
||||
CompositeIndex(self).heads(candidates)
|
||||
}
|
||||
|
||||
pub fn topo_order(&self, input: &mut dyn Iterator<Item = &CommitId>) -> Vec<IndexEntry> {
|
||||
CompositeIndex(self).topo_order(input)
|
||||
}
|
||||
|
||||
fn graph_entry(&self, local_pos: u32) -> CommitGraphEntry {
|
||||
let offset = (local_pos as usize) * self.commit_graph_entry_size;
|
||||
CommitGraphEntry {
|
||||
|
@ -1704,11 +1682,66 @@ impl ReadonlyIndex {
|
|||
}
|
||||
}
|
||||
|
||||
impl Index for ReadonlyIndex {
|
||||
fn num_commits(&self) -> u32 {
|
||||
CompositeIndex(self).num_commits()
|
||||
}
|
||||
|
||||
fn stats(&self) -> IndexStats {
|
||||
CompositeIndex(self).stats()
|
||||
}
|
||||
|
||||
fn commit_id_to_pos(&self, commit_id: &CommitId) -> Option<IndexPosition> {
|
||||
CompositeIndex(self).commit_id_to_pos(commit_id)
|
||||
}
|
||||
|
||||
fn shortest_unique_commit_id_prefix_len(&self, commit_id: &CommitId) -> usize {
|
||||
CompositeIndex(self).shortest_unique_commit_id_prefix_len(commit_id)
|
||||
}
|
||||
|
||||
fn resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> {
|
||||
CompositeIndex(self).resolve_prefix(prefix)
|
||||
}
|
||||
|
||||
fn entry_by_id(&self, commit_id: &CommitId) -> Option<IndexEntry> {
|
||||
CompositeIndex(self).entry_by_id(commit_id)
|
||||
}
|
||||
|
||||
fn entry_by_pos(&self, pos: IndexPosition) -> IndexEntry {
|
||||
CompositeIndex(self).entry_by_pos(pos)
|
||||
}
|
||||
|
||||
fn has_id(&self, commit_id: &CommitId) -> bool {
|
||||
CompositeIndex(self).has_id(commit_id)
|
||||
}
|
||||
|
||||
fn is_ancestor(&self, ancestor_id: &CommitId, descendant_id: &CommitId) -> bool {
|
||||
CompositeIndex(self).is_ancestor(ancestor_id, descendant_id)
|
||||
}
|
||||
|
||||
fn common_ancestors(&self, set1: &[CommitId], set2: &[CommitId]) -> Vec<CommitId> {
|
||||
CompositeIndex(self).common_ancestors(set1, set2)
|
||||
}
|
||||
|
||||
fn walk_revs(&self, wanted: &[CommitId], unwanted: &[CommitId]) -> RevWalk {
|
||||
CompositeIndex(self).walk_revs(wanted, unwanted)
|
||||
}
|
||||
|
||||
fn heads(&self, candidates: &mut dyn Iterator<Item = &CommitId>) -> Vec<CommitId> {
|
||||
CompositeIndex(self).heads(candidates)
|
||||
}
|
||||
|
||||
fn topo_order(&self, input: &mut dyn Iterator<Item = &CommitId>) -> Vec<IndexEntry> {
|
||||
CompositeIndex(self).topo_order(input)
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use test_case::test_case;
|
||||
|
||||
use super::*;
|
||||
use crate::index::Index;
|
||||
|
||||
/// Generator of unique 16-byte ChangeId excluding root id
|
||||
fn change_id_generator() -> impl FnMut() -> ChangeId {
|
||||
|
@ -1721,12 +1754,11 @@ mod tests {
|
|||
fn index_empty(on_disk: bool) {
|
||||
let temp_dir = testutils::new_temp_dir();
|
||||
let index = MutableIndex::full(3, 16);
|
||||
let mut _saved_index = None;
|
||||
let index = if on_disk {
|
||||
_saved_index = Some(index.save_in(temp_dir.path().to_owned()).unwrap());
|
||||
IndexRef::Readonly(_saved_index.as_ref().unwrap())
|
||||
let index: Box<dyn Index> = if on_disk {
|
||||
let saved_index = index.save_in(temp_dir.path().to_owned()).unwrap();
|
||||
Box::new(Arc::try_unwrap(saved_index).unwrap())
|
||||
} else {
|
||||
IndexRef::Mutable(&index)
|
||||
Box::new(index)
|
||||
};
|
||||
|
||||
// Stats are as expected
|
||||
|
@ -1752,12 +1784,11 @@ mod tests {
|
|||
let id_0 = CommitId::from_hex("000000");
|
||||
let change_id0 = new_change_id();
|
||||
index.add_commit_data(id_0.clone(), change_id0.clone(), &[]);
|
||||
let mut _saved_index = None;
|
||||
let index = if on_disk {
|
||||
_saved_index = Some(index.save_in(temp_dir.path().to_owned()).unwrap());
|
||||
IndexRef::Readonly(_saved_index.as_ref().unwrap())
|
||||
let index: Box<dyn Index> = if on_disk {
|
||||
let saved_index = index.save_in(temp_dir.path().to_owned()).unwrap();
|
||||
Box::new(Arc::try_unwrap(saved_index).unwrap())
|
||||
} else {
|
||||
IndexRef::Mutable(&index)
|
||||
Box::new(index)
|
||||
};
|
||||
|
||||
// Stats are as expected
|
||||
|
@ -1834,12 +1865,11 @@ mod tests {
|
|||
index.add_commit_data(id_3.clone(), change_id3.clone(), &[id_2.clone()]);
|
||||
index.add_commit_data(id_4.clone(), change_id4, &[id_1.clone()]);
|
||||
index.add_commit_data(id_5.clone(), change_id5, &[id_4.clone(), id_2.clone()]);
|
||||
let mut _saved_index = None;
|
||||
let index = if on_disk {
|
||||
_saved_index = Some(index.save_in(temp_dir.path().to_owned()).unwrap());
|
||||
IndexRef::Readonly(_saved_index.as_ref().unwrap())
|
||||
let index: Box<dyn Index> = if on_disk {
|
||||
let saved_index = index.save_in(temp_dir.path().to_owned()).unwrap();
|
||||
Box::new(Arc::try_unwrap(saved_index).unwrap())
|
||||
} else {
|
||||
IndexRef::Mutable(&index)
|
||||
Box::new(index)
|
||||
};
|
||||
|
||||
// Stats are as expected
|
||||
|
@ -1925,12 +1955,11 @@ mod tests {
|
|||
new_change_id(),
|
||||
&[id_1, id_2, id_3, id_4, id_5],
|
||||
);
|
||||
let mut _saved_index = None;
|
||||
let index = if on_disk {
|
||||
_saved_index = Some(index.save_in(temp_dir.path().to_owned()).unwrap());
|
||||
IndexRef::Readonly(_saved_index.as_ref().unwrap())
|
||||
let index: Box<dyn Index> = if on_disk {
|
||||
let saved_index = index.save_in(temp_dir.path().to_owned()).unwrap();
|
||||
Box::new(Arc::try_unwrap(saved_index).unwrap())
|
||||
} else {
|
||||
IndexRef::Mutable(&index)
|
||||
Box::new(index)
|
||||
};
|
||||
|
||||
// Stats are as expected
|
||||
|
|
|
@ -26,7 +26,7 @@ use crate::backend::CommitId;
|
|||
use crate::commit::Commit;
|
||||
use crate::dag_walk;
|
||||
use crate::file_util::persist_content_addressed_temp_file;
|
||||
use crate::index::{IndexLoadError, MutableIndex, ReadonlyIndex};
|
||||
use crate::index::{Index, IndexLoadError, MutableIndex, ReadonlyIndex};
|
||||
use crate::op_store::OperationId;
|
||||
use crate::operation::Operation;
|
||||
use crate::store::Store;
|
||||
|
|
|
@ -30,7 +30,8 @@ use crate::commit_builder::CommitBuilder;
|
|||
use crate::dag_walk::topo_order_reverse;
|
||||
use crate::git_backend::GitBackend;
|
||||
use crate::index::{
|
||||
HexPrefix, IndexEntry, IndexPosition, IndexRef, MutableIndex, PrefixResolution, ReadonlyIndex,
|
||||
HexPrefix, Index, IndexEntry, IndexPosition, IndexRef, MutableIndex, PrefixResolution,
|
||||
ReadonlyIndex,
|
||||
};
|
||||
use crate::index_store::IndexStore;
|
||||
use crate::local_backend::LocalBackend;
|
||||
|
|
|
@ -2234,7 +2234,7 @@ fn has_diff_from_parent(repo: RepoRef<'_>, entry: &IndexEntry<'_>, matcher: &dyn
|
|||
mod tests {
|
||||
use super::*;
|
||||
use crate::backend::ChangeId;
|
||||
use crate::index::MutableIndex;
|
||||
use crate::index::{Index, MutableIndex};
|
||||
|
||||
/// Generator of unique 16-byte ChangeId excluding root id
|
||||
fn change_id_generator() -> impl FnMut() -> ChangeId {
|
||||
|
|
|
@ -19,6 +19,7 @@ use itertools::{process_results, Itertools};
|
|||
use crate::backend::{BackendError, BackendResult, CommitId, ObjectId};
|
||||
use crate::commit::Commit;
|
||||
use crate::dag_walk;
|
||||
use crate::index::Index;
|
||||
use crate::op_store::RefTarget;
|
||||
use crate::repo::{MutableRepo, RepoRef};
|
||||
use crate::repo_path::RepoPath;
|
||||
|
|
|
@ -17,7 +17,7 @@ use std::sync::Arc;
|
|||
use jujutsu_lib::backend::CommitId;
|
||||
use jujutsu_lib::commit::Commit;
|
||||
use jujutsu_lib::commit_builder::CommitBuilder;
|
||||
use jujutsu_lib::index::ReadonlyIndex;
|
||||
use jujutsu_lib::index::{Index, ReadonlyIndex};
|
||||
use jujutsu_lib::repo::{MutableRepo, ReadonlyRepo, StoreFactories};
|
||||
use jujutsu_lib::settings::UserSettings;
|
||||
use test_case::test_case;
|
||||
|
|
|
@ -12,6 +12,7 @@
|
|||
// See the License for the specific language governing permissions and
|
||||
// limitations under the License.
|
||||
|
||||
use jujutsu_lib::index::Index;
|
||||
use jujutsu_lib::op_store::{RefTarget, WorkspaceId};
|
||||
use maplit::hashset;
|
||||
use test_case::test_case;
|
||||
|
|
|
@ -13,6 +13,7 @@
|
|||
// limitations under the License.
|
||||
|
||||
use itertools::Itertools;
|
||||
use jujutsu_lib::index::Index;
|
||||
use jujutsu_lib::revset::revset_for_commits;
|
||||
use jujutsu_lib::revset_graph_iterator::RevsetGraphEdge;
|
||||
use test_case::test_case;
|
||||
|
|
|
@ -33,6 +33,7 @@ use jujutsu_lib::commit::Commit;
|
|||
use jujutsu_lib::git::{GitExportError, GitImportError};
|
||||
use jujutsu_lib::gitignore::GitIgnoreFile;
|
||||
use jujutsu_lib::hex_util::to_reverse_hex;
|
||||
use jujutsu_lib::index::Index;
|
||||
use jujutsu_lib::matchers::{EverythingMatcher, Matcher, PrefixMatcher, Visit};
|
||||
use jujutsu_lib::op_heads_store::{self, OpHeadResolutionError, OpHeadsStore};
|
||||
use jujutsu_lib::op_store::{OpStore, OpStoreError, OperationId, RefTarget, WorkspaceId};
|
||||
|
|
|
@ -3,6 +3,7 @@ use std::collections::BTreeSet;
|
|||
use clap::builder::NonEmptyStringValueParser;
|
||||
use itertools::Itertools;
|
||||
use jujutsu_lib::backend::{CommitId, ObjectId};
|
||||
use jujutsu_lib::index::Index;
|
||||
use jujutsu_lib::op_store::RefTarget;
|
||||
use jujutsu_lib::repo::RepoRef;
|
||||
use jujutsu_lib::view::View;
|
||||
|
|
|
@ -10,6 +10,7 @@ use clap::{ArgGroup, Subcommand};
|
|||
use itertools::Itertools;
|
||||
use jujutsu_lib::backend::ObjectId;
|
||||
use jujutsu_lib::git::{self, GitFetchError, GitRefUpdate};
|
||||
use jujutsu_lib::index::Index;
|
||||
use jujutsu_lib::op_store::{BranchTarget, RefTarget};
|
||||
use jujutsu_lib::refs::{classify_branch_push_action, BranchPushAction, BranchPushUpdate};
|
||||
use jujutsu_lib::repo::RepoRef;
|
||||
|
|
|
@ -29,7 +29,7 @@ use itertools::Itertools;
|
|||
use jujutsu_lib::backend::{CommitId, ObjectId, TreeValue};
|
||||
use jujutsu_lib::commit::Commit;
|
||||
use jujutsu_lib::dag_walk::topo_order_reverse;
|
||||
use jujutsu_lib::index::IndexEntry;
|
||||
use jujutsu_lib::index::{Index, IndexEntry};
|
||||
use jujutsu_lib::matchers::EverythingMatcher;
|
||||
use jujutsu_lib::op_store::{RefTarget, WorkspaceId};
|
||||
use jujutsu_lib::repo::ReadonlyRepo;
|
||||
|
|
Loading…
Reference in a new issue