9 Commits

Author SHA1 Message Date
42fce0ae43 Drop unreachable cases of updateDepth
All checks were successful
Test and Build / test (push) Successful in 2m27s
Test and Build / build (push) Successful in 1m39s
2025-01-21 16:16:04 -06:00
51b1eb070f Add more explicit error handling for mismatched groupings 2025-01-21 16:06:10 -06:00
c2e5a8985a Inline pattern matching in Parser 2025-01-21 14:21:47 -06:00
9d7e4daa41 CI/CD for tests and builds (broken caching)
All checks were successful
Test and Build / test (push) Successful in 2m35s
Test and Build / build (push) Successful in 1m39s
2025-01-21 13:29:52 -06:00
edde0a80c9 Actually readable Level Order Traversal 2025-01-20 20:10:14 -06:00
35163a5d54 Allow multiline expressions 2025-01-20 19:20:29 -06:00
ca7f09e2ac Eliminate redundant eager calls of elimLambda 2025-01-20 16:05:06 -06:00
82e29440b0 Reduce duplication of elimLambda calls 2025-01-20 15:16:27 -06:00
ad02c8b86a General refactor for legibility
Priming to update all source to lhs and document extensively
2025-01-19 14:41:25 -06:00
12 changed files with 585 additions and 431 deletions

View File

@ -0,0 +1,86 @@
name: Test and Build
on:
push:
branches:
- main
pull_request:
types:
- opened
- synchronize
jobs:
test:
container:
image: docker.matri.cx/nix-runner:latest
credentials:
username: ${{ secrets.REGISTRY_USERNAME }}
password: ${{ secrets.REGISTRY_PASSWORD }}
steps:
- name: Checkout code
uses: actions/checkout@v3
- name: Set up cache for Cabal
uses: actions/cache@v4
with:
path: |
~/.cabal
~/.ghc
key: cabal-${{ runner.os }}-${{ hashFiles('tricu.cabal') }}
restore-keys: |
cabal-${{ runner.os }}-
- name: Set up cache for Nix
uses: actions/cache@v4
with:
path: |
/nix/store
/nix/var/nix/cache
key: nix-${{ runner.os }}-${{ hashFiles('flake.lock') }}
restore-keys: |
nix-${{ runner.os }}-
- name: Initialize Cabal and update package list
run: |
nix develop --command cabal update
- name: Install dependencies and run tests
run: |
nix develop --command cabal test
build:
needs: test
container:
image: docker.matri.cx/nix-runner:latest
credentials:
username: ${{ secrets.REGISTRY_USERNAME }}
password: ${{ secrets.REGISTRY_PASSWORD }}
steps:
- name: Checkout code
uses: actions/checkout@v3
- name: Set up cache for Cabal
uses: actions/cache@v4
with:
path: |
~/.cabal
~/.ghc
key: cabal-${{ runner.os }}-${{ hashFiles('tricu.cabal') }}
restore-keys: |
cabal-${{ runner.os }}-
- name: Set up cache for Nix
uses: actions/cache@v4
with:
path: |
/nix/store
/nix/var/nix/cache
key: nix-${{ runner.os }}-${{ hashFiles('flake.lock') }}
restore-keys: |
nix-${{ runner.os }}-
- name: Build binary
run: |
nix build
ls -alh ./result/bin/tricu

View File

@ -2,7 +2,7 @@
## Introduction ## Introduction
tricu (pronounced like "tree-shoe" in English) is a purely functional interpreted language implemented in Haskell. [I'm](https://eversole.co) developing tricu to further research the possibilities offered by the various forms of [Tree Calculi](https://github.com/barry-jay-personal/typed_tree_calculus/blob/main/typed_program_analysis.pdf). tricu (pronounced "tree-shoe") is a purely functional interpreted language implemented in Haskell. [I'm](https://eversole.co) developing tricu to further research the possibilities offered by the various forms of [Tree Calculi](https://github.com/barry-jay-personal/typed_tree_calculus/blob/main/typed_program_analysis.pdf).
tricu offers minimal syntax sugar yet manages to provide a complete, intuitive, and familiar programming environment. There is great power in simplicity. tricu offers: tricu offers minimal syntax sugar yet manages to provide a complete, intuitive, and familiar programming environment. There is great power in simplicity. tricu offers:

View File

@ -17,18 +17,52 @@
-- 4 5 6 -- 4 5 6
-- --
isLeaf = (\node : lOr (emptyList node) (emptyList (tail node))) isLeaf = (\node :
getLabel = (\node : head node) lOr
getLeft = (\node : if (emptyList node) [] (if (emptyList (tail node)) [] (head (tail node)))) (emptyList node)
getRight = (\node : if (emptyList node) [] (if (emptyList (tail node)) [] (if (emptyList (tail (tail node))) [] (head (tail (tail node)))))) (emptyList (tail node)))
getLabel = (\node : head node)
getLeft = (\node : if (emptyList node)
[]
(if (emptyList (tail node))
[]
(head (tail node))))
getRight = (\node : if (emptyList node)
[]
(if (emptyList (tail node))
[]
(if (emptyList (tail (tail node)))
[]
(head (tail (tail node))))))
processLevel = y (\self queue : if (emptyList queue)
[]
(pair (map getLabel queue) (self (filter
(\node : not (emptyList node))
(lconcat (map getLeft queue) (map getRight queue))))))
processLevel = y (\self queue : if (emptyList queue) [] (pair (map getLabel queue) (self (filter (\node : not (emptyList node)) (lconcat (map getLeft queue) (map getRight queue))))))
levelOrderTraversal = (\a : processLevel (t a t)) levelOrderTraversal = (\a : processLevel (t a t))
toLineString = y (\self levels : if (emptyList levels) "" (lconcat (lconcat (map (\x : lconcat x " ") (head levels)) "") (if (emptyList (tail levels)) "" (lconcat (t (t 10 t) t) (self (tail levels))))))
toLineString = y (\self levels : if (emptyList levels)
""
(lconcat
(lconcat (map (\x : lconcat x " ") (head levels)) "")
(if (emptyList (tail levels)) "" (lconcat (t (t 10 t) t) (self (tail levels))))))
levelOrderToString = (\s : toLineString (levelOrderTraversal s)) levelOrderToString = (\s : toLineString (levelOrderTraversal s))
flatten = foldl (\acc x : lconcat acc x) "" flatten = foldl (\acc x : lconcat acc x) ""
flatLOT = (\s : lconcat (t 10 t) (flatten (levelOrderToString s))) flatLOT = (\s : lconcat (t 10 t) (flatten (levelOrderToString s)))
exampleOne = flatLOT [("1") [("2") [("4") t t] t] [("3") [("5") t t] [("6") t t]]]] exampleOne = flatLOT [("1")
exampleTwo = flatLOT [("1") [("2") [("4") [("8") t t] [("9") t t]] [("6") [("10") t t] [("12") t t]]] [("3") [("5") [("11") t t] t] [("7") t t]]] [("2") [("4") t t] t]
[("3") [("5") t t] [("6") t t]]]
exampleTwo = flatLOT [("1")
[("2") [("4") [("8") t t] [("9") t t]] [("6") [("10") t t] [("12") t t]]]
[("3") [("5") [("11") t t] t] [("7") t t]]]
exampleTwo

View File

@ -17,25 +17,77 @@ yD = yi iE
id = (\a : a) id = (\a : a)
triage = (\a b c : t (t a b) c) triage = (\a b c : t (t a b) c)
pair = t pair = t
matchBool = (\ot of : triage of (\_ : ot) (\_ _ : ot))
matchList = (\oe oc : triage oe _ oc) matchBool = (\ot of : triage
matchPair = (\op : triage _ _ op) of
(\_ : ot)
(\_ _ : ot)
)
matchList = (\oe oc : triage
oe
_
oc
)
matchPair = (\op : triage
_
_
op
)
not = matchBool false true not = matchBool false true
and = matchBool id (\z : false) and = matchBool id (\z : false)
if = (\cond then else : t (t else (t t then)) t cond) if = (\cond then else : t (t else (t t then)) t cond)
test = triage "Leaf" (\z : "Stem") (\a b : "Fork") test = triage "Leaf" (\z : "Stem") (\a b : "Fork")
emptyList = matchList true (\y z : false) emptyList = matchList true (\y z : false)
head = matchList t (\hd tl : hd) head = matchList t (\hd tl : hd)
tail = matchList t (\hd tl : tl) tail = matchList t (\hd tl : tl)
lconcat = y (\self : matchList (\k : k) (\h r k : pair h (self r k)))
lAnd = triage (\x : false) (\_ x : x) (\_ _ x : x) lconcat = y (\self : matchList
lOr = triage (\x : x) (\_ _ : true) (\_ _ x : true) (\k : k)
hmap = y (\self : matchList (\f : t) (\hd tl f : pair (f hd) (self tl f))) (\h r k : pair h (self r k)))
lAnd = (triage
(\x : false)
(\_ x : x)
(\_ _ x : x)
)
lOr = (triage
(\x : x)
(\_ _ : true)
(\_ _ x : true)
)
hmap = y (\self :
matchList
(\f : t)
(\hd tl f : pair
(f hd)
(self tl f)))
map = (\f l : hmap l f) map = (\f l : hmap l f)
equal = y (\self : triage (triage true (\z : false) (\y z : false)) (\ax : triage false (self ax) (\y z : false)) (\ax ay : triage false (\z : false) (\bx by : lAnd (self ax bx) (self ay by))))
equal = y (\self : triage
(triage
true
(\z : false)
(\y z : false))
(\ax : triage
false
(self ax)
(\y z : false))
(\ax ay : triage
false
(\z : false)
(\bx by : lAnd (self ax bx) (self ay by))))
hfilter = y (\self : matchList (\f : t) (\hd tl f : matchBool (t hd) i (f hd) (self tl f))) hfilter = y (\self : matchList (\f : t) (\hd tl f : matchBool (t hd) i (f hd) (self tl f)))
filter = (\f l : hfilter l f) filter = (\f l : hfilter l f)
hfoldl = y (\self f l x : matchList (\acc : acc) (\hd tl acc : self f tl (f acc hd)) l x)
foldl = (\f x l : hfoldl f l x) hfoldl = y (\self f l x : matchList (\acc : acc) (\hd tl acc : self f tl (f acc hd)) l x)
hfoldr = y (\self x f l : matchList x (\hd tl : f (self x f tl) hd) l) foldl = (\f x l : hfoldl f l x)
foldr = (\f x l : hfoldr x f l)
hfoldr = y (\self x f l : matchList x (\hd tl : f (self x f tl) hd) l)
foldr = (\f x l : hfoldr x f l)

View File

@ -8,110 +8,85 @@ import Data.Map (Map)
import qualified Data.Map as Map import qualified Data.Map as Map
import qualified Data.Set as Set import qualified Data.Set as Set
evalSingle :: Map String T -> TricuAST -> Map String T evalSingle :: Env -> TricuAST -> Env
evalSingle env term = case term of evalSingle env term
SFunc name [] body -> | SFunc name [] body <- term =
let lineNoLambda = eliminateLambda body let res = evalAST env body
result = evalAST env lineNoLambda in Map.insert "__result" res (Map.insert name res env)
in Map.insert "__result" result (Map.insert name result env) | SApp func arg <- term =
SLambda _ body -> let res = apply (evalAST env func) (evalAST env arg)
let result = evalAST env body in Map.insert "__result" res env
in Map.insert "__result" result env | SVar name <- term =
SApp func arg -> case Map.lookup name env of
let result = apply (evalAST env $ eliminateLambda func) (evalAST env $ eliminateLambda arg) Just v -> Map.insert "__result" v env
in Map.insert "__result" result env Nothing -> errorWithoutStackTrace $ "Variable " ++ name ++ " not defined"
SVar name -> | otherwise =
case Map.lookup name env of Map.insert "__result" (evalAST env term) env
Just value -> Map.insert "__result" value env
Nothing -> errorWithoutStackTrace $ "Variable " ++ name ++ " not defined"
_ ->
let result = evalAST env term
in Map.insert "__result" result env
evalTricu :: Map String T -> [TricuAST] -> Map String T evalTricu :: Env -> [TricuAST] -> Env
evalTricu env list = evalTricu' env (filter (/= SEmpty) list) evalTricu env [] = env
where evalTricu env [x] =
evalTricu' :: Map String T -> [TricuAST] -> Map String T let updatedEnv = evalSingle env x
evalTricu' env [] = env in Map.insert "__result" (result updatedEnv) updatedEnv
evalTricu' env [lastLine] = evalTricu env (x:xs) =
let lastLineNoLambda = eliminateLambda lastLine evalTricu (evalSingle env x) xs
updatedEnv = evalSingle env lastLineNoLambda
in Map.insert "__result" (result updatedEnv) updatedEnv
evalTricu' env (line:rest) =
let lineNoLambda = eliminateLambda line
updatedEnv = evalSingle env lineNoLambda
in evalTricu updatedEnv rest
evalAST :: Map String T -> TricuAST -> T evalAST :: Env -> TricuAST -> T
evalAST env term = case term of evalAST env term
SVar name -> case Map.lookup name env of | SLambda _ _ <- term = evalAST env (elimLambda term)
Just value -> value | SVar name <- term = evalVar name
Nothing -> errorWithoutStackTrace $ "Variable " ++ name ++ " not defined" | TLeaf <- term = Leaf
TLeaf -> Leaf | TStem t <- term = Stem (evalAST env t)
TStem t -> Stem (evalAST env t) | TFork t u <- term = Fork (evalAST env t) (evalAST env u)
TFork t1 t2 -> Fork (evalAST env t1) (evalAST env t2) | SApp t u <- term = apply (evalAST env t) (evalAST env u)
SApp t1 t2 -> apply (evalAST env t1) (evalAST env t2) | SStr s <- term = ofString s
SStr str -> ofString str | SInt n <- term = ofNumber n
SInt num -> ofNumber num | SList xs <- term = ofList (map (evalAST env) xs)
SList elems -> ofList (map (evalAST env) elems) | SEmpty <- term = Leaf
SEmpty -> Leaf | otherwise = errorWithoutStackTrace "Unexpected AST term"
SFunc name args body -> where
errorWithoutStackTrace $ "Unexpected function definition " ++ name evalVar name = Map.findWithDefault
SLambda {} -> errorWithoutStackTrace "Internal error: SLambda found in evalAST after elimination." (errorWithoutStackTrace $ "Variable " ++ name ++ " not defined")
name env
eliminateLambda :: TricuAST -> TricuAST
eliminateLambda (SLambda (v:vs) body)
| null vs = lambdaToT v (eliminateLambda body)
| otherwise = eliminateLambda (SLambda [v] (SLambda vs body))
eliminateLambda (SApp f arg) = SApp (eliminateLambda f) (eliminateLambda arg)
eliminateLambda (TStem t) = TStem (eliminateLambda t)
eliminateLambda (TFork l r) = TFork (eliminateLambda l) (eliminateLambda r)
eliminateLambda (SList xs) = SList (map eliminateLambda xs)
eliminateLambda other = other
-- https://github.com/barry-jay-personal/typed_tree_calculus/blob/main/typed_program_analysis.pdf -- https://github.com/barry-jay-personal/typed_tree_calculus/blob/main/typed_program_analysis.pdf
-- Chapter 4: Lambda-Abstraction -- Chapter 4: Lambda-Abstraction
lambdaToT :: String -> TricuAST -> TricuAST elimLambda :: TricuAST -> TricuAST
lambdaToT x (SVar y) elimLambda = go
| x == y = tI where
lambdaToT x (SVar y) go (SLambda (v:vs) body)
| x /= y = SApp tK (SVar y) | null vs = toSKI v (elimLambda body)
lambdaToT x t | otherwise = elimLambda (SLambda [v] (SLambda vs body))
| not (isFree x t) = SApp tK t go (SApp f g) = SApp (elimLambda f) (elimLambda g)
lambdaToT x (SApp n u) go x = x
| not (isFree x (SApp n u)) = SApp tK (SApp (eliminateLambda n) (eliminateLambda u))
lambdaToT x (SApp n u) = SApp (SApp tS (lambdaToT x (eliminateLambda n))) (lambdaToT x (eliminateLambda u))
lambdaToT x body
| not (isFree x body) = SApp tK body
| otherwise = SApp (SApp tS (lambdaToT x body)) TLeaf
freeVars :: TricuAST -> Set.Set String toSKI x (SVar y)
freeVars (SVar v) = Set.singleton v | x == y = _I
freeVars (SInt _) = Set.empty | otherwise = SApp _K (SVar y)
freeVars (SStr _) = Set.empty toSKI x t@(SApp n u)
freeVars (SList xs) = foldMap freeVars xs | not (isFree x t) = SApp _K t
freeVars (SApp f arg) = freeVars f <> freeVars arg | otherwise = SApp (SApp _S (toSKI x n)) (toSKI x u)
freeVars TLeaf = Set.empty toSKI x t
freeVars (SFunc _ _ b) = freeVars b | not (isFree x t) = SApp _K t
freeVars (TStem t) = freeVars t | otherwise = SApp (SApp _S (toSKI x t)) TLeaf
freeVars (TFork l r) = freeVars l <> freeVars r
freeVars (SLambda vs b) = foldr Set.delete (freeVars b) vs
isFree :: String -> TricuAST -> Bool _S = parseSingle "t (t (t t t)) t"
isFree x = Set.member x . freeVars _K = parseSingle "t t"
_I = parseSingle "t (t (t t)) t"
-- We need the SKI operators in an unevaluated TricuAST tree form so that we isFree x = Set.member x . freeVars
-- can keep the evaluation functions straightforward freeVars (SVar v ) = Set.singleton v
tI :: TricuAST freeVars (SInt _ ) = Set.empty
tI = SApp (SApp TLeaf (SApp TLeaf (SApp TLeaf TLeaf))) TLeaf freeVars (SStr _ ) = Set.empty
freeVars (SList s ) = foldMap freeVars s
freeVars (SApp f a ) = freeVars f <> freeVars a
freeVars (TLeaf ) = Set.empty
freeVars (SFunc _ _ b) = freeVars b
freeVars (TStem t ) = freeVars t
freeVars (TFork l r ) = freeVars l <> freeVars r
freeVars (SLambda v b ) = foldr Set.delete (freeVars b) v
tK :: TricuAST result :: Env -> T
tK = SApp TLeaf TLeaf
tS :: TricuAST
tS = SApp (SApp TLeaf (SApp TLeaf (SApp (SApp TLeaf TLeaf) TLeaf))) TLeaf
result :: Map String T -> T
result r = case Map.lookup "__result" r of result r = case Map.lookup "__result" r of
Just a -> a Just a -> a
Nothing -> errorWithoutStackTrace "No __result field found in provided environment" Nothing -> errorWithoutStackTrace "No __result field found in provided environment"

View File

@ -61,7 +61,10 @@ lnewline :: Lexer LToken
lnewline = char '\n' *> pure LNewline lnewline = char '\n' *> pure LNewline
sc :: Lexer () sc :: Lexer ()
sc = space space1 (skipLineComment "--") (skipBlockComment "|-" "-|") sc = space
(void $ takeWhile1P (Just "space") (\c -> c == ' ' || c == '\t'))
(skipLineComment "--")
(skipBlockComment "|-" "-|")
tricuLexer :: Lexer [LToken] tricuLexer :: Lexer [LToken]
tricuLexer = do tricuLexer = do
@ -75,7 +78,8 @@ tricuLexer = do
pure tokens pure tokens
where where
tricuLexer' = tricuLexer' =
[ try identifier [ try lnewline
, try identifier
, try keywordT , try keywordT
, try integerLiteral , try integerLiteral
, try stringLiteral , try stringLiteral

View File

@ -81,4 +81,7 @@ main = do
putStrLn $ decodeResult $ result $ evalTricu library $ parseTricu value putStrLn $ decodeResult $ result $ evalTricu library $ parseTricu value
runTricu :: String -> T runTricu :: String -> T
runTricu = result . evalTricu Map.empty . parseTricu runTricu input =
let asts = parseTricu input
finalEnv = evalTricu Map.empty asts
in result finalEnv

View File

@ -1,277 +1,304 @@
module Parser where module Parser where
import Lexer import Lexer
import Research hiding (toList) import Research
import Data.List.NonEmpty (toList) import Control.Monad (void)
import Control.Monad.State
import Data.List.NonEmpty (toList)
import Data.Void (Void) import Data.Void (Void)
import Text.Megaparsec import Text.Megaparsec
import Text.Megaparsec.Char
import Text.Megaparsec.Error (ParseErrorBundle, errorBundlePretty) import Text.Megaparsec.Error (ParseErrorBundle, errorBundlePretty)
import qualified Data.Set as Set import qualified Data.Set as Set
type Parser = Parsec Void [LToken] data PState = PState
type AltParser = Parsec Void String { parenDepth :: Int
, bracketDepth :: Int
} deriving (Show)
type ParserM = StateT PState (Parsec Void [LToken])
satisfyM :: (LToken -> Bool) -> ParserM LToken
satisfyM f = do
token <- lift (satisfy f)
modify' (updateDepth token)
return token
updateDepth :: LToken -> PState -> PState
updateDepth LOpenParen st = st { parenDepth = parenDepth st + 1 }
updateDepth LOpenBracket st = st { bracketDepth = bracketDepth st + 1 }
updateDepth LCloseParen st = st { parenDepth = parenDepth st - 1 }
updateDepth LCloseBracket st = st { bracketDepth = bracketDepth st - 1 }
updateDepth _ st = st
topLevelNewline :: ParserM ()
topLevelNewline = do
st <- get
if parenDepth st == 0 && bracketDepth st == 0
then void (satisfyM (== LNewline))
else fail "Top-level exit in nested context (paren or bracket)"
parseProgram :: [LToken] -> Either (ParseErrorBundle [LToken] Void) [TricuAST]
parseProgram tokens =
runParser (evalStateT (parseProgramM <* finalizeDepth <* eof) (PState 0 0)) "" tokens
parseSingleExpr :: [LToken] -> Either (ParseErrorBundle [LToken] Void) TricuAST
parseSingleExpr tokens =
runParser (evalStateT (scnParserM *> parseExpressionM <* finalizeDepth <* eof) (PState 0 0)) "" tokens
finalizeDepth :: ParserM ()
finalizeDepth = do
st <- get
case (parenDepth st, bracketDepth st) of
(0, 0) -> pure ()
(p, b) -> fail $ "Unmatched tokens: " ++ show (p, b)
parseTricu :: String -> [TricuAST] parseTricu :: String -> [TricuAST]
parseTricu input parseTricu input =
| null tokens = [] case lexTricu input of
| otherwise = map parseSingle tokens [] -> []
where toks ->
tokens = case lexTricu input of case parseProgram toks of
[] -> [] Left err -> errorWithoutStackTrace (handleParseError err)
tokens -> lines input Right asts -> asts
parseSingle :: String -> TricuAST parseSingle :: String -> TricuAST
parseSingle input = case lexTricu input of parseSingle input =
[] -> SEmpty case lexTricu input of
tokens -> case runParser parseExpression "" tokens of [] -> SEmpty
Left err -> error $ handleParseError err toks ->
Right ast -> ast case parseSingleExpr toks of
Left err -> errorWithoutStackTrace (handleParseError err)
Right ast -> ast
parseExpression :: Parser TricuAST parseProgramM :: ParserM [TricuAST]
parseExpression = choice parseProgramM = do
[ try parseFunction skipMany topLevelNewline
, try parseLambda exprs <- sepEndBy parseOneExpression (some topLevelNewline)
, try parseLambdaExpression skipMany topLevelNewline
, try parseListLiteral return exprs
, try parseApplication
, try parseTreeTerm parseOneExpression :: ParserM TricuAST
, parseLiteral parseOneExpression = scnParserM *> parseExpressionM
scnParserM :: ParserM ()
scnParserM = skipMany $ do
t <- lookAhead anySingle
st <- get
if | (parenDepth st > 0 || bracketDepth st > 0) && case t of
LNewline -> True
_ -> False -> void $ satisfyM $ \case
LNewline -> True
_ -> False
| otherwise -> fail "In nested context or no space token" <|> empty
eofM :: ParserM ()
eofM = lift eof
parseExpressionM :: ParserM TricuAST
parseExpressionM = choice
[ try parseFunctionM
, try parseLambdaM
, try parseLambdaExpressionM
, try parseListLiteralM
, try parseApplicationM
, try parseTreeTermM
, parseLiteralM
] ]
scnParser :: Parser () parseFunctionM :: ParserM TricuAST
scnParser = skipMany (satisfy isNewline) parseFunctionM = do
LIdentifier name <- satisfyM $ \case
LIdentifier _ -> True
_ -> False
args <- many $ satisfyM $ \case
LIdentifier _ -> True
_ -> False
_ <- satisfyM (== LAssign)
scnParserM
body <- parseExpressionM
pure (SFunc name (map getIdentifier args) body)
parseFunction :: Parser TricuAST parseLambdaM :: ParserM TricuAST
parseFunction = do parseLambdaM =
LIdentifier name <- satisfy isIdentifier between (satisfyM (== LOpenParen)) (satisfyM (== LCloseParen)) $ do
args <- many (satisfy isIdentifier) _ <- satisfyM (== LBackslash)
satisfy (== LAssign) param <- satisfyM $ \case
body <- parseExpression LIdentifier _ -> True
return (SFunc name (map getIdentifier args) body) _ -> False
rest <- many $ satisfyM $ \case
LIdentifier _ -> True
_ -> False
_ <- satisfyM (== LColon)
scnParserM
body <- parseLambdaExpressionM
let nested = foldr (\v acc -> SLambda [getIdentifier v] acc) body rest
pure (SLambda [getIdentifier param] nested)
parseAtomicBase :: Parser TricuAST parseLambdaExpressionM :: ParserM TricuAST
parseAtomicBase = choice parseLambdaExpressionM = choice
[ parseTreeLeaf [ try parseLambdaApplicationM
, parseGrouped , parseAtomicLambdaM
]
parseLambda :: Parser TricuAST
parseLambda = between (satisfy (== LOpenParen)) (satisfy (== LCloseParen)) $ do
satisfy (== LBackslash)
param <- satisfy isIdentifier
rest <- many (satisfy isIdentifier)
satisfy (== LColon)
body <- parseLambdaExpression
let nestedLambda = foldr (\v acc -> SLambda [v] acc) body (map getIdentifier rest)
return (SLambda [getIdentifier param] nestedLambda)
parseLambdaExpression :: Parser TricuAST
parseLambdaExpression = choice
[ try parseLambdaApplication
, parseAtomicLambda
] ]
parseAtomicLambda :: Parser TricuAST parseAtomicLambdaM :: ParserM TricuAST
parseAtomicLambda = choice parseAtomicLambdaM = choice
[ parseVar [ parseVarM
, parseTreeLeaf , parseTreeLeafM
, parseLiteral , parseLiteralM
, parseListLiteral , parseListLiteralM
, try parseLambda , try parseLambdaM
, between (satisfy (== LOpenParen)) (satisfy (== LCloseParen)) parseLambdaExpression , between (satisfyM (== LOpenParen)) (satisfyM (== LCloseParen)) parseLambdaExpressionM
] ]
parseApplication :: Parser TricuAST parseApplicationM :: ParserM TricuAST
parseApplication = do parseApplicationM = do
func <- parseAtomicBase func <- parseAtomicBaseM
args <- many parseAtomic scnParserM
return $ foldl (\acc arg -> SApp acc arg) func args args <- many $ do
scnParserM
arg <- parseAtomicM
return arg
return $ foldl SApp func args
parseLambdaApplication :: Parser TricuAST parseLambdaApplicationM :: ParserM TricuAST
parseLambdaApplication = do parseLambdaApplicationM = do
func <- parseAtomicLambda func <- parseAtomicLambdaM
args <- many parseAtomicLambda scnParserM
return $ foldl (\acc arg -> SApp acc arg) func args args <- many $ do
arg <- parseAtomicLambdaM
scnParserM
pure arg
pure $ foldl SApp func args
isTreeTerm :: TricuAST -> Bool parseAtomicBaseM :: ParserM TricuAST
isTreeTerm TLeaf = True parseAtomicBaseM = choice
isTreeTerm (TStem _) = True [ parseTreeLeafM
isTreeTerm (TFork _ _) = True , parseGroupedM
isTreeTerm _ = False ]
parseTreeLeaf :: Parser TricuAST parseTreeLeafM :: ParserM TricuAST
parseTreeLeaf = satisfy isKeywordT *> notFollowedBy (satisfy (== LAssign)) *> pure TLeaf parseTreeLeafM = do
_ <- satisfyM $ \case
LKeywordT -> True
_ -> False
notFollowedBy $ lift $ satisfy (== LAssign)
pure TLeaf
parseTreeTermM :: ParserM TricuAST
parseTreeTermM = do
base <- parseTreeLeafOrParenthesizedM
rest <- many parseTreeLeafOrParenthesizedM
pure (foldl combine base rest)
where
combine acc next
| TLeaf <- acc = TStem next
| TStem t <- acc = TFork t next
| TFork _ _ <- acc = TFork acc next
parseTreeLeafOrParenthesizedM :: ParserM TricuAST
parseTreeLeafOrParenthesizedM = choice
[ between (satisfyM (== LOpenParen)) (satisfyM (== LCloseParen)) parseTreeTermM
, parseTreeLeafM
]
parseAtomicM :: ParserM TricuAST
parseAtomicM = choice
[ parseVarM
, parseTreeLeafM
, parseListLiteralM
, parseGroupedM
, parseLiteralM
]
parseGroupedM :: ParserM TricuAST
parseGroupedM = between (satisfyM (== LOpenParen)) (satisfyM (== LCloseParen)) $
scnParserM *> parseExpressionM <* scnParserM
parseLiteralM :: ParserM TricuAST
parseLiteralM = choice
[ parseIntLiteralM
, parseStrLiteralM
]
parseListLiteralM :: ParserM TricuAST
parseListLiteralM = do
_ <- satisfyM (== LOpenBracket)
elements <- many $ do
scnParserM
parseListItemM
scnParserM
_ <- satisfyM (== LCloseBracket)
pure (SList elements)
parseListItemM :: ParserM TricuAST
parseListItemM = choice
[ parseGroupedItemM
, parseListLiteralM
, parseSingleItemM
]
parseGroupedItemM :: ParserM TricuAST
parseGroupedItemM = do
_ <- satisfyM (== LOpenParen)
inner <- parseExpressionM
_ <- satisfyM (== LCloseParen)
pure inner
parseSingleItemM :: ParserM TricuAST
parseSingleItemM = do
token <- satisfyM $ \case
LIdentifier _ -> True
LKeywordT -> True
_ -> False
case token of
LIdentifier name -> pure (SVar name)
LKeywordT -> pure TLeaf
_ -> fail "Unexpected token in list item"
parseVarM :: ParserM TricuAST
parseVarM = do
LIdentifier name <- satisfyM $ \case
LIdentifier _ -> True
_ -> False
if name == "t" || name == "__result"
then fail ("Reserved keyword: " ++ name ++ " cannot be assigned.")
else pure (SVar name)
parseIntLiteralM :: ParserM TricuAST
parseIntLiteralM = do
LIntegerLiteral value <- satisfyM $ \case
LIntegerLiteral _ -> True
_ -> False
pure (SInt value)
parseStrLiteralM :: ParserM TricuAST
parseStrLiteralM = do
LStringLiteral value <- satisfyM $ \case
LStringLiteral _ -> True
_ -> False
pure (SStr value)
getIdentifier :: LToken -> String getIdentifier :: LToken -> String
getIdentifier (LIdentifier name) = name getIdentifier (LIdentifier name) = name
getIdentifier _ = error "Expected identifier" getIdentifier _ = errorWithoutStackTrace "Expected identifier"
parseTreeTerm :: Parser TricuAST
parseTreeTerm = do
base <- parseTreeLeafOrParenthesized
rest <- many parseTreeLeafOrParenthesized
pure $ foldl combine base rest
where
combine acc next = case acc of
TLeaf -> TStem next
TStem t -> TFork t next
TFork _ _ -> TFork acc next
parseTreeLeafOrParenthesized :: Parser TricuAST
parseTreeLeafOrParenthesized = choice
[ between (satisfy (== LOpenParen)) (satisfy (== LCloseParen)) parseTreeTerm
, parseTreeLeaf
]
foldTree :: [TricuAST] -> TricuAST
foldTree [] = TLeaf
foldTree [x] = x
foldTree (x:y:rest) = TFork x (foldTree (y:rest))
parseAtomic :: Parser TricuAST
parseAtomic = choice
[ parseVar
, parseTreeLeaf
, parseListLiteral
, parseGrouped
, parseLiteral
]
parseGrouped :: Parser TricuAST
parseGrouped = between (satisfy (== LOpenParen)) (satisfy (== LCloseParen)) parseExpression
parseLiteral :: Parser TricuAST
parseLiteral = choice
[ parseIntLiteral
, parseStrLiteral
]
parens :: Parser TricuAST -> Parser TricuAST
parens p = do
satisfy (== LOpenParen)
result <- p
satisfy (== LCloseParen)
return result
parseListLiteral :: Parser TricuAST
parseListLiteral = do
satisfy (== LOpenBracket)
elements <- many parseListItem
satisfy (== LCloseBracket)
return (SList elements)
parseListItem :: Parser TricuAST
parseListItem = choice
[ parseGroupedItem
, parseListLiteral
, parseSingleItem
]
parseGroupedItem :: Parser TricuAST
parseGroupedItem = do
satisfy (== LOpenParen)
inner <- parseExpression
satisfy (== LCloseParen)
return inner
parseSingleItem :: Parser TricuAST
parseSingleItem = do
token <- satisfy isListItem
case token of
LIdentifier name -> return (SVar name)
LKeywordT -> return TLeaf
_ -> fail "Unexpected token in list item"
isListItem :: LToken -> Bool
isListItem (LIdentifier _) = True
isListItem LKeywordT = True
isListItem _ = False
parseVar :: Parser TricuAST
parseVar = do
LIdentifier name <- satisfy isIdentifier
if (name == "t" || name == "__result")
then fail $ "Reserved keyword: " ++ name ++ " cannot be assigned."
else return (SVar name)
parseIntLiteral :: Parser TricuAST
parseIntLiteral = do
LIntegerLiteral value <- satisfy isIntegerLiteral
return (SInt value)
parseStrLiteral :: Parser TricuAST
parseStrLiteral = do
LStringLiteral value <- satisfy isStringLiteral
return (SStr value)
-- Boolean Helpers
isKeywordT (LKeywordT) = True
isKeywordT _ = False
isIdentifier (LIdentifier _) = True
isIdentifier _ = False
isIntegerLiteral (LIntegerLiteral _) = True
isIntegerLiteral _ = False
isStringLiteral (LStringLiteral _) = True
isStringLiteral _ = False
isLiteral (LIntegerLiteral _) = True
isLiteral (LStringLiteral _) = True
isLiteral _ = False
isNewline (LNewline) = True
isNewline _ = False
-- Alternative parsers
altSC :: AltParser ()
altSC = skipMany (char ' ' <|> char '\t' <|> char '\n')
parseTernaryTerm :: AltParser TricuAST
parseTernaryTerm = do
altSC
term <- choice parseTernaryTerm'
altSC
pure term
where
parseTernaryTerm' =
[ try (between (char '(') (char ')') parseTernaryTerm)
, try parseTernaryLeaf
, try parseTernaryStem
, try parseTernaryFork
]
parseTernaryLeaf :: AltParser TricuAST
parseTernaryLeaf = char '0' *> pure TLeaf
parseTernaryStem :: AltParser TricuAST
parseTernaryStem = char '1' *> (TStem <$> parseTernaryTerm)
parseTernaryFork :: AltParser TricuAST
parseTernaryFork = do
char '2'
term1 <- parseTernaryTerm
term2 <- parseTernaryTerm
pure $ TFork term1 term2
parseTernary :: String -> Either String TricuAST
parseTernary input = case runParser (parseTernaryTerm <* eof) "" input of
Left err -> Left (errorBundlePretty err)
Right ast -> Right ast
-- Error Handling
handleParseError :: ParseErrorBundle [LToken] Void -> String handleParseError :: ParseErrorBundle [LToken] Void -> String
handleParseError bundle = handleParseError bundle =
let errors = bundleErrors bundle let errors = bundleErrors bundle
errorList = toList errors formattedErrors = map formatError (Data.List.NonEmpty.toList errors)
formattedErrors = map showError errorList
in unlines ("Parse error(s) encountered:" : formattedErrors) in unlines ("Parse error(s) encountered:" : formattedErrors)
showError :: ParseError [LToken] Void -> String formatError :: ParseError [LToken] Void -> String
showError (TrivialError offset (Just (Tokens tokenStream)) expected) = formatError (TrivialError offset unexpected expected) =
"Parse error at offset " ++ show offset ++ ": unexpected token " let unexpectedMsg = case unexpected of
++ show tokenStream ++ ", expected one of " ++ show (Set.toList expected) Just x -> "unexpected token " ++ show x
showError (FancyError offset fancy) = Nothing -> "unexpected end of input"
"Parse error at offset " ++ show offset ++ ":\n " ++ unlines (map show (Set.toList fancy)) expectedMsg = if null expected
showError (TrivialError offset Nothing expected) = then ""
"Parse error at offset " ++ show offset ++ ": expected one of " else "expected " ++ show (Set.toList expected)
++ show (Set.toList expected) in "Parse error at offset " ++ show offset ++ ": " ++ unexpectedMsg ++
if null expectedMsg then "" else " " ++ expectedMsg
formatError (FancyError offset _) =
"Parse error at offset " ++ show offset ++ ": unexpected FancyError"

View File

@ -20,37 +20,36 @@ repl env = runInputT defaultSettings (loop env)
loop :: Env -> InputT IO () loop :: Env -> InputT IO ()
loop env = do loop env = do
minput <- getInputLine "tricu < " minput <- getInputLine "tricu < "
case minput of if
Nothing -> outputStrLn "Exiting tricu" | Nothing <- minput -> outputStrLn "Exiting tricu"
Just s -> case strip s of | Just s <- minput, strip s == "!exit" -> outputStrLn "Exiting tricu"
"!exit" -> outputStrLn "Exiting tricu" | Just s <- minput, strip s == "" -> do
"!load" -> do outputStrLn ""
path <- getInputLine "File path to load < " loop env
case path of | Just s <- minput, strip s == "!load" -> do
Nothing -> do path <- getInputLine "File path to load < "
outputStrLn "No input received; stopping import." if
loop env | Nothing <- path -> do
Just path -> do outputStrLn "No input received; stopping import."
loadedEnv <- liftIO $ evaluateFileWithContext env (strip path) loop env
loop $ Map.delete "__result" (Map.union loadedEnv env) | Just p <- path -> do
"" -> do loadedEnv <- liftIO $ evaluateFileWithContext env (strip p) `catch` \e -> errorHandler env e
outputStrLn "" loop $ Map.delete "__result" (Map.union loadedEnv env)
loop env | Just s <- minput -> do
input -> do if
case (take 2 input) of | take 2 s == "--" -> loop env
"--" -> loop env | otherwise -> do
_ -> do newEnv <- liftIO $ processInput env s `catch` errorHandler env
newEnv <- liftIO $ (processInput env input `catch` errorHandler env) loop newEnv
loop newEnv
processInput :: Env -> String -> IO Env processInput :: Env -> String -> IO Env
processInput env input = do processInput env input = do
let asts = parseTricu input let asts = parseTricu input
newEnv = evalTricu env asts newEnv = evalTricu env asts
case Map.lookup "__result" newEnv of if
Just r -> do | Just r <- Map.lookup "__result" newEnv -> do
putStrLn $ "tricu > " ++ decodeResult r putStrLn $ "tricu > " ++ decodeResult r
Nothing -> return () | otherwise -> return ()
return newEnv return newEnv
errorHandler :: Env -> SomeException -> IO (Env) errorHandler :: Env -> SomeException -> IO (Env)
@ -62,10 +61,8 @@ repl env = runInputT defaultSettings (loop env)
strip = dropWhileEnd isSpace . dropWhile isSpace strip = dropWhileEnd isSpace . dropWhile isSpace
decodeResult :: T -> String decodeResult :: T -> String
decodeResult tc = case toNumber tc of decodeResult tc
Right num -> show num | Right num <- toNumber tc = show num
Left _ -> case toString tc of | Right str <- toString tc = "\"" ++ str ++ "\""
Right str -> "\"" ++ str ++ "\"" | Right list <- toList tc = "[" ++ intercalate ", " (map decodeResult list) ++ "]"
Left _ -> case toList tc of | otherwise = formatResult TreeCalculus tc
Right list -> "[" ++ intercalate ", " (map decodeResult list) ++ "]"
Left _ -> formatResult TreeCalculus tc

View File

@ -28,7 +28,7 @@ data TricuAST
| SEmpty | SEmpty
deriving (Show, Eq, Ord) deriving (Show, Eq, Ord)
-- Tokens from Lexer -- Lexer Tokens
data LToken data LToken
= LKeywordT = LKeywordT
| LIdentifier String | LIdentifier String
@ -61,19 +61,6 @@ apply (Fork (Fork a1 a2) a3) Leaf = a1
apply (Fork (Fork a1 a2) a3) (Stem u) = apply a2 u apply (Fork (Fork a1 a2) a3) (Stem u) = apply a2 u
apply (Fork (Fork a1 a2) a3) (Fork u v) = apply (apply a3 u) v apply (Fork (Fork a1 a2) a3) (Fork u v) = apply (apply a3 u) v
-- SKI Combinators
_S :: T
_S = Fork (Stem (Fork Leaf Leaf)) Leaf
_K :: T
_K = Stem Leaf
-- Identity
-- We use the "point-free" style which drops a redundant node
-- Full I form (SKK): Fork (Stem (Stem Leaf)) (Stem Leaf)
_I :: T
_I = Fork (Stem (Stem Leaf)) Leaf
-- Booleans -- Booleans
_false :: T _false :: T
_false = Leaf _false = Leaf

View File

@ -31,7 +31,6 @@ tests = testGroup "Tricu Tests"
, lambdaEvalTests , lambdaEvalTests
, libraryTests , libraryTests
, fileEvaluationTests , fileEvaluationTests
, propertyTests
] ]
lexerTests :: TestTree lexerTests :: TestTree
@ -72,9 +71,9 @@ lexerTests = testGroup "Lexer Tests"
parserTests :: TestTree parserTests :: TestTree
parserTests = testGroup "Parser Tests" parserTests = testGroup "Parser Tests"
[ testCase "Error when assigning a value to T" $ do [ testCase "Error when assigning a value to T" $ do
let input = lexTricu "t = x" let tokens = lexTricu "t = x"
case (runParser parseExpression "" input) of case parseSingleExpr tokens of
Left _ -> return () Left _ -> return ()
Right _ -> assertFailure "Expected failure when trying to assign the value of T" Right _ -> assertFailure "Expected failure when trying to assign the value of T"
, testCase "Parse function definitions" $ do , testCase "Parse function definitions" $ do
let input = "x = (\\a b c : a)" let input = "x = (\\a b c : a)"
@ -150,10 +149,6 @@ parserTests = testGroup "Parser Tests"
let input = "(t) -- (t) -- (t)" let input = "(t) -- (t) -- (t)"
expect = [TLeaf] expect = [TLeaf]
parseTricu input @?= expect parseTricu input @?= expect
, testCase "Comments with no terms" $ do
let input = unlines ["-- (t)", "(t t)"]
expect = [SEmpty,SApp TLeaf TLeaf]
parseTricu input @?= expect
] ]
evaluationTests :: TestTree evaluationTests :: TestTree
@ -414,13 +409,3 @@ fileEvaluationTests = testGroup "Evaluation tests"
res <- liftIO $ evaluateFileWithContext library "./test/string.tri" res <- liftIO $ evaluateFileWithContext library "./test/string.tri"
decodeResult (result res) @?= "\"String test!\"" decodeResult (result res) @?= "\"String test!\""
] ]
propertyTests :: TestTree
propertyTests = testGroup "Property Tests"
[ testProperty "Lexing and parsing round-trip" $ \input ->
case runParser tricuLexer "" input of
Left _ -> property True
Right tokens -> case runParser parseExpression "" tokens of
Left _ -> property True
Right ast -> parseSingle input === ast
]

View File

@ -1,7 +1,7 @@
cabal-version: 1.12 cabal-version: 1.12
name: tricu name: tricu
version: 0.5.0 version: 0.6.0
description: A micro-language for exploring Tree Calculus description: A micro-language for exploring Tree Calculus
author: James Eversole author: James Eversole
maintainer: james@eversole.co maintainer: james@eversole.co
@ -18,6 +18,8 @@ executable tricu
src src
default-extensions: default-extensions:
DeriveDataTypeable DeriveDataTypeable
LambdaCase
MultiWayIf
OverloadedStrings OverloadedStrings
ghc-options: -threaded -rtsopts -with-rtsopts=-N -optl-pthread -fPIC ghc-options: -threaded -rtsopts -with-rtsopts=-N -optl-pthread -fPIC
build-depends: build-depends:
@ -43,6 +45,8 @@ test-suite tricu-tests
hs-source-dirs: test, src hs-source-dirs: test, src
default-extensions: default-extensions:
DeriveDataTypeable DeriveDataTypeable
LambdaCase
MultiWayIf
OverloadedStrings OverloadedStrings
build-depends: build-depends:
base base