From 9666addfaa1eb8092954d816f47d3a70be7d56de Mon Sep 17 00:00:00 2001 From: Joey Hess Date: Wed, 12 Jun 2013 14:54:23 -0400 Subject: [PATCH] sync: Better support for bare git remotes. Now pushes directly to the master branch on such a remote, instead of to synced/master. This makes it easier to clone from a bare git remote that has been populated with git annex sync or by the assistant. --- Command/Sync.hs | 27 +++++++++++++++++++++------ debian/changelog | 4 ++++ 2 files changed, 25 insertions(+), 6 deletions(-) diff --git a/Command/Sync.hs b/Command/Sync.hs index 091431bda0..a6ae610f84 100644 --- a/Command/Sync.hs +++ b/Command/Sync.hs @@ -138,7 +138,8 @@ pullRemote remote branch = do {- The remote probably has both a master and a synced/master branch. - Which to merge from? Well, the master has whatever latest changes - - were committed, while the synced/master may have changes that some + - were committed (or pushed changes, if this is a bare remote), + - while the synced/master may have changes that some - other remote synced to this remote. So, merge them both. -} mergeRemote :: Remote -> (Maybe Git.Ref) -> CommandCleanup mergeRemote remote b = case b of @@ -163,15 +164,29 @@ pushRemote remote branch = go =<< needpush showOutput inRepo $ pushBranch remote branch +{- If the remote is a bare git repository, it's best to push the branch + - directly to it. On the other hand, if it's not bare, pushing to the + - checked out branch will fail, and this is why we use the syncBranch. + - + - Git offers no way to tell if a remote is bare or not, so both methods + - are tried. + - + - The direct push is likely to spew an ugly error message, so stderr is + - elided. Since progress is output to stderr too, the sync push is done + - first, and actually sends the data. Then the direct push is tried, + - with stderr discarded, to update the branch ref on the remote. + -} pushBranch :: Remote -> Git.Ref -> Git.Repo -> IO Bool -pushBranch remote branch g = - Git.Command.runBool +pushBranch remote branch g = tryIO directpush `after` syncpush + where + syncpush = Git.Command.runBool (pushparams (refspec branch)) g + directpush = Git.Command.runQuiet (pushparams (show $ Git.Ref.base branch)) g + pushparams b = [ Param "push" , Param $ Remote.name remote , Param $ refspec Annex.Branch.name - , Param $ refspec branch - ] g - where + , Param b + ] refspec b = concat [ show $ Git.Ref.base b , ":" diff --git a/debian/changelog b/debian/changelog index 696d007928..ae3c058172 100644 --- a/debian/changelog +++ b/debian/changelog @@ -14,6 +14,10 @@ git-annex (4.20130602) UNRELEASED; urgency=low duplicate uploads to the same glacier repository by `git annex copy`. * Direct mode: No longer temporarily remove write permission bit of files when adding them. + * sync: Better support for bare git remotes. Now pushes directly to the + master branch on such a remote, instead of to synced/master. This + makes it easier to clone from a bare git remote that has been populated + with git annex sync or by the assistant. -- Joey Hess Mon, 10 Jun 2013 12:52:44 -0400