- The payload data type for POST requests is
application/x-www-form-urlencoded
instead ofapplication/json
. - You can specify a list of custom user ids to be used by the
generator
mode instead of random ones (see Simon Eismann'smicroservices-demo
). - The
-c
option indirector
mode allows the choice between 2 power communicator implementationshipstershop
anddocker
.- The
hipstershop
mode is meant to be used for themicroservices-demo
applications and its power daemons. The addresses of the daemons in-p
are comma-separated values<IP>:<PORT>,<IP>:<PORT>
. - The
docker
mode requests resource utilization from a docker API over HTTP. The addresses of the daemons in-p
are comma-separated values<IP>:<PORT>:<CONTAINER_ID>,<IP>:<PORT>:<CONTAINER_ID>
.
- The
# https://serverfault.com/questions/517906/how-to-expose-a-unix-domain-socket-directly-over-tcp
# creates a proxy to forward TCP (HTTP) traffic (e.g., port 12345) to the docker API
sudo socat TCP-LISTEN:12345,reuseaddr,fork UNIX-CLIENT:/var/run/docker.sock &
Download the binary here. Also, please consider citing us.
The HTTP Load Generator is a load generator designed to generate HTTP loads with varying load intensities. It uses load intensity specifications as specified by LIMBO to generate loads that vary in intensity (number of requests per second) over time. The load generator logs application level data and supports connecting to external power measurement daemons. It specifies the http requests themselves using LUA scripts, which are read at run-time.
The HTTP Load Generator tool can be run in two modes: As director and as load generator. The director mode starts the tool in a mode where it parses necessary profiles and scripts, connects to power measurement devices, and collects data. The load generator mode receives instructions from the director and generates the actual requests. The mode is set using command line switches, which means that at least two instances of the HTTP Load Generator must be running for a test, one in each mode. The HTTP Load Generator also supports multi-node load generation, where one director connects to multiple instances running in load generator mode.
Structure of this README:
- Features and Application Scenarios
- Getting Started with the Load Generator
- Creating Custom Request Profies
- Using Power Daemons
- All Command Line Switches
- Results
- Cite Us
The load generator has the following three primary features:
- Generation of loads with a varying load intensity (number of requests per second).
- Generation of loads with context sensitive requests that can be scripted to be dependent on previous responses.
- An infrastructure that allows for collection of power measurements from external measurement devices (optional).
The load generator can be used for testing of web applications regarding testing of energy efficiency, testing of scaling behavior over time, and testing of time-dependent scenarios, such as the effect of sudden bursts, seasonal variations in request patterns, trends, etc.
First build or download the httploadgenetor.jar. Deploy the httploadgenerator on two machines:
- The director machine (experiment controller): Usually your PC. This machine must have access to the load profile and request script to be run. In addition, this machine must be able to communicate with the power meters (optional, we are not using a power meter in this getting started section).
- The load generator machine: The machine that sends the network loads. Usually a quite powerful machine. No additional files, except for the jar itself, are required on this machine. You may also choose to use multiple load generation machines if you do not have a single machine with sufficient power or network capabilities.
In addition to the jar, you need a load intensity profile and a LUA script for generating the actual requests. We provide an example for each in the examplefiles directory:
- Example Load Intensity Profile: An example load intensity profile that runs for one minute. Its arrival rate increases for 30 seconds before decreasing again in a sinoid shape.
- Example Minimal Request Generation LUA Script: A minimal example script for generating requests. Alternates between calls on index.html and index.htm. For a more complex example see here. This second scripts specifies calls for the Dell DVD Store. However, use the minimal script for now. It is far easier to run with almost any web application.
Download both files and place them on the director machine. For simplicity, we will assume that you place them in the same directory as the httploadgenerator.jar. You would now also modify the LUA script with calls for your web application. For the minimal example, just make sure that index.html and index.htm are accessible and enter the adress of your server hosting the web application in line 7 of the script.
Now, on the load generator machine start the HTTP Load Generator in load generator mode, using the following command line with the -l switch. If you use multiple load generator machines, do this on each of them:
$ java -jar httploadgenerator.jar loadgenerator
Next, on the director machine start the HTTP Load Generator in director mode:
$ java -jar httploadgenerator.jar director --ip IP_OF_THE_LOAD_GENERATOR_MACHINE --load curveArrivalRates.csv -o testlog.csv --lua http_calls_minimal.lua
The director call does the following:
director
starts the director mode.--ip
specifies the address of the load generator machine. For multiple load generators, use a comma delimiter (no white spaces!).--load
specifies the load intensity (arrival rate) profile.-o
specifies the name of the output log, containing the results.--lua
specifies the LUA script.
The director will now connect with the load generator, send the load intensity profile, script, and other settings. It will then prompt you to press enter to start the test. Once the test has concluded, the output log file will appear in the directory.
Since you don't always want to be running our example profiles, you can specify your own. We specify the load intensity (arrival rate) and the requests separately in separate files.
The easiest way of creating a load intensity profile is using the LIMBO tool. LIMBO is an Eclipse plugin for modeling arrival rates. You can model the arrival rates using a graphical editor and then export them to various formats. The HTTP Load Generator supports two of those formats:
- Simple Arrival Rate File (.txt)
- Simple Arrival Rate File (.csv)
To get files of this format, right click on a .dlim model's file in the Eclipse Package Explorer, then select Generate Time Stamps. A dialog with export options appears and our two supported options should be among them. In the following dialog, you get to pick the sampling interval, this is the interval at which LIMBO samples the arrival rate and is also the interval at which the HTTP Load Generator re-adjusts the arrival rate and reports results. It can be freely configured, but an interval of 1 is recommended.
If you do not wish to use LIMBO (you should, though) you can also speficy the arrival rates manually. The format of the file is a simple CSV format with the first column being the middle time stamp of each scheduling interval and the second column being the load intensity. Intervals must always have the same step (e.g. always increment by 1) and you may not skip any!
Example:
0.5,10
1.5,20
2.5,30
Note, that the time stamp is always the middle of the interval. Meaning that it is 0.5, 1.5, ... instead of 0, 1, ... This is for compatibility with LIMBO, where this design decision makes more sense. Again, intervals with a period of 1 second (0.5, 1.5, 2.5, ...) are recommended.
The requests are specified using a LUA script. We recommend modifying one of the examples, such as the minimal example or the Dell DVD Store example. The examples contain explanations in their code comments.
Two LUA functions in the script are called from by HTTP Load Generator:
- onCycle(): Is called at the beginning of each call cycle. No return value is expected. Initialize all global variables here. Note that math.random is already initialized using a fixed seed (5) for reproducibility.
- onCall(callnum): Is called for each HTTP request. Must return the URL to call. This function is called with an index for the call, starting at 1 (LUA convention). The index increases with each call and resets once onCall returns nil.
You can parse the HTTP response in the onCall function using regular expressions. We provide HTML helper functions (considering the response is usually html). Specifically, we offer:
- html.getMatches( regex ) : Returns all lines in the returned text stream that match a provided regex.
- html.extractMatches( prefixRegex, postfixRegex ) : Returns all matches that are preceeded by a prefixRegex match and followed by a postfixRegex match. The regexes must have one unique match for each line in which they apply.
- html.extractMatches( prefixRegex, matchingRegex, postfixRegex ) : Variant of extractMatches with a matching regex defining the string that is to be extracted.
Note that all regular expressions are passed directly to the Java backend. They must be specified, as if they were specified directly in the Java code. I.e., use "\\" instead of a single "\".
URLs returned by onCall are called using HTTP GET. To send a HTTP POST request, prepend [POST] (including the brackets) before the returned URL.
You can test your LUA scripts using our HTTP Script Tester (download the binary here). The HTTP Script Tester is a graphical application that runs the script and renders HTML responses in a graphical web view to check for correct functionality of the script. Lauch the script tester using your graphical user interface or using the following command line:
$ java -jar httpscripttester.jar ./MYSCRIPTFILE.lua
We recommend a command line lauch, as the command line in the background will show potential errors and LUA print statements, which is very helpful when debugging.
The HTTP Load Generator supports connecting to power analyzer daemons. The general idea behind the infrastructure is to connect to a network daemon that may run on a separate machine with the power analyzer. Unfortunately, most power analyzer daemons, such as the SPEC PTDaemon, have restrictive licenses, prohibiting their use with the HTTP Load Generator or preventing us from providing them to you.
We provide a IPowerCommunicator interface in the tools.descartes.dlim.httploadgenerator.power package. Implement your own power daemon communicator against this interface. The HIOKICommunicator is a functioning reference implementation that supports ethernet-capable HIOKI power measurement devices. The TMCTLDCommunicator can be used as a further example implementation.
To start the HTTP Load Generator with your power communicator, add it to the classpath and then specify the fully quailified class name of your communicator using the -c switch of the HTTP Load Generator in director mode. Use the -p switch to specify the network address of your power daemon. You may enter multiple comma separated addresses. If you do, the director will instantiate a power communicator for each of those addresses and log its results in a separate column.
Example (with the power communicator compiled into the httploadgenerator.jar):
$ java -jar httploadgenerator.jar director --ip LOADGENIP --load myArrivalRates.csv -o myLog.csv -p PWRRDAEEMONIP:PWRDAEMONPORT -c my.fully.qualified.Classname --lua./http_calls.lua
Example (with the power communicator compiled into a separate jar):
$ java -cp "MYJAR.jar;httploadgenerator.jar" tools.descartes.dlim.httploadgenerator.runner.Main director --ip LOADGENIP --load myArrivalRates.csv -o myLog.csv -p PWRRDAEEMONIP:PWRDAEMONPORT -c my.fully.qualified.Classname --lua ./http_calls.lua
Use the -h
switch to show the help page:
Usage: java -jar httploadgenerator.jar COMMAND [<options>...]
HTTP load generator for varying load intensities.
-h, --help Display this help message.
Commands:
director Run in director mode.
loadgenerator Run in director mode.
Run java -jar httploadgenerator.jar director -h
for the director's help page:
Run in director mode.
Usage: java -jar httploadgenerator.jar director [<options>...]
Runs the load generator in director mode. The director parses configuration
files, connects to one or multiple load generators, and writes the results to
the result csv file.
--randomize-users With this flag, threads will not pick users (HTTP input
generators, LUA script contexts) in order. Instead, each
request will pick a random user. This setting can
compensate for burstiness, caused by the fixed order of
LUA calls. It is highly recommended to configure long
warmup times when randomizing users.
--wd, --warmupduration, --warmup-duration=WARMUP_DURATION
Duration of the warmup period in seconds. Warmup is
skipped if set to 0.
Default: 30
--wp, --warmupause, --warmup-pause=WARMUP_PAUSE
Duration of the pause after conclusion of the warmup
period in seconds. Ignored if warmup is skipped.
Default: 5
--wr, --warmup, --warmuprate, --warmup-rate=WARMUP_RATE
Load intensity for warmup period. Warmup is skipped if set
to < 1.
Default: 0.0
-a, --load, --arrivals, --loadintensity=ARRIVALRATE_FILE
Path of the (LIMBO-generated) arrival rate file.
Default: arrivalrates.csv
-c, --class, --classname, --powerclass=POWER_CLASS
Fully qualified classname of the power communicator. Must
be on the classpath.
-h, --help Display this help message.
-l, --lua, --script=LUASCRIPT
Path of the lua script that generates the call URLs.
Default: http_calls.lua
-o, --out, --log, --csv, --outfile=OUT_FILE
Name of output log relative to directory of arrival rate
file.
Default: default_log.txt
-p, --power, --poweraddress=POWER_IP[:POWER_PORT]
Adress of powerDaemon. Multiple addresses are delimited
with ",". No address => no power measurements.
Default: []
-r, --seed, --random, --randomseed=SEED
Integer seed for the random generator. Seed of 0 =>
Equi-distant dispatch times.
Default: 5
-s, --ip, --adress, --generator=IP
Adress of load generator(s). Multiple addresses are
delimited with ",".
Default: [127.0.0.1]
-t, --threads, --threadcount=NUM_THREADS
Number of threads used by the load generator. Increase
this number in case of dropped transactions.
Default: 128
-u, --timeout=TIMEOUT Url connection timeout in ms. Timout of 0 => no timout.
Default: 0
Additional Example:
$ java -jar httploadgenerator.jar director -s 192.168.0.201 -a ./arrivalRates/test.txt -o myLog.csv -p 127.0.0.1:8888 -c tools.descartes.dlim.httploadgenerator.power.TMCTLDCommunicator -l ./http_calls.lua
Results are written to the output CSV file. They contain the following metrics for each time interval:
- Target Time: The current time interval.
- Load Intensity: The target load intensity for the interval, as specified in the arrival rate file.
- Successful Transactions: The number of successful transactions that concluded in this time interval.
- Failed Transactions: Number of transaction that failed in this time interval. Failed transactions can have one of three causes (in descending order of likelyhood):
- Timout: The transaction was interrupted by a timout, as specified using the
-u, --timout
command line switch. - Error Code: The transaction returned an HTTP error code. Error codes are logged if the logging level is set to FINEST.
- Exception in the LUA script: Any exception in the load generator also causes a failed transaction. This can be caused if the LUA lua script expected a different response.
- Dropped Transactions: Number of dropped transactions. Dropped transactions are transactions that are never sent out. This is the case if a transaction would already have exceeded its timout time at the time it was started. Dropped transactions are usually an indicator of too few threads in the load generator or other bottlenecks in the load generation machine.
- Avg Response Time: Average response time of all transactions completed in this time interval. Note the response time only measures the time the transaction waited for a response by the server. It does not measure the queueing time at the load generator before being sent out.
- Final Batch Time: A control metric that logs the time when the las transaction of this time interval was queued up in the transaction queue.
Please condider citing us if you use the HTTP Load Generator in your work:
@inproceedings{KiDeKo2018-ICAC-PowerPrediction,
author = {J{\'o}akim von Kistowski and Maximilian Deffner and Samuel Kounev},
booktitle = {Proceedings of the 15th IEEE International Conference on Autonomic Computing (ICAC 2018)},
location = {Trento, Italy},
month = {September},
title = {{Run-time Prediction of Power Consumption for Component Deployments}},
year = {2018},
}