git-annex/Logs/Location.hs

174 lines
4.9 KiB
Haskell
Raw Normal View History

{-# LANGUAGE BangPatterns #-}
2010-10-09 23:22:40 +00:00
{- git-annex location log
-
2011-04-02 19:50:51 +00:00
- git-annex keeps track of which repositories have the contents of annexed
- files.
2010-10-09 23:22:40 +00:00
-
2010-10-12 22:06:34 +00:00
- Repositories record their UUID and the date when they --get or --drop
2010-10-13 00:04:36 +00:00
- a value.
2010-10-10 16:31:14 +00:00
-
- Copyright 2010-2018 Joey Hess <id@joeyh.name>
2010-10-27 20:53:54 +00:00
-
- Licensed under the GNU AGPL version 3 or higher.
2010-10-09 23:22:40 +00:00
-}
2011-10-15 20:21:08 +00:00
module Logs.Location (
2010-10-12 22:25:41 +00:00
LogStatus(..),
2012-12-12 23:20:38 +00:00
logStatus,
logStatusAfter,
logChange,
loggedLocations,
loggedLocationsHistorical,
2016-07-17 19:15:08 +00:00
loggedLocationsRef,
isKnownKey,
checkDead,
setDead,
Unchecked,
finishCheck,
2011-07-01 21:23:01 +00:00
loggedKeys,
loggedKeysFor,
loggedKeysFor',
2010-10-11 21:52:46 +00:00
) where
2010-10-09 23:22:40 +00:00
import Annex.Common
2011-10-04 04:40:47 +00:00
import qualified Annex.Branch
import Logs
2011-10-15 20:21:08 +00:00
import Logs.Presence
2012-12-12 23:20:38 +00:00
import Annex.UUID
2016-07-17 19:15:08 +00:00
import Annex.CatFile
import Annex.VectorClock
2016-07-17 19:15:08 +00:00
import Git.Types (RefDate, Ref)
import qualified Annex
2012-12-12 23:20:38 +00:00
import Data.Time.Clock
2012-12-12 23:20:38 +00:00
{- Log a change in the presence of a key's value in current repository. -}
logStatus :: Key -> LogStatus -> Annex ()
2013-09-05 01:37:13 +00:00
logStatus key s = do
2012-12-12 23:20:38 +00:00
u <- getUUID
2013-09-05 01:37:13 +00:00
logChange key u s
2010-10-09 23:22:40 +00:00
{- Run an action that gets the content of a key, and update the log
- when it succeeds. -}
logStatusAfter :: Key -> Annex Bool -> Annex Bool
logStatusAfter key a = ifM a
( do
logStatus key InfoPresent
return True
, return False
)
2011-06-22 20:01:32 +00:00
{- Log a change in the presence of a key's value in a repository. -}
2011-11-09 05:15:51 +00:00
logChange :: Key -> UUID -> LogStatus -> Annex ()
logChange = logChange' logNow
logChange' :: (LogStatus -> LogInfo -> Annex LogLine) -> Key -> UUID -> LogStatus -> Annex ()
logChange' mklog key u@(UUID _) s = do
config <- Annex.getGitConfig
maybeAddLog (locationLogFile config key) =<< mklog s (LogInfo (fromUUID u))
logChange' _ _ NoUUID _ = noop
2010-10-12 22:25:41 +00:00
{- Returns a list of repository UUIDs that, according to the log, have
- the value of a key. -}
loggedLocations :: Key -> Annex [UUID]
loggedLocations = getLoggedLocations currentLogInfo
{- Gets the location log on a particular date. -}
loggedLocationsHistorical :: RefDate -> Key -> Annex [UUID]
loggedLocationsHistorical = getLoggedLocations . historicalLogInfo
2016-07-17 19:15:08 +00:00
{- Gets the locations contained in a git ref. -}
loggedLocationsRef :: Ref -> Annex [UUID]
loggedLocationsRef ref = map (toUUID . fromLogInfo) . getLog <$> catObject ref
2016-07-17 19:15:08 +00:00
getLoggedLocations :: (RawFilePath -> Annex [LogInfo]) -> Key -> Annex [UUID]
getLoggedLocations getter key = do
config <- Annex.getGitConfig
map (toUUID . fromLogInfo) <$> getter (locationLogFile config key)
{- Is there a location log for the key? True even for keys with no
- remaining locations. -}
isKnownKey :: Key -> Annex Bool
isKnownKey key = do
config <- Annex.getGitConfig
not . null <$> readLog (locationLogFile config key)
{- For a key to be dead, all locations that have location status for the key
- must have InfoDead set. -}
checkDead :: Key -> Annex Bool
checkDead key = do
config <- Annex.getGitConfig
ls <- compactLog <$> readLog (locationLogFile config key)
return $! all (\l -> status l == InfoDead) ls
{- Updates the log to say that a key is dead.
-
- Changes all logged lines for the key, in any location, that are
- currently InfoMissing, to be InfoDead.
-}
setDead :: Key -> Annex ()
setDead key = do
config <- Annex.getGitConfig
let logfile = locationLogFile config key
ls <- compactLog <$> readLog logfile
mapM_ (go logfile) (filter (\l -> status l == InfoMissing) ls)
where
go logfile l = addLog logfile $ setDead' l
{- Note that the timestamp in the log is updated minimally, so that this
- can be overruled by other location log changes. -}
setDead' :: LogLine -> LogLine
setDead' l = l
{ status = InfoDead
, date = case date l of
VectorClock c -> VectorClock $
c + realToFrac (picosecondsToDiffTime 1)
Unknown -> Unknown
}
2011-04-02 19:50:51 +00:00
data Unchecked a = Unchecked (Annex (Maybe a))
finishCheck :: Unchecked a -> Annex (Maybe a)
finishCheck (Unchecked a) = a
{- Finds all keys that have location log information.
- (There may be duplicate keys in the list.)
-
- Keys that have been marked as dead are not included.
-}
loggedKeys :: Annex ([Unchecked Key], IO Bool)
loggedKeys = loggedKeys' (not <$$> checkDead)
loggedKeys' :: (Key -> Annex Bool) -> Annex ([Unchecked Key], IO Bool)
loggedKeys' check = do
config <- Annex.getGitConfig
(bfs, cleanup) <- Annex.Branch.files
let l = mapMaybe (defercheck <$$> locationLogFileKey config) bfs
return (l, cleanup)
where
defercheck k = Unchecked $ ifM (check k)
( return (Just k)
, return Nothing
)
2011-07-01 21:23:01 +00:00
{- Finds all keys that have location log information indicating
- they are present in the specified repository.
-
- This does not stream well; use loggedKeysFor' for lazy streaming.
-}
loggedKeysFor :: UUID -> Annex [Key]
loggedKeysFor u = do
(l, cleanup) <- loggedKeysFor' u
l' <- catMaybes <$> mapM finishCheck l
liftIO $ void cleanup
return l'
loggedKeysFor' :: UUID -> Annex ([Unchecked Key], IO Bool)
loggedKeysFor' u = loggedKeys' isthere
2012-11-11 04:51:07 +00:00
where
isthere k = do
us <- loggedLocations k
let !there = u `elem` us
return there