Skip to content

Latest commit

 

History

History
151 lines (114 loc) · 9.23 KB

README.md

File metadata and controls

151 lines (114 loc) · 9.23 KB

DEPRECATED - please see 'Bench-Swift' in: https://github.com/IBM-Swift/Kitura-Benchmarks.git instead.

A performance benchmark harness for testing web frameworks, which offers the convenience of running CPU and RSS (resident set) monitoring in the background. CPU affinity is supported on Linux.

It is essentially a small collection of bash scripts, calling out to standard tools such as sed, awk, some optional system utilities such as mpstat, and various workload drivers such as wrk.

This is firmly a work in progress and growing features as I need them. Contributions, fixes and improvements are welcome!

Requirements:

On Mac, there are no prereqs other than a suitable workload driver (see below). On Linux, you may additionally want to install:

  • mpstat (Ubuntu: sudo apt-get install sysstat)
  • numactl if not already installed (sudo apt-get install numactl)

Usage:

./drive.sh run_name [cpu list] [clients list] [duration] [app] [url] [instances]

  • cpu list = comma-separated list of CPUs to affinitize the application to (eg: 0,1,2,3)
    • only supported on Linux. This param is ignored on Mac
  • clients list = number of simultaneous clients to simulate. This can be a comma-separated list, for example to simulate ramp-up. Separate statistics will be reported for each load period
  • duration = time (seconds) for each load period
  • app = full path name to the executable under test
  • url = URL to drive load against (or for JMeter, the driver script to use)
  • instances = number of concurrent instances of app to start (default 1)

Results and output files are stored under a subdirectory runs/<run name>/

Environment variables:

The following options further influence how the script behaves:

  • CLIENT - remote client server (see 'Driving requests remotely' below)
  • DRIVER - the load driver to use (see 'Workload driver' below)
  • PROFILER - the profiling tool to run (see 'Profiling' below)

Workload driver options - for JMeter:

  • JMETER_SCRIPT - the .jmx file to use (required)
  • USER_PROPS - the user.properties file (default: jmeter/user.properties.sample)
  • SYSTEM_PROPS - the system.properties file (default: jmeter/system.properties.sample)

Workload driver options - for wrk2:

  • WORK_RATE = comma-separated list of constant load levels (rps) to drive

Output format options:

  • INTERVAL - frequency of RSS measurements (seconds), and throughput (if supported)

Customizing the drive.sh script

Before using this script, there are a number of hard-coded settings that you should review and customize for your system:

  • On Linux, you will need to adjust the affinity settings (numactl ...) appropriately for the NUMA topology of your system, or remove them if you do not want to use this facility. These have no effect on Mac.
  • A number of operating system tuning parameters will be set by the script during workload execution if your userid has sudo permission. This works best if sudo can be executed without a password. If you cannot or do not want to use these facilities, change the SUDO_PERMISSIONS setting to no.
  • On Mac, the script will attempt to disable the firewall during testing. If you do not want it to do this, change the OSX_DISABLE_FIREWALL setting to no.

Comparing multiple implementations:

./compare.sh [app1] ... [appN]

  • Wrapper for drive.sh, running a compare of multiple applications and repeating for a number of iterations. At the end of the runs, a table of results is produced for easy consumption.
  • The output from the runs is stored in compares/<date>/runs/compare_<iteration no>_<app no>/
  • The original output from drive.sh is preserved in a series of files named compares/<date>/compare_<iteration no>_<app no>.out.
  • Compares can be customized by setting various environment variables:
  DRIVER: workload driver to use (default: wrk)
  ITERATIONS: number of repetitions of each implementation (default: 5)
  URL: url to drive load against (default: http://127.0.0.1:8080/plaintext)
  CLIENT: server to use to execute load driver (default: localhost)
  CPUS: list of CPUs to affinitize to (default: 0,1,2,3)
  CLIENTS: # of concurrent clients (default: 128)
  DURATION: time (sec) to apply load (default: 30)
  SLEEP: time (sec) to wait between tests (default: 5)
  RUNNAME: name of directory to store results (default: compares/YYYYMMDD-HHmmss)
  PRE_RUN_SCRIPT: a script to run before each test (eg. cleanup / setup actions)
Output control:
  VERBOSE_TRACE: set to echo periodic CPU, RSS and throughput values in CSV format to the console
  JSONFILE: fully-qualified filename to write results to in JSON format

...in addition, the environment variables consumed by drive.sh can also be specified.

Per-implementation options may be specified. These follow the executable, comma-separated:

  instances=N: Run multiple instances of an application
  label=SomeName: Used to name the application in the output
  pwd=/some/directory: Specify the PWD for the application

Example: /path/to/myExecutable,label=MyProg1,pwd=/tmp

Example output

Output from compare.sh is in the following format:

Implementation | Avg Throughput | Max Throughput | Avg CPU | Avg RSS (kb)
---------------|----------------|----------------|---------|--------------
             1 |        38122.0 |       38609.66 |    88.6 |       793990
             2 |        33068.6 |       33508.57 |    99.9 |        53444
             3 |         3196.4 |        3196.90 |    10.1 |        42287

This is a simple summarization of the output from each run of each implementation. Further details for each measurement can be found by examining the compare_<iteration no>_<app no>.out files.

If JSONFILE is specified, the JSON-formatted output will be written to this file in addition to the normal output described above. The JSON output always contains the detailed trace information: periodic RSS, CPU and Throughput measurements, and CPU time consumed.

Regenerating the output from a previous compare

As a convenience, to regenerate the output from a previous compare, use: ./recompare.sh <date> where <date> is a directory containing the original output (eg. compares/<date>).

Recompare will automatically determine which applications you compared. However, if you set any environment variables for your original compare (such as changing the number of iterations), you must set them to the same values when running recompare. This is useful if you want to fix or workaround a problem with one of the original post-processing scripts.

Workload driver

You can set the environment variable DRIVER to either

By default, 'wrk' is used to drive load. Ensure that the command is available in your PATH.

Also, a couple of 'wrk' variants:

  • wrk-nokeepalive - sets the Connection: close header when talking to the server
  • wrk-pipeline - enables HTTP pipelining of requests

Driving requests remotely

You can set the environment variable CLIENT to the hostname of another server which will act as the client driver. In this case, you must also set the URL parmeter to be appropriate from the client server's perspective. As an example:

  • myserver.mydomain.com - system under test (execute the driver script here)
  • myclient.mydomain.com - system to use as a client
  • env CLIENT=myclient ./drive.sh ..... http://myserver:8080/plaintext

When CLIENT is set, the driver command will be issued via ssh -t $CLIENT. In order for this to work, you must allow myserver to issue SSH commands on myclient with the same userid and without prompting for a password. This can typically be achieved through the use of public key authentication. Also, you must ensure that the workload driver wrk is installed and present on your PATH on the client system.

Profiling

Various profiling options are provided for Linux (you must install these tools as appropriate on your system, all are readily available through the package manager).

To enable, set environment variable PROFILER to one of the following:

  • valgrind - produces a report of memory leaks using massif
  • oprofile - profiles the application only, using 'oprofile'
    • The report is generated in two formats: plain text (flat and callgraph), and an XML format.
  • oprofile-sys - profiles the whole system (requires sudo)
  • perf - profiles the application only, using 'perf', generating a flat profile.
  • perf-cg - two reports are generated: a flat profile, and a detailed call graph.
  • perf-idle - profiles scheduler to get insight into why the application is idle (requires sudo)
    • this is experimental and requires kernel debug symbols installed to /usr/lib/debug/boot/vmlinux-'uname -r'

Installing profiler prerequisites

Profiling with perf requires the linux-tools and kernel dbgsym packages to be installed:

sudo apt-get install linux-tools-`uname -r` linux-image-`uname -r`-dbgsym