This repository contains a series of components of the RETORCH End-to-End (E2E) test orchestration framework. It's primary goal is to optimize E2E test execution by reducing both the execution time and the number of unnecessary Resource1 redeployment's.
NOTE: The repository is a work in progress, the initial version only made available the annotations, and currently we're migrating the orchestration generator module. Additional components will be added in future releases.
- Add the dependency
io.github.giis-uniovi:retorch-annotations
to the pom.xml of your SUT. - Add the annotations to the test classes as indicated below
- Configure the E2E test suite as indicated below
- Execute the orchestration generator and generate the pipelining-scripting code
- [TO-DO]
The RETORCH framework provides a set of custom annotations to define and manage Resources used in end-to-end testing. These annotations allow testers to group, schedule, and characterize Resources. To execute test cases using RETORCH, each test case must be annotated with at least one access mode and resource.
The tester needs to specify the access mode using the following attributes:
resID
: Resource identifier for the access mode.concurrency
: The upper bound of test cases that can access the resource concurrently.sharing
: Allows sharing the resource between multiple test cases.accessMode
: The type of access mode performed by the test case.
@AccessMode(resID = "LoginService", concurrency = 10, sharing = true, accessMode = "READONLY")
Each access mode annotation corresponds to a specific resource, which must be annotated with the following attributes:
resID
: A unique identifier for the resource.replaceable
: A list of Resources that can replace the current one.
@Resource(resID = "LoginService", replaceable = {})
The following code snippets illustrate a test case annotated with multiple Resources and access modes:
@Resource(resID = "LoginService", replaceable = {})
@AccessMode(resID = "LoginService", concurrency = 10, sharing = true, accessMode = "READONLY")
@Resource(resID = "OpenVidu", replaceable = {"OpenViduMock"})
@AccessMode(resID = "OpenVidu", concurrency = 10, sharing = true, accessMode = "NOACCESS")
@Resource(resID = "Course", replaceable = {"Forum"})
@AccessMode(resID = "Course", concurrency = 10, sharing = true, accessMode = "READONLY")
@ParameterizedTest
@MethodSource("data")
void forumLoadEntriesTest(String usermail,String password,String role){
this.user=setupBrowser("chrome",TJOB_NAME+"_"+TEST_NAME,usermail,WAIT_SECONDS);
driver=user.getDriver();
this.slowLogin(user,usermail,password);
}
The RETORCH framework provides a generator that creates the Execution Plan, along with the required pipelining and script
files for execution in a CI environment. The generation of scripts and pipelining code is based on the Access Modes
annotated within the test cases and the Resource information specified in [SUT_NAME]SystemResources.json
.
The RETORCH orchestration generator requires 4 inputs:
- The annotated E2E test cases with the RETORCH access modes into a single module Maven project.
- A file with the Resources in JSON format.
- A properties file with the Environment configuration.
- A custom
docker-compose.yml
file.
Given these inputs, the generator gives as output the necessary scripting code and the Jenkinsfile
to execute the E2E test
suite into a Continuous Integration system.
The first step is to create several folders to store the configurations and place the docker-compose.yml
in the single module project root.
The resulting directory tree might look like as:
.
├── src
├── docker-compose.yml
├── retorchfiles/
│ ├── configurations/
│ └── customscriptscode/
- The
retorchfiles/
directory would contain all the configuration files and scripting snippets that would be used to generate the pipelining code and the scripts to set up, deploy, and tear down the different Resources and TJob. Contains two subdirectories:configurations/
: stores the Resource and CI configuration files.customscriptscode/
: stores the different script snippets for the tear down, set up and environment.
- The
docker-compose.yml
in the root of the project. - The different project directories and files.
The following subsections explain how to create each configuration file and how to prepare the docker-compose.yml
file.
The Resource file must be placed in the retorchfiles/configurations/
and named with the system or test suite name, followed
by SystemResources.json
. This file contains a map with a series of Resources, using their unique ResourceID as a key. For each
Resource the tester needs to specify the following attributes:
resourceID
: A unique identifier for the Resource.replaceable
: A list of Resources that can replace the current one.hierarchyParent
: A resourceID of the hierarchical parent of the Resource.elasticityModel
: The elasticity model of the Resource, is composed by the following attributes:elasticityID
: A unique identifier for the elasticity model.elasticity
: Integer with the available Resources.elasticityCost
: Instantiation cost of each Resource.
resourceType
: String with the type of the Resource(e.g. LOGICAL, PHYSICAL or COMPUTATIONAL).minimalCapacities
: List with the Minimal Capacities required by the Resource; each Capacity is composed by:name
: String between "memory", "processor" and "storage".quantity
: float with the amount of Capacity Required.
dockerImage
: String with the concatenation of the placeholder name in the docker-compose, "[IMG:]" and the image name.
The following snippet shows an example of two Resources declared in the JSON file:
{
"userservice": {
"hierarchyParent": ["mysql"],
"replaceable": [],
"elasticityModel": {"elasticityID": "elasmodeluserservice", "elasticity": 5, "elasticityCost": 30.0},
"resourceType": "LOGICAL", "resourceID": "userservice",
"minimalCapacities": [
{"name": "memory", "quantity": 0.2929},
{"name": "processor", "quantity": 0.2},
{"name": "storage", "quantity": 0.5}],
"dockerImage": "userservice[IMG:]wigo4it/identityserver4:latest"
},
"frontend": {
"hierarchyParent": [], "replaceable": [],
"elasticityModel": {"elasticityID": "elasmodelfrontend", "elasticity": 1, "elasticityCost": 300.0},
"resourceType": "LOGICAL", "resourceID": "frontend",
"minimalCapacities": [
{"name": "memory", "quantity": 2},
{"name": "processor", "quantity": 1},
{"name": "storage", "quantity": 0.88}],
"dockerImage": "frontend[IMG:]nginx:latest"
}
}
The CI file must be placed in retorchfiles/configurations/
, namely retorchCI.properties
containing several parameters
related to the SUT and the Continuous Integration Infrastructure, these parameters are the following:
agentCIName
: the specific Jenkins agent used to execute the test suite.sut-wait-html
: state in the frontend (HTML displayed) when the SUT is ready to execute the test SUITE.sut-location
: location of thedocker-compose.yml
file used to deploy the SUT.docker-frontend-name
: ID of the container used as frontend .docker-frontend-port
: PORT on which the frontend container is available.external-binded-port
: EXTERNAL PORT where the frontend is made available (if its available).external-frontend-url
: EXTERNAL URI where the frontend is made available.testsBasePath
: Path to the Java project root.
The following snippet provides an example of how this file looks like:
agentCIName=any
sut-wait-html=<title>Hello World</title>
sut-location=$WORKSPACE
docker-frontend-name=https://sutexample-
docker-frontend-port=5000
external-binded-port=
external-frontend-url=
testsBasePath=./
The orchestration generator also requires to parametrize the docker-compose.yml
used to deploy the application by means including the
necessary environment variables in the containers names and URIs, as well as the placeholders of the images specified above.
Examples of the necessary changes in the docker-compose.yml
can consulted in the FullTeaching and eShopOnContainers repositories:
- FullTeaching:
- EshopContainers:
The RETORCH orchestration generator allows to specify scripting code/commands to be included in the generated set up, tear down, and
the environment declaration of each TJob. To include it, the tester must create the following files in retorch\customscriptscode
:
custom-tjob-setup
: Contains the custom set up code (e.g. declare some environment variable specific for each TJob) or custom logging systems.custom-tjob-teardown
: Contains the custom tear down code (e.g. save some generated outputs).custom.env
: Contains configurations and environment variables common to all TJobs.
Examples of the three snippets files can be consulted in FullTeaching Test Suite and eShopOnContainers.
Once created the different properties and configuration files, the single module directory tree might look like:
.
├── src
├── docker-compose.yml
├── retorchfiles/
│ ├── configurations/
│ │ ├── [SUTNAME]SystemResource.json
│ │ └── retorchCI.properties
│ └── customscriptscode/
├── custom-tjob-setup
├── custom-tjob-teardown
└── custom.env
Once all the files created and the docker-compose.yml
is prepared, to execute the generator we only need to create a
main class and instantiate an OrchestratinGenericToolbox
object. Calling the generateJenkinsFile()
method with the following
parameters:
packageRoute
: String that specifies the complete package name to the E2E annotated tests folder.systemName
: String that specifies the system name, must correspond with the name used in the Resources JSON file.jenkinsFilePath
: String with the location where theJenkinsfile
will be created (usually the root of the project:./
).
The following code snippet shows an example of the method invocation:
import giis.retorch.orchestration.main.OrchestrationGenericToolBox;
public class SutExampleRetorchMain {
public static void main(String[] args) {
OrchestrationGenericToolBox toolBox = new OrchestrationGenericToolBox();
toolBox.generateJenkinsfile("giis.sutexample.e2e.functional.tests", "sutexample", "./");
}
}
The generator provides four different outputs: the pipelining code, the necessary scripts to set up, tear down and execute the TJobs(/retorchfiles/tjoblifecycles
),
the infrastructure(/retorchfiles/coilifecycles
) and the different environment files of each TJob (/retorchfiles/envfiles
) :
Jenkinsfile
: located in the root of the project, contains the pipelining code with the different stages in sequential-parallel that perform the different TJob lifecycle stages./retorchfiles/tjoblifecycles
and/retorchfiles/coilifecycles
contains the set up, execution, and tear down scripts for the TJobs and infrastructure/retorchfiles/envfiles
: contains the generated custom environment of each TJob.
See the general contribution policies and guidelines for giis-uniovi at CONTRIBUTING.md.
Cristian Augusto - [email protected] - Software Engineering Research Group (GIIS) - University of Oviedo, ES
RETORCH E2E Test Orchestration framework:
- Cristian Augusto, Jesús Morán, Antonia Bertolino, Claudio de la Riva, and Javier Tuya, “RETORCH: an approach for resource-aware orchestration of end-to-end test cases,” Software Quality Journal, vol. 28, no. 3, 2020. https://doi.org/10.1007/s11219-020-09505-2 - Full Article available - Authors version - Download citation
RETORCH*: A Cost and Resource aware Model for E2E Testing in the Cloud:
- Cristian Augusto, Jesús Morán, Antonia Bertolino, Claudio de la Riva, and Javier Tuya, “RETORCH*: A Cost and Resource aware Model for E2E Testing in the Cloud”, Journal of Systems and Software, vol. 221 , pages. 112237, 2025. https://doi.org/10.1016/j.jss.2024.112237 - Full Paper available - Authors version - Download citation
This work has been developed under the TestBUS (PID2019-105455GB-C32) and project supported by the Ministry of Science and Innovation (SPAIN)
Footnotes
-
Henceforth, we will use the term "Resources" (capitalized) when referring to the ones required by the E2E test suite. ↩