{-# LANGUAGE FlexibleContexts, FlexibleInstances, GADTs, GeneralizedNewtypeDeriving, InstanceSigs, RankNTypes, ScopedTypeVariables, StandaloneDeriving, TypeApplications, TypeFamilies, TypeOperators, UndecidableInstances #-} {-# OPTIONS -fno-full-laziness #-} module Text.Grampa.ContextFree.LeftRecursive (Fixed, Parser, SeparatedParser(..), FallibleWithExpectations(..), longest, peg, terminalPEG, liftPositive, liftPure, mapPrimitive, parseSeparated, separated) where import Control.Applicative import Control.Monad (Monad(..), MonadPlus(..), void) import Control.Monad.Trans.State.Lazy (State, evalState, get, put) import Data.Functor.Compose (Compose(..)) import Data.Maybe (isJust) import Data.Semigroup (Semigroup(..)) import Data.Monoid (Monoid(mempty), All(..), Any(..)) import Data.Monoid.Null (MonoidNull(null)) import Data.Monoid.Factorial (FactorialMonoid) import Data.Monoid.Textual (TextualMonoid) import Data.Semigroup.Cancellative (LeftReductive) import qualified Data.Monoid.Factorial as Factorial import qualified Data.Monoid.Textual as Textual import Data.String (fromString) import Data.Type.Equality ((:~:)(Refl)) import qualified Text.Parser.Char as Char import Text.Parser.Char (CharParsing) import Text.Parser.Combinators (Parsing(..)) import Text.Parser.LookAhead (LookAheadParsing(..)) import qualified Rank2 import Text.Grampa.Class (GrammarParsing(..), InputParsing(..), InputCharParsing(..), MultiParsing(..), AmbiguousParsing(..), Ambiguous(..), ConsumedInputParsing(..), DeterministicParsing(..), TailsParsing(parseTails, parseAllTails), Expected(..)) import Text.Grampa.Internal (ResultList(..), FailureInfo(..), AmbiguousAlternative(ambiguousOr)) import qualified Text.Grampa.ContextFree.SortedMemoizing as Memoizing import qualified Text.Grampa.PEG.Backtrack.Measured as Backtrack import Prelude hiding (cycle, null, span, take, takeWhile) type Parser = Fixed Memoizing.Parser type ResultAppend p (g :: (* -> *) -> *) s = GrammarFunctor (p g s) Rank2.~> GrammarFunctor (p g s) Rank2.~> GrammarFunctor (p g s) data Fixed p g s a = Parser { complete, direct, direct0, direct1, indirect :: p g s a, isAmbiguous :: Maybe (AmbiguityWitness a), cyclicDescendants :: Rank2.Apply g => g (Const (ParserFlags g)) -> ParserFlags g} | DirectParser { complete, direct0, direct1 :: p g s a} | PositiveDirectParser { complete :: p g s a} data AmbiguityWitness a where AmbiguityWitness :: (a :~: Ambiguous b) -> AmbiguityWitness a data SeparatedParser p (g :: (* -> *) -> *) s a = FrontParser (p g s a) | CycleParser { cycleParser :: p g s a, backParser :: p g s a, appendResultsArrow :: ResultAppend p g s a, dependencies :: g (Const Bool)} | BackParser { backParser :: p g s a} data ParserFlags g = ParserFlags { nullable :: Bool, dependsOn :: g (Const Bool)} deriving instance Show (g (Const Bool)) => Show (ParserFlags g) data ParserFunctor p g s a = ParserResultsFunctor {parserResults :: GrammarFunctor (p g s) a} | ParserFlagsFunctor {parserFlags :: ParserFlags g} newtype Union (g :: (* -> *) -> *) = Union{getUnion :: g (Const Bool)} --instance Rank2.Applicative g => Monoid (Union g) where -- mempty = Union (Rank2.pure $ Const False) instance (Rank2.Apply g, Rank2.Distributive g) => Semigroup (Union g) where Union g1 <> Union g2 = Union (Rank2.liftA2 union g1 g2) instance (Rank2.Apply g, Rank2.Distributive g) => Monoid (Union g) where mempty = Union (Rank2.cotraverse (Const . getConst) (Const False)) mappend (Union g1) (Union g2) = Union (Rank2.liftA2 union g1 g2) mapPrimitive :: (p g s a -> p g s a) -> Fixed p g s a -> Fixed p g s a mapPrimitive f p@PositiveDirectParser{} = PositiveDirectParser{complete= f (complete p)} mapPrimitive f p@DirectParser{} = DirectParser{complete= f (complete p), direct0= f (direct0 p), direct1= f (direct1 p)} mapPrimitive f p@Parser{} = Parser{complete= f (complete p), isAmbiguous= isAmbiguous p, cyclicDescendants= cyclicDescendants p, indirect= f (indirect p), direct= f (direct p), direct0= f (direct0 p), direct1= f (direct1 p)} general, general' :: Alternative (p g s) => Fixed p g s a -> Fixed p g s a general p = Parser{ complete= complete p, direct = direct p', direct0= direct0 p', direct1= direct1 p', indirect= indirect p', isAmbiguous= case p of Parser{isAmbiguous= a} -> a _ -> Nothing, cyclicDescendants= cyclicDescendants p'} where p' = general' p general' p@PositiveDirectParser{} = Parser{ complete= complete p, direct = complete p, direct0= empty, direct1= complete p, indirect= empty, isAmbiguous= Nothing, cyclicDescendants= \cd-> ParserFlags False (const (Const False) Rank2.<$> cd)} general' p@DirectParser{} = Parser{ complete= complete p, direct = complete p, direct0= direct0 p, direct1= direct1 p, indirect= empty, isAmbiguous= Nothing, cyclicDescendants= \cd-> ParserFlags True (const (Const False) Rank2.<$> cd)} general' p@Parser{} = p -- | Parser of general context-free grammars, including left recursion. -- -- @ -- 'parseComplete' :: ("Rank2".'Rank2.Apply' g, "Rank2".'Rank2.Traversable' g, 'FactorialMonoid' s) => -- g (LeftRecursive.'Fixed g s) -> s -> g ('Compose' ('ParseResults' s) []) -- @ instance (Eq s, LeftReductive s, FactorialMonoid s, Alternative (p g s), TailsParsing (p g s), GrammarConstraint (p g s) g, ParserGrammar (p g s) ~ g, Functor (ResultFunctor (p g s)), s ~ ParserInput (p g s), GrammarFunctor (p g s) ~ rl s, FallibleWithExpectations rl, AmbiguousAlternative (GrammarFunctor (p g s))) => MultiParsing (Fixed p g s) where type GrammarConstraint (Fixed p g s) g' = (GrammarConstraint (p g s) g', g ~ g', Rank2.Apply g, Rank2.Distributive g, Rank2.Traversable g) type ResultFunctor (Fixed p g s) = ResultFunctor (p g s) parsePrefix :: (Rank2.Apply g, Rank2.Distributive g, Rank2.Traversable g, Eq s, FactorialMonoid s) => g (Fixed p g s) -> s -> g (Compose (ResultFunctor (p g s)) ((,) s)) parsePrefix g input = Rank2.fmap (Compose . parsingResult @(p g s) input) (snd $ head $ parseRecursive g input) {-# INLINE parsePrefix #-} parseComplete :: (Rank2.Apply g, Rank2.Distributive g, Rank2.Traversable g, Eq s, FactorialMonoid s) => g (Fixed p g s) -> s -> g (ResultFunctor (p g s)) parseComplete g = \input-> let close :: g (p g s) close = Rank2.fmap (<* eof) selfReferring in Rank2.fmap ((snd <$>) . parsingResult @(p g s) input) (snd $ head $ parseAllTails close $ parseSeparated g' input) where g' = separated g {-# INLINE parseComplete #-} instance (Eq s, LeftReductive s, FactorialMonoid s, Alternative (p g s), TailsParsing (p g s), GrammarConstraint (p g s) g, ParserGrammar (p g s) ~ g, Functor (ResultFunctor (p g s)), s ~ ParserInput (p g s), GrammarFunctor (p g s) ~ rl s, FallibleWithExpectations rl, AmbiguousAlternative (GrammarFunctor (p g s))) => GrammarParsing (Fixed p g s) where type ParserGrammar (Fixed p g s) = g type GrammarFunctor (Fixed p g s) = ParserFunctor p g s parsingResult :: s -> ParserFunctor p g s a -> ResultFunctor (p g s) (s, a) parsingResult s = parsingResult @(p g s) s . parserResults nonTerminal :: (Rank2.Apply g, Rank2.Distributive g, Rank2.Traversable g) => (g (ParserFunctor p g s) -> ParserFunctor p g s a) -> Fixed p g s a nonTerminal f = Parser{ complete= ind, direct= empty, direct0= empty, direct1= empty, indirect= ind, isAmbiguous= Nothing, cyclicDescendants= parserFlags . f . Rank2.fmap (ParserFlagsFunctor . getConst) . addSelf} where ind = nonTerminal (parserResults . f . Rank2.fmap ParserResultsFunctor) addSelf g = Rank2.liftA2 adjust bits g adjust :: forall b. Const (g (Const Bool)) b -> Const (ParserFlags g) b -> Const (ParserFlags g) b adjust (Const bit) (Const (ParserFlags n d)) = Const ParserFlags{ nullable= n, dependsOn= Rank2.liftA2 union bit d} {-# INLINE nonTerminal #-} recursive = general bits :: forall (g :: (* -> *) -> *). (Rank2.Distributive g, Rank2.Traversable g) => g (Const (g (Const Bool))) bits = start `seq` Rank2.fmap oneBit start where start = evalState (Rank2.traverse next (Rank2.distributeJoin Nothing)) 0 oneBit :: Const Int a -> Const (g (Const Bool)) a next :: f a -> State Int (Const Int a) oneBit (Const i) = Const (Rank2.fmap (Const . (i ==) . getConst) start) next _ = do {i <- get; let {i' = succ i}; seq i' (put i'); return (Const i)} instance Functor (p g s) => Functor (Fixed p g s) where fmap f (PositiveDirectParser p) = PositiveDirectParser (fmap f p) fmap f p@DirectParser{} = DirectParser{ complete= fmap f (complete p), direct0= fmap f (direct0 p), direct1= fmap f (direct1 p)} fmap f p@Parser{} = p{ complete= fmap f (complete p), direct= fmap f (direct p), direct0= fmap f (direct0 p), direct1= fmap f (direct1 p), indirect= fmap f (indirect p), isAmbiguous= Nothing} {-# INLINABLE fmap #-} instance Alternative (p g s) => Applicative (Fixed p g s) where pure a = DirectParser{complete= pure a, direct0= pure a, direct1= empty} p@PositiveDirectParser{} <*> q = PositiveDirectParser{ complete= complete p <*> complete q} p@DirectParser{} <*> q@PositiveDirectParser{} = PositiveDirectParser{ complete= complete p <*> complete q} p@DirectParser{} <*> q@DirectParser{} = DirectParser{ complete= complete p <*> complete q, direct0= direct0 p <*> direct0 q, direct1= direct0 p <*> direct1 q <|> direct1 p <*> complete q} p <*> q@Parser{} = Parser{ complete= complete p' <*> complete q, direct= direct0 p' <*> direct q <|> direct1 p' <*> complete q, direct0= direct0 p' <*> direct0 q, direct1= direct0 p' <*> direct1 q <|> direct1 p' <*> complete q, indirect= direct0 p' <*> indirect q <|> indirect p' <*> complete q, isAmbiguous= Nothing, cyclicDescendants= \deps-> let pcd@(ParserFlags pn pd) = cyclicDescendants p' deps ParserFlags qn qd = cyclicDescendants q deps in if pn then ParserFlags qn (Rank2.liftA2 union pd qd) else pcd} where p'@Parser{} = general' p p <*> q = Parser{ complete= complete p' <*> complete q', direct= direct p' <*> complete q', direct0= direct0 p' <*> direct0 q', direct1= direct0 p' <*> direct1 q' <|> direct1 p' <*> complete q', indirect= indirect p' <*> complete q', isAmbiguous= Nothing, cyclicDescendants= \deps-> let pcd@(ParserFlags pn pd) = cyclicDescendants p' deps ParserFlags qn qd = cyclicDescendants q' deps in if pn then ParserFlags qn (Rank2.liftA2 union pd qd) else pcd} where p'@Parser{} = general' p q'@Parser{} = general' q {-# INLINABLE pure #-} {-# INLINABLE (<*>) #-} instance Alternative (p g s) => Alternative (Fixed p g s) where empty = PositiveDirectParser{complete= empty} p@PositiveDirectParser{} <|> q@PositiveDirectParser{} = PositiveDirectParser{complete= complete p <|> complete q} p@PositiveDirectParser{} <|> q@DirectParser{} = DirectParser{ complete= complete p <|> complete q, direct0 = direct0 q, direct1= complete p <|> direct1 q} p@DirectParser{} <|> q@PositiveDirectParser{} = DirectParser{ complete= complete p <|> complete q, direct0 = direct0 p, direct1= direct1 p <|> complete q} p@DirectParser{} <|> q@DirectParser{} = DirectParser{ complete= complete p <|> complete q, direct0 = direct0 p <|> direct0 q, direct1= direct1 p <|> direct1 q} p <|> q = Parser{complete= complete p' <|> complete q', direct= direct p' <|> direct q', direct0= direct0 p' <|> direct0 q', direct1= direct1 p' <|> direct1 q', indirect= indirect p' <|> indirect q', isAmbiguous= Nothing, cyclicDescendants= \deps-> let ParserFlags pn pd = cyclicDescendants p' deps ParserFlags qn qd = cyclicDescendants q' deps in ParserFlags (pn || qn) (Rank2.liftA2 union pd qd)} where p'@Parser{} = general p q'@Parser{} = general q many (PositiveDirectParser p) = DirectParser{ complete= many p, direct0= pure [], direct1= some p} many p@DirectParser{} = DirectParser{ complete= many (complete p), direct0= pure [] <|> (:[]) <$> direct0 p, direct1= (:) <$> direct1 p <*> many (complete p)} many p@Parser{} = Parser{ complete= mcp, direct= d0 <|> d1, direct0= d0, direct1= d1, indirect= (:) <$> indirect p <*> mcp, isAmbiguous= Nothing, cyclicDescendants= \deps-> (cyclicDescendants p deps){nullable= True}} where d0 = pure [] <|> (:[]) <$> direct0 p d1 = (:) <$> direct1 p <*> mcp mcp = many (complete p) some (PositiveDirectParser p) = PositiveDirectParser{complete= some p} some p@DirectParser{} = DirectParser{ complete= some (complete p), direct0= (:[]) <$> direct0 p, direct1= (:) <$> direct1 p <*> many (complete p)} some p@Parser{} = Parser{ complete= some (complete p), direct= d0 <|> d1, direct0= d0, direct1= d1, indirect= (:) <$> indirect p <*> many (complete p), isAmbiguous= Nothing, cyclicDescendants= cyclicDescendants p} where d0 = (:[]) <$> direct0 p d1= (:) <$> direct1 p <*> many (complete p) {-# INLINABLE (<|>) #-} {-# INLINABLE many #-} {-# INLINABLE some #-} union :: Const Bool x -> Const Bool x -> Const Bool x union (Const False) d = d union (Const True) _ = Const True instance (Alternative (p g s), Monad (p g s)) => Monad (Fixed p g s) where return = pure (>>) = (*>) PositiveDirectParser p >>= cont = PositiveDirectParser (p >>= complete . cont) p@DirectParser{} >>= cont = Parser{ complete= complete p >>= complete . cont, direct= d0 <|> d1, direct0= d0, direct1= d1, indirect= direct0 p >>= indirect . general' . cont, isAmbiguous= Nothing, cyclicDescendants= \cd-> (ParserFlags True $ Rank2.fmap (const $ Const True) cd)} where d0 = direct0 p >>= direct0 . general' . cont d1 = (direct0 p >>= direct1 . general' . cont) <|> (direct1 p >>= complete . cont) p >>= cont = Parser{ complete= complete p >>= complete . cont, direct= d0 <|> d1, direct0= d0, direct1= d1, indirect= (indirect p >>= complete . cont) <|> (direct0 p >>= indirect . general' . cont), isAmbiguous= Nothing, cyclicDescendants= \cd-> let pcd@(ParserFlags pn _) = cyclicDescendants p' cd in if pn then ParserFlags True (Rank2.fmap (const $ Const True) cd) else pcd} where d0 = direct0 p >>= direct0 . general' . cont d1 = (direct0 p >>= direct1 . general' . cont) <|> (direct1 p >>= complete . cont) p'@Parser{} = general' p instance MonadPlus (p g s) => MonadPlus (Fixed p g s) where mzero = empty mplus = (<|>) instance (Alternative (p g s), Semigroup x) => Semigroup (Fixed p g s x) where (<>) = liftA2 (<>) instance (Alternative (p g s), Monoid x) => Monoid (Fixed p g s x) where mempty = pure mempty mappend = liftA2 mappend primitive :: p g s a -> p g s a -> p g s a -> Fixed p g s a primitive d0 d1 d = DirectParser{complete= d, direct0= d0, direct1= d1} {-# INLINE primitive #-} -- | Lifts a primitive positive parser (/i.e./, one that always consumes some input) into a left-recursive one liftPositive :: p g s a -> Fixed p g s a liftPositive p = PositiveDirectParser{complete= p} {-# INLINE liftPositive #-} -- | Lifts a primitive pure parser (/i.e./, one that consumes no input) into a left-recursive one liftPure :: Alternative (p g s) => p g s a -> Fixed p g s a liftPure p = DirectParser{complete= p, direct0= p, direct1= empty} {-# INLINE liftPure #-} instance (Parsing (p g s), InputParsing (Fixed p g s)) => Parsing (Fixed p g s) where eof = primitive eof empty eof try (PositiveDirectParser p) = PositiveDirectParser (try p) try p@DirectParser{} = DirectParser{ complete= try (complete p), direct0= try (direct0 p), direct1= try (direct1 p)} try p@Parser{} = p{ complete= try (complete p), direct= try (direct p), direct0= try (direct0 p), direct1= try (direct1 p), indirect= try (indirect p)} PositiveDirectParser p msg = PositiveDirectParser (p msg) p@DirectParser{} msg = DirectParser{ complete= complete p msg, direct0= direct0 p msg, direct1= direct1 p msg} p@Parser{} msg = p{ complete= complete p msg, direct= direct p msg, direct0= direct0 p msg, direct1= direct1 p msg, indirect= indirect p msg} notFollowedBy p@PositiveDirectParser{} = DirectParser{ complete= notFollowedBy (complete p), direct0= notFollowedBy (complete p), direct1= empty} notFollowedBy p@DirectParser{} = DirectParser{ complete= notFollowedBy (complete p), direct0= notFollowedBy (complete p), direct1= empty} notFollowedBy p@Parser{} = Parser{ complete= notFollowedBy (complete p), direct= notFollowedBy (direct p), direct0= notFollowedBy (direct p), direct1= empty, isAmbiguous= Nothing, indirect= notFollowedBy (indirect p), cyclicDescendants= \deps-> (cyclicDescendants p deps){nullable= True}} unexpected msg = liftPositive (unexpected msg) instance (InputParsing (Fixed p g s), DeterministicParsing (p g s)) => DeterministicParsing (Fixed p g s) where p@DirectParser{} <<|> q@PositiveDirectParser{} = DirectParser{ complete= complete p <<|> complete q, direct0 = direct0 p, direct1= direct1 p <<|> complete q} p@DirectParser{} <<|> q@DirectParser{} = DirectParser{ complete= complete p <<|> complete q, direct0 = direct0 p <<|> direct0 q, direct1= direct1 p <<|> direct1 q} p <<|> q = Parser{complete= complete p' <<|> complete q', direct= direct p' <<|> direct q', direct0= direct0 p' <<|> direct0 q', direct1= direct1 p' <<|> direct1 q', indirect= indirect p' <<|> indirect q', isAmbiguous= Nothing, cyclicDescendants= \deps-> let ParserFlags pn pd = cyclicDescendants p' deps ParserFlags qn qd = cyclicDescendants q' deps in ParserFlags (pn || qn) (Rank2.liftA2 union pd qd)} where p'@Parser{} = general p q'@Parser{} = general q takeSome p = (:) <$> p <*> takeMany p takeMany (PositiveDirectParser p) = DirectParser{ complete = takeMany p, direct0= [] <$ notFollowedBy (void p), direct1= takeSome p} takeMany p@DirectParser{} = DirectParser{ complete = takeMany (complete p), direct0= (:[]) <$> direct0 p <<|> [] <$ notFollowedBy (void $ complete p), direct1= (:) <$> direct1 p <*> takeMany (complete p)} takeMany p@Parser{} = Parser{ complete= mcp, direct= d1 <<|> d0, direct0= d0, direct1= d1, indirect= (:) <$> indirect p <*> mcp, isAmbiguous= Nothing, cyclicDescendants= \deps-> (cyclicDescendants p deps){nullable= True}} where d0 = (:[]) <$> direct0 p <<|> [] <$ notFollowedBy (void $ direct p) d1 = (:) <$> direct1 p <*> mcp mcp = takeMany (complete p) skipAll (PositiveDirectParser p) = DirectParser{ complete = skipAll p, direct0= () <$ notFollowedBy (void p), direct1= p *> skipAll p} skipAll p@DirectParser{} = DirectParser{ complete = skipAll (complete p), direct0= void (direct0 p) <<|> notFollowedBy (void $ complete p), direct1= direct1 p *> skipAll (complete p)} skipAll p@Parser{} = Parser{ complete= mcp, direct= d1 <<|> d0, direct0= d0, direct1= d1, indirect= indirect p *> mcp, isAmbiguous= Nothing, cyclicDescendants= \deps-> (cyclicDescendants p deps){nullable= True}} where d0 = () <$ direct0 p <<|> notFollowedBy (void $ direct p) d1 = direct1 p *> mcp mcp = skipAll (complete p) instance (LookAheadParsing (p g s), InputParsing (Fixed p g s)) => LookAheadParsing (Fixed p g s) where lookAhead p@PositiveDirectParser{} = DirectParser{ complete= lookAhead (complete p), direct0= lookAhead (complete p), direct1= empty} lookAhead p@DirectParser{} = DirectParser{ complete= lookAhead (complete p), direct0= lookAhead (complete p), direct1= empty} lookAhead p@Parser{} = Parser{ complete= lookAhead (complete p), direct= lookAhead (direct p), direct0= lookAhead (direct p), direct1= empty, isAmbiguous= isAmbiguous p, indirect= lookAhead (indirect p), cyclicDescendants= \deps-> (cyclicDescendants p deps){nullable= True}} instance (LeftReductive s, FactorialMonoid s, InputParsing (p g s), ParserInput (p g s) ~ s) => InputParsing (Fixed p g s) where type ParserInput (Fixed p g s) = s getInput = primitive getInput empty getInput anyToken = liftPositive anyToken satisfy predicate = liftPositive (satisfy predicate) notSatisfy predicate = primitive (notSatisfy predicate) empty (notSatisfy predicate) scan s0 f = primitive (mempty <$ notSatisfy test) (lookAhead (satisfy test) *> p) p where p = scan s0 f test = isJust . f s0 string s | null s = primitive (string s) empty (string s) | otherwise = liftPositive (string s) take 0 = mempty take n = liftPositive (take n) takeWhile predicate = primitive (mempty <$ notSatisfy predicate) (takeWhile1 predicate) (takeWhile predicate) takeWhile1 predicate = liftPositive (takeWhile1 predicate) {-# INLINABLE string #-} instance (LeftReductive s, FactorialMonoid s, ConsumedInputParsing (p g s), ParserInput (p g s) ~ s) => ConsumedInputParsing (Fixed p g s) where match (PositiveDirectParser p) = PositiveDirectParser (match p) match p@DirectParser{} = DirectParser{ complete= match (complete p), direct0 = match (direct0 p), direct1 = match (direct1 p)} match p@Parser{} = Parser{ complete= match (complete p), direct = match (direct p), direct0 = match (direct0 p), direct1 = match (direct1 p), indirect= match (indirect p), isAmbiguous= Nothing, cyclicDescendants= cyclicDescendants p} instance (Show s, TextualMonoid s, InputCharParsing (p g s), ParserInput (p g s) ~ s) => InputCharParsing (Fixed p g s) where satisfyCharInput predicate = liftPositive (satisfyCharInput predicate) notSatisfyChar predicate = primitive (notSatisfyChar predicate) empty (notSatisfyChar predicate) scanChars s0 f = primitive (mempty <$ notSatisfyChar test) (lookAhead (Char.satisfy test) *> p) p where p = scanChars s0 f test = isJust . f s0 takeCharsWhile predicate = primitive (mempty <$ notSatisfyChar predicate) (takeCharsWhile1 predicate) (takeCharsWhile predicate) takeCharsWhile1 predicate = liftPositive (takeCharsWhile1 predicate) instance (CharParsing (p g s), InputCharParsing (Fixed p g s), TextualMonoid s, s ~ ParserInput (Fixed p g s), Show s) => CharParsing (Fixed p g s) where satisfy predicate = liftPositive (Char.satisfy predicate) string s = Textual.toString (error "unexpected non-character") <$> string (fromString s) text t = (fromString . Textual.toString (error "unexpected non-character")) <$> string (Textual.fromText t) instance AmbiguousParsing (p g s) => AmbiguousParsing (Fixed p g s) where ambiguous (PositiveDirectParser p) = PositiveDirectParser (ambiguous p) ambiguous p@DirectParser{} = DirectParser{complete= ambiguous (complete p), direct0= ambiguous (direct0 p), direct1= ambiguous (direct1 p)} ambiguous p@Parser{} = Parser{complete= ambiguous (complete p), direct= ambiguous (direct p), direct0= ambiguous (direct0 p), direct1= ambiguous (direct1 p), indirect= ambiguous (indirect p), isAmbiguous= Just (AmbiguityWitness Refl), cyclicDescendants= cyclicDescendants p} {-# INLINABLE ambiguous #-} -- | Turns a context-free parser into a backtracking PEG parser that consumes the longest possible prefix of the list -- of input tails, opposite of 'peg' longest :: Fixed Memoizing.Parser g s a -> Fixed Backtrack.Parser g [(s, g (ResultList g s))] a longest (PositiveDirectParser p) = PositiveDirectParser (Memoizing.longest p) longest p@DirectParser{} = DirectParser{complete= Memoizing.longest (complete p), direct0= Memoizing.longest (direct0 p), direct1= Memoizing.longest (direct1 p)} longest p@Parser{} = Parser{complete= Memoizing.longest (complete p), direct= Memoizing.longest (direct p), direct0= Memoizing.longest (direct0 p), direct1= Memoizing.longest (direct1 p), indirect= Memoizing.longest (indirect p), isAmbiguous= Nothing, cyclicDescendants= cyclicDescendants p} -- | Turns a backtracking PEG parser of the list of input tails into a context-free parser, opposite of 'longest' peg :: Fixed Backtrack.Parser g [(s, g (ResultList g s))] a -> Fixed Memoizing.Parser g s a peg (PositiveDirectParser p) = PositiveDirectParser (Memoizing.peg p) peg p@DirectParser{} = DirectParser{complete= Memoizing.peg (complete p), direct0= Memoizing.peg (direct0 p), direct1= Memoizing.peg (direct1 p)} peg p@Parser{} = Parser{complete= Memoizing.peg (complete p), direct= Memoizing.peg (direct p), direct0= Memoizing.peg (direct0 p), direct1= Memoizing.peg (direct1 p), indirect= Memoizing.peg (indirect p), isAmbiguous= Nothing, cyclicDescendants= cyclicDescendants p} -- | Turns a backtracking PEG parser into a context-free parser terminalPEG :: Monoid s => Fixed Backtrack.Parser g s a -> Fixed Memoizing.Parser g s a terminalPEG (PositiveDirectParser p) = PositiveDirectParser (Memoizing.terminalPEG p) terminalPEG p@DirectParser{} = DirectParser{complete= Memoizing.terminalPEG (complete p), direct0= Memoizing.terminalPEG (direct0 p), direct1= Memoizing.terminalPEG (direct1 p)} terminalPEG p@Parser{} = Parser{complete= Memoizing.terminalPEG (complete p), direct= Memoizing.terminalPEG (direct p), direct0= Memoizing.terminalPEG (direct0 p), direct1= Memoizing.terminalPEG (direct1 p), indirect= Memoizing.terminalPEG (indirect p), isAmbiguous= Nothing, cyclicDescendants= cyclicDescendants p} parseRecursive :: forall p g s rl. (Rank2.Apply g, Rank2.Distributive g, Rank2.Traversable g, Eq s, FactorialMonoid s, LeftReductive s, Alternative (p g s), TailsParsing (p g s), GrammarConstraint (p g s) g, s ~ ParserInput (p g s), GrammarFunctor (p g s) ~ rl s, FallibleWithExpectations rl, AmbiguousAlternative (GrammarFunctor (p g s))) => g (Fixed p g s) -> s -> [(s, g (GrammarFunctor (p g s)))] parseRecursive = parseSeparated . separated {-# INLINE parseRecursive #-} separated :: forall p g s. (Alternative (p g s), Rank2.Apply g, Rank2.Distributive g, Rank2.Traversable g, AmbiguousAlternative (GrammarFunctor (p g s))) => g (Fixed p g s) -> g (SeparatedParser p g s) separated g = Rank2.liftA4 reseparate circulars cycleFollowers descendants g where descendants :: g (Const (g (Const Bool))) cycleFollowers, circulars :: g (Const Bool) cyclicDescendantses :: g (Const (ParserFlags g)) appendResults :: forall a. Maybe (AmbiguityWitness a) -> GrammarFunctor (p g s) a -> GrammarFunctor (p g s) a -> GrammarFunctor (p g s) a leftRecursive :: forall a. Const (g (Const Bool)) a -> Const (ParserFlags g) a -> Const Bool a leftRecursiveDeps :: forall a. Const Bool a -> Const (ParserFlags g) a -> Const (g (Const Bool)) a reseparate :: forall a. Const Bool a -> Const Bool a -> Const (g (Const Bool)) a -> Fixed p g s a -> SeparatedParser p g s a reseparate (Const circular) (Const follower) (Const deps) p | circular || leader && follower = CycleParser (indirect p) (direct p) (Rank2.Arrow (Rank2.Arrow . appendResults (isAmbiguous p))) deps | follower = BackParser (complete p) | otherwise = FrontParser (complete p) where leader = getAny (Rank2.foldMap (Any . getConst) $ Rank2.liftA2 intersection circulars deps) appendResults (Just (AmbiguityWitness Refl)) = ambiguousOr appendResults Nothing = (<|>) descendants = Rank2.fmap (Const . dependsOn . getConst) cyclicDescendantses cyclicDescendantses = fixDescendants (Rank2.fmap (Const . cyclicDescendants . general) g) circulars = Rank2.liftA2 leftRecursive bits cyclicDescendantses cycleFollowers = getUnion (Rank2.foldMap (Union . getConst) $ Rank2.liftA2 leftRecursiveDeps circulars cyclicDescendantses) leftRecursive (Const bit) (Const flags) = Const (getAny $ Rank2.foldMap (Any . getConst) $ Rank2.liftA2 intersection bit $ dependsOn flags) leftRecursiveDeps (Const True) (Const flags) = Const (dependsOn flags) leftRecursiveDeps (Const False) (Const flags) = Const (Rank2.fmap (const $ Const False) (dependsOn flags)) intersection (Const a) (Const b) = Const (a && b) {-# INLINABLE separated #-} fixDescendants :: forall g. (Rank2.Apply g, Rank2.Traversable g) => g (Const (g (Const (ParserFlags g)) -> (ParserFlags g))) -> g (Const (ParserFlags g)) fixDescendants gf = go initial where go :: g (Const (ParserFlags g)) -> g (Const (ParserFlags g)) go cd | getAll (Rank2.foldMap (All . getConst) $ Rank2.liftA2 agree cd cd') = cd | otherwise = go cd' where cd' = Rank2.liftA2 depsUnion cd (Rank2.fmap (\(Const f)-> Const (f cd)) gf) agree (Const (ParserFlags _xn xd)) (Const (ParserFlags _yn yd)) = Const (getAll (Rank2.foldMap (All . getConst) (Rank2.liftA2 agree' xd yd))) agree' (Const x) (Const y) = Const (x == y) depsUnion (Const ParserFlags{dependsOn= old}) (Const (ParserFlags n new)) = Const (ParserFlags n $ Rank2.liftA2 union old new) initial = Rank2.liftA2 (\_ (Const n)-> Const (ParserFlags n (const (Const False) Rank2.<$> gf))) gf nullabilities nullabilities = fixNullabilities gf {-# INLINABLE fixDescendants #-} fixNullabilities :: forall g. (Rank2.Apply g, Rank2.Traversable g) => g (Const (g (Const (ParserFlags g)) -> (ParserFlags g))) -> g (Const Bool) fixNullabilities gf = Rank2.fmap (Const . nullable . getConst) (go initial) where go :: g (Const (ParserFlags g)) -> g (Const (ParserFlags g)) go cd | getAll (Rank2.foldMap (All . getConst) $ Rank2.liftA2 agree cd cd') = cd | otherwise = go cd' where cd' = Rank2.fmap (\(Const f)-> Const (f cd)) gf agree (Const flags1) (Const flags2) = Const (nullable flags1 == nullable flags2) initial = const (Const (ParserFlags True (const (Const False) Rank2.<$> gf))) Rank2.<$> gf {-# INLINABLE fixNullabilities #-} -- | Parse the given input using a context-free grammar separated into two parts: the first specifying all the -- left-recursive productions, the second all others. The first function argument specifies the left-recursive -- dependencies among the grammar productions. parseSeparated :: forall p g rl s. (Rank2.Apply g, Rank2.Foldable g, Eq s, FactorialMonoid s, LeftReductive s, TailsParsing (p g s), GrammarConstraint (p g s) g, GrammarFunctor (p g s) ~ rl s, FallibleWithExpectations rl, s ~ ParserInput (p g s)) => g (SeparatedParser p g s) -> s -> [(s, g (GrammarFunctor (p g s)))] parseSeparated parsers input = foldr parseTail [] (Factorial.tails input) where parseTail s parsedTail = parsed where parsed = (s,d''):parsedTail d = Rank2.fmap (($ (s,d):parsedTail) . parseTails) directs d' = fixRecursive s parsedTail d d'' = Rank2.liftA2 f parsers d' f :: forall a. SeparatedParser p g s a -> GrammarFunctor (p g s) a -> GrammarFunctor (p g s) a f (FrontParser p) _ = parseTails p ((s,d''):parsedTail) f _ result = result fixRecursive :: s -> [(s, g (GrammarFunctor (p g s)))] -> g (GrammarFunctor (p g s)) -> g (GrammarFunctor (p g s)) whileAnyContinues :: (g (GrammarFunctor (p g s)) -> g (GrammarFunctor (p g s))) -> (g (GrammarFunctor (p g s)) -> g (GrammarFunctor (p g s))) -> g (GrammarFunctor (p g s)) -> g (GrammarFunctor (p g s)) -> g (GrammarFunctor (p g s)) recurseTotal :: s -> g (GrammarFunctor (p g s) Rank2.~> GrammarFunctor (p g s)) -> [(s, g (GrammarFunctor (p g s)))] -> g (GrammarFunctor (p g s)) -> g (GrammarFunctor (p g s)) recurseMarginal :: s -> [(s, g (GrammarFunctor (p g s)))] -> g (GrammarFunctor (p g s)) -> g (GrammarFunctor (p g s)) maybeDependencies :: g (Const (Maybe (g (Const Bool)))) maybeDependency :: SeparatedParser p g s r -> Const (Maybe (g (Const Bool))) r appends :: g (ResultAppend p g s) parserAppend :: SeparatedParser p g s r -> ResultAppend p g s r directs = Rank2.fmap backParser parsers indirects = Rank2.fmap (\p-> case p of {CycleParser{}-> cycleParser p; _ -> empty}) parsers appends = Rank2.fmap parserAppend parsers parserAppend p@CycleParser{} = appendResultsArrow p parserAppend _ = Rank2.Arrow (Rank2.Arrow . const) maybeDependencies = Rank2.fmap maybeDependency parsers maybeDependency p@CycleParser{} = Const (Just $ dependencies p) maybeDependency _ = Const Nothing fixRecursive s parsedTail initial = whileAnyContinues (recurseTotal s (appends Rank2.<*> initial) parsedTail) (recurseMarginal s parsedTail) initial initial whileAnyContinues ft fm total marginal = Rank2.liftA3 choiceWhile maybeDependencies total (whileAnyContinues ft fm (ft total) (fm marginal)) where choiceWhile :: Const (Maybe (g (Const Bool))) x -> GrammarFunctor (p g s) x -> GrammarFunctor (p g s) x -> GrammarFunctor (p g s) x choiceWhile (Const Nothing) t _ = t choiceWhile (Const (Just deps)) t t' | getAny (Rank2.foldMap (Any . getConst) (Rank2.liftA2 combine deps marginal)) = t' | hasSuccess t = t | otherwise = let expected = expectations t FailureInfo pos expected' = failureOf (if getAny (Rank2.foldMap (Any . getConst) $ Rank2.liftA2 (combineFailures expected) deps marginal) then t' else t) in failWith (FailureInfo pos expected') where combine :: Const Bool x -> GrammarFunctor (p g s) x -> Const Bool x combineFailures :: [Expected s] -> Const Bool x -> GrammarFunctor (p g s) x -> Const Bool x combine (Const False) _ = Const False combine (Const True) results = Const (hasSuccess results) combineFailures _ (Const False) _ = Const False combineFailures expected (Const True) rl = Const (any (`notElem` expected) $ expectations rl) recurseTotal s initialAppends parsedTail total = Rank2.liftA2 reparse initialAppends indirects where reparse :: (GrammarFunctor (p g s) Rank2.~> GrammarFunctor (p g s)) a -> p g s a -> GrammarFunctor (p g s) a reparse append p = Rank2.apply append (parseTails p $ (s, total) : parsedTail) recurseMarginal s parsedTail marginal = flip parseTails ((s, marginal) : parsedTail) Rank2.<$> indirects {-# NOINLINE parseSeparated #-} class FallibleWithExpectations f where hasSuccess :: f s a -> Bool failureOf :: f s a -> FailureInfo s failWith :: FailureInfo s -> f s a expectations :: f s a -> [Expected s] instance FallibleWithExpectations (ResultList g) where hasSuccess (ResultList [] _) = False hasSuccess _ = True failureOf (ResultList _ failure) = failure failWith = ResultList [] expectations (ResultList _ (FailureInfo _ expected)) = expected