From 0d8585001735d786e800505d9a31abd6b85cffbe Mon Sep 17 00:00:00 2001 From: Martin von Zweigbergk Date: Mon, 28 Dec 2020 23:38:20 -0800 Subject: [PATCH] git: return a new repo instance from the store instead of the store's instance Returning the store's internal `git2::Repository` instance wrapped in a `Mutex` makes it easy to run into deadlocks. Let's return a freshly loaded repo instance instead. --- lib/src/git.rs | 20 +++++++++++--------- lib/src/git_store.rs | 5 +++-- lib/src/local_store.rs | 3 +-- lib/src/store.rs | 3 +-- lib/src/store_wrapper.rs | 5 ++--- 5 files changed, 18 insertions(+), 18 deletions(-) diff --git a/lib/src/git.rs b/lib/src/git.rs index 0148d699f..696f00652 100644 --- a/lib/src/git.rs +++ b/lib/src/git.rs @@ -30,10 +30,9 @@ pub fn push_commit( remote_branch: &str, ) -> Result<(), GitPushError> { let git_repo = commit.store().git_repo().ok_or(GitPushError::NotAGitRepo)?; - let locked_git_repo = git_repo.lock().unwrap(); // Create a temporary ref to work around https://github.com/libgit2/libgit2/issues/3178 let temp_ref_name = format!("refs/jj/git-push/{}", commit.id().hex()); - let mut temp_ref = locked_git_repo + let mut temp_ref = git_repo .reference( &temp_ref_name, git2::Oid::from_bytes(&commit.id().0).unwrap(), @@ -46,13 +45,16 @@ pub fn push_commit( err )) })?; - let mut remote = locked_git_repo.find_remote(remote_name).map_err(|err| { - match (err.class(), err.code()) { - (git2::ErrorClass::Config, git2::ErrorCode::NotFound) => GitPushError::NoSuchRemote, - (git2::ErrorClass::Config, git2::ErrorCode::InvalidSpec) => GitPushError::NoSuchRemote, - _ => panic!("unhandled git error: {:?}", err), - } - })?; + let mut remote = + git_repo + .find_remote(remote_name) + .map_err(|err| match (err.class(), err.code()) { + (git2::ErrorClass::Config, git2::ErrorCode::NotFound) => GitPushError::NoSuchRemote, + (git2::ErrorClass::Config, git2::ErrorCode::InvalidSpec) => { + GitPushError::NoSuchRemote + } + _ => panic!("unhandled git error: {:?}", err), + })?; // Need to add "refs/heads/" prefix due to https://github.com/libgit2/libgit2/issues/1125 let refspec = format!("{}:refs/heads/{}", temp_ref_name, remote_branch); remote diff --git a/lib/src/git_store.rs b/lib/src/git_store.rs index b822ad625..26d02f8f6 100644 --- a/lib/src/git_store.rs +++ b/lib/src/git_store.rs @@ -155,8 +155,9 @@ impl Store for GitStore { 20 } - fn git_repo(&self) -> Option<&Mutex> { - Some(&self.repo) + fn git_repo(&self) -> Option { + let path = self.repo.lock().unwrap().path().to_owned(); + Some(git2::Repository::open(&path).unwrap()) } fn read_file(&self, _path: &FileRepoPath, id: &FileId) -> StoreResult> { diff --git a/lib/src/local_store.rs b/lib/src/local_store.rs index 81ed55376..7ceb79578 100644 --- a/lib/src/local_store.rs +++ b/lib/src/local_store.rs @@ -28,7 +28,6 @@ use crate::store::{ ChangeId, Commit, CommitId, Conflict, ConflictId, ConflictPart, FileId, MillisSinceEpoch, Signature, Store, StoreError, StoreResult, SymlinkId, Timestamp, Tree, TreeId, TreeValue, }; -use std::sync::Mutex; impl From for StoreError { fn from(err: std::io::Error) -> Self { @@ -111,7 +110,7 @@ impl Store for LocalStore { 64 } - fn git_repo(&self) -> Option<&Mutex> { + fn git_repo(&self) -> Option { None } diff --git a/lib/src/store.rs b/lib/src/store.rs index 78d1fd183..788227821 100644 --- a/lib/src/store.rs +++ b/lib/src/store.rs @@ -21,7 +21,6 @@ use std::vec::Vec; use crate::repo_path::DirRepoPath; use crate::repo_path::FileRepoPath; use std::borrow::Borrow; -use std::sync::Mutex; use thiserror::Error; #[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Hash)] @@ -333,7 +332,7 @@ impl Tree { pub trait Store: Send + Sync + Debug { fn hash_length(&self) -> usize; - fn git_repo(&self) -> Option<&Mutex>; + fn git_repo(&self) -> Option; fn read_file(&self, path: &FileRepoPath, id: &FileId) -> StoreResult>; diff --git a/lib/src/store_wrapper.rs b/lib/src/store_wrapper.rs index 0bde25ff1..2304c74ee 100644 --- a/lib/src/store_wrapper.rs +++ b/lib/src/store_wrapper.rs @@ -13,7 +13,7 @@ // limitations under the License. use std::collections::HashMap; -use std::sync::{Arc, Mutex, RwLock, Weak}; +use std::sync::{Arc, RwLock, Weak}; use crate::commit::Commit; use crate::repo_path::{DirRepoPath, FileRepoPath}; @@ -24,7 +24,6 @@ use crate::store::{ }; use crate::tree::Tree; use crate::tree_builder::TreeBuilder; -use git2::Repository; use std::io::Read; /// Wraps the low-level store and makes it return more convenient types. Also @@ -60,7 +59,7 @@ impl StoreWrapper { self.store.hash_length() } - pub fn git_repo(&self) -> Option<&Mutex> { + pub fn git_repo(&self) -> Option { self.store.git_repo() }