Skip to content

Commit

Permalink
Add disambiguation example
Browse files Browse the repository at this point in the history
  • Loading branch information
expipiplus1 committed May 13, 2023
1 parent ec21f6f commit 63e5523
Show file tree
Hide file tree
Showing 2 changed files with 88 additions and 0 deletions.
9 changes: 9 additions & 0 deletions Earley.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -117,6 +117,15 @@ executable earley-infinite
default-language: Haskell2010
build-depends: base, Earley

executable earley-disambiguate
if !flag(examples)
buildable: False
main-is: Disambiguate.hs
ghc-options: -Wall
hs-source-dirs: examples
default-language: Haskell2010
build-depends: base, Earley, containers

benchmark bench
type: exitcode-stdio-1.0
hs-source-dirs: . bench
Expand Down
79 changes: 79 additions & 0 deletions examples/Disambiguate.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
{-# LANGUAGE LambdaCase #-}
{-# LANGUAGE PatternSynonyms #-}
{-# LANGUAGE RecursiveDo #-}

import Control.Applicative
import Data.Char
import Data.Foldable (traverse_)
import Data.Tree
import System.Environment
import Text.Earley
import Text.Earley.Grammar

type Expr = Tree String
pattern Add, Mul :: Tree String -> Tree String -> Tree String
pattern Add x y = Node "+" [x, y]
pattern Mul x y = Node "*" [x, y]
pattern Var :: a -> Tree a
pattern Var n = Node n []
pattern Amb :: [Tree String] -> Tree String
pattern Amb xs = Node "Ambiguous" xs

expr :: Grammar r (Prod r String String Expr)
expr = mdo
let exprProd = disambiguate $ \case
[x] -> x
xs -> Amb xs
e <-
exprProd $
Add <$> e <* namedToken "+" <*> e
<|> Mul <$> e <* namedToken "*" <*> e
<|> Var <$> satisfy ident
<|> namedToken "(" *> e <* namedToken "("
return e
where
ident (x : _) = isAlpha x
ident _ = False

-- λ> :main "A + B * C * G"
-- Ambiguous
--
-- +- ((A+B)*(C*G))
--
-- +- *
-- │ │
-- │ +- Ambiguous
-- │ │ │
-- │ │ +- ((A+B)*C)
-- │ │ │
-- │ │ `- (A+(B*C))
-- │ │
-- │ `- G
--
-- `- +
--
-- +- A
--
-- `- Ambiguous
--
-- +- (B*(C*G))
--
-- `- ((B*C)*G)
main :: IO ()
main = do
x : _ <- getArgs
let (ps, r) = fullParses (parser expr) (words x)
traverse_ (putStrLn . drawTree . simplifyTree) ps
print r

-- | render non-ambiguous expressions on one line to make the printed tree
-- smaller
simplifyTree :: Expr -> Expr
simplifyTree =
foldTree
( \op -> \case
[Node n [], Node m []] | op /= "Ambiguous" -> Node (parens (n <> op <> m)) []
ns -> Node op ns
)
where
parens x = "(" <> x <> ")"

0 comments on commit 63e5523

Please sign in to comment.