diff options
Diffstat (limited to 'Command/Vicfg.hs')
-rw-r--r-- | Command/Vicfg.hs | 80 |
1 files changed, 70 insertions, 10 deletions
diff --git a/Command/Vicfg.hs b/Command/Vicfg.hs index 7e073a00a..3170488e5 100644 --- a/Command/Vicfg.hs +++ b/Command/Vicfg.hs @@ -11,6 +11,7 @@ import qualified Data.Map as M import qualified Data.Set as S import System.Environment (getEnv) import Data.Tuple (swap) +import Data.Char (isSpace) import Common.Annex import Command @@ -31,23 +32,24 @@ seek = [withNothing start] start :: CommandStart start = do f <- fromRepo gitAnnexTmpCfgFile - createAnnexDirectory (parentDir f) - liftIO . writeFile f =<< genCfg <$> getCfg - vicfg f + createAnnexDirectory $ parentDir f + cfg <- getCfg + liftIO $ writeFile f $ genCfg cfg + vicfg cfg f stop -vicfg :: FilePath -> Annex () -vicfg f = do +vicfg :: Cfg -> FilePath -> Annex () +vicfg curcfg f = do vi <- liftIO $ catchDefaultIO "vi" $ getEnv "EDITOR" -- Allow EDITOR to be processed by the shell, so it can contain options. unlessM (liftIO $ boolSystem "sh" [Param "-c", Param $ unwords [vi, f]]) $ error $ vi ++ " exited nonzero; aborting" - r <- parseCfg <$> liftIO (readFileStrict f) + r <- parseCfg curcfg <$> liftIO (readFileStrict f) liftIO $ nukeFile f case r of Left s -> do liftIO $ writeFile f s - vicfg f + vicfg curcfg f Right c -> setCfg c data Cfg = Cfg @@ -62,6 +64,9 @@ getCfg = Cfg <*> (groupsByUUID <$> groupMap) <*> uuidDescriptions +emptyCfg :: Cfg +emptyCfg = Cfg M.empty M.empty M.empty + setCfg :: Cfg -> Annex () setCfg = error "TODO setCfg" @@ -97,13 +102,13 @@ genCfg cfg = unlines $ concat sort $ map swap $ M.toList $ cfgGroupMap cfg defaultgroups = map (\u -> pcom $ line "group" u "") $ missing cfgGroupMap + line setting u value = unwords [ setting , showu u , "=" , value ] - com s = "# " ++ s pcom s = "#" ++ s showu u = fromMaybe (fromUUID u) $ M.lookup u (cfgDescriptions cfg) @@ -111,5 +116,60 @@ genCfg cfg = unlines $ concat {- If there's a parse error, returns a new version of the file, - with the problem lines noted. -} -parseCfg :: String -> Either String Cfg -parseCfg = undefined +parseCfg :: Cfg -> String -> Either String Cfg +parseCfg curcfg = go [] curcfg . lines + where + go c cfg [] + | null (catMaybes $ map fst c) = Right cfg + | otherwise = Left $ unlines $ + badheader ++ concatMap showerr (reverse c) + go c cfg (l:ls) = case parse (dropWhile isSpace l) cfg of + Left msg -> go ((Just msg, l):c) cfg ls + Right cfg' -> go ((Nothing, l):c) cfg' ls + + parse l cfg + | null l = Right cfg + | "#" `isPrefixOf` l = Right cfg + | null setting || null repo' = Left "missing repository name" + | otherwise = case M.lookup repo' name2uuid of + Nothing -> badval "repository" repo' + Just u -> handle cfg u setting value' + where + (setting, rest) = separate isSpace l + (repo, value) = separate (== '=') rest + value' = dropWhile isSpace value + repo' = reverse $ dropWhile isSpace $ + reverse $ dropWhile isSpace repo + + + handle cfg u setting value + | setting == "trust" = case readTrustLevel value of + Nothing -> badval "trust value" value + Just t -> + let m = M.insert u t (cfgTrustMap cfg) + in Right $ cfg { cfgTrustMap = m } + | setting == "group" = + let m = M.insert u (S.fromList $ words value) (cfgGroupMap cfg) + in Right $ cfg { cfgGroupMap = m } + | otherwise = badval "setting" setting + + name2uuid = M.fromList $ map swap $ + M.toList $ cfgDescriptions curcfg + + showerr (Just msg, l) = [parseerr ++ msg, l] + showerr (Nothing, l) + -- filter out the header and parse error lines + -- from any previous parse failure + | any (`isPrefixOf` l) (parseerr:badheader) = [] + | otherwise = [l] + + badval desc val = Left $ "unknown " ++ desc ++ " \"" ++ val ++ "\"" + badheader = + [ com "There was a problem parsing your input." + , com "Search for \"Parse error\" to find the bad lines." + , com "Either fix the bad lines, or delete them (to discard your changes)." + ] + parseerr = com "Parse error in next line: " + +com :: String -> String +com s = "# " ++ s |