2024-08-21 17:32:20 +03:00
|
|
|
use anyhow::Result;
|
|
|
|
use gitbutler_branch::VirtualBranchesHandle;
|
|
|
|
use gitbutler_command_context::CommandContext;
|
2024-08-23 15:32:31 +03:00
|
|
|
use gitbutler_commit::commit_ext::CommitExt;
|
2024-08-23 00:29:13 +03:00
|
|
|
use gitbutler_repo::{
|
2024-09-06 12:03:57 +03:00
|
|
|
create_change_reference, list_branch_references, push_change_reference,
|
2024-09-10 19:51:47 +03:00
|
|
|
update_change_reference, LogUntil, RepositoryExt as _,
|
2024-08-21 17:32:20 +03:00
|
|
|
};
|
|
|
|
use tempfile::TempDir;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn create_success() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let reference = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/success".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
|
|
|
assert_eq!(reference.branch_id, test_ctx.branch.id);
|
2024-08-26 18:02:38 +03:00
|
|
|
assert_eq!(reference.name, "refs/remotes/origin/success".into());
|
2024-08-23 15:32:31 +03:00
|
|
|
assert_eq!(
|
|
|
|
reference.change_id,
|
2024-08-26 18:02:38 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap()
|
2024-08-23 15:32:31 +03:00
|
|
|
);
|
2024-08-21 17:32:20 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn create_multiple() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let first = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/first".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
|
|
|
assert_eq!(first.branch_id, test_ctx.branch.id);
|
2024-08-26 18:02:38 +03:00
|
|
|
assert_eq!(first.name, "refs/remotes/origin/first".into());
|
2024-08-23 15:32:31 +03:00
|
|
|
assert_eq!(
|
|
|
|
first.change_id,
|
2024-08-26 18:02:38 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap()
|
2024-08-23 15:32:31 +03:00
|
|
|
);
|
2024-08-26 18:02:38 +03:00
|
|
|
let last = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/last".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.last().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
|
|
|
assert_eq!(last.branch_id, test_ctx.branch.id);
|
2024-08-26 18:02:38 +03:00
|
|
|
assert_eq!(last.name, "refs/remotes/origin/last".into());
|
|
|
|
assert_eq!(
|
|
|
|
last.change_id,
|
|
|
|
test_ctx.commits.last().unwrap().change_id().unwrap()
|
|
|
|
);
|
2024-08-21 17:32:20 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn create_fails_with_non_remote_reference() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let result = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"foo".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
result.unwrap_err().to_string(),
|
|
|
|
"Failed to parse the provided reference",
|
|
|
|
);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn create_fails_when_branch_reference_with_name_exists() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/taken".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let result = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/taken".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.last().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
result.unwrap_err().to_string(),
|
|
|
|
format!("A reference refs/remotes/origin/taken already exists",),
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn create_fails_when_commit_already_referenced() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/one".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let result = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/two".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
);
|
|
|
|
assert_eq!(
|
|
|
|
result.unwrap_err().to_string(),
|
|
|
|
format!(
|
2024-08-26 21:49:01 +03:00
|
|
|
"A reference for change {} already exists",
|
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap()
|
2024-08-21 17:32:20 +03:00
|
|
|
),
|
|
|
|
);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn create_fails_when_commit_in_anothe_branch() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 21:49:01 +03:00
|
|
|
let wrong_change = test_ctx.other_commits.first().unwrap().change_id().unwrap();
|
2024-08-26 18:02:38 +03:00
|
|
|
let result = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/asdf".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
wrong_change.clone(),
|
2024-08-21 17:32:20 +03:00
|
|
|
);
|
2024-08-26 21:49:01 +03:00
|
|
|
assert!(result.is_err());
|
2024-08-21 17:32:20 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2024-08-26 21:49:01 +03:00
|
|
|
fn create_fails_when_change_id_not_found() -> Result<()> {
|
2024-08-21 17:32:20 +03:00
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let result = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/baz".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
"does-not-exist".into(),
|
2024-08-21 17:32:20 +03:00
|
|
|
);
|
2024-08-26 18:02:38 +03:00
|
|
|
assert!(result.is_err());
|
2024-08-21 17:32:20 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn list_success() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let first_ref = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/first".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let second_ref = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/second".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.last().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
|
|
|
let result = list_branch_references(&ctx, test_ctx.branch.id)?;
|
|
|
|
assert_eq!(result.len(), 2);
|
|
|
|
assert_eq!(result[0], first_ref);
|
|
|
|
assert_eq!(result[1], second_ref);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn update_success() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/first".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let updated = update_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/first".into(),
|
2024-08-26 18:02:38 +03:00
|
|
|
test_ctx.commits.last().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
2024-08-23 15:32:31 +03:00
|
|
|
assert_eq!(
|
|
|
|
updated.change_id,
|
2024-08-26 18:02:38 +03:00
|
|
|
test_ctx.commits.last().unwrap().change_id().unwrap()
|
2024-08-23 15:32:31 +03:00
|
|
|
);
|
2024-08-21 17:32:20 +03:00
|
|
|
let list = list_branch_references(&ctx, test_ctx.branch.id)?;
|
|
|
|
assert_eq!(list.len(), 1);
|
2024-08-26 18:02:38 +03:00
|
|
|
assert_eq!(
|
|
|
|
list[0].change_id,
|
|
|
|
test_ctx.commits.last().unwrap().change_id().unwrap()
|
|
|
|
);
|
2024-08-21 17:32:20 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn push_success() -> Result<()> {
|
|
|
|
let (ctx, _temp_dir) = command_ctx("multiple-commits")?;
|
|
|
|
let test_ctx = test_ctx(&ctx)?;
|
2024-08-26 18:02:38 +03:00
|
|
|
let reference = create_change_reference(
|
2024-08-21 17:32:20 +03:00
|
|
|
&ctx,
|
|
|
|
test_ctx.branch.id,
|
|
|
|
"refs/remotes/origin/first".into(),
|
2024-08-26 21:49:01 +03:00
|
|
|
test_ctx.commits.first().unwrap().change_id().unwrap(),
|
2024-08-21 17:32:20 +03:00
|
|
|
)?;
|
2024-09-06 12:03:57 +03:00
|
|
|
let result = push_change_reference(&ctx, reference.branch_id, reference.name, false);
|
2024-08-21 17:32:20 +03:00
|
|
|
assert!(result.is_ok());
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn command_ctx(name: &str) -> Result<(CommandContext, TempDir)> {
|
|
|
|
gitbutler_testsupport::writable::fixture("stacking.sh", name)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_ctx(ctx: &CommandContext) -> Result<TestContext> {
|
|
|
|
let handle = VirtualBranchesHandle::new(ctx.project().gb_dir());
|
|
|
|
let branches = handle.list_all_branches()?;
|
|
|
|
let branch = branches.iter().find(|b| b.name == "virtual").unwrap();
|
|
|
|
let other_branch = branches.iter().find(|b| b.name != "virtual").unwrap();
|
|
|
|
let target = handle.get_default_target()?;
|
2024-09-10 19:51:47 +03:00
|
|
|
let branch_commits = ctx
|
|
|
|
.repository()
|
|
|
|
.log(branch.head, LogUntil::Commit(target.sha))?;
|
|
|
|
let other_commits = ctx
|
|
|
|
.repository()
|
|
|
|
.log(other_branch.head, LogUntil::Commit(target.sha))?;
|
2024-08-21 17:32:20 +03:00
|
|
|
Ok(TestContext {
|
|
|
|
branch: branch.clone(),
|
|
|
|
commits: branch_commits,
|
2024-08-26 18:02:38 +03:00
|
|
|
// other_branch: other_branch.clone(),
|
2024-08-21 17:32:20 +03:00
|
|
|
other_commits,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
struct TestContext<'a> {
|
|
|
|
branch: gitbutler_branch::Branch,
|
|
|
|
commits: Vec<git2::Commit<'a>>,
|
|
|
|
other_commits: Vec<git2::Commit<'a>>,
|
|
|
|
}
|