2022-11-26 23:57:50 +00:00
|
|
|
|
// Copyright 2021 The Jujutsu Authors
|
2021-04-05 05:56:10 +00:00
|
|
|
|
//
|
|
|
|
|
// 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.
|
|
|
|
|
|
2024-07-07 07:23:20 +00:00
|
|
|
|
use std::iter;
|
2022-10-23 04:14:00 +00:00
|
|
|
|
use std::path::Path;
|
|
|
|
|
|
2022-12-31 08:25:40 +00:00
|
|
|
|
use assert_matches::assert_matches;
|
2024-06-08 00:42:15 +00:00
|
|
|
|
use chrono::DateTime;
|
2023-03-14 05:14:06 +00:00
|
|
|
|
use itertools::Itertools;
|
2024-08-22 18:18:15 +00:00
|
|
|
|
use jj_lib::backend::CommitId;
|
|
|
|
|
use jj_lib::backend::MillisSinceEpoch;
|
|
|
|
|
use jj_lib::backend::Signature;
|
|
|
|
|
use jj_lib::backend::Timestamp;
|
2023-06-28 14:12:40 +00:00
|
|
|
|
use jj_lib::commit::Commit;
|
2024-04-05 11:35:31 +00:00
|
|
|
|
use jj_lib::fileset::FilesetExpression;
|
2023-06-28 14:12:40 +00:00
|
|
|
|
use jj_lib::git;
|
|
|
|
|
use jj_lib::git_backend::GitBackend;
|
2024-08-22 18:18:15 +00:00
|
|
|
|
use jj_lib::graph::GraphEdge;
|
|
|
|
|
use jj_lib::graph::ReverseGraphIterator;
|
revset: pass separate repo to disambiguation index
The idea is that the disambiguation index can be loaded from a repo which is
different from the symbol resolution context.
Suppose we add at_operation(op, expr) revset, a symbol inside at_operation()
expression will have to be resolved within that operation, whereas the
disambiguation index is cached globally by WorkspaceCommandHelper. We could
build temporary disambiguation index for each at-op repo, but that would be
complicated implementation-wise, and wouldn't be useful. For example, a query
"x | at_operation(@-, x)" might be resolved to "xy | at_operation(@-, xz)"
if disambiguation index were reloaded for the @- operation. Instead, the
short change ID "x" can be disambiguated to "xy", then resolved to the
corresponding commit IDs at each operation.
2024-09-30 04:14:59 +00:00
|
|
|
|
use jj_lib::id_prefix::IdPrefixContext;
|
2024-01-07 16:35:19 +00:00
|
|
|
|
use jj_lib::object_id::ObjectId;
|
2024-08-22 18:18:15 +00:00
|
|
|
|
use jj_lib::op_store::RefTarget;
|
|
|
|
|
use jj_lib::op_store::RemoteRef;
|
|
|
|
|
use jj_lib::op_store::RemoteRefState;
|
|
|
|
|
use jj_lib::op_store::WorkspaceId;
|
2023-06-28 14:12:40 +00:00
|
|
|
|
use jj_lib::repo::Repo;
|
2024-08-22 18:18:15 +00:00
|
|
|
|
use jj_lib::repo_path::RepoPath;
|
|
|
|
|
use jj_lib::repo_path::RepoPathUiConverter;
|
|
|
|
|
use jj_lib::revset::parse;
|
|
|
|
|
use jj_lib::revset::DefaultSymbolResolver;
|
|
|
|
|
use jj_lib::revset::FailingSymbolResolver;
|
|
|
|
|
use jj_lib::revset::Revset;
|
|
|
|
|
use jj_lib::revset::RevsetAliasesMap;
|
2024-09-19 06:02:10 +00:00
|
|
|
|
use jj_lib::revset::RevsetDiagnostics;
|
2024-08-22 18:18:15 +00:00
|
|
|
|
use jj_lib::revset::RevsetExpression;
|
|
|
|
|
use jj_lib::revset::RevsetExtensions;
|
|
|
|
|
use jj_lib::revset::RevsetFilterPredicate;
|
|
|
|
|
use jj_lib::revset::RevsetParseContext;
|
|
|
|
|
use jj_lib::revset::RevsetResolutionError;
|
|
|
|
|
use jj_lib::revset::RevsetWorkspaceContext;
|
revset: pass separate repo to disambiguation index
The idea is that the disambiguation index can be loaded from a repo which is
different from the symbol resolution context.
Suppose we add at_operation(op, expr) revset, a symbol inside at_operation()
expression will have to be resolved within that operation, whereas the
disambiguation index is cached globally by WorkspaceCommandHelper. We could
build temporary disambiguation index for each at-op repo, but that would be
complicated implementation-wise, and wouldn't be useful. For example, a query
"x | at_operation(@-, x)" might be resolved to "xy | at_operation(@-, xz)"
if disambiguation index were reloaded for the @- operation. Instead, the
short change ID "x" can be disambiguated to "xy", then resolved to the
corresponding commit IDs at each operation.
2024-09-30 04:14:59 +00:00
|
|
|
|
use jj_lib::revset::SymbolResolver;
|
2024-08-22 18:18:15 +00:00
|
|
|
|
use jj_lib::revset::SymbolResolverExtension;
|
2023-06-28 14:12:40 +00:00
|
|
|
|
use jj_lib::settings::GitSettings;
|
|
|
|
|
use jj_lib::workspace::Workspace;
|
2021-04-05 05:56:10 +00:00
|
|
|
|
use test_case::test_case;
|
2024-08-22 18:18:15 +00:00
|
|
|
|
use testutils::create_random_commit;
|
|
|
|
|
use testutils::create_tree;
|
|
|
|
|
use testutils::write_random_commit;
|
|
|
|
|
use testutils::CommitGraphBuilder;
|
|
|
|
|
use testutils::TestRepo;
|
|
|
|
|
use testutils::TestRepoBackend;
|
|
|
|
|
use testutils::TestWorkspace;
|
2021-04-05 05:56:10 +00:00
|
|
|
|
|
2024-04-23 22:11:58 +00:00
|
|
|
|
fn resolve_symbol_with_extensions(
|
|
|
|
|
repo: &dyn Repo,
|
2024-05-03 15:50:50 +00:00
|
|
|
|
extensions: &RevsetExtensions,
|
2024-04-23 22:11:58 +00:00
|
|
|
|
symbol: &str,
|
|
|
|
|
) -> Result<Vec<CommitId>, RevsetResolutionError> {
|
2024-05-03 15:50:50 +00:00
|
|
|
|
let aliases_map = RevsetAliasesMap::default();
|
2024-06-08 00:42:15 +00:00
|
|
|
|
let now = chrono::Local::now();
|
|
|
|
|
let context =
|
|
|
|
|
RevsetParseContext::new(&aliases_map, String::new(), now.into(), extensions, None);
|
2024-09-19 06:02:10 +00:00
|
|
|
|
let expression = parse(&mut RevsetDiagnostics::new(), symbol, &context).unwrap();
|
2023-08-25 01:13:47 +00:00
|
|
|
|
assert_matches!(*expression, RevsetExpression::CommitRef(_));
|
2024-04-23 22:11:58 +00:00
|
|
|
|
let symbol_resolver = DefaultSymbolResolver::new(repo, extensions.symbol_resolvers());
|
2024-11-05 07:34:17 +00:00
|
|
|
|
match expression
|
|
|
|
|
.resolve_user_expression(repo, &symbol_resolver)?
|
|
|
|
|
.as_ref()
|
|
|
|
|
{
|
|
|
|
|
RevsetExpression::Commits(commits) => Ok(commits.clone()),
|
2023-08-25 01:13:47 +00:00
|
|
|
|
expression => panic!("symbol resolved to compound expression: {expression:?}"),
|
|
|
|
|
}
|
2023-05-12 07:07:40 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-04-23 22:11:58 +00:00
|
|
|
|
fn resolve_symbol(repo: &dyn Repo, symbol: &str) -> Result<Vec<CommitId>, RevsetResolutionError> {
|
2024-05-03 15:50:50 +00:00
|
|
|
|
resolve_symbol_with_extensions(repo, &RevsetExtensions::default(), symbol)
|
2024-04-23 22:11:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-03-22 19:47:46 +00:00
|
|
|
|
fn revset_for_commits<'index>(
|
|
|
|
|
repo: &'index dyn Repo,
|
|
|
|
|
commits: &[&Commit],
|
2024-01-08 02:28:23 +00:00
|
|
|
|
) -> Box<dyn Revset + 'index> {
|
2024-04-23 22:11:58 +00:00
|
|
|
|
let symbol_resolver =
|
|
|
|
|
DefaultSymbolResolver::new(repo, &([] as [&Box<dyn SymbolResolverExtension>; 0]));
|
2023-03-22 19:47:46 +00:00
|
|
|
|
RevsetExpression::commits(commits.iter().map(|commit| commit.id().clone()).collect())
|
2023-05-05 22:50:37 +00:00
|
|
|
|
.resolve_user_expression(repo, &symbol_resolver)
|
2023-04-07 10:31:32 +00:00
|
|
|
|
.unwrap()
|
2023-03-22 19:47:46 +00:00
|
|
|
|
.evaluate(repo)
|
|
|
|
|
.unwrap()
|
|
|
|
|
}
|
2023-05-12 07:07:40 +00:00
|
|
|
|
|
2023-05-31 04:07:48 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_resolve_symbol_empty_string() {
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-05-31 04:07:48 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
assert_matches!(
|
2023-08-25 01:13:47 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), r#""""#),
|
2023-05-31 04:07:48 +00:00
|
|
|
|
Err(RevsetResolutionError::EmptyString)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-05 05:56:10 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_resolve_symbol_commit_id() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
// Test only with git so we can get predictable commit ids
|
2023-09-18 04:26:31 +00:00
|
|
|
|
let test_repo = TestRepo::init_with_backend(TestRepoBackend::Git);
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-05 05:56:10 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-05 05:56:10 +00:00
|
|
|
|
let signature = Signature {
|
|
|
|
|
name: "test".to_string(),
|
|
|
|
|
email: "test".to_string(),
|
|
|
|
|
timestamp: Timestamp {
|
|
|
|
|
timestamp: MillisSinceEpoch(0),
|
|
|
|
|
tz_offset: 0,
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let mut commits = vec![];
|
|
|
|
|
for i in &[1, 167, 895] {
|
2022-12-25 16:36:13 +00:00
|
|
|
|
let commit = mut_repo
|
|
|
|
|
.new_commit(
|
|
|
|
|
&settings,
|
|
|
|
|
vec![repo.store().root_commit_id().clone()],
|
2023-08-30 14:57:13 +00:00
|
|
|
|
repo.store().empty_merged_tree_id(),
|
2022-12-25 16:36:13 +00:00
|
|
|
|
)
|
|
|
|
|
.set_description(format!("test {i}"))
|
|
|
|
|
.set_author(signature.clone())
|
|
|
|
|
.set_committer(signature.clone())
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2021-04-05 05:56:10 +00:00
|
|
|
|
commits.push(commit);
|
|
|
|
|
}
|
2024-11-13 23:19:58 +00:00
|
|
|
|
let repo = tx.commit("test").unwrap();
|
2021-04-05 05:56:10 +00:00
|
|
|
|
|
|
|
|
|
// Test the test setup
|
|
|
|
|
assert_eq!(
|
|
|
|
|
commits[0].id().hex(),
|
|
|
|
|
"0454de3cae04c46cda37ba2e8873b4c17ff51dcb"
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
commits[1].id().hex(),
|
|
|
|
|
"045f56cd1b17e8abde86771e2705395dcde6a957"
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
commits[2].id().hex(),
|
|
|
|
|
"0468f7da8de2ce442f512aacf83411d26cd2e0cf"
|
|
|
|
|
);
|
|
|
|
|
|
2023-01-14 07:46:30 +00:00
|
|
|
|
// Change ids should never have prefix "04"
|
|
|
|
|
insta::assert_snapshot!(commits[0].change_id().hex(), @"781199f9d55d18e855a7aa84c5e4b40d");
|
|
|
|
|
insta::assert_snapshot!(commits[1].change_id().hex(), @"a2c96fc88f32e487328f04927f20c4b1");
|
|
|
|
|
insta::assert_snapshot!(commits[2].change_id().hex(), @"4399e4f3123763dfe7d68a2809ecc01b");
|
|
|
|
|
|
2021-04-05 05:56:10 +00:00
|
|
|
|
// Test lookup by full commit id
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), "0454de3cae04c46cda37ba2e8873b4c17ff51dcb",).unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![commits[0].id().clone()]
|
2021-04-05 05:56:10 +00:00
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), "045f56cd1b17e8abde86771e2705395dcde6a957",).unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![commits[1].id().clone()]
|
2021-04-05 05:56:10 +00:00
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), "0468f7da8de2ce442f512aacf83411d26cd2e0cf",).unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![commits[2].id().clone()]
|
2021-04-05 05:56:10 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Test commit id prefix
|
2021-05-30 17:26:23 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), "046").unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![commits[2].id().clone()]
|
2021-05-30 17:26:23 +00:00
|
|
|
|
);
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), "04"),
|
2023-05-31 04:07:48 +00:00
|
|
|
|
Err(RevsetResolutionError::AmbiguousCommitIdPrefix(s)) if s == "04"
|
2021-04-05 05:56:10 +00:00
|
|
|
|
);
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), "040"),
|
2023-06-05 04:35:34 +00:00
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision{name, candidates}) if name == "040" && candidates.is_empty()
|
2021-04-05 05:56:10 +00:00
|
|
|
|
);
|
|
|
|
|
|
2021-04-10 06:51:37 +00:00
|
|
|
|
// Test non-hex string
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo.as_ref(), "foo"),
|
2023-06-05 04:35:34 +00:00
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision{name, candidates}) if name == "foo" && candidates.is_empty()
|
2021-04-10 06:51:37 +00:00
|
|
|
|
);
|
2022-11-07 07:29:35 +00:00
|
|
|
|
|
|
|
|
|
// Test present() suppresses only NoSuchRevision error
|
2023-03-21 06:02:29 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(repo.as_ref(), "present(foo)"), []);
|
2024-04-23 22:11:58 +00:00
|
|
|
|
let symbol_resolver = DefaultSymbolResolver::new(
|
|
|
|
|
repo.as_ref(),
|
|
|
|
|
&([] as [&Box<dyn SymbolResolverExtension>; 0]),
|
|
|
|
|
);
|
2024-05-03 15:50:50 +00:00
|
|
|
|
let aliases_map = RevsetAliasesMap::default();
|
|
|
|
|
let extensions = RevsetExtensions::default();
|
2024-06-08 00:42:15 +00:00
|
|
|
|
let context = RevsetParseContext::new(
|
|
|
|
|
&aliases_map,
|
|
|
|
|
settings.user_email(),
|
|
|
|
|
chrono::Utc::now().fixed_offset().into(),
|
|
|
|
|
&extensions,
|
|
|
|
|
None,
|
|
|
|
|
);
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2024-11-05 07:34:17 +00:00
|
|
|
|
parse(&mut RevsetDiagnostics::new(), "present(04)", &context).unwrap()
|
2024-09-19 06:02:10 +00:00
|
|
|
|
.resolve_user_expression(repo.as_ref(), &symbol_resolver),
|
2023-05-31 04:07:48 +00:00
|
|
|
|
Err(RevsetResolutionError::AmbiguousCommitIdPrefix(s)) if s == "04"
|
2022-11-07 07:29:35 +00:00
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-03-21 06:02:29 +00:00
|
|
|
|
resolve_commit_ids(repo.as_ref(), "present(046)"),
|
2022-11-07 07:29:35 +00:00
|
|
|
|
vec![commits[2].id().clone()]
|
|
|
|
|
);
|
2021-04-05 05:56:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-01-25 08:58:20 +00:00
|
|
|
|
#[test_case(false ; "mutable")]
|
|
|
|
|
#[test_case(true ; "readonly")]
|
|
|
|
|
fn test_resolve_symbol_change_id(readonly: bool) {
|
2022-11-13 06:29:06 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2022-12-13 00:18:19 +00:00
|
|
|
|
let git_settings = GitSettings::default();
|
2021-05-30 06:52:50 +00:00
|
|
|
|
// Test only with git so we can get predictable change ids
|
2023-09-18 04:26:31 +00:00
|
|
|
|
let test_repo = TestRepo::init_with_backend(TestRepoBackend::Git);
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-05-30 06:52:50 +00:00
|
|
|
|
|
2023-05-12 13:05:32 +00:00
|
|
|
|
let git_repo = repo
|
|
|
|
|
.store()
|
|
|
|
|
.backend_impl()
|
|
|
|
|
.downcast_ref::<GitBackend>()
|
|
|
|
|
.unwrap()
|
2023-10-03 09:54:32 +00:00
|
|
|
|
.open_git_repo()
|
|
|
|
|
.unwrap();
|
2021-05-30 06:52:50 +00:00
|
|
|
|
// Add some commits that will end up having change ids with common prefixes
|
|
|
|
|
let empty_tree_id = git_repo.treebuilder(None).unwrap().write().unwrap();
|
|
|
|
|
let git_author = git2::Signature::new(
|
|
|
|
|
"git author",
|
|
|
|
|
"git.author@example.com",
|
|
|
|
|
&git2::Time::new(1000, 60),
|
|
|
|
|
)
|
|
|
|
|
.unwrap();
|
|
|
|
|
let git_committer = git2::Signature::new(
|
|
|
|
|
"git committer",
|
|
|
|
|
"git.committer@example.com",
|
|
|
|
|
&git2::Time::new(2000, -480),
|
|
|
|
|
)
|
|
|
|
|
.unwrap();
|
|
|
|
|
let git_tree = git_repo.find_tree(empty_tree_id).unwrap();
|
|
|
|
|
let mut git_commit_ids = vec![];
|
2022-11-27 09:39:35 +00:00
|
|
|
|
for i in &[133, 664, 840, 5085] {
|
2021-05-30 06:52:50 +00:00
|
|
|
|
let git_commit_id = git_repo
|
|
|
|
|
.commit(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
Some(&format!("refs/heads/bookmark{i}")),
|
2021-05-30 06:52:50 +00:00
|
|
|
|
&git_author,
|
|
|
|
|
&git_committer,
|
2022-12-15 02:30:06 +00:00
|
|
|
|
&format!("test {i}"),
|
2021-05-30 06:52:50 +00:00
|
|
|
|
&git_tree,
|
|
|
|
|
&[],
|
|
|
|
|
)
|
|
|
|
|
.unwrap();
|
|
|
|
|
git_commit_ids.push(git_commit_id);
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
git::import_refs(tx.repo_mut(), &git_settings).unwrap();
|
2021-05-30 06:52:50 +00:00
|
|
|
|
|
|
|
|
|
// Test the test setup
|
|
|
|
|
assert_eq!(
|
2021-11-17 22:20:54 +00:00
|
|
|
|
hex::encode(git_commit_ids[0]),
|
2021-05-30 06:52:50 +00:00
|
|
|
|
// "04e12a5467bba790efb88a9870894ec208b16bf1" reversed
|
|
|
|
|
"8fd68d104372910e19511df709e5dde62a548720"
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2021-11-17 22:20:54 +00:00
|
|
|
|
hex::encode(git_commit_ids[1]),
|
2021-05-30 06:52:50 +00:00
|
|
|
|
// "040b3ba3a51d8edbc4c5855cbd09de71d4c29cca" reversed
|
|
|
|
|
"5339432b8e7b90bd3aa1a323db71b8a5c5dcd020"
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2021-11-17 22:20:54 +00:00
|
|
|
|
hex::encode(git_commit_ids[2]),
|
2021-05-30 06:52:50 +00:00
|
|
|
|
// "04e1c7082e4e34f3f371d8a1a46770b861b9b547" reversed
|
|
|
|
|
"e2ad9d861d0ee625851b8ecfcf2c727410e38720"
|
|
|
|
|
);
|
2022-11-27 09:39:35 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
hex::encode(git_commit_ids[3]),
|
|
|
|
|
// "911d7e52fd5ba04b8f289e14c3d30b52d38c0020" reversed
|
|
|
|
|
"040031cb4ad0cbc3287914f1d205dabf4a7eb889"
|
|
|
|
|
);
|
2021-05-30 06:52:50 +00:00
|
|
|
|
|
2023-02-13 17:52:21 +00:00
|
|
|
|
let _readonly_repo;
|
|
|
|
|
let repo: &dyn Repo = if readonly {
|
2024-11-13 23:19:58 +00:00
|
|
|
|
_readonly_repo = tx.commit("test").unwrap();
|
2023-03-21 06:02:29 +00:00
|
|
|
|
_readonly_repo.as_ref()
|
2023-01-25 08:58:20 +00:00
|
|
|
|
} else {
|
2024-09-07 15:51:02 +00:00
|
|
|
|
tx.repo_mut()
|
2023-01-25 08:58:20 +00:00
|
|
|
|
};
|
|
|
|
|
|
2021-05-30 06:52:50 +00:00
|
|
|
|
// Test lookup by full change id
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvlyxpuvtsoopsqzlkorrpqrszrqvlnx").unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![CommitId::from_hex(
|
2021-05-30 06:52:50 +00:00
|
|
|
|
"8fd68d104372910e19511df709e5dde62a548720"
|
2022-12-31 08:25:40 +00:00
|
|
|
|
)]
|
2021-05-30 06:52:50 +00:00
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvzowopwpuymrlmonvnuruunomzqmlsy").unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![CommitId::from_hex(
|
2021-05-30 06:52:50 +00:00
|
|
|
|
"5339432b8e7b90bd3aa1a323db71b8a5c5dcd020"
|
2022-12-31 08:25:40 +00:00
|
|
|
|
)]
|
2021-05-30 06:52:50 +00:00
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvlynszrxlvlwvkwkwsymrpypvtsszor").unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![CommitId::from_hex(
|
2021-05-30 06:52:50 +00:00
|
|
|
|
"e2ad9d861d0ee625851b8ecfcf2c727410e38720"
|
2022-12-31 08:25:40 +00:00
|
|
|
|
)]
|
2021-05-30 06:52:50 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Test change id prefix
|
2023-02-12 18:21:29 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvlyx").unwrap(),
|
2023-02-12 18:21:29 +00:00
|
|
|
|
vec![CommitId::from_hex(
|
|
|
|
|
"8fd68d104372910e19511df709e5dde62a548720"
|
|
|
|
|
)]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvlyn").unwrap(),
|
2023-02-12 18:21:29 +00:00
|
|
|
|
vec![CommitId::from_hex(
|
|
|
|
|
"e2ad9d861d0ee625851b8ecfcf2c727410e38720"
|
|
|
|
|
)]
|
|
|
|
|
);
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvly"),
|
2023-05-31 04:07:48 +00:00
|
|
|
|
Err(RevsetResolutionError::AmbiguousChangeIdPrefix(s)) if s == "zvly"
|
2021-05-30 06:52:50 +00:00
|
|
|
|
);
|
2023-02-12 18:21:29 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvlyw"),
|
2023-06-05 04:35:34 +00:00
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision{name, candidates}) if name == "zvlyw" && candidates.is_empty()
|
2023-02-12 18:21:29 +00:00
|
|
|
|
);
|
2022-11-27 09:39:35 +00:00
|
|
|
|
|
2023-02-13 04:21:34 +00:00
|
|
|
|
// Test that commit and changed id don't conflict ("040" and "zvz" are the
|
|
|
|
|
// same).
|
2022-11-27 09:39:35 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "040").unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![CommitId::from_hex(
|
2023-02-13 04:21:34 +00:00
|
|
|
|
"040031cb4ad0cbc3287914f1d205dabf4a7eb889"
|
2022-12-31 08:25:40 +00:00
|
|
|
|
)]
|
2022-11-27 09:39:35 +00:00
|
|
|
|
);
|
2023-02-12 18:21:29 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "zvz").unwrap(),
|
2023-02-12 18:21:29 +00:00
|
|
|
|
vec![CommitId::from_hex(
|
|
|
|
|
"5339432b8e7b90bd3aa1a323db71b8a5c5dcd020"
|
|
|
|
|
)]
|
|
|
|
|
);
|
|
|
|
|
|
2021-05-30 06:52:50 +00:00
|
|
|
|
// Test non-hex string
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(repo, "foo"),
|
2023-06-05 04:35:34 +00:00
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision{
|
|
|
|
|
name,
|
|
|
|
|
candidates
|
|
|
|
|
}) if name == "foo" && candidates.is_empty()
|
2021-05-30 06:52:50 +00:00
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
revset: pass separate repo to disambiguation index
The idea is that the disambiguation index can be loaded from a repo which is
different from the symbol resolution context.
Suppose we add at_operation(op, expr) revset, a symbol inside at_operation()
expression will have to be resolved within that operation, whereas the
disambiguation index is cached globally by WorkspaceCommandHelper. We could
build temporary disambiguation index for each at-op repo, but that would be
complicated implementation-wise, and wouldn't be useful. For example, a query
"x | at_operation(@-, x)" might be resolved to "xy | at_operation(@-, xz)"
if disambiguation index were reloaded for the @- operation. Instead, the
short change ID "x" can be disambiguated to "xy", then resolved to the
corresponding commit IDs at each operation.
2024-09-30 04:14:59 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_resolve_symbol_in_different_disambiguation_context() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo0 = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo0.start_transaction(&settings);
|
|
|
|
|
let commit1 = write_random_commit(tx.repo_mut(), &settings);
|
|
|
|
|
// Create more commits that are likely to conflict with 1-char hex prefix.
|
|
|
|
|
for _ in 0..50 {
|
|
|
|
|
write_random_commit(tx.repo_mut(), &settings);
|
|
|
|
|
}
|
2024-11-13 23:19:58 +00:00
|
|
|
|
let repo1 = tx.commit("test").unwrap();
|
revset: pass separate repo to disambiguation index
The idea is that the disambiguation index can be loaded from a repo which is
different from the symbol resolution context.
Suppose we add at_operation(op, expr) revset, a symbol inside at_operation()
expression will have to be resolved within that operation, whereas the
disambiguation index is cached globally by WorkspaceCommandHelper. We could
build temporary disambiguation index for each at-op repo, but that would be
complicated implementation-wise, and wouldn't be useful. For example, a query
"x | at_operation(@-, x)" might be resolved to "xy | at_operation(@-, xz)"
if disambiguation index were reloaded for the @- operation. Instead, the
short change ID "x" can be disambiguated to "xy", then resolved to the
corresponding commit IDs at each operation.
2024-09-30 04:14:59 +00:00
|
|
|
|
|
|
|
|
|
let mut tx = repo1.start_transaction(&settings);
|
|
|
|
|
let commit2 = tx
|
|
|
|
|
.repo_mut()
|
|
|
|
|
.rewrite_commit(&settings, &commit1)
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
tx.repo_mut().rebase_descendants(&settings).unwrap();
|
2024-11-13 23:19:58 +00:00
|
|
|
|
let repo2 = tx.commit("test").unwrap();
|
revset: pass separate repo to disambiguation index
The idea is that the disambiguation index can be loaded from a repo which is
different from the symbol resolution context.
Suppose we add at_operation(op, expr) revset, a symbol inside at_operation()
expression will have to be resolved within that operation, whereas the
disambiguation index is cached globally by WorkspaceCommandHelper. We could
build temporary disambiguation index for each at-op repo, but that would be
complicated implementation-wise, and wouldn't be useful. For example, a query
"x | at_operation(@-, x)" might be resolved to "xy | at_operation(@-, xz)"
if disambiguation index were reloaded for the @- operation. Instead, the
short change ID "x" can be disambiguated to "xy", then resolved to the
corresponding commit IDs at each operation.
2024-09-30 04:14:59 +00:00
|
|
|
|
|
|
|
|
|
// Set up disambiguation index which only contains the commit2.id().
|
|
|
|
|
let id_prefix_context = IdPrefixContext::new(Default::default())
|
|
|
|
|
.disambiguate_within(RevsetExpression::commit(commit2.id().clone()));
|
|
|
|
|
let symbol_resolver =
|
|
|
|
|
DefaultSymbolResolver::new(repo2.as_ref(), &[] as &[Box<dyn SymbolResolverExtension>])
|
|
|
|
|
.with_id_prefix_context(&id_prefix_context);
|
|
|
|
|
|
|
|
|
|
// Sanity check
|
2024-10-15 08:29:00 +00:00
|
|
|
|
let change_hex = commit2.change_id().reverse_hex();
|
revset: pass separate repo to disambiguation index
The idea is that the disambiguation index can be loaded from a repo which is
different from the symbol resolution context.
Suppose we add at_operation(op, expr) revset, a symbol inside at_operation()
expression will have to be resolved within that operation, whereas the
disambiguation index is cached globally by WorkspaceCommandHelper. We could
build temporary disambiguation index for each at-op repo, but that would be
complicated implementation-wise, and wouldn't be useful. For example, a query
"x | at_operation(@-, x)" might be resolved to "xy | at_operation(@-, xz)"
if disambiguation index were reloaded for the @- operation. Instead, the
short change ID "x" can be disambiguated to "xy", then resolved to the
corresponding commit IDs at each operation.
2024-09-30 04:14:59 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
symbol_resolver
|
|
|
|
|
.resolve_symbol(repo2.as_ref(), &change_hex[0..1])
|
|
|
|
|
.unwrap(),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
symbol_resolver
|
|
|
|
|
.resolve_symbol(repo2.as_ref(), &commit2.id().hex()[0..1])
|
|
|
|
|
.unwrap(),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Change ID is disambiguated within repo2, then resolved in repo1.
|
|
|
|
|
assert_eq!(
|
|
|
|
|
symbol_resolver
|
|
|
|
|
.resolve_symbol(repo1.as_ref(), &change_hex[0..1])
|
|
|
|
|
.unwrap(),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Commit ID can be found in the disambiguation index, but doesn't exist in
|
|
|
|
|
// repo1.
|
|
|
|
|
assert_matches!(
|
|
|
|
|
symbol_resolver.resolve_symbol(repo1.as_ref(), &commit2.id().hex()[0..1]),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { .. })
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_resolve_working_copy() {
|
2021-04-05 05:56:10 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-05 05:56:10 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-05 05:56:10 +00:00
|
|
|
|
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
2021-04-05 05:56:10 +00:00
|
|
|
|
|
2022-02-02 16:15:25 +00:00
|
|
|
|
let ws1 = WorkspaceId::new("ws1".to_string());
|
|
|
|
|
let ws2 = WorkspaceId::new("ws2".to_string());
|
|
|
|
|
|
2023-04-09 23:18:32 +00:00
|
|
|
|
// Cannot resolve a working-copy commit for an unknown workspace
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2023-11-24 18:52:12 +00:00
|
|
|
|
RevsetExpression::working_copy(ws1.clone())
|
|
|
|
|
.resolve_user_expression(mut_repo, &FailingSymbolResolver),
|
2023-04-09 23:18:32 +00:00
|
|
|
|
Err(RevsetResolutionError::WorkspaceMissingWorkingCopy { name }) if name == "ws1"
|
2022-02-02 16:15:25 +00:00
|
|
|
|
);
|
|
|
|
|
|
2024-09-30 11:08:24 +00:00
|
|
|
|
// The error can be suppressed by present()
|
|
|
|
|
assert_eq!(
|
2024-11-02 07:10:36 +00:00
|
|
|
|
RevsetExpression::working_copy(ws1.clone())
|
|
|
|
|
.present()
|
2024-11-04 05:46:00 +00:00
|
|
|
|
.resolve_user_expression(mut_repo, &FailingSymbolResolver)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.evaluate(mut_repo)
|
2024-11-02 07:10:36 +00:00
|
|
|
|
.unwrap()
|
|
|
|
|
.iter()
|
|
|
|
|
.map(Result::unwrap)
|
|
|
|
|
.collect_vec(),
|
2024-09-30 11:08:24 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
|
2022-02-02 16:15:25 +00:00
|
|
|
|
// Add some workspaces
|
2022-10-20 23:33:14 +00:00
|
|
|
|
mut_repo
|
|
|
|
|
.set_wc_commit(ws1.clone(), commit1.id().clone())
|
|
|
|
|
.unwrap();
|
2023-04-09 23:18:32 +00:00
|
|
|
|
mut_repo
|
|
|
|
|
.set_wc_commit(ws2.clone(), commit2.id().clone())
|
|
|
|
|
.unwrap();
|
|
|
|
|
let resolve = |ws_id: WorkspaceId| -> Vec<CommitId> {
|
|
|
|
|
RevsetExpression::working_copy(ws_id)
|
2024-11-04 05:46:00 +00:00
|
|
|
|
.resolve_user_expression(mut_repo, &FailingSymbolResolver)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.evaluate(mut_repo)
|
2023-04-09 23:18:32 +00:00
|
|
|
|
.unwrap()
|
|
|
|
|
.iter()
|
2024-10-11 14:54:45 +00:00
|
|
|
|
.map(Result::unwrap)
|
2023-04-09 23:18:32 +00:00
|
|
|
|
.collect()
|
|
|
|
|
};
|
|
|
|
|
|
2022-02-02 16:15:25 +00:00
|
|
|
|
// Can resolve "@" shorthand with a default workspace ID
|
2023-04-09 23:18:32 +00:00
|
|
|
|
assert_eq!(resolve(ws1), vec![commit1.id().clone()]);
|
2022-02-02 16:15:25 +00:00
|
|
|
|
// Can resolve an explicit checkout
|
2023-04-09 23:18:32 +00:00
|
|
|
|
assert_eq!(resolve(ws2), vec![commit2.id().clone()]);
|
2021-04-05 05:56:10 +00:00
|
|
|
|
}
|
2021-04-10 06:40:52 +00:00
|
|
|
|
|
2024-04-01 01:40:19 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_resolve_working_copies() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-04-01 01:40:19 +00:00
|
|
|
|
|
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
|
|
|
|
|
// Add some workspaces
|
|
|
|
|
let ws1 = WorkspaceId::new("ws1".to_string());
|
|
|
|
|
let ws2 = WorkspaceId::new("ws2".to_string());
|
|
|
|
|
|
|
|
|
|
// add one commit to each working copy
|
|
|
|
|
mut_repo
|
|
|
|
|
.set_wc_commit(ws1.clone(), commit1.id().clone())
|
|
|
|
|
.unwrap();
|
|
|
|
|
mut_repo
|
|
|
|
|
.set_wc_commit(ws2.clone(), commit2.id().clone())
|
|
|
|
|
.unwrap();
|
|
|
|
|
let resolve = || -> Vec<CommitId> {
|
|
|
|
|
RevsetExpression::working_copies()
|
2024-11-04 05:46:00 +00:00
|
|
|
|
.resolve_user_expression(mut_repo, &FailingSymbolResolver)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.evaluate(mut_repo)
|
2024-04-01 01:40:19 +00:00
|
|
|
|
.unwrap()
|
|
|
|
|
.iter()
|
2024-10-11 14:54:45 +00:00
|
|
|
|
.map(Result::unwrap)
|
2024-04-01 01:40:19 +00:00
|
|
|
|
.collect()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ensure our output has those two commits
|
|
|
|
|
assert_eq!(resolve(), vec![commit2.id().clone(), commit1.id().clone()]);
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-02 04:33:52 +00:00
|
|
|
|
#[test]
|
2024-08-21 19:59:15 +00:00
|
|
|
|
fn test_resolve_symbol_bookmarks() {
|
2023-07-02 04:33:52 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-07-02 04:33:52 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2023-10-16 01:04:29 +00:00
|
|
|
|
let new_remote_ref = |target| RemoteRef {
|
2023-10-11 17:15:17 +00:00
|
|
|
|
target,
|
2023-10-16 01:04:29 +00:00
|
|
|
|
state: RemoteRefState::New,
|
2023-10-11 17:15:17 +00:00
|
|
|
|
};
|
2023-10-16 01:04:29 +00:00
|
|
|
|
let tracking_remote_ref = |target| RemoteRef {
|
|
|
|
|
target,
|
|
|
|
|
state: RemoteRefState::Tracking,
|
|
|
|
|
};
|
|
|
|
|
let normal_tracking_remote_ref =
|
|
|
|
|
|id: &CommitId| tracking_remote_ref(RefTarget::normal(id.clone()));
|
2023-07-02 04:33:52 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2023-07-02 04:33:52 +00:00
|
|
|
|
|
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit4 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit5 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_local_bookmark_target("local", RefTarget::normal(commit1.id().clone()));
|
|
|
|
|
mut_repo.set_remote_bookmark("remote", "origin", normal_tracking_remote_ref(commit2.id()));
|
|
|
|
|
mut_repo.set_local_bookmark_target("local-remote", RefTarget::normal(commit3.id().clone()));
|
|
|
|
|
mut_repo.set_remote_bookmark(
|
2023-10-16 01:04:29 +00:00
|
|
|
|
"local-remote",
|
|
|
|
|
"origin",
|
|
|
|
|
normal_tracking_remote_ref(commit4.id()),
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_local_bookmark_target(
|
|
|
|
|
"local-remote@origin", // not a remote bookmark
|
2023-08-24 03:21:39 +00:00
|
|
|
|
RefTarget::normal(commit5.id().clone()),
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
2023-07-11 14:01:54 +00:00
|
|
|
|
"local-remote",
|
|
|
|
|
"mirror",
|
2024-08-21 19:59:15 +00:00
|
|
|
|
tracking_remote_ref(mut_repo.get_local_bookmark("local-remote")),
|
2023-10-16 01:04:29 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
2023-10-16 01:04:29 +00:00
|
|
|
|
"local-remote",
|
|
|
|
|
"untracked",
|
2024-08-21 19:59:15 +00:00
|
|
|
|
new_remote_ref(mut_repo.get_local_bookmark("local-remote")),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
2023-09-25 14:34:09 +00:00
|
|
|
|
"local-remote",
|
|
|
|
|
git::REMOTE_NAME_FOR_LOCAL_GIT_REPO,
|
2024-08-21 19:59:15 +00:00
|
|
|
|
tracking_remote_ref(mut_repo.get_local_bookmark("local-remote")),
|
2023-07-09 09:17:48 +00:00
|
|
|
|
);
|
2023-07-02 04:33:52 +00:00
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_local_bookmark_target(
|
2023-07-11 14:29:45 +00:00
|
|
|
|
"local-conflicted",
|
2023-07-11 15:22:21 +00:00
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit1.id().clone()],
|
|
|
|
|
[commit3.id().clone(), commit2.id().clone()],
|
|
|
|
|
),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
2023-07-11 14:01:54 +00:00
|
|
|
|
"remote-conflicted",
|
|
|
|
|
"origin",
|
2023-10-16 01:04:29 +00:00
|
|
|
|
tracking_remote_ref(RefTarget::from_legacy_form(
|
2023-07-11 15:22:21 +00:00
|
|
|
|
[commit3.id().clone()],
|
|
|
|
|
[commit5.id().clone(), commit4.id().clone()],
|
2023-10-13 14:37:17 +00:00
|
|
|
|
)),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Local only
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local").unwrap(),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
vec![commit1.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local@origin").unwrap_err(), @r###"
|
2023-07-02 04:33:52 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "local@origin",
|
|
|
|
|
candidates: [
|
|
|
|
|
"local",
|
2023-07-09 09:17:48 +00:00
|
|
|
|
"local-remote@git",
|
2024-02-09 21:55:09 +00:00
|
|
|
|
"local-remote@mirror",
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"local-remote@origin",
|
|
|
|
|
"remote@origin",
|
2023-07-02 04:33:52 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
|
|
|
|
|
// Remote only (or locally deleted)
|
2023-07-02 00:59:34 +00:00
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "remote").unwrap_err(), @r###"
|
2023-07-02 00:59:34 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "remote",
|
|
|
|
|
candidates: [
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"remote-conflicted@origin",
|
|
|
|
|
"remote@origin",
|
2023-07-02 00:59:34 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
2023-07-02 04:33:52 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "remote@origin").unwrap(),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
vec![commit2.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
|
2023-07-09 09:17:48 +00:00
|
|
|
|
// Local/remote/git
|
2023-07-02 04:33:52 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-remote").unwrap(),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
vec![commit3.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-remote@origin").unwrap(),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
vec![commit4.id().clone()],
|
|
|
|
|
);
|
2023-08-24 03:21:39 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_symbol(mut_repo, r#""local-remote@origin""#).unwrap(),
|
|
|
|
|
vec![commit5.id().clone()],
|
|
|
|
|
);
|
2023-07-02 04:33:52 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-remote@mirror").unwrap(),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
vec![commit3.id().clone()],
|
|
|
|
|
);
|
2023-07-09 09:17:48 +00:00
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-remote@git").unwrap(),
|
2023-07-09 09:17:48 +00:00
|
|
|
|
vec![commit3.id().clone()],
|
|
|
|
|
);
|
2023-07-02 04:33:52 +00:00
|
|
|
|
|
|
|
|
|
// Conflicted
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-conflicted").unwrap(),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "remote-conflicted@origin").unwrap(),
|
2023-07-02 04:33:52 +00:00
|
|
|
|
vec![commit5.id().clone(), commit4.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Typo of local/remote bookmark name:
|
2023-07-09 09:17:48 +00:00
|
|
|
|
// For "local-emote" (without @remote part), "local-remote@mirror"/"@git" aren't
|
2023-10-16 01:04:29 +00:00
|
|
|
|
// suggested since they point to the same target as "local-remote". OTOH,
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// "local-remote@untracked" is suggested because non-tracking bookmark is
|
|
|
|
|
// unrelated to the local bookmark of the same name.
|
2023-07-02 04:33:52 +00:00
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-emote").unwrap_err(), @r###"
|
2023-07-02 04:33:52 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "local-emote",
|
|
|
|
|
candidates: [
|
|
|
|
|
"local",
|
|
|
|
|
"local-conflicted",
|
|
|
|
|
"local-remote",
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"local-remote@origin",
|
2023-10-16 01:04:29 +00:00
|
|
|
|
"local-remote@untracked",
|
2023-07-02 04:33:52 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-emote@origin").unwrap_err(), @r###"
|
2023-07-02 04:33:52 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "local-emote@origin",
|
|
|
|
|
candidates: [
|
|
|
|
|
"local",
|
|
|
|
|
"local-remote",
|
2023-07-09 09:17:48 +00:00
|
|
|
|
"local-remote@git",
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"local-remote@mirror",
|
|
|
|
|
"local-remote@origin",
|
2023-10-16 01:04:29 +00:00
|
|
|
|
"local-remote@untracked",
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"remote-conflicted@origin",
|
|
|
|
|
"remote@origin",
|
2023-07-02 04:33:52 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "local-remote@origine").unwrap_err(), @r###"
|
2023-07-02 04:33:52 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "local-remote@origine",
|
|
|
|
|
candidates: [
|
|
|
|
|
"local",
|
|
|
|
|
"local-remote",
|
2023-07-09 09:17:48 +00:00
|
|
|
|
"local-remote@git",
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"local-remote@mirror",
|
|
|
|
|
"local-remote@origin",
|
2023-10-16 01:04:29 +00:00
|
|
|
|
"local-remote@untracked",
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"remote-conflicted@origin",
|
|
|
|
|
"remote@origin",
|
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
// "local-remote@mirror" shouldn't be omitted just because it points to the same
|
|
|
|
|
// target as "local-remote".
|
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "remote@mirror").unwrap_err(), @r###"
|
2023-07-02 04:57:44 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "remote@mirror",
|
|
|
|
|
candidates: [
|
|
|
|
|
"local-remote@mirror",
|
|
|
|
|
"remote@origin",
|
2023-07-02 04:33:52 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Typo of remote-only bookmark name
|
2023-07-02 04:33:52 +00:00
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "emote").unwrap_err(), @r###"
|
2023-07-02 04:33:52 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "emote",
|
|
|
|
|
candidates: [
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"remote-conflicted@origin",
|
|
|
|
|
"remote@origin",
|
2023-07-02 04:33:52 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "emote@origin").unwrap_err(), @r###"
|
2023-07-02 04:33:52 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "emote@origin",
|
|
|
|
|
candidates: [
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"local-remote@origin",
|
|
|
|
|
"remote@origin",
|
2023-07-02 04:33:52 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
insta::assert_debug_snapshot!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "remote@origine").unwrap_err(), @r###"
|
2023-07-02 04:33:52 +00:00
|
|
|
|
NoSuchRevision {
|
|
|
|
|
name: "remote@origine",
|
|
|
|
|
candidates: [
|
2023-07-02 04:57:44 +00:00
|
|
|
|
"local-remote@origin",
|
|
|
|
|
"remote-conflicted@origin",
|
|
|
|
|
"remote@origin",
|
2023-07-02 04:33:52 +00:00
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
"###);
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-11 04:44:17 +00:00
|
|
|
|
#[test]
|
2023-09-06 01:08:32 +00:00
|
|
|
|
fn test_resolve_symbol_tags() {
|
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-09-06 01:08:32 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2023-09-06 01:08:32 +00:00
|
|
|
|
|
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_tag_target("tag-bookmark", RefTarget::normal(commit1.id().clone()));
|
|
|
|
|
mut_repo.set_local_bookmark_target("tag-bookmark", RefTarget::normal(commit2.id().clone()));
|
2023-09-06 01:08:32 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
|
|
|
|
"refs/tags/unimported",
|
|
|
|
|
RefTarget::normal(commit3.id().clone()),
|
|
|
|
|
);
|
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Tag precedes bookmark
|
2023-09-06 01:08:32 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_symbol(mut_repo, "tag-bookmark").unwrap(),
|
2023-09-06 01:08:32 +00:00
|
|
|
|
vec![commit1.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
|
2023-09-06 01:24:11 +00:00
|
|
|
|
assert_matches!(
|
|
|
|
|
resolve_symbol(mut_repo, "unimported"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { .. })
|
2023-09-06 01:08:32 +00:00
|
|
|
|
);
|
2023-09-10 11:18:02 +00:00
|
|
|
|
|
|
|
|
|
// "@" (quoted) can be resolved, and root is a normal symbol.
|
|
|
|
|
let ws_id = WorkspaceId::default();
|
|
|
|
|
mut_repo
|
|
|
|
|
.set_wc_commit(ws_id.clone(), commit1.id().clone())
|
|
|
|
|
.unwrap();
|
|
|
|
|
mut_repo.set_tag_target("@", RefTarget::normal(commit2.id().clone()));
|
|
|
|
|
mut_repo.set_tag_target("root", RefTarget::normal(commit3.id().clone()));
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_symbol(mut_repo, r#""@""#).unwrap(),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_symbol(mut_repo, "root").unwrap(),
|
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
2023-09-06 01:08:32 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-10 06:40:52 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_resolve_symbol_git_refs() {
|
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-10 06:40:52 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-10 06:40:52 +00:00
|
|
|
|
|
|
|
|
|
// Create some commits and refs to work with and so the repo is not empty
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit4 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit5 = write_random_commit(mut_repo, &settings);
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
"refs/heads/bookmark1",
|
2023-07-11 13:14:59 +00:00
|
|
|
|
RefTarget::normal(commit1.id().clone()),
|
2021-07-11 17:58:01 +00:00
|
|
|
|
);
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
"refs/heads/bookmark2",
|
2023-07-11 13:14:59 +00:00
|
|
|
|
RefTarget::normal(commit2.id().clone()),
|
2021-07-11 17:58:01 +00:00
|
|
|
|
);
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
|
|
|
|
"refs/heads/conflicted",
|
2023-07-11 15:22:21 +00:00
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit2.id().clone()],
|
|
|
|
|
[commit1.id().clone(), commit3.id().clone()],
|
|
|
|
|
),
|
2021-07-11 17:58:01 +00:00
|
|
|
|
);
|
2023-07-11 13:14:59 +00:00
|
|
|
|
mut_repo.set_git_ref_target("refs/tags/tag1", RefTarget::normal(commit2.id().clone()));
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
"refs/tags/remotes/origin/bookmark1",
|
2023-07-11 13:14:59 +00:00
|
|
|
|
RefTarget::normal(commit3.id().clone()),
|
2021-04-10 06:40:52 +00:00
|
|
|
|
);
|
|
|
|
|
|
2022-09-09 17:31:43 +00:00
|
|
|
|
// Nonexistent ref
|
2022-12-31 08:25:40 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "nonexistent"),
|
2023-06-12 18:59:15 +00:00
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision{name, candidates})
|
|
|
|
|
if name == "nonexistent" && candidates.is_empty()
|
2021-04-10 06:40:52 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Full ref
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
|
|
|
|
"refs/heads/bookmark",
|
|
|
|
|
RefTarget::normal(commit4.id().clone()),
|
|
|
|
|
);
|
2021-04-10 06:40:52 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_symbol(mut_repo, "refs/heads/bookmark").unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![commit4.id().clone()]
|
2021-04-10 06:40:52 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Qualified with only heads/
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
|
|
|
|
"refs/heads/bookmark",
|
|
|
|
|
RefTarget::normal(commit5.id().clone()),
|
|
|
|
|
);
|
|
|
|
|
mut_repo.set_git_ref_target(
|
|
|
|
|
"refs/tags/bookmark",
|
|
|
|
|
RefTarget::normal(commit4.id().clone()),
|
|
|
|
|
);
|
|
|
|
|
// bookmark alone is not recognized
|
2023-07-09 09:17:48 +00:00
|
|
|
|
insta::assert_debug_snapshot!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_symbol(mut_repo, "bookmark").unwrap_err(), @r###"
|
2023-07-09 09:17:48 +00:00
|
|
|
|
NoSuchRevision {
|
2024-08-21 19:59:15 +00:00
|
|
|
|
name: "bookmark",
|
2023-09-25 14:34:09 +00:00
|
|
|
|
candidates: [],
|
2023-07-09 09:17:48 +00:00
|
|
|
|
}
|
|
|
|
|
"###);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// heads/bookmark does get resolved to the git ref refs/heads/bookmark
|
2023-06-12 18:59:15 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_symbol(mut_repo, "heads/bookmark").unwrap(),
|
2023-06-12 18:59:15 +00:00
|
|
|
|
vec![commit5.id().clone()]
|
2021-04-10 06:40:52 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Unqualified tag name
|
2023-07-11 13:14:59 +00:00
|
|
|
|
mut_repo.set_git_ref_target("refs/tags/tag", RefTarget::normal(commit4.id().clone()));
|
2023-09-06 01:24:11 +00:00
|
|
|
|
assert_matches!(
|
|
|
|
|
resolve_symbol(mut_repo, "tag"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { .. })
|
2021-04-10 06:40:52 +00:00
|
|
|
|
);
|
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Unqualified remote-tracking bookmark name
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
"refs/remotes/origin/remote-bookmark",
|
2023-07-11 13:14:59 +00:00
|
|
|
|
RefTarget::normal(commit2.id().clone()),
|
2021-04-10 06:40:52 +00:00
|
|
|
|
);
|
2023-09-06 01:24:11 +00:00
|
|
|
|
assert_matches!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_symbol(mut_repo, "origin/remote-bookmark"),
|
2023-09-06 01:24:11 +00:00
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { .. })
|
2021-04-10 06:40:52 +00:00
|
|
|
|
);
|
|
|
|
|
|
2021-07-11 17:58:01 +00:00
|
|
|
|
// Conflicted ref resolves to its "adds"
|
|
|
|
|
assert_eq!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "refs/heads/conflicted").unwrap(),
|
2022-12-31 08:25:40 +00:00
|
|
|
|
vec![commit1.id().clone(), commit3.id().clone()]
|
2021-07-11 17:58:01 +00:00
|
|
|
|
);
|
2021-04-10 06:40:52 +00:00
|
|
|
|
}
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
2023-02-13 17:52:21 +00:00
|
|
|
|
fn resolve_commit_ids(repo: &dyn Repo, revset_str: &str) -> Vec<CommitId> {
|
2024-09-28 10:20:06 +00:00
|
|
|
|
try_resolve_commit_ids(repo, revset_str).unwrap()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn try_resolve_commit_ids(
|
|
|
|
|
repo: &dyn Repo,
|
|
|
|
|
revset_str: &str,
|
|
|
|
|
) -> Result<Vec<CommitId>, RevsetResolutionError> {
|
2023-08-14 00:35:11 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2024-05-03 15:50:50 +00:00
|
|
|
|
let aliases_map = RevsetAliasesMap::default();
|
|
|
|
|
let revset_extensions = RevsetExtensions::default();
|
|
|
|
|
let context = RevsetParseContext::new(
|
|
|
|
|
&aliases_map,
|
|
|
|
|
settings.user_email(),
|
2024-06-08 00:42:15 +00:00
|
|
|
|
chrono::Utc::now().fixed_offset().into(),
|
2024-05-03 15:50:50 +00:00
|
|
|
|
&revset_extensions,
|
|
|
|
|
None,
|
|
|
|
|
);
|
2024-11-05 07:34:17 +00:00
|
|
|
|
let expression = parse(&mut RevsetDiagnostics::new(), revset_str, &context).unwrap();
|
2024-04-23 22:11:58 +00:00
|
|
|
|
let symbol_resolver = DefaultSymbolResolver::new(repo, revset_extensions.symbol_resolvers());
|
2024-09-28 10:20:06 +00:00
|
|
|
|
let expression = expression.resolve_user_expression(repo, &symbol_resolver)?;
|
2024-10-11 14:54:45 +00:00
|
|
|
|
Ok(expression
|
|
|
|
|
.evaluate(repo)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.iter()
|
|
|
|
|
.map(Result::unwrap)
|
|
|
|
|
.collect())
|
2022-02-02 16:15:25 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn resolve_commit_ids_in_workspace(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
repo: &dyn Repo,
|
2022-02-02 16:15:25 +00:00
|
|
|
|
revset_str: &str,
|
2022-10-23 06:51:44 +00:00
|
|
|
|
workspace: &Workspace,
|
2022-10-23 04:14:00 +00:00
|
|
|
|
cwd: Option<&Path>,
|
2022-02-02 16:15:25 +00:00
|
|
|
|
) -> Vec<CommitId> {
|
2023-08-14 00:35:11 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2024-05-26 12:33:54 +00:00
|
|
|
|
let path_converter = RepoPathUiConverter::Fs {
|
|
|
|
|
cwd: cwd.unwrap_or_else(|| workspace.workspace_root()).to_owned(),
|
|
|
|
|
base: workspace.workspace_root().to_owned(),
|
|
|
|
|
};
|
2022-10-23 06:51:44 +00:00
|
|
|
|
let workspace_ctx = RevsetWorkspaceContext {
|
2024-05-26 12:33:54 +00:00
|
|
|
|
path_converter: &path_converter,
|
2022-10-23 06:51:44 +00:00
|
|
|
|
workspace_id: workspace.workspace_id(),
|
|
|
|
|
};
|
2024-05-03 15:50:50 +00:00
|
|
|
|
let aliases_map = RevsetAliasesMap::default();
|
|
|
|
|
let extensions = RevsetExtensions::default();
|
|
|
|
|
let context = RevsetParseContext::new(
|
|
|
|
|
&aliases_map,
|
|
|
|
|
settings.user_email(),
|
2024-06-08 00:42:15 +00:00
|
|
|
|
chrono::Utc::now().fixed_offset().into(),
|
2024-05-03 15:50:50 +00:00
|
|
|
|
&extensions,
|
|
|
|
|
Some(workspace_ctx),
|
|
|
|
|
);
|
2024-11-05 07:34:17 +00:00
|
|
|
|
let expression = parse(&mut RevsetDiagnostics::new(), revset_str, &context).unwrap();
|
2024-04-23 22:11:58 +00:00
|
|
|
|
let symbol_resolver =
|
|
|
|
|
DefaultSymbolResolver::new(repo, &([] as [&Box<dyn SymbolResolverExtension>; 0]));
|
2023-04-07 10:31:32 +00:00
|
|
|
|
let expression = expression
|
2023-05-05 21:01:25 +00:00
|
|
|
|
.resolve_user_expression(repo, &symbol_resolver)
|
2023-04-07 10:31:32 +00:00
|
|
|
|
.unwrap();
|
2024-10-11 14:54:45 +00:00
|
|
|
|
expression
|
|
|
|
|
.evaluate(repo)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.iter()
|
|
|
|
|
.map(Result::unwrap)
|
|
|
|
|
.collect()
|
2021-04-10 17:18:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_root_and_checkout() {
|
2021-04-10 17:18:30 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_workspace = TestWorkspace::init(&settings);
|
2022-10-23 06:51:44 +00:00
|
|
|
|
let repo = &test_workspace.repo;
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
2024-10-07 01:32:59 +00:00
|
|
|
|
let root_operation = repo.loader().root_operation();
|
2024-10-01 03:29:25 +00:00
|
|
|
|
let root_repo = repo.reload_at(&root_operation).unwrap();
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
|
|
|
|
let root_commit = repo.store().root_commit();
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
|
|
|
|
// Can find the root commit
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "root()"),
|
2021-04-10 17:18:30 +00:00
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
2024-10-07 02:00:37 +00:00
|
|
|
|
// Can find the root commit in the root view
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(root_repo.as_ref(), "root()"),
|
|
|
|
|
vec![root_commit.id().clone()]
|
2024-10-01 03:29:25 +00:00
|
|
|
|
);
|
|
|
|
|
|
2023-01-25 16:29:15 +00:00
|
|
|
|
// Can find the current working-copy commit
|
2022-10-20 23:33:14 +00:00
|
|
|
|
mut_repo
|
|
|
|
|
.set_wc_commit(WorkspaceId::default(), commit1.id().clone())
|
|
|
|
|
.unwrap();
|
2021-04-10 17:18:30 +00:00
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids_in_workspace(mut_repo, "@", &test_workspace.workspace, None),
|
2021-04-10 17:18:30 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
2024-10-07 02:00:37 +00:00
|
|
|
|
|
|
|
|
|
// Shouldn't panic by unindexed commit ID
|
2024-11-04 05:46:00 +00:00
|
|
|
|
let expression = RevsetExpression::commit(commit1.id().clone())
|
|
|
|
|
.resolve_user_expression(tx.repo(), &FailingSymbolResolver)
|
|
|
|
|
.unwrap();
|
2024-10-07 02:00:37 +00:00
|
|
|
|
assert!(expression.evaluate(tx.base_repo().as_ref()).is_err());
|
2021-04-10 17:18:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_heads() {
|
2021-10-10 19:30:00 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-10-10 19:30:00 +00:00
|
|
|
|
|
|
|
|
|
let root_commit = repo.store().root_commit();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-10-10 19:30:00 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
2023-08-11 04:51:01 +00:00
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1]);
|
2021-10-10 19:30:00 +00:00
|
|
|
|
|
|
|
|
|
// Heads of an empty set is an empty set
|
2023-02-13 17:52:21 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "heads(none())"), vec![]);
|
2021-10-10 19:30:00 +00:00
|
|
|
|
|
|
|
|
|
// Heads of the root is the root
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "heads(root())"),
|
2021-10-10 19:30:00 +00:00
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Heads of a single commit is that commit
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("heads({})", commit2.id())),
|
2021-10-10 19:30:00 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Heads of a parent and a child is the child
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("heads({} | {})", commit2.id(), commit3.id())
|
2021-10-10 19:30:00 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Heads of a grandparent and a grandchild is the grandchild (unlike Mercurial's
|
|
|
|
|
// heads() revset, which would include both)
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("heads({} | {})", commit1.id(), commit3.id())
|
2021-10-10 19:30:00 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
2023-08-11 04:51:01 +00:00
|
|
|
|
// Heads should be sorted in reverse index position order
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("heads({} | {})", commit3.id(), commit4.id())
|
2023-08-11 04:51:01 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
2023-03-28 06:13:31 +00:00
|
|
|
|
// Heads of all commits is the set of visible heads in the repo
|
2021-10-10 19:30:00 +00:00
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "heads(all())"),
|
2023-03-28 06:13:31 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "visible_heads()")
|
2021-10-10 19:30:00 +00:00
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_roots() {
|
2022-04-13 20:55:47 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-04-13 20:55:47 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let root_commit = repo.store().root_commit();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2022-04-13 20:55:47 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
|
|
|
|
|
// Roots of an empty set is an empty set
|
2023-02-13 17:52:21 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "roots(none())"), vec![]);
|
2022-04-13 20:55:47 +00:00
|
|
|
|
|
|
|
|
|
// Roots of the root is the root
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "roots(root())"),
|
2022-04-13 20:55:47 +00:00
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Roots of a single commit is that commit
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("roots({})", commit2.id())),
|
2022-04-13 20:55:47 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Roots of a parent and a child is the parent
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("roots({} | {})", commit2.id(), commit3.id())
|
2022-04-13 20:55:47 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Roots of a grandparent and a grandchild is the grandparent (unlike
|
|
|
|
|
// Mercurial's roots() revset, which would include both)
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("roots({} | {})", commit1.id(), commit3.id())
|
2022-04-13 20:55:47 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Roots of all commits is the root commit
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "roots(all())"),
|
2022-04-13 20:55:47 +00:00
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_parents() {
|
2021-04-10 17:18:30 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_workspace = TestWorkspace::init(&settings);
|
2022-10-23 06:51:44 +00:00
|
|
|
|
let repo = &test_workspace.repo;
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let root_commit = repo.store().root_commit();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.initial_commit();
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit2, &commit3]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit2]);
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
|
|
|
|
// The root commit has no parents
|
2023-09-03 03:47:23 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "root()-"), vec![]);
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
2023-01-25 16:29:15 +00:00
|
|
|
|
// Can find parents of the current working-copy commit
|
2022-10-20 23:33:14 +00:00
|
|
|
|
mut_repo
|
|
|
|
|
.set_wc_commit(WorkspaceId::default(), commit2.id().clone())
|
|
|
|
|
.unwrap();
|
2021-04-10 17:18:30 +00:00
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids_in_workspace(mut_repo, "@-", &test_workspace.workspace, None,),
|
2021-04-10 17:18:30 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Can find parents of a merge commit
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}-", commit4.id())),
|
2021-04-18 22:09:29 +00:00
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Parents of all commits in input are returned
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("({} | {})-", commit2.id(), commit3.id())),
|
2021-04-18 22:09:29 +00:00
|
|
|
|
vec![commit1.id().clone(), root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Parents already in input set are returned
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("({} | {})-", commit1.id(), commit2.id())),
|
2021-04-18 22:09:29 +00:00
|
|
|
|
vec![commit1.id().clone(), root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Parents shared among commits in input are not repeated
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("({} | {})-", commit4.id(), commit5.id())),
|
2021-04-18 22:09:29 +00:00
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()]
|
2021-04-10 17:18:30 +00:00
|
|
|
|
);
|
2022-12-09 13:55:59 +00:00
|
|
|
|
|
|
|
|
|
// Can find parents of parents, which may be optimized to single query
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}--", commit4.id())),
|
2022-12-09 13:55:59 +00:00
|
|
|
|
vec![commit1.id().clone(), root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("({} | {})--", commit4.id(), commit5.id())
|
2022-12-09 13:55:59 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit1.id().clone(), root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("({} | {})--", commit4.id(), commit2.id())
|
2022-12-09 13:55:59 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit1.id().clone(), root_commit.id().clone()]
|
|
|
|
|
);
|
2021-04-10 17:18:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_children() {
|
2021-04-21 21:51:23 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-21 21:51:23 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-21 21:51:23 +00:00
|
|
|
|
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
2021-04-21 21:51:23 +00:00
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
2021-09-25 21:36:34 +00:00
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit4 = create_random_commit(mut_repo, &settings)
|
2021-04-21 21:51:23 +00:00
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit5 = create_random_commit(mut_repo, &settings)
|
2021-09-25 21:36:34 +00:00
|
|
|
|
.set_parents(vec![commit3.id().clone(), commit4.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2023-04-19 11:42:10 +00:00
|
|
|
|
let commit6 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit5.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2021-04-21 21:51:23 +00:00
|
|
|
|
|
|
|
|
|
// Can find children of the root commit
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "root()+"),
|
2022-02-05 23:02:16 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
2021-04-21 21:51:23 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Children of all commits in input are returned, including those already in the
|
|
|
|
|
// input set
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("({} | {})+", commit1.id(), commit2.id())),
|
2021-04-21 21:51:23 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
2021-09-25 21:36:34 +00:00
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone()
|
2021-04-21 21:51:23 +00:00
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Children shared among commits in input are not repeated
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("({} | {})+", commit3.id(), commit4.id())),
|
2021-09-25 21:36:34 +00:00
|
|
|
|
vec![commit5.id().clone()]
|
2021-04-21 21:51:23 +00:00
|
|
|
|
);
|
2023-04-07 01:37:49 +00:00
|
|
|
|
|
2023-04-19 11:42:10 +00:00
|
|
|
|
// Can find children of children, which may be optimized to single query
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "root()++"),
|
2023-04-19 11:42:10 +00:00
|
|
|
|
vec![commit4.id().clone(), commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("(root() | {})++", commit1.id())),
|
2023-04-19 11:42:10 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("({} | {})++", commit4.id(), commit2.id())
|
2023-04-19 11:42:10 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit6.id().clone(), commit5.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
2023-04-07 01:37:49 +00:00
|
|
|
|
// Empty root
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "none()+"), vec![]);
|
2021-04-21 21:51:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_ancestors() {
|
2021-04-10 17:18:30 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let root_commit = repo.store().root_commit();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1, &commit3]);
|
2021-04-10 17:18:30 +00:00
|
|
|
|
|
|
|
|
|
// The ancestors of the root commit is just the root commit itself
|
|
|
|
|
assert_eq!(
|
2024-02-13 05:56:46 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "::root()"),
|
2021-04-10 17:18:30 +00:00
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Can find ancestors of a specific commit. Commits reachable via multiple paths
|
|
|
|
|
// are not repeated.
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("::{}", commit4.id())),
|
2021-04-10 17:18:30 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2022-12-09 13:55:59 +00:00
|
|
|
|
|
|
|
|
|
// Can find ancestors of parents or parents of ancestors, which may be optimized
|
|
|
|
|
// to single query
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("::({}-)", commit4.id())),
|
2022-12-09 13:55:59 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("(::({}|{}))-", commit3.id(), commit2.id()),
|
2022-12-09 13:55:59 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("::(({}|{})-)", commit3.id(), commit2.id()),
|
2022-12-09 13:55:59 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2023-09-02 06:02:35 +00:00
|
|
|
|
|
|
|
|
|
// Can find last n ancestors of a commit
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("ancestors({}, 0)", commit2.id())),
|
2023-09-02 06:02:35 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("ancestors({}, 1)", commit3.id())),
|
2023-09-02 06:02:35 +00:00
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("ancestors({}, 3)", commit3.id())),
|
2023-09-02 06:02:35 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2021-04-10 17:18:30 +00:00
|
|
|
|
}
|
2021-04-16 05:19:08 +00:00
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_range() {
|
2021-04-23 18:11:39 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-23 18:11:39 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1, &commit3]);
|
2021-04-23 18:11:39 +00:00
|
|
|
|
|
|
|
|
|
// The range from the root to the root is empty (because the left side of the
|
|
|
|
|
// range is exclusive)
|
2023-09-03 03:47:23 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "root()..root()"), vec![]);
|
2021-04-23 18:11:39 +00:00
|
|
|
|
|
|
|
|
|
// Linear range
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}..{}", commit1.id(), commit3.id())),
|
2021-04-23 18:11:39 +00:00
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Empty range (descendant first)
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}..{}", commit3.id(), commit1.id())),
|
2021-04-23 18:11:39 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Range including a merge
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}..{}", commit1.id(), commit4.id())),
|
2021-04-23 18:11:39 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
2024-03-12 10:22:16 +00:00
|
|
|
|
// Range including merge ancestors: commit4-- == root | commit2
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}--..{}", commit4.id(), commit3.id())),
|
2024-03-12 10:22:16 +00:00
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
2021-04-23 18:11:39 +00:00
|
|
|
|
// Sibling commits
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}..{}", commit2.id(), commit3.id())),
|
2021-04-23 18:11:39 +00:00
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
2023-09-04 23:38:28 +00:00
|
|
|
|
|
|
|
|
|
// Left operand defaults to root()
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("..{}", commit2.id())),
|
2023-09-04 23:38:28 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Right operand defaults to visible_heads()
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}..", commit2.id())),
|
2023-09-04 23:38:28 +00:00
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
2023-09-03 08:08:16 +00:00
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, ".."),
|
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2021-04-23 18:11:39 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_dag_range() {
|
2021-04-23 05:45:58 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-23 05:45:58 +00:00
|
|
|
|
|
2021-11-17 20:29:08 +00:00
|
|
|
|
let root_commit_id = repo.store().root_commit_id().clone();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit3, &commit4]);
|
2021-04-23 05:45:58 +00:00
|
|
|
|
|
|
|
|
|
// Can get DAG range of just the root commit
|
|
|
|
|
assert_eq!(
|
2024-02-13 05:56:46 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "root()::root()"),
|
2021-11-17 20:29:08 +00:00
|
|
|
|
vec![root_commit_id.clone()]
|
2021-04-23 05:45:58 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Linear range
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}::{}", root_commit_id, commit2.id())),
|
2023-09-03 08:08:16 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit_id.clone(),
|
|
|
|
|
]
|
2021-04-23 05:45:58 +00:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Empty range
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}::{}", commit2.id(), commit4.id())),
|
2021-04-23 05:45:58 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
|
2023-04-06 09:43:39 +00:00
|
|
|
|
// Empty root
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("none()::{}", commit5.id())),
|
2023-04-06 09:43:39 +00:00
|
|
|
|
vec![],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Multiple root, commit1 shouldn't be hidden by commit2
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("({}|{})::{}", commit1.id(), commit2.id(), commit3.id())
|
2023-04-06 09:43:39 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
2021-04-23 05:45:58 +00:00
|
|
|
|
// Including a merge
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}::{}", commit1.id(), commit5.id())),
|
2021-04-23 05:45:58 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
2022-04-14 04:26:21 +00:00
|
|
|
|
// Including a merge, but ancestors only from one side
|
2021-04-23 05:45:58 +00:00
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}::{}", commit2.id(), commit5.id())),
|
2021-04-23 05:45:58 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
2022-04-14 04:26:21 +00:00
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2023-09-03 08:08:16 +00:00
|
|
|
|
|
|
|
|
|
// Full range meaning all()
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "::"),
|
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit_id.clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2022-04-14 04:26:21 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_connected() {
|
2022-04-14 04:26:21 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-04-14 04:26:21 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let root_commit_id = repo.store().root_commit_id().clone();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2022-04-14 04:26:21 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit3, &commit4]);
|
|
|
|
|
|
|
|
|
|
// Connecting an empty set yields an empty set
|
2023-02-13 17:52:21 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "connected(none())"), vec![]);
|
2022-04-14 04:26:21 +00:00
|
|
|
|
|
|
|
|
|
// Can connect just the root commit
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "connected(root())"),
|
2022-04-14 04:26:21 +00:00
|
|
|
|
vec![root_commit_id.clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Can connect linearly
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("connected({} | {})", root_commit_id, commit2.id())
|
2022-04-14 04:26:21 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone(), root_commit_id]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Siblings don't get connected
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("connected({} | {})", commit2.id(), commit4.id())
|
2022-04-14 04:26:21 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit4.id().clone(), commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Including a merge
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("connected({} | {})", commit1.id(), commit5.id())
|
2022-04-14 04:26:21 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Including a merge, but ancestors only from one side
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("connected({} | {})", commit2.id(), commit5.id())
|
2022-04-14 04:26:21 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
2021-04-23 05:45:58 +00:00
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-14 21:01:28 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_reachable() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-05-14 21:01:28 +00:00
|
|
|
|
|
|
|
|
|
// Construct 3 separate subgraphs off the root commit.
|
|
|
|
|
// 1 is a chain, 2 is a merge, 3 is a pyramidal monstrosity
|
|
|
|
|
let graph1commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let graph1commit2 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![graph1commit1.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let graph1commit3 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![graph1commit2.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let graph2commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let graph2commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let graph2commit3 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![graph2commit1.id().clone(), graph2commit2.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let graph3commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let graph3commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let graph3commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let graph3commit4 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![graph3commit1.id().clone(), graph3commit2.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let graph3commit5 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![graph3commit2.id().clone(), graph3commit3.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let graph3commit6 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![graph3commit3.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let graph3commit7 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![graph3commit4.id().clone(), graph3commit5.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
// Domain is respected.
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!(
|
|
|
|
|
"reachable({}, all() ~ ::{})",
|
2023-12-28 04:22:40 +00:00
|
|
|
|
graph1commit2.id(),
|
|
|
|
|
graph1commit1.id()
|
2024-05-14 21:01:28 +00:00
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
vec![graph1commit3.id().clone(), graph1commit2.id().clone(),]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!(
|
|
|
|
|
"reachable({}, all() ~ ::{})",
|
2023-12-28 04:22:40 +00:00
|
|
|
|
graph1commit2.id(),
|
|
|
|
|
graph1commit3.id()
|
2024-05-14 21:01:28 +00:00
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Each graph is identifiable from any node in it.
|
|
|
|
|
for (i, commit) in [&graph1commit1, &graph1commit2, &graph1commit3]
|
|
|
|
|
.iter()
|
|
|
|
|
.enumerate()
|
|
|
|
|
{
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("reachable({}, all() ~ root())", commit.id())
|
2024-05-14 21:01:28 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
graph1commit3.id().clone(),
|
|
|
|
|
graph1commit2.id().clone(),
|
|
|
|
|
graph1commit1.id().clone(),
|
|
|
|
|
],
|
|
|
|
|
"commit {}",
|
|
|
|
|
i + 1
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i, commit) in [&graph2commit1, &graph2commit2, &graph2commit3]
|
|
|
|
|
.iter()
|
|
|
|
|
.enumerate()
|
|
|
|
|
{
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("reachable({}, all() ~ root())", commit.id())
|
2024-05-14 21:01:28 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
graph2commit3.id().clone(),
|
|
|
|
|
graph2commit2.id().clone(),
|
|
|
|
|
graph2commit1.id().clone(),
|
|
|
|
|
],
|
|
|
|
|
"commit {}",
|
|
|
|
|
i + 1
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i, commit) in [
|
|
|
|
|
&graph3commit1,
|
|
|
|
|
&graph3commit2,
|
|
|
|
|
&graph3commit3,
|
|
|
|
|
&graph3commit4,
|
|
|
|
|
&graph3commit5,
|
|
|
|
|
&graph3commit6,
|
|
|
|
|
&graph3commit7,
|
|
|
|
|
]
|
|
|
|
|
.iter()
|
|
|
|
|
.enumerate()
|
|
|
|
|
{
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("reachable({}, all() ~ root())", commit.id())
|
2024-05-14 21:01:28 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
graph3commit7.id().clone(),
|
|
|
|
|
graph3commit6.id().clone(),
|
|
|
|
|
graph3commit5.id().clone(),
|
|
|
|
|
graph3commit4.id().clone(),
|
|
|
|
|
graph3commit3.id().clone(),
|
|
|
|
|
graph3commit2.id().clone(),
|
|
|
|
|
graph3commit1.id().clone(),
|
|
|
|
|
],
|
|
|
|
|
"commit {}",
|
|
|
|
|
i + 1
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Test a split of the pyramidal monstrosity.
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!(
|
|
|
|
|
"reachable({}, all() ~ ::{})",
|
2023-12-28 04:22:40 +00:00
|
|
|
|
graph3commit4.id(),
|
|
|
|
|
graph3commit5.id()
|
2024-05-14 21:01:28 +00:00
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
graph3commit7.id().clone(),
|
|
|
|
|
graph3commit4.id().clone(),
|
|
|
|
|
graph3commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_descendants() {
|
2021-04-21 21:51:23 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-21 21:51:23 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-21 21:51:23 +00:00
|
|
|
|
|
2021-11-17 20:29:08 +00:00
|
|
|
|
let root_commit_id = repo.store().root_commit_id().clone();
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
2021-04-21 21:51:23 +00:00
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
2021-09-25 21:36:34 +00:00
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit4 = create_random_commit(mut_repo, &settings)
|
2021-04-21 21:51:23 +00:00
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit5 = create_random_commit(mut_repo, &settings)
|
2021-09-25 21:36:34 +00:00
|
|
|
|
.set_parents(vec![commit3.id().clone(), commit4.id().clone()])
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2023-04-19 11:42:10 +00:00
|
|
|
|
let commit6 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit5.id().clone()])
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2021-04-21 21:51:23 +00:00
|
|
|
|
|
2021-09-25 21:36:34 +00:00
|
|
|
|
// The descendants of the root commit are all the commits in the repo
|
2021-04-21 21:51:23 +00:00
|
|
|
|
assert_eq!(
|
2024-02-13 05:56:46 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "root()::"),
|
2021-04-21 21:51:23 +00:00
|
|
|
|
vec![
|
2023-04-19 11:42:10 +00:00
|
|
|
|
commit6.id().clone(),
|
2021-04-21 21:51:23 +00:00
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
2021-09-25 21:36:34 +00:00
|
|
|
|
commit2.id().clone(),
|
2021-04-21 21:51:23 +00:00
|
|
|
|
commit1.id().clone(),
|
2021-11-17 20:29:08 +00:00
|
|
|
|
root_commit_id,
|
2021-04-21 21:51:23 +00:00
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
2021-08-29 05:03:49 +00:00
|
|
|
|
// Can find descendants of a specific commit
|
2021-04-21 21:51:23 +00:00
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}::", commit2.id())),
|
2021-04-21 21:51:23 +00:00
|
|
|
|
vec![
|
2023-04-19 11:42:10 +00:00
|
|
|
|
commit6.id().clone(),
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Can find descendants of children or children of descendants, which may be
|
|
|
|
|
// optimized to single query
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("({}+)::", commit1.id())),
|
2023-04-19 11:42:10 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit6.id().clone(),
|
2021-09-25 21:36:34 +00:00
|
|
|
|
commit5.id().clone(),
|
2023-04-19 11:42:10 +00:00
|
|
|
|
commit4.id().clone(),
|
2021-04-21 21:51:23 +00:00
|
|
|
|
commit3.id().clone(),
|
2021-09-25 21:36:34 +00:00
|
|
|
|
commit2.id().clone(),
|
2021-04-21 21:51:23 +00:00
|
|
|
|
]
|
|
|
|
|
);
|
2023-04-19 11:42:10 +00:00
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("({}++)::", commit1.id())),
|
2023-04-19 11:42:10 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit6.id().clone(),
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("(({}|{})::)+", commit4.id(), commit2.id()),
|
2023-04-19 11:42:10 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit6.id().clone(),
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("(({}|{})+)::", commit4.id(), commit2.id()),
|
2023-04-19 11:42:10 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit6.id().clone(),
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2024-06-13 16:38:02 +00:00
|
|
|
|
|
|
|
|
|
// Can find next n descendants of a commit
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("descendants({}, 0)", commit2.id())),
|
2024-06-13 16:38:02 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("descendants({}, 1)", commit3.id())),
|
2024-06-13 16:38:02 +00:00
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("descendants({}, 3)", commit3.id())),
|
2024-06-13 16:38:02 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit6.id().clone(),
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2021-04-21 21:51:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_none() {
|
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-10-13 16:07:18 +00:00
|
|
|
|
|
|
|
|
|
// none() is empty (doesn't include the checkout, for example)
|
2023-03-21 06:02:29 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(repo.as_ref(), "none()"), vec![]);
|
2021-10-13 16:07:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_all() {
|
2021-10-13 16:12:50 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-10-13 16:12:50 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-11-17 20:29:08 +00:00
|
|
|
|
let root_commit_id = repo.store().root_commit_id().clone();
|
2021-10-13 16:12:50 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit2, &commit3]);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "all()"),
|
2021-10-13 16:12:50 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
2021-11-17 20:29:08 +00:00
|
|
|
|
root_commit_id,
|
2021-10-13 16:12:50 +00:00
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_visible_heads() {
|
2021-04-16 05:19:08 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-16 05:19:08 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit1]);
|
2021-04-16 05:19:08 +00:00
|
|
|
|
|
|
|
|
|
assert_eq!(
|
2023-03-28 06:13:31 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "visible_heads()"),
|
2022-02-05 23:02:16 +00:00
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()]
|
2021-04-16 05:19:08 +00:00
|
|
|
|
);
|
|
|
|
|
}
|
2021-04-16 05:52:33 +00:00
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_git_refs() {
|
2021-04-26 06:38:23 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-26 06:38:23 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-26 06:38:23 +00:00
|
|
|
|
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit4 = write_random_commit(mut_repo, &settings);
|
2021-04-26 06:38:23 +00:00
|
|
|
|
|
|
|
|
|
// Can get git refs when there are none
|
2023-02-13 17:52:21 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "git_refs()"), vec![]);
|
2021-04-26 06:38:23 +00:00
|
|
|
|
// Can get a mix of git refs
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
"refs/heads/bookmark1",
|
2023-07-11 13:14:59 +00:00
|
|
|
|
RefTarget::normal(commit1.id().clone()),
|
2021-07-11 17:58:01 +00:00
|
|
|
|
);
|
2023-07-11 13:14:59 +00:00
|
|
|
|
mut_repo.set_git_ref_target("refs/tags/tag1", RefTarget::normal(commit2.id().clone()));
|
2021-04-26 06:38:23 +00:00
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "git_refs()"),
|
2021-04-26 06:38:23 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// Two refs pointing to the same commit does not result in a duplicate in the
|
|
|
|
|
// revset
|
2023-07-11 13:14:59 +00:00
|
|
|
|
mut_repo.set_git_ref_target("refs/tags/tag2", RefTarget::normal(commit2.id().clone()));
|
2021-04-26 06:38:23 +00:00
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "git_refs()"),
|
2021-04-26 06:38:23 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2021-07-11 17:58:01 +00:00
|
|
|
|
// Can get git refs when there are conflicted refs
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
"refs/heads/bookmark1",
|
2023-07-11 15:22:21 +00:00
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit1.id().clone()],
|
|
|
|
|
[commit2.id().clone(), commit3.id().clone()],
|
|
|
|
|
),
|
2021-07-11 17:58:01 +00:00
|
|
|
|
);
|
2023-07-11 13:42:35 +00:00
|
|
|
|
mut_repo.set_git_ref_target(
|
|
|
|
|
"refs/tags/tag1",
|
2023-07-11 15:22:21 +00:00
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit2.id().clone()],
|
|
|
|
|
[commit3.id().clone(), commit4.id().clone()],
|
|
|
|
|
),
|
2021-07-11 17:58:01 +00:00
|
|
|
|
);
|
2023-07-12 16:56:02 +00:00
|
|
|
|
mut_repo.set_git_ref_target("refs/tags/tag2", RefTarget::absent());
|
2021-07-11 17:58:01 +00:00
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "git_refs()"),
|
2021-07-11 17:58:01 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
2021-04-26 06:38:23 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_git_head() {
|
2021-11-29 07:02:35 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-11-29 07:02:35 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-11-29 07:02:35 +00:00
|
|
|
|
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
2021-11-29 07:02:35 +00:00
|
|
|
|
|
|
|
|
|
// Can get git head when it's not set
|
2023-02-13 17:52:21 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "git_head()"), vec![]);
|
2023-07-11 13:14:59 +00:00
|
|
|
|
mut_repo.set_git_head_target(RefTarget::normal(commit1.id().clone()));
|
2021-11-29 07:02:35 +00:00
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "git_head()"),
|
2021-11-29 07:02:35 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
2024-08-21 19:59:15 +00:00
|
|
|
|
fn test_evaluate_expression_bookmarks() {
|
2021-10-10 16:39:40 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-10-10 16:39:40 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-10-10 16:39:40 +00:00
|
|
|
|
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit4 = write_random_commit(mut_repo, &settings);
|
2021-10-10 16:39:40 +00:00
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks when there are none
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "bookmarks()"), vec![]);
|
|
|
|
|
// Can get a few bookmarks
|
|
|
|
|
mut_repo.set_local_bookmark_target("bookmark1", RefTarget::normal(commit1.id().clone()));
|
|
|
|
|
mut_repo.set_local_bookmark_target("bookmark2", RefTarget::normal(commit2.id().clone()));
|
2021-10-10 16:39:40 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks()"),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks with matching names
|
2023-01-12 10:01:35 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks(bookmark1)"),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks(bookmark)"),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2023-08-16 09:41:21 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks(exact:bookmark1)"),
|
2023-08-16 09:41:21 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
2023-10-19 17:46:26 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"bookmarks(glob:"Bookmark?")"#),
|
2024-06-28 09:35:50 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"bookmarks(glob-i:"Bookmark?")"#),
|
2023-10-19 17:46:26 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2024-07-21 12:44:43 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks(regex:'ookmark')"),
|
2024-07-21 12:44:43 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks(regex:'^[Bb]ookmark1$')"),
|
2024-07-21 12:44:43 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
2023-01-12 10:01:35 +00:00
|
|
|
|
// Can silently resolve to an empty set if there's no matches
|
2024-08-21 19:59:15 +00:00
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "bookmarks(bookmark3)"), vec![]);
|
2023-08-16 09:41:21 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks(exact:ookmark1)"),
|
2023-08-16 09:41:21 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Two bookmarks pointing to the same commit does not result in a duplicate in
|
2021-10-10 16:39:40 +00:00
|
|
|
|
// the revset
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_local_bookmark_target("bookmark3", RefTarget::normal(commit2.id().clone()));
|
2021-10-10 16:39:40 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks()"),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks when there are conflicted refs
|
|
|
|
|
mut_repo.set_local_bookmark_target(
|
|
|
|
|
"bookmark1",
|
2023-07-11 15:22:21 +00:00
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit1.id().clone()],
|
|
|
|
|
[commit2.id().clone(), commit3.id().clone()],
|
|
|
|
|
),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_local_bookmark_target(
|
|
|
|
|
"bookmark2",
|
2023-07-11 15:22:21 +00:00
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit2.id().clone()],
|
|
|
|
|
[commit3.id().clone(), commit4.id().clone()],
|
|
|
|
|
),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_local_bookmark_target("bookmark3", RefTarget::absent());
|
2021-10-10 16:39:40 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "bookmarks()"),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
2024-08-21 19:59:15 +00:00
|
|
|
|
fn test_evaluate_expression_remote_bookmarks() {
|
2021-10-10 16:39:40 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2024-07-07 23:38:29 +00:00
|
|
|
|
let tracking_remote_ref = |target| RemoteRef {
|
2023-10-11 17:15:17 +00:00
|
|
|
|
target,
|
2024-07-07 23:38:29 +00:00
|
|
|
|
state: RemoteRefState::Tracking,
|
2023-10-11 17:15:17 +00:00
|
|
|
|
};
|
2024-07-07 23:38:29 +00:00
|
|
|
|
let normal_tracking_remote_ref =
|
|
|
|
|
|id: &CommitId| tracking_remote_ref(RefTarget::normal(id.clone()));
|
2021-10-10 16:39:40 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-10-10 16:39:40 +00:00
|
|
|
|
|
2022-12-24 15:38:20 +00:00
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit4 = write_random_commit(mut_repo, &settings);
|
2023-11-07 08:47:13 +00:00
|
|
|
|
let commit_git_remote = write_random_commit(mut_repo, &settings);
|
2021-10-10 16:39:40 +00:00
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks when there are none
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "remote_bookmarks()"), vec![]);
|
2024-09-11 16:11:50 +00:00
|
|
|
|
// Bookmark 1 is untracked on remote origin
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
|
|
|
|
"bookmark1",
|
2024-07-07 23:38:29 +00:00
|
|
|
|
"origin",
|
|
|
|
|
RemoteRef {
|
|
|
|
|
target: RefTarget::normal(commit1.id().clone()),
|
|
|
|
|
state: RemoteRefState::New,
|
|
|
|
|
},
|
|
|
|
|
);
|
2024-09-11 16:11:50 +00:00
|
|
|
|
// Bookmark 2 is tracked on remote private
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
|
|
|
|
"bookmark2",
|
2024-07-07 23:38:29 +00:00
|
|
|
|
"private",
|
|
|
|
|
normal_tracking_remote_ref(commit2.id()),
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Git-tracking bookmarks aren't included
|
|
|
|
|
mut_repo.set_remote_bookmark(
|
|
|
|
|
"bookmark",
|
2023-11-07 08:47:13 +00:00
|
|
|
|
git::REMOTE_NAME_FOR_LOCAL_GIT_REPO,
|
2024-07-07 23:38:29 +00:00
|
|
|
|
normal_tracking_remote_ref(commit_git_remote.id()),
|
2023-11-07 08:47:13 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get a few bookmarks
|
2021-10-10 16:39:40 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks()"),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks with matching names
|
2023-01-12 10:01:35 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks(bookmark1)"),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks(bookmark)"),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2023-08-16 09:41:21 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks(exact:bookmark1)"),
|
2023-08-16 09:41:21 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks from matching remotes
|
2023-01-12 10:01:35 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks("", origin)"#),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks("", ri)"#),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2023-08-16 09:41:21 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks("", exact:origin)"#),
|
2023-08-16 09:41:21 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks with matching names from matching remotes
|
2023-01-12 10:01:35 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks(bookmark1, ri)"),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks(bookmark, private)"#),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
2023-08-16 09:41:21 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
r#"remote_bookmarks(exact:bookmark1, exact:origin)"#
|
|
|
|
|
),
|
2023-08-16 09:41:21 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can filter bookmarks by tracked and untracked
|
2024-07-07 23:38:29 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "tracked_remote_bookmarks()"),
|
2024-07-07 23:38:29 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "untracked_remote_bookmarks()"),
|
2024-07-07 23:38:29 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "untracked_remote_bookmarks(bookmark1, origin)"),
|
2024-07-07 23:38:29 +00:00
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "tracked_remote_bookmarks(bookmark2, private)"),
|
2024-07-07 23:38:29 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
2023-01-12 10:01:35 +00:00
|
|
|
|
// Can silently resolve to an empty set if there's no matches
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks(bookmark3)"),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks("", upstream)"#),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks(bookmark1, private)"#),
|
2023-01-12 10:01:35 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
2023-08-16 09:41:21 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks(exact:ranch1, exact:origin)"#),
|
2023-08-16 09:41:21 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, r#"remote_bookmarks(exact:bookmark1, exact:orig)"#),
|
2023-08-16 09:41:21 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
2024-07-07 23:38:29 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "tracked_remote_bookmarks(bookmark1)"),
|
2024-07-07 23:38:29 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "untracked_remote_bookmarks(bookmark2)"),
|
2024-07-07 23:38:29 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Two bookmarks pointing to the same commit does not result in a duplicate in
|
2021-10-10 16:39:40 +00:00
|
|
|
|
// the revset
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
|
|
|
|
"bookmark3",
|
2024-07-07 23:38:29 +00:00
|
|
|
|
"origin",
|
|
|
|
|
normal_tracking_remote_ref(commit2.id()),
|
|
|
|
|
);
|
2021-10-10 16:39:40 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks()"),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2021-10-23 06:52:16 +00:00
|
|
|
|
// The commits don't have to be in the current set of heads to be included.
|
|
|
|
|
mut_repo.remove_head(commit2.id());
|
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks()"),
|
2021-10-23 06:52:16 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
// Can get bookmarks when there are conflicted refs
|
|
|
|
|
mut_repo.set_remote_bookmark(
|
|
|
|
|
"bookmark1",
|
2023-07-11 14:01:54 +00:00
|
|
|
|
"origin",
|
2024-07-07 23:38:29 +00:00
|
|
|
|
tracking_remote_ref(RefTarget::from_legacy_form(
|
2023-07-11 15:22:21 +00:00
|
|
|
|
[commit1.id().clone()],
|
|
|
|
|
[commit2.id().clone(), commit3.id().clone()],
|
2023-10-13 14:37:17 +00:00
|
|
|
|
)),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark(
|
|
|
|
|
"bookmark2",
|
2023-07-11 14:01:54 +00:00
|
|
|
|
"private",
|
2024-07-07 23:38:29 +00:00
|
|
|
|
tracking_remote_ref(RefTarget::from_legacy_form(
|
2023-07-11 15:22:21 +00:00
|
|
|
|
[commit2.id().clone()],
|
|
|
|
|
[commit3.id().clone(), commit4.id().clone()],
|
2023-10-13 14:37:17 +00:00
|
|
|
|
)),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
);
|
2024-08-21 19:59:15 +00:00
|
|
|
|
mut_repo.set_remote_bookmark("bookmark3", "origin", RemoteRef::absent());
|
2021-10-10 16:39:40 +00:00
|
|
|
|
assert_eq!(
|
2024-08-21 19:59:15 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "remote_bookmarks()"),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
2024-11-18 20:44:05 +00:00
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_tags() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
|
|
|
|
let mut_repo = tx.repo_mut();
|
|
|
|
|
|
|
|
|
|
let commit1 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit2 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit3 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit4 = write_random_commit(mut_repo, &settings);
|
|
|
|
|
|
|
|
|
|
// Can get tags when there are none
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "tags()"), vec![]);
|
|
|
|
|
// Can get a few tags
|
|
|
|
|
mut_repo.set_tag_target("tag1", RefTarget::normal(commit1.id().clone()));
|
|
|
|
|
mut_repo.set_tag_target("tag2", RefTarget::normal(commit2.id().clone()));
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags()"),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// Can get tags with matching names
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags(tag1)"),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags(tag)"),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags(exact:tag1)"),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, r#"tags(glob:"Tag?")"#), vec![]);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, r#"tags(glob-i:"Tag?")"#),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags(regex:'ag')"),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags(regex:'^[Tt]ag1$')"),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// Can silently resolve to an empty set if there's no matches
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "tags(tag3)"), vec![]);
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "tags(exact:ag1)"), vec![]);
|
|
|
|
|
// Two tags pointing to the same commit does not result in a duplicate in
|
|
|
|
|
// the revset
|
|
|
|
|
mut_repo.set_tag_target("tag3", RefTarget::normal(commit2.id().clone()));
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags()"),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// Can get tags when there are conflicted refs
|
|
|
|
|
mut_repo.set_tag_target(
|
|
|
|
|
"tag1",
|
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit1.id().clone()],
|
|
|
|
|
[commit2.id().clone(), commit3.id().clone()],
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
mut_repo.set_tag_target(
|
|
|
|
|
"tag2",
|
|
|
|
|
RefTarget::from_legacy_form(
|
|
|
|
|
[commit2.id().clone()],
|
|
|
|
|
[commit3.id().clone(), commit4.id().clone()],
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
mut_repo.set_tag_target("tag3", RefTarget::absent());
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "tags()"),
|
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
2021-10-10 16:39:40 +00:00
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_latest() {
|
2023-03-25 03:14:26 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-03-25 03:14:26 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2023-03-25 03:14:26 +00:00
|
|
|
|
|
2023-05-11 23:01:09 +00:00
|
|
|
|
let mut write_commit_with_committer_timestamp = |sec: i64| {
|
2023-03-25 03:14:26 +00:00
|
|
|
|
let builder = create_random_commit(mut_repo, &settings);
|
|
|
|
|
let mut committer = builder.committer().clone();
|
2023-05-11 23:01:09 +00:00
|
|
|
|
committer.timestamp.timestamp = MillisSinceEpoch(sec * 1000);
|
2023-03-25 03:14:26 +00:00
|
|
|
|
builder.set_committer(committer).write().unwrap()
|
|
|
|
|
};
|
|
|
|
|
let commit1_t3 = write_commit_with_committer_timestamp(3);
|
|
|
|
|
let commit2_t2 = write_commit_with_committer_timestamp(2);
|
|
|
|
|
let commit3_t2 = write_commit_with_committer_timestamp(2);
|
|
|
|
|
let commit4_t1 = write_commit_with_committer_timestamp(1);
|
|
|
|
|
|
|
|
|
|
// Pick the latest entry by default (count = 1)
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "latest(all())"),
|
|
|
|
|
vec![commit1_t3.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Should not panic with count = 0 or empty set
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "latest(all(), 0)"), vec![]);
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "latest(none())"), vec![]);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "latest(all(), 1)"),
|
|
|
|
|
vec![commit1_t3.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Tie-breaking: pick the later entry in position
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "latest(all(), 2)"),
|
|
|
|
|
vec![commit3_t2.id().clone(), commit1_t3.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "latest(all(), 3)"),
|
|
|
|
|
vec![
|
|
|
|
|
commit3_t2.id().clone(),
|
|
|
|
|
commit2_t2.id().clone(),
|
|
|
|
|
commit1_t3.id().clone(),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "latest(all(), 4)"),
|
|
|
|
|
vec![
|
|
|
|
|
commit4_t1.id().clone(),
|
|
|
|
|
commit3_t2.id().clone(),
|
|
|
|
|
commit2_t2.id().clone(),
|
|
|
|
|
commit1_t3.id().clone(),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "latest(all(), 5)"),
|
|
|
|
|
vec![
|
|
|
|
|
commit4_t1.id().clone(),
|
|
|
|
|
commit3_t2.id().clone(),
|
|
|
|
|
commit2_t2.id().clone(),
|
|
|
|
|
commit1_t3.id().clone(),
|
|
|
|
|
mut_repo.store().root_commit_id().clone(),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Should not panic if count is larger than the candidates size
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "latest(~root(), 5)"),
|
2023-03-25 03:14:26 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4_t1.id().clone(),
|
|
|
|
|
commit3_t2.id().clone(),
|
|
|
|
|
commit2_t2.id().clone(),
|
|
|
|
|
commit1_t3.id().clone(),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-07 19:59:03 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_fork_point() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
// 5 6
|
|
|
|
|
// |/|
|
|
|
|
|
// 4 |
|
|
|
|
|
// | |
|
|
|
|
|
// 1 2 3
|
|
|
|
|
// | |/
|
|
|
|
|
// |/
|
|
|
|
|
// 0
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
|
|
|
|
let mut_repo = tx.repo_mut();
|
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let root_commit = repo.store().root_commit();
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.initial_commit();
|
|
|
|
|
let commit3 = graph_builder.initial_commit();
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit4]);
|
|
|
|
|
let commit6 = graph_builder.commit_with_parents(&[&commit4, &commit2]);
|
|
|
|
|
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "fork_point(none())"), vec![]);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "fork_point(root())"),
|
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, &format!("fork_point({})", commit1.id())),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, &format!("fork_point({})", commit2.id())),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, &format!("fork_point({})", commit3.id())),
|
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, &format!("fork_point({})", commit4.id())),
|
|
|
|
|
vec![commit4.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, &format!("fork_point({})", commit5.id())),
|
|
|
|
|
vec![commit5.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, &format!("fork_point({})", commit6.id())),
|
|
|
|
|
vec![commit6.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit1.id(), commit2.id())
|
|
|
|
|
),
|
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit2.id(), commit3.id())
|
|
|
|
|
),
|
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!(
|
|
|
|
|
"fork_point({} | {} | {})",
|
|
|
|
|
commit1.id(),
|
|
|
|
|
commit2.id(),
|
|
|
|
|
commit3.id()
|
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit1.id(), commit4.id())
|
|
|
|
|
),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit2.id(), commit5.id())
|
|
|
|
|
),
|
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit3.id(), commit6.id())
|
|
|
|
|
),
|
|
|
|
|
vec![root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit1.id(), commit5.id())
|
|
|
|
|
),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit4.id(), commit5.id())
|
|
|
|
|
),
|
|
|
|
|
vec![commit4.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit5.id(), commit6.id())
|
|
|
|
|
),
|
|
|
|
|
vec![commit4.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_fork_point_criss_cross() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
// 3 4
|
|
|
|
|
// |X|
|
|
|
|
|
// 1 2
|
|
|
|
|
// |/
|
|
|
|
|
// 0
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
|
|
|
|
let mut_repo = tx.repo_mut();
|
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.initial_commit();
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit1, &commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1, &commit2]);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit3.id(), commit4.id())
|
|
|
|
|
),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_fork_point_merge_with_ancestor() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
// 4 5
|
|
|
|
|
// |\ /|
|
|
|
|
|
// 1 2 3
|
|
|
|
|
// \|/
|
|
|
|
|
// 0
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
|
|
|
|
let mut_repo = tx.repo_mut();
|
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.initial_commit();
|
|
|
|
|
let commit3 = graph_builder.initial_commit();
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1, &commit2]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit2, &commit3]);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
&format!("fork_point({} | {})", commit4.id(), commit5.id())
|
|
|
|
|
),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_merges() {
|
2021-05-01 21:29:43 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-05-01 21:29:43 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 21:29:43 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.initial_commit();
|
|
|
|
|
let commit3 = graph_builder.initial_commit();
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit1, &commit2]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit1, &commit2, &commit3]);
|
|
|
|
|
|
|
|
|
|
// Finds all merges by default
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "merges()"),
|
2021-05-01 21:29:43 +00:00
|
|
|
|
vec![commit5.id().clone(), commit4.id().clone(),]
|
|
|
|
|
);
|
|
|
|
|
// Searches only among candidates if specified
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("::{} & merges()", commit5.id())),
|
2021-05-01 21:29:43 +00:00
|
|
|
|
vec![commit5.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_description() {
|
2021-04-16 17:24:22 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-16 17:24:22 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-04-16 17:24:22 +00:00
|
|
|
|
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit1 = create_random_commit(mut_repo, &settings)
|
2022-12-21 09:13:56 +00:00
|
|
|
|
.set_description("commit 1")
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
2021-04-16 17:24:22 +00:00
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
2022-12-21 09:13:56 +00:00
|
|
|
|
.set_description("commit 2")
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
2021-04-16 17:24:22 +00:00
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
2022-12-21 09:13:56 +00:00
|
|
|
|
.set_description("commit 3")
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2021-04-16 17:24:22 +00:00
|
|
|
|
|
|
|
|
|
// Can find multiple matches
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "description(commit)"),
|
2021-04-16 17:24:22 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
// Can find a unique match
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "description(\"commit 2\")"),
|
2021-04-16 17:24:22 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
2021-05-01 21:29:43 +00:00
|
|
|
|
// Searches only among candidates if specified
|
2021-04-16 17:24:22 +00:00
|
|
|
|
assert_eq!(
|
2023-03-28 06:13:31 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "visible_heads() & description(\"commit 2\")"),
|
2021-04-16 17:24:22 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
}
|
2021-04-17 22:25:52 +00:00
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_author() {
|
2021-12-16 06:16:22 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-12-16 06:16:22 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-12-16 06:16:22 +00:00
|
|
|
|
|
|
|
|
|
let timestamp = Timestamp {
|
|
|
|
|
timestamp: MillisSinceEpoch(0),
|
|
|
|
|
tz_offset: 0,
|
|
|
|
|
};
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit1 = create_random_commit(mut_repo, &settings)
|
2021-12-16 06:16:22 +00:00
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp,
|
2021-12-16 06:16:22 +00:00
|
|
|
|
})
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
2021-12-16 06:16:22 +00:00
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name2".to_string(),
|
|
|
|
|
email: "email2".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp,
|
2021-12-16 06:16:22 +00:00
|
|
|
|
})
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
2021-12-16 06:16:22 +00:00
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name3".to_string(),
|
|
|
|
|
email: "email3".to_string(),
|
|
|
|
|
timestamp,
|
|
|
|
|
})
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2021-12-16 06:16:22 +00:00
|
|
|
|
|
|
|
|
|
// Can find multiple matches
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "author(name)"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
// Can find a unique match by either name or email
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "author(\"name2\")"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-06-28 09:35:50 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "author(\"email3\")"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
2024-06-28 09:35:50 +00:00
|
|
|
|
// Can match case‐insensitively
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "author(substring-i:Name)"),
|
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2021-12-16 06:16:22 +00:00
|
|
|
|
// Searches only among candidates if specified
|
|
|
|
|
assert_eq!(
|
2023-03-28 06:13:31 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "visible_heads() & author(\"name2\")"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
2022-11-30 09:17:06 +00:00
|
|
|
|
// Filter by union of pure predicate and set
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("root().. & (author(name1) | {})", commit3.id())
|
2022-11-30 09:17:06 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit3.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2021-12-16 06:16:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-06-08 00:42:15 +00:00
|
|
|
|
fn parse_timestamp(s: &str) -> Timestamp {
|
|
|
|
|
Timestamp::from_datetime(s.parse::<DateTime<chrono::FixedOffset>>().unwrap())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_author_date() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-06-08 00:42:15 +00:00
|
|
|
|
|
|
|
|
|
let timestamp1 = parse_timestamp("2023-03-25T11:30:00Z");
|
|
|
|
|
let timestamp2 = parse_timestamp("2023-03-25T12:30:00Z");
|
|
|
|
|
let timestamp3 = parse_timestamp("2023-03-25T13:30:00Z");
|
|
|
|
|
|
|
|
|
|
let root_commit = repo.store().root_commit();
|
|
|
|
|
let commit1 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp1,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name2".to_string(),
|
|
|
|
|
email: "email2".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name3".to_string(),
|
|
|
|
|
email: "email3".to_string(),
|
|
|
|
|
timestamp: timestamp3,
|
|
|
|
|
})
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
// Can find multiple matches
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "author_date(after:'2023-03-25 12:00')"),
|
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "author_date(before:'2023-03-25 12:00')"),
|
|
|
|
|
vec![commit1.id().clone(), root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_committer_date() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-06-08 00:42:15 +00:00
|
|
|
|
|
|
|
|
|
let timestamp1 = parse_timestamp("2023-03-25T11:30:00Z");
|
|
|
|
|
let timestamp2 = parse_timestamp("2023-03-25T12:30:00Z");
|
|
|
|
|
let timestamp3 = parse_timestamp("2023-03-25T13:30:00Z");
|
|
|
|
|
|
|
|
|
|
let root_commit = repo.store().root_commit();
|
|
|
|
|
let commit1 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp1,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name2".to_string(),
|
|
|
|
|
email: "email2".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name3".to_string(),
|
|
|
|
|
email: "email3".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp: timestamp2,
|
2024-06-08 00:42:15 +00:00
|
|
|
|
})
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
|
|
|
|
timestamp: timestamp3,
|
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
// Can find multiple matches
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "committer_date(after:'2023-03-25 12:00')"),
|
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "committer_date(before:'2023-03-25 12:00')"),
|
|
|
|
|
vec![commit1.id().clone(), root_commit.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_mine() {
|
2023-08-14 00:35:11 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-08-14 00:35:11 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2023-08-14 00:35:11 +00:00
|
|
|
|
|
|
|
|
|
let timestamp = Timestamp {
|
|
|
|
|
timestamp: MillisSinceEpoch(0),
|
|
|
|
|
tz_offset: 0,
|
|
|
|
|
};
|
|
|
|
|
let commit1 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp,
|
2023-08-14 00:35:11 +00:00
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name2".to_string(),
|
|
|
|
|
email: settings.user_email(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp,
|
2023-08-14 00:35:11 +00:00
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2024-06-30 08:46:56 +00:00
|
|
|
|
// Can find a unique match
|
2023-08-14 00:35:11 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "mine()"),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
|
|
|
|
.set_author(Signature {
|
|
|
|
|
name: "name3".to_string(),
|
2024-06-28 09:35:50 +00:00
|
|
|
|
// Test that matches are case‐insensitive
|
|
|
|
|
email: settings.user_email().to_ascii_uppercase(),
|
2023-08-14 00:35:11 +00:00
|
|
|
|
timestamp,
|
|
|
|
|
})
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2024-06-30 08:46:56 +00:00
|
|
|
|
// Can find multiple matches
|
2023-08-14 00:35:11 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "mine()"),
|
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// Searches only among candidates if specified
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "visible_heads() & mine()"),
|
|
|
|
|
vec![commit3.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
// Filter by union of pure predicate and set
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("root().. & (mine() | {})", commit1.id())),
|
2023-08-14 00:35:11 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_committer() {
|
2021-12-16 06:16:22 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-12-16 06:16:22 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-12-16 06:16:22 +00:00
|
|
|
|
|
|
|
|
|
let timestamp = Timestamp {
|
|
|
|
|
timestamp: MillisSinceEpoch(0),
|
|
|
|
|
tz_offset: 0,
|
|
|
|
|
};
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit1 = create_random_commit(mut_repo, &settings)
|
2021-12-16 06:16:22 +00:00
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name1".to_string(),
|
|
|
|
|
email: "email1".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp,
|
2021-12-16 06:16:22 +00:00
|
|
|
|
})
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
2021-12-16 06:16:22 +00:00
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name2".to_string(),
|
|
|
|
|
email: "email2".to_string(),
|
2024-09-14 16:49:18 +00:00
|
|
|
|
timestamp,
|
2021-12-16 06:16:22 +00:00
|
|
|
|
})
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-12-24 17:01:11 +00:00
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
2021-12-16 06:16:22 +00:00
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
|
|
|
|
.set_committer(Signature {
|
|
|
|
|
name: "name3".to_string(),
|
|
|
|
|
email: "email3".to_string(),
|
|
|
|
|
timestamp,
|
|
|
|
|
})
|
2022-12-24 05:09:19 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2021-12-16 06:16:22 +00:00
|
|
|
|
|
|
|
|
|
// Can find multiple matches
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "committer(name)"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
// Can find a unique match by either name or email
|
|
|
|
|
assert_eq!(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "committer(\"name2\")"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-06-28 09:35:50 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "committer(\"email3\")"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![commit3.id().clone()]
|
|
|
|
|
);
|
2024-06-28 09:35:50 +00:00
|
|
|
|
// Can match case‐insensitively
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "committer(substring-i:Name)"),
|
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
2021-12-16 06:16:22 +00:00
|
|
|
|
// Searches only among candidates if specified
|
|
|
|
|
assert_eq!(
|
2023-03-28 06:13:31 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "visible_heads() & committer(\"name2\")"),
|
2021-12-16 06:16:22 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-28 10:20:06 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_at_operation() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo0 = &test_repo.repo;
|
|
|
|
|
let root_commit = repo0.store().root_commit();
|
|
|
|
|
|
|
|
|
|
let mut tx = repo0.start_transaction(&settings);
|
|
|
|
|
let commit1_op1 = create_random_commit(tx.repo_mut(), &settings)
|
|
|
|
|
.set_description("commit1@op1")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit2_op1 = create_random_commit(tx.repo_mut(), &settings)
|
|
|
|
|
.set_description("commit2@op1")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
tx.repo_mut()
|
|
|
|
|
.set_local_bookmark_target("commit1_ref", RefTarget::normal(commit1_op1.id().clone()));
|
2024-11-13 23:19:58 +00:00
|
|
|
|
let repo1 = tx.commit("test").unwrap();
|
2024-09-28 10:20:06 +00:00
|
|
|
|
|
|
|
|
|
let mut tx = repo1.start_transaction(&settings);
|
|
|
|
|
let commit1_op2 = tx
|
|
|
|
|
.repo_mut()
|
|
|
|
|
.rewrite_commit(&settings, &commit1_op1)
|
|
|
|
|
.set_description("commit1@op2")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit3_op2 = create_random_commit(tx.repo_mut(), &settings)
|
|
|
|
|
.set_description("commit3@op2")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
tx.repo_mut().rebase_descendants(&settings).unwrap();
|
2024-11-13 23:19:58 +00:00
|
|
|
|
let repo2 = tx.commit("test").unwrap();
|
2024-09-28 10:20:06 +00:00
|
|
|
|
|
|
|
|
|
let mut tx = repo2.start_transaction(&settings);
|
|
|
|
|
let _commit4_op3 = create_random_commit(tx.repo_mut(), &settings)
|
|
|
|
|
.set_description("commit4@op3")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
// Symbol resolution:
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@, commit1_ref)"),
|
|
|
|
|
vec![commit1_op2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@-, commit1_ref)"),
|
|
|
|
|
vec![commit1_op1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_matches!(
|
|
|
|
|
try_resolve_commit_ids(repo2.as_ref(), "at_operation(@--, commit1_ref)"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { .. })
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "present(at_operation(@--, commit1_ref))"),
|
|
|
|
|
vec![]
|
|
|
|
|
);
|
2024-11-03 09:25:39 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@--, present(commit1_ref))"),
|
|
|
|
|
vec![]
|
|
|
|
|
);
|
2024-09-28 10:20:06 +00:00
|
|
|
|
|
|
|
|
|
// Visibility resolution:
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@, all())"),
|
|
|
|
|
vec![
|
|
|
|
|
commit3_op2.id().clone(),
|
|
|
|
|
commit1_op2.id().clone(),
|
|
|
|
|
commit2_op1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@-, all())"),
|
|
|
|
|
vec![
|
|
|
|
|
commit2_op1.id().clone(),
|
|
|
|
|
commit1_op1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Operation is resolved relative to the outer ReadonlyRepo.
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@-, at_operation(@-, all()))"),
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@--, all())")
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// TODO: It might make more sense to resolve "@" to the current MutableRepo
|
|
|
|
|
// state. However, doing that isn't easy because there's no Operation object
|
|
|
|
|
// representing a MutableRepo state. For now, "@" is resolved to the base
|
|
|
|
|
// operation.
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(tx.repo(), "at_operation(@, all())"),
|
|
|
|
|
vec![
|
|
|
|
|
commit3_op2.id().clone(),
|
|
|
|
|
commit1_op2.id().clone(),
|
|
|
|
|
commit2_op1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Filter should be evaluated within the at-op repo. Note that this can
|
|
|
|
|
// populate hidden commits without explicitly referring them by commit refs.
|
|
|
|
|
// It might be better to intersect inner results again with the outer repo.
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(repo2.as_ref(), "at_operation(@-, description('commit'))"),
|
|
|
|
|
vec![commit2_op1.id().clone(), commit1_op1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// For the same reason, commit1_op1 isn't filtered out. The following query
|
|
|
|
|
// is effectively evaluated as "description('commit1') & commit1_op1".
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
repo2.as_ref(),
|
|
|
|
|
"description('commit1') & at_operation(@-, description('commit'))"
|
|
|
|
|
),
|
|
|
|
|
vec![commit1_op1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// If we have an explicit ::visible_heads(), commit1_op1 is filtered out.
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
repo2.as_ref(),
|
|
|
|
|
"::visible_heads() & description('commit1') & at_operation(@-, description('commit'))"
|
|
|
|
|
),
|
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Bad operation:
|
|
|
|
|
// TODO: should we suppress NoSuchOperation error by present()?
|
|
|
|
|
assert_matches!(
|
|
|
|
|
try_resolve_commit_ids(repo2.as_ref(), "at_operation(000000000000-, all())"),
|
|
|
|
|
Err(RevsetResolutionError::Other(_))
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-13 14:40:34 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_coalesce() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
let root_commit_id = repo.store().root_commit_id().clone();
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
|
|
|
|
let mut_repo = tx.repo_mut();
|
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
mut_repo.set_local_bookmark_target("commit1", RefTarget::normal(commit1.id().clone()));
|
|
|
|
|
mut_repo.set_local_bookmark_target("commit2", RefTarget::normal(commit2.id().clone()));
|
|
|
|
|
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "coalesce()"), vec![]);
|
|
|
|
|
assert_eq!(resolve_commit_ids(mut_repo, "coalesce(none())"), vec![]);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "coalesce(all())"),
|
|
|
|
|
vec![
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit_id.clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "coalesce(all(), commit1)"),
|
|
|
|
|
vec![
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit_id.clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "coalesce(none(), commit1)"),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "coalesce(commit1, commit2)"),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "coalesce(none(), none(), commit2)"),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "coalesce(none(), commit1, commit2)"),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
// Should resolve invalid symbols regardless of whether a specific revset is
|
|
|
|
|
// evaluated.
|
|
|
|
|
assert_matches!(
|
|
|
|
|
try_resolve_commit_ids(mut_repo, "coalesce(all(), commit1_invalid)"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { name, .. })
|
|
|
|
|
if name == "commit1_invalid"
|
|
|
|
|
);
|
|
|
|
|
assert_matches!(
|
|
|
|
|
try_resolve_commit_ids(mut_repo, "coalesce(none(), commit1_invalid)"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { name, .. })
|
|
|
|
|
if name == "commit1_invalid"
|
|
|
|
|
);
|
|
|
|
|
assert_matches!(
|
|
|
|
|
try_resolve_commit_ids(mut_repo, "coalesce(all(), commit1, commit2_invalid)"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { name, .. })
|
|
|
|
|
if name == "commit2_invalid"
|
|
|
|
|
);
|
|
|
|
|
assert_matches!(
|
|
|
|
|
try_resolve_commit_ids(mut_repo, "coalesce(none(), commit1, commit2_invalid)"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { name, .. })
|
|
|
|
|
if name == "commit2_invalid"
|
|
|
|
|
);
|
|
|
|
|
assert_matches!(
|
|
|
|
|
try_resolve_commit_ids(mut_repo, "coalesce(none(), commit1, commit2, commit2_invalid)"),
|
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { name, .. })
|
|
|
|
|
if name == "commit2_invalid"
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_union() {
|
2021-04-18 22:09:29 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-18 22:09:29 +00:00
|
|
|
|
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let root_commit = repo.store().root_commit();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit3]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit2]);
|
2021-04-18 22:09:29 +00:00
|
|
|
|
|
|
|
|
|
// Union between ancestors
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("::{} | ::{}", commit4.id(), commit5.id())
|
2021-04-18 22:09:29 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Unioning can add back commits removed by difference
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2021-04-18 22:09:29 +00:00
|
|
|
|
&format!(
|
2024-02-13 05:56:46 +00:00
|
|
|
|
"(::{} ~ ::{}) | ::{}",
|
2023-12-28 04:22:40 +00:00
|
|
|
|
commit4.id(),
|
|
|
|
|
commit2.id(),
|
|
|
|
|
commit5.id()
|
2021-04-18 22:09:29 +00:00
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Unioning of disjoint sets
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2021-04-18 22:09:29 +00:00
|
|
|
|
&format!(
|
2024-02-13 05:56:46 +00:00
|
|
|
|
"(::{} ~ ::{}) | {}",
|
2023-12-28 04:22:40 +00:00
|
|
|
|
commit4.id(),
|
|
|
|
|
commit2.id(),
|
|
|
|
|
commit5.id(),
|
2021-04-18 22:09:29 +00:00
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit5.id().clone(),
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-07 07:23:20 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_machine_generated_union() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-07-07 07:23:20 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
|
|
|
|
|
// This query shouldn't trigger stack overflow. Here we use "x::y" in case
|
|
|
|
|
// we had optimization path for trivial "commit_id|.." expression.
|
2023-12-28 04:22:40 +00:00
|
|
|
|
let revset_str = iter::repeat(format!("({}::{})", commit1.id(), commit2.id()))
|
2024-07-07 07:23:20 +00:00
|
|
|
|
.take(5000)
|
|
|
|
|
.join("|");
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, &revset_str),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_intersection() {
|
2021-04-19 00:10:17 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-19 00:10:17 +00:00
|
|
|
|
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let root_commit = repo.store().root_commit();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit3]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit2]);
|
2021-04-19 00:10:17 +00:00
|
|
|
|
|
|
|
|
|
// Intersection between ancestors
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("::{} & ::{}", commit4.id(), commit5.id())
|
2021-04-19 00:10:17 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Intersection of disjoint sets
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{} & {}", commit4.id(), commit2.id())),
|
2021-04-19 00:10:17 +00:00
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_difference() {
|
2021-04-17 22:25:52 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2022-02-05 23:02:16 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
2021-04-17 22:25:52 +00:00
|
|
|
|
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let root_commit = repo.store().root_commit();
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2021-05-01 04:41:27 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, mut_repo);
|
|
|
|
|
let commit1 = graph_builder.initial_commit();
|
|
|
|
|
let commit2 = graph_builder.commit_with_parents(&[&commit1]);
|
|
|
|
|
let commit3 = graph_builder.commit_with_parents(&[&commit2]);
|
|
|
|
|
let commit4 = graph_builder.commit_with_parents(&[&commit3]);
|
|
|
|
|
let commit5 = graph_builder.commit_with_parents(&[&commit2]);
|
2021-04-17 22:25:52 +00:00
|
|
|
|
|
2022-11-17 12:58:51 +00:00
|
|
|
|
// Difference from all
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("~::{}", commit5.id())),
|
2022-11-17 12:58:51 +00:00
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
2021-04-17 22:25:52 +00:00
|
|
|
|
// Difference between ancestors
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("::{} ~ ::{}", commit4.id(), commit5.id())
|
2021-04-17 22:25:52 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("::{} ~ ::{}", commit5.id(), commit4.id())
|
2021-04-17 22:25:52 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit5.id().clone()]
|
|
|
|
|
);
|
2022-11-17 12:58:51 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("~::{} & ::{}", commit4.id(), commit5.id())
|
2022-11-17 12:58:51 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit5.id().clone()]
|
|
|
|
|
);
|
2021-04-17 22:25:52 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("::{} ~ ::{}", commit4.id(), commit2.id())
|
2021-04-17 22:25:52 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Associativity
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("::{} ~ {} ~ {}", commit4.id(), commit2.id(), commit3.id())
|
2021-04-17 22:25:52 +00:00
|
|
|
|
),
|
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
root_commit.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Subtracting a difference does not add back any commits
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2021-04-17 22:25:52 +00:00
|
|
|
|
&format!(
|
2024-02-13 05:56:46 +00:00
|
|
|
|
"(::{} ~ ::{}) ~ (::{} ~ ::{})",
|
2023-12-28 04:22:40 +00:00
|
|
|
|
commit4.id(),
|
|
|
|
|
commit1.id(),
|
|
|
|
|
commit3.id(),
|
|
|
|
|
commit1.id(),
|
2021-04-17 22:25:52 +00:00
|
|
|
|
)
|
|
|
|
|
),
|
|
|
|
|
vec![commit4.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
2022-09-13 06:39:06 +00:00
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_filter_combinator() {
|
2023-04-01 07:41:47 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-04-01 07:41:47 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2023-04-01 07:41:47 +00:00
|
|
|
|
|
|
|
|
|
let root_commit_id = repo.store().root_commit_id();
|
|
|
|
|
let commit1 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_description("commit 1")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit2 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit1.id().clone()])
|
|
|
|
|
.set_description("commit 2")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit3 = create_random_commit(mut_repo, &settings)
|
|
|
|
|
.set_parents(vec![commit2.id().clone()])
|
|
|
|
|
.set_description("commit 3")
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
// Not intersected with a set node
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "~description(1)"),
|
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
root_commit_id.clone(),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(mut_repo, "description(1) | description(2)"),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
|
|
|
|
"description(commit) ~ (description(2) | description(3))",
|
|
|
|
|
),
|
|
|
|
|
vec![commit1.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Intersected with a set node
|
|
|
|
|
assert_eq!(
|
2023-09-03 03:47:23 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "root().. & ~description(1)"),
|
2023-04-01 07:41:47 +00:00
|
|
|
|
vec![commit3.id().clone(), commit2.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids(
|
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!("{}.. & (description(1) | description(2))", commit1.id()),
|
2023-04-01 07:41:47 +00:00
|
|
|
|
),
|
|
|
|
|
vec![commit2.id().clone()],
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_file() {
|
2022-09-13 06:39:06 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_workspace = TestWorkspace::init(&settings);
|
2022-10-23 04:14:00 +00:00
|
|
|
|
let repo = &test_workspace.repo;
|
2022-09-13 06:39:06 +00:00
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2022-09-13 06:39:06 +00:00
|
|
|
|
|
2023-11-26 10:21:46 +00:00
|
|
|
|
let added_clean_clean = RepoPath::from_internal_string("added_clean_clean");
|
|
|
|
|
let added_modified_clean = RepoPath::from_internal_string("added_modified_clean");
|
|
|
|
|
let added_modified_removed = RepoPath::from_internal_string("added_modified_removed");
|
2023-08-28 14:55:48 +00:00
|
|
|
|
let tree1 = create_tree(
|
2022-09-13 06:39:06 +00:00
|
|
|
|
repo,
|
|
|
|
|
&[
|
2023-11-26 07:12:36 +00:00
|
|
|
|
(added_clean_clean, "1"),
|
|
|
|
|
(added_modified_clean, "1"),
|
|
|
|
|
(added_modified_removed, "1"),
|
2022-09-13 06:39:06 +00:00
|
|
|
|
],
|
|
|
|
|
);
|
2023-08-28 14:55:48 +00:00
|
|
|
|
let tree2 = create_tree(
|
2022-09-13 06:39:06 +00:00
|
|
|
|
repo,
|
|
|
|
|
&[
|
2023-11-26 07:12:36 +00:00
|
|
|
|
(added_clean_clean, "1"),
|
|
|
|
|
(added_modified_clean, "2"),
|
|
|
|
|
(added_modified_removed, "2"),
|
2022-09-13 06:39:06 +00:00
|
|
|
|
],
|
|
|
|
|
);
|
2023-08-28 14:55:48 +00:00
|
|
|
|
let tree3 = create_tree(
|
2022-09-13 06:39:06 +00:00
|
|
|
|
repo,
|
|
|
|
|
&[
|
2023-11-26 07:12:36 +00:00
|
|
|
|
(added_clean_clean, "1"),
|
|
|
|
|
(added_modified_clean, "2"),
|
2022-09-13 06:39:06 +00:00
|
|
|
|
// added_modified_removed,
|
|
|
|
|
],
|
|
|
|
|
);
|
2022-12-25 16:36:13 +00:00
|
|
|
|
let commit1 = mut_repo
|
|
|
|
|
.new_commit(
|
|
|
|
|
&settings,
|
|
|
|
|
vec![repo.store().root_commit_id().clone()],
|
2023-08-28 14:55:48 +00:00
|
|
|
|
tree1.id(),
|
2022-12-25 16:36:13 +00:00
|
|
|
|
)
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit2 = mut_repo
|
2023-08-28 14:55:48 +00:00
|
|
|
|
.new_commit(&settings, vec![commit1.id().clone()], tree2.id())
|
2022-12-25 16:36:13 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit3 = mut_repo
|
2023-08-28 14:55:48 +00:00
|
|
|
|
.new_commit(&settings, vec![commit2.id().clone()], tree3.id())
|
2022-12-25 16:36:13 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit4 = mut_repo
|
2023-08-28 14:55:48 +00:00
|
|
|
|
.new_commit(&settings, vec![commit3.id().clone()], tree3.id())
|
2022-12-25 16:36:13 +00:00
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
2022-09-13 06:39:06 +00:00
|
|
|
|
|
|
|
|
|
let resolve = |file_path: &RepoPath| -> Vec<CommitId> {
|
2023-02-13 17:52:21 +00:00
|
|
|
|
let mut_repo = &*mut_repo;
|
2024-04-05 11:35:31 +00:00
|
|
|
|
let expression = RevsetExpression::filter(RevsetFilterPredicate::File(
|
|
|
|
|
FilesetExpression::prefix_path(file_path.to_owned()),
|
|
|
|
|
));
|
2024-11-05 07:34:17 +00:00
|
|
|
|
let revset = expression.evaluate(mut_repo).unwrap();
|
2024-10-11 14:54:45 +00:00
|
|
|
|
revset.iter().map(Result::unwrap).collect()
|
2022-09-13 06:39:06 +00:00
|
|
|
|
};
|
|
|
|
|
|
2023-11-26 07:12:36 +00:00
|
|
|
|
assert_eq!(resolve(added_clean_clean), vec![commit1.id().clone()]);
|
2022-09-13 06:39:06 +00:00
|
|
|
|
assert_eq!(
|
2023-11-26 07:12:36 +00:00
|
|
|
|
resolve(added_modified_clean),
|
2022-09-13 06:39:06 +00:00
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2023-11-26 07:12:36 +00:00
|
|
|
|
resolve(added_modified_removed),
|
2022-09-13 06:39:06 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone()
|
|
|
|
|
]
|
|
|
|
|
);
|
2022-10-23 04:14:00 +00:00
|
|
|
|
|
|
|
|
|
// file() revset:
|
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids_in_workspace(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2024-08-24 17:53:00 +00:00
|
|
|
|
r#"files("repo/added_clean_clean")"#,
|
2022-10-23 04:14:00 +00:00
|
|
|
|
&test_workspace.workspace,
|
|
|
|
|
Some(test_workspace.workspace.workspace_root().parent().unwrap()),
|
|
|
|
|
),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
2024-07-06 10:50:04 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids_in_workspace(
|
|
|
|
|
mut_repo,
|
2024-08-24 17:53:00 +00:00
|
|
|
|
r#"files("added_clean_clean"|"added_modified_clean")"#,
|
2024-07-06 10:50:04 +00:00
|
|
|
|
&test_workspace.workspace,
|
|
|
|
|
Some(test_workspace.workspace.workspace_root()),
|
|
|
|
|
),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
2022-10-23 04:14:00 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
resolve_commit_ids_in_workspace(
|
2023-02-13 17:52:21 +00:00
|
|
|
|
mut_repo,
|
2023-12-28 04:22:40 +00:00
|
|
|
|
&format!(r#"{}:: & files("added_modified_clean")"#, commit2.id()),
|
2022-10-23 04:14:00 +00:00
|
|
|
|
&test_workspace.workspace,
|
|
|
|
|
Some(test_workspace.workspace.workspace_root()),
|
|
|
|
|
),
|
|
|
|
|
vec![commit2.id().clone()]
|
|
|
|
|
);
|
2022-11-15 08:12:37 +00:00
|
|
|
|
|
|
|
|
|
// empty() revset, which is identical to ~file(".")
|
|
|
|
|
assert_eq!(
|
2023-12-28 04:22:40 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, &format!("{}:: & empty()", commit1.id())),
|
2022-11-15 08:12:37 +00:00
|
|
|
|
vec![commit4.id().clone()]
|
|
|
|
|
);
|
2022-09-13 06:39:06 +00:00
|
|
|
|
}
|
2023-03-14 05:14:06 +00:00
|
|
|
|
|
2024-07-08 11:20:41 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_diff_contains() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_workspace = TestWorkspace::init(&settings);
|
|
|
|
|
let repo = &test_workspace.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-07-08 11:20:41 +00:00
|
|
|
|
|
|
|
|
|
let empty_clean_inserted_deleted =
|
|
|
|
|
RepoPath::from_internal_string("empty_clean_inserted_deleted");
|
|
|
|
|
let blank_clean_inserted_clean = RepoPath::from_internal_string("blank_clean_inserted_clean");
|
|
|
|
|
let noeol_modified_modified_clean =
|
|
|
|
|
RepoPath::from_internal_string("noeol_modified_modified_clean");
|
|
|
|
|
let normal_inserted_modified_removed =
|
|
|
|
|
RepoPath::from_internal_string("normal_inserted_modified_removed");
|
|
|
|
|
let tree1 = create_tree(
|
|
|
|
|
repo,
|
|
|
|
|
&[
|
|
|
|
|
(empty_clean_inserted_deleted, ""),
|
|
|
|
|
(blank_clean_inserted_clean, "\n"),
|
|
|
|
|
(noeol_modified_modified_clean, "1"),
|
|
|
|
|
(normal_inserted_modified_removed, "1\n"),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
let tree2 = create_tree(
|
|
|
|
|
repo,
|
|
|
|
|
&[
|
|
|
|
|
(empty_clean_inserted_deleted, ""),
|
|
|
|
|
(blank_clean_inserted_clean, "\n"),
|
|
|
|
|
(noeol_modified_modified_clean, "2"),
|
|
|
|
|
(normal_inserted_modified_removed, "1\n2\n"),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
let tree3 = create_tree(
|
|
|
|
|
repo,
|
|
|
|
|
&[
|
|
|
|
|
(empty_clean_inserted_deleted, "3"),
|
|
|
|
|
(blank_clean_inserted_clean, "\n3\n"),
|
|
|
|
|
(noeol_modified_modified_clean, "2 3"),
|
|
|
|
|
(normal_inserted_modified_removed, "1 3\n2\n"),
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
let tree4 = create_tree(
|
|
|
|
|
repo,
|
|
|
|
|
&[
|
|
|
|
|
(empty_clean_inserted_deleted, ""),
|
|
|
|
|
(blank_clean_inserted_clean, "\n3\n"),
|
|
|
|
|
(noeol_modified_modified_clean, "2 3"),
|
|
|
|
|
// normal_inserted_modified_removed
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
let commit1 = mut_repo
|
|
|
|
|
.new_commit(
|
|
|
|
|
&settings,
|
|
|
|
|
vec![repo.store().root_commit_id().clone()],
|
|
|
|
|
tree1.id(),
|
|
|
|
|
)
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit2 = mut_repo
|
|
|
|
|
.new_commit(&settings, vec![commit1.id().clone()], tree2.id())
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit3 = mut_repo
|
|
|
|
|
.new_commit(&settings, vec![commit2.id().clone()], tree3.id())
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
let commit4 = mut_repo
|
|
|
|
|
.new_commit(&settings, vec![commit3.id().clone()], tree4.id())
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
let query = |revset_str: &str| {
|
|
|
|
|
resolve_commit_ids_in_workspace(
|
|
|
|
|
mut_repo,
|
|
|
|
|
revset_str,
|
|
|
|
|
&test_workspace.workspace,
|
|
|
|
|
Some(test_workspace.workspace.workspace_root()),
|
|
|
|
|
)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// should match both inserted and deleted lines
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query("diff_contains('2')"),
|
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query("diff_contains('3')"),
|
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(query("diff_contains('2 3')"), vec![commit3.id().clone()]);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query("diff_contains('1 3')"),
|
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// should match line with eol
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query(&format!(
|
|
|
|
|
"diff_contains(exact:'1', {normal_inserted_modified_removed:?})",
|
|
|
|
|
)),
|
|
|
|
|
vec![commit3.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// should match line without eol
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query(&format!(
|
|
|
|
|
"diff_contains(exact:'1', {noeol_modified_modified_clean:?})",
|
|
|
|
|
)),
|
|
|
|
|
vec![commit2.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// exact:'' should match blank line
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query(&format!(
|
|
|
|
|
"diff_contains(exact:'', {empty_clean_inserted_deleted:?})",
|
|
|
|
|
)),
|
|
|
|
|
vec![]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query(&format!(
|
|
|
|
|
"diff_contains(exact:'', {blank_clean_inserted_clean:?})",
|
|
|
|
|
)),
|
|
|
|
|
vec![commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// '' should match anything but clean
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query(&format!(
|
|
|
|
|
"diff_contains('', {empty_clean_inserted_deleted:?})",
|
|
|
|
|
)),
|
|
|
|
|
vec![commit4.id().clone(), commit3.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query(&format!(
|
|
|
|
|
"diff_contains('', {blank_clean_inserted_clean:?})",
|
|
|
|
|
)),
|
|
|
|
|
vec![commit3.id().clone(), commit1.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2024-08-08 16:36:57 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_file_merged_parents() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_workspace = TestWorkspace::init(&settings);
|
|
|
|
|
let repo = &test_workspace.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-08-08 16:36:57 +00:00
|
|
|
|
|
|
|
|
|
// file2 can be merged automatically, file1 can't.
|
|
|
|
|
let file_path1 = RepoPath::from_internal_string("file1");
|
|
|
|
|
let file_path2 = RepoPath::from_internal_string("file2");
|
|
|
|
|
let tree1 = create_tree(repo, &[(file_path1, "1\n"), (file_path2, "1\n")]);
|
|
|
|
|
let tree2 = create_tree(repo, &[(file_path1, "1\n2\n"), (file_path2, "2\n1\n")]);
|
|
|
|
|
let tree3 = create_tree(repo, &[(file_path1, "1\n3\n"), (file_path2, "1\n3\n")]);
|
|
|
|
|
let tree4 = create_tree(repo, &[(file_path1, "1\n4\n"), (file_path2, "2\n1\n3\n")]);
|
|
|
|
|
|
|
|
|
|
let mut create_commit = |parent_ids, tree_id| {
|
|
|
|
|
mut_repo
|
|
|
|
|
.new_commit(&settings, parent_ids, tree_id)
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap()
|
|
|
|
|
};
|
|
|
|
|
let commit1 = create_commit(vec![repo.store().root_commit_id().clone()], tree1.id());
|
|
|
|
|
let commit2 = create_commit(vec![commit1.id().clone()], tree2.id());
|
|
|
|
|
let commit3 = create_commit(vec![commit1.id().clone()], tree3.id());
|
|
|
|
|
let commit4 = create_commit(vec![commit2.id().clone(), commit3.id().clone()], tree4.id());
|
|
|
|
|
|
|
|
|
|
let query = |revset_str: &str| {
|
|
|
|
|
resolve_commit_ids_in_workspace(
|
|
|
|
|
mut_repo,
|
|
|
|
|
revset_str,
|
|
|
|
|
&test_workspace.workspace,
|
|
|
|
|
Some(test_workspace.workspace.workspace_root()),
|
|
|
|
|
)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
2024-08-24 17:53:00 +00:00
|
|
|
|
query("files('file1')"),
|
2024-08-08 16:36:57 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
2024-08-24 17:53:00 +00:00
|
|
|
|
query("files('file2')"),
|
2024-08-08 16:36:57 +00:00
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query("diff_contains(regex:'[1234]', 'file1')"),
|
|
|
|
|
vec![
|
|
|
|
|
commit4.id().clone(),
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
query("diff_contains(regex:'[1234]', 'file2')"),
|
|
|
|
|
vec![
|
|
|
|
|
commit3.id().clone(),
|
|
|
|
|
commit2.id().clone(),
|
|
|
|
|
commit1.id().clone(),
|
|
|
|
|
]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-09-19 12:49:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_evaluate_expression_conflict() {
|
2023-04-05 03:05:51 +00:00
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_workspace = TestWorkspace::init(&settings);
|
2023-04-05 03:05:51 +00:00
|
|
|
|
let repo = &test_workspace.repo;
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2023-04-05 03:05:51 +00:00
|
|
|
|
|
|
|
|
|
// Create a few trees, including one with a conflict in `file1`
|
2023-11-26 10:21:46 +00:00
|
|
|
|
let file_path1 = RepoPath::from_internal_string("file1");
|
|
|
|
|
let file_path2 = RepoPath::from_internal_string("file2");
|
2023-11-26 07:12:36 +00:00
|
|
|
|
let tree1 = create_tree(repo, &[(file_path1, "1"), (file_path2, "1")]);
|
|
|
|
|
let tree2 = create_tree(repo, &[(file_path1, "2"), (file_path2, "2")]);
|
|
|
|
|
let tree3 = create_tree(repo, &[(file_path1, "3"), (file_path2, "1")]);
|
2023-08-28 14:55:48 +00:00
|
|
|
|
let tree4 = tree2.merge(&tree1, &tree3).unwrap();
|
2023-04-05 03:05:51 +00:00
|
|
|
|
|
|
|
|
|
let mut create_commit = |parent_ids, tree_id| {
|
|
|
|
|
mut_repo
|
|
|
|
|
.new_commit(&settings, parent_ids, tree_id)
|
|
|
|
|
.write()
|
|
|
|
|
.unwrap()
|
|
|
|
|
};
|
2023-08-28 14:55:48 +00:00
|
|
|
|
let commit1 = create_commit(vec![repo.store().root_commit_id().clone()], tree1.id());
|
|
|
|
|
let commit2 = create_commit(vec![commit1.id().clone()], tree2.id());
|
|
|
|
|
let commit3 = create_commit(vec![commit2.id().clone()], tree3.id());
|
|
|
|
|
let commit4 = create_commit(vec![commit3.id().clone()], tree4.id());
|
2023-04-05 03:05:51 +00:00
|
|
|
|
|
|
|
|
|
// Only commit4 has a conflict
|
|
|
|
|
assert_eq!(
|
2024-08-24 17:53:00 +00:00
|
|
|
|
resolve_commit_ids(mut_repo, "conflicts()"),
|
2023-04-05 03:05:51 +00:00
|
|
|
|
vec![commit4.id().clone()]
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-14 05:14:06 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_reverse_graph_iterator() {
|
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-03-14 05:14:06 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
// Tests that merges, forks, direct edges, indirect edges, and "missing" edges
|
|
|
|
|
// are correct in reversed graph. "Missing" edges (i.e. edges to commits not
|
|
|
|
|
// in the input set) won't be part of the reversed graph. Conversely, there
|
|
|
|
|
// won't be missing edges to children not in the input.
|
|
|
|
|
//
|
|
|
|
|
// F
|
|
|
|
|
// |\
|
|
|
|
|
// D E
|
|
|
|
|
// |/
|
|
|
|
|
// C
|
|
|
|
|
// |
|
|
|
|
|
// b
|
|
|
|
|
// |
|
|
|
|
|
// A
|
|
|
|
|
// |
|
|
|
|
|
// root
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut graph_builder = CommitGraphBuilder::new(&settings, tx.repo_mut());
|
2023-03-14 05:14:06 +00:00
|
|
|
|
let commit_a = graph_builder.initial_commit();
|
|
|
|
|
let commit_b = graph_builder.commit_with_parents(&[&commit_a]);
|
|
|
|
|
let commit_c = graph_builder.commit_with_parents(&[&commit_b]);
|
|
|
|
|
let commit_d = graph_builder.commit_with_parents(&[&commit_c]);
|
|
|
|
|
let commit_e = graph_builder.commit_with_parents(&[&commit_c]);
|
|
|
|
|
let commit_f = graph_builder.commit_with_parents(&[&commit_d, &commit_e]);
|
2024-11-13 23:19:58 +00:00
|
|
|
|
let repo = tx.commit("test").unwrap();
|
2023-03-14 05:14:06 +00:00
|
|
|
|
|
|
|
|
|
let revset = revset_for_commits(
|
2023-03-21 06:02:29 +00:00
|
|
|
|
repo.as_ref(),
|
2023-03-14 05:14:06 +00:00
|
|
|
|
&[&commit_a, &commit_c, &commit_d, &commit_e, &commit_f],
|
|
|
|
|
);
|
2024-10-11 14:54:45 +00:00
|
|
|
|
let commits: Vec<_> = ReverseGraphIterator::new(revset.iter_graph())
|
|
|
|
|
.unwrap()
|
|
|
|
|
.try_collect()
|
|
|
|
|
.unwrap();
|
2023-03-14 05:14:06 +00:00
|
|
|
|
assert_eq!(commits.len(), 5);
|
2023-03-20 00:44:50 +00:00
|
|
|
|
assert_eq!(commits[0].0, *commit_a.id());
|
|
|
|
|
assert_eq!(commits[1].0, *commit_c.id());
|
|
|
|
|
assert_eq!(commits[2].0, *commit_d.id());
|
|
|
|
|
assert_eq!(commits[3].0, *commit_e.id());
|
|
|
|
|
assert_eq!(commits[4].0, *commit_f.id());
|
2023-03-19 23:52:33 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
commits[0].1,
|
2024-05-03 11:57:47 +00:00
|
|
|
|
vec![GraphEdge::indirect(commit_c.id().clone())]
|
2023-03-19 23:52:33 +00:00
|
|
|
|
);
|
2023-03-14 05:14:06 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
commits[1].1,
|
|
|
|
|
vec![
|
2024-05-03 11:57:47 +00:00
|
|
|
|
GraphEdge::direct(commit_e.id().clone()),
|
|
|
|
|
GraphEdge::direct(commit_d.id().clone()),
|
2023-03-14 05:14:06 +00:00
|
|
|
|
]
|
|
|
|
|
);
|
2024-05-03 11:57:47 +00:00
|
|
|
|
assert_eq!(commits[2].1, vec![GraphEdge::direct(commit_f.id().clone())]);
|
|
|
|
|
assert_eq!(commits[3].1, vec![GraphEdge::direct(commit_f.id().clone())]);
|
2023-03-14 05:14:06 +00:00
|
|
|
|
assert_eq!(commits[4].1, vec![]);
|
|
|
|
|
}
|
2023-03-20 23:19:19 +00:00
|
|
|
|
|
2023-06-05 04:35:34 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_no_such_revision_suggestion() {
|
|
|
|
|
let settings = testutils::user_settings();
|
2023-09-19 12:49:42 +00:00
|
|
|
|
let test_repo = TestRepo::init();
|
2023-06-05 04:35:34 +00:00
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
2023-12-12 05:42:05 +00:00
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2023-06-05 04:35:34 +00:00
|
|
|
|
let commit = write_random_commit(mut_repo, &settings);
|
|
|
|
|
|
2024-08-21 19:59:15 +00:00
|
|
|
|
for bookmark_name in ["foo", "bar", "baz"] {
|
|
|
|
|
mut_repo.set_local_bookmark_target(bookmark_name, RefTarget::normal(commit.id().clone()));
|
2023-06-05 04:35:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-09 23:18:32 +00:00
|
|
|
|
assert_matches!(resolve_symbol(mut_repo, "bar"), Ok(_));
|
2023-06-05 04:35:34 +00:00
|
|
|
|
assert_matches!(
|
2023-04-09 23:18:32 +00:00
|
|
|
|
resolve_symbol(mut_repo, "bax"),
|
2023-06-05 04:35:34 +00:00
|
|
|
|
Err(RevsetResolutionError::NoSuchRevision { name, candidates })
|
|
|
|
|
if name == "bax" && candidates == vec!["bar".to_string(), "baz".to_string()]
|
|
|
|
|
);
|
|
|
|
|
}
|
2024-03-08 09:15:27 +00:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_revset_containing_fn() {
|
|
|
|
|
let settings = testutils::user_settings();
|
|
|
|
|
let test_repo = TestRepo::init();
|
|
|
|
|
let repo = &test_repo.repo;
|
|
|
|
|
|
|
|
|
|
let mut tx = repo.start_transaction(&settings);
|
2024-09-07 15:51:02 +00:00
|
|
|
|
let mut_repo = tx.repo_mut();
|
2024-03-08 09:15:27 +00:00
|
|
|
|
let commit_a = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit_b = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit_c = write_random_commit(mut_repo, &settings);
|
|
|
|
|
let commit_d = write_random_commit(mut_repo, &settings);
|
2024-11-13 23:19:58 +00:00
|
|
|
|
let repo = tx.commit("test").unwrap();
|
2024-03-08 09:15:27 +00:00
|
|
|
|
|
|
|
|
|
let revset = revset_for_commits(repo.as_ref(), &[&commit_b, &commit_d]);
|
|
|
|
|
|
|
|
|
|
let revset_has_commit = revset.containing_fn();
|
2024-10-19 09:48:11 +00:00
|
|
|
|
assert!(!revset_has_commit(commit_a.id()).unwrap());
|
|
|
|
|
assert!(revset_has_commit(commit_b.id()).unwrap());
|
|
|
|
|
assert!(!revset_has_commit(commit_c.id()).unwrap());
|
|
|
|
|
assert!(revset_has_commit(commit_d.id()).unwrap());
|
2024-03-08 09:15:27 +00:00
|
|
|
|
}
|