{-# LANGUAGE GeneralizedNewtypeDeriving #-} {-# LANGUAGE TemplateHaskell #-} {-# LANGUAGE TypeFamilies #-} -- | Pinchot internals. Ordinarily the "Pinchot" module should have -- everything you need. module Pinchot.Internal where import Pinchot.Intervals import Control.Applicative ((<|>), liftA2) import Control.Exception (Exception) import qualified Control.Lens as Lens import Control.Monad (join, when) import Control.Monad.Fix (MonadFix, mfix) import Control.Monad.Trans.Class (lift) import Control.Monad.Trans.Except (ExceptT, throwE, runExceptT) import Control.Monad.Trans.State (State, runState, get, put) import Data.Char (isUpper) import Data.Foldable (toList) import Data.Map (Map) import qualified Data.Map as M import Data.Monoid ((<>)) import Data.Set (Set) import Data.Sequence (Seq, ViewL(EmptyL, (:<)), viewl, (<|)) import qualified Data.Sequence as Seq import qualified Data.Set as Set import Data.Typeable (Typeable) import Language.Haskell.TH (ExpQ, ConQ, normalC, mkName, strictType, notStrict, newtypeD, cxt, conT, Name, dataD, appT, DecsQ, appE, Q, uInfixE, varE, varP, conE, Pat, Exp, recC, varStrictType, dyn) import qualified Language.Haskell.TH as TH import qualified Language.Haskell.TH.Syntax as Syntax import Text.Earley (satisfy, rule, symbol) import qualified Text.Earley -- | Type synonym for the name of a production rule. This will be the -- name of the type constructor for the corresponding type that will -- be created, so this must be a valid Haskell type constructor name. -- -- If you are creating a 'terminal', 'option', 'list', 'list1', or -- 'wrap', the 'RuleName' will also be used for the name of the single -- data construtor. If you are creating a 'nonTerminal', you will -- specify the name of each data constructor with 'AlternativeName'. type RuleName = String -- | Type synonym the the name of an alternative in a 'nonTerminal'. -- This name must not conflict with any other data constructor, either -- one specified as an 'AlternativeName' or one that was created using -- 'terminal', 'option', 'list', or 'list1'. type AlternativeName = String -- | A branch in a sum rule. In @Branch s ls@, @s@ is the name of the -- data constructor, and @ls@ is the list of rules that this branch -- produces. data Branch t = Branch String (Seq (Rule t)) deriving (Eq, Ord, Show) data RuleType t = RTerminal (Intervals t) | RBranch (Branch t, Seq (Branch t)) | RUnion (Rule t, Seq (Rule t)) | RSeqTerm (Seq t) | ROptional (Rule t) | RList (Rule t) | RList1 (Rule t) | RWrap (Rule t) | RRecord (Seq (Rule t)) deriving (Eq, Ord, Show) -- Rule n d t, where -- -- n is the name of the rule. This is used as the name of the -- corresponding data type. -- -- d is the description of the rule. This is optional and is used for -- the parser's error messages. If there is no description, the name -- is used for error messages. -- -- t is the type of rule (terminal, branch, etc.) -- | A single production rule. It may be a terminal or a non-terminal. data Rule t = Rule String (Maybe String) (RuleType t) deriving (Eq, Ord, Show) -- | Name a 'Rule' for use in error messages. If you do not name a -- rule using this combinator, the rule's type name will be used in -- error messages. label :: String -> Rule t -> Rule t label s (Rule n _ t) = Rule n (Just s) t -- | Infix form of 'label' for use in a 'Pinchot'; handy for use in -- @do@ or @mdo@ notation. () :: Pinchot t (Rule t) -> String -> Pinchot t (Rule t) p s = fmap (label s) p infixr 0 data Names t = Names { tyConNames :: Set RuleName , dataConNames :: Set String , nextIndex :: Int , allRules :: Map Int (Rule t) } deriving (Eq, Ord, Show) -- | Errors that may arise when constructing an AST. data Error = InvalidName String -- ^ A name was invalid. The field is the invalid name. The name -- might be invalid because it was already used, or because it does -- not begin with a capital letter. | EmptyNonTerminal String -- ^ A non-terminal must have at least one summand. The field is -- the name of the empty non-terminal. deriving (Show, Typeable) instance Exception Error -- | Constructs new 'Rule's. @t@ is the type of the token; often this -- will be 'Char'. -- -- 'Pinchot' is a 'Monad' and an 'Applicative' so you can combine -- computations using the usual methods of those classes. Also, -- 'Pinchot' is a 'MonadFix'. This allows you to construct a 'Rule' -- that depends on itself, and to construct sets of 'Rule's that have -- mutually recursive dependencies. 'MonadFix' also allows you to use -- the GHC @RecursiveDo@ extension. Put -- -- @ -- {-\# LANGUAGE RecursiveDo \#-} -- @ -- -- at the top of your module, then use @mdo@ instead of @do@. Because -- an @mdo@ block is recursive, you can use a binding before it is -- defined, just as you can in a set of @let@ bindings. newtype Pinchot t a = Pinchot { runPinchot :: ExceptT Error (State (Names t)) a } deriving (Functor, Applicative, Monad, MonadFix) -- | Runs a 'Pinchot' with a starting empty state. Fails in the Q -- monad if the grammar is bad. goPinchot :: Pinchot t a -> Q (Names t, a) goPinchot (Pinchot pinc) = case fst pair of Left err -> fail $ "pinchot: bad grammar: " ++ show err Right g -> return (snd pair, g) where pair = runState (runExceptT pinc) (Names Set.empty Set.empty 0 M.empty) addRuleName :: RuleName -> Pinchot t () addRuleName name = Pinchot $ do old@(Names tyNames _ _ _) <- lift get case name of [] -> throw x:_ -> do when (not (isUpper x)) throw when (Set.member name tyNames) throw lift $ put (old { tyConNames = Set.insert name tyNames }) where throw = throwE $ InvalidName name addDataConName :: AlternativeName -> Pinchot t () addDataConName name = Pinchot $ do old@(Names _ dcNames _ _) <- lift get case name of [] -> throw x:_ -> do when (not (isUpper x)) throw when (Set.member name dcNames) throw lift $ put (old { dataConNames = Set.insert name dcNames }) where throw = throwE $ InvalidName name newRule :: RuleName -> RuleType t -> Pinchot t (Rule t) newRule name ty = Pinchot $ do runPinchot (addRuleName name) st <- lift get let r = Rule name Nothing ty newSt = st { nextIndex = succ (nextIndex st) , allRules = M.insert (nextIndex st) r (allRules st) } lift (put newSt) runPinchot $ addDataConNames r return r -- | Creates a terminal production rule. terminal :: RuleName -> Intervals t -- ^ Valid terminal symbols -> Pinchot t (Rule t) terminal name ivls = newRule name (RTerminal ivls) splitNonTerminal :: String -> Seq (String, Seq (Rule t)) -> Pinchot t ((String, Seq (Rule t)), Seq (String, Seq (Rule t))) splitNonTerminal n sq = Pinchot $ case viewl sq of EmptyL -> throwE $ EmptyNonTerminal n x :< xs -> return (x, xs) -- | Creates a production for a sequence of terminals. Useful for -- parsing specific words. terminalSeq :: RuleName -> Seq t -- ^ Sequence of terminal symbols to recognize -> Pinchot t (Rule t) terminalSeq name sq = newRule name (RSeqTerm sq) -- | Creates a new non-terminal production rule. nonTerminal :: RuleName -> Seq (AlternativeName, Seq (Rule t)) -- ^ Alternatives. There must be at least one alternative; -- otherwise, an error will result. In each pair @(a, b)@, @a@ will -- be the data constructor, so this must be a valid Haskell data -- constructor name. @b@ is the sequence of production rules, which -- can be empty (this is how to create an epsilon production). -> Pinchot t (Rule t) nonTerminal name sq = do (b1, bs) <- splitNonTerminal name sq let branches = RBranch (uncurry Branch b1, fmap (uncurry Branch) bs) newRule name branches ruleConstructorNames :: Rule t -> Seq AlternativeName ruleConstructorNames (Rule n _ t) = case t of RTerminal _ -> Seq.singleton n RBranch (b1, bs) -> branchName b1 <| fmap branchName bs where branchName (Branch x _) = x RUnion (b1, bs) -> branchName b1 <| fmap branchName bs where branchName (Rule x _ _) = unionBranchName n x RSeqTerm _ -> Seq.singleton n ROptional _ -> Seq.singleton n RList _ -> Seq.singleton n RList1 _ -> Seq.singleton n RWrap _ -> Seq.singleton n RRecord _ -> Seq.singleton n unionBranchName :: RuleName -- ^ Name of the parent rule -> RuleName -- ^ Name of the branch rule -> AlternativeName unionBranchName p b = p ++ '\'' : b addDataConNames :: Rule t -> Pinchot t () addDataConNames = mapM_ addDataConName . ruleConstructorNames -- | Creates a new non-terminal production rule where each alternative -- produces only one rule. The constructor name for each alternative -- is -- -- @RULE_NAME'PRODUCTION_NAME@ -- -- where @RULE_NAME@ is the name of the rule itself, and -- @PRODUCTION_NAME@ is the rule name for what is being produced. For -- an example, see 'Pinchot.Examples.PostalAstAllRules.Suffix'. -- -- Currently there is no way to change the names of the constructors; -- however, you can use 'nonTerminal', which is more flexible. union :: RuleName -> Seq (Rule t) -- ^ List of alternatives. There must be at least one alternative; -- otherwise a compile-time error will occur. -> Pinchot t (Rule t) union name sq = Pinchot $ case viewl sq of EmptyL -> throwE $ EmptyNonTerminal name x :< xs -> runPinchot $ newRule name (RUnion (x, xs)) -- | Creates a new non-terminal production rule with only one -- alternative where each field has a record name. The name of each -- record is: -- -- @_r\'RULE_NAME\'INDEX\'FIELD_TYPE@ -- -- where @RULE_NAME@ is the name of this rule, @INDEX@ is the index number -- for this field (starting with 0), and @FIELD_TYPE@ is the type of the -- field itself. For an example, see -- 'Pinchot.Examples.PostalAstAllRules.Address'. -- -- Currently there is no way to change the names of the record fields. record :: RuleName -- ^ The name of this rule, which is used both as the type name and -- the name of the sole data constructor. -> Seq (Rule t) -- ^ The right-hand side of this rule. This sequence can be empty, -- which results in an epsilon production. -> Pinchot t (Rule t) record name sq = newRule name (RRecord sq) -- | Creates a rule for the production of a sequence of other rules. -- The name for the created 'Rule' is the name of the 'Rule' to which -- this function is applied, with @'Seq@ appended. list :: Rule t -- ^ The resulting 'Rule' is a sequence of productions of this -- 'Rule'; that is, this 'Rule' may appear zero or more times. -> Pinchot t (Rule t) list r@(Rule inner _ _) = newRule name (RList r) where name = inner ++ "'Seq" -- | Creates a rule for a production that appears at least once. The -- name for the created 'Rule' is the name of the 'Rule' to which this -- function is applied, with @'Seq1@ appended. list1 :: Rule t -- ^ The resulting 'Rule' produces this 'Rule' at least once. -> Pinchot t (Rule t) list1 r@(Rule inner _ _) = newRule name (RList1 r) where name = inner ++ "'Seq1" -- | Creates a rule for a production that optionally produces another -- rule. The name for the created 'Rule' is the name of the 'Rule' to -- which this function is applied, with @'Maybe@ appended to the end. option :: Rule t -- ^ The resulting 'Rule' optionally produces this 'Rule'; that is, -- this 'Rule' may appear once or not at all. -> Pinchot t (Rule t) option r@(Rule inner _ _) = newRule name (ROptional r) where name = inner ++ "'Maybe" -- | Creates a newtype wrapper. wrap :: RuleName -> Rule t -- ^ The resulting 'Rule' simply wraps this 'Rule'. -> Pinchot t (Rule t) wrap name r = newRule name (RWrap r) -- | Gets all ancestor 'Rule's. Skips duplicates. getAncestors :: Rule t -> State (Set String) (Seq (Rule t)) getAncestors r@(Rule name _ ei) = do set <- get if Set.member name set then return Seq.empty else do put (Set.insert name set) case ei of RTerminal _ -> return (Seq.singleton r) RBranch (b1, bs) -> do as1 <- branchAncestors b1 ass <- fmap join . mapM branchAncestors $ bs return $ r <| as1 <> ass RUnion (b1, bs) -> do c1 <- getAncestors b1 cs <- fmap join . mapM getAncestors $ bs return $ r <| c1 <> cs RSeqTerm _ -> return (Seq.singleton r) ROptional c -> do cs <- getAncestors c return $ r <| cs RList c -> do cs <- getAncestors c return $ r <| cs RList1 c -> do cs <- getAncestors c return $ r <| cs RWrap c -> do cs <- getAncestors c return $ r <| cs RRecord ls -> do cs <- fmap join . mapM getAncestors $ ls return $ r <| cs where branchAncestors (Branch _ rs) = fmap join . mapM getAncestors $ rs -- | Returns both this 'Rule' and any 'Rule's that are ancestors. ruleAndAncestors :: Rule t -> Seq (Rule t) ruleAndAncestors r = fst $ runState (getAncestors r) Set.empty -- | Given a sequence of 'Rule', determine which rules are on a -- right-hand side before they are defined. rulesDemandedBeforeDefined :: Foldable f => f (Rule t) -> Set Name rulesDemandedBeforeDefined = snd . foldl f (Set.empty, Set.empty) where f (lhsDefined, results) (Rule nm _ ty) = (Set.insert nm lhsDefined, results') where results' = case ty of RTerminal _ -> results RBranch (b1, bs) -> foldr checkBranch (checkBranch b1 results) bs where checkBranch (Branch _ rls) rslts = foldr checkRule rslts rls RUnion (b1, bs) -> foldr checkRule (checkRule b1 results) bs RSeqTerm _ -> results ROptional r -> checkRule r results RList r -> addHelper $ checkRule r results RList1 r -> addHelper $ checkRule r results RWrap r -> checkRule r results RRecord sq -> foldr checkRule results $ sq checkRule (Rule name _ _) rslts | Set.member name lhsDefined = rslts | otherwise = Set.insert (ruleName name) rslts addHelper = Set.insert (helperName nm) thBranch :: Branch t -> ConQ thBranch (Branch nm rules) = normalC name fields where name = mkName nm mkField (Rule n _ _) = strictType notStrict (conT (mkName n)) fields = toList . fmap mkField $ rules thUnionBranch :: RuleName -- ^ Parent rule name -> Rule t -- ^ Child rule -> ConQ thUnionBranch parent (Rule child _ _) = normalC name fields where name = mkName (unionBranchName parent child) fields = [strictType notStrict (conT (mkName child))] thRule :: Syntax.Lift t => Bool -- ^ If True, make lenses. -> Name -- ^ Name of terminal type -> Seq Name -- ^ What to derive -> Rule t -> TH.Q [TH.Dec] thRule doLenses typeName derives (Rule nm _ ruleType) = do ty <- makeType typeName derives nm ruleType lenses <- if doLenses then ruleToOptics typeName nm ruleType else return [] inst <- productionDecl nm typeName ruleType return (ty : inst ++ lenses) makeType :: Name -- ^ Name of terminal type -> Seq Name -- ^ What to derive -> String -- ^ Name of rule -> RuleType t -> TH.Q TH.Dec makeType typeName derivesSeq nm ruleType = case ruleType of RTerminal _ -> newtypeD (cxt []) name [] newtypeCon derives where newtypeCon = normalC name [strictType notStrict (conT typeName)] RBranch (b1, bs) -> dataD (cxt []) name [] cons derives where cons = thBranch b1 : toList (fmap thBranch bs) RUnion (b1, bs) -> dataD (cxt []) name [] cons derives where cons = thUnionBranch nm b1 : toList (fmap (thUnionBranch nm) bs) RSeqTerm _ -> newtypeD (cxt []) name [] cons derives where cons = normalC name [strictType notStrict (appT [t| Seq |] (conT typeName))] ROptional (Rule inner _ _) -> newtypeD (cxt []) name [] newtypeCon derives where newtypeCon = normalC name [strictType notStrict (appT [t| Maybe |] (conT (mkName inner)))] RList (Rule inner _ _) -> newtypeD (cxt []) name [] newtypeCon derives where newtypeCon = normalC name [strictType notStrict (appT [t| Seq |] (conT (mkName inner)))] RList1 (Rule inner _ _) -> newtypeD (cxt []) name [] cons derives where cons = normalC name [ strictType notStrict (TH.tupleT 2 `appT` (conT (mkName inner)) `appT` ([t| Seq |] `appT` (conT (mkName inner)))) ] RWrap (Rule inner _ _) -> newtypeD (cxt []) name [] newtypeCon derives where newtypeCon = normalC name [ strictType notStrict (conT (mkName inner)) ] RRecord sq -> dataD (cxt []) name [] [ctor] derives where ctor = recC name . zipWith mkField [(0 :: Int) ..] . toList $ sq mkField num (Rule rn _ _) = varStrictType (mkName fldNm) (strictType notStrict (conT (mkName rn))) where fldNm = '_' : fieldName num nm rn where name = mkName nm derives = toList derivesSeq -- | Field name - without a leading underscore fieldName :: Int -- ^ Index -> String -- ^ Parent type name -> String -- ^ Inner type name -> String fieldName idx par inn = "r'" ++ par ++ "'" ++ show idx ++ "'" ++ inn thAllRules :: Syntax.Lift t => Bool -- ^ If True, make optics as well. -> Name -- ^ Terminal type constructor name -> Seq Name -- ^ What to derive -> Map Int (Rule t) -> DecsQ thAllRules doOptics typeName derives = fmap join . sequence . fmap (thRule doOptics typeName derives) . fmap snd . M.toAscList makeWrapped :: TH.Type -- ^ Name of wrapped type -> String -- ^ Name of wrapper type -> TH.Dec makeWrapped wrappedType nm = TH.InstanceD [] typ decs where name = TH.mkName nm local = mkName "_x" typ = (TH.ConT ''Lens.Wrapped) `TH.AppT` (TH.ConT name) decs = [assocType, wrapper] where assocType = TH.TySynInstD ''Lens.Unwrapped (TH.TySynEqn [TH.ConT name] wrappedType) wrapper = TH.FunD 'Lens._Wrapped [TH.Clause [] (TH.NormalB body) []] where body = (TH.VarE 'Lens.iso) `TH.AppE` unwrap `TH.AppE` doWrap where unwrap = TH.LamE [lambPat] (TH.VarE local) where lambPat = TH.ConP name [TH.VarP local] doWrap = TH.LamE [lambPat] expn where expn = (TH.ConE name) `TH.AppE` (TH.VarE local) lambPat = TH.VarP local -- | TH helper like 'dyn' but for patterns dynP :: String -> TH.PatQ dynP = TH.varP . TH.mkName seqTermToOptics :: Syntax.Lift t => Name -- ^ Terminal type name -> String -- ^ Rule name -> Seq t -> TH.Q [TH.Dec] seqTermToOptics termName nm sq = do e1 <- TH.sigD (TH.mkName ('_':nm)) (TH.conT ''Lens.Prism' `TH.appT` (TH.conT ''Seq `TH.appT` TH.conT termName) `TH.appT` TH.conT (TH.mkName nm)) e2 <- TH.valD prismName (TH.normalB expn) [] return [e1, e2] where prismName = TH.varP (TH.mkName ('_' : nm)) fetchPat = TH.conP (TH.mkName nm) [TH.varP (TH.mkName "_x")] fetchName = TH.varE (TH.mkName "_x") ctor = TH.conE (TH.mkName nm) expn = [| let fetch $fetchPat = $fetchName store _term | $(liftSeq sq) == _term = Right ($ctor _term) | otherwise = Left _term in Lens.prism fetch store |] -- | Creates a prism for a terminal type. Although a newtype wraps -- each terminal, do not make a Wrapped or an Iso, because the -- relationship between the outer type and the type that it wraps -- typically is not isometric. Thus, use a Prism instead, which -- captures this relationship properly. terminalToOptics :: Syntax.Lift t => Name -- ^ Terminal type name -> String -- ^ Rule name -> Intervals t -> TH.Q [TH.Dec] terminalToOptics termName nm ivls = do e1 <- TH.sigD (TH.mkName ('_':nm)) (TH.conT ''Lens.Prism' `TH.appT` TH.conT termName `TH.appT` TH.conT (TH.mkName nm)) e2 <- TH.valD prismName (TH.normalB expn) [] return [e1, e2] where prismName = TH.varP (TH.mkName ('_' : nm)) fetchPat = TH.conP (TH.mkName nm) [TH.varP (TH.mkName "_x")] fetchName = TH.varE (TH.mkName "_x") ctor = TH.conE (TH.mkName nm) expn = [| let fetch $fetchPat = $fetchName store _term | inIntervals ivls _term = Right ($ctor _term) | otherwise = Left _term in Lens.prism fetch store |] optionalToOptics :: String -- ^ Wrapped rule name -> String -- ^ Wrapping Rule name -> TH.Dec optionalToOptics wrappedName = makeWrapped maybeName where maybeName = (TH.ConT ''Maybe) `TH.AppT` (TH.ConT (TH.mkName wrappedName)) many1ToOptics :: String -- ^ Wrapped rule name -> String -- ^ Wrapping Rule name -> TH.Dec many1ToOptics wrappedName = makeWrapped tupName where tupName = (TH.TupleT 2) `TH.AppT` (TH.ConT (TH.mkName wrappedName)) `TH.AppT` ((TH.ConT ''Seq) `TH.AppT` (TH.ConT (TH.mkName wrappedName))) manyToOptics :: String -- ^ Wrapped rule name -> String -- ^ Wrapping Rule name -> TH.Dec manyToOptics wrappedName = makeWrapped innerName where innerName = (TH.ConT ''Seq) `TH.AppT` (TH.ConT (TH.mkName wrappedName)) wrapToOptics :: String -- ^ Wrapped rule name -> String -- ^ Wrapping Rule name -> TH.Dec wrapToOptics wrappedName = makeWrapped innerName where innerName = TH.ConT (TH.mkName wrappedName) terminalSeqToOptics :: Name -- ^ Terminal type name -> String -- ^ Rule name -> TH.Dec terminalSeqToOptics terminalName = makeWrapped sqType where sqType = (TH.ConT ''Seq) `TH.AppT` (TH.ConT terminalName) branchesToOptics :: String -- ^ Rule name -> Branch t -> Seq (Branch t) -> [TH.Dec] branchesToOptics nm b1 bsSeq = concat $ makePrism b1 : fmap makePrism bs where bs = toList bsSeq makePrism (Branch inner rulesSeq) = [ signature, binding ] where rules = toList rulesSeq prismName = TH.mkName ('_' : inner) signature = TH.SigD prismName $ (TH.ConT ''Lens.Prism') `TH.AppT` (TH.ConT (TH.mkName nm)) `TH.AppT` fieldsType where fieldsType = case rules of [] -> TH.TupleT 0 Rule r1 _ _ : [] -> TH.ConT (TH.mkName r1) rs -> foldl addType (TH.TupleT (length rs)) rs where addType soFar (Rule r _ _) = soFar `TH.AppT` (TH.ConT (TH.mkName r)) binding = TH.ValD (TH.VarP prismName) body [] where body = TH.NormalB $ (TH.VarE 'Lens.prism) `TH.AppE` setter `TH.AppE` getter where setter = TH.LamE [pat] expn where (pat, expn) = case rules of [] -> (TH.TupP [], TH.ConE (TH.mkName inner)) _ : [] -> (TH.VarP local, TH.ConE (TH.mkName inner) `TH.AppE` TH.VarE local) where local = TH.mkName "_x" ls -> (TH.TupP pats, set) where pats = fmap (\i -> TH.VarP (mkName ("_x" ++ show i))) . take (length ls) $ [(0 :: Int) ..] set = foldl addVar start . take (length ls) $ [(0 :: Int) ..] where addVar acc i = acc `TH.AppE` (TH.VarE (TH.mkName ("_x" ++ show i))) start = TH.ConE (TH.mkName inner) getter = TH.LamE [pat] expn where local = TH.mkName "_x" pat = TH.VarP local expn = TH.CaseE (TH.VarE (TH.mkName "_x")) $ TH.Match patCtor bodyCtor [] : rest where patCtor = TH.ConP (TH.mkName inner) . fmap (\i -> TH.VarP (TH.mkName $ "_y" ++ show i)) . take (length rules) $ [(0 :: Int) ..] bodyCtor = TH.NormalB . (TH.ConE 'Right `TH.AppE`) $ case rules of [] -> TH.TupE [] _:[] -> TH.VarE (TH.mkName "_y0") _ -> TH.TupE . fmap (\i -> TH.VarE (TH.mkName $ "_y" ++ show i)) . take (length rules) $ [(0 :: Int) ..] rest = case bs of [] -> [] _ -> [TH.Match patBlank bodyBlank []] where patBlank = TH.VarP (TH.mkName "_z") bodyBlank = TH.NormalB $ TH.ConE ('Left) `TH.AppE` TH.VarE (TH.mkName "_z") unionToOptics :: String -- ^ Rule name -> Rule t -- ^ First rule -> Seq (Rule t) -- ^ Remaining rules -> TH.DecsQ unionToOptics parentName r1 rs = fmap concat . sequence $ optics r1 : fmap optics (toList rs) where optics (Rule r _ _) = sequence $ sig : prism : [] where sig = TH.sigD prismName [t| Lens.Prism' $bigType $innerType |] prismName = TH.mkName $ "_" ++ parentName ++ "'" ++ r bigType = TH.conT (TH.mkName parentName) innerType = TH.conT (TH.mkName r) prism = TH.valD (TH.varP prismName) (TH.normalB [| Lens.prism $dataCtor $sToA |] ) [] sToA = TH.lamE [pat] expn where pat = dynP "_x" expn = TH.caseE (dyn "_x") [ TH.match (TH.conP (TH.mkName (unionBranchName parentName r)) [TH.varP (TH.mkName "_a")]) (TH.normalB [| Right $(dyn "_a") |]) [] , TH.match (dynP "_b") (TH.normalB [| Left $(dyn "_b") |]) [] ] dataCtor = TH.conE (TH.mkName (unionBranchName parentName r)) recordsToOptics :: String -- ^ Rule name -> Seq (Rule t) -> [TH.Dec] recordsToOptics nm = concat . zipWith makeLens [(0 :: Int) ..] . toList where makeLens index (Rule inner _ _) = [ signature, function ] where fieldNm = fieldName index nm inner lensName = mkName fieldNm signature = TH.SigD lensName $ (TH.ConT ''Lens.Lens') `TH.AppT` (TH.ConT (TH.mkName nm)) `TH.AppT` (TH.ConT (TH.mkName inner)) function = TH.FunD lensName [TH.Clause [] (TH.NormalB body) []] where namedRec = TH.mkName "_namedRec" namedNewVal = TH.mkName "_namedNewVal" body = (TH.VarE 'Lens.lens) `TH.AppE` getter `TH.AppE` setter where getter = TH.LamE [pat] expn where pat = TH.VarP namedRec expn = (TH.VarE (TH.mkName ('_' : fieldNm))) `TH.AppE` (TH.VarE namedRec) setter = TH.LamE [patRec, patNewVal] expn where patRec = TH.VarP namedRec patNewVal = TH.VarP namedNewVal expn = TH.RecUpdE (TH.VarE namedRec) [ (TH.mkName ('_' : fieldNm), TH.VarE namedNewVal) ] ruleToOptics :: Syntax.Lift t => Name -- ^ Terminal type name -> String -- ^ Rule name -> RuleType t -> TH.DecsQ ruleToOptics terminalName nm ty = case ty of RTerminal ivl -> terminalToOptics terminalName nm ivl RBranch (b1, bs) -> return $ branchesToOptics nm b1 bs RUnion (r1, rs) -> unionToOptics nm r1 rs RSeqTerm sq -> seqTermToOptics terminalName nm sq ROptional (Rule inner _ _) -> return [optionalToOptics inner nm] RList (Rule inner _ _) -> return [manyToOptics inner nm] RList1 (Rule inner _ _) -> return [many1ToOptics inner nm] RWrap (Rule inner _ _) -> return [wrapToOptics inner nm] RRecord recs -> return $ recordsToOptics nm recs -- | Should optics be made? type MakeOptics = Bool -- | Creates optics. -- -- If you use this option, you will need -- @ -- \{\-\# LANGUAGE TypeFamilies \#\-\} -- @ -- -- at the top of the module into which you splice in the -- declarations, because you will get instances of 'Lens.Wrapped'. -- -- Creates the listed optics for each kind of -- 'Rule', as follows: -- -- * 'terminal': @'Lens.Prism'' a b@, where @a@ is the type of the -- terminal token (often 'Char') and @b@ is the type of this -- particular production. For an example, see -- 'Pinchot.Examples.PostalAstAllRules._Comma'. -- -- >>> ',' ^? _Comma -- Just (Comma ',') -- >>> 'a' ^? _Comma -- Nothing -- >>> Comma ',' ^. re _Comma -- ',' -- -- Thus this gives you a safe way to insert tokens into types made -- with 'terminal' (useful if you want to construct a syntax tree.) -- -- * 'terminalSeq': @'Lens.Prism'' ('Seq' a) b@, where @a@ is the type -- of the terminal token (often 'Char') and @b@ is the type of this -- particular production. As with 'terminal' this gives you a safe -- way to insert values into the types made with 'terminalSeq'. -- -- * 'nonTerminal': one 'Lens.Prism'' for each data constructor (even if -- there is only one data constructor) -- -- * 'union': one 'Lens.Prism' for each data constructor (even if -- there is only one data constructor) -- -- * 'record': one 'Lens.Lens' for each field -- -- * 'list': 'Lens.Wrapped', wrapping a @'Seq' a@ -- -- * 'list1': 'Lens.Wrapped', wrapping a pair @(a, 'Seq' a)@ -- -- * 'option': 'Lens.Wrapped', wrapping a @'Maybe' a@ -- -- * 'wrap': 'Lens.Wrapped', wrapping the underlying type makeOptics :: MakeOptics makeOptics = True -- | Do not make any optics. noOptics :: MakeOptics noOptics = False -- | Creates data types for every 'Rule' created in the 'Pinchot'. The data -- types are created in the same order in which they were created in -- the 'Pinchot'. When spliced, the 'DecsQ' is a list of -- declarations, each of which is an appropriate @data@ or @newtype@. -- For an example use of 'allRulesToTypes', see -- "Pinchot.Examples.PostalAstAllRules". -- -- Also creates bindings whose names are prefixed with @t'@. Each -- of these is a function that, when given a particular production, -- reduces it to a sequence of terminal symbols. allRulesToTypes :: Syntax.Lift t => MakeOptics -> Name -- ^ Terminal type constructor name. Typically you will use the -- Template Haskell quoting mechanism to get this. -> Seq Name -- ^ What to derive. For instance, you might use @Eq@, @Ord@, and -- @Show@ here. Each created data type will derive these instances. -> Pinchot t a -- ^ The return value from the 'Pinchot' is ignored. -> DecsQ allRulesToTypes doOptics typeName derives pinchot = do st' <- fmap fst $ goPinchot pinchot thAllRules doOptics typeName derives (allRules st') -- | Creates data types only for the 'Rule' returned from the 'Pinchot', and -- for its ancestors. -- -- Also creates bindings whose names are prefixed with @t'@. Each -- of these is a function that, when given a particular production, -- reduces it to a sequence of terminal symbols. ruleTreeToTypes :: Syntax.Lift t => MakeOptics -> Name -- ^ Terminal type constructor name. Typically you will use the -- Template Haskell quoting mechanism to get this. -> Seq Name -- ^ What to derive. For instance, you might use @Eq@, @Ord@, and -- @Show@ here. Each created data type will derive these instances. -> Pinchot t (Rule t) -- ^ A data type is created for the 'Rule' that the 'Pinchot' -- returns, and for the ancestors of the 'Rule'. -> DecsQ ruleTreeToTypes doOptics typeName derives pinchot = do r <- fmap snd $ goPinchot pinchot fmap join . sequence . toList . fmap (thRule doOptics typeName derives) . runCalc . getAncestors $ r where runCalc stateCalc = fst $ runState stateCalc Set.empty addPrefix :: String -> String -> String addPrefix pfx suf | null pfx = suf | otherwise = pfx ++ '.':suf ruleToParser :: Syntax.Lift t => String -- ^ Module prefix -> Rule t -> [(Name, ExpQ)] ruleToParser prefix (Rule nm mayDescription rt) = case rt of RTerminal ivls -> [makeRule expression] where expression = [| fmap $constructor (satisfy (inIntervals ivls)) |] RBranch (b1, bs) -> [makeRule expression] where expression = foldl addBranch (branchToParser prefix b1) bs where addBranch tree branch = [| $tree <|> $(branchToParser prefix branch) |] RUnion (Rule r1 _ _, rs) -> [makeRule expression] where expression = foldl adder start rs where branch r = [| $(conE (mkName (addPrefix prefix . unionBranchName nm $ r))) <$> $(varE (ruleName r)) |] start = branch r1 adder soFar (Rule r _ _) = [| $soFar <|> $(branch r) |] RSeqTerm sq -> [nestRule, topRule] where nestRule = (helper, [| rule $(foldl addTerm start sq) |]) where start = [|pure Seq.empty|] addTerm acc x = [| liftA2 (<|) (symbol x) $acc |] topRule = makeRule (wrapper helper) ROptional (Rule innerNm _ _) -> [makeRule expression] where expression = [| fmap $constructor (pure Nothing <|> $(just)) |] where just = [| fmap Just $(varE (ruleName innerNm)) |] RList (Rule innerNm _ _) -> [nestRule, makeRule (wrapper helper)] where nestRule = (helper, ([|rule|] `appE` parseSeq)) where parseSeq = uInfixE [|pure Seq.empty|] [|(<|>)|] pSeq where pSeq = [|liftA2 (<|) $(varE (ruleName innerNm)) $(varE helper) |] RList1 (Rule innerNm _ _) -> [nestRule, makeRule topExpn] where nestRule = (helper, [|rule $(parseSeq)|]) where parseSeq = [| pure Seq.empty <|> $pSeq |] where pSeq = [| (<|) <$> $(varE (ruleName innerNm)) <*> $(varE helper) |] topExpn = [| $constructor <$> ( (,) <$> $(varE (ruleName innerNm)) <*> $(varE helper) ) |] RWrap (Rule innerNm _ _) -> [makeRule expression] where expression = [|fmap $constructor $(varE (ruleName innerNm)) |] RRecord sq -> [makeRule expression] where expression = case viewl sq of EmptyL -> [| pure $constructor |] Rule r1 _ _ :< restFields -> foldl addField fstField restFields where fstField = [| $constructor <$> $(varE (ruleName r1)) |] addField soFar (Rule r _ _) = [| $soFar <*> $(varE (ruleName r)) |] where makeRule expression = (ruleName nm, [|rule ($expression Text.Earley. $(textToExp desc))|]) desc = maybe nm id mayDescription textToExp txt = [| $(Syntax.lift txt) |] constructor = constructorName prefix nm wrapper wrapRule = [|fmap $constructor $(varE wrapRule) |] helper = helperName nm constructorName :: String -- ^ Module prefix -> String -- ^ Name of constructor -> ExpQ constructorName pfx nm = conE (mkName name) where name = pfx' ++ nm pfx' | null pfx = "" | otherwise = pfx ++ "." ruleName :: String -> Name ruleName suffix = mkName ("_rule'" ++ suffix) helperName :: String -> Name helperName suffix = mkName ("_helper'" ++ suffix) branchToParser :: Syntax.Lift t => String -- ^ Module prefix -> Branch t -> ExpQ branchToParser prefix (Branch name rules) = case viewl rules of EmptyL -> [| pure $constructor |] (Rule rule1 _ _) :< xs -> foldl f z xs where z = [| $constructor <$> $(varE (ruleName rule1)) |] f soFar (Rule rule2 _ _) = [| $soFar <*> $(varE (ruleName rule2)) |] where constructor = constructorName prefix name -- # lazyPattern and bigTuple - because TH has no support for -- mdo notation -- | Creates a lazy pattern for all the given names. Adds an empty -- pattern onto the front. This is the counterpart of 'bigTuple'. -- All of the given names are bound. In addition, a single, -- wildcard pattern is bound to the front. -- -- For example, @lazyPattern (map mkName ["x", "y", "z"])@ gives a -- pattern that looks like -- -- @~(_, (x, (y, (z, ()))))@ -- -- The idea is that the named patterns are needed so that the -- recursive @do@ notation works, and that the wildcard pattern is -- the return value, which is not needed here. lazyPattern :: Foldable c => c Name -> Q Pat lazyPattern = finish . foldr gen [p| () |] where gen name rest = [p| ($(varP name), $rest) |] finish pat = [p| ~(_, $pat) |] -- | Creates a big tuple. It is nested in the second element, such -- as (1, (2, (3, (4, ())))). Thus, the big tuple is terminated -- with a unit value. It resembles a list where each tuple is a -- cons cell and the terminator is unit. bigTuple :: Foldable c => ExpQ -- ^ This expression will be the first one in the tuple. -> c ExpQ -- ^ Remaining expressions in the tuple. -> ExpQ bigTuple top = finish . foldr f [| () |] where f n rest = [| ( $(n), $rest) |] finish tup = [| ($(top), $tup) |] -- | Creates an Earley grammar for a given 'Rule'. For examples of how -- to use this, see the source code for -- "Pinchot.Examples.PostalAstRuleTree" and for -- "Pinchot.Examples.PostalAstAllRules". earleyGrammar :: Syntax.Lift t => Qualifier -- ^ Qualifier for data types crated with 'ruleTreeToTypes' or -- 'allRulesToTypes' -> Pinchot t (Rule t) -- ^ Creates an Earley parser for the 'Rule' that the 'Pinchot' -- returns. -> Q Exp -- ^ When spliced, this expression has type -- @'Text.Earley.Grammar' r ('Text.Earley.Prod' r 'String' t a)@ -- -- where -- -- @r@ is left universally quantified -- -- @t@ is the type of the token (usually 'Char') -- -- @a@ is the type defined by the 'Rule'. earleyGrammar prefix pinc = do r <- fmap snd $ goPinchot pinc earleyGrammarFromRule prefix r -- | Builds a recursive @do@ expression (because TH has no support -- for @mdo@ notation). recursiveDo :: [(Name, ExpQ)] -- ^ Binding statements -> ExpQ -- ^ Final return value from @do@ block. The type of this 'ExpQ' -- must be in the same monad as the @do@ block; it must not be a -- pure value. -> ExpQ -- ^ Returns an expression whose value is the final return value -- from the @do@ block. recursiveDo binds final = [| fmap fst $ mfix $(fn) |] where fn = [| \ $(lazyPattern (fmap fst binds)) -> $doBlock |] doBlock = TH.doE (bindStmts ++ returnStmts) bindStmts = map mkBind binds where mkBind (name, exp) = TH.bindS (TH.varP name) exp returnStmts = [bindRtnVal, returner] where rtnValName = TH.mkName "_returner" bindRtnVal = TH.bindS (TH.varP rtnValName) final returner = TH.noBindS [| return $(bigTuple (TH.varE rtnValName) (fmap (TH.varE . fst) binds)) |] earleyGrammarFromRule :: Syntax.Lift t => String -- ^ Module prefix -> Rule t -> Q Exp earleyGrammarFromRule prefix r@(Rule top _ _) = recursiveDo binds final where binds = concatMap (ruleToParser prefix) . toList . ruleAndAncestors $ r final = [| return $(TH.varE $ ruleName top) |] -- | Creates an Earley grammar for each 'Rule' created in a -- 'Pinchot'. For a 'Pinchot' with a large number of 'Rule's, this -- can create a large number of declarations that can take a long -- time to compile--sometimes several minutes. For lower -- compilation times, try 'earleyProduct'. allEarleyGrammars :: Syntax.Lift t => Qualifier -- ^ Qualifier for data types created with 'ruleTreeToTypes' or -- 'allRulesToTypes' -> Name -- ^ Name for the terminal type; often this is 'Char'. Typically -- you will use the Template Haskell quoting mechanism--for -- example, @\'\'Char@. -> Pinchot t a -- ^ Creates an Earley grammar for each 'Rule' created in the -- 'Pinchot'. The return value of the 'Pinchot' computation is -- ignored. -> DecsQ -- ^ When spliced, this is a list of declarations. Each -- declaration has type -- @'Text.Earley.Grammar' r ('Text.Earley.Prod' r 'String' t a)@ -- -- where -- -- @r@ is left universally quantified -- -- @t@ is the type of the token (usually 'Char') -- -- @a@ is the type defined by the 'Rule'. -- -- The name of each declaration is -- g'TYPE_NAME -- -- where TYPE_NAME is the name of the type defined in the -- corresponding 'Rule'. allEarleyGrammars prefix termName pinc = do st <- fmap fst $ goPinchot pinc sequence . concat . fmap makeDecl . fmap snd . M.toList . allRules $ st where makeDecl rule@(Rule nm _ _) = [signature, TH.valD pat body []] where signature = TH.sigD name types r = TH.mkName "r" types = TH.forallT [TH.PlainTV r] (return []) $ [t| Text.Earley.Grammar $(TH.varT r) (Text.Earley.Prod $(TH.varT r) String $(TH.conT termName) $(TH.conT qualRuleName)) |] name = TH.mkName $ "g'" ++ nm pat = TH.varP name body = TH.normalB (earleyGrammarFromRule prefix rule) qualRuleName | null prefix = TH.mkName nm | otherwise = TH.mkName (prefix ++ "." ++ nm) prodDeclName :: String -> TH.Name prodDeclName name = TH.mkName $ "t'" ++ name prodFn :: String -> TH.ExpQ prodFn = TH.varE . prodDeclName addIndices :: Foldable c => c a -> [(Int, a)] addIndices = zip [0..] . toList -- | Creates a production declaration for a 'Rule'. productionDecl :: String -- ^ Rule name -> Name -- ^ Name of terminal type -> RuleType t -> TH.DecsQ productionDecl n termType t = sequence [signature, TH.funD (prodDeclName n) clauses] where signature = TH.sigD (prodDeclName n) types where types = TH.appT (TH.appT TH.arrowT (TH.conT (TH.mkName n))) (TH.appT (TH.conT ''Seq) (TH.conT termType)) clauses = case t of RTerminal _ -> [TH.clause [pat] bdy []] where pat = TH.conP (TH.mkName n) [TH.varP (TH.mkName "_x")] bdy = TH.normalB [| Seq.singleton $(TH.varE (TH.mkName "_x")) |] RBranch (b1, bs) -> branchToClause b1 : fmap branchToClause (toList bs) RSeqTerm _ -> [TH.clause [pat] bdy []] where pat = TH.conP (TH.mkName n) [TH.varP (TH.mkName "_x")] bdy = TH.normalB (dyn "_x") ROptional (Rule inner _ _) -> [justClause, nothingClause] where justClause = TH.clause [TH.conP (TH.mkName n) [TH.conP 'Just [TH.varP (TH.mkName "_b")]]] (TH.normalB [| $(prodFn inner) $(TH.varE (TH.mkName "_b")) |]) [] nothingClause = TH.clause [TH.conP (TH.mkName n) [TH.conP 'Nothing []]] (TH.normalB [| Seq.empty |]) [] RList (Rule inner _ _) -> [TH.clause [pat] bdy []] where pat = TH.conP (TH.mkName n) [TH.varP (TH.mkName "_a")] bdy = TH.normalB [| join $ fmap $(prodFn inner) $(TH.varE (TH.mkName "_a")) |] RList1 (Rule inner _ _) -> [TH.clause [pat] bdy []] where pat = TH.conP (TH.mkName n) [TH.tupP [ dynP "_x1", dynP "_xs" ]] bdy = TH.normalB [| $lft `mappend` (join (fmap $(prodFn inner) $(dyn "_xs"))) |] where lft = [| $(prodFn inner) $(dyn "_x1") |] RWrap (Rule inner _ _) -> [TH.clause [pat] bdy []] where pat = TH.conP (TH.mkName n) [dynP "_x"] bdy = TH.normalB [| $(prodFn inner) $(dyn "_x") |] RRecord sq -> [TH.clause [pat] (TH.normalB bdy) []] where pat = TH.conP (TH.mkName n) . fmap mkPat . fmap fst . addIndices $ sq where mkPat idx = dynP ("_x'" ++ show idx) bdy = foldr addField [| Seq.empty |] . addIndices $ sq where addField (idx, (Rule nm _ _)) acc = [| $this `mappend` $acc |] where this = [| $(prodFn nm) $(dyn ("_x'" ++ show idx)) |] RUnion (r1, rs) -> mkClause r1 : fmap mkClause (toList rs) where mkClause (Rule inner _ _) = TH.clause [pat] bdy [] where pat = TH.conP (TH.mkName (unionBranchName n inner)) [dynP "_x"] bdy = TH.normalB [| $(prodFn inner) $(dyn "_x") |] branchToClause :: Branch t -> TH.ClauseQ branchToClause (Branch n rs) = TH.clause [pat] bdy [] where pat = TH.conP (TH.mkName n) fields where fields = fmap mkField . fmap fst . addIndices $ rs where mkField idx = TH.varP (TH.mkName ("_x'" ++ show idx)) bdy = TH.normalB [| join $sq |] where sq = foldr addField (TH.varE 'Seq.empty) . addIndices $ rs where addField (idx, (Rule inner _ _)) acc = [| $newTerm <| $acc |] where newTerm = [| $(prodFn inner) $(TH.varE (TH.mkName ("_x'" ++ show idx))) |] -- | Many functions take an argument that holds the name qualifier -- for the module that contains the data types created by applying -- 'ruleTreeToTypes' or 'allRulesToTypes' to the 'Pinchot.' -- -- You have to make sure that the data types you created with -- 'ruleTreeToTypes', 'allRulesToTypes', or 'allRulesRecord' are in -- scope. The spliced Template Haskell code has to know where to -- look for these data types. If you did an unqualified @import@ or -- if the types are in the same module as is the splice of -- 'earleyParser', just pass the empty string here. If you did a -- qualified import, use the appropriate qualifier here. -- -- For example, if you used @import qualified MyAst@, pass -- @\"MyAst\"@ here. If you used @import qualified -- Data.MyLibrary.MyAst as MyLibrary.MyAst@, pass -- @\"MyLibrary.MyAst\"@ here. -- -- I recommend that you always create a new module and that all you -- do in that module is apply 'ruleTreeToTypes' or -- 'allRulesToTypes', and that you then perform an @import -- qualified@ to bring those names into scope in the module in which -- you use a function that takes a 'Qualifier' argument. This -- avoids unlikely, but possible, issues that could otherwise arise -- due to naming conflicts. type Qualifier = String -- | Creates a record data type that holds a value of type -- -- @'Text.Earley.Prod' r 'String' t a@ -- -- for every rule created in the 'Pinchot'. @r@ is left -- universally quantified, @t@ is the token type (typically 'Char') -- and @a@ is the type of the rule. -- -- This always creates a single product type whose name is -- @Productions@; currently the name cannot be configured. -- -- For an example of the use of 'allRulesRecord', please see -- "Pinchot.Examples.AllRulesRecord". allRulesRecord :: Qualifier -- ^ Qualifier for data types created with 'ruleTreeToTypes' or -- 'allRulesToTypes' -> Name -- ^ Name of terminal type. Typically you will get this through -- the Template Haskell quoting mechanism, such as @''Char@. -> Pinchot t a -- ^ A record is created that holds a value for each 'Rule' -- created in the 'Pinchot'; the return value of the 'Pinchot' is -- ignored. -> DecsQ -- ^ When spliced, this will create a single declaration that is a -- record with the name @Productions@. It will have one type variable, -- @r@. Each record in the declaration will have a name like so: -- -- @a'NAME@ -- -- where @NAME@ is the name of the type. Don't count on these -- records being in any particular order. allRulesRecord prefix termName pinc = sequence [TH.dataD (return []) (TH.mkName nameStr) tys [con] []] where nameStr = "Productions" tys = [TH.PlainTV (TH.mkName "r")] con = do names <- fmap fst $ goPinchot pinc TH.recC (TH.mkName nameStr) (fmap (mkRecord . snd) . M.assocs . allRules $ names) mkRecord (Rule ruleNm _ _) = TH.varStrictType recName st where recName = TH.mkName ("a'" ++ ruleNm) st = TH.strictType TH.notStrict ty where ty = (TH.conT ''Text.Earley.Prod) `TH.appT` (TH.varT (TH.mkName "r")) `TH.appT` (TH.conT ''String) `TH.appT` (TH.conT termName) `TH.appT` (TH.conT (TH.mkName nameWithPrefix)) nameWithPrefix = case prefix of [] -> ruleNm _ -> prefix ++ '.' : ruleNm -- | Creates a 'Text.Earley.Grammar' that contains a -- 'Text.Earley.Prod' for every 'Rule' created in the 'Pinchot'. earleyProduct :: Syntax.Lift t => Qualifier -- ^ Qualifier for data types created with 'ruleTreeToTypes' or -- 'allRulesToTypes' -> Qualifier -- ^ Module prefix for the type created with 'allRulesRecord' -> Pinchot t a -- ^ Creates an Earley grammar that contains a 'Text.Earley.Prod' -- for each 'Rule' in the 'Pinchot'. The return value from the -- 'Pinchot' is ignored. -> ExpQ -- ^ When spliced, 'earleyProduct' creates an expression whose -- type is @'Text.Earley.Grammar' r (Productions r)@, where -- @Productions@ is -- the type created by 'allRulesRecord'. earleyProduct pfxRule pfxRec pinc = do names <- fmap fst $ goPinchot pinc let binds = concatMap (ruleToParser pfxRule) . fmap snd . M.assocs . allRules $ names final = [| return $(TH.recConE (TH.mkName rulesRecName) (recs names)) |] recursiveDo binds final where rulesRecName | null pfxRec = "Productions" | otherwise = pfxRec ++ ".Productions" recs = fmap mkRec . fmap snd . M.assocs . allRules where mkRec (Rule n _ _) = return (TH.mkName recName, recVal) where recName | null pfxRec = "a'" ++ n | otherwise = pfxRec ++ ".a'" ++ n recVal = TH.VarE . ruleName $ n