index: make HexPrefix accessor simply return "min" prefix as bytes slice

This is low-level function, so I think using &[u8] should be good here.
This commit is contained in:
Yuya Nishihara 2023-01-26 15:45:23 +09:00
parent 770ca72a1f
commit d771c12637
2 changed files with 19 additions and 21 deletions

View file

@ -310,6 +310,13 @@ impl HexPrefix {
hex_string hex_string
} }
/// Minimum bytes that would match this prefix. (e.g. "abc0" for "abc")
///
/// Use this to partition a sorted slice, and test `matches(id)` from there.
pub fn min_prefix_bytes(&self) -> &[u8] {
&self.min_prefix_bytes
}
fn split_odd_byte(&self) -> (Option<u8>, &[u8]) { fn split_odd_byte(&self) -> (Option<u8>, &[u8]) {
if self.has_odd_byte { if self.has_odd_byte {
let (&odd, prefix) = self.min_prefix_bytes.split_last().unwrap(); let (&odd, prefix) = self.min_prefix_bytes.split_last().unwrap();
@ -319,11 +326,6 @@ impl HexPrefix {
} }
} }
pub fn bytes_prefixes<Q: ObjectId>(&self) -> (Q, Q) {
let (_, prefix) = self.split_odd_byte();
(Q::from_bytes(prefix), Q::from_bytes(&self.min_prefix_bytes))
}
pub fn matches<Q: ObjectId>(&self, id: &Q) -> bool { pub fn matches<Q: ObjectId>(&self, id: &Q) -> bool {
let id_bytes = id.as_bytes(); let id_bytes = id.as_bytes();
let (maybe_odd, prefix) = self.split_odd_byte(); let (maybe_odd, prefix) = self.split_odd_byte();
@ -1333,7 +1335,7 @@ impl IndexSegment for ReadonlyIndex {
} }
fn segment_resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> { fn segment_resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> {
let (_, min_bytes_prefix) = prefix.bytes_prefixes(); let min_bytes_prefix = CommitId::from_bytes(prefix.min_prefix_bytes());
let lookup_pos = self let lookup_pos = self
.commit_id_byte_prefix_to_lookup_pos(&min_bytes_prefix) .commit_id_byte_prefix_to_lookup_pos(&min_bytes_prefix)
.unwrap_or(self.num_local_commits); .unwrap_or(self.num_local_commits);
@ -1428,7 +1430,7 @@ impl IndexSegment for MutableIndex {
} }
fn segment_resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> { fn segment_resolve_prefix(&self, prefix: &HexPrefix) -> PrefixResolution<CommitId> {
let (_, min_bytes_prefix) = prefix.bytes_prefixes(); let min_bytes_prefix = CommitId::from_bytes(prefix.min_prefix_bytes());
let mut matches = self let mut matches = self
.lookup .lookup
.range((Bound::Included(&min_bytes_prefix), Bound::Unbounded)) .range((Bound::Included(&min_bytes_prefix), Bound::Unbounded))
@ -2598,21 +2600,17 @@ mod tests {
#[test] #[test]
fn test_hex_prefix_prefixes() { fn test_hex_prefix_prefixes() {
let (prefix, min_prefix) = HexPrefix::new("").unwrap().bytes_prefixes::<CommitId>(); let prefix = HexPrefix::new("").unwrap();
assert_eq!(prefix, CommitId::from_hex("")); assert_eq!(prefix.min_prefix_bytes(), b"");
assert_eq!(min_prefix, CommitId::from_hex(""));
let (prefix, min_prefix) = HexPrefix::new("1").unwrap().bytes_prefixes::<CommitId>(); let prefix = HexPrefix::new("1").unwrap();
assert_eq!(prefix, CommitId::from_hex("")); assert_eq!(prefix.min_prefix_bytes(), b"\x10");
assert_eq!(min_prefix, CommitId::from_hex("10"));
let (prefix, min_prefix) = HexPrefix::new("12").unwrap().bytes_prefixes::<CommitId>(); let prefix = HexPrefix::new("12").unwrap();
assert_eq!(prefix, CommitId::from_hex("12")); assert_eq!(prefix.min_prefix_bytes(), b"\x12");
assert_eq!(min_prefix, CommitId::from_hex("12"));
let (prefix, min_prefix) = HexPrefix::new("123").unwrap().bytes_prefixes::<CommitId>(); let prefix = HexPrefix::new("123").unwrap();
assert_eq!(prefix, CommitId::from_hex("12")); assert_eq!(prefix.min_prefix_bytes(), b"\x12\x30");
assert_eq!(min_prefix, CommitId::from_hex("1230"));
} }
#[test] #[test]

View file

@ -1306,8 +1306,8 @@ where
&'a self, &'a self,
prefix: &'b HexPrefix, prefix: &'b HexPrefix,
) -> impl Iterator<Item = (&'a K, &'a V)> + 'b { ) -> impl Iterator<Item = (&'a K, &'a V)> + 'b {
let (_, min_bytes_prefix) = prefix.bytes_prefixes::<K>(); let min_bytes = prefix.min_prefix_bytes();
let pos = self.0.partition_point(|(k, _)| k < &min_bytes_prefix); let pos = self.0.partition_point(|(k, _)| k.as_bytes() < min_bytes);
self.0[pos..] self.0[pos..]
.iter() .iter()
.take_while(|(k, _)| prefix.matches(k)) .take_while(|(k, _)| prefix.matches(k))