Initial pass on Antithesis testing

This adds a "limbo_stress" tool for stress testing Limbo in
non-deterministic way together with support code to run the tests under
Antithesis (which makes them deterministic). The stress tester does not
really do anything useful yet, this is just a step to make sure we can
run tests under Antithesis.
This commit is contained in:
Pekka Enberg 2025-02-14 12:57:27 +02:00
parent 672fe066c1
commit be4014a1df
15 changed files with 330 additions and 2 deletions

View file

@ -184,6 +184,39 @@ Once Maturin is installed, you can build the crate and install it as a Python mo
cd bindings/python && maturin develop
```
## Antithesis
Antithesis is a testing platform for finding bugs with reproducibility. In
Limbo, we use Antithesis in addition to our own deterministic simulation
testing (DST) tool for the following:
- Discovering bugs that the DST did not catch (and improve the DST)
- Discovering bugs that the DST does not cover (for example, non-simulated I/O)
If you have an Antithesis account, you first need to configure some
environment variables:
```bash
export ANTITHESIS_USER=
export ANTITHESIS_TENANT=
export ANTITHESIS_PASSWD=
export ANTITHESIS_DOCKER_HOST=
export ANTITHESIS_DOCKER_REPO=
export ANTITHESIS_EMAIL=
```
You can then publish a new Antithesis workflow with:
```bash
scripts/antithesis/publish-workload.sh
```
And launch an Antithesis test run with:
```bash
scripts/antithesis/launch.sh
```
## Adding Third Party Dependencies
When you want to add third party dependencies, please follow these steps:

61
Cargo.lock generated
View file

@ -128,6 +128,22 @@ dependencies = [
"windows-sys 0.59.0",
]
[[package]]
name = "antithesis_sdk"
version = "0.2.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "201eba73b76341631014baf9c0018e703af204a1e0f15d7664d8a0947f6be74d"
dependencies = [
"libc",
"libloading",
"linkme",
"once_cell",
"rand 0.8.5",
"rustc_version_runtime",
"serde",
"serde_json",
]
[[package]]
name = "anyhow"
version = "1.0.95"
@ -1765,6 +1781,17 @@ dependencies = [
"uncased",
]
[[package]]
name = "limbo_stress"
version = "0.0.15"
dependencies = [
"antithesis_sdk",
"clap",
"limbo",
"serde_json",
"tokio",
]
[[package]]
name = "limbo_time"
version = "0.0.15"
@ -1786,6 +1813,26 @@ dependencies = [
"uuid",
]
[[package]]
name = "linkme"
version = "0.3.31"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "566336154b9e58a4f055f6dd4cbab62c7dc0826ce3c0a04e63b2d2ecd784cdae"
dependencies = [
"linkme-impl",
]
[[package]]
name = "linkme-impl"
version = "0.3.31"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "edbe595006d355eaf9ae11db92707d4338cd2384d16866131cc1afdbdd35d8d9"
dependencies = [
"proc-macro2",
"quote",
"syn 2.0.96",
]
[[package]]
name = "linux-raw-sys"
version = "0.4.15"
@ -2654,6 +2701,16 @@ dependencies = [
"semver",
]
[[package]]
name = "rustc_version_runtime"
version = "0.3.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2dd18cd2bae1820af0b6ad5e54f4a51d0f3fcc53b05f845675074efcc7af071d"
dependencies = [
"rustc_version",
"semver",
]
[[package]]
name = "rustix"
version = "0.38.43"
@ -2745,9 +2802,9 @@ dependencies = [
[[package]]
name = "serde_json"
version = "1.0.135"
version = "1.0.139"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2b0d7ba2887406110130a978386c4e1befb98c674b4fba677954e4db976630d9"
checksum = "44f86c3acccc9c65b153fe1b85a3be07fe5515274ec9f0653b4a0875731c72a6"
dependencies = [
"indexmap",
"itoa",

View file

@ -21,6 +21,7 @@ members = [
"macros",
"simulator",
"sqlite3",
"stress",
"tests",
]
exclude = ["perf/latency/limbo"]

73
Dockerfile.antithesis Normal file
View file

@ -0,0 +1,73 @@
FROM lukemathwalker/cargo-chef:0.1.68-rust-1.84.0-slim-bullseye AS chef
RUN apt update \
&& apt install -y git libssl-dev pkg-config\
&& apt clean \
&& rm -rf /var/lib/apt/lists/*
WORKDIR /app
#
# Cache dependencies
#
FROM chef AS planner
COPY ./Cargo.lock ./Cargo.lock
COPY ./Cargo.toml ./Cargo.toml
COPY ./bindings/go ./bindings/go/
COPY ./bindings/java ./bindings/java/
COPY ./bindings/python ./bindings/python/
COPY ./bindings/rust ./bindings/rust/
COPY ./bindings/wasm ./bindings/wasm/
COPY ./cli ./cli/
COPY ./core ./core/
COPY ./extensions ./extensions/
COPY ./macros ./macros/
COPY ./simulator ./simulator/
COPY ./sqlite3 ./sqlite3/
COPY ./tests ./tests/
COPY ./stress ./stress/
COPY ./vendored ./vendored/
RUN cargo chef prepare --bin limbo_stress --recipe-path recipe.json
#
# Build the project.
#
FROM chef AS builder
ARG antithesis=true
# Source: https://antithesis.com/assets/instrumentation/libvoidstar.so
COPY stress/libvoidstar.so /opt/antithesis/libvoidstar.so
COPY --from=planner /app/recipe.json recipe.json
RUN cargo chef cook --bin limbo_stress --release --recipe-path recipe.json
COPY --from=planner /app/bindings/rust ./bindings/rust/
COPY --from=planner /app/core ./core/
COPY --from=planner /app/extensions ./extensions/
COPY --from=planner /app/macros ./macros/
COPY --from=planner /app/stress ./stress/
COPY --from=planner /app/vendored ./vendored/
RUN if [ "$antithesis" = "true" ]; then \
cp /opt/antithesis/libvoidstar.so /usr/lib/libvoidstar.so && \
export RUSTFLAGS="-Ccodegen-units=1 -Cpasses=sancov-module -Cllvm-args=-sanitizer-coverage-level=3 -Cllvm-args=-sanitizer-coverage-trace-pc-guard -Clink-args=-Wl,--build-id -L/usr/lib/ -lvoidstar" && \
cargo build --bin limbo_stress --release; \
else \
cargo build --bin limbo_stress --release; \
fi
#
# The final image.
#
FROM debian:bullseye-slim AS runtime
RUN apt-get update && apt-get install -y bash && rm -rf /var/lib/apt/lists/*
WORKDIR /app
EXPOSE 8080
COPY --from=builder /usr/lib/libvoidstar.so* /usr/lib/
COPY --from=builder /app/target/release/limbo_stress /bin/limbo_stress
COPY stress/docker-entrypoint.sh /bin
RUN chmod +x /bin/docker-entrypoint.sh
ENTRYPOINT ["/bin/docker-entrypoint.sh"]
CMD ["/bin/limbo_stress"]

10
scripts/antithesis/launch.sh Executable file
View file

@ -0,0 +1,10 @@
#!/bin/sh
curl --fail -u "$ANTITHESIS_USER:$ANTITHESIS_PASSWD" \
-X POST https://$ANTITHESIS_TENANT.antithesis.com/api/v1/launch/basic_test \
-d "{\"params\": { \"antithesis.description\":\"basic_test on main\",
\"antithesis.duration\":\"1\",
\"antithesis.config_image\":\"$ANTITHESIS_DOCKER_REPO/limbo-config:antithesis-latest\",
\"antithesis.images\":\"$ANTITHESIS_DOCKER_REPO/limbo-workload:antithesis-latest\",
\"antithesis.report.recipients\":\"$ANTITHESIS_EMAIL\"
} }"

View file

@ -0,0 +1,19 @@
#!/bin/sh
set -e
export DOCKER_REPO_URL=$ANTITHESIS_DOCKER_REPO
export IMAGE_NAME=limbo-config
export DOCKER_IMAGE_VERSION=antithesis-latest
export DOCKER_BUILD_ARGS="--build-arg antithesis=true"
export DOCKERFILE=stress/Dockerfile.antithesis-config
export DOCKER_DIR=stress
cat turso.key.json | docker login -u _json_key https://$ANTITHESIS_DOCKER_HOST --password-stdin
${BASH_SOURCE%/*}/publish-docker.sh

View file

@ -0,0 +1,23 @@
#!/bin/sh
set -e
if [ -z "$DOCKER_REPO_URL" ]; then
echo "Error: DOCKER_REPO_URL is not set."
exit 1
fi
if [ -z "$IMAGE_NAME" ]; then
echo "Error: IMAGE_NAME is not set."
exit 1
fi
if [ -z "$DOCKER_IMAGE_VERSION" ]; then
DOCKER_IMAGE_VERSION=$(git rev-parse HEAD)
fi
DOCKER_IMAGE=$DOCKER_REPO_URL/$IMAGE_NAME:$DOCKER_IMAGE_VERSION
docker build -f $DOCKERFILE -t $DOCKER_IMAGE $DOCKER_BUILD_ARGS $DOCKER_DIR
docker push $DOCKER_IMAGE

View file

@ -0,0 +1,19 @@
#!/bin/sh
set -e
export DOCKER_REPO_URL=$ANTITHESIS_DOCKER_REPO
export IMAGE_NAME=limbo-workload
export DOCKER_IMAGE_VERSION=antithesis-latest
export DOCKER_BUILD_ARGS="--build-arg antithesis=true"
export DOCKERFILE=Dockerfile.antithesis
export DOCKER_DIR=.
cat turso.key.json | docker login -u _json_key https://$ANTITHESIS_DOCKER_HOST --password-stdin
${BASH_SOURCE%/*}/publish-docker.sh

22
stress/Cargo.toml Normal file
View file

@ -0,0 +1,22 @@
# Copyright 2025 the Limbo authors. All rights reserved. MIT license.
[package]
name = "limbo_stress"
version.workspace = true
authors.workspace = true
edition.workspace = true
license.workspace = true
repository.workspace = true
description = "The Limbo stress tester"
publish = false
[[bin]]
name = "limbo_stress"
path = "main.rs"
[dependencies]
antithesis_sdk = "0.2.5"
clap = { version = "4.5", features = ["derive"] }
limbo = { path = "../bindings/rust" }
serde_json = "1.0.139"
tokio = { version = "1.29.1", features = ["full"] }

View file

@ -0,0 +1,2 @@
FROM scratch
COPY docker-compose.yaml docker-compose.yaml

View file

@ -0,0 +1,4 @@
services:
workload:
image: us-central1-docker.pkg.dev/molten-verve-216720/turso-repository/limbo-workload:antithesis-latest
command: [ "/bin/limbo_stress" ]

View file

@ -0,0 +1,5 @@
#!/bin/bash
set -Eeuo pipefail
exec "$@"

BIN
stress/libvoidstar.so Normal file

Binary file not shown.

44
stress/main.rs Normal file
View file

@ -0,0 +1,44 @@
mod opts;
use antithesis_sdk::*;
use clap::Parser;
use limbo::{Builder, Value};
use opts::Opts;
use serde_json::json;
use std::sync::Arc;
#[tokio::main]
async fn main() {
let (num_nodes, main_id) = (1, "n-001");
let startup_data = json!({
"num_nodes": num_nodes,
"main_node_id": main_id,
});
lifecycle::setup_complete(&startup_data);
antithesis_init();
let opts = Opts::parse();
let mut handles = Vec::new();
for _ in 0..opts.nr_threads {
// TODO: share the database between threads
let db = Arc::new(Builder::new_local(":memory:").build().await.unwrap());
let nr_iterations = opts.nr_iterations;
let db = db.clone();
let handle = tokio::spawn(async move {
let conn = db.connect().unwrap();
for _ in 0..nr_iterations {
let mut rows = conn.query("select 1", ()).await.unwrap();
let row = rows.next().await.unwrap().unwrap();
let value = row.get_value(0).unwrap();
assert_always!(matches!(value, Value::Integer(1)), "value is incorrect");
}
});
handles.push(handle);
}
for handle in handles {
handle.await.unwrap();
}
println!("Done.");
}

16
stress/opts.rs Normal file
View file

@ -0,0 +1,16 @@
use clap::{command, Parser};
#[derive(Parser)]
#[command(name = "limbo_stress")]
#[command(author, version, about, long_about = None)]
pub struct Opts {
#[clap(short = 't', long, help = "the number of threads", default_value_t = 8)]
pub nr_threads: usize,
#[clap(
short = 'i',
long,
help = "the number of iterations",
default_value_t = 1000
)]
pub nr_iterations: usize,
}