A tiny language, a subset of Haskell aimed at aiding teachers teach Haskell
Go to file
2019-11-15 16:57:46 +01:00
app Doc steps 2019-11-15 16:55:51 +01:00
examples More run docs 2019-11-15 16:55:03 +01:00
logo Logos 2017-05-15 14:53:52 +01:00
src Add exe 2019-11-15 16:08:02 +01:00
test Add exe 2019-11-15 16:08:02 +01:00
.dir-locals.el Add exe 2019-11-15 16:08:02 +01:00
.gitignore Stackify; builds on LTS-8.9 (GHC 8.0.2) 2017-04-11 09:37:49 +01:00
duet.cabal Remove type from exe 2019-11-15 16:17:36 +01:00
LICENSE.md Rewrite README 2017-04-19 10:16:50 +01:00
README.md Add note on no modules 2019-11-15 16:57:46 +01:00
stack.yaml Remove ghcjs stuff 2019-01-26 23:03:47 +00:00
stack.yaml.lock Add exe 2019-11-15 16:08:02 +01:00

Duet

A tiny language, a subset of Haskell aimed at aiding teachers teach Haskell

Usage

This comes as a library and as a program capable of running Duet programs.

$ duet --help
Duet interpreter

Usage: duet [--version] [--help] COMMAND
  This is the interpreter for the Duet mini-Haskell educational language

Available options:
  --version                Show version
  --help                   Show this help text

Available commands:
  run                      Run the given program source

Run

The help for this command:

Usage: duet run FILEPATH [--main NAME] [--concise] [-n|--steps steps]
  Run the given program source

Available options:
  -h,--help                Show this help text
  FILEPATH                 The .hs file to interpret
  --main NAME              The main value to run
  --concise                Concise view
  -n,--steps steps         Maximum number of steps to run (default: 100)

Running code in Duet literally performs one substitution step at time. For example, evaluating (\x -> x + 5) (2 * 3), we get:

[1]
(\x -> x + 5) (2 * 3)
[2]
(2 * 3) + 5
[3]
6 + 5
[4]
11

Note that this demonstrates basic argument application and non-strictness.

Example integers.hs:

main = 3 + ((2 + -3) - 3)

Output for this program:

$ duet run examples/integers.hs
[1]
3 + ((2 + -3) - 3)
[2]
3 + (-1 - 3)
[3]
3 + -4
[4]
-1

Differences from Haskell

See also the next section for a complete example using all the available syntax.

  • Duet is non-strict, but is not lazy. There is no sharing and no thunks.
  • No module or import module system whatsoever.
  • No let syntax, no parameters in definitions e.g. f x = .. you must use a lambda. Representing let in the stepper presents a design challenge not currently met.
  • Kinds * are written Type: e.g. class Functor (f :: Type -> Type).
  • Kind inference is not implemented, so if you want a kind other than Type (aka * in Haskell), you have to put a kind signature on the type variable.
  • Indentation is stricter, a case's alts must be at a column larger than the case.
  • Infix operators are stricter: an infix operator must have spaces around it. You cannot have more than one operator without parentheses, therefore operator precedence does not come into play in Duet (this is intentional). This also permits you to write -5 without worrying about where it rests.
  • Superclasses are not supported.
  • Operator definitions are not supported.
  • There is only Integer and Rational number types: they are written as 1 or 1.0.
  • Any _ or _foo means "hole" and the interpreter does not touch them, it continues performing rewrites without caring. This is good for teaching.
  • There is no standard Prelude. The only defined base types are:
    • String
    • Char
    • Integer
    • Rational
    • Bool
  • You don't need a Show instance to inspect values; the interpreter shows them as they are, including lambdas.

View examples/syntax-buffet.hs for an example featuring all the syntax supported in Duet.