improve alwayscommit=false mode

Now changes are staged into the branch's index, but not committed,
which avoids growing a large journal. And sync and merge always
explicitly commit, ensuring that even when they do nothing else,
they commit the staged changes.

Added a flag file to indicate that the branch's journal contains
uncommitted changes. (Could use git ls-files, but don't want to run
that every time.)

In the future, this ability to have uncommitted changes staged in the
journal might be used on remotes after a series of oneshot commands.
This commit is contained in:
Joey Hess 2012-02-25 16:11:47 -04:00
parent b49c0c2633
commit 1f73db3469
6 changed files with 44 additions and 7 deletions

View file

@ -18,6 +18,7 @@ module Annex.Branch (
get, get,
change, change,
commit, commit,
stage,
files, files,
) where ) where
@ -114,14 +115,14 @@ updateTo pairs = do
-- ensure branch exists, and get its current ref -- ensure branch exists, and get its current ref
branchref <- getBranch branchref <- getBranch
-- check what needs updating before taking the lock -- check what needs updating before taking the lock
dirty <- journalDirty dirty <- unCommitted
(refs, branches) <- unzip <$> filterM isnewer pairs (refs, branches) <- unzip <$> filterM isnewer pairs
if not dirty && null refs if not dirty && null refs
then updateIndex branchref then updateIndex branchref
else withIndex $ lockJournal $ do else withIndex $ lockJournal $ do
when dirty stageJournal when dirty stageJournal
let merge_desc = if null branches let merge_desc = if null branches
then "update" then "update"
else "merging " ++ else "merging " ++
unwords (map Git.Ref.describe branches) ++ unwords (map Git.Ref.describe branches) ++
" into " ++ show name " into " ++ show name
@ -182,11 +183,17 @@ set file content = do
{- Stages the journal, and commits staged changes to the branch. -} {- Stages the journal, and commits staged changes to the branch. -}
commit :: String -> Annex () commit :: String -> Annex ()
commit message = whenM journalDirty $ lockJournal $ do commit message = whenM unCommitted $ lockJournal $ do
stageJournal stageJournal
ref <- getBranch ref <- getBranch
withIndex $ commitBranch ref message [fullname] withIndex $ commitBranch ref message [fullname]
{- Stages the journal, not making a commit to the branch. -}
stage :: Annex ()
stage = whenM journalDirty $ lockJournal $ do
stageJournal
setUnCommitted
{- Commits the staged changes in the index to the branch. {- Commits the staged changes in the index to the branch.
- -
- Ensures that the branch's index file is first updated to the state - Ensures that the branch's index file is first updated to the state
@ -213,6 +220,7 @@ commitBranch branchref message parents = do
parentrefs <- commitparents <$> catObject committedref parentrefs <- commitparents <$> catObject committedref
when (racedetected branchref parentrefs) $ when (racedetected branchref parentrefs) $
fixrace committedref parentrefs fixrace committedref parentrefs
setCommitted
where where
-- look for "parent ref" lines and return the refs -- look for "parent ref" lines and return the refs
commitparents = map (Git.Ref . snd) . filter isparent . commitparents = map (Git.Ref . snd) . filter isparent .
@ -301,6 +309,25 @@ setIndexSha ref = do
lock <- fromRepo gitAnnexIndexLock lock <- fromRepo gitAnnexIndexLock
liftIO $ writeFile lock $ show ref ++ "\n" liftIO $ writeFile lock $ show ref ++ "\n"
{- Checks if there are uncommitted changes in the branch's index or journal. -}
unCommitted :: Annex Bool
unCommitted = do
d <- liftIO . doesFileExist =<< fromRepo gitAnnexIndexDirty
if d
then return d
else journalDirty
setUnCommitted :: Annex ()
setUnCommitted = do
file <- fromRepo gitAnnexIndexDirty
liftIO $ writeFile file "1"
setCommitted :: Annex ()
setCommitted = do
file <- fromRepo gitAnnexIndexDirty
_ <- liftIO $ tryIO $ removeFile file
return ()
{- Stages the journal into the index. -} {- Stages the journal into the index. -}
stageJournal :: Annex () stageJournal :: Annex ()
stageJournal = do stageJournal = do

View file

@ -312,9 +312,11 @@ getKeysPresent' dir = do
saveState :: Bool -> Annex () saveState :: Bool -> Annex ()
saveState oneshot = do saveState oneshot = do
Annex.Queue.flush False Annex.Queue.flush False
alwayscommit <- Git.configTrue <$> fromRepo (Git.Config.get "annex.alwayscommit" "true") unless oneshot $ do
unless (oneshot || not alwayscommit) $ do alwayscommit <- Git.configTrue <$> fromRepo (Git.Config.get "annex.alwayscommit" "true")
Annex.Branch.commit "update" if alwayscommit
then Annex.Branch.commit "update"
else Annex.Branch.stage
{- Downloads content from any of a list of urls. -} {- Downloads content from any of a list of urls. -}
downloadUrl :: [Url.URLString] -> FilePath -> Annex Bool downloadUrl :: [Url.URLString] -> FilePath -> Annex Bool

View file

@ -26,4 +26,6 @@ start = do
perform :: CommandPerform perform :: CommandPerform
perform = do perform = do
Annex.Branch.update Annex.Branch.update
-- commit explicitly, in case no remote branches were merged
Annex.Branch.commit "update"
next $ return True next $ return True

View file

@ -75,6 +75,7 @@ commit = do
showStart "commit" "" showStart "commit" ""
next $ next $ do next $ next $ do
showOutput showOutput
Annex.Branch.commit "update"
-- 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 "git-annex automatic sync"] [Param "-a", Param "-m", Param "git-annex automatic sync"]

View file

@ -22,6 +22,7 @@ module Locations (
gitAnnexJournalLock, gitAnnexJournalLock,
gitAnnexIndex, gitAnnexIndex,
gitAnnexIndexLock, gitAnnexIndexLock,
gitAnnexIndexDirty,
gitAnnexSshDir, gitAnnexSshDir,
isLinkToAnnex, isLinkToAnnex,
annexHashes, annexHashes,
@ -143,6 +144,10 @@ gitAnnexIndex r = gitAnnexDir r </> "index"
gitAnnexIndexLock :: Git.Repo -> FilePath gitAnnexIndexLock :: Git.Repo -> FilePath
gitAnnexIndexLock r = gitAnnexDir r </> "index.lck" gitAnnexIndexLock r = gitAnnexDir r </> "index.lck"
{- Flag file for .git/annex/index. -}
gitAnnexIndexDirty :: Git.Repo -> FilePath
gitAnnexIndexDirty r = gitAnnexDir r </> "index.dirty"
{- .git/annex/ssh/ is used for ssh connection caching -} {- .git/annex/ssh/ is used for ssh connection caching -}
gitAnnexSshDir :: Git.Repo -> FilePath gitAnnexSshDir :: Git.Repo -> FilePath
gitAnnexSshDir r = addTrailingPathSeparator $ gitAnnexDir r </> "ssh" gitAnnexSshDir r = addTrailingPathSeparator $ gitAnnexDir r </> "ssh"

2
debian/changelog vendored
View file

@ -37,7 +37,7 @@ git-annex (3.20120124) UNRELEASED; urgency=low
a remote's location log. a remote's location log.
* To avoid commits of data to the git-annex branch after each command * To avoid commits of data to the git-annex branch after each command
is run, set annex.alwayscommit=false. Its data will then be committed is run, set annex.alwayscommit=false. Its data will then be committed
less frequently, when a merge or sync is done. less frequently, when a merge or sync is done.
-- Joey Hess <joeyh@debian.org> Tue, 24 Jan 2012 16:21:55 -0400 -- Joey Hess <joeyh@debian.org> Tue, 24 Jan 2012 16:21:55 -0400