module Language.Parser where import Language.Ast import Language.Error (errorInMappy) import qualified Data.Map.Strict as M import Text.ParserCombinators.Parsec import Data.Maybe (catMaybes) parseFile :: String -> Either ParseError [Definition] parseFile = parse file "Error parsing file" defOrExpr :: Parser (Maybe (Either Definition Expression)) defOrExpr = let validRepl cons p = Just . cons <$> try (fullTerm p) fullTerm p = whiteSpace *> p <* whiteSpace <* eof in validRepl Left definition <|> validRepl Right expression <|> whiteSpace *> eof *> pure Nothing file :: Parser [Definition] file = let validTopLevel = choice [lineComment *> pure Nothing, Just <$> definition] end = eof <|> lineComment fileContents = catMaybes <$> many ( validTopLevel <* whiteSpace) in whiteSpace *> fileContents <* end lineComment :: Parser () lineComment = string "--" *> manyTill anyChar (newline *> pure () <|> eof) *> pure () <?> "line comment" expression :: Parser Expression expression = specialForm <|> map' <|> application <|> lambda <|> keyword <|> namedValue definition :: Parser Definition definition = (do name <- namedValue whiteSpace valueDefinition name <|> functionDefinition name) <?> "definition" valueDefinition :: Expression -> Parser Definition valueDefinition name = MappyDef name <$> (char '=' *> whiteSpace *> expression) functionDefinition :: Expression -> Parser Definition functionDefinition name = do names <- namesEndingWith $ char '=' whiteSpace expr <- expression return $ DefSugar $ SugaredFnDefinition name names expr specialForm :: Parser Expression specialForm = letExpression <|> list <|> character <|> string' character :: Parser Expression character = ExprSugar . SugaredChar <$> (char '\'' *> characterInternal <* char '\'') <?> "character" string' :: Parser Expression string' = ExprSugar . SugaredString <$> (char '"' *> manyTill characterInternal (char '"')) <?> "string" characterInternal :: Parser Char characterInternal = escapedChar <|> anyChar where escapedChar = do b <- char '\\' c <- anyChar return $ read $ '\'':b:c:"'" list :: Parser Expression list = ExprSugar . SugaredList <$> between (try $ string "(|" <* whiteSpace) (string "|)") (expression `sepEndBy` whiteSpace) <?> "list" letExpression :: Parser Expression letExpression = (do _ <- try $ string "let" <* whiteSpace firstDef <- definition <* whiteSpace restDefs <- manyTill (definition <* whiteSpace) $ string "in" whiteSpace expr <- expression return $ ExprSugar $ SugaredLet (firstDef:restDefs) expr) <?> "let expression" lazyArgument :: Parser Expression lazyArgument = (fmap MappyLazyArgument $ char '(' *> whiteSpace *> identifier <* whiteSpace <* char ')') <?> "lazy argument" lambda :: Parser Expression lambda = lambda' <?> "lambda" where lambda' = do _ <- char '\\' whiteSpace names <- namesEndingWith $ string "->" whiteSpace expr <- expression return $ MappyLambda names expr namesEndingWith :: Parser a -> Parser [Expression] namesEndingWith = manyTill ((namedValue <|> lazyArgument) <* whiteSpace) pairs :: Parser (M.Map Expression Expression) pairs = do whiteSpace ps <- expression `sepEndBy` whiteSpace if even $ length ps then pure $ toMap ps else unexpected "odd number of values in literal map" where toMap [] = M.empty toMap [_] = errorInMappy "Impossible, odd valued map escaped guards." toMap (k:v:rest) = M.insert k v $ toMap rest map' :: Parser Expression map' = MappyMap <$> StandardMap <$> between (char '(') (char ')') pairs <?> "map" application :: Parser Expression application = (between (char '[') (char ']') $ do whiteSpace fn <- namedValue <|> application <|> keyword <|> lambda whiteSpace args <- expression `sepEndBy` whiteSpace return $ MappyApp fn args) <?> "application" identifier :: Parser String identifier = many1 $ letter <|> digit <|> oneOf "_/-+<>!@#$%^&*;.?=" keyword :: Parser Expression keyword = char ':' *> (MappyKeyword <$> identifier) <?> "keyword" namedValue :: Parser Expression namedValue = MappyNamedValue <$> identifier <?> "name" whiteSpace :: Parser () whiteSpace = many (oneOf " \n\r\t,") *> pure () <?> "whitespace"