git-annex/Logs/PreferredContent.hs

126 lines
4.2 KiB
Haskell
Raw Normal View History

{- git-annex preferred content matcher configuration
-
- Copyright 2012-2014 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Logs.PreferredContent (
preferredContentLog,
preferredContentSet,
groupPreferredContentSet,
isPreferredContent,
preferredContentMap,
preferredContentMapLoad,
preferredContentMapRaw,
groupPreferredContentMapRaw,
checkPreferredContentExpression,
setStandardGroup,
) where
import qualified Data.Map as M
import qualified Data.Set as S
import Data.Either
import Common.Annex
import Logs.PreferredContent.Raw
import qualified Annex.Branch
import qualified Annex
import Logs
import Logs.UUIDBased
import qualified Utility.Matcher
import Annex.FileMatcher
import Annex.UUID
2013-10-28 18:05:55 +00:00
import Types.Limit
2012-10-08 19:18:58 +00:00
import Types.Group
2013-04-26 03:44:55 +00:00
import Types.Remote (RemoteConfig)
2012-10-08 19:18:58 +00:00
import Logs.Group
2013-04-26 03:44:55 +00:00
import Logs.Remote
import Types.StandardGroups
import Limit
{- Checks if a file is preferred content for the specified repository
- (or the current repository if none is specified). -}
isPreferredContent :: Maybe UUID -> AssumeNotPresent -> Maybe Key -> AssociatedFile -> Bool -> Annex Bool
isPreferredContent mu notpresent mkey afile def = do
u <- maybe getUUID return mu
m <- preferredContentMap
case M.lookup u m of
Nothing -> return def
Just matcher -> checkMatcher matcher mkey afile notpresent def
{- The map is cached for speed. -}
preferredContentMap :: Annex Annex.PreferredContentMap
preferredContentMap = maybe preferredContentMapLoad return
=<< Annex.getState Annex.preferredcontentmap
{- Loads the map, updating the cache. -}
preferredContentMapLoad :: Annex Annex.PreferredContentMap
preferredContentMapLoad = do
2012-10-08 19:18:58 +00:00
groupmap <- groupMap
2013-04-26 03:44:55 +00:00
configmap <- readRemoteLog
2014-03-15 21:08:55 +00:00
groupwantedmap <- groupPreferredContentMapRaw
m <- simpleMap
2014-03-15 21:08:55 +00:00
. parseLogWithUUID ((Just .) . makeMatcher groupmap configmap groupwantedmap)
<$> Annex.Branch.get preferredContentLog
Annex.changeState $ \s -> s { Annex.preferredcontentmap = Just m }
return m
{- This intentionally never fails, even on unparsable expressions,
2013-12-19 09:57:50 +00:00
- because the configuration is shared among repositories and newer
- versions of git-annex may add new features. -}
makeMatcher
:: GroupMap
-> M.Map UUID RemoteConfig
-> M.Map Group PreferredContentExpression
-> UUID
-> PreferredContentExpression
-> FileMatcher
2014-03-15 21:08:55 +00:00
makeMatcher groupmap configmap groupwantedmap u = go True True
2012-11-11 04:51:07 +00:00
where
2014-03-15 21:08:55 +00:00
go expandstandard expandgroupwanted expr
| null (lefts tokens) = Utility.Matcher.generate $ rights tokens
| otherwise = unknownMatcher u
where
2014-03-15 21:08:55 +00:00
tokens = exprParser matchstandard matchgroupwanted groupmap configmap (Just u) expr
matchstandard
| expandstandard = maybe (unknownMatcher u) (go False False)
2014-03-15 21:08:55 +00:00
(standardPreferredContent <$> getStandardGroup mygroups)
| otherwise = unknownMatcher u
2014-03-15 21:08:55 +00:00
matchgroupwanted
| expandgroupwanted = maybe (unknownMatcher u) (go True False)
2014-03-15 21:08:55 +00:00
(groupwanted mygroups)
| otherwise = unknownMatcher u
2014-03-15 21:08:55 +00:00
mygroups = fromMaybe S.empty (u `M.lookup` groupsByUUID groupmap)
groupwanted s = case M.elems $ M.filterWithKey (\k _ -> S.member k s) groupwantedmap of
[pc] -> Just pc
_ -> Nothing
2012-10-10 19:15:56 +00:00
{- When a preferred content expression cannot be parsed, but is already
- in the log (eg, put there by a newer version of git-annex),
- the fallback behavior is to match only files that are currently present.
-
- This avoid unwanted/expensive changes to the content, until the problem
- is resolved. -}
unknownMatcher :: UUID -> FileMatcher
unknownMatcher u = Utility.Matcher.generate [present]
where
present = Utility.Matcher.Operation $ matchPresent (Just u)
{- Checks if an expression can be parsed, if not returns Just error -}
2014-01-01 23:58:02 +00:00
checkPreferredContentExpression :: PreferredContentExpression -> Maybe String
checkPreferredContentExpression expr = case parsedToMatcher tokens of
Left e -> Just e
Right _ -> Nothing
2012-11-11 04:51:07 +00:00
where
2014-03-15 21:08:55 +00:00
tokens = exprParser matchAll matchAll emptyGroupMap M.empty Nothing expr
{- Puts a UUID in a standard group, and sets its preferred content to use
- the standard expression for that group, unless something is already set. -}
setStandardGroup :: UUID -> StandardGroup -> Annex ()
setStandardGroup u g = do
groupSet u $ S.singleton $ fromStandardGroup g
m <- preferredContentMap
unless (isJust $ M.lookup u m) $
preferredContentSet u "standard"