Introduction
In class we looked at depython
, some types for representing python abstract syntax trees.
In python you can introspect code using the ast
module.
import ast
ast.dump(ast.parse("123"), annotate_fields=False)
The result is
'Module([Expr(Constant(123))], [])'
A more complicated example:
ast.dump(ast.parse("2+3"), annotate_fields=False)
Results in
'Module([Expr(BinOp(Constant(2, None), Add(), Constant(3, None)))], [])'
Types
We can define OCaml types similar to the above representations of python ast
(* depython.ml - interpretador de python simplificado
Copyright 2025 - Humberto Ortiz-Zuazaga - humberto.ortiz@upr.edu
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*)
type op = Add | Mul
type expr =
| Constant of int
| BinOp of expr * op * expr
| Name of string
type stm =
| Assign of string * expr
| Expr of expr
type program =
| Module of stm list
let e1 = BinOp (Constant 1, Add, Constant 3)
let p1 = Module [Expr (BinOp (Constant 1, Add, Constant 3))]
let p2 = Module [Assign ("x", Constant 6) ;
Expr (BinOp (Name "x", Add, Constant 1))]
type env = (string * int) list
let update (x : string) (v : int) (env : env) : env =
(x, v) :: env
let rec lookup (x : string) (env : env) : int =
match env with
| [] -> failwith ("Undefined " ^ x)
| (id, value) :: env' ->
if x = id then value else lookup x env'
let interp_stm s env =
let rec calc e =
match e with
| Constant n -> n
| BinOp (e1, Add, e2) -> calc e1 + calc e2
| BinOp (e1, Mul, e2) -> calc e1 * calc e2
| Name x -> lookup x env
in
match s with
| Assign (v, e) -> update v (calc e) env
| Expr e -> print_endline (string_of_int (calc e)) ; env
let rec interp_stms stms env =
match stms with
| [] -> env
| (s::ss) -> let env' = interp_stm s env in interp_stms ss env'
let interp (p : program) =
match p with
| Module stms -> let _ = interp_stms stms [] in ()
Homework
Add a conditional statement like If
to depython, and implement the case in
calc
. An If
statement should have a test, which is compared to 0, if the
test is non-zero, evaluate the then branch, and if it is zero, evaluate the else
branch. These if statements must have both arms (the then case and the else
case).
Here's the python ast for a small example if:
>>> import ast
>>> prog = """if 0:
... 1
... else:
... 2
...
... """
>>> ast.dump(ast.parse(prog), annotate_fields=False)
'Module([If(Constant(0), [Expr(Constant(1))], [Expr(Constant(2))])], [])'
We can simplify this by saying an If
will take a tuple of 1 expression: the test, and two statements: the then and the else.
type stm =
...
| If of expr * stm * stm
Here's an example depython program with an if:
let unif = Module [If (BinOp (Const 1, Add, Const 2), Expr (Const 4), Expr (Const 5))];;
running interp unif
should result in 4.
WARNING
Look at the follwing program:
if 0:
launch_the_missiles()
else:
do_not_launch_missiles()
Try not to launch the missiles.