The adaptive load controller is a standalone library with a proto based API. It is an abstraction built on top of Nighthawk that implements an adaptive execution mode.
When executing Nighthawk in closed-loop or open-loop mode, the test configuration must specify the load level to deliver to the system under test, i.e. the number of requests per second (RPS). When running tests continuously over a longer time period, this configuration needs to be manually maintained and may get obsolete as the performance of the system under test changes over time.
When executing Nighthawk in the adaptive mode, the adaptive load controller automatically determines the maximum load specification the system under test can sustain and still meet some defined criteria (e.g. success rate, latency, etc...). This decreases test maintenance and allows to monitor how the performance of the system under test evolves over time. The maximum load specification is further referred to as the optimal configuration.
The execution is broken into two stages in the adaptive mode:
- The adaptive (search) stage.
- The testing stage.
In the adaptive stage the adaptive load controller iteratively executes Nighthawk with varying test parameters in search for the optimal configuration. The individual Nighthawk executions use the open-loop mode. The controller collects metrics after each iteration and evaluates them according to configured thresholds in order to determine if the configuration used in the iteration is optimal. Once the controller finds the optimal test configuration, it enters the testing stage.
In the testing stage the adaptive load controller delivers load to the system under test using the optimal test configuration found in the adaptive stage.
The following diagram illustrates an execution when using the adaptive mode.
This diagram outlines the architecture of the adaptive load controller:
The adaptive load controller binary talks to a Nighthawk server over gRPC. The controller accepts its configuration in protocol buffer format.
The config validator validates the provided configuration, ensuring that the adaptive mode can successfully execute.
The step controller executes the chosen search algorithm (e.g. exponential search) and uses the variable setter to alter Nighthawk's configuration used in individual iterations.
The metric evaluator decides whether metrics observed in an iteration are within bounds specified in the configuration. It utilizes a scoring function that represents the observed metrics as a score usable by the search algorithm.
The metric evaluator uses the metric plugins to receive the metric values.
The yellow components in the architecture diagram above are implemented as plugins using the Envoy plugin architecture. Users of Nigthhawk can create and use their own implementations of these components. This section of the document lists the plugin implementations that are included in this repository.
The existing metrics plugins are implemented in the metrics_plugin_impl.cc file.
Metric Plugin | Plugin Name | Metric Name | Description |
---|---|---|---|
Attempted RPS | builtin |
attempted-rps |
The request per seconds (RPS) value Nighthawk attempted in the iteration. |
Achieved RPS | builtin |
achieved-rps |
The request per seconds (RPS) value Nighthawk achieved in the iteration. |
Send rate | builtin |
send-rate |
A percentage of planned requests that Nighthawk managed to send in the iteration. |
Success rate | builtin |
success-rate |
A percentage of the sent requests that resulted in HTTP code 200 in the iteration. |
Minimum latency | builtin |
latency-ns-min |
The minimum latency in nanoseconds observed by Nighthawk in the iteration. |
Mean latency | builtin |
latency-ns-mean |
The average latency in nanoseconds observed by Nighthawk in the iteration. |
Maximum latency | builtin |
latency-ns-max |
The maximum latency in nanoseconds observed by Nighthawk in the iteration. |
Mean + 1x stdev latency | builtin |
latency-ns-mean-plus-1stdev |
Calculated as average latency plus one standard deviation in nanoseconds as observed by Nighthawk in the iteration. |
Mean + 2x stdev latency | builtin |
latency-ns-mean-plus-2stdev |
Calculated as average latency plus two standard deviations in nanoseconds as observed by Nighthawk in the iteration. |
Mean + 3x stdev latency | builtin |
latency-ns-mean-plus-3stdev |
Calculated as average latency plus three standard deviations in nanoseconds as observed by Nighthawk in the iteration. |
Stdev latency | builtin |
latency-ns-pstdev |
The standard deviation of latencies in nanoseconds as observed by Nighthawk in the iteration. |
The existing step controllers are implemented in the step_controller_impl.cc file.
Step controller | Plugin Name | Description |
---|---|---|
Exponential search | nighthawk.exponential_search |
Implements the exponential search algorithm. |
The existing metric scoring functions are implemented in the scoring_function_impl.cc file.
Scoring function | Plugin Name | Description |
---|---|---|
Binary scoring | nighthawk.binary_scoring |
A boolean (true/false) evaluation of the metric value. |
Linear scoring | nighthawk.linear_scoring |
Evaluation of the metric value with a linear scoring constant. |
The existing input variable setters are implemented in the input_variable_setter_impl.cc file.
Input variable setter | Plugin Name | Description |
---|---|---|
RPS | nighthawk.rps |
Sets the RPS value in the Nighthawk configuration for an iteration. |
The controller accepts its configuration in protocol buffer format.
A valid configuration contains a nighthawk_traffic_template
which is the base
configuration for Nighthawk to be used in the iterations. The chosen input
variable setter alters variables in this template. Note that the adaptive load
controller overrides some values in the configuration, e.g. it will always set
open_loop
to true. The full set of overrides can be found in the
session_spec_proto_helper_impl.h.
The configuration also contains a selection of metrics, their individual configuration and a set of timers that determine the duration and deadline of the individual stages used in the adaptive mode.
The following outlines an example configuration for the adaptive mode that uses
the exponential search algorithm to find an RPS value that achieves more than
99% of the send-rate
metric:
metric_thresholds {
metric_spec { metric_name: "send-rate" }
threshold_spec {
scoring_function {
name: "nighthawk.binary_scoring"
typed_config {
[type.googleapis.com/nighthawk.adaptive_load.BinaryScoringFunctionConfig] {
lower_threshold { value: 0.99 }
}
}
}
}
}
step_controller_config {
name: "nighthawk.exponential_search"
typed_config {
[type.googleapis.com/nighthawk.adaptive_load.ExponentialSearchStepControllerConfig] {
initial_value: 200
input_variable_setter {
typed_config {
[type.googleapis.com/nighthawk.adaptive_load.RequestsPerSecondInputVariableSetterConfig] { }
}
}
}
}
}
measuring_period { seconds: 30 }
benchmark_cooldown_duration { seconds: 60 }
convergence_deadline { seconds: 1200 }
testing_stage_duration { seconds: 60 }
It is recommended to use configuration that contains at least two metric thresholds to ensure stable test results. A typical adaptive mode configuration will contain the following metric thresholds:
- Send rate to ensure Nighthawk has enough resources to deliver the load.
- Success rate to ensure the system under test is responding.
- Latency to ensure the system under test responds in reasonable time.
- CPU usage of the system under test to get a direct feedback on how much load is landing on it.