Python client for Loadero API
Project description
Loadero-Python
Python client for managing Loadero tests.
Loadero-Python provides easy-to-use programmatic access to Loadero API. Allows to manage tests, participants, asserts, runs and other Loadero resources, start and stop tests and extract test run results. Example usage might be running Loadero tests as a part of CI/CD.
Table of Contents
Installation
Loadero-Python is available on PyPI and can be installed by running
pip install loadero-python
Usage
API Access
Before using the Loadero-Python client an API access token needs to be acquired. This can be done in the projects settings page in Loadero web-app. More information about the API can be found in the Loadero wiki
Note Access tokens are project specific and cannot be used across multiple projects. Make sure to specify the project ID in the request for an access token.
Initialization
After acquiring the access token Loadero-Python needs to be initialized with it.
Loadero-Python uses a singleton object APIClient from
loadero_python.api_client module to perform all requests to the API. Since
APIClient is a singleton it needs to be initialized once like so
from loadero_python.api_client import APIClient
APIClient(
project_id=1234,
access_token="your_access_token",
)
Further examples will not include APIClient initialization. It is assumed that
the client has been initialized at an earlier step.
Working with Existing Resources
Loadero resources have a tree-like structure hence there can be child and parent resources.
project
|
|----tests----groups-----participants
| |
| |------asserts----assert_preconditions
|
|----files
|
|----runs-----results
Every parent resource can read all of its child resources.
Project class from loadero_python.resources.project module provides an entry
point to access all resources.
Project class can be imported with
from loadero_python.resources.project import Project
All tests in a project can be read with
tests, pagination, filters = Project().tests()
Notice, that project ID was not specified in this example, this is because the
APIClient has already been initialized with the project ID and corresponding
access token.
testsis a list ofTestobjects fromloadero_python.resources.testmodulepaginationis aPaginationParamsobject fromloadero_python.resources.paginationmodule.filtersis a python dictionary of applied filters.
A more detailed explanation of pagination and filters return values is
available in the Filtering and Pagination section.
Creating a Test
With an initialized APIClient Loadero-Python can now manage resources in the
project. Test is a resource one of many in Loadero-Python. More information
about all the resources that Loadero-Python provides can be found in the
Structure section. This usage guide cannot demonstrate all of
Loadero-Python's functionality hence will cover only common use case scenarios
starting with creating a test.
Test resource is contained within the loadero_python.resources.test module.
From it Test, TestParams, and Script classes need to be imported.
from loadero_python.resources.test import Test, TestParams, Script
Additionally, TestMode and IncrementStrategy classificator constant
enumerations need to be imported from loadero_python.resources.classificator.
They will be used for test attribute definitions.
from loadero_python.resources.classificator import TestMode, IncrementStrategy
Test attributes can be specified in two ways. Directly as arguments in params initialization.
test = Test(
params=TestParams(
name="my first loadero python test",
start_interval=1,
participant_timeout=10 * 60, # ten minutes
mode=TestMode.TM_LOAD,
increment_strategy=IncrementStrategy.IS_LINEAR,
script=Script(content='print("hello test script")'),
)
).create()
or with builder methods.
test = Test(
params=TestParams()
.with_name("my second loadero python test")
.with_start_interval(1)
.with_participant_timeout(10 * 60) # ten minutes
.with_mode(TestMode.TM_PERFORMANCE)
.with_increment_strategy(IncrementStrategy.IS_RANDOM)
.with_script(Script(content='print("hello test script")'))
).create()
Resource create and update operations have required and optional attributes. If a required attribute is missing the API call will fail. Loadero-Python checks if all of the required attributes have been populated before making the API call and raises an exception if one or more required attributes are missing.
For test resources, the required attributes are:
namestart_intervalparticipant_timeoutmodeincrement_strategy
After the create operation completes the test object will have a few more of
its attributes populated. Any resource attributes can be simply printed.
print(test)
This will output a JSON object representation of the resource.
{
"id": 1234,
"name": "my first loadero python test",
"start_interval": 1,
"participant_timeout": 600,
"mode": "load",
"increment_strategy": "linear",
"script": "print(\"hello test script\")",
"created": "2022-08-25 15:33:04+00:00",
"updated": "2022-08-25 15:33:04+00:00",
"script_file_id": 12345
}
Different output formats can be achieved by using to_dict and to_dict_full
resource params methods. Both methods return a python dictionary representation
of the resource. to_dict will return only the required attributes and optional
attributes if present. to_dict_full will return all attributes present.
Note to_dict will raise an exception if one or more required attribute is
missing.
import yaml
print(yaml.dump(test.params.to_dict_full()))
created: "2022-08-25 15:33:04+00:00"
id: 1234
increment_strategy: linear
mode: load
name: my first loadero python test
participant_timeout: 600
script: print("hello test script")
script_file_id: 12345
start_interval: 1
updated: "2022-08-25 15:33:04+00:00"
Running a Test
To run a test the only required attribute is test ID.
For the test object from previous examples, the test_id attribute has been
populated by create operation, so it can simply be run by calling the launch
method.
run = test.launch()
If a test ID is known it can be run directly.
run = Test(test_id=1234).launch()
All tests in a project can be run with
for test in Project().tests()[0]:
test.launch()
Polling
After a test has been launched, waiting for the test to finish can be achieved
with the poll method.
run.poll()
By default, the poll method will make an API call to check if the test
execution has finished every 15 seconds and will wait up to 12 hours. This
functionality can be customized with the interval and timeout arguments.
# will poll every 5 seconds and will wait up to 10 minutes.
run.poll(interval=5.0, timeout=10 * 60.0)
If test execution does not finish within the specified timeout, the poll
method will raise an exception.
Stopping Test Execution
If test execution needs to be prematurely stopped, it can be done with the
stop method.
run.stop()
Note stop only sends an API request that starts a Loadero procedure of
stopping the test. This process is NOT immediate. Even though the stop API
request completes relatively quickly, the test can remain running for a while
longer.
Note If another process is polling the test execution, it will automatically stop if the test is stopped.
Getting Results
After the test run finishes execution, the run object already contains many
useful attributes that may be used in result analysis. The attributes are stored
on the run.params field. run.params is an RunParams object from
loadero_python.resources.run module.
print(run.params.success_rate)
Participant Results
run object describes a result overview of the whole test. To get a more
detailed result information about each test participant's results needs to be
read.
results, _, _ = run.Results()
result = results[0]
The ignored return values are pagination and filters. They are not relevant for result retrieval, hence they are omitted. A more detailed explanation of these values is available in the Filtering and Pagination section.
results is a list of Result objects from loadero_python.resources.result
module. A single result corresponds to a single participant in the test.
Result just like a regular resource object has a params field of type
ResultParams that contains its attributes. The result resource has the largest
amount of attributes, so this showcase will cover only common use cases.
Log Retrieval
import requests
resp = requests.get(result.params.log_paths.selenium)
if not resp:
print("failed to download selenium log")
exit(1)
with open(f"selenium_log_of_result_{result.params.result_id}", "w") as f:
f.write(resp.text)
result.params.log_paths.selenium is an URL to an Selenium log. It first needs
to be downloaded using the HTTP library requests. Then it can be written to a
file.
Extracting Failed Asserts
Before extracting failed asserts. AssertStatus classificator constant
enumeration needs to be imported.
from loadero_python.resources.classificator import AssertStatus
failed_asserts = []
for result_assert in result.params.asserts:
if result_assert.status == AssertStatus.AS_FAIL
failed_asserts.append(result_assert)
Checking metrics
Loadero tests collect various different metrics from CPU, RAM, and network usage
to video and audio quality indicators. Loadero organizes these metrics with
metric base paths - a path-like string that uniquely identifies metric data. For
example, CPU usage metric data is described by the metric base path
machine/cpu/used.
After test execution finishes Loadero processes the collected metric data by applying aggregator functions.
- total
- minimum
- maximum
- average
- standard deviation
- relative standard deviation
- 1st percentile
- 5th percentile
- 25th percentile
- 50th percentile
- 75th percentile
- 95th percentile
- 99th percentile
The result is a single float value identified by a metric path. For example, the
maximal CPU usage is described by the metric path - machine/cpu/used/max
In Loadero-Python metric base paths - MetricBasePath and metric paths -
MetricPath are constant enumerations of all the available metric and metric
base paths. Contained within the loadero_python.resources.metric_path module.
To access a specific metric MetricBasePath enumeration needs to be imported.
from loadero_python.resources.metric_path import MetricBasePath
Then a specific metric can be checked like this
if result.params.metrics is None or result.metrics.machine is None:
print("result has no machine metrics")
exit(1)
if MetricBasePath.MACHINE_CPU_AVAILABLE not in result.params.metrics.machine:
print("result has no machine cpu available metric")
exit(1)
if (
result.params.metrics.machine[MetricBasePath.MACHINE_CPU_AVAILABLE].average
< 10.0
):
print("test is well configured. efficient usage of CPU resources")
The not None checks are required because some or all metrics for a result can
be missing. For example, non-WebRTC tests will not have any webrtc metrics.
Filtering and Pagination
Read-all operations have the option to limit the number of resources returned, offset a limited read-all operation by some amount of resources and filter out undesired resources.
This is done by passing a query params argument when performing a read-all operation.
QueryParams class is contained in loadero_python.resources.resource module
and can be imported with
from loadero_python.resources.resource import QueryParams
Filtering
Filters are resource-specific and are defined in each resource module. For
example, test resource filters are defined in the TestFilterKey constant
enumeration in the loadero_python.resources.test module.
TestFilterKey can be imported with
from loadero_python.resources.test import TestFilterKey
Now test read all operations can be filtered like this
tests, _, filters = Project().tests(
query_params=QueryParams().filter(
TestFilterKey.PARTICIPANT_TIMEOUT_TO, 10 * 60 # ten minutes
)
)
The ignored value is pagination. It can be ignored because pagination will contain useful page information when limit and offset query_params have been applied.
This will return tests whose participant timeout attribute is smaller than ten minutes.
filters is a python dictionary with the applied filters.
Pagination
When performing a read-all operation that will return many resources it is good practice to limit the number of resources returned and perform multiple smaller reads. This can be achieved by limiting and offsetting the number of resources returned.
tests, pagination, _ = Project().tests(
query_params=QueryParams().limit(20).offset(10)
)
This time the ignored value is filters. It can be ignored because no filters were applied.
Let's assume that the project has 28 tests numbered from 1 to 28, then this read-all operation would return tests with numbers from 11 to 28. This happens because the returned resources were offset by 10 and the next resource after the 10th is the 11th. Only 18 resources were returned because the remaining resources after offset were smaller than the defined limit - 20.
pagination is an instance PaginationParams class from
loadero_python.resources.resource module. It contains information about the
applied limit and offset, plus additional information describing how many
resources remain to be read.
Structure
The Loadero-Python library structure is similar to the Loadero API structure. The main structural components are:
- API client
- resources and operations with them
API client
Contained within loadero_python.api_client module is the APIClient singleton
object. It needs to be initialized once with the project ID and access token.
All requests to Loadero API are done with the APIClient object. It adds the
required headers to make valid API requests. Additionally, the APIClient rate
limits all requests to be compliant with Loadero API's access limits. Rate
limiting can be opted out on initialization.
APIClient(
project_id=1234,
access_token="your_access_token",
rate_limit=False,
)
Resources
Each resource has a separate module.
| Resource class | Module |
|---|---|
AssertPrecondition |
loadero_python.resources.assert_precondition |
Assert |
loadero_python.resources.assert_resource |
File |
loadero_python.resources.file |
Group |
loadero_python.resources.group |
Participant |
loadero_python.resources.participant |
Project |
loadero_python.resources.project |
Result |
loadero_python.resources.result |
RunParticipant |
loadero_python.resources.run_participant |
Run |
loadero_python.resources.run |
Test |
loadero_python.resources.test |
All resource classes have a similar structure:
- Resource classes have an attribute
paramsthat is used to store the data of a single instance of the resource. Read more about resource params here. - most resources implement common CRUD manipulation methods -
create,read,update,delete,duplicate. Some resources do not have these methods because they are impossible or not available via API access. For example,Projectresource has onlyreadmethod because API access prohibits updates to this resource. - resources that can have child resources have methods for reading all the child
resources.
# reads all groups in test groups, _, _ = Test(test_id=123).groups()
Resource Params
Every resource has a resource params class that stores the data of a single resource instance.
Resource params class for each resource is available in the resources module.
| Resource params class | Module |
|---|---|
AssertPreconditionParams |
loadero_python.resources.assert_precondition |
AssertParams |
loadero_python.resources.assert_resource |
FileParams |
loadero_python.resources.file |
GroupParams |
loadero_python.resources.group |
ParticipantParams |
loadero_python.resources.participant |
ProjectParams |
loadero_python.resources.project |
ResultParams |
loadero_python.resources.result |
RunParticipantParams |
loadero_python.resources.run_participant |
RunParams |
loadero_python.resources.run |
TestParams |
loadero_python.resources.test |
Resource params classes provide access to the resource attributes.
# read a test and print its name
print(Test(test_id=123).read().params.name)
Constants
Loadero-Python has two modules for constants.
loadero_python.resources.classificatorfor classificator constant enumerations.loadero_python.resources.metric_pathfor metric path and metric base path constant enumerations.
Resource API
Every resource has its own API class also available in the resources module.
| Resource class | Module |
|---|---|
AssertPreconditionAPI |
loadero_python.resources.assert_precondition |
AssertAPI |
loadero_python.resources.assert_resource |
FileAPI |
loadero_python.resources.file |
GroupAPI |
loadero_python.resources.group |
ParticipantAPI |
loadero_python.resources.participant |
ProjectAPI |
loadero_python.resources.project |
ResultAPI |
loadero_python.resources.result |
RunParticipantAPI |
loadero_python.resources.run_participant |
RunAPI |
loadero_python.resources.run |
TestAPI |
loadero_python.resources.test |
Resource API class implements all the available API operations of that resource. Resource API classes are internally used by all resources, but the class on its own is not very useful.
Contributing
Found a bug? - Feel free to open an issue.
Would like to request a feature? - Open an issue describing the request and the reason for it or contact Loadero support.
Want to contribute? - Open issues is a good place where to find stuff that needs to be worked on.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file loadero_python-1.2.0.tar.gz.
File metadata
- Download URL: loadero_python-1.2.0.tar.gz
- Upload date:
- Size: 66.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.11.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3476dcafa949f38114fdfdb21cedd3f5b19b47a984a8f5ca1edeafde7293f8f0
|
|
| MD5 |
da14560992d2a80041c38bf68b2ab530
|
|
| BLAKE2b-256 |
2a2fab9d78ff61bde6ad055fb7df4e12cf082b4c23b4b4f6b4b159a04ab5f59c
|
File details
Details for the file loadero_python-1.2.0-py3-none-any.whl.
File metadata
- Download URL: loadero_python-1.2.0-py3-none-any.whl
- Upload date:
- Size: 70.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.11.4
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e61710586ba1a29279b8292800a92af91f073f363155176c2935013cf3361964
|
|
| MD5 |
49431a732dbe73593b42d8a1191cf746
|
|
| BLAKE2b-256 |
2662dab081703b4a1a26f3859fdbc95394cceb7d255a8ac08b01ea9e16334cd0
|