Update working tree files fully atomically

This avoids commit churn by the assistant when eg,
replacing a file with a symlink.

But, just as importantly, it prevents the working tree being left with a
deleted file if git-annex, or perhaps the whole system, crashes at the
wrong time.

(It also probably avoids confusing displays in file managers.)
This commit is contained in:
Joey Hess 2013-04-02 13:13:42 -04:00
parent 8c52b20cc7
commit 38d61f934d
8 changed files with 34 additions and 21 deletions

View file

@ -49,6 +49,7 @@ import Config
import Annex.Exception import Annex.Exception
import Git.SharedRepository import Git.SharedRepository
import Annex.Perms import Annex.Perms
import Annex.Link
import Annex.Content.Direct import Annex.Content.Direct
import Backend import Backend
@ -256,20 +257,33 @@ moveAnnex key src = withObjectLoc key storeobject storedirect
updateInodeCache key src updateInodeCache key src
thawContent src thawContent src
replaceFile dest $ liftIO . moveFile src replaceFile dest $ liftIO . moveFile src
{- Copy to any other locations. -}
forM_ fs $ \f -> replaceFile f $ forM_ fs $ \f -> replaceFile f $
void . liftIO . copyFileExternal dest liftIO . void . copyFileExternal dest
{- Replaces any existing file with a new version, by running an action. {- Replaces a possibly already existing file with a new version,
- First, makes sure the file is deleted. Or, if it didn't already exist, - atomically, by running an action.
- makes sure the parent directory exists. -}
- The action is passed a temp file, which it can write to, and once
- done the temp file is moved into place.
-}
replaceFile :: FilePath -> (FilePath -> Annex ()) -> Annex () replaceFile :: FilePath -> (FilePath -> Annex ()) -> Annex ()
replaceFile file a = do replaceFile file a = do
tmpdir <- fromRepo gitAnnexTmpDir
createAnnexDirectory tmpdir
tmpfile <- liftIO $ do
(tmpfile, h) <- openTempFileWithDefaultPermissions tmpdir $
takeFileName file
hClose h
return tmpfile
a tmpfile
liftIO $ do liftIO $ do
r <- tryIO $ removeFile file r <- tryIO $ rename tmpfile file
case r of case r of
Left _ -> createDirectoryIfMissing True $ parentDir file Left _ -> do
createDirectoryIfMissing True $ parentDir file
rename tmpfile file
_ -> noop _ -> noop
a file
{- Runs an action to transfer an object's content. {- Runs an action to transfer an object's content.
- -
@ -366,8 +380,7 @@ removeAnnex key = withObjectLoc key remove removedirect
cwd <- liftIO getCurrentDirectory cwd <- liftIO getCurrentDirectory
let top' = fromMaybe top $ absNormPath cwd top let top' = fromMaybe top $ absNormPath cwd top
let l' = relPathDirToFile top' (fromMaybe l $ absNormPath top' l) let l' = relPathDirToFile top' (fromMaybe l $ absNormPath top' l)
replaceFile f $ const $ replaceFile f $ makeAnnexLink l'
makeAnnexLink l' f
{- Moves a key's file out of .git/annex/objects/ -} {- Moves a key's file out of .git/annex/objects/ -}
fromAnnex :: Key -> FilePath -> Annex () fromAnnex :: Key -> FilePath -> Annex ()

View file

@ -153,8 +153,7 @@ mergeDirectCleanup d oldsha newsha = do
- Symlinks are replaced with their content, if it's available. -} - Symlinks are replaced with their content, if it's available. -}
movein k f = do movein k f = do
l <- calcGitLink f k l <- calcGitLink f k
replaceFile f $ replaceFile f $ makeAnnexLink l
makeAnnexLink l
toDirect k f toDirect k f
{- Any new, modified, or renamed files were written to the temp {- Any new, modified, or renamed files were written to the temp
@ -179,15 +178,14 @@ toDirectGen k f = do
{- Move content from annex to direct file. -} {- Move content from annex to direct file. -}
updateInodeCache k loc updateInodeCache k loc
thawContent loc thawContent loc
replaceFile f $ replaceFile f $ liftIO . moveFile loc
liftIO . moveFile loc
, return Nothing , return Nothing
) )
(loc':_) -> ifM (isNothing <$> getAnnexLinkTarget loc') (loc':_) -> ifM (isNothing <$> getAnnexLinkTarget loc')
{- Another direct file has the content; copy it. -} {- Another direct file has the content; copy it. -}
( return $ Just $ ( return $ Just $
replaceFile f $ replaceFile f $
void . liftIO . copyFileExternal loc' liftIO . void . copyFileExternal loc'
, return Nothing , return Nothing
) )

View file

@ -60,7 +60,9 @@ getAnnexLinkTarget file = do
-} -}
makeAnnexLink :: LinkTarget -> FilePath -> Annex () makeAnnexLink :: LinkTarget -> FilePath -> Annex ()
makeAnnexLink linktarget file = ifM (coreSymlinks <$> Annex.getGitConfig) makeAnnexLink linktarget file = ifM (coreSymlinks <$> Annex.getGitConfig)
( liftIO $ createSymbolicLink linktarget file ( liftIO $ do
void $ tryIO $ removeFile file
createSymbolicLink linktarget file
, liftIO $ writeFile file linktarget , liftIO $ writeFile file linktarget
) )

View file

@ -222,9 +222,9 @@ onAddSymlink isdirect file filestatus = go =<< liftAnnex (Backend.lookupFile fil
ifM ((==) (Just link) <$> liftIO (catchMaybeIO $ readSymbolicLink file)) ifM ((==) (Just link) <$> liftIO (catchMaybeIO $ readSymbolicLink file))
( ensurestaged (Just link) (Just key) =<< getDaemonStatus ( ensurestaged (Just link) (Just key) =<< getDaemonStatus
, do , do
unless isdirect $ do unless isdirect $
liftIO $ removeFile file liftAnnex $ replaceFile file $
liftAnnex $ Backend.makeAnnexLink link file makeAnnexLink link
addLink file link (Just key) addLink file link (Just key)
) )
go Nothing = do -- other symlink go Nothing = do -- other symlink

View file

@ -11,7 +11,6 @@ module Backend (
genKey, genKey,
lookupFile, lookupFile,
isAnnexLink, isAnnexLink,
makeAnnexLink,
chooseBackend, chooseBackend,
lookupBackendName, lookupBackendName,
maybeLookupBackendName maybeLookupBackendName

View file

@ -175,7 +175,7 @@ undo file key e = do
link :: FilePath -> Key -> Bool -> Annex String link :: FilePath -> Key -> Bool -> Annex String
link file key hascontent = handle (undo file key) $ do link file key hascontent = handle (undo file key) $ do
l <- calcGitLink file key l <- calcGitLink file key
makeAnnexLink l file replaceFile file $ makeAnnexLink l
#ifndef __ANDROID__ #ifndef __ANDROID__
when hascontent $ do when hascontent $ do

View file

@ -148,7 +148,7 @@ gitAnnexObjectDir r = addTrailingPathSeparator $ Git.localGitDir r </> objectDir
gitAnnexTmpDir :: Git.Repo -> FilePath gitAnnexTmpDir :: Git.Repo -> FilePath
gitAnnexTmpDir r = addTrailingPathSeparator $ gitAnnexDir r </> "tmp" gitAnnexTmpDir r = addTrailingPathSeparator $ gitAnnexDir r </> "tmp"
{- The temp file to use for a given key. -} {- The temp file to use for a given key's content. -}
gitAnnexTmpLocation :: Key -> Git.Repo -> FilePath gitAnnexTmpLocation :: Key -> Git.Repo -> FilePath
gitAnnexTmpLocation key r = gitAnnexTmpDir r </> keyFile key gitAnnexTmpLocation key r = gitAnnexTmpDir r </> keyFile key

1
debian/changelog vendored
View file

@ -26,6 +26,7 @@ git-annex (4.20130324) UNRELEASED; urgency=low
repositories. repositories.
* assistant: Fix bug that could cause direct mode files to be unstaged * assistant: Fix bug that could cause direct mode files to be unstaged
from git. from git.
* Update working tree files fully atomically.
-- Joey Hess <joeyh@debian.org> Mon, 25 Mar 2013 10:21:46 -0400 -- Joey Hess <joeyh@debian.org> Mon, 25 Mar 2013 10:21:46 -0400