{-# LANGUAGE BangPatterns, Haskell2010 #-} -- | -- Module : Data.Picoparsec.Internal -- Copyright : Bryan O'Sullivan 2012, Mario Blažević <blamario@yahoo.com> 2014 -- License : BSD3 -- -- Maintainer : Mario Blažević -- Stability : experimental -- Portability : unknown -- -- Simple, efficient parser combinators, loosely based on the Parsec -- library. module Data.Picoparsec.Internal ( compareResults , get , put , prompt , demandInput , wantInput ) where import Prelude hiding (null) import Data.Picoparsec.Internal.Types import Data.ByteString (ByteString) import Data.Monoid ((<>)) import Data.Monoid.Null (MonoidNull(null)) import Data.Text (Text) -- | Compare two 'IResult' values for equality. -- -- If both 'IResult's are 'Partial', the result will be 'Nothing', as -- they are incomplete and hence their equality cannot be known. -- (This is why there is no 'Eq' instance for 'IResult'.) compareResults :: (Eq t, Eq r) => IResult t r -> IResult t r -> Maybe Bool compareResults (Fail i0 ctxs0 msg0) (Fail i1 ctxs1 msg1) = Just (i0 == i1 && ctxs0 == ctxs1 && msg0 == msg1) compareResults (Done i0 r0) (Done i1 r1) = Just (i0 == i1 && r0 == r1) compareResults (Partial _) (Partial _) = Nothing compareResults _ _ = Just False get :: Parser t t get = Parser $ \i0 a0 m0 _kf ks -> ks i0 a0 m0 (unI i0) {-# INLINE get #-} put :: t -> Parser t () put c = Parser $ \_i0 a0 m0 _kf ks -> ks (I c) a0 m0 () {-# INLINE put #-} -- | Ask for input. If we receive any, pass it to a success -- continuation, otherwise to a failure continuation. prompt :: MonoidNull t => Input t -> Added t -> More -> (Input t -> Added t -> More -> IResult t r) -> (Input t -> Added t -> More -> IResult t r) -> IResult t r prompt i0 a0 _m0 kf ks = Partial $ \s -> if null s then kf i0 a0 Complete else ks (i0 <> I s) (a0 <> A s) Incomplete {-# SPECIALIZE prompt :: Input ByteString -> Added ByteString -> More -> (Input ByteString -> Added ByteString -> More -> IResult ByteString r) -> (Input ByteString -> Added ByteString -> More -> IResult ByteString r) -> IResult ByteString r #-} {-# SPECIALIZE prompt :: Input Text -> Added Text -> More -> (Input Text -> Added Text -> More -> IResult Text r) -> (Input Text -> Added Text-> More -> IResult Text r) -> IResult Text r #-} -- | Immediately demand more input via a 'Partial' continuation -- result. demandInput :: MonoidNull t => Parser t () demandInput = Parser $ \i0 a0 m0 kf ks -> if m0 == Complete then kf i0 a0 m0 ["demandInput"] "not enough input" else let kf' i a m = kf i a m ["demandInput"] "not enough input" ks' i a m = ks i a m () in prompt i0 a0 m0 kf' ks' {-# SPECIALIZE demandInput :: Parser ByteString () #-} {-# SPECIALIZE demandInput :: Parser Text () #-} -- | This parser always succeeds. It returns 'True' if any input is -- available either immediately or on demand, and 'False' if the end -- of all input has been reached. wantInput :: MonoidNull t => Parser t Bool wantInput = Parser $ \i0 a0 m0 _kf ks -> case () of _ | not (null (unI i0)) -> ks i0 a0 m0 True | m0 == Complete -> ks i0 a0 m0 False | otherwise -> let kf' i a m = ks i a m False ks' i a m = ks i a m True in prompt i0 a0 m0 kf' ks' {-# SPECIALIZE wantInput :: Parser ByteString Bool #-} {-# SPECIALIZE wantInput :: Parser Text Bool #-}