module Distribution.ReadE (
ReadE(..), succeedReadE, failReadE,
parseReadE, readEOrFail,
readP_to_E
) where
import Distribution.Compat.ReadP
import Data.Char ( isSpace )
newtype ReadE a = ReadE {runReadE :: String -> Either ErrorMsg a}
type ErrorMsg = String
instance Functor ReadE where
fmap f (ReadE p) = ReadE $ \txt -> case p txt of
Right a -> Right (f a)
Left err -> Left err
succeedReadE :: (String -> a) -> ReadE a
succeedReadE f = ReadE (Right . f)
failReadE :: ErrorMsg -> ReadE a
failReadE = ReadE . const . Left
parseReadE :: ReadE a -> ReadP r a
parseReadE (ReadE p) = do
txt <- look
either fail return (p txt)
readEOrFail :: ReadE a -> String -> a
readEOrFail r = either error id . runReadE r
readP_to_E :: (String -> ErrorMsg) -> ReadP a a -> ReadE a
readP_to_E err r =
ReadE $ \txt -> case [ p | (p, s) <- readP_to_S r txt
, all isSpace s ]
of [] -> Left (err txt)
(p:_) -> Right p