This document features basic guidelines and recommendations on how to do bpftrace development. Please read it carefully before submitting pull requests to simplify reviewing and to speed up the merge process.
There are 3 basic kinds of tests in bpftrace:
-
Unit tests for individual components (semantic analyser, codegen, etc.) based on the GoogleTest framework. These are located in
tests/*.cpp
and are executed by<builddir>/tests/bpftrace_test
. -
Runtime tests that execute bpftrace in various scenarios. These are located in
tests/runtime
and can be executed using<builddir>/tests/runtime-tests.sh
. The tests are managed by a custom framework. -
Tools parsing tests that run every tool in the
tools/
directory and check whether it executes without issues. The validity of tools outputs is not checked at the moment. Tests can be executed by<builddir>/tests/tools-parsing-test.sh
.
Every contribution should (1) not break the existing tests and (2) introduce new tests if relevant. See existing tests for inspiration on how to write new ones.
CI executes the above tests in a matrix of different environments:
- Standard (dynamically linked) bpftrace built on Ubuntu 20.04 LTS with different versions of LLVM.
- bpftrace with all dependencies, except for libc, statically linked. Uses Ubuntu 20.04, LLVM 12, and is linked dynamically to two different versions of libc.
- bpftrace with all dependencies, including libc, statically linked. Uses Alpine and LLVM 10.
The first matrix is defined in .github/workflows/ci.yml
and the latter two in
.github/workflows/embedded.yml
.
CI is automatically run on all branches and pull requests on the main repo. We recommend to enable the CI (GitHub Actions) on your own fork, too, which will allow you to run the CI against your testing branches.
It may often happen that tests pass on your local setup but fail in one of the CI environments (especially the embedded ones). In such a case, it is useful to reproduce the environment to debug the issue.
All CI tests run in Docker containers created from our custom images. See
.github/workflows/*.yml
for exact docker build
and docker run
commands.
Note: the images use docker/build.sh
as the entrypoint so you may want to
override it (--entrypoint=
) and build bpftrace manually in the container.
Some tests are known to be flaky and sometimes fail in the CI environment. The list of known such tests:
- runtime test
usdt."usdt probes - file based semaphore activation multi process"
(#2410)
What usually helps, is restarting the CI. This is simple on your own fork but requires one of the maintainers for pull requests.
We use clang-format with our custom config for formatting code. This was
introduced after a lot of code
was already written. Instead of formatting the whole code base at once and
breaking git blame
we're taking an incremental approach, each new/modified bit
of code needs to be formatted.
The CI checks this too, if the changes don't adhere to our style the job will fail.
git clang-format
can be used to easily format commits, e.g. git clang-format upstream/master
We want to avoid fix formatting
commits. Instead every commit should be
formatted correctly.
The changelog is for end users. It should provide them with a quick summary of all changes important to them. Internal changes like refactoring or test changes do not belong to it.
To avoid having write a changelog when we do a release (which leads to useless changelog or a lot of work) we write them as we go. That means that every PR that has a user impacting change must also include a changelog entry.
As we include the PR number in the changelog format this can only be done after the PR has been opened.
If it is a single commit PR we include the changelog in that commit, when the PR consists of multiple commits it is OK to add a separate commit for the changelog.
For more details on bpftrace internals, see internals_development.md.