Preferred content path matching bugfix.
When in a subdir, both the normal filepath, and the filepath relative to the top of the git repo are needed for matching. The former for key lookup, and the latter for include/exclude to match against. Previously, key lookup didn't work in this situation.
This commit is contained in:
parent
59170c4187
commit
e7780a39f5
6 changed files with 37 additions and 29 deletions
10
Annex.hs
10
Annex.hs
|
@ -10,6 +10,7 @@
|
||||||
module Annex (
|
module Annex (
|
||||||
Annex,
|
Annex,
|
||||||
AnnexState(..),
|
AnnexState(..),
|
||||||
|
FileInfo(..),
|
||||||
PreferredContentMap,
|
PreferredContentMap,
|
||||||
new,
|
new,
|
||||||
newState,
|
newState,
|
||||||
|
@ -77,7 +78,12 @@ instance MonadBaseControl IO Annex where
|
||||||
|
|
||||||
type Matcher a = Either [Utility.Matcher.Token a] (Utility.Matcher.Matcher a)
|
type Matcher a = Either [Utility.Matcher.Token a] (Utility.Matcher.Matcher a)
|
||||||
|
|
||||||
type PreferredContentMap = M.Map UUID (Utility.Matcher.Matcher (S.Set UUID -> FilePath -> Annex Bool))
|
data FileInfo = FileInfo
|
||||||
|
{ relFile :: FilePath -- may be relative to cwd
|
||||||
|
, matchFile :: FilePath -- filepath to match on; may be relative to top
|
||||||
|
}
|
||||||
|
|
||||||
|
type PreferredContentMap = M.Map UUID (Utility.Matcher.Matcher (S.Set UUID -> FileInfo -> Annex Bool))
|
||||||
|
|
||||||
-- internal state storage
|
-- internal state storage
|
||||||
data AnnexState = AnnexState
|
data AnnexState = AnnexState
|
||||||
|
@ -94,7 +100,7 @@ data AnnexState = AnnexState
|
||||||
, checkattrhandle :: Maybe CheckAttrHandle
|
, checkattrhandle :: Maybe CheckAttrHandle
|
||||||
, forcebackend :: Maybe String
|
, forcebackend :: Maybe String
|
||||||
, forcenumcopies :: Maybe Int
|
, forcenumcopies :: Maybe Int
|
||||||
, limit :: Matcher (FilePath -> Annex Bool)
|
, limit :: Matcher (FileInfo -> Annex Bool)
|
||||||
, preferredcontentmap :: Maybe PreferredContentMap
|
, preferredcontentmap :: Maybe PreferredContentMap
|
||||||
, shared :: Maybe SharedRepository
|
, shared :: Maybe SharedRepository
|
||||||
, forcetrust :: TrustMap
|
, forcetrust :: TrustMap
|
||||||
|
|
|
@ -9,7 +9,6 @@ module Annex.Wanted where
|
||||||
|
|
||||||
import Common.Annex
|
import Common.Annex
|
||||||
import Logs.PreferredContent
|
import Logs.PreferredContent
|
||||||
import Git.FilePath
|
|
||||||
import Annex.UUID
|
import Annex.UUID
|
||||||
import Types.Remote
|
import Types.Remote
|
||||||
|
|
||||||
|
@ -18,22 +17,17 @@ import qualified Data.Set as S
|
||||||
{- Check if a file is preferred content for the local repository. -}
|
{- Check if a file is preferred content for the local repository. -}
|
||||||
wantGet :: AssociatedFile -> Annex Bool
|
wantGet :: AssociatedFile -> Annex Bool
|
||||||
wantGet Nothing = return True
|
wantGet Nothing = return True
|
||||||
wantGet (Just file) = do
|
wantGet (Just file) = isPreferredContent Nothing S.empty file
|
||||||
fp <- inRepo $ toTopFilePath file
|
|
||||||
isPreferredContent Nothing S.empty fp
|
|
||||||
|
|
||||||
{- Check if a file is preferred content for a remote. -}
|
{- Check if a file is preferred content for a remote. -}
|
||||||
wantSend :: AssociatedFile -> UUID -> Annex Bool
|
wantSend :: AssociatedFile -> UUID -> Annex Bool
|
||||||
wantSend Nothing _ = return True
|
wantSend Nothing _ = return True
|
||||||
wantSend (Just file) to = do
|
wantSend (Just file) to = isPreferredContent (Just to) S.empty file
|
||||||
fp <- inRepo $ toTopFilePath file
|
|
||||||
isPreferredContent (Just to) S.empty fp
|
|
||||||
|
|
||||||
{- Check if a file can be dropped, maybe from a remote.
|
{- Check if a file can be dropped, maybe from a remote.
|
||||||
- Don't drop files that are preferred content. -}
|
- Don't drop files that are preferred content. -}
|
||||||
wantDrop :: Maybe UUID -> AssociatedFile -> Annex Bool
|
wantDrop :: Maybe UUID -> AssociatedFile -> Annex Bool
|
||||||
wantDrop _ Nothing = return True
|
wantDrop _ Nothing = return True
|
||||||
wantDrop from (Just file) = do
|
wantDrop from (Just file) = do
|
||||||
fp <- inRepo $ toTopFilePath file
|
|
||||||
u <- maybe getUUID (return . id) from
|
u <- maybe getUUID (return . id) from
|
||||||
not <$> isPreferredContent (Just u) (S.singleton u) fp
|
not <$> isPreferredContent (Just u) (S.singleton u) file
|
||||||
|
|
29
Limit.hs
29
Limit.hs
|
@ -28,7 +28,7 @@ import Logs.Group
|
||||||
import Utility.HumanTime
|
import Utility.HumanTime
|
||||||
import Utility.DataUnits
|
import Utility.DataUnits
|
||||||
|
|
||||||
type MatchFiles = AssumeNotPresent -> FilePath -> Annex Bool
|
type MatchFiles = AssumeNotPresent -> Annex.FileInfo -> Annex Bool
|
||||||
type MkLimit = String -> Either String MatchFiles
|
type MkLimit = String -> Either String MatchFiles
|
||||||
type AssumeNotPresent = S.Set UUID
|
type AssumeNotPresent = S.Set UUID
|
||||||
|
|
||||||
|
@ -38,10 +38,10 @@ limited = (not . Utility.Matcher.matchesAny) <$> getMatcher'
|
||||||
|
|
||||||
{- Gets a matcher for the user-specified limits. The matcher is cached for
|
{- Gets a matcher for the user-specified limits. The matcher is cached for
|
||||||
- speed; once it's obtained the user-specified limits can't change. -}
|
- speed; once it's obtained the user-specified limits can't change. -}
|
||||||
getMatcher :: Annex (FilePath -> Annex Bool)
|
getMatcher :: Annex (Annex.FileInfo -> Annex Bool)
|
||||||
getMatcher = Utility.Matcher.matchM <$> getMatcher'
|
getMatcher = Utility.Matcher.matchM <$> getMatcher'
|
||||||
|
|
||||||
getMatcher' :: Annex (Utility.Matcher.Matcher (FilePath -> Annex Bool))
|
getMatcher' :: Annex (Utility.Matcher.Matcher (Annex.FileInfo -> Annex Bool))
|
||||||
getMatcher' = do
|
getMatcher' = do
|
||||||
m <- Annex.getState Annex.limit
|
m <- Annex.getState Annex.limit
|
||||||
case m of
|
case m of
|
||||||
|
@ -52,7 +52,7 @@ getMatcher' = do
|
||||||
return matcher
|
return matcher
|
||||||
|
|
||||||
{- Adds something to the limit list, which is built up reversed. -}
|
{- Adds something to the limit list, which is built up reversed. -}
|
||||||
add :: Utility.Matcher.Token (FilePath -> Annex Bool) -> Annex ()
|
add :: Utility.Matcher.Token (Annex.FileInfo -> Annex Bool) -> Annex ()
|
||||||
add l = Annex.changeState $ \s -> s { Annex.limit = prepend $ Annex.limit s }
|
add l = Annex.changeState $ \s -> s { Annex.limit = prepend $ Annex.limit s }
|
||||||
where
|
where
|
||||||
prepend (Left ls) = Left $ l:ls
|
prepend (Left ls) = Left $ l:ls
|
||||||
|
@ -80,8 +80,9 @@ addExclude = addLimit . limitExclude
|
||||||
limitExclude :: MkLimit
|
limitExclude :: MkLimit
|
||||||
limitExclude glob = Right $ const $ return . not . matchglob glob
|
limitExclude glob = Right $ const $ return . not . matchglob glob
|
||||||
|
|
||||||
matchglob :: String -> FilePath -> Bool
|
matchglob :: String -> Annex.FileInfo -> Bool
|
||||||
matchglob glob f = isJust $ match cregex f []
|
matchglob glob (Annex.FileInfo { Annex.matchFile = f }) =
|
||||||
|
isJust $ match cregex f []
|
||||||
where
|
where
|
||||||
cregex = compile regex []
|
cregex = compile regex []
|
||||||
regex = '^':wildToRegex glob
|
regex = '^':wildToRegex glob
|
||||||
|
@ -97,7 +98,7 @@ limitIn name = Right $ \notpresent -> check $
|
||||||
then inhere notpresent
|
then inhere notpresent
|
||||||
else inremote notpresent
|
else inremote notpresent
|
||||||
where
|
where
|
||||||
check a = Backend.lookupFile >=> handle a
|
check a = lookupFile >=> handle a
|
||||||
handle _ Nothing = return False
|
handle _ Nothing = return False
|
||||||
handle a (Just (key, _)) = a key
|
handle a (Just (key, _)) = a key
|
||||||
inremote notpresent key = do
|
inremote notpresent key = do
|
||||||
|
@ -127,8 +128,8 @@ limitCopies want = case split ":" want of
|
||||||
where
|
where
|
||||||
go num good = case readish num of
|
go num good = case readish num of
|
||||||
Nothing -> Left "bad number for copies"
|
Nothing -> Left "bad number for copies"
|
||||||
Just n -> Right $ \notpresent ->
|
Just n -> Right $ \notpresent f ->
|
||||||
Backend.lookupFile >=> handle n good notpresent
|
lookupFile f >>= handle n good notpresent
|
||||||
handle _ _ _ Nothing = return False
|
handle _ _ _ Nothing = return False
|
||||||
handle n good notpresent (Just (key, _)) = do
|
handle n good notpresent (Just (key, _)) = do
|
||||||
us <- filter (`S.notMember` notpresent)
|
us <- filter (`S.notMember` notpresent)
|
||||||
|
@ -147,8 +148,7 @@ addInAllGroup groupname = do
|
||||||
limitInAllGroup :: GroupMap -> MkLimit
|
limitInAllGroup :: GroupMap -> MkLimit
|
||||||
limitInAllGroup m groupname
|
limitInAllGroup m groupname
|
||||||
| S.null want = Right $ const $ const $ return True
|
| S.null want = Right $ const $ const $ return True
|
||||||
| otherwise = Right $ \notpresent ->
|
| otherwise = Right $ \notpresent -> lookupFile >=> check notpresent
|
||||||
Backend.lookupFile >=> check notpresent
|
|
||||||
where
|
where
|
||||||
want = fromMaybe S.empty $ M.lookup groupname $ uuidsByGroup m
|
want = fromMaybe S.empty $ M.lookup groupname $ uuidsByGroup m
|
||||||
check _ Nothing = return False
|
check _ Nothing = return False
|
||||||
|
@ -164,7 +164,7 @@ addInBackend :: String -> Annex ()
|
||||||
addInBackend = addLimit . limitInBackend
|
addInBackend = addLimit . limitInBackend
|
||||||
|
|
||||||
limitInBackend :: MkLimit
|
limitInBackend :: MkLimit
|
||||||
limitInBackend name = Right $ const $ Backend.lookupFile >=> check
|
limitInBackend name = Right $ const $ lookupFile >=> check
|
||||||
where
|
where
|
||||||
wanted = Backend.lookupBackendName name
|
wanted = Backend.lookupBackendName name
|
||||||
check = return . maybe False ((==) wanted . snd)
|
check = return . maybe False ((==) wanted . snd)
|
||||||
|
@ -179,7 +179,7 @@ addSmallerThan = addLimit . limitSize (<)
|
||||||
limitSize :: (Maybe Integer -> Maybe Integer -> Bool) -> MkLimit
|
limitSize :: (Maybe Integer -> Maybe Integer -> Bool) -> MkLimit
|
||||||
limitSize vs s = case readSize dataUnits s of
|
limitSize vs s = case readSize dataUnits s of
|
||||||
Nothing -> Left "bad size"
|
Nothing -> Left "bad size"
|
||||||
Just sz -> Right $ const $ Backend.lookupFile >=> check sz
|
Just sz -> Right $ const $ lookupFile >=> check sz
|
||||||
where
|
where
|
||||||
check _ Nothing = return False
|
check _ Nothing = return False
|
||||||
check sz (Just (key, _)) = return $ keySize key `vs` Just sz
|
check sz (Just (key, _)) = return $ keySize key `vs` Just sz
|
||||||
|
@ -196,3 +196,6 @@ addTimeLimit s = do
|
||||||
warning $ "Time limit (" ++ s ++ ") reached!"
|
warning $ "Time limit (" ++ s ++ ") reached!"
|
||||||
liftIO $ exitWith $ ExitFailure 101
|
liftIO $ exitWith $ ExitFailure 101
|
||||||
else return True
|
else return True
|
||||||
|
|
||||||
|
lookupFile :: Annex.FileInfo -> Annex (Maybe (Key, Backend))
|
||||||
|
lookupFile = Backend.lookupFile . Annex.relFile
|
||||||
|
|
|
@ -46,15 +46,19 @@ preferredContentSet NoUUID _ = error "unknown UUID; cannot modify"
|
||||||
|
|
||||||
{- Checks if a file is preferred content for the specified repository
|
{- Checks if a file is preferred content for the specified repository
|
||||||
- (or the current repository if none is specified). -}
|
- (or the current repository if none is specified). -}
|
||||||
isPreferredContent :: Maybe UUID -> AssumeNotPresent -> TopFilePath -> Annex Bool
|
isPreferredContent :: Maybe UUID -> AssumeNotPresent -> FilePath -> Annex Bool
|
||||||
isPreferredContent mu notpresent file = do
|
isPreferredContent mu notpresent file = do
|
||||||
|
matchfile <- getTopFilePath <$> inRepo (toTopFilePath file)
|
||||||
|
let fi = Annex.FileInfo
|
||||||
|
{ Annex.matchFile = matchfile
|
||||||
|
, Annex.relFile = file
|
||||||
|
}
|
||||||
u <- maybe getUUID return mu
|
u <- maybe getUUID return mu
|
||||||
m <- preferredContentMap
|
m <- preferredContentMap
|
||||||
case M.lookup u m of
|
case M.lookup u m of
|
||||||
Nothing -> return True
|
Nothing -> return True
|
||||||
Just matcher ->
|
Just matcher -> Utility.Matcher.matchMrun matcher $ \a ->
|
||||||
Utility.Matcher.matchMrun matcher $ \a ->
|
a notpresent fi
|
||||||
a notpresent (getTopFilePath file)
|
|
||||||
|
|
||||||
{- Read the preferredContentLog into a map. The map is cached for speed. -}
|
{- Read the preferredContentLog into a map. The map is cached for speed. -}
|
||||||
preferredContentMap :: Annex Annex.PreferredContentMap
|
preferredContentMap :: Annex Annex.PreferredContentMap
|
||||||
|
|
2
Seek.hs
2
Seek.hs
|
@ -113,7 +113,7 @@ prepFiltered a fs = do
|
||||||
map (process matcher) <$> fs
|
map (process matcher) <$> fs
|
||||||
where
|
where
|
||||||
process matcher f = do
|
process matcher f = do
|
||||||
ok <- matcher f
|
ok <- matcher $ Annex.FileInfo f f
|
||||||
if ok then a f else return Nothing
|
if ok then a f else return Nothing
|
||||||
|
|
||||||
notSymlink :: FilePath -> IO Bool
|
notSymlink :: FilePath -> IO Bool
|
||||||
|
|
1
debian/changelog
vendored
1
debian/changelog
vendored
|
@ -1,6 +1,7 @@
|
||||||
git-annex (3.20121018) UNRELEASED; urgency=low
|
git-annex (3.20121018) UNRELEASED; urgency=low
|
||||||
|
|
||||||
* Fix handling of GIT_DIR when it refers to a git submodule.
|
* Fix handling of GIT_DIR when it refers to a git submodule.
|
||||||
|
* Preferred content path matching bugfix.
|
||||||
|
|
||||||
-- Joey Hess <joeyh@debian.org> Wed, 17 Oct 2012 14:24:10 -0400
|
-- Joey Hess <joeyh@debian.org> Wed, 17 Oct 2012 14:24:10 -0400
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue