Chialisp
For this section of the course, you will learn how to set up your development environment, write Chialisp code, and execute it on the command-line.
If you are using PowerShell, make sure to install the PowerShell 7.3 preview version:
winget install --id Microsoft.Powershell.Preview --source winget
This version fixes nested quoting, which is required for many of the commands on this page.
Dev Tools
To get started with Chialisp, you will first want to install Chia Dev Tools.
Here is a summary of the instructions:
- Windows
- Linux/MacOS
mkdir learning
cd learning
py -m venv venv
./venv/Scripts/activate
pip install chia-dev-tools
cdv --version
mkdir learning
cd learning
python3 -m venv venv
. ./venv/bin/activate
pip install chia-dev-tools
cdv --version
This will install the Chia Dev Tools within your activated virtual environment. You'll want to make sure this virtual environment is activated before working on Chialisp. You'll see a (venv)
on the left of your terminal prompt.
Virtual environments allow you to install specific Python packages that will only be usable with the environment that is currently active. This allows you to switch between different environments for different projects, or if you just want to use different software versions.
Chia Dev Tools
The cdv
command provides a set of useful commands for building and running Chialisp programs, as well as some utilities for deploying smart coins on the Chia blockchain, which we will cover later on.
Run the following to see what commands it provides:
cdv
For example, a Chialisp file can be built like so:
cdv clsp build something.clsp
Run
You also have access to the run
command that can be used to compile Chialisp code directly.
If Chialisp code doesn't depend on any external parameters, the compiler will simplify it to the smallest form it can, which often means that this command will return the final output of the program.
If this is the case, you can skip the brun
command.
Run the following command:
run '(+ 2 3)'
Which should return the following result:
5
The syntax (+ 2 3)
may look confusing. In Chialisp, we place the operator first, followed by the operands. This is known as prefix notation. Think of this as the equivalent to 2 + 3
in math and most other programming languages.
It is set up this way because every program in Chialisp is written as a list, where the first item is the operator. (+ 2 3)
is a list of three elements with the first being the +
operator, and thus it's a valid Chialisp program.
Brun
The brun
command is different from the run
command in that it doesn't compile code. Instead, it takes the result of the run
command and executes it on the Chialisp Virtual Machine (CLVM) directly.
If you need to pass external parameters into the program, you will need to first compile it with run
, then use the brun
command with the parameters.
For example, let's say that the run
command produced the following CLVM bytecode output:
2
You could run it like so:
brun '2' '(42)'
Which should produce the following output:
42
So Chialisp can calculate the meaning of life!
Writing a Chialisp Puzzle
Let's start off with some terminology. Firstly, coins on the Chia blockchain use Chialisp programs named puzzles to secure the value stored within. The parameters to a puzzle are called its solution.
To create puzzles that require a solution, we will use the mod
operator. It allows us to take arguments passed in to customize the functionality and result of the puzzle.
A very basic example would be:
run '(mod (arg1 arg2) (+ arg1 arg2))'
Which should return the following result:
(+ 2 5)
What in the world is (+ 2 5)
that run
returned? This is an example of Chialisp bytecode that is later executed by the Chialisp Virtual Machine (CLVM). It is not very human-readable, but don't worry about that, as you are not required to understand CLVM bytecode in order to use it.
Our first command, run
, will take Chialisp code and compile it to bytecode. Next, brun
will take that bytecode and execute it.
We will then run this puzzle with the brun
command, followed by a solution of your choice:
brun '(+ 2 5)' '(10 5)'
Which should return the following result:
15
We are now using mod
to demand a solution for our puzzle. Whenever this is the case, you will be required to use the brun
command after run
.
Pay close attention to the location of quotes and parenthesis. It's easy to get lost! With brun
, the solution is passed in as a distinct value surrounded by quotes. (10, 5)
is the solution in this example and translates to arg1 = 10
and arg2 = 5
.
You can run it again with a different solution:
brun '(+ 2 5)' '(20 7)'
Which should return the following result:
27
At this point you have a working Chialisp puzzle that will take inputs and give back an output. Congrats on making it this far!
Comparisons and If
Going with a contrived example, let's say we wanted to add two numbers and return large
if they were > 100
, or small
if they were <= 100
.
You can compare two values like so:
(> apples oranges)
If apples
is larger than oranges
, this returns 1
. Otherwise, the output is ()
, which is equivalent to 0
.
You can then use an if statement to return one of two different things depending on the result.
(if <comparison> <result if true> <result if false>)
A concrete example of an if
would be:
run '(if 1 "true" "false")'
Which should return the following result:
"true"
Now, we will add arg1
and arg2
with the code (+ arg1 arg2)
and compare it to the literal value 100
. This comparison will determine whether the if
is true
or false
. We end up with:
run '(mod (arg1 arg2) (if (> (+ arg1 arg2) 100) "large" "small"))'
Which should return the following result:
(a (i (> (+ 2 5) (q . 100)) (q 1 . "large") (q 1 . "small")) 1)
Next, let's put this bytecode through brun
, giving it a solution:
brun '(a (i (> (+ 2 5) (q . 100)) (q 1 . "large") (q 1 . "small")) 1)' '(10 90)'
Which should return the following result:
small
Now, again with a different solution:
brun '(a (i (> (+ 2 5) (q . 100)) (q 1 . "large") (q 1 . "small")) 1)' '(10 91)'
Which should return the following result:
large
The difference here being the new solution of (10 91)
. When added together, 10
and 91
are greater than 100
.
Text Editor
Up to this point, we've been using the command line to write and run Chialisp programs. While this is efficient for quickly prototyping and testing small programs, it doesn't scale very well.
When writing larger programs in Chialisp, it'll be much easier to use a text editor of your choice and save them to a file. Both Atom and Visual Studio Code have extensions to improve the quality of life of writing Chialisp code. However, any LISP-based syntax highlighting will help as well.
If you decide to use Visual Studio Code, we have begun development on a Chialisp language server extension.
You may need to click the dropdown in the editor to install the prerelease version.
Chialisp Files
We will be storing Chialisp code in files, then building and running the files on the command line using Chia Dev Tools. There are a few commands that we can use more effectively after setting up a project in this way.
Conventions
The following file extensions are used for Chialisp:
Extension | Description |
---|---|
.clsp | Chialisp source code |
.clvm | Deserialized CLVM bytecode |
.clvm.hex | Serialized CLVM bytecode |
.clsp.hex | Generated CLVM bytecode |
.sym | Generated Chialisp symbol table |
You will be writing code in .clsp
files.
Putting it Together
Using a text editor and files allows us to format our code nicely.
Put the following in a file named first.clsp
:
(mod (arg1 arg2)
(if (> (+ arg1 arg2) 100) 'large' 'small')
)
Now, we can execute the file by name:
run first.clsp