op_store: add special case for root id resolution

This commit is contained in:
Yuya Nishihara 2024-01-14 17:16:15 +09:00
parent 660806ffed
commit fb3e006a45
2 changed files with 22 additions and 4 deletions

View file

@ -182,10 +182,11 @@ impl OpStore for SimpleOpStore {
) -> OpStoreResult<PrefixResolution<OperationId>> { ) -> OpStoreResult<PrefixResolution<OperationId>> {
let op_dir = self.path.join("operations"); let op_dir = self.path.join("operations");
let find = || -> io::Result<_> { let find = || -> io::Result<_> {
let matches_root = prefix.matches(&self.root_operation_id);
let hex_prefix = prefix.hex(); let hex_prefix = prefix.hex();
if hex_prefix.len() == OPERATION_ID_LENGTH * 2 { if hex_prefix.len() == OPERATION_ID_LENGTH * 2 {
// Fast path for full-length ID // Fast path for full-length ID
if op_dir.join(hex_prefix).try_exists()? { if matches_root || op_dir.join(hex_prefix).try_exists()? {
let id = OperationId::from_bytes(prefix.as_full_bytes().unwrap()); let id = OperationId::from_bytes(prefix.as_full_bytes().unwrap());
return Ok(PrefixResolution::SingleMatch(id)); return Ok(PrefixResolution::SingleMatch(id));
} else { } else {
@ -193,7 +194,7 @@ impl OpStore for SimpleOpStore {
} }
} }
let mut matched = None; let mut matched = matches_root.then(|| self.root_operation_id.clone());
for entry in op_dir.read_dir()? { for entry in op_dir.read_dir()? {
let Ok(name) = entry?.file_name().into_string() else { let Ok(name) = entry?.file_name().into_string() else {
continue; // Skip invalid UTF-8 continue; // Skip invalid UTF-8

View file

@ -434,14 +434,15 @@ fn test_resolve_op_id() {
let settings = stable_op_id_settings(); let settings = stable_op_id_settings();
let test_repo = TestRepo::init_with_settings(&settings); let test_repo = TestRepo::init_with_settings(&settings);
let repo = test_repo.repo; let repo = test_repo.repo;
let op_store = repo.op_store();
let mut operations = Vec::new(); let mut operations = Vec::new();
for i in 0..6 { for i in (0..6).chain([16]) {
let tx = repo.start_transaction(&settings); let tx = repo.start_transaction(&settings);
let repo = tx.commit(format!("transaction {i}")); let repo = tx.commit(format!("transaction {i}"));
operations.push(repo.operation().clone()); operations.push(repo.operation().clone());
} }
// "c" is ambiguous // "c" and "0" are ambiguous
insta::assert_debug_snapshot!(operations.iter().map(|op| op.id().hex()).collect_vec(), @r###" insta::assert_debug_snapshot!(operations.iter().map(|op| op.id().hex()).collect_vec(), @r###"
[ [
"4ff2007de55a2f649f7ab0c98618e4126ef49f0d40a086c8e0a4612a0d5ab4992e1baf4b4fa0a2a224fab39fc5e5b200ac4cddf964db29c6be1379ab2b6d4572", "4ff2007de55a2f649f7ab0c98618e4126ef49f0d40a086c8e0a4612a0d5ab4992e1baf4b4fa0a2a224fab39fc5e5b200ac4cddf964db29c6be1379ab2b6d4572",
@ -450,6 +451,7 @@ fn test_resolve_op_id() {
"cdb35f2826be9a561ae452f06a86e020feec43419d38406f731190732fe143bd69b0e8496ee23817ce13ff6abf9202ec3279b9cb21222be89d5592faa779ff6c", "cdb35f2826be9a561ae452f06a86e020feec43419d38406f731190732fe143bd69b0e8496ee23817ce13ff6abf9202ec3279b9cb21222be89d5592faa779ff6c",
"19971a76da2927c916c079813a0e1e8d91fab52065f926018973b6b5e9d0a22cfdefe3d937ed3ec8e323074a50f6e747a2d6cee0e95185980594ffda8c438a84", "19971a76da2927c916c079813a0e1e8d91fab52065f926018973b6b5e9d0a22cfdefe3d937ed3ec8e323074a50f6e747a2d6cee0e95185980594ffda8c438a84",
"689a23c147a58d70a6f30005d64e49e68fe96a2e9143d78b5957bf26fd9cf06d218279430d7c87b6c5ba163f1557fe2f3b951f0ad126bbe1b804e992c590616a", "689a23c147a58d70a6f30005d64e49e68fe96a2e9143d78b5957bf26fd9cf06d218279430d7c87b6c5ba163f1557fe2f3b951f0ad126bbe1b804e992c590616a",
"09baae9e4e6aeaa9a6107ec4b2a6e86350dbefc227d962983a32631e8f9c65a5cdd73a7420f79eb139ee10f01cf9fcab579abc4596f1844499fdeed75a382db6",
] ]
"###); "###);
@ -489,6 +491,21 @@ fn test_resolve_op_id() {
OpsetResolutionError::NoSuchOperation(_) OpsetResolutionError::NoSuchOperation(_)
)) ))
); );
// Virtual root id
let root_operation = {
let id = op_store.root_operation_id();
let data = op_store.read_operation(id).unwrap();
Operation::new(op_store.clone(), id.clone(), data)
};
assert_eq!(resolve(&root_operation.id().hex()).unwrap(), root_operation);
assert_eq!(resolve("00").unwrap(), root_operation);
assert_eq!(resolve("09").unwrap(), operations[6]);
assert_matches!(
resolve("0"),
Err(OpsetEvaluationError::OpsetResolution(
OpsetResolutionError::AmbiguousIdPrefix(_)
))
);
} }
#[test] #[test]