-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathAwful.hs
221 lines (221 loc) · 9.46 KB
/
Awful.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
--------------------------------------------------------------------------------------------------------------------------------
{-# OPTIONS_GHC -Wall #-}
module Main where
import Control.Monad.Trans (liftIO)
import Control.Monad.Except (ExceptT (..), MonadError (..), liftEither, runExceptT)
import Control.Monad.State (MonadState (..), StateT, runStateT)
import Data.Foldable (traverse_)
import Data.Map (empty, fromList, insert, lookup, union, unions)
import Dictionary (Dictionary)
import Errors (Err, Error (..), Error' (..), L (..), Language_or_location (..), add_file_name, write_error)
import Eval (eval, run)
import Naming (naming_file, naming_term, naming_tests)
import Parser (
Code_file,
Code_file' (..),
Kind (..),
Input (..),
Term,
Test,
Test_file (..),
parse_code_file,
parse_input,
parse_test_file)
import System.Directory (findFile)
import System.Environment (getArgs)
import Typing (
(!->),
(!=>),
Polymorphic_term (..),
Polymorphic_type (..),
Term (..),
Term_pattern (..),
Type (..),
type_file,
type_term,
type_tests)
type Env = (Dictionary Language_or_location, Dictionary Polymorphic_type, Dictionary Polymorphic_term)
type File_checker =
StateT
(Dictionary Language_or_location, Dictionary (Dictionary Polymorphic_type), Dictionary Polymorphic_term)
(ExceptT Error IO)
check :: [L String] -> ExceptT Error IO Env
check file_names =
do
(term_types, (term_locations, _, term_defs)) <-
runStateT (check_files [] "input" file_names) (init_term_locations, empty, init_term_defs)
return (term_locations, term_types, term_defs)
check_circular_dependencies :: [String] -> String -> File_checker ()
check_circular_dependencies file_names file_name =
case elem file_name file_names of
False -> return ()
True -> throwError (Circular_dependency_between_files file_names)
check_file :: [String] -> String -> L String -> File_checker (Dictionary Polymorphic_type)
check_file requesting_file_names requesting_file_name (L line_and_char requested_file_name) =
do
files <- get_files
case Data.Map.lookup requested_file_name files of
Nothing ->
do
check_circular_dependencies requesting_file_names requested_file_name
maybe_file <- liftIO (read_file requested_file_name)
file <-
case maybe_file of
Nothing -> throwError (Error requesting_file_name (Failed_to_find_file requested_file_name line_and_char))
Just file' -> return file'
Code_file' imports file' <- liftEither (add_file_name requested_file_name (parse_code_file file))
term_types <- check_files (requested_file_name : requesting_file_names) requested_file_name imports
term_locations <- get_term_locations
term_defs <- get_term_defs
(term_locations', term_types', term_defs') <-
liftEither (naming_and_type_file requested_file_name term_locations term_types file')
put (term_locations', insert requested_file_name term_types' files, union term_defs term_defs')
return term_types'
Just term_types -> return term_types
check_files :: [String] -> String -> [L String] -> File_checker (Dictionary Polymorphic_type)
check_files requested_file_names requesting_file_name imports =
do
term_types <- traverse (check_file requested_file_names requesting_file_name) imports
return (unions (init_term_types : term_types))
eval :: [L String] -> Parser.Term -> ExceptT Error IO String
eval file_names term =
do
env <- check file_names
liftEither (naming_type_and_eval_term env term)
get_files :: File_checker (Dictionary (Dictionary Polymorphic_type))
get_files =
do
(_, files, _) <- get
return files
get_term_defs :: File_checker (Dictionary Polymorphic_term)
get_term_defs =
do
(_, _, term_defs) <- get
return term_defs
get_term_locations :: File_checker (Dictionary Language_or_location)
get_term_locations =
do
(term_locations, _, _) <- get
return term_locations
{-
init_constructors :: Set String
init_constructors = Data.Set.fromList ["EQ", "GT", "LT"]
-}
init_term_defs :: Dictionary Polymorphic_term
init_term_defs =
Data.Map.fromList
[
(
"Add",
Def_term
[]
(Name_term_pattern "x" !-> Name_term_pattern "y" !-> Add_Int_term (Local_name_term "x") (Local_name_term "y"))),
-- ("Compare", Name_term_pattern "x" !-> Name_term_pattern "y" !-> Compare_Int_term (Name_term "x") (Name_term "y")),
("Convert", Def_term [] (Name_term_pattern "x" !-> Local_name_term "x")),
(
"Div",
Def_term
[]
(Name_term_pattern "x" !-> Name_term_pattern "y" !-> Div_term (Local_name_term "x") (Local_name_term "y"))),
(
"Mod",
Def_term
[]
(Name_term_pattern "x" !-> Name_term_pattern "y" !-> Mod_term (Local_name_term "x") (Local_name_term "y"))),
(
"Times",
Def_term
[]
(Name_term_pattern "x" !-> Name_term_pattern "y" !-> Times_Int_term (Local_name_term "x") (Local_name_term "y")))]
init_term_locations :: Dictionary Language_or_location
init_term_locations = Data.Map.fromList ((\name -> (name, Language)) <$> ["Add", "Convert", "Div", "Mod", "Times"])
init_term_types :: Dictionary Polymorphic_type
init_term_types =
Data.Map.fromList
[
("Add", Def_type [] (Name_type "Int" !=> Name_type "Int" !=> Name_type "Int")),
-- ("Compare", Int_type !=> Int_type !=> Ordering_type),
("Convert", Def_type [] (Name_type "Int" !=> Name_type "Int")),
("Div", Def_type [] (Name_type "Int" !=> Name_type "Int" !=> Name_type "Int")),
{-
("EQ", Ordering_type),
("GT", Ordering_type),
("LT", Ordering_type),
-}
("Mod", Def_type [] (Name_type "Int" !=> Name_type "Int" !=> Name_type "Int")),
("Times", Def_type [] (Name_type "Int" !=> Name_type "Int" !=> Name_type "Int"))]
init_type_kinds :: Dictionary Kind
init_type_kinds = Data.Map.fromList [("Arrow", Arrow_kind Type_kind (Arrow_kind Type_kind Type_kind)), ("Int", Type_kind)]
init_type_locations :: Dictionary Language_or_location
init_type_locations = Data.Map.fromList ((\name -> (name, Language)) <$> ["Arrow", "Int"])
main :: IO ()
main =
do
args <- getArgs
case args of
[arg] ->
case parse_input arg of
Left err -> putStrLn (write_error err)
Right input ->
case input of
Check file_names ->
do
result <- runExceptT (check file_names)
case result of
Left err -> putStrLn (write_error err)
Right _ -> return ()
Eval file_names term ->
do
maybe_term' <- runExceptT (Main.eval file_names term)
putStrLn
(case maybe_term' of
Left err -> write_error err
Right term' -> term')
Run file_names ->
do
result <- runExceptT (traverse_ Main.run file_names)
case result of
Left err -> putStrLn (write_error err)
Right () -> return ()
_ -> putStrLn (write_error Input_error)
naming_and_type_file :: String -> Dictionary Language_or_location -> Dictionary Polymorphic_type -> Code_file -> Err Env
naming_and_type_file file_name term_locations term_types file =
do
(term_locations', file') <- naming_file file_name (init_type_locations, term_locations) file
(term_types', term_defs) <- add_file_name file_name (type_file init_type_kinds term_types file')
return (term_locations', term_types', term_defs)
naming_type_and_eval_term :: Env -> Parser.Term -> Err String
naming_type_and_eval_term (term_locations, term_types, term_defs) term =
do
term' <- naming_term term_locations term
term'' <- type_term term_types term'
Eval.eval term_defs term''
naming_type_and_run_tests :: String -> Env -> [Parser.Test] -> Err ()
naming_type_and_run_tests file_name (term_locations, term_types, term_defs) tests =
do
tests' <- naming_tests file_name term_locations tests
tests'' <- add_file_name file_name (type_tests term_types tests')
add_file_name file_name (Eval.run term_defs tests'')
read_file :: String -> IO (Maybe String)
read_file file_name =
do
maybe_file_path <- findFile [""] file_name
case maybe_file_path of
Nothing -> return Nothing
Just file_path ->
do
file <- readFile file_path
return (Just file)
run :: L String -> ExceptT Error IO ()
run (L line_and_char file_name) =
-- todo: failiotsimiskood kokku check ja run jaoks?
do
maybe_file <- liftIO (read_file file_name)
file <-
case maybe_file of
Nothing -> throwError (Error "input" (Failed_to_find_file file_name line_and_char))
Just file' -> return file'
Test_file imports tests <- liftEither (add_file_name file_name (parse_test_file file))
env <- check imports
liftEither (naming_type_and_run_tests file_name env tests)
--------------------------------------------------------------------------------------------------------------------------------