diff options
Diffstat (limited to 'Types')
-rw-r--r-- | Types/RefSpec.hs | 44 |
1 files changed, 44 insertions, 0 deletions
diff --git a/Types/RefSpec.hs b/Types/RefSpec.hs new file mode 100644 index 000000000..42f4c6226 --- /dev/null +++ b/Types/RefSpec.hs @@ -0,0 +1,44 @@ +{- This is not the same as git's fetch/push refspecs. + - + - Copyright 2015 Joey Hess <id@joeyh.name> + - + - Licensed under the GNU GPL version 3 or higher. + -} + +module Types.RefSpec where + +import Common +import Utility.Glob +import Git.Types + +import Data.Either + +type RefSpec = [RefSpecPart] + +data RefSpecPart = AddRef Ref | AddMatching Glob | RemoveMatching Glob + +allRefSpec :: RefSpec +allRefSpec = [AddMatching $ compileGlob "*" CaseSensative] + +parseRefSpec :: String -> Either String RefSpec +parseRefSpec v = case partitionEithers (map mk $ split ":" v) of + ([],refspec) -> Right refspec + (e:_,_) -> Left e + where + mk ('+':s) + | any (`elem` s) "*?" = + Right $ AddMatching $ compileGlob s CaseSensative + | otherwise = Right $ AddRef $ Ref s + mk ('-':s) = Right $ RemoveMatching $ compileGlob s CaseSensative + mk s = Left $ "bad refspec item \"" ++ s ++ "\" (expected + or - prefix)" + +applyRefSpec :: RefSpec -> [Ref] -> [Ref] +applyRefSpec refspec rs = go [] refspec + where + go c [] = reverse c + go c (AddRef r : rest) = go (r:c) rest + go c (AddMatching g : rest) = + let add = filter (matchGlob g . fromRef) rs + in go (add ++ c) rest + go c (RemoveMatching g : rest) = + go (filter (not . matchGlob g . fromRef) c) rest |