mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-15 00:44:33 +00:00
revset: allow resolving change id using hex digits from reverse alphabet
By separating the value spaces change ids and commit ids, we can simplify lookup of a prefix. For example, if we know that a prefix is for a change id, we don't have to try to find matching commit ids. I think it might also help new users more quickly understand that change ids are not commit ids. This commit is a step towards that separation. It allows resolving change ids by using hex digits from the back of the alphabet instead of 0-f, so 'z'='0', 'y'='1', etc, and 'k'='f'. Thanks to @ilyagr for the idea. The regular hex digits are still allowed.
This commit is contained in:
parent
b44148871a
commit
39640cc288
4 changed files with 108 additions and 1 deletions
80
lib/src/hex_util.rs
Normal file
80
lib/src/hex_util.rs
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
// Copyright 2023 The Jujutsu Authors
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
fn to_reverse_hex_digit(b: u8) -> Option<u8> {
|
||||||
|
let value = match b {
|
||||||
|
b'0'..=b'9' => b - b'0',
|
||||||
|
b'A'..=b'F' => b - b'A' + 10,
|
||||||
|
b'a'..=b'f' => b - b'a' + 10,
|
||||||
|
_ => return None,
|
||||||
|
};
|
||||||
|
Some(b'z' - value)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn to_forward_hex_digit(b: u8) -> Option<u8> {
|
||||||
|
let value = match b {
|
||||||
|
b'k'..=b'z' => b'z' - b,
|
||||||
|
b'K'..=b'Z' => b'Z' - b,
|
||||||
|
_ => return None,
|
||||||
|
};
|
||||||
|
if value < 10 {
|
||||||
|
Some(b'0' + value)
|
||||||
|
} else {
|
||||||
|
Some(b'a' + value - 10)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn to_forward_hex(reverse_hex: &str) -> Option<String> {
|
||||||
|
reverse_hex
|
||||||
|
.bytes()
|
||||||
|
.map(|b| to_forward_hex_digit(b).map(char::from))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn to_reverse_hex(forward_hex: &str) -> Option<String> {
|
||||||
|
forward_hex
|
||||||
|
.bytes()
|
||||||
|
.map(|b| to_reverse_hex_digit(b).map(char::from))
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_reverse_hex() {
|
||||||
|
// Empty string
|
||||||
|
assert_eq!(to_reverse_hex(""), Some("".to_string()));
|
||||||
|
assert_eq!(to_forward_hex(""), Some("".to_string()));
|
||||||
|
|
||||||
|
// Single digit
|
||||||
|
assert_eq!(to_reverse_hex("0"), Some("z".to_string()));
|
||||||
|
assert_eq!(to_forward_hex("z"), Some("0".to_string()));
|
||||||
|
|
||||||
|
// All digits
|
||||||
|
assert_eq!(
|
||||||
|
to_reverse_hex("0123456789abcdefABCDEF"),
|
||||||
|
Some("zyxwvutsrqponmlkponmlk".to_string())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
to_forward_hex("zyxwvutsrqponmlkPONMLK"),
|
||||||
|
Some("0123456789abcdefabcdef".to_string())
|
||||||
|
);
|
||||||
|
|
||||||
|
// Invalid digit
|
||||||
|
assert_eq!(to_reverse_hex("g"), None);
|
||||||
|
assert_eq!(to_forward_hex("j"), None);
|
||||||
|
}
|
||||||
|
}
|
|
@ -28,6 +28,7 @@ pub mod files;
|
||||||
pub mod git;
|
pub mod git;
|
||||||
pub mod git_backend;
|
pub mod git_backend;
|
||||||
pub mod gitignore;
|
pub mod gitignore;
|
||||||
|
pub mod hex_util;
|
||||||
pub mod index;
|
pub mod index;
|
||||||
pub mod index_store;
|
pub mod index_store;
|
||||||
#[cfg(feature = "legacy-thrift")]
|
#[cfg(feature = "legacy-thrift")]
|
||||||
|
|
|
@ -32,6 +32,7 @@ use thiserror::Error;
|
||||||
|
|
||||||
use crate::backend::{BackendError, BackendResult, CommitId, ObjectId};
|
use crate::backend::{BackendError, BackendResult, CommitId, ObjectId};
|
||||||
use crate::commit::Commit;
|
use crate::commit::Commit;
|
||||||
|
use crate::hex_util::to_forward_hex;
|
||||||
use crate::index::{HexPrefix, IndexEntry, PrefixResolution};
|
use crate::index::{HexPrefix, IndexEntry, PrefixResolution};
|
||||||
use crate::matchers::{EverythingMatcher, Matcher, PrefixMatcher};
|
use crate::matchers::{EverythingMatcher, Matcher, PrefixMatcher};
|
||||||
use crate::op_store::WorkspaceId;
|
use crate::op_store::WorkspaceId;
|
||||||
|
@ -119,7 +120,8 @@ fn resolve_short_commit_id(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn resolve_change_id(repo: RepoRef, symbol: &str) -> Result<Option<Vec<CommitId>>, RevsetError> {
|
fn resolve_change_id(repo: RepoRef, symbol: &str) -> Result<Option<Vec<CommitId>>, RevsetError> {
|
||||||
if let Some(prefix) = HexPrefix::new(symbol) {
|
let forward_hex = to_forward_hex(symbol);
|
||||||
|
if let Some(prefix) = HexPrefix::new(forward_hex.as_deref().unwrap_or(symbol)) {
|
||||||
match repo.resolve_change_id_prefix(&prefix) {
|
match repo.resolve_change_id_prefix(&prefix) {
|
||||||
PrefixResolution::NoMatch => Ok(None),
|
PrefixResolution::NoMatch => Ok(None),
|
||||||
PrefixResolution::AmbiguousMatch => {
|
PrefixResolution::AmbiguousMatch => {
|
||||||
|
|
|
@ -257,12 +257,24 @@ fn test_resolve_symbol_change_id(readonly: bool) {
|
||||||
"8fd68d104372910e19511df709e5dde62a548720"
|
"8fd68d104372910e19511df709e5dde62a548720"
|
||||||
)]
|
)]
|
||||||
);
|
);
|
||||||
|
assert_eq!(
|
||||||
|
resolve_symbol(repo_ref, "zvlyx", None).unwrap(),
|
||||||
|
vec![CommitId::from_hex(
|
||||||
|
"8fd68d104372910e19511df709e5dde62a548720"
|
||||||
|
)]
|
||||||
|
);
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
resolve_symbol(repo_ref, "04e1c", None).unwrap(),
|
resolve_symbol(repo_ref, "04e1c", None).unwrap(),
|
||||||
vec![CommitId::from_hex(
|
vec![CommitId::from_hex(
|
||||||
"e2ad9d861d0ee625851b8ecfcf2c727410e38720"
|
"e2ad9d861d0ee625851b8ecfcf2c727410e38720"
|
||||||
)]
|
)]
|
||||||
);
|
);
|
||||||
|
assert_eq!(
|
||||||
|
resolve_symbol(repo_ref, "zvlyn", None).unwrap(),
|
||||||
|
vec![CommitId::from_hex(
|
||||||
|
"e2ad9d861d0ee625851b8ecfcf2c727410e38720"
|
||||||
|
)]
|
||||||
|
);
|
||||||
assert_matches!(
|
assert_matches!(
|
||||||
resolve_symbol(repo_ref, "04e1", None),
|
resolve_symbol(repo_ref, "04e1", None),
|
||||||
Err(RevsetError::AmbiguousIdPrefix(s)) if s == "04e1"
|
Err(RevsetError::AmbiguousIdPrefix(s)) if s == "04e1"
|
||||||
|
@ -275,6 +287,10 @@ fn test_resolve_symbol_change_id(readonly: bool) {
|
||||||
resolve_symbol(repo_ref, "04e13", None),
|
resolve_symbol(repo_ref, "04e13", None),
|
||||||
Err(RevsetError::NoSuchRevision(s)) if s == "04e13"
|
Err(RevsetError::NoSuchRevision(s)) if s == "04e13"
|
||||||
);
|
);
|
||||||
|
assert_matches!(
|
||||||
|
resolve_symbol(repo_ref, "zvlyw", None),
|
||||||
|
Err(RevsetError::NoSuchRevision(s)) if s == "zvlyw"
|
||||||
|
);
|
||||||
|
|
||||||
// Test commit/changed id conflicts.
|
// Test commit/changed id conflicts.
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
|
@ -288,6 +304,14 @@ fn test_resolve_symbol_change_id(readonly: bool) {
|
||||||
Err(RevsetError::AmbiguousIdPrefix(s)) if s == "040"
|
Err(RevsetError::AmbiguousIdPrefix(s)) if s == "040"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Can disambiguate by using reverse hex for change id
|
||||||
|
assert_eq!(
|
||||||
|
resolve_symbol(repo_ref, "zvz", None).unwrap(),
|
||||||
|
vec![CommitId::from_hex(
|
||||||
|
"5339432b8e7b90bd3aa1a323db71b8a5c5dcd020"
|
||||||
|
)]
|
||||||
|
);
|
||||||
|
|
||||||
// Test non-hex string
|
// Test non-hex string
|
||||||
assert_matches!(
|
assert_matches!(
|
||||||
resolve_symbol(repo_ref, "foo", None),
|
resolve_symbol(repo_ref, "foo", None),
|
||||||
|
|
Loading…
Reference in a new issue