This guide will walk through how to develop, update and test builtin-actors from within Lotus. The aim of this guide is to make it easier to contribute to builtin-actors by providing a clear path to follow through the somewhat complex web of dependencies and processes involved in integrating changes through to Lotus.
- Context
- Builtin-actors-only changes
- Builtin-actors and chain or state type (go-state-types) changes
- Builtin-actors and (go-state-types) state migration changes
- Builtin-actors, (go-state-types) state migration and FVM changes
builtin-actors is a Rust codebase that compiles to a set of WASM binaries that are bundled into CAR files and executed by the FVM as required by messages in the Filecoin network. These actors are used to manage the state transitions of the Filecoin network and are critical to the operation of the network. The builtin-actors bundles are not just used by Lotus, but are shared by all Filecoin implementations. The compiled bundles are referenced by their CID for each network version. That CID must match what is in the CAR file and this process is repeated by each Filecoin implementation. Differences in the compiled actors can lead to consensus failures.
The file builtin_actors_gen.go
is generated (by the actors-gen
Makefile target, i.e. make actors-gen
) and contains a mapping of CID to the compiled actor code for each network version. build/actors/
contains compressed tar files with CAR files for each network type (mainnet, calibnet, devnet, etc.) supported by Lotus. When executing Filecoin messages, the correct actors bundle for the network type and version for the particular epoch is loaded using this mapping and the WASM code is executed by the FVM to process the message.
Changes to builtin-actors code can have subtle impacts on message execution, primarily through gas accounting, which, among other things, accounts for the cost of executing individual WASM instructions. Executing a message for one network version using a different builtin-actors bundle can lead to different gas costs, which will lead to consensus failures because your local node will not agree with the rest of the network on the cost of executing a message and the state transition that results. Therefore, changes to builtin-actors code must either be executed on a separate network (e.g. a devnet), or be included in a coordinated network upgrade.
Testing integration of builtin-actors code typically involves bundling pre-release builds of the actors bundles into Lotus and executing an isolated devnet, primarily using the itests
suite. In this way, you can simulate a version of the network that executes your custom bundle and exercise the changes in functionality you are developing. You can even simulate a network upgrade process where the network switches from one bundle to another. This is described in more detail below.
When developing builtin-actors, the bundle-devnet
Makefile target can be used to compile a devenet ("2k") version of builtin-actors into a CAR file in the output/
directory. This bundle is immediately usable by Lotus for testing using the LOTUS_BUILTIN_ACTORS_VXX
environment variable. For a given actors version, XX
, which corresponds to the network version you want to replace, you can execute itests using the LOTUS_BUILTIN_ACTORS_VXX
environment variable to test the changes you have made. The file chain/consensus/filcns/upgrades.go
contains the full list of mappings from network versions to actors versions. For example, LOTUS_BUILTIN_ACTORS_V15=/path/to/builtin-actors/output/builtin-actors-devnet.car go test ./itests/test_foo.go
will run test_foo.go
using the builtin-actors-devnet.car
bundle if the test expects to load actors version 15. By default, itests will run using the latest network / actors version. So if the current network version is NV15, then actors version 15 will be loaded by default and LOTUS_BUILTIN_ACTORS_V15
will intercept the load and use the specified bundle instead of the devnet bundle packaged in build/actors/
. In this way you can run integration tests manually off a custom actors build.
This method does not extend to being able to run your Lotus changes in CI with a pull request as this override is not possible in that environment. This method is only useful for local development and testing, but can be used until a tagged and released version of builtin-actors containing your changes is checked into the Lotus repository. For a development workflow that works with CI, see Replacing actors bundles.
An alternative method to the environment variable override is to simply replace the current vX.tar.zst
file in build/actors/
with your custom bundle. This method can be used to check in code for pull requests to run in CI.
Important
Do not attempt to check your replaced file in to the Lotus codebase unless it is a tagged (and verified) builtin-actors bundle. This process is typically handled separately during the network upgrade process, so be prepared to remove your custom bundle in your commit tree before having your pull request merged.
First you will need a complete set of builtin-actors bundles for all of the network types. The builtin-actors all-bundles
Makefile target (make all-bundles
) will generate these for you and place them into the output/
directory. You can then use the build/actors/pack.sh
script to compress these into the vX.tar.zst
files that Lotus expects in that directory.
If you want to bundle a tagged builtin-actors version:
For example, following the Lotus dependency update guide for v15.0.0-rc1
:
cd build/actors/
./pack.sh v15 v15.0.0-rc1
This will fetch the CAR files from the GitHub release page for v15.0.0-rc1
and create a replacement for build/actors/v15.tar.zst
and generate an update to builtin_actors_gen.go
. If v15 is the correct actors version for the current network version (as per upgrades.go
, then running itests, by default, will use your new bundle.
If you want to bundle a custom build of builtin-actors from your local filesystem:
cd build/actors/
./pack.sh v15 local/20240930 /path/to/builtin-actors/output/
Where local/20240930
is some arbitrary identifying string that is used for the BundleGitTag
in build/builtin_actors_gen.go
. Instead of fetching from the network, it will expect the CAR files to exist in /path/to/builtin-actors/output/
and will create a replacement for build/actors/v15.tar.zst
and generate an update to build/builtin_actors_gen.go
.
By default, the itests framework will use the current latest network version. Edits to the build/builtin_actors_gen.go
file will result in an updated latestNetworkVersion
in gen/inlinegen-data.json
, which will result in a new TestNetworkVersion
in build/buildconstants/params.go
which is used in DefaultEnsembleOpts
in itests/kit/ensemble_opts.go
which will set the genesis network version of itests unless it is overridden.
This means that once the latest bundle is installed and build/builtin_actors_gen.go
has an entry with the actors Version
you want to test, you can proceed to write an itest using the same format as the existing itests without needing to provide additional options.
Sometimes, testing the upgrade boundary, or before and after upgrades, is desireable. For example, wanting to test that a feature works one way (or doesn't exist) before an upgrade, and another way after the upgrade, your itest will need an "upgrade schedule" which matches what is currently found in chain/consensus/filcns/upgrades.go
. The kit.UpgradeSchedule
function in the itests framework can be used to set this up.
For example, to test a feature that is only available after the network upgrade to version 24, you would instantiate the itest ensemble with the following option to dictate that the network starts on NV23 (using -1
as the Height
) but will perform an upgrade at height 100 to NV24:
kit.UpgradeSchedule(
stmgr.Upgrade{
Network: network.Version23,
Height: -1,
},
stmgr.Upgrade{
Network: network.Version24,
Height: 100,
Migration: filcns.UpgradeActorsV15,
},
)
There also exists a kit.LatestActorsAt(100)
short-hand for this which will produce the same effect. However, this is tied to the latestNetworkVersion
in gen/inlinegen-data.json
and will not work when the network version is incremented in the future, so it is best to be explicit about network versions you are testing.
Using client.WaitTillChain(ctx, kit.HeightAtLeast(105))
within your test will provide a boundary between the two network versions where you can test the feature before and after the upgrade, in this instance we are aiming at testing the feature at height 105
to provide a small buffer after the upgrade at 100
.
Similar to builtin-actors only, but requires additional steps to match go-state-types to builtin-actors types such that serialisation of message parameters, return types and/or state properties are consistent between the two so that Lotus can communicate with the actors correctly. Such changes require coordinating changes to both the builtin-actors and the go-state-types codebases as well as their integration into Lotus.
TODO: More detail is needed on how to coordinate changes between builtin-actors and go-state-types
Similar to builtin-actors and chain or state type changes, but requires additional steps to implement a state migration such that the state tree can be upgraded at the correct upgrade epoch by Lotus and go-state-types and match what the new version of builtin-actors expects. This requires implementing a migration to be invoked in builtin/vXX/migration/top.go
(where XX
is the new actors version), properly calling it from chain/consensus/filcns/upgrades.go
at the upgrade epoch and then implementing appropriate tests for the upgrade in itests/migration_test.go
(and elsewhere as required).
TODO: More detail is needed on how to coordinate changes between builtin-actors, go-state-types and Lotus for state migration
Similar to builtin-actors and state migration changes, but requires additional coordination to integrate a new version of ref-fvm with builtin-actors and filecoin-ffi and then integrate that into Lotus along with builtin-actors and go-state-types, along with appropriate tests.
TODO: More detail is needed on how to coordinate changes between builtin-actors, go-state-types, ref-fvm and Lotus for state migration