module Text.Grampa.PEG.Backtrack (Parser(..), Result(..), alt) where
import Control.Applicative (Applicative(..), Alternative(..), liftA2)
import Control.Monad (Monad(..), MonadPlus(..))
import Data.Char (isSpace)
import Data.Functor.Classes (Show1(..))
import Data.Functor.Compose (Compose(..))
import Data.List (nub)
import Data.Monoid (Monoid(mappend, mempty), (<>))
import Data.Monoid.Factorial(FactorialMonoid)
import Data.String (fromString)
import qualified Data.Monoid.Cancellative as Cancellative
import qualified Data.Monoid.Factorial as Factorial
import qualified Data.Monoid.Null as Null
import qualified Data.Monoid.Textual as Textual
import qualified Rank2
import qualified Text.Parser.Char
import Text.Parser.Char (CharParsing)
import Text.Parser.Combinators (Parsing(..))
import Text.Parser.LookAhead (LookAheadParsing(..))
import Text.Parser.Token (TokenParsing(someSpace))
import Text.Grampa.Class (MonoidParsing(..), MultiParsing(..), ParseResults, ParseFailure(..))
import Text.Grampa.Internal (FailureInfo(..))
data Result (g :: (* -> *) -> *) s v = Parsed{parsedPrefix :: !v,
parsedSuffix :: !s}
| NoParse FailureInfo
newtype Parser g s r = Parser{applyParser :: s -> Result g s r}
instance Show1 (Result g s) where
liftShowsPrec showsPrecSub _showList prec Parsed{parsedPrefix= r} rest = "Parsed " ++ showsPrecSub prec r rest
liftShowsPrec _showsPrec _showList _prec (NoParse f) rest = "NoParse " ++ shows f rest
instance Functor (Result g s) where
fmap f (Parsed a rest) = Parsed (f a) rest
fmap _ (NoParse failure) = NoParse failure
instance Functor (Parser g s) where
fmap f (Parser p) = Parser (fmap f . p)
instance Applicative (Parser g s) where
pure a = Parser (Parsed a)
Parser p <*> Parser q = Parser r where
r rest = case p rest
of Parsed f rest' -> f <$> q rest'
NoParse failure -> NoParse failure
instance Factorial.FactorialMonoid s => Alternative (Parser g s) where
empty = Parser (\rest-> NoParse $ FailureInfo 0 (fromIntegral $ Factorial.length rest) ["empty"])
(<|>) = alt
alt :: Parser g s a -> Parser g s a -> Parser g s a
Parser p `alt` Parser q = Parser r where
r rest = case p rest
of x@Parsed{} -> x
NoParse{} -> q rest
instance Monad (Parser g s) where
return = pure
Parser p >>= f = Parser r where
r rest = case p rest
of Parsed a rest' -> applyParser (f a) rest'
NoParse failure -> NoParse failure
instance Factorial.FactorialMonoid s => MonadPlus (Parser g s) where
mzero = empty
mplus = (<|>)
instance Monoid x => Monoid (Parser g s x) where
mempty = pure mempty
mappend = liftA2 mappend
instance Factorial.FactorialMonoid s => Parsing (Parser g s) where
try = id
(<?>) = const
eof = endOfInput
unexpected msg = Parser (\t-> NoParse $ FailureInfo 0 (fromIntegral $ Factorial.length t) [msg])
notFollowedBy (Parser p) = Parser (\input-> rewind input (p input))
where rewind t Parsed{} = NoParse (FailureInfo 1 (fromIntegral $ Factorial.length t) ["notFollowedBy"])
rewind t NoParse{} = Parsed () t
instance Factorial.FactorialMonoid s => LookAheadParsing (Parser g s) where
lookAhead (Parser p) = Parser (\input-> rewind input (p input))
where rewind t (Parsed r _) = Parsed r t
rewind _ r@NoParse{} = r
instance (Show s, Textual.TextualMonoid s) => CharParsing (Parser g s) where
satisfy = satisfyChar
string s = Textual.toString (error "unexpected non-character") <$> string (fromString s)
char = satisfyChar . (==)
notChar = satisfyChar . (/=)
anyChar = satisfyChar (const True)
text t = (fromString . Textual.toString (error "unexpected non-character")) <$> string (Textual.fromText t)
instance (Show s, Textual.TextualMonoid s) => TokenParsing (Parser g s) where
someSpace = () <$ takeCharsWhile1 isSpace
instance MonoidParsing (Parser g) where
endOfInput = Parser p
where p rest
| Null.null rest = Parsed () rest
| otherwise = NoParse (FailureInfo 1 (fromIntegral $ Factorial.length rest) ["endOfInput"])
getInput = Parser p
where p rest = Parsed rest mempty
anyToken = Parser p
where p rest = case Factorial.splitPrimePrefix rest
of Just (first, suffix) -> Parsed first suffix
_ -> NoParse (FailureInfo 1 (fromIntegral $ Factorial.length rest) ["anyToken"])
satisfy predicate = Parser p
where p rest =
case Factorial.splitPrimePrefix rest
of Just (first, suffix) | predicate first -> Parsed first suffix
_ -> NoParse (FailureInfo 1 (fromIntegral $ Factorial.length rest) ["satisfy"])
satisfyChar predicate = Parser p
where p rest =
case Textual.splitCharacterPrefix rest
of Just (first, suffix) | predicate first -> Parsed first suffix
_ -> NoParse (FailureInfo 1 (fromIntegral $ Factorial.length rest) ["satisfyChar"])
satisfyCharInput predicate = Parser p
where p rest =
case Textual.splitCharacterPrefix rest
of Just (first, suffix) | predicate first -> Parsed (Factorial.primePrefix rest) suffix
_ -> NoParse (FailureInfo 1 (fromIntegral $ Factorial.length rest) ["satisfyChar"])
notSatisfy predicate = Parser p
where p s = case Factorial.splitPrimePrefix s
of Just (first, _)
| predicate first -> NoParse (FailureInfo 1 (fromIntegral $ Factorial.length s) ["notSatisfy"])
_ -> Parsed () s
notSatisfyChar predicate = Parser p
where p s = case Textual.characterPrefix s
of Just first | predicate first
-> NoParse (FailureInfo 1 (fromIntegral $ Factorial.length s) ["notSatisfyChar"])
_ -> Parsed () s
scan s0 f = Parser (p s0)
where p s rest = Parsed prefix suffix
where (prefix, suffix, _) = Factorial.spanMaybe' s f rest
scanChars s0 f = Parser (p s0)
where p s rest = Parsed prefix suffix
where (prefix, suffix, _) = Textual.spanMaybe_' s f rest
takeWhile predicate = Parser p
where p rest | (prefix, suffix) <- Factorial.span predicate rest = Parsed prefix suffix
takeWhile1 predicate = Parser p
where p rest | (prefix, suffix) <- Factorial.span predicate rest =
if Null.null prefix
then NoParse (FailureInfo 1 (fromIntegral $ Factorial.length rest) ["takeWhile1"])
else Parsed prefix suffix
takeCharsWhile predicate = Parser p
where p rest | (prefix, suffix) <- Textual.span_ False predicate rest = Parsed prefix suffix
takeCharsWhile1 predicate = Parser p
where p rest | (prefix, suffix) <- Textual.span_ False predicate rest =
if Null.null prefix
then NoParse (FailureInfo 1 (fromIntegral $ Factorial.length rest) ["takeCharsWhile1"])
else Parsed prefix suffix
string s = Parser p where
p s' | Just suffix <- Cancellative.stripPrefix s s' = Parsed s suffix
| otherwise = NoParse (FailureInfo 1 (fromIntegral $ Factorial.length s') ["string " ++ show s])
whiteSpace = () <$ takeCharsWhile isSpace
concatMany (Parser p) = Parser q
where q rest = case p rest
of Parsed prefix suffix -> let Parsed prefix' suffix' = q suffix
in Parsed (prefix <> prefix') suffix'
NoParse{} -> Parsed mempty rest
instance MultiParsing Parser where
type ResultFunctor Parser = ParseResults
parsePrefix g input = Rank2.fmap (Compose . fromResult input . (`applyParser` input)) g
parseComplete g input = Rank2.fmap ((snd <$>) . fromResult input . (`applyParser` input))
(Rank2.fmap (<* endOfInput) g)
fromResult :: FactorialMonoid s => s -> Result g s r -> ParseResults (s, r)
fromResult s (NoParse (FailureInfo _ pos msgs)) =
Left (ParseFailure (Factorial.length s fromIntegral pos + 1) (nub msgs))
fromResult _ (Parsed prefix suffix) = Right (suffix, prefix)