Support for nested variable definitions
This commit is contained in:
parent
3dc3e3b842
commit
d3f63b1d15
2 changed files with 40 additions and 20 deletions
23
app/Main.hs
23
app/Main.hs
|
@ -2,7 +2,7 @@ module Main (main) where
|
||||||
|
|
||||||
import Text.Parsec
|
import Text.Parsec
|
||||||
|
|
||||||
import Lib (exprparser, evaluate, replaceVars, Expr, extractVariableDefinitions)
|
import Lib (exprparser, evaluate, replaceVars, Expr, extractVariableDefinitions, updateVariables)
|
||||||
|
|
||||||
import Data.Map (Map)
|
import Data.Map (Map)
|
||||||
import qualified Data.Map as Map
|
import qualified Data.Map as Map
|
||||||
|
@ -24,17 +24,24 @@ showRatio p r = (if (r < 0) then "-" else "") ++ prepoint_digits ++ (if (length
|
||||||
prepoint_digits = init . show . round . abs $ (r * 10)
|
prepoint_digits = init . show . round . abs $ (r * 10)
|
||||||
postpoint_digits = reverse . dropWhile (=='0') . reverse .(take p) . (drop (length prepoint_digits)) . show . round . abs $ (r * 10^p)
|
postpoint_digits = reverse . dropWhile (=='0') . reverse .(take p) . (drop (length prepoint_digits)) . show . round . abs $ (r * 10^p)
|
||||||
|
|
||||||
useResult :: Map String Rational -> Either ParseError Expr -> String
|
useResult :: Map String Rational -> Expr -> String
|
||||||
useResult vs (Right expr) = either id ((showRatio precision) . evaluate) $ replaceVars expr vs
|
useResult vs expr = either id ((showRatio precision) . evaluate) $ replaceVars expr vs
|
||||||
useResult vs (Left error) = show error
|
|
||||||
|
|
||||||
ioLoop :: Map String Rational -> IO ()
|
ioLoop :: Map String Rational -> IO ()
|
||||||
ioLoop vs = do done <- isEOF
|
ioLoop vs = do done <- isEOF
|
||||||
if done
|
if done
|
||||||
then putStrLn "Quit!"
|
then putStrLn "Quit!"
|
||||||
else do inp <- getLine
|
else do inp <- getLine
|
||||||
let expr = parse exprparser "<stdin>" inp
|
let expr_res = parse exprparser "<stdin>" inp
|
||||||
let uvs = either (const vs) (\e -> Map.unionWith (flip const) vs (extractVariableDefinitions e)) expr
|
case expr_res of
|
||||||
putStrLn $ useResult uvs expr
|
Left err -> do
|
||||||
ioLoop uvs
|
putStrLn . show $ err
|
||||||
|
ioLoop vs
|
||||||
|
Right expr -> do
|
||||||
|
let vardefs = extractVariableDefinitions expr
|
||||||
|
let uvs_res = updateVariables vardefs vs
|
||||||
|
case uvs_res of
|
||||||
|
Left err -> putStrLn err
|
||||||
|
Right uvs -> putStrLn $ useResult uvs expr
|
||||||
|
ioLoop (either (const vs) (id) uvs_res)
|
||||||
|
|
||||||
|
|
31
src/Lib.hs
31
src/Lib.hs
|
@ -1,5 +1,5 @@
|
||||||
module Lib
|
module Lib
|
||||||
( exprparser, evaluate, replaceVars, Expr, extractVariableDefinitions
|
( exprparser, evaluate, replaceVars, Expr, extractVariableDefinitions, updateVariables
|
||||||
) where
|
) where
|
||||||
import Control.Applicative((<*))
|
import Control.Applicative((<*))
|
||||||
|
|
||||||
|
@ -14,7 +14,7 @@ import Text.Parsec.Expr
|
||||||
import Text.Parsec.Token
|
import Text.Parsec.Token
|
||||||
import Text.Parsec.Language
|
import Text.Parsec.Language
|
||||||
|
|
||||||
data Expr = Variable String | Constant Rational | Binary BinaryOperator Expr Expr | VariableDefinition String Rational
|
data Expr = Variable String | Constant Rational | Binary BinaryOperator Expr Expr | VariableDefinition String Expr
|
||||||
deriving Show
|
deriving Show
|
||||||
data BinaryOperator = Plus | Minus | Multiply | Divide | Power
|
data BinaryOperator = Plus | Minus | Multiply | Divide | Power
|
||||||
deriving Show
|
deriving Show
|
||||||
|
@ -80,7 +80,7 @@ variableDefinition :: Parser Expr
|
||||||
variableDefinition = try (do
|
variableDefinition = try (do
|
||||||
name <- m_identifier
|
name <- m_identifier
|
||||||
_ <- char '='
|
_ <- char '='
|
||||||
value <- constantInteger <|> constantRational
|
value <- exprparser
|
||||||
return (VariableDefinition name value)
|
return (VariableDefinition name value)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -90,16 +90,29 @@ term = m_parens exprparser
|
||||||
<|> fmap Constant constantInteger
|
<|> fmap Constant constantInteger
|
||||||
<|> fmap Constant constantRational
|
<|> fmap Constant constantRational
|
||||||
|
|
||||||
extractVariableDefinitions :: Expr -> Map.Map String Rational
|
extractVariableDefinitions :: Expr -> [(String,Expr)]
|
||||||
extractVariableDefinitions (VariableDefinition name value) = Map.singleton name value
|
extractVariableDefinitions (VariableDefinition name value) = [(name, value)]
|
||||||
extractVariableDefinitions (Binary _ a b) = Map.union (extractVariableDefinitions a) (extractVariableDefinitions b)
|
extractVariableDefinitions (Binary _ a b) = extractVariableDefinitions a ++ extractVariableDefinitions b
|
||||||
extractVariableDefinitions (Constant _) = Map.empty
|
extractVariableDefinitions (Constant _) = []
|
||||||
extractVariableDefinitions (Variable _) = Map.empty
|
extractVariableDefinitions (Variable _) = []
|
||||||
|
|
||||||
|
updateVariables :: [(String, Expr)] -> Map String Rational -> Either String (Map String Rational)
|
||||||
|
updateVariables ds vs = foldl updateVariable (Right vs) ds
|
||||||
|
|
||||||
|
updateVariable :: Either String (Map String Rational) -> (String, Expr) -> Either String (Map String Rational)
|
||||||
|
updateVariable (Left e) _ = Left e
|
||||||
|
updateVariable (Right vs) (name, e) = case updateVariables nvs vs of
|
||||||
|
Left e -> Left e
|
||||||
|
Right uvs -> case replaceVars e uvs of
|
||||||
|
Left e -> Left e
|
||||||
|
Right exp -> Right $ Map.insert name (evaluate exp) uvs
|
||||||
|
where
|
||||||
|
nvs = extractVariableDefinitions e
|
||||||
|
|
||||||
replaceVars :: Expr -> Map.Map String Rational -> Either String Expr
|
replaceVars :: Expr -> Map.Map String Rational -> Either String Expr
|
||||||
replaceVars (Constant c) vs = Right $ Constant c
|
replaceVars (Constant c) vs = Right $ Constant c
|
||||||
replaceVars (Variable name) vs = maybe (Left ("Usage of unknown variable: '" ++ name ++ "'\nSuggestion: define the variable\n++ " ++ name ++ "=42")) (Right . Constant . id) $ Map.lookup name vs
|
replaceVars (Variable name) vs = maybe (Left ("Usage of unknown variable: '" ++ name ++ "'\nSuggestion: define the variable\n++ " ++ name ++ "=42")) (Right . Constant . id) $ Map.lookup name vs
|
||||||
replaceVars (VariableDefinition _ value ) vs = Right $ Constant value
|
replaceVars (VariableDefinition name _) vs = replaceVars (Variable name) vs
|
||||||
replaceVars (Binary op l r) vs = case leftBranch of
|
replaceVars (Binary op l r) vs = case leftBranch of
|
||||||
Left s -> Left s
|
Left s -> Left s
|
||||||
Right a -> case rightBranch of
|
Right a -> case rightBranch of
|
||||||
|
|
Loading…
Reference in a new issue