summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGravatar Joey Hess <joey@kitenet.net>2013-02-19 16:26:07 -0400
committerGravatar Joey Hess <joey@kitenet.net>2013-02-19 17:31:03 -0400
commit490211a7f9cf9080bc913a481d8e11865931e46b (patch)
tree71fe44cb34462002aa58cb43618384eb77efc5e0
parent4e40b54a7516bbc9070faaef31254bd22154684e (diff)
Direct mode: Support filesystems like FAT which can change their inodes each time they are mounted.
-rw-r--r--Annex.hs2
-rw-r--r--Annex/Content.hs6
-rw-r--r--Annex/Content/Direct.hs51
-rw-r--r--Annex/Direct.hs2
-rw-r--r--Locations.hs8
-rw-r--r--Utility/InodeCache.hs13
-rw-r--r--debian/changelog2
7 files changed, 69 insertions, 15 deletions
diff --git a/Annex.hs b/Annex.hs
index 454969619..2a17fffe1 100644
--- a/Annex.hs
+++ b/Annex.hs
@@ -116,6 +116,7 @@ data AnnexState = AnnexState
, flags :: M.Map String Bool
, fields :: M.Map String String
, cleanup :: M.Map String (Annex ())
+ , inodeschanged :: Maybe Bool
}
newState :: Git.Repo -> AnnexState
@@ -145,6 +146,7 @@ newState gitrepo = AnnexState
, flags = M.empty
, fields = M.empty
, cleanup = M.empty
+ , inodeschanged = Nothing
}
{- Makes an Annex state object for the specified git repo.
diff --git a/Annex/Content.hs b/Annex/Content.hs
index 871cedc42..0439cb367 100644
--- a/Annex/Content.hs
+++ b/Annex/Content.hs
@@ -304,8 +304,8 @@ prepSendAnnex key = withObjectLoc key indirect direct
direct (f:fs) = do
cache <- recordedInodeCache key
-- check that we have a good file
- ifM (liftIO $ compareInodeCache f cache)
- ( return $ Just (f, liftIO $ compareInodeCache f cache)
+ ifM (sameInodeCache f cache)
+ ( return $ Just (f, sameInodeCache f cache)
, direct fs
)
@@ -356,7 +356,7 @@ removeAnnex key = withObjectLoc key remove removedirect
cache <- recordedInodeCache key
removeInodeCache key
mapM_ (resetfile cache) fs
- resetfile cache f = whenM (liftIO $ compareInodeCache f cache) $ do
+ resetfile cache f = whenM (sameInodeCache f cache) $ do
l <- calcGitLink f key
top <- fromRepo Git.repoPath
cwd <- liftIO getCurrentDirectory
diff --git a/Annex/Content/Direct.hs b/Annex/Content/Direct.hs
index 67a966142..1ae127ef6 100644
--- a/Annex/Content/Direct.hs
+++ b/Annex/Content/Direct.hs
@@ -1,6 +1,6 @@
{- git-annex file content managing for direct mode
-
- - Copyright 2010,2012 Joey Hess <joey@kitenet.net>
+ - Copyright 2012-2013 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
@@ -14,12 +14,13 @@ module Annex.Content.Direct (
recordedInodeCache,
updateInodeCache,
writeInodeCache,
- compareInodeCache,
+ sameInodeCache,
removeInodeCache,
toInodeCache,
) where
import Common.Annex
+import qualified Annex
import Annex.Perms
import qualified Git
import Utility.TempFile
@@ -94,9 +95,7 @@ normaliseAssociatedFile file = do
- expected mtime and inode.
-}
goodContent :: Key -> FilePath -> Annex Bool
-goodContent key file = do
- old <- recordedInodeCache key
- liftIO $ compareInodeCache file old
+goodContent key file = sameInodeCache file =<< recordedInodeCache key
changedFileStatus :: Key -> FileStatus -> Annex Bool
changedFileStatus key status = do
@@ -128,3 +127,45 @@ removeInodeCache key = withInodeCacheFile key $ \f -> do
withInodeCacheFile :: Key -> (FilePath -> Annex a) -> Annex a
withInodeCacheFile key a = a =<< inRepo (gitAnnexInodeCache key)
+
+{- Checks if a file's InodeCache matches its current info.
+ -
+ - If the inodes have changed, only the size and mtime are compared.
+ -}
+sameInodeCache :: FilePath -> Maybe InodeCache -> Annex Bool
+sameInodeCache _ Nothing = return False
+sameInodeCache file (Just old) = go =<< liftIO (genInodeCache file)
+ where
+ go Nothing = return False
+ go (Just curr)
+ | curr == old = return True
+ | otherwise = ifM inodesChanged
+ ( return $ compareWeak curr old
+ , return False
+ )
+
+{- Some filesystems get new inodes each time they are mounted.
+ - In order to work on such a filesystem, a sentinal file is used to detect
+ - when the inodes have changed. -}
+inodesChanged :: Annex Bool
+inodesChanged = maybe calc return =<< Annex.getState Annex.inodeschanged
+ where
+ calc = do
+ sentinalfile <- fromRepo gitAnnexInodeSentinal
+ sentinalcachefile <- fromRepo gitAnnexInodeSentinalCache
+ scache <- liftIO $ genInodeCache sentinalfile
+ scached <- liftIO $ catchMaybeIO $ readInodeCache <$> readFile sentinalcachefile
+ case (scache, scached) of
+ (Just c1, Just (Just c2)) -> changed $ c1 /= c2
+ _ -> do
+ writesentinal
+ changed True
+ changed v = do
+ Annex.changeState $ \s -> s { Annex.inodeschanged = Just v }
+ return v
+ writesentinal = do
+ sentinalfile <- fromRepo gitAnnexInodeSentinal
+ sentinalcachefile <- fromRepo gitAnnexInodeSentinalCache
+ liftIO $ writeFile sentinalfile ""
+ liftIO $ maybe noop (writeFile sentinalcachefile . showInodeCache)
+ =<< genInodeCache sentinalfile
diff --git a/Annex/Direct.hs b/Annex/Direct.hs
index 5d616e450..c6f12a7b8 100644
--- a/Annex/Direct.hs
+++ b/Annex/Direct.hs
@@ -85,7 +85,7 @@ addDirect file cache = do
got Nothing = do
showEndFail
return False
- got (Just (key, _)) = ifM (liftIO $ compareInodeCache file $ Just cache)
+ got (Just (key, _)) = ifM (sameInodeCache file $ Just cache)
( do
stageSymlink file =<< hashSymlink =<< calcGitLink file key
writeInodeCache key cache
diff --git a/Locations.hs b/Locations.hs
index 49ccb350c..fcf516bdc 100644
--- a/Locations.hs
+++ b/Locations.hs
@@ -13,6 +13,8 @@ module Locations (
gitAnnexLocation,
gitAnnexMapping,
gitAnnexInodeCache,
+ gitAnnexInodeSentinal,
+ gitAnnexInodeSentinalCache,
annexLocations,
annexLocation,
gitAnnexDir,
@@ -128,6 +130,12 @@ gitAnnexInodeCache key r = do
loc <- gitAnnexLocation key r
return $ loc ++ ".cache"
+gitAnnexInodeSentinal :: Git.Repo -> FilePath
+gitAnnexInodeSentinal r = gitAnnexDir r </> "sentinal"
+
+gitAnnexInodeSentinalCache :: Git.Repo -> FilePath
+gitAnnexInodeSentinalCache r = gitAnnexInodeSentinal r ++ ".cache"
+
{- The annex directory of a repository. -}
gitAnnexDir :: Git.Repo -> FilePath
gitAnnexDir r = addTrailingPathSeparator $ Git.localGitDir r </> annexDir
diff --git a/Utility/InodeCache.hs b/Utility/InodeCache.hs
index 023a203f8..321125bf4 100644
--- a/Utility/InodeCache.hs
+++ b/Utility/InodeCache.hs
@@ -13,6 +13,13 @@ import System.Posix.Types
data InodeCache = InodeCache FileID FileOffset EpochTime
deriving (Eq, Show)
+{- Weak comparison of the inode caches, comparing the size and mtime, but
+ - not the actual inode. Useful when inodes have changed, perhaps
+ - due to some filesystems being remounted. -}
+compareWeak :: InodeCache -> InodeCache -> Bool
+compareWeak (InodeCache _ size1 mtime1) (InodeCache _ size2 mtime2) =
+ size1 == size2 && mtime1 == mtime2
+
showInodeCache :: InodeCache -> String
showInodeCache (InodeCache inode size mtime) = unwords
[ show inode
@@ -42,9 +49,3 @@ toInodeCache s
(fileSize s)
(modificationTime s)
| otherwise = Nothing
-
-{- Compares an inode cache with the current inode of file. -}
-compareInodeCache :: FilePath -> Maybe InodeCache -> IO Bool
-compareInodeCache file old = do
- curr <- genInodeCache file
- return $ isJust curr && curr == old
diff --git a/debian/changelog b/debian/changelog
index 76c83056b..b4bc100f5 100644
--- a/debian/changelog
+++ b/debian/changelog
@@ -8,6 +8,8 @@ git-annex (3.20130217) UNRELEASED; urgency=low
the main local repository.
* Android: Bundle now includes openssh.
* Android: Support ssh connection caching.
+ * Direct mode: Support filesystems like FAT which can change their inodes
+ each time they are mounted.
-- Joey Hess <joeyh@debian.org> Sun, 17 Feb 2013 16:42:16 -0400