Skip to content

Commit

Permalink
git: make export_refs() act on mutable repo
Browse files Browse the repository at this point in the history
When exporting refs, we should update our record of Git's refs. For
that, we need a mutable repo.
  • Loading branch information
martinvonz committed Nov 9, 2022
1 parent f23ba0c commit 27df21d
Show file tree
Hide file tree
Showing 4 changed files with 67 additions and 93 deletions.
8 changes: 4 additions & 4 deletions lib/src/git.rs
Original file line number Diff line number Diff line change
Expand Up @@ -252,12 +252,12 @@ fn export_changes(
/// Git repo. If this is the first export, nothing will be exported. The
/// exported view is recorded in the repo (`.jj/repo/git_export_view`).
pub fn export_refs(
repo: &Arc<ReadonlyRepo>,
mut_repo: &mut MutableRepo,
git_repo: &git2::Repository,
) -> Result<(), GitExportError> {
upgrade_old_export_state(repo);
upgrade_old_export_state(mut_repo.base_repo());

let last_export_path = repo.repo_path().join("git_export_view");
let last_export_path = mut_repo.base_repo().repo_path().join("git_export_view");
let last_export_store_view =
if let Ok(mut last_export_file) = OpenOptions::new().read(true).open(&last_export_path) {
let thrift_view = simple_op_store::read_thrift(&mut last_export_file)
Expand All @@ -267,7 +267,7 @@ pub fn export_refs(
op_store::View::default()
};
let last_export_view = View::new(last_export_store_view);
let new_export_store_view = export_changes(&last_export_view, repo.view(), git_repo)?;
let new_export_store_view = export_changes(&last_export_view, mut_repo.view(), git_repo)?;
if let Ok(mut last_export_file) = OpenOptions::new()
.write(true)
.create(true)
Expand Down
140 changes: 57 additions & 83 deletions lib/tests/test_git.rs
Original file line number Diff line number Diff line change
Expand Up @@ -406,19 +406,17 @@ fn test_import_refs_detached_head() {
fn test_export_refs_no_detach() {
// When exporting the branch that's current checked out, don't detach HEAD if
// the target already matches
let mut test_data = GitRepoData::create();
let test_data = GitRepoData::create();
let git_repo = test_data.git_repo;
let commit1 = empty_git_commit(&git_repo, "refs/heads/main", &[]);
git_repo.set_head("refs/heads/main").unwrap();
let mut tx = test_data.repo.start_transaction("test");
git::import_refs(tx.mut_repo(), &git_repo).unwrap();
tx.mut_repo()
.rebase_descendants(&test_data.settings)
.unwrap();
test_data.repo = tx.commit();
let mut_repo = tx.mut_repo();
git::import_refs(mut_repo, &git_repo).unwrap();
mut_repo.rebase_descendants(&test_data.settings).unwrap();

// Do an initial export to make sure `main` is considered
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));
assert_eq!(git_repo.head().unwrap().name(), Some("refs/heads/main"));
assert_eq!(
git_repo.find_reference("refs/heads/main").unwrap().target(),
Expand All @@ -429,22 +427,20 @@ fn test_export_refs_no_detach() {
#[test]
fn test_export_refs_no_op() {
// Nothing changes on the git side if nothing changed on the jj side
let mut test_data = GitRepoData::create();
let test_data = GitRepoData::create();
let git_repo = test_data.git_repo;
let commit1 = empty_git_commit(&git_repo, "refs/heads/main", &[]);
git_repo.set_head("refs/heads/main").unwrap();

let mut tx = test_data.repo.start_transaction("test");
git::import_refs(tx.mut_repo(), &git_repo).unwrap();
tx.mut_repo()
.rebase_descendants(&test_data.settings)
.unwrap();
test_data.repo = tx.commit();
let mut_repo = tx.mut_repo();
git::import_refs(mut_repo, &git_repo).unwrap();
mut_repo.rebase_descendants(&test_data.settings).unwrap();

assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));
// The export should be a no-op since nothing changed on the jj side since last
// export
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));
assert_eq!(git_repo.head().unwrap().name(), Some("refs/heads/main"));
assert_eq!(
git_repo.find_reference("refs/heads/main").unwrap().target(),
Expand All @@ -455,7 +451,7 @@ fn test_export_refs_no_op() {
#[test]
fn test_export_refs_branch_changed() {
// We can export a change to a branch
let mut test_data = GitRepoData::create();
let test_data = GitRepoData::create();
let git_repo = test_data.git_repo;
let commit = empty_git_commit(&git_repo, "refs/heads/main", &[]);
git_repo
Expand All @@ -467,24 +463,20 @@ fn test_export_refs_branch_changed() {
git_repo.set_head("refs/heads/feature").unwrap();

let mut tx = test_data.repo.start_transaction("test");
git::import_refs(tx.mut_repo(), &git_repo).unwrap();
tx.mut_repo()
.rebase_descendants(&test_data.settings)
.unwrap();
test_data.repo = tx.commit();
let mut_repo = tx.mut_repo();
git::import_refs(mut_repo, &git_repo).unwrap();
mut_repo.rebase_descendants(&test_data.settings).unwrap();
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));

assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
let mut tx = test_data.repo.start_transaction("test");
let new_commit = create_random_commit(&test_data.settings, &test_data.repo)
.set_parents(vec![jj_id(&commit)])
.write_to_repo(tx.mut_repo());
tx.mut_repo().set_local_branch(
.write_to_repo(mut_repo);
mut_repo.set_local_branch(
"main".to_string(),
RefTarget::Normal(new_commit.id().clone()),
);
tx.mut_repo().remove_local_branch("delete-me");
test_data.repo = tx.commit();
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
mut_repo.remove_local_branch("delete-me");
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));
assert_eq!(
git_repo
.find_reference("refs/heads/main")
Expand All @@ -502,28 +494,24 @@ fn test_export_refs_branch_changed() {
#[test]
fn test_export_refs_current_branch_changed() {
// If we update a branch that is checked out in the git repo, HEAD gets detached
let mut test_data = GitRepoData::create();
let test_data = GitRepoData::create();
let git_repo = test_data.git_repo;
let commit1 = empty_git_commit(&git_repo, "refs/heads/main", &[]);
git_repo.set_head("refs/heads/main").unwrap();
let mut tx = test_data.repo.start_transaction("test");
git::import_refs(tx.mut_repo(), &git_repo).unwrap();
tx.mut_repo()
.rebase_descendants(&test_data.settings)
.unwrap();
test_data.repo = tx.commit();
let mut_repo = tx.mut_repo();
git::import_refs(mut_repo, &git_repo).unwrap();
mut_repo.rebase_descendants(&test_data.settings).unwrap();
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));

assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
let mut tx = test_data.repo.start_transaction("test");
let new_commit = create_random_commit(&test_data.settings, &test_data.repo)
.set_parents(vec![jj_id(&commit1)])
.write_to_repo(tx.mut_repo());
tx.mut_repo().set_local_branch(
.write_to_repo(mut_repo);
mut_repo.set_local_branch(
"main".to_string(),
RefTarget::Normal(new_commit.id().clone()),
);
test_data.repo = tx.commit();
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));
assert_eq!(
git_repo
.find_reference("refs/heads/main")
Expand All @@ -539,26 +527,22 @@ fn test_export_refs_current_branch_changed() {
#[test]
fn test_export_refs_unborn_git_branch() {
// Can export to an empty Git repo (we can handle Git's "unborn branch" state)
let mut test_data = GitRepoData::create();
let test_data = GitRepoData::create();
let git_repo = test_data.git_repo;
git_repo.set_head("refs/heads/main").unwrap();
let mut tx = test_data.repo.start_transaction("test");
git::import_refs(tx.mut_repo(), &git_repo).unwrap();
tx.mut_repo()
.rebase_descendants(&test_data.settings)
.unwrap();
test_data.repo = tx.commit();
let mut_repo = tx.mut_repo();
git::import_refs(mut_repo, &git_repo).unwrap();
mut_repo.rebase_descendants(&test_data.settings).unwrap();
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));

assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
let mut tx = test_data.repo.start_transaction("test");
let new_commit =
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(tx.mut_repo());
tx.mut_repo().set_local_branch(
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(mut_repo);
mut_repo.set_local_branch(
"main".to_string(),
RefTarget::Normal(new_commit.id().clone()),
);
test_data.repo = tx.commit();
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));
assert_eq!(
git_repo
.find_reference("refs/heads/main")
Expand All @@ -579,45 +563,39 @@ fn test_export_import_sequence() {
// Import a branch pointing to A, modify it in jj to point to B, export it,
// modify it in git to point to C, then import it again. There should be no
// conflict.
let mut test_data = GitRepoData::create();
let test_data = GitRepoData::create();
let git_repo = test_data.git_repo;
let mut tx = test_data.repo.start_transaction("test");
let mut_repo = tx.mut_repo();
let commit_a =
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(tx.mut_repo());
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(mut_repo);
let commit_b =
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(tx.mut_repo());
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(mut_repo);
let commit_c =
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(tx.mut_repo());
test_data.repo = tx.commit();
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(mut_repo);

// Import the branch pointing to A
git_repo
.reference("refs/heads/main", git_id(&commit_a), true, "test")
.unwrap();
let mut tx = test_data.repo.start_transaction("test");
git::import_refs(tx.mut_repo(), &git_repo).unwrap();
test_data.repo = tx.commit();
git::import_refs(mut_repo, &git_repo).unwrap();

// Modify the branch in jj to point to B
let mut tx = test_data.repo.start_transaction("test");
tx.mut_repo()
.set_local_branch("main".to_string(), RefTarget::Normal(commit_b.id().clone()));
test_data.repo = tx.commit();
mut_repo.set_local_branch("main".to_string(), RefTarget::Normal(commit_b.id().clone()));

// Export the branch to git
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));

// Modify the branch in git to point to C
git_repo
.reference("refs/heads/main", git_id(&commit_c), true, "test")
.unwrap();

// Import from git
let mut tx = test_data.repo.start_transaction("test");
git::import_refs(tx.mut_repo(), &git_repo).unwrap();
git::import_refs(mut_repo, &git_repo).unwrap();
// TODO: The branch should point to C, it shouldn't be a conflict
assert_eq!(
tx.mut_repo().view().get_local_branch("main"),
mut_repo.view().get_local_branch("main"),
Some(RefTarget::Conflict {
removes: vec![commit_a.id().clone()],
adds: vec![commit_b.id().clone(), commit_c.id().clone()],
Expand All @@ -628,38 +606,34 @@ fn test_export_import_sequence() {
#[test]
fn test_export_conflicts() {
// We skip export of conflicted branches
let mut test_data = GitRepoData::create();
let test_data = GitRepoData::create();
let git_repo = test_data.git_repo;
let mut tx = test_data.repo.start_transaction("test");
let mut_repo = tx.mut_repo();
let commit_a =
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(tx.mut_repo());
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(mut_repo);
let commit_b =
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(tx.mut_repo());
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(mut_repo);
let commit_c =
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(tx.mut_repo());
tx.mut_repo()
.set_local_branch("main".to_string(), RefTarget::Normal(commit_a.id().clone()));
tx.mut_repo().set_local_branch(
create_random_commit(&test_data.settings, &test_data.repo).write_to_repo(mut_repo);
mut_repo.set_local_branch("main".to_string(), RefTarget::Normal(commit_a.id().clone()));
mut_repo.set_local_branch(
"feature".to_string(),
RefTarget::Normal(commit_a.id().clone()),
);
test_data.repo = tx.commit();
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));

// Create a conflict and export. It should not be exported, but other changes
// should be.
let mut tx = test_data.repo.start_transaction("test");
tx.mut_repo()
.set_local_branch("main".to_string(), RefTarget::Normal(commit_b.id().clone()));
tx.mut_repo().set_local_branch(
mut_repo.set_local_branch("main".to_string(), RefTarget::Normal(commit_b.id().clone()));
mut_repo.set_local_branch(
"feature".to_string(),
RefTarget::Conflict {
removes: vec![commit_a.id().clone()],
adds: vec![commit_b.id().clone(), commit_c.id().clone()],
},
);
test_data.repo = tx.commit();
assert_eq!(git::export_refs(&test_data.repo, &git_repo), Ok(()));
assert_eq!(git::export_refs(mut_repo, &git_repo), Ok(()));
assert_eq!(
git_repo
.find_reference("refs/heads/feature")
Expand Down
6 changes: 2 additions & 4 deletions src/cli_util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -835,6 +835,8 @@ impl WorkspaceCommandHelper {
}
if self.working_copy_shared_with_git {
self.export_head_to_git(mut_repo)?;
let git_repo = self.repo.store().git_repo().unwrap();
git::export_refs(mut_repo, &git_repo)?;
}
let maybe_old_commit = tx
.base_repo()
Expand All @@ -855,10 +857,6 @@ impl WorkspaceCommandHelper {
print_checkout_stats(ui, stats)?;
}
}
if self.working_copy_shared_with_git {
let git_repo = self.repo.store().git_repo().unwrap();
git::export_refs(&self.repo, &git_repo)?;
}
let settings = ui.settings();
if settings.user_name() == UserSettings::user_name_placeholder()
|| settings.user_email() == UserSettings::user_email_placeholder()
Expand Down
6 changes: 4 additions & 2 deletions src/commands.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4511,10 +4511,12 @@ fn cmd_git_export(
command: &CommandHelper,
_args: &GitExportArgs,
) -> Result<(), CommandError> {
let workspace_command = command.workspace_helper(ui)?;
let mut workspace_command = command.workspace_helper(ui)?;
let repo = workspace_command.repo();
let git_repo = get_git_repo(repo.store())?;
git::export_refs(repo, &git_repo)?;
let mut tx = workspace_command.start_transaction("export git refs");
git::export_refs(tx.mut_repo(), &git_repo)?;
workspace_command.finish_transaction(ui, tx)?;
Ok(())
}

Expand Down

0 comments on commit 27df21d

Please sign in to comment.