aboutsummaryrefslogtreecommitdiff
path: root/Types
diff options
context:
space:
mode:
Diffstat (limited to 'Types')
-rw-r--r--Types/RefSpec.hs44
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