Zebra Continuous Integration
Zebra has extensive continuous integration tests for node syncing and
On every PR change, Zebra runs these Docker tests:
- Zebra update syncs from a cached state Google Cloud tip image
- lightwalletd full syncs from a cached state Google Cloud tip image
- lightwalletd update syncs from a cached state Google Cloud tip image
- lightwalletd integration with Zebra JSON-RPC and Light Wallet gRPC calls
When a PR is merged to the
main branch, we also run a Zebra full sync test from genesis.
Some of our builds and tests are repeated on the
main branch, due to:
- GitHub's cache sharing rules,
- our cached state sharing rules, or
- generating base coverage for PR coverage reports.
Currently, each Zebra and lightwalletd full and update sync will updates cached state images, which are shared by all tests. Tests prefer the latest image generated from the same commit. But if a state from the same commit is not available, tests will use the latest image from any branch and commit, as long as the state version is the same.
Zebra also does a smaller set of tests on tier 2 platforms using GitHub actions runners.
We use Mergify to automatically merge most pull requests.
To merge, a PR has to pass all required
main branch protection checks, and be approved by a Zebra developer.
We try to use Mergify as much as we can, so all PRs get consistent checks.
Some PRs don't use Mergify:
- Mergify config updates
- Admin merges, which happen when there are multiple failures on the
- Manual merges
We use workflow conditions to skip some checks on PRs, Mergify, or the
For example, some workflow changes skip Rust code checks.
Pull Requests from Forked Repositories
GitHub doesn't allow PRs from forked repositories to have access to our repository secret keys, even after we approve their CI. This means that Google Cloud CI fails on these PRs.
Unril we fix this CI bug, we can merge external PRs by:
- Reviewing the code to make sure it won't give our secret keys to anyone
- Pushing a copy of the branch to the Zebra repository
- Opening a PR using that branch
- Closing the original PR with a note that it will be merged (this is reauired by Mergify)
- Asking another Zebra developer to approve the new PR
Manual Testing Using Google Cloud
Some Zebra developers have access to the Zcash Foundation's Google Cloud instance, which also runs our automatic CI.
Please shut down large instances when they are not being used.
The Delete GCP Resources workflow automatically deletes test instances, instance templates, disks, and images older than a few days.
If you want to keep instances, instance templates, disks, or images in Google Cloud, name them so they don't match the automated names:
- deleted instances, instance templates and disks end in a commit hash, so use a name that doesn't end in
- deleted disks and images start with
lwd-, so use a name starting with anything else
Our production Google Cloud project doesn't have automated deletion.
To improve CI performance, some Docker tests are stateful.
Tests can depend on:
- built Zebra and
- cached state images in Google cloud
- jobs that launch Google Cloud instances for each test
- multiple jobs that follow the logs from Google Cloud (to work around the 6 hour GitHub actions limit)
- a final "Run" job that checks the exit status of the Rust acceptance test
- the current height and user-submitted transactions on the blockchain, which changes every minute
To support this test state, some Docker tests depend on other tests finishing first. This means that the entire workflow must be re-run when a single test fails.
Check if the same failure is happening on the
mainbranch or multiple PRs. If it is, open a ticket and tell the Zebra team lead.
Look for the earliest job that failed, and find the earliest failure.
For example, this failure doesn't tell us what actually went wrong:
Error: The template is not valid. ZcashFoundation/zebra/.github/workflows/build-docker-image.yml@8bbc5b21c97fafc83b70fbe7f3b5e9d0ffa19593 (Line: 52, Col: 19): Error reading JToken from JsonReader. Path '', line 0, position 0.
But the specific failure is a few steps earlier:
#24 2117.3 error[E0308]: mismatched types ...
The earliest failure can also be in another job or pull request: a. check the whole workflow run (use the "Summary" button on the top left of the job details, and zoom in) b. if Mergify failed with "The pull request embarked with main cannot be merged", look at the PR "Conversation" tab, and find the latest Mergify PR that tried to merge this PR. Then start again from step 1.
If that doesn't help, try looking for the latest failure. In Rust tests, the "failure:" notice contains the failed test names.
Fixing CI Sync Timeouts
CI sync jobs near the tip will take different amounts of time as:
- the blockchain grows, and
- Zebra's checkpoints are updated.
To fix a CI sync timeout, follow these steps until the timeouts are fixed:
Check for recent PRs that could have caused a performance decrease
Wait for a full or update sync to finish with the new checkpoints
The GitHub actions job limit is 6 hours, so the ideal job time is 4-5 hours. If any GitHub actions job times out, or takes over 5 hours:
a. Split the job based on the sync height, or
b. Adjust the sync heights in existing jobs.
If a Rust test fails with "command did not log any matches for the given regex, within the ... timeout":
a. If it's the full sync test, increase the full sync timeout
b. If it's an update sync test, increase the update sync timeouts
Fixing Duplicate Dependencies in
Check deny.toml bans
Zebra's CI checks for duplicate crate dependencies: multiple dependencies on different versions of the same crate.
If a developer or dependabot adds a duplicate dependency, the
Check deny.toml bans CI job will fail.
You can view Zebra's entire dependency tree using
cargo tree. It can also show the active features on each dependency.
To fix duplicate dependencies, follow these steps until the duplicate dependencies are fixed:
Check for updates to the crates mentioned in the
Check deny.toml banslogs, and try doing them in the same PR. For an example, see PR #5009.
a. Check for open dependabot PRs, and
b. Manually check for updates to those crates on https://crates.io .
If there are still duplicate dependencies, try removing those dependencies by disabling crate features:
a. Check for features that Zebra activates in its
Cargo.tomlfiles, and try turning them off, then
b. Try adding
default-features = falseto Zebra's dependencies (see PR #4082).
If there are still duplicate dependencies, add or update
a. Prefer exceptions for dependencies that are closer to Zebra in the dependency tree (sometimes this resolves other duplicates as well),
b. Add or update exceptions for the earlier version of duplicate dependencies, not the later version, and
c. Add a comment about why the dependency exception is needed: what was the direct Zebra dependency that caused it?
d. For an example, see PR #4890.
Repeat step 3 until the dependency warnings are fixed. Adding a single
skip-treeexception can resolve multiple warnings.
Fixing Disk Full Errors and Zcash Parameter Errors
If the Docker cached state disks are full, increase the disk sizes in:
If the GitHub Actions disks are full, or the Zcash parameter downloads time out without any network messages or errors, follow these steps until the errors are fixed:
- Check if error is also happening on the
mainbranch. If it is, skip the next step.
- Update your branch to the latest
mainbranch, this builds with all the latest dependencies in the
- Clear the GitHub Actions code cache for the failing branch. Code caches are named after the compiler version.
- Clear the GitHub Actions code caches for all the branches and the
These errors often happen after a new compiler version is released, because the caches can end up with files from both compiler versions.
If the Zcash Parameter downloads have an error loading the parameters:
- Clear the Zcash parameter caches for all branches, including
*-sprout-and-sapling-params caches should be around 765 MB.
You can find a list of caches using:
gh api -H "Accept: application/vnd.github+json" repos/ZcashFoundation/Zebra/actions/caches
And delete a cache by
gh api --method DELETE -H "Accept: application/vnd.github+json" /repos/ZcashFoundation/Zebra/actions/caches/<id>
These commands are from the GitHub Actions Cache API reference.
Retrying After Temporary Errors
Some errors happen due to network connection issues, high load, or other rare situations.
If it looks like a failure might be temporary, try re-running all the jobs on the PR using one of these methods:
@dependabot recreate(for dependabot PRs only)
- click on the failed job, and select "re-run all jobs". If the workflow hasn't finished, you might need to cancel it, and wait for it to finish.
Here are some of the rare and temporary errors that should be retried:
- Docker: "buildx failed with ... cannot reuse body, request must be retried"
- Failure in
local_listener_fixed_port_localhost_addr_v4Rust test, mention ticket #4999 on the PR
- any network connection or download failures
We track some rare errors using tickets, so we know if they are becoming more common and we need to fix them.