Fray is a tool that helps you to test multithreaded code. It is especially useful when you want to test the behavior of your code under different thread interleavings. Here is an example of how to add Fray to a Gradle project. You may find the complete source code here
Add the following plugin to your build.gradle file:
plugins {
id("org.pastalab.fray.gradle") version "0.7.3"
}Fray does not require any special annotations or modifications to your code. You can write your test code as usual. Here is an example of a simple class that you want to test:
import java.util.concurrent.atomic.AtomicInteger;
public class BankAccount {
public AtomicInteger balance = new AtomicInteger(1000);
public void withdraw(int amount) {
// Check if there is enough balance
if (balance.get() >= amount) {
// Deduct the amount
balance.set(balance.get() - amount);
}
}
}Here is an example of a test code that tests the BankAccount class without Fray:
import org.junit.jupiter.api.Test;
public class BankAccountTest {
public void myBankAccountTest() throws InterruptedException {
BankAccount account = new BankAccount();
Thread t1 = new Thread(() -> {
account.withdraw(500);
assert(account.balance.get() > 0);
});
Thread t2 = new Thread(() -> {
account.withdraw(700);
assert(account.balance.get() > 0);
});
t1.start();
t2.start();
t1.join();
t2.join();
}
@Test
public void runTestUsingJunit() throws InterruptedException {
myBankAccountTest();
}
}Here is an example of a test code that tests the BankAccount class with Fray:
...
import org.junit.jupiter.api.extension.ExtendWith;
import org.pastalab.fray.junit.junit5.FrayTestExtension;
import org.pastalab.fray.junit.junit5.annotations.ConcurrencyTest;
@ExtendWith(FrayTestExtension.class)
public class BankAccountTest {
public void myBankAccountTest() throws InterruptedException {
...
}
@ConcurrencyTest(
iterations = 1000
)
public void runTestUsingFray() throws InterruptedException {
myBankAccountTest();
}
}- First you need to add the
@ExtendWith(FrayTestExtension.class)annotation to the test class so that Fray can run the test. - Then you need to add the
@ConcurrencyTestannotation to the test method. Theiterationsparameter specifies how many times the test method should be executed.- You may also specify scheduling algorithms and other parameters in the
@ConcurrencyTestannotation. For more information, see the ConcurrencyTest.kt
- You may also specify scheduling algorithms and other parameters in the
You can run the test from the command line using the following command:
./gradlew frayTestFray will launch all tests annotated with @ConcurrencyTest and run them multiple times.
If you are using an IDE, you can run the test as you would run any other JUnit test.
If a test fails, Fray will automatically generate a test case that reproduces the failure. Fray prints the path of the recording in the standard output.
Bug found in iteration test runTestUsingFray() repetition 0 of 1000, you may find detailed report and replay files in PATH_TO_FRAY_REPORTIn the report folder, Fray logs the detailed information of the test failure in fray.log file.
2025-02-17 13:43:40 [INFO]: Error found at iter: 1, step: 19, Elapsed time: 22ms
2025-02-17 13:43:40 [INFO]: Error: java.lang.AssertionError
Thread: Thread[#20037,Thread-20003,5,main]
java.lang.AssertionError
at BankAccountTest.lambda$myBankAccountTest$0(BankAccountTest.java:14)
at java.base/java.lang.Thread.run(Thread.java:1583)
2025-02-17 13:43:40 [INFO]: The recording is saved to PATH_TO_FRAY_REPORT/recordingFray provides two ways to reproduce a failure:
You can rerun the program with the same scheduler and the recorded random choices used in the failing execution.
This technique is described in detail in our NSDI paper.
By default, Fray uses this method to reproduce failures.
Each report folder includes both the scheduler type and the random choices used during the run.
To replay the failure, simply provide the path to the replay file—no need to specify the scheduler explicitly:
@ConcurrencyTest(
replay = "PATH_TO_FRAY_REPORT/recording"
)Alternatively, you can reproduce the failure by replaying the exact thread schedule observed during the original execution. This requires that the schedule was recorded when the test was run.
To enable schedule recording, set the following system property: -Dfray.recordSchedule=true.
Note
This approach is less reliable if your program contains other sources of nondeterminism (e.g., random number generators, I/O operations, etc.) that influence the execution path.
Once the schedule is recorded, configure your test to use the ReplayScheduler:
@ConcurrencyTest(
scheduler = ReplayScheduler.class,
replay = "PATH_TO_FRAY_REPORT/recording"
)Fray downloads Corretto JDK 25 and runs ConcurrencyTest with it by default. However, NixOS cannot run dynamically
linked executables. To run Fray on NixOS, you can provide environment variable JDK25_HOME and Fray will use provided
JDK 25 instead of downloading it.
packages = with pkgs; [
...
javaPackages.compiler.openjdk25
];
shellHook = ''
export JDK25_HOME="${pkgs.javaPackages.compiler.openjdk25.home}"
''Fray also provides a Java agent that lets you run Fray with existing Java applications without using the Fray launcher. This is useful when your application is already running inside a deterministic environment (such as Antithesis) and you only want Fray to control thread interleavings.
To use the agent, you can start from Fray's prebuilt Docker image.
FROM ghcr.io/cmu-pasta/fray:0.7.3 as fray
COPY --from=fray /nix /nix
COPY --from=fray /opt/fray /opt/frayAfter you have the image, run your application with the Fray agent:
- Replace the
javacommand with the instrumented/opt/fray/java-inst/bin/java. - If you use launchers such as Gradle or Maven, set
JAVA_HOMEto/opt/fray/java-inst. - Add the following two agents:
-javaagent:/opt/fray/libs/fray-core-FRAY_VERSION-SNAPSHOT-all.jar=FRAY_ARGSFRAY_ARGSare the same arguments you would pass to the Fray launcher, separated by colons (:).- For example, to use the
posscheduler and enable memory interleaving:-javaagent:/opt/fray/libs/fray-core-0.7.3-SNAPSHOT-all.jar=-m:--scheduler:pos
-agentpath:/opt/fray/native-libs/libjvmti.so
To fully leverage Antithesis's fuzzing capabilities, add the following argument to the Fray agent: --randomness-provider:antithesis-sdk-random. You can also set this system property to have Fray report errors through the Antithesis SDK: -Dfray.antithesisSdk=true.
Note
Ensure the Antithesis SDK is available on your application's classpath. Fray does not package the Antithesis SDK.