summaryrefslogtreecommitdiff
path: root/Annex/Branch.hs
diff options
context:
space:
mode:
Diffstat (limited to 'Annex/Branch.hs')
-rw-r--r--Annex/Branch.hs79
1 files changed, 37 insertions, 42 deletions
diff --git a/Annex/Branch.hs b/Annex/Branch.hs
index e6c92fbe5..c657525b1 100644
--- a/Annex/Branch.hs
+++ b/Annex/Branch.hs
@@ -79,20 +79,14 @@ withIndex' bootstrapping a = do
- Compares the ref stored in the lock file with the current
- ref of the branch to see if an update is needed.
-}
-updateIndex :: Annex (Maybe Git.Ref)
-updateIndex = do
- branchref <- getRef fullname
- go branchref
- return branchref
- where
- go Nothing = return ()
- go (Just branchref) = do
- lock <- fromRepo gitAnnexIndexLock
- lockref <- Git.Ref . firstLine <$>
- liftIO (catchDefaultIO (readFileStrict lock) "")
- when (lockref /= branchref) $ do
- withIndex $ mergeIndex [fullname]
- setIndexRef branchref
+updateIndex :: Git.Ref -> Annex ()
+updateIndex branchref = do
+ lock <- fromRepo gitAnnexIndexLock
+ lockref <- Git.Ref . firstLine <$>
+ liftIO (catchDefaultIO (readFileStrict lock) "")
+ when (lockref /= branchref) $ do
+ withIndex $ mergeIndex [fullname]
+ setIndexRef branchref
{- Record that the branch's index has been updated to correspond to a
- given ref of the branch. -}
@@ -115,13 +109,13 @@ setIndexRef ref = do
- change is reverted. This race is detected and another commit made
- to fix it.
-}
-commitBranch :: String -> [Git.Ref] -> Annex ()
-commitBranch message parents = do
- expected <- updateIndex
+commitBranch :: Git.Ref -> String -> [Git.Ref] -> Annex ()
+commitBranch branchref message parents = do
+ updateIndex branchref
committedref <- inRepo $ Git.commit message fullname parents
setIndexRef committedref
parentrefs <- commitparents <$> catObject committedref
- when (racedetected expected parentrefs) $
+ when (racedetected branchref parentrefs) $
fixrace committedref parentrefs
where
-- look for "parent ref" lines and return the refs
@@ -133,10 +127,7 @@ commitBranch message parents = do
{- The race can be detected by checking the commit's
- parent, which will be the newly pushed branch,
- instead of the expected ref that the index was updated to. -}
- racedetected Nothing parentrefs
- | null parentrefs = False -- first commit, no parents
- | otherwise = True -- race on first commit
- racedetected (Just expectedref) parentrefs
+ racedetected expectedref parentrefs
| expectedref `elem` parentrefs = False -- good parent
| otherwise = True -- race!
@@ -144,7 +135,7 @@ commitBranch message parents = do
- into the index, and recommit on top of the bad commit. -}
fixrace committedref lostrefs = do
mergeIndex lostrefs
- commitBranch racemessage [committedref]
+ commitBranch committedref racemessage [committedref]
racemessage = message ++ " (recovery from race)"
@@ -179,20 +170,31 @@ getCache file = getState >>= go
{- Creates the branch, if it does not already exist. -}
create :: Annex ()
-create = unlessM hasBranch $ hasOrigin >>= go
+create = do
+ _ <- getBranch
+ return ()
+
+{- Returns the ref of the branch, creating it first if necessary. -}
+getBranch :: Annex (Git.Ref)
+getBranch = maybe (hasOrigin >>= go >>= use) (return) =<< getRef fullname
where
go True = do
inRepo $ Git.run "branch"
[Param $ show name, Param $ show originname]
- maybe (return ()) setIndexRef =<< getRef fullname
- go False = withIndex' True $
- setIndexRef =<< (inRepo $ Git.commit "branch created" fullname [])
+ fromMaybe (error $ "failed to create " ++ show name)
+ <$> getRef fullname
+ go False = withIndex' True $ do
+ inRepo $ Git.commit "branch created" fullname []
+ use ref = do
+ setIndexRef ref
+ return ref
{- Stages the journal, and commits staged changes to the branch. -}
commit :: String -> Annex ()
commit message = whenM journalDirty $ lockJournal $ do
stageJournalFiles
- withIndex $ commitBranch message [fullname]
+ ref <- getBranch
+ withIndex $ commitBranch ref message [fullname]
{- Ensures that the branch and index are is up-to-date; should be
- called before data is read from it. Runs only once per git-annex run.
@@ -209,14 +211,14 @@ commit message = whenM journalDirty $ lockJournal $ do
-}
update :: Annex ()
update = onceonly $ do
- -- ensure branch exists
- create
+ -- ensure branch exists, and get its current ref
+ branchref <- getBranch
-- check what needs updating before taking the lock
dirty <- journalDirty
c <- filterM (changedBranch fullname . snd) =<< siblingBranches
let (refs, branches) = unzip c
if (not dirty && null refs)
- then simpleupdate
+ then updateIndex branchref
else withIndex $ lockJournal $ do
when dirty stageJournalFiles
let merge_desc = if null branches
@@ -229,17 +231,15 @@ update = onceonly $ do
mergeIndex branches
ff <- if dirty then return False else tryFastForwardTo refs
if ff
- then simpleupdate
- else commitBranch merge_desc (nub $ fullname:refs)
+ then updateIndex branchref
+ else commitBranch branchref merge_desc
+ (nub $ fullname:refs)
invalidateCache
where
onceonly a = unlessM (branchUpdated <$> getState) $ do
r <- a
disableUpdate
return r
- simpleupdate = do
- _ <- updateIndex
- return ()
{- Checks if the second branch has any commits not present on the first
- branch. -}
@@ -306,21 +306,16 @@ refExists :: Git.Ref -> Annex Bool
refExists ref = inRepo $ Git.runBool "show-ref"
[Param "--verify", Param "-q", Param $ show ref]
-{- Get the ref of a branch. -}
+{- Get the ref of a branch. (Must be a fully qualified branch name) -}
getRef :: Git.Branch -> Annex (Maybe Git.Ref)
getRef branch = process . L.unpack <$> showref
where
showref = inRepo $ Git.pipeRead [Param "show-ref",
Param "--hash", -- get the hash
- Params "--verify", -- only exact match
Param $ show branch]
process [] = Nothing
process s = Just $ Git.Ref $ firstLine s
-{- Does the main git-annex branch exist? -}
-hasBranch :: Annex Bool
-hasBranch = refExists fullname
-
{- Does origin/git-annex exist? -}
hasOrigin :: Annex Bool
hasOrigin = refExists originname