Implement branch-syncing in Command.Sync

as described in the previous commit to the documentation. The loggin UI
is not great yet.
This commit is contained in:
Joachim Breitner 2011-12-29 18:37:30 +01:00
parent 559bbdb424
commit 0ee1141f30

View file

@ -9,20 +9,41 @@ module Command.Sync where
import Common.Annex import Common.Annex
import Command import Command
import qualified Remote
import qualified Annex.Branch import qualified Annex.Branch
import qualified Git.Command import qualified Git.Command
import qualified Git.Config import qualified Git.Config
import qualified Git.Ref import qualified Git.Ref
import qualified Git import qualified Git
import qualified Command.Merge
import qualified Data.ByteString.Lazy.Char8 as L import qualified Data.ByteString.Lazy.Char8 as L
def :: [Command] def :: [Command]
def = [command "sync" paramPaths seek "synchronize local repository with remote"] def = [command "sync" (paramOptional (paramRepeating paramRemote))
[seek] "synchronize local repository with remote repositories"]
-- syncing involves several operations, any of which can independantly fail -- syncing involves several operations, any of which can independantly fail
seek :: [CommandSeek] seek :: CommandSeek
seek = map withNothing [commit, pull, push] seek args = do
remotes <- if null args
then defaultSyncRemotes
else mapM Remote.byName args
branch <- currentBranch
showStart "syncing" $ "branch " ++ Git.Ref.describe branch ++ " with remote repositories " ++ intercalate "," (map Remote.name remotes)
showOutput
return $
[ commit
, mergeLocal branch
] ++
[ fetch remote | remote <- remotes ] ++
[ mergeRemote remote branch | remote <- remotes ] ++
[ Command.Merge.start ] ++
[ pushLocal branch ] ++
[ pushRemote remote branch | remote <- remotes ]
defaultSyncRemotes :: Annex [Remote.Remote Annex]
defaultSyncRemotes = undefined
commit :: CommandStart commit :: CommandStart
commit = do commit = do
@ -31,44 +52,72 @@ commit = do
showOutput showOutput
-- Commit will fail when the tree is clean, so ignore failure. -- Commit will fail when the tree is clean, so ignore failure.
_ <- inRepo $ Git.Command.runBool "commit" _ <- inRepo $ Git.Command.runBool "commit"
[Param "-a", Param "-m", Param "sync"] [Param "-a", Param "-m", Param "git-annex automatic sync"]
return True return True
pull :: CommandStart mergeLocal :: Git.Ref -> CommandStart
pull = do mergeLocal branch =
remote <- defaultRemote mergeFromIfExists $ Git.Ref $ "refs/heads/synced/" ++ Git.Ref.describe branch
showStart "pull" remote
pushLocal :: Git.Ref -> CommandStart
pushLocal branch = do
let syncBranch = Git.Ref $ "refs/heads/synced/" ++ Git.Ref.describe branch
ex <- inRepo $ Git.Ref.exists syncBranch
if ex then do
showStart "updateing" $
Git.Ref.describe syncBranch ++
" to the state of " ++ Git.Ref.describe branch ++ "..."
next $ next $
inRepo $ Git.Command.runBool "branch" [Param "-f", Param (Git.Ref.describe syncBranch)]
else
return Nothing
mergeFromIfExists :: Git.Ref -> CommandStart
mergeFromIfExists fromBranch = do
ex <- inRepo $ Git.Ref.exists fromBranch
if ex then do
showStart "merging" $ Git.Ref.describe fromBranch ++ "..."
next $ next $
inRepo $ Git.Command.runBool "merge" [Param (show fromBranch)]
else do
showNote $ Git.Ref.describe fromBranch ++ " does not exist, not merging."
showOutput
return Nothing
fetch :: Remote.Remote Annex -> CommandStart
fetch remote = do
showStart "fetching from" (Remote.name remote)
next $ next $ do next $ next $ do
showOutput showOutput
checkRemote remote checkRemote remote
inRepo $ Git.Command.runBool "pull" [Param remote] inRepo $ Git.Command.runBool "fetch" [Param (Remote.name remote)]
push :: CommandStart mergeRemote :: Remote.Remote Annex -> Git.Ref -> CommandStart
push = do mergeRemote remote branch =
remote <- defaultRemote mergeFromIfExists $ Git.Ref $ "refs/remotes/" ++ Remote.name remote ++ "/synced/" ++ Git.Ref.describe branch
showStart "push" remote
next $ next $ do
Annex.Branch.update
showOutput
inRepo $ Git.Command.runBool "push" [Param remote, matchingbranches]
where
-- git push may be configured to not push matching
-- branches; this should ensure it always does.
matchingbranches = Param ":"
-- the remote defaults to origin when not configured pushRemote :: Remote.Remote Annex -> Git.Ref -> CommandStart
defaultRemote :: Annex String pushRemote remote branch = do
defaultRemote = do showStart "pushing to" (Remote.name remote)
branch <- currentBranch let syncbranch = Git.Ref $ "refs/heads/synced/" ++ Git.Ref.describe branch
fromRepo $ Git.Config.get ("branch." ++ branch ++ ".remote") "origin" let syncbranchRemote = Git.Ref $ "refs/remotes/" ++ Remote.name remote ++ "/" ++ Git.Ref.describe syncbranch
let refspec = Git.Ref.describe branch ++ ":" ++ Git.Ref.describe syncbranch
ex <- inRepo $ Git.Ref.exists syncbranchRemote
next $ next $ do
showOutput
inRepo $ Git.Command.runBool "push" $
[ Param (Remote.name remote)
, Param (Git.Ref.describe Annex.Branch.name) ] ++
[ Param refspec | ex ]
currentBranch :: Annex String currentBranch :: Annex Git.Ref
currentBranch = Git.Ref.describe . Git.Ref . firstLine . L.unpack <$> currentBranch = Git.Ref . firstLine . L.unpack <$>
inRepo (Git.Command.pipeRead [Param "symbolic-ref", Param "HEAD"]) inRepo (Git.Command.pipeRead [Param "symbolic-ref", Param "HEAD"])
checkRemote :: String -> Annex () checkRemote :: Remote.Remote Annex -> Annex ()
checkRemote remote = do checkRemote remote = do
remoteurl <- fromRepo $ remoteurl <- fromRepo $
Git.Config.get ("remote." ++ remote ++ ".url") "" Git.Config.get ("remote." ++ Remote.name remote ++ ".url") ""
when (null remoteurl) $ do when (null remoteurl) $
error $ "No url is configured for the remote: " ++ remote error $ "No url is configured for the remote: " ++ Remote.name remote