This document explains how to benchmark the codebase and read benchmark results. It also provides a step-by-step tutorial to run benchmarks on Amazon Web Services (AWS) across multiple data centers (WAN).
When running benchmarks, the codebase is automatically compiled with the feature flag benchmark
. This enables the node to print special log entries that are then read by the Python scripts and used to compute performance. These special log entries are clearly indicated with comments in the code: make sure to not alter them. (Otherwise, the benchmark scripts will fail to interpret the logs.)
After cloning the repo and installing all dependencies, you can use Fabric to run benchmarks on your local machine. Locate the task called local
in the file fabfile.py:
@task
def local(ctx):
...
The task specifies two types of parameters, the benchmark parameters and the nodes parameters. The benchmark parameters appear as follows:
bench_params = {
'nodes': 4,
'workers': 1,
'rate': 50_000,
'tx_size': 512,
'faults': 0,
'duration': 300,
}
They specify the number of primaries (nodes
) and workers per primary (workers
) to deploy, the input rate (transactions per second, or tx/s) at which the clients submit transactions to the system (rate
), the size of each transaction in bytes (tx_size
), the number of faulty nodes (faults
), and the duration of the benchmark in seconds (duration
). The minimum transaction size is 9 bytes; this ensures the transactions of a client are all different.
The benchmarking script will deploy as many clients as workers and divide the input rate equally amongst each client. For instance, if you configure the testbed with four nodes, one worker per node, and an input rate of 1,000 tx/s (as in the example above), the scripts will deploy four clients each submitting transactions to one node at a rate of 250 tx/s. When the parameter faults
is set to f > 0
, the last f
nodes and clients are not booted; the system will thus run with n-f
nodes (and n-f
clients).
The nodes parameters determine the configuration for the primaries and workers:
node_params = {
'header_num_of_batches_threshold': 32,
'max_header_num_of_batches': 1000,
'max_header_delay': '2000ms',
'min_header_delay': '1000ms',
'gc_depth': 50,
'sync_retry_delay': '10000ms',
'sync_retry_nodes': 3,
'batch_size': 500_000,
'max_batch_delay': '100ms',
'block_synchronizer': {
'range_synchronize_timeout': '30s',
'certificates_synchronize_timeout': '30s',
'payload_synchronize_timeout': '30s',
'payload_availability_timeout': '30s',
'handler_certificate_deliver_timeout': '30s'
},
"consensus_api_grpc": {
"socket_addr": "/ip4/127.0.0.1/tcp/0/http",
"get_collections_timeout": "5_000ms",
"remove_collections_timeout": "5_000ms"
},
'max_concurrent_requests': 500_000
}
They are defined as follows:
header_num_of_batches_threshold
: The minimum number of batches that will trigger a new header creationmax_header_num_of_batches
: The maximum number of batch digests included in a header.max_header_delay
: The maximum delay that the primary waits between generating two headers, even if the header does not reachheader_num_of_batches_threshold
or contain leader in parents. Denominated in milliseconds (ms).min_header_delay
: The delay that the primary waits between generating two headers, even ifheader_num_of_batches_threshold
is not reached. Denominated in ms.gc_depth
: The depth of the garbage collection. Denominated in number of rounds.sync_retry_delay
: The delay after which the synchronizer retries to send sync requests. Denominated in ms.sync_retry_nodes
: How many nodes to sync when re-trying to send sync-request. These nodes are picked at random from the committee.batch_size
: The preferred batch size. The workers seal a batch of transactions when it reaches this size. Denominated in bytes.max_batch_delay
: The delay after which the workers seal a batch of transactions, even ifmax_batch_size
is not reached. Denominated in ms.range_synchronize_timeout
: The timeout configuration when synchronizing a range of certificates from peers.certificates_synchronize_timeout
: The timeout configuration when requesting certificates from peers.payload_synchronize_timeout
: Timeout when has requested the payload for a certificate and is waiting to receive them.payload_availability_timeout
: The timeout configuration when for when we ask the other peers to discover who has the payload available for the dictated certificates.socket_addr
: The socket address the consensus api gRPC server should be listening to.get_collections_timeout
: The timeout configuration when requesting batches from workers.remove_collections_timeout
: The timeout configuration when removing batches from workers.handler_certificate_deliver_timeout
: When a certificate is fetched on the fly from peers, it is submitted from the block synchronizer handler for further processing to core to validate and ensure parents are available and history is causal complete. This property is the timeout while we wait for core to perform this processes and the certificate to become available to the handler to consume.max_concurrent_requests
: The maximum number of concurrent requests for primary-to-primary and worker-to-worker messages.
Once you specified both bench_params
and node_params
as desired, run:
$ fab local
This command first recompiles your code in release
mode (and with the benchmark
feature flag activated), thus ensuring you always benchmark the latest version of your code. This may take a long time the first time you run it. The command then generates the configuration files and keys for each node, and runs the benchmarks with the specified parameters. Finally, fab local
parses the logs and displays a summary of the execution resembling the one below. All the configuration and key files are hidden JSON files; i.e., their names start with a dot (.
), such as .committee.json
.
Here is sample output:
-----------------------------------------
SUMMARY:
-----------------------------------------
+ CONFIG:
Faults: 0 node(s)
Committee size: 4 node(s)
Worker(s) per node: 1 worker(s)
Collocate primary and workers: True
Input rate: 50,000 tx/s
Transaction size: 512 B
Execution time: 19 s
Header size: 1,000 B
Max header delay: 100 ms
GC depth: 50 round(s)
Sync retry delay: 10,000 ms
Sync retry nodes: 3 node(s)
batch size: 500,000 B
Max batch delay: 100 ms
+ RESULTS:
Consensus TPS: 46,478 tx/s
Consensus BPS: 23,796,531 B/s
Consensus latency: 464 ms
End-to-end TPS: 46,149 tx/s
End-to-end BPS: 23,628,541 B/s
End-to-end latency: 557 ms
-----------------------------------------
The 'Consensus TPS' and 'Consensus latency' report the average throughput and latency without considering the client, respectively. The consensus latency thus refers to the time elapsed between the block's creation and its commit. In contrast, End-to-end TPS
and End-to-end latency
report the performance of the whole system, starting from when the client submits the transaction. The end-to-end latency is often called 'client-perceived latency'. To accurately measure this value without degrading performance, the client periodically submits 'sample' transactions that are tracked across all the modules until they get committed into a block; the benchmark scripts use sample transactions to estimate the end-to-end latency.
Memory profiling for benchmarks are possible via jemalloc
on Linux. It can be enabled in the following way:
- Install
jemalloc
, e.g.sudo apt install libjemalloc-dev
- Enable
jemalloc
by settingexport MALLOC_CONF=prof:true,prof_prefix:jeprof.out,lg_prof_interval:33
before launching the benchmark script.
Memory profiles with names jeprof.out*
will be written to the currently directory.
To visualize the profile,
- Install
graphviz
, e.g.sudo apt install graphviz
sudo jeprof --svg <path to Narwhal binary> <path to profile> > prof.svg
This repo integrates various Python scripts to deploy and benchmark the codebase on Amazon Web Services (AWS). They are particularly useful to run benchmarks in the WAN, across multiple data centers. This section provides a step-by-step tutorial explaining how to use them.
Set up your AWS credentials to enable programmatic access to your account from your local machine. These credentials will authorize your machine to create, delete, and edit instances on your AWS account programmatically. First of all, find your 'access key id' and 'secret access key'. Then, create a file ~/.aws/credentials
with the following content:
[default]
aws_access_key_id = YOUR_ACCESS_KEY_ID
aws_secret_access_key = YOUR_SECRET_ACCESS_KEY
Do not specify any AWS region in that file as the Python scripts will allow you to handle multiple regions programmatically.
You must now add your ED25519 SSH public key to your AWS account.
PasswordRequiredException
errors when running fab
later.
If you don't have an SSH key, you can create one through Amazon or by using ssh-keygen:
$ ssh-keygen -t ed25519 -f ~/.ssh/aws
Then follow the instructions for Amazon EC2 key pairs and Linux instances to import the public key to Amazon EC2 (using the Actions > Import key pair function).
This operation is manual (AWS exposes APIs to manipulate keys) and needs to be repeated for each AWS region that you plan to use. Upon importing your key, AWS requires you to choose a 'name' for your key; ensure you set the same name on all AWS regions. This SSH key will be used by the Python scripts to execute commands and upload/download files to your AWS instances.
The file settings.json (located in narwhal/benchmarks) contains all the configuration parameters of the testbed to deploy. Its content looks as follows:
{
"key": {
"name": "aws",
"path": "/absolute/key/path"
},
"port": 5000,
"repo": {
"name": "narwhal",
"url": "https://github.com/mystenlabs/narwhal.git",
"branch": "main"
},
"instances": {
"type": "m5d.8xlarge",
"regions": [
"us-east-1",
"eu-north-1",
"ap-southeast-2",
"us-west-1",
"ap-northeast-1"
]
}
}
The first block (key
) contains information regarding your SSH key:
"key": {
"name": "aws",
"path": "/absolute/key/path"
},
Enter the name of your SSH key; this is the name you specified in the AWS web console in step 2. Also, enter the absolute path of your SSH private key (using a relative path won't work).
The second block (ports
) specifies the TCP ports to use:
"port": 5000,
Narwhal requires a number of TCP and UDP ports, depending on the number of workers per node. Each primary requires one udp ports: one to receive messages from other primaries and its own workers. And each worker requires one udp and one tcp ports: one tpc port to receive client transactions, and one udp port to receive messages from its primary from other workers. Note that the script will open a large port range (5000-7000) to the WAN on all your AWS instances.
The third block (repo
) contains the information regarding the repository's name, the URL of the repo, and the branch containing the code to deploy:
"repo": {
"name": "narwhal",
"url": "https://github.com/mystenlabs/narwhal.git",
"branch": "main"
},
Remember to update the url
field to the name of your repo. You can modify the branch name when testing new functionality without having to check out the code locally.
Then the last block (instances
) specifies the AWS instance type and the AWS regions to use:
"instances": {
"type": "m5d.8xlarge",
"regions": ["us-east-1", "eu-north-1", "ap-southeast-2", "us-west-1", "ap-northeast-1"]
}
The instance type selects the hardware on which to deploy the testbed. For example, m5d.8xlarge
instances come with 32 vCPUs (16 physical cores), 128 GB of RAM, and guarantee 10 Gbps of bandwidth. The Python scripts will configure each instance with 300 GB of SSD hard drive. The regions
field specifies the data centers to use.
If you require more nodes than data centers, the Python scripts will distribute the nodes as equally as possible amongst the data centers. All machines run a fresh install of Ubuntu Server 20.04.
The AWS instances are orchestrated with Fabric from the file fabfile.py (located in narwhal/benchmarks); you can list all possible commands as follows:
$ cd narwhal/benchmark
$ fab --list
The command fab create
creates new AWS instances; open fabfile.py and locate the create
task:
@task
def create(ctx, nodes=2):
...
The parameter nodes
determines how many instances to create in each AWS region. That is, if you specified five AWS regions as in the example of step 3, setting nodes=2
will creates a total of 10 machines:
$ fab create
Creating 10 instances |██████████████████████████████| 100.0%
Waiting for all instances to boot...
Successfully created 10 new instances
You can then clone the repo and install Rust on the remote instances with fab install
:
$ fab install
Installing rust and cloning the repo...
Initialized testbed of 10 nodes
This may take a long time as the command will first update all instances.
The commands fab stop
and fab start
respectively stop and start the testbed without destroying it. (It is good practice to stop the testbed when not in use as AWS can be quite expensive.) And fab destroy
terminates all instances and destroys the testbed. Note that, depending on the instance types, AWS instances may take up to several minutes to fully start or stop. The command fab info
displays a nice summary of all available machines and information to manually connect to them (for debugging).
After setting up the testbed, running a benchmark on AWS is similar to running it locally (see Run Local Benchmarks). Locate the task remote
in fabfile.py:
@task
def remote(ctx):
...
The benchmark parameters are similar to local benchmarks but allow you to specify the number of nodes and the input rate as arrays to automate multiple benchmarks with a single command. The parameter runs
specifies the number of times to repeat each benchmark (to later compute the average and stdev of the results), and the parameter collocate
specifies whether to collocate all the node's workers and the primary on the same machine. If collocate
is set to False
, the script will run one node per data center (AWS region), with its primary and each of its worker running on a dedicated instance.
bench_params = {
'nodes': [10, 20, 30],
'workers: 2,
'collocate': True,
'rate': [20_000, 30_000, 40_000],
'tx_size': 512,
'faults': 0,
'duration': 300,
'runs': 2,
}
Similarly to local benchmarks, the scripts will deploy as many clients as workers and divide the input rate equally amongst each client. Each client is colocated with a worker and submits transactions only to the worker with whom they share the machine.
Once you specified both bench_params
and node_params
as desired, run:
$ fab remote
This command first updates all machines with the latest commit of the GitHub repo and branch specified in your file settings.json (step 3); this ensures that benchmarks are always run with the latest version of the code. It then generates and uploads the configuration files to each machine, runs the benchmarks with the specified parameters, and downloads the logs. It finally parses the logs and prints the results into a folder called results
(which is automatically created if it doesn't already exist). You can run fab remote
multiple times without fear of overriding previous results; the command either appends new results to a file containing existing results or prints them in separate files. If anything goes wrong during a benchmark, you can always stop it by running fab kill
.
Once you have enough results, you can aggregate and plot them:
$ fab plot
This command creates a latency graph, a throughput graph, and a robustness graph in a folder called plots
(which is automatically created if it doesn't already exist). You can adjust the plot parameters to filter which curves to add to the plot:
plot_params = {
'faults': [0],
'nodes': [10, 20, 50],
'workers': [1],
'collocate': True,
'tx_size': 512,
'max_latency': [3_500, 4_500]
}
The first graph ('latency') plots the latency versus the throughput. It shows that the latency is low until a fairly neat threshold after which it drastically increases. Determining this threshold is crucial to understanding the limits of the system.
Another challenge is comparing apples-to-apples between different deployments of the system. The challenge here is again that latency and throughput are interdependent, as a result a throughput/number of nodes chart could be tricky to produce fairly. The way to do it is to define a maximum latency and measure the throughput at this point instead of simply pushing every system to its peak throughput (where latency is meaningless). The second graph ('tps') plots the maximum achievable throughput under a maximum latency for different numbers of nodes.
Fail points are code instrumentations that allow errors and other behavior to be injected dynamically at runtime, primarily for testing purposes. Fail points are flexible and can be configured to exhibit a variety of behavior, including panics, early returns, and sleeping. They can be controlled both programmatically and via the environment, and can be triggered conditionally and probabilistically.
Search the code base for fail_point!()
macro to see the list instrumented failpoints.
Example usage
$ FAILPOINTS='rpc-response-delay=5%sleep(10000);request-batch=5%return;report-our-batch=5%return;request-vote=5%return;certificate-store-panic=.01%return;certificate-store=5%return' \
fab failpoints