As noted in Setting load profiles with executors, this particular executor relegates the control of VUs and the running state of tests to external processes. Feel free to use Bash, Python, or some automation component; the source of these processes is of no consequence for the executor.
The focus of the executor will be to set up the test scenario and provide constraints on the overall duration and allowable number of virtual users. From this point, a running test will be in somewhat of a holding pattern waiting for further instructions.
Interaction with the running test utilizes either the REST APIs exposed by the k6 process, or by using the k6
command line interface (CLI).
For our exercises, we're going to start by using a very basic script which simply performs an HTTP request then waits three seconds before completing the test iteration. We're providing some console output as things change.
The configured options
will be the absolute minimum required for the externally-controlled
executor.
Create a file named test.js with the following content:
import http from 'k6/http';
import { sleep } from 'k6';
export const options = {
scenarios: {
k6_workshop: {
executor: 'externally-controlled',
duration: '5m',
},
},
};
export default function () {
console.log(`[VU: ${__VU}, iteration: ${__ITER}] Starting iteration...`);
http.get('https://test.k6.io/contacts.php');
sleep(3);
}
Open a terminal window within the same directory as the test.js
script. We're now going to execute our script using the k6 binary:
k6 run test.js
k6 should now start. You should see a timer counting up as the test is running, but nothing much happening. We're not seeing our expected console message included in our script! We'll stay in this holding pattern until the timer reaches the configured duration
from the script options.
No tests were actually performed due to there being 0
virtual users (VUs) started by the script. k6 waits for the external process to scale up VUs. To scale up, we're going to use the 'k6' commandline.
Let's open another terminal window, this time however, the directory should not matter. If your script timed out in the meantime, start it back up again using k6 run test.js
.
k6 scale --vus 2 --max 10
☝️ If you don't specify the
maxVUs
in your script options, any request to scale up will fail unless you provide a--max
with the scale request!
Now that VUs have been scaled up, you should now see console output showing that each virtual user is now executing the test code.
Continuing with the currently running test from the previous step, let's play with other options available to control the overall test execution.
Using an idle terminal window (one not currently running the k6 test), we'll issue the following commands:
# Halt the currently running test; console output should stop
k6 pause
# Go ahead and scale up the desired number of VUs
k6 scale --vus 5
# Resume the test; output should confirm the test is running with additional VUs
k6 resume
At any time, you can use the k6
command to inquire as to the state of a running instance:
$ k6 status
status: 7
paused: "false"
vus: "5"
vus-max: "10"
stopped: false
running: true
tainted: false
While your script is running, whether paused or active, you can poll the current metrics from the running script. This will dump the current metrics in YAML format to your console which can be piped to an output file.
$ k6 stats
...
- name: http_req_duration
type:
type: trend
valid: true
contains:
type: time
valid: true
tainted: ""
sample:
avg: 60.06574999999998
max: 71.202
med: 59.515
min: 55.203
p(90): 63.559000000000005
p(95): 65.21095
...
If you wish to complete a test before the duration
timeframe has been met, you will have to use the Ctrl+C
keyboard command in the terminal running your test or use the REST API:
curl -X PATCH \
http://localhost:6565/v1/status \
-H 'Content-Type: application/json' \
-d '{
"data": {
"attributes": {
"stopped": true
},
"id": "default",
"type": "status"
}
}'
The
k6
CLI does not provide an equivalentk6 stop
command. Simply useCtrl+C
to end a test.
Our initial script provides the bare minimum to begin your test.
Consider a best-practice of specifying your maxVUs
within the script. As noted previously, the maxVUs
is not required, however any attempt to scale will be met with an error unless a --max
is specified with the initial scaling request. The maxVUs
value may be overridden using the --max
argument should the controlling script deem necessary.
The final script option is the vus
setting. When provided, this number of VUs will begin processing once the script is started thereby eliminating the need for an initial scale up.
Let's update the options declaration in our test.js script to include these additional settings:
export const options = {
scenarios: {
k6_workshop: {
executor: 'externally-controlled',
duration: '5m',
maxVUs: 10,
vus: 2,
},
},
};
Running our script now will immediately show that our tests are being executed by 2 virtual users:
$ k6 run test.js
/\ |‾‾| /‾‾/ /‾‾/
/\ / \ | |/ / / /
/ \/ \ | ( / ‾‾\
/ \ | |\ \ | (‾) |
/ __________ \ |__| \__\ \_____/ .io
execution: local
script: scripts/test.js
output: -
scenarios: (100.00%) 1 scenario, 10 max VUs, 5m0s max duration (incl. graceful stop):
* k6_workshop: Externally controlled execution with 2 VUs, 10 max VUs, 5m0s duration
INFO[0000] [VU: 4, iteration: 0] Starting iteration... source=console
INFO[0000] [VU: 1, iteration: 0] Starting iteration... source=console
INFO[0003] [VU: 1, iteration: 1] Starting iteration... source=console
INFO[0003] [VU: 4, iteration: 1] Starting iteration... source=console
INFO[0006] [VU: 1, iteration: 2] Starting iteration... source=console
INFO[0006] [VU: 4, iteration: 2] Starting iteration... source=console
So that's it! Hopefully you see the additional power in being able to control your k6 load test from external source!