Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ymyzk/lambda-dti
Interpreter of the ITGL with dynamic type inference
https://github.com/ymyzk/lambda-dti
dune functional-programming gradual-typing hindley-milner interpreter ocaml repl type-inferece
Last synced: 3 months ago
JSON representation
Interpreter of the ITGL with dynamic type inference
- Host: GitHub
- URL: https://github.com/ymyzk/lambda-dti
- Owner: ymyzk
- License: mit
- Created: 2018-10-13T07:38:43.000Z (over 6 years ago)
- Default Branch: master
- Last Pushed: 2024-07-08T11:37:03.000Z (7 months ago)
- Last Synced: 2024-07-08T16:19:55.578Z (7 months ago)
- Topics: dune, functional-programming, gradual-typing, hindley-milner, interpreter, ocaml, repl, type-inferece
- Language: OCaml
- Homepage: https://dl.acm.org/do/10.1145/3291628/full/
- Size: 227 KB
- Stars: 21
- Watchers: 5
- Forks: 2
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
- Citation: CITATION.cff
Awesome Lists containing this project
README
# lambda-dti
[![CI](https://github.com/ymyzk/lambda-dti/actions/workflows/ci.yml/badge.svg)](https://github.com/ymyzk/lambda-dti/actions/workflows/ci.yml)
**lambda-dti** is an interpreter of the implicitly typed gradual language (ITGL) which uses **dynamic type inference** for evaluating programs.
This implementation consists of:- Garcia and Cimini's type inference algorithm;
- a cast-inserting translator from the ITGL to the blame calculus;
- an evaluator of the blame calculus with dynamic type inference; and
- some extensions (recursion, operators, and libraries) to the ITGL.This is the artifact of the following paper in POPL 2019.
- Yusuke Miyazaki, Taro Sekiyama, and Atsushi Igarashi. [Dynamic Type Inference for Gradual Hindley–Milner Typing](https://doi.org/10.1145/3290331). POPL 2019.
## Requirements
- opam 2.0.0+
- OCaml 4.03.0+
- Dune 2.0.0+ (formerly known as Jbuilder)
- Menhir
- OUnit 2 (optional for running unit tests)
- [rlwrap](https://github.com/hanslub42/rlwrap) (optional for line editing and input history)## Getting started
### A. Building from source
```console
dune build
./_build/default/bin/main.exe
```
Run `$ ./_build/default/bin/main.exe --help` for command line options.(Optional) Run the following command to install the application:
```
$ dune install
$ ldti
```### B. Running a Docker image
Docker images are available on [GitHub](https://github.com/ymyzk/lambda-dti/pkgs/container/lambda-dti).```console
docker run -it --rm ghcr.io/ymyzk/lambda-dti:latest
```### C. Running a virtual machine
Please see [HOW_TO_USE_ARTIFACT.md](./HOW_TO_USE_ARTIFACT.md) for details.
The virtual machine image contains [lambda-dti v2.1](https://github.com/ymyzk/lambda-dti/tree/v2.1).## Tips
### Running tests
```console
dune runtest
```### Debug mode
By enabling the debug mode, our interpreter show various messages to stderr.
```console
ldti -d
```### Non-interactive mode
You can specify a file as a command line argument. Our interpreter executes the programs in the file then exits.
```console
ldti ./sample.ldti
```### Line editing
You may want to use rlwrap for line editing and input history.
```console
rlwrap ldti
```## Syntax
### Top-level
- Let declaration: `let x ... = e;;`
- Recursion declaration: `let rec f x ... = e;;`
- Expression: `e;;`### Expressions `e`
- Variables: lowercase alphabet followed by lowercase alphabets, numbers, `_`, or `'`
- Constants:
- Integers: `0`, `1`, `2`, ...
- Booleans: `true`, `false`
- Unit: `()`
- Unary operators for integers: `+`, `-`
- Binary operators (from higher precedence to lower precedence):
- Integer multiplication, division, remainder (left): `*`, `/`, `mod`
- Integer addition, subtraction (left): `+`, `-`
- Integer comparators (left): `=`, `<>`, `<`, `<=`, `>`, `>=`
- Boolean and (right): `&&`
- Boolean or (right): `||`
- Abstraction:
- Simple: `fun x -> e`
- Multiple parameters: `fun x y z ... -> e`
- With type annotations: `fun (x: U1) y (z: U3) ...: U -> e`
- Application: `e1 e2`
- Let expression:
- Simple: `let x = e1 in e2`
- Multiple parameters: `let x y z ... = e1 in e2`
- With type annotations: `let (x:U1) y (z: U3) ... : U ... = e1 in e2`
- Recursion (requires at least one parameter):
- Simple: `let rec f x = e1 in e2`
- Multiple parameters: `let rec f x y z ... = e1 in e2`
- With type annotations: `let rec f (x: U1) y (z: U3) ... : U = e1 in e2`
- If-then-else expression: `if e1 then e2 else e3`
- Sequence of expressions: `e1; e2`
- Type ascription: `(e : U)`### Types `U`
- Dynamic type: `?`
- Base types: `bool`, `int`, and `unit`
- Function type: `U -> U`
- Type variables: `'a`, `'b`, ...### Comments
- Simple: `(* comments *)`
- Nested comments: `(* leave comments here (* nested comments are also supported *) *)`## Standard library
Some useful functions and values are available:
```
# is_bool;;
- : ? -> bool =
# is_int;;
- : ? -> bool =
# is_unit;;
- : ? -> bool =
# is_fun;;
- : ? -> bool =# succ;;
- : int -> int =
# pred;;
- : int -> int =
# max;;
- : int -> int -> int =
# min;;
- : int -> int -> int =
# abs;;
- : int -> int =
# max_int;;
- : int = 4611686018427387903
# min_int;;
- : int = -4611686018427387904# not;;
- : bool -> bool =# print_bool;;
- : bool -> unit =
# print_int;;
- : int -> unit =
# print_newline;;
- : unit -> unit =# ignore;;
- : 'a -> unit =# exit;;
- : int -> unit =
```## Examples
You can check more examples in `sample.ldti` and `test/test_examples.ml`.
```
(* Simple examples which use the dynamic type *)
# (fun (x:?) -> x + 2) 3;;
- : int = 5# (fun (x:?) -> x + 2) true;;
Blame on the expression side:
line 2, character 14 -- line 2, character 15# (fun (x:?) -> x) (fun y -> y);;
- : ? = : ? -> ? => ?(* DTI: a type of y is instantiated to int *)
# (fun (x:?) -> x 2) (fun y -> y);;
- : ? = 2: int => ?(* DTI: a type of x is instantiated to X1->X2 where X1 and X2 are fresh,
then X1 and X2 are instantiated to int *)
# (fun (f:?) -> f 2) ((fun x -> x) ((fun (y:?) -> y) (fun z -> z + 1)));;
- : ? = 3: int => ?(* DTI: a type of x is instantiated to unit, then raises blame
because a cast "true: bool => ? => unit" fails *)
# (fun (f:?) -> f (); f true) (fun x -> x);;
Blame on the environment side:
line 6, character 29 -- line 6, character 39(* Let polymorphism *)
# let id x = x;;
id : 'a -> 'a =# let dynid (x:?) = x;;
dynid : ? -> ? =(* succ is in the standard library *)
# succ;;
- : int -> int =# (fun (f:?) -> f 2) (id (dynid succ));;
- : ? = 3: int => ?# (fun (f:?) -> f true) (id (dynid succ));;
Blame on the environment side:
line 11, character 33 -- line 11, character 37(* A polymorphic function which does not behave parametric *)
# let succ_non_para x = 1 + dynid x;;
succ_non_para : 'a -> int =(* Returns a value when applied to an interger value *)
# succ_non_para 3;;
- : int = 4(* Returns a value when applied to a non-interger value *)
# succ_non_para true;;
Blame on the expression side:
line 12, character 26 -- line 12, character 33(* "let x = v in e" and "e[x:=v]" should behave the same way *)
# (fun x -> 1 + dynid x) 3;;
- : int = 4(* The following example uses ν during the evaluation *)
# let nu_fun x = ((fun y -> y): ? -> ?) x;;
nu_fun : 'a -> ? =(* The following expression is translated into "nu_fun[unit,ν]; nu_fun[int,ν];;"
and returns a value *)
# nu_fun (); nu_fun 3;;
- : ? = 3: int => ?(* Recursion *)
# let rec sum (n:?) = if n < 1 then 0 else n + sum (n - 1);;
sum : ? -> int =# sum 100;;
- : int = 5050# sum true;;
Blame on the expression side:
line 18, character 23 -- line 18, character 24# exit 0;;
```## Contents
- `bin`: Entry point of the interpreter
- `lib`: Implementation of the calculus
- `test`: Unit tests## License
MIT License. See [LICENSE](LICENSE).## References
- [Ronald Garcia and Matteo Cimini. Principal Type Schemes for Gradual Programs. In Proc. of ACM POPL, 2015.](https://dl.acm.org/citation.cfm?id=2676992)