2012-12-07 21:28:23 +00:00
|
|
|
{- git-annex file content managing for direct mode
|
|
|
|
-
|
2013-02-19 20:26:07 +00:00
|
|
|
- Copyright 2012-2013 Joey Hess <joey@kitenet.net>
|
2012-12-07 21:28:23 +00:00
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
module Annex.Content.Direct (
|
|
|
|
associatedFiles,
|
2012-12-12 23:20:38 +00:00
|
|
|
removeAssociatedFile,
|
|
|
|
addAssociatedFile,
|
2012-12-08 21:03:39 +00:00
|
|
|
goodContent,
|
2013-02-14 20:17:40 +00:00
|
|
|
recordedInodeCache,
|
|
|
|
updateInodeCache,
|
|
|
|
writeInodeCache,
|
2013-02-19 20:26:07 +00:00
|
|
|
sameInodeCache,
|
2013-02-22 21:01:48 +00:00
|
|
|
sameFileStatus,
|
2013-02-15 20:37:57 +00:00
|
|
|
removeInodeCache,
|
2013-02-14 20:17:40 +00:00
|
|
|
toInodeCache,
|
2013-02-20 17:55:53 +00:00
|
|
|
inodesChanged,
|
|
|
|
createInodeSentinalFile,
|
2012-12-07 21:28:23 +00:00
|
|
|
) where
|
|
|
|
|
|
|
|
import Common.Annex
|
2013-02-19 20:26:07 +00:00
|
|
|
import qualified Annex
|
2013-01-26 09:09:15 +00:00
|
|
|
import Annex.Perms
|
2012-12-07 21:28:23 +00:00
|
|
|
import qualified Git
|
2012-12-10 18:37:24 +00:00
|
|
|
import Utility.TempFile
|
2012-12-12 23:20:38 +00:00
|
|
|
import Logs.Location
|
2013-02-14 20:17:40 +00:00
|
|
|
import Utility.InodeCache
|
2012-12-07 21:28:23 +00:00
|
|
|
|
2013-01-18 16:20:08 +00:00
|
|
|
{- Absolute FilePaths of Files in the tree that are associated with a key. -}
|
2012-12-07 21:28:23 +00:00
|
|
|
associatedFiles :: Key -> Annex [FilePath]
|
|
|
|
associatedFiles key = do
|
2012-12-12 17:11:59 +00:00
|
|
|
files <- associatedFilesRelative key
|
|
|
|
top <- fromRepo Git.repoPath
|
|
|
|
return $ map (top </>) files
|
2012-12-07 21:28:23 +00:00
|
|
|
|
2012-12-12 17:11:59 +00:00
|
|
|
{- List of files in the tree that are associated with a key, relative to
|
|
|
|
- the top of the repo. -}
|
|
|
|
associatedFilesRelative :: Key -> Annex [FilePath]
|
|
|
|
associatedFilesRelative key = do
|
2012-12-10 19:02:44 +00:00
|
|
|
mapping <- inRepo $ gitAnnexMapping key
|
2013-01-18 16:26:45 +00:00
|
|
|
liftIO $ catchDefaultIO [] $ do
|
|
|
|
h <- openFile mapping ReadMode
|
|
|
|
fileEncoding h
|
2013-01-18 17:16:16 +00:00
|
|
|
lines <$> hGetContents h
|
2012-12-10 19:02:44 +00:00
|
|
|
|
2012-12-10 18:37:24 +00:00
|
|
|
{- Changes the associated files information for a key, applying a
|
2013-01-18 16:20:08 +00:00
|
|
|
- transformation to the list. Returns new associatedFiles value. -}
|
2012-12-12 23:20:38 +00:00
|
|
|
changeAssociatedFiles :: Key -> ([FilePath] -> [FilePath]) -> Annex [FilePath]
|
2012-12-10 18:37:24 +00:00
|
|
|
changeAssociatedFiles key transform = do
|
|
|
|
mapping <- inRepo $ gitAnnexMapping key
|
2012-12-12 17:11:59 +00:00
|
|
|
files <- associatedFilesRelative key
|
2012-12-10 19:02:44 +00:00
|
|
|
let files' = transform files
|
2013-01-26 09:09:15 +00:00
|
|
|
when (files /= files') $ do
|
|
|
|
createContentDir mapping
|
2013-01-18 16:26:45 +00:00
|
|
|
liftIO $ viaTmp write mapping $ unlines files'
|
2013-01-18 16:20:08 +00:00
|
|
|
top <- fromRepo Git.repoPath
|
|
|
|
return $ map (top </>) files'
|
2013-01-18 16:26:45 +00:00
|
|
|
where
|
|
|
|
write file content = do
|
|
|
|
h <- openFile file WriteMode
|
|
|
|
fileEncoding h
|
|
|
|
hPutStr h content
|
|
|
|
hClose h
|
2012-12-10 18:37:24 +00:00
|
|
|
|
2013-01-18 16:20:08 +00:00
|
|
|
{- Removes an associated file. Returns new associatedFiles value. -}
|
2012-12-12 23:20:38 +00:00
|
|
|
removeAssociatedFile :: Key -> FilePath -> Annex [FilePath]
|
|
|
|
removeAssociatedFile key file = do
|
2013-01-18 16:20:08 +00:00
|
|
|
file' <- normaliseAssociatedFile file
|
|
|
|
fs <- changeAssociatedFiles key $ filter (/= file')
|
2012-12-12 23:20:38 +00:00
|
|
|
when (null fs) $
|
|
|
|
logStatus key InfoMissing
|
|
|
|
return fs
|
2012-12-10 18:37:24 +00:00
|
|
|
|
2013-01-18 16:20:08 +00:00
|
|
|
{- Adds an associated file. Returns new associatedFiles value. -}
|
2012-12-12 23:20:38 +00:00
|
|
|
addAssociatedFile :: Key -> FilePath -> Annex [FilePath]
|
2013-01-18 16:20:08 +00:00
|
|
|
addAssociatedFile key file = do
|
|
|
|
file' <- normaliseAssociatedFile file
|
2013-02-18 06:39:40 +00:00
|
|
|
changeAssociatedFiles key $ \files ->
|
2013-01-18 16:20:08 +00:00
|
|
|
if file' `elem` files
|
|
|
|
then files
|
|
|
|
else file':files
|
|
|
|
|
|
|
|
{- Associated files are always stored relative to the top of the repository.
|
|
|
|
- The input FilePath is relative to the CWD. -}
|
|
|
|
normaliseAssociatedFile :: FilePath -> Annex FilePath
|
|
|
|
normaliseAssociatedFile file = do
|
|
|
|
top <- fromRepo Git.repoPath
|
|
|
|
liftIO $ relPathDirToFile top <$> absPath file
|
2012-12-10 18:37:24 +00:00
|
|
|
|
2012-12-07 21:28:23 +00:00
|
|
|
{- Checks if a file in the tree, associated with a key, has not been modified.
|
|
|
|
-
|
|
|
|
- To avoid needing to fsck the file's content, which can involve an
|
|
|
|
- expensive checksum, this relies on a cache that contains the file's
|
|
|
|
- expected mtime and inode.
|
|
|
|
-}
|
2012-12-08 21:03:39 +00:00
|
|
|
goodContent :: Key -> FilePath -> Annex Bool
|
2013-02-19 20:26:07 +00:00
|
|
|
goodContent key file = sameInodeCache file =<< recordedInodeCache key
|
2012-12-08 21:03:39 +00:00
|
|
|
|
2013-02-14 20:17:40 +00:00
|
|
|
{- Gets the recorded inode cache for a key. -}
|
|
|
|
recordedInodeCache :: Key -> Annex (Maybe InodeCache)
|
|
|
|
recordedInodeCache key = withInodeCacheFile key $ \f ->
|
|
|
|
liftIO $ catchDefaultIO Nothing $ readInodeCache <$> readFile f
|
2012-12-08 17:13:36 +00:00
|
|
|
|
|
|
|
{- Stores a cache of attributes for a file that is associated with a key. -}
|
2013-02-14 20:17:40 +00:00
|
|
|
updateInodeCache :: Key -> FilePath -> Annex ()
|
|
|
|
updateInodeCache key file = maybe noop (writeInodeCache key)
|
|
|
|
=<< liftIO (genInodeCache file)
|
2012-12-12 23:20:38 +00:00
|
|
|
|
|
|
|
{- Writes a cache for a key. -}
|
2013-02-14 20:17:40 +00:00
|
|
|
writeInodeCache :: Key -> InodeCache -> Annex ()
|
|
|
|
writeInodeCache key cache = withInodeCacheFile key $ \f -> do
|
|
|
|
createContentDir f
|
|
|
|
liftIO $ writeFile f $ showInodeCache cache
|
2012-12-10 18:37:24 +00:00
|
|
|
|
2013-02-15 20:37:57 +00:00
|
|
|
{- Removes an inode cache. -}
|
|
|
|
removeInodeCache :: Key -> Annex ()
|
|
|
|
removeInodeCache key = withInodeCacheFile key $ \f -> do
|
|
|
|
createContentDir f -- also thaws directory
|
|
|
|
liftIO $ nukeFile f
|
|
|
|
|
2013-02-14 20:17:40 +00:00
|
|
|
withInodeCacheFile :: Key -> (FilePath -> Annex a) -> Annex a
|
|
|
|
withInodeCacheFile key a = a =<< inRepo (gitAnnexInodeCache key)
|
2013-02-19 20:26:07 +00:00
|
|
|
|
2013-02-22 19:19:28 +00:00
|
|
|
{- Checks if a InodeCache matches the current version of a file. -}
|
2013-02-19 20:26:07 +00:00
|
|
|
sameInodeCache :: FilePath -> Maybe InodeCache -> Annex Bool
|
|
|
|
sameInodeCache _ Nothing = return False
|
|
|
|
sameInodeCache file (Just old) = go =<< liftIO (genInodeCache file)
|
|
|
|
where
|
|
|
|
go Nothing = return False
|
2013-02-22 19:19:28 +00:00
|
|
|
go (Just curr) = compareInodeCaches curr old
|
|
|
|
|
2013-02-22 21:01:48 +00:00
|
|
|
{- Checks if a FileStatus matches the recorded InodeCache of a file. -}
|
|
|
|
sameFileStatus :: Key -> FileStatus -> Annex Bool
|
|
|
|
sameFileStatus key status = do
|
|
|
|
old <- recordedInodeCache key
|
|
|
|
let curr = toInodeCache status
|
|
|
|
r <- case (old, curr) of
|
|
|
|
(Just o, Just c) -> compareInodeCaches o c
|
|
|
|
(Nothing, Nothing) -> return True
|
|
|
|
_ -> return False
|
|
|
|
return r
|
|
|
|
|
2013-02-22 19:19:28 +00:00
|
|
|
{- If the inodes have changed, only the size and mtime are compared. -}
|
|
|
|
compareInodeCaches :: InodeCache -> InodeCache -> Annex Bool
|
|
|
|
compareInodeCaches x y
|
|
|
|
| x == y = return True
|
|
|
|
| otherwise = ifM inodesChanged
|
|
|
|
( return $ compareWeak x y
|
|
|
|
, return False
|
|
|
|
)
|
2013-02-19 20:26:07 +00:00
|
|
|
|
|
|
|
{- 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
|
2013-02-20 17:55:53 +00:00
|
|
|
- when the inodes have changed.
|
|
|
|
-
|
|
|
|
- If the sentinal file does not exist, we have to assume that the
|
|
|
|
- inodes have changed.
|
|
|
|
-}
|
2013-02-19 20:26:07 +00:00
|
|
|
inodesChanged :: Annex Bool
|
|
|
|
inodesChanged = maybe calc return =<< Annex.getState Annex.inodeschanged
|
|
|
|
where
|
|
|
|
calc = do
|
2013-02-20 17:55:53 +00:00
|
|
|
scache <- liftIO . genInodeCache
|
|
|
|
=<< fromRepo gitAnnexInodeSentinal
|
|
|
|
scached <- readInodeSentinalFile
|
|
|
|
let changed = case (scache, scached) of
|
|
|
|
(Just c1, Just c2) -> c1 /= c2
|
|
|
|
_ -> True
|
|
|
|
Annex.changeState $ \s -> s { Annex.inodeschanged = Just changed }
|
|
|
|
return changed
|
|
|
|
|
|
|
|
readInodeSentinalFile :: Annex (Maybe InodeCache)
|
|
|
|
readInodeSentinalFile = do
|
|
|
|
sentinalcachefile <- fromRepo gitAnnexInodeSentinalCache
|
|
|
|
liftIO $ catchDefaultIO Nothing $
|
|
|
|
readInodeCache <$> readFile sentinalcachefile
|
|
|
|
|
|
|
|
writeInodeSentinalFile :: Annex ()
|
|
|
|
writeInodeSentinalFile = do
|
|
|
|
sentinalfile <- fromRepo gitAnnexInodeSentinal
|
|
|
|
sentinalcachefile <- fromRepo gitAnnexInodeSentinalCache
|
|
|
|
liftIO $ writeFile sentinalfile ""
|
|
|
|
liftIO $ maybe noop (writeFile sentinalcachefile . showInodeCache)
|
|
|
|
=<< genInodeCache sentinalfile
|
|
|
|
|
|
|
|
{- The sentinal file is only created when first initializing a repository.
|
|
|
|
- If there are any annexed objects in the repository already, creating
|
|
|
|
- the file would invalidate their inode caches. -}
|
|
|
|
createInodeSentinalFile :: Annex ()
|
|
|
|
createInodeSentinalFile =
|
|
|
|
unlessM (alreadyexists <||> hasobjects)
|
|
|
|
writeInodeSentinalFile
|
|
|
|
where
|
|
|
|
alreadyexists = isJust <$> readInodeSentinalFile
|
|
|
|
hasobjects = liftIO . doesDirectoryExist =<< fromRepo gitAnnexObjectDir
|