Grinder is a distributed load testing tool described here
This code defines implementations of grinder worker threads meant to repeatedly invoke the required number of BF api calls during each "reporting interval"
It also includes the infrastructure to divide the total work described in the grinder properties file across all the workers in the distributed system.
The BF grinder system is designed to take a single properties file that lists the total number of threads to be used to generate load, and allocate each of those threads to generate a certain kind of http request.
The BF grinder code creates different types of threads to handle the different kinds of requests to be generated. Currently the thread types are:
- IngestThread - create requests for the "/ingest/multi" endpoint
- AnnotationsIngestThread - ingests to the "events" endpoint
- Various query threads:
- SinglePlotQuery - send GET requests to
<tenant>/views/<metric>
withfrom
,to
, andresolution
params - MultiPlotQuery - send POST requests to
<tenant>/views
withfrom
,to
, andresolution
params - SearchQuery - send requests to
<tenant>/metrics/search
- AnnotationsQuery - send requests to
<tenant>/events/getEvents
withfrom
,until
params
- SinglePlotQuery - send GET requests to
The grinder.threads
property determines the number of threads started by each worker process.
How those threads are divided up between the different *Thread
classes is determined by weight properties.
A given thread type will be created for a number of threads proportional to the weight specified for that thread type divided by the total weight for all thread types.
Each thread generates one HTTP request for each "run" of Grinder.
Grinder-specific properties are discussed in more detail here
-
grinder.runs
- The number of times the grinder script should be run. If0
, then it will run indefinitely, until manually stopped. Default is1
. On each run, every thread will make its request exactly once, so if runs is set to 1, then you'll get as many requests as you have threads. -
grinder.processes
- The number of worker processes started by each agent. Default is1
. -
grinder.threads
- The number of threads per worker process. Default is1
. -
[grinder.bf.]url
- The HTTP Url for ingestion-based traffic. Default ishttp://localhost:19000
. -
[grinder.bf.]query_url
- The HTTP Url for query-based traffic. Default ishttp://localhost:20000
. -
[grinder.bf.]max_multiplot_metrics
- Default is10
. -
[grinder.bf.]name_fmt
- Default isorg.example.metric.%d
. -
[grinder.bf.]throttling_group.<name>.max_requests_per_minute
- Create a throttling group with the givenname
. The value of the property is taken as the throttling group'smax_requests_per_minute
parameter. By default, no throttling groups are created if no properties are specified. -
[grinder.bf.]ingest_weight
- Default is15
. -
[grinder.bf.]ingest_num_tenants
- Ingestion threads randomly generate a numerical tenant id in the range of[0,ingest_num_tenants)
. Change this property to control how many different tenant id's are used when sending standard metrics to the service. Default is4
. This isn't the total number of tenants that will be used, only the number that can be used. -
[grinder.bf.]ingest_metrics_per_tenant
- Ingestion threads randomly generate a numerical metric name suffix in the range of[0,ingest_metrics_per_tenant)
. Metric names will generally be of the formorg.example.metric.%d
, or whatever thename_fmt
property is set to. Change this property to control how many different metrics will have data generated for them. Default is15
. As iningest_num_tenants
, this isn't the number of metrics that will be sent. It's the number of distinct metric names than can be sent. -
[grinder.bf.]ingest_batch_size
- Ingestion threads will generate this many metrics in a single payload (HTTP request body). Default is5
. This is the property that actually controls how many metrics are ingested per request or thread. With 5 ingest threads and a batch size of 10, 50 metrics total will be ingested. The names and tenants of those metrics are randomized within the bounds of theingest_num_tenants
andingest_metrics_per_tenant
properties. -
[grinder.bf.]ingest_delay_millis
- Configures delayed metrics. Default is""
, which doesn't produce any delayed metrics. Set to a comma-separated list of delays, in milliseconds, to produce some delayed metrics. For half of the tenants (the even numbered ones), a delay value will be randomly selected from the list and applied to each metric produced for that tenant, making it look like the metric was produced that many milliseconds ago but has just now been ingested. Example:"0,0,0,60000"
would make about 1/4 of the metrics of even-numbered tenants appear delayed by 1 minute. -
[grinder.bf.]ingest_throttling_group
- Name of an above-defined throttling group. The named tgroup will be assigned to allIngestThread
objects. Default isNone
. If the tgroup name is blank, or is not defined among the throttling groups (or if there is a spelling error), then no throttling will be performed for this thread type. -
[grinder.bf.]ingest_count_raw_metrics
-True
to create a secondary GrinderTest
object to track the total number of metrics ingested, not just the number of HTTP requests. The count is increased by the number of metrics in a given POST payload (which should be equal toingest_batch_size
), when the given request is successful. Note that this will skew the total TPS and other statistics that Grinder collects. Default is False. -
[grinder.bf.]annotations_weight
- Default is5
. Note: "annotations" here == "events" in Blueflood. -
[grinder.bf.]annotations_num_tenants
- Exactly likeingest_num_tenants
, except that this property controls the number of tenant id's for theAnnotationsIngestThread
class. This property is provided so that ingest and annotation ingest threads can be configured independently. Default is5
. -
[grinder.bf.]annotations_per_tenant
- Exactly likeingest_metrics_per_tenant
, except that this property controls the number of metric name suffixes for theAnnotationsIngestThread
class. This property is provided so that ingest threads can be configured independently. Default is10
. -
[grinder.bf.]annotations_throttling_group
- Name of an above-defined throttling group. The named tgroup will be assigned to allAnnotationsIngestThread
objects. Default isNone
. If the tgroup name is blank, or is not defined among the throttling groups (or if there is a spelling error), then no throttling will be performed for this thread type. -
[grinder.bf.]singleplot_query_weight
- Default is10
. -
[grinder.bf.]singleplot_query_throttling_group
- Name of an above-defined throttling group. The named tgroup will be assigned to allSinglePlotQuery
objects. Default isNone
. If the tgroup name is blank, or is not defined among the throttling groups (or if there is a spelling error), then no throttling will be performed for this thread type. -
[grinder.bf.]multiplot_query_weight
- Default is10
. -
[grinder.bf.]multiplot_query_throttling_group
- Name of an above-defined throttling group. The named tgroup will be assigned to allMultiPlotQuery
objects. Default isNone
. If the tgroup name is blank, or is not defined among the throttling groups (or if there is a spelling error), then no throttling will be performed for this thread type. -
[grinder.bf.]search_query_weight
- Default is10
. -
[grinder.bf.]search_query_throttling_group
- Name of an above-defined throttling group. The named tgroup will be assigned to allSearchQuery
objects. Default isNone
. If the tgroup name is blank, or is not defined among the throttling groups (or if there is a spelling error), then no throttling will be performed for this thread type. -
[grinder.bf.]annotations_query_weight
- Default is8
. -
[grinder.bf.]annotations_query_throttling_group
- Name of an above-defined throttling group. The named tgroup will be assigned to allAnnotationsQuery
objects. Default isNone
. If the tgroup name is blank, or is not defined among the throttling groups (or if there is a spelling error), then no throttling will be performed for this thread type. -
[grinder.bf.]auth_url
- URL to use to authenticate against before running the perf test. Should be the url to an OpenStack compatible identity service. -
[grinder.bf.]auth_username
- The username to authenticate with before running the perf test. -
[grinder.bf.]auth_api_key
- The API key to authenticate with before running the perf test. -
[grinder.bf.]auth_properties_path
- Path to a.properties
file that contains the user credentials. If any ofauth_url
,auth_username
, orauth_api_key
is not specified in the main config file, then this property will be checked for credentials. The property file referred to by this property will only be checked for user credentials; any other properties defined in it will not be used for any purpose, nor will this.properties
file in any way override the main config. -
[grinder.bf.]auth_properties_encr_key_file
- Path to a.properties
file that hass apassword
entry giving a simple encryption key. If this property is specified, then properties in theauth_properties_path
file can be encrypted using jasypt, e.g.ENC(abc123...)
. The property file referred to byauth_properties_encr_key_file
will only be checked for apassword
property; any other properties defined in it will not be used for any purpose, nor will this property file in any way override the main config.
For a quick, self-contained environment, run this project in Docker. Start by building the image and running it:
docker build . -t bf-perf
docker run --rm -it --name bf-perf bf-perf
Then treat it like a virtual machine that you log into with docker exec
:
docker exec -it bf-perf bash
Be sure to set Blueflood URLs as appropriate in the properties file so that the container can reach it.
Note: This image has the source files copied into it, not mounted, so changes to the source aren't reflected in your running container. If you make changes, make them in the container, or else you have to rebuild the image and start a new container.
The following command will download the necessary software packages and place them under the dependencies/
folder:
./setup-dependencies.bash
Note this needs to be run on each node in the cluster, as well as the console.
The GUI can be started with the provided script:
./run-console.bash
The console can also be run headless, with another provided script. You'll need to use this one if you're using Docker:
./run-headless-console.bash
The headless console starts in the foreground and will give a few lines of output when started successfully, including successfully starting a Jetty server. To control the headless console, you can interact with a rest api like so:
curl -X POST http://localhost:6373/agents/stop-workers
curl -X POST http://localhost:6373/agents/start-workers
Once the tests are started, the console terminal keeps outputting "collecting samples" every second, even when no tests
are currently running. Grinder writes test output to log files in resources/logs
by default (overridable in the
properties file).
If you see connection errors in the logs, make sure the blueflood URLs set in the properties file have the correct IP and port for wherever you have Blueflood running.
The graphical console gives some useful status info, so you may prefer using that.
Each agent is started with the provided script, like so:
./run-agent.bash $GRINDER_PROPERTIES_FILE
There are currently some example properties files in the properties
folder:
grinder-local.properties
has some configs for running on your localhostgrinder-unittests.properties
holds the configs used by the unit tests
The agent is headless and starts in the foreground. You'll see a few lines of output on a successful start, including a line indicating that it's waiting for the console.
There is a set of unit tests to check the function of the individual components in the scripts.
The relevant file is scripts/tests.py
.
To run the unit tests under Jython and Grinder, you can simply run ./run-unit-tests.bash
at the bash prompt.
Alternately, there is a ./run-unit-tests-with-coverage.bash
command that will run the unit tests under Python, collect code coverage numbers, and produce a report at htmlcov/index.html
.
-
Blueflood has a feature called
ENABLE_TOKEN_SEARCH_IMPROVEMENTS
. When turned on, it does additional indexing of the separate tokens that compose a metric name. This indexing powers the/v2.0/{tenant}/metric_name/search
endpoint. That endpoint isn't exercised in this project right now. Add a generator and appropriate configurations for it so that it gets tested. -
All metrics generated by this project are the same, except for the final token, like
org.example.metric.<number>
. In real life, metric names are far more varied, which could have a big impact on Blueflood, especially withENABLE_TOKEN_SEARCH_IMPROVEMENTS
turned on. Try to make this project add more randomness into the metric names to increase the total number of tokens that have to be indexed.