This repository contains the formal specifications, executable models, and implementations of the Bcc Ledger.
The documents are built in our CI and can be readily accessed using the following links:
Era | Design Documents | Formal Specification | CDDL |
---|---|---|---|
Cole | Chain Spec, Ledger Spec | CDDL | |
Sophie | Design | Spec | CDDL |
Evie & Jen | Multi-Currency, UTXOma | Spec | CDDL |
Aurum | eUTXO | Spec | CDDL |
Other Documents:
- Non-integer calculations specification: details on the parts of the Sophie specification that use real numbers.
- Stake pool ranking specification: details for a robust stake pool ranking mechanism.
- Explanation of the small-step-semantics framework: a guide to the notation and style used by our ledger rules.
In addition, there is a formalization of the Ledger Specification in Isabelle/HOL which can be found here.
The directory structure of this repository is as follows:
- cole
- sophie
- design-spec
- chain-and-ledger (specs are combined in Sophie era)
- cddl
- Timelocks and Multi-Assets
For building LaTeX documents we use
nix
. Haskell files can be built either
with nix
or cabal
.
When using nix
it is recommended that you setup the cache, so that it can
reuse built artifacts, reducing the compilation times dramatically:
If you are using NixOS add the snippet below to your
/etc/nixos/configuration.nix
:
nix.binaryCaches = [
"https://cache.nixos.org"
"https://hydra.tbco.io"
];
nix.binaryCachePublicKeys = [
"hydra.tbco.io:f/Ea+s+dFdN+3Y/G+FDgSq+a5NEWhJGzdjvKNGv0/EQ="
];
If you are using the nix
package manager next to another operating system put
the following in /etc/nix/nix.conf
if you have a system-wide nix
installation , or in ~/.config/nix/nix.conf
if you have a local installation:
substituters = https://hydra.tbco.io https://cache.nixos.org/
trusted-public-keys = hydra.tbco.io:f/Ea+s+dFdN+3Y/G+FDgSq+a5NEWhJGzdjvKNGv0/EQ= cache.nixos.org-1:6NCHdD59X431o0gWypbMrAURkbJ16ZPMQFGspcDShjY=
When using nix
the documents and Haskell code can be readily
built by running:
nix build
The LaTeX documents will be places inside directories named result*
, e.g.:
result-2/ledger-spec.pdf
result-3/delegation_design_spec.pdf
result-4/non-integer-calculations.pdf
result-5/small-step-semantics.pdf
result-6/ledger-spec.pdf
result/blockchain-spec.pdf
Change to the latex directory where the latex document is (e.g. sophie/chain-and-ledger/formal-spec
for the ledger specification corresponding to the Sophie release, or
cole/ledger/formal-spec
for the ledger specification corresponding to
the Cole release). Then, build the latex document by running:
nix-shell --pure --run make
For a continuous compilation of the LaTeX
file run:
nix-shell --pure --run "make watch"
The tests can be run with cabal. For example the Sophie tests can be run with:
cabal test sophie-spec-ledger-test
Note that the tests in sophie-spec-ledger-test
require two Ruby gems,
cbor-diag and
cddl.
It can be helpful to use the --test-show-details=streaming
option for seeing
the output of the tests while they run:
cabal test sophie-spec-ledger-test --test-show-details=streaming
The test suites use Tasty,
which allows for running specific tests.
This is done by passing the -p
flag to the test program, followed by an awk
pattern.
You can alternatively use the TASTY_PATTERN
environment variable with a pattern.
For example, the Sophie golden tests can be run with:
cabal test sophie-spec-ledger-test --test-options="-p golden"
or
TASTY_PATTERN=golden cabal test sophie-spec-ledger-test
Tasty
allows for more
complex patterns.
For instance, to run only the Cole update mechanism tests for the ledger
that classify traces, we can pass the
-p $1 ~ /Ledger/ && $2 ~ /Update/ && $3 ~ /classified/
option.
Here each $i
refers to a level in the tests names hierarchy.
Passing -l
to tasty
will list the available test names.
When testing using cabal
, pay special attention to escaping the right symbols, e.g.:
cabal test cole-spec-ledger:test:cole-spec-ledger-test --test-options "-p \"\$1 ~ /Ledger/ && \$2 ~ /Update/ && \$3 ~ /classified/\""
When a QuickCheck test fails, the seed which produced the failure is reported. The failure can be replayed with:
cabal test sophie-spec-ledger-test --test-options "--quickcheck-replay=42"
(where 42 is an example seed).
Most of the test suites are grouped into test scenarios.
For example, the Sophie test suite contains
ContinuousIntegration
, Development
, Nightly
, and Fast
,
which can be run with the --scenario
flag. For example:
cabal test sophie-spec-ledger-test --test-options --scenario=Fast
We have support for running
ghcid
from inside of nix-shell.
Enter nix-shell from the base directory of the repository,
change directories to the cabal package that you wish to check,
then run ghcid
.
For example:
nix-shell
cd sophie/chain-and-ledger/executable-spec/
ghcid
The artifacts in this repository can be built and tested using nix. This is additionally used by the Hydra CI to test building, including cross-compilation for other systems.
To add a new Haskell project, you should do the following:
- Create the project in the usual way. It should have an appropriate
.cabal
file. - Add the project to the top-level stack.yaml, configuring dependencies etc as needed. If your project's configuration deviates too far from the snapshot in ``bcc-prelude`, then you may have to submit a PR there to update that snapshot.
- At this point, test that your new project builds using
stack build <project_name>
. - Run nix-shell ./nix -A tbcoNix.stack-cabal-sync-shell --run scripts/stack-cabal_config_check.sh script to check and report your change from stack.yaml to cabal.project.
- Run the regenerate script to update sha256 checksums in cabal.project.
- Test that you can build your new project by running the following:
nix build -f default.nix libs.<project_name>
. If you have executables, then you may also try building these using theexes.<executable_name>
attribute path. A good way to see what's available is to execute:l default.nix
innix repl
. This will allow you to explore the potential attribute names. - If you want your product to be tested by CI, add it to release.nix using the format specified in that file.
To add a new LaTeX specification, the easiest way is to copy from one of the
existing specifications. You will want the Makefile
and default.nix
(say
from the Sophie ledger spec).
- Copy these files into the root of your new LaTeX specification.
- Modify the
DOCNAME
in theMakefile
. - Update
default.nix
to:- Make sure that the relative path in the first line is pointing to
(default.nix)[./default.nix]. This is used to pin the
nixpkgs
version used to build the LaTeX specifications. - Update the
buildInputs
to add in any LaTeX packages you need in your document, and remove any unneeded ones. - Alter the
meta
description field to reflect the nature of this document.
- Make sure that the relative path in the first line is pointing to
(default.nix)[./default.nix]. This is used to pin the
- Add a link to the package at the bottom of default.nix, following the existing examples.
- To require that your specification be built in CI, add it at the end of the list in default.nix following the existing examples.
You can find additional documentation on the nix infrastructure used in this repo in the following places:
Note that the user guide linked above is incomplete and does not correctly refer
to projects built using tbco-nix
, as this one is. A certain amount of trial
and error may be required to make substantive changes!
We use editorconfig
to ensure consistency in the format of our
Haskell code. There are editorconfig plugins for several text editors, so make sure that your editor
honors the configuration in .editorconfig
.
Additionally, we use ormolu
for formatting.
There is a script here
which uses nix to format the appropriate directories.