Skip to content

🦀 λ Overly-documented Rust-powered Lambda Calculus Interpreter.

License

Notifications You must be signed in to change notification settings

orsinium-labs/rlci

Repository files navigation

RLCI

🦀 Overly-documented Rust-powered Lambda Calculus Interpreter.

😎 Features

  • 📚 Overly-documented. There are comments and docstrings for everything, almost every line. Our top priority is to provide you with a good example of building your own programming language.
  • 🐦 Simple. We kept only essentials and ditched anything unneeded. Comments? Strings? System calls? Who needs it?
  • ⚙️ Working. Everything possible in lambda calculus is also possible with RLCI.
  • 🏃 REPL. We have an interactive input with autocomplete and a bit of syntax highlighting.
  • 🧩 Standard library. We have lots of useful functions available out of the box.
  • ⚠️ Friendly error messages. And we even got tracebacks!
  • 🦀 Rust. Rust is a good choice for writing a programming language, thanks to binary distributions and great performance.

🧠 Learn Lambda Calculus

The main idea of Lambda calculus is pretty simple. What if you have a programming language where you can only define a single-argument function and call other functions? That's it. No integers, no strings, no loops, no conditions. Just you and functions. And turns out, you can do quite a few things. In fact, anything that any other programming language can do.

The best way to learn lambda calculus is the Lambda Calculus from the Ground Up workshop by David Beazley. I recommend you open the Python interpreter, follow along, pause, and try to think for yourself when the speaker asks a question. It's a great exercise and this is the most mind-boggling thing I ever learned.

When you go through the workshop, take a look at the python-lambda-calculus project. There, I've implemented everything covered in the workshop and a few more things, like filter, map, reduce, and signed integers.

🤔 Motivation

I had several attempts to write a programming language before. And each time I ended up with scope creep and a spaghetti of features breaking each other and in the end, nothing works. This time, I decided to approach things differently and make the smallest possible working programming language.

If you want a Turing complete programming language, your best options are either Turing machine or Lambda calculus. The thing about the Turing machine, though, is that it's quite hard to program on it anything meaningful (see Brainfuck). But Lambda calculus is different. Lambda calculus is a real functional language and has almost everything you have in LISP or Clojure. So, here we are.

📦 Installation

If you have cargo:

cargo install rlci

If you don't, go into releases and grab a binary for your system.

📝 Syntax

Everything is a function. Every function accepts exactly one argument (you give it a name) and returns an expression. Here is one that just returns the given argument:

\x x

Or the one that accepts 2 arguments and returns the first one:

\a \b a

Technically, it accepts only one argument and returns another function that returns that first argument, but that works about the same as if we had a function accepting 2 arguments. This is known as currying.

You can assign expressions to variables to be referenced later:

id = \x \x
true = \a \b a

To call a function, specify the function you want to call and then space-separate arguments:

id true     # calls `id` with true, returns `true`
true id id  # cals `true` with `id` and `id`, returns the first `id`

You can use parenthesis to specify the order in which they should be executed:

(\x x) (not true)  # returns `false`

And that's it. Many functions are available out-of-the-box, such as boolean operations, natural numbers, lists, a few recursive funtions, and combinators.

🛠️ Usage

Run REPL:

rlci repl

Parse a module and print the result of the last expression:

cat module.txt | rlci eval

Parse and print the AST of a module:

echo 'id = λx x' | rlci parse

⚙️ Dependencies

  • pest is for parsing the language grammar into AST.
  • rustyline is for making a friendly REPL.
  • clap is for making a friendly CLI.
  • anyhow is for easy error handling and nice error messages.
  • include_dir is for including stdlib into the binary.
  • colored is for colorizing terminal output. Errors should be red!

❓ Questions and Answers

Q: Should I use it on the production?

A: No. Modern computers are designed after the Turing machine rather than Lambda calculus. So, it will be slow. Very slow. Still better than DrRacket, though.

Q: Why there are no type annotations?

A: Because there is only one type: a function.

Q: Why does it exist?

Because I can and I needed a break from tossing JSONs from one API to another.

Q: When is the next release?

A: Lambda Calculus became feature-complete in the 1930s. I'll let you know if there is anything new.

Q: Why is it on Rust?

Because writing things on Python or Go is too easy. Programming on Rust is like solving riddles, and I love riddles.

Q: Should I star the project?

A: Yes, it will make me look superior to the other people in the office.