The CustomChecks project implements checks for packages and projects that can be provided individually as UserPyModules and thus extend the Built-In-Checks that are delivered directly with ecu.test. All checks are available as open source and can be used as a basis for project specific CustomChecks.
ecu.test is a test automation software for the validation of embedded systems in automotive environments developed by
tracetronic GmbH.
This software executes regression tests which are essential for validating complex technical products such as electronic
control units (ECUs).
It supports standardized access to a broad range of test tools and provides automation of distributed test
environments (SiL – MiL – HiL – vehicle).
The tracetronic CustomChecks project is part of
the Automotive DevOps Platform by tracetronic. With
the Automotive DevOps Platform, we go from the big picture to the details and unite all phases of vehicle software
testing – from planning the test scopes to summarizing the test results. At the same time, continuous monitoring across
all test phases always provides an overview of all activities – even with several thousand test executions per day and
in different test environments.
Please consider other open-source solutions by tracetronic, especially the ecu.test Execution Plugin and the CX Templates.
The recommended way to install the CustomChecks for ecu.test is by putting it into a library workspace, ensuring maximum flexibility.
- Clone this repository
- Add the path to the cloned repository to the Manifest:
You should now be able to use the CustomChecks (see First Check Run). Note that, upon first execution, the configuration file will be copied into the Parameters/CustomChecks folder of your main ecu.test workspace.
You can change the namespace and the name of the CustomChecks library within ecu.test by modifying the manifest file.
Alternatively, you can put the CustomChecks folders under UserPyModules and Parameters into
the respective folders in your main ecu.test workspace. If ecu.test was already running, press Ctrl + Shift + R
to reload the UserPyModules.
Create an empty ecu.test package and click on the check button:
A notification box should appear:
Congratulations, you just completed your first check run successfully. 😃
The CustomChecks consist of two major parts:
- the implementation scripts for the checks (under the UserPyModules folder) determining what should be checked - these always start with Check...
- a configuration file with which you control if and how the checks are executed
For a customized usage of the delivered checks, you need to adjust the configuration file according to your needs. The configuration file will be copied to Parameters/CustomChecks/config.yaml in the main workspace when first executing the CustomChecks. For a simple usage, this is basically the only thing you need to customize. For more involved features (such as writing your own checks), see Customization and Extension. You can automate the checks using the ecu.test REST-API and COM-API, see Workflows.
With CustomChecks, you can check both ecu.test Packages and Projects. This repository contains checks for
- package attributes
- desired attribute is set
- attribute value follows regex pattern
- attribute value is one of a limited set of choices
- project attributes (similar to package attributes)
- general information about the package, such as description length and existence of required parameters
- package name
- with specified regex pattern
- package variables
- variable name follows regex pattern
- allowed and disallowed package content (such as test steps)
Further features:
- enable/disable certain checks completely
- specify the message which should be printed if a check fails
- set conditions to determine when a check should be executed
- define your own checks with your own custom configuration options, or extend existing ones
The configuration determines the specifics of the check executions. A configuration item for one specific check has the following structure:
<CheckName>:
Enabled: <true/false>
<PackageType>
Conditions:
<PackageFeature>
<Key>: <Value>
Parameters:
<ParameterName>
<Key>: <Value>
The different entries shall be explained in the following:
Element | Explanation |
---|---|
<CheckName> | the name of the check, for instance CheckPackageAttributes |
Enabled | enable/disable this check (true or false) |
<PackageType> | determines the type of package to be checked (e.g. CheckTestCase for test case package, and CheckLibrary for library packages) |
Conditions | the conditions under which this check is executed are grouped under this element |
<PackageFeature> | the specific feature(s) of the package which is taken for the evaluation of the condition - for instance PackageName (only packages with a certain name are checked) |
Parameters | This is the main part of the check, where you determine what is checked. |
The <CheckName> element corresponds to the equally named Python module. For more information, please refer to the documentation within the configuration file.
For a manual execution of the CustomChecks for a specific package or project, simply click on the button shown under First Check Run. There, you can also choose whether the whole package tree should be checked recursively.
For automation purposes, such as usage within a CI, you can use the ecu.test REST-API. You can find the REST-API Swagger documentation of you local ecu.test instance under http://127.0.0.1:5050/api/v1/ui/. With the /checks endpoint, you can create check execution orders and retrieve information about the result of the checks. The following code snippet shows an example in Python:
import requests
import time
BASE_URL = "http://127.0.0.1:5050/api/v1"
CHECKS_ENDPOINT = f"{BASE_URL}/checks"
PKG_PATH = "Package.pkg"
if __name__ == "__main__":
#----- Send execution order -----
headers = {"accept": "application/json", "Content-Type": "application/json"}
data = f'{{"filePath": "{PKG_PATH}"}}'
response = requests.post(url=f"{CHECKS_ENDPOINT}/executions", headers=headers, data=data)
if (response.status_code > 399):
print("[ERROR] Posting execution order failed.")
exit(1)
checkExecutionId = response.json().get("checkExecutionId")
#----- Poll the status of the order -----
max_tries = 10
headers = {"accept": "application/json"}
while (max_tries > 0):
response = requests.get(url=f"{CHECKS_ENDPOINT}/executions/{checkExecutionId}",
headers=headers)
if (response.status_code > 399):
print("[ERROR] Polling status failed.")
exit(1)
if response.json().get("status") == "FINISHED":
break
time.sleep(2**(10-max_tries))
print("tick...")
max_tries -= 1
if max_tries == 0:
raise TimeoutError("[ERROR]: Execution order was not processed within the permissible "
"timeout.")
#----- Set exit status depending on the result -----
headers = {"accept": "application/json"}
response = requests.get(url=f"{CHECKS_ENDPOINT}/executions/{checkExecutionId}/result",
headers=headers)
if response.status_code > 399:
print("[ERROR] Getting execution order result failed.")
exit(1)
issues = response.json().get("issues")
if not issues:
exit(0)
print("The following issues were found:")
print(issues)
exit(1)
In your CI, the exit code can be used, for instance, to pass or fail the pipeline.
A similar functionality can be achieved over the COM-API COMPackage endpoint (Check(), CheckNG()).
A check comprises three parts:
- a Python script with the name pattern "Check" + item type + check description, for instance CheckPackageAttributes.py
- the config.yaml where the checks are configured and included/excluded
- the ConfigKeys.py mediating between entries in the config.yaml and Python scripts, where these entries should be used as variables
There are already predefined scripts and configurations in the checks folder, which can be used as an orientation when writing new checks. There are also several predefined keys in the ConfigKeys.py, which can simply be used in the checks, but can also be extended with new keys.
The anatomy of a check script is as follows:
### (1) ###
from typing import List
from .api.CheckResult import CheckResult # used if a check error was found
from .api.AbstractPackageCheck import AbstractPackageCheck # for a package check
from .helper.Configuration import Configuration # read the config.yaml
from .helper.CheckType import CheckType # enum for the type of check
from .helper.ConfigKeys import ParameterKeys as pk # the keys available for usage in the script - can be extended
from .helper.CheckResultWithMessage import check_result_with_message # used like CheckResult, but with custom message
# some ecu.test imports
try:
from tts.core.logging import SPrint, WPrint, EPrint
from tts.core.api.internalApi.Api import Api
api = Api()
except:
from logging import info as SPrint, warning as WPrint, error as EPrint
### (2) ###
# module type: mandatory
MODULE_TYPE = CheckType.PACKAGE.value
### (3) ###
# naming scheme: "Check" + item type + description - must inherit from Abstract[item type]Check
class CheckPackageMyExampleCheck(AbstractPackageCheck):
"""
Check Package MyExampleCheck
========================
Description
-----------
Add description.
Instructions:
-----------
Give some instructions on how to configure this check with the config.yaml
Return messages:
---------------------
- Possible return messages of the check, displayed in ecu.test
Limitations
-----------
...
"""
### (4) ###
# needs to be there, leave untouched
def __init__(self, internalApi):
"""
Constructor to load the check parameters from config.yaml
"""
super().__init__()
# needs to be there, leave untouched
def GetName(self) -> str:
"""
Name to be shown in UI and used in the config.yaml
"""
return type(self).__name__
### (5) ###
# This is the main method of the check and must be there. You can either write the check directly
# into this method, or define further methods (see below) to be called here.
# test_item: Package, Project, AnalysisPackage
# parameters: the Parameters from the config.yaml, belonging to this check
def check(self, test_item, parameters) -> List:
return self.check_something(test_item, parameters)
### (6) ###
def check_something(self, package, parameters):
"""
Checks something
"""
### (7) ###
# init clean check result list
checkResults = []
### (8) ###
package_name = package.GetName() # Package name from Object API
### (9) ###
if package_name != parameters[pk.HARDCODED_NAME]:
checkResults.append(CheckResult("Package name is not valid."))
### (10) ###
package_version = package.GetVersion() # version from Object API
### (11) ###
if package_version != parameters[pk.VERSION]:
msg = parameters.get(pk.CUSTOM_MESSAGE)
default_msg = f"Package version {package_version} invalid."
# if CUSTOM_MESSAGE is set, use it, otherwise use default message
checkResults.append(check_result_with_message(parameters, msg, default_msg))
return checkResults
The script must be in the CustomChecks folder. The file is structured like this (see numbered comments):
- necessary imports
- CheckResult and CheckResultWithMessage are needed to add results for unsuccessful checks
- AbstractPackageCheck: for packages - for projects, import AbstractProjectCheck from .api.AbstractProjectCheck
- CheckType: necessary to define MODULE_TYPE (mandatory!)
- ParameterKeys: the .yaml keys from the config.yaml mapped to corresponding variables
- Configuration: parses the config.yaml
- MODULE_TYPE is mandatory and depends on the test item to be checked
- the class name of the check - must be the same as the corresponding top-level item in the config.yaml, and should be the same as the Python module name
- constructor and GetName method: must not be changed
- check(self, test_item, parameters): returns the check results
- method signature must not be changed
- may directly implement the check, or alternatively may call other methods of the class, e.g. in multi-step checks
- the method containing most of the check logic
- checkResults is an array of type CheckResult
- the ecu.test Object API is called on the Package class
- a check which accesses the config key 'HardcodedName' as Python variable - if unsuccessful, append check result
- the ecu.test Object API is called on the Package class
- check with custom message
A corresponding configuration in the config.yaml might look like this:
CheckPackageMyExampleCheck:
# Checks the attributes of a package depending on the conditions
Enabled: true
CheckTestCases: # check for test case package
Conditions:
ProjectName:
RegexPattern: '^.*'
Parameters:
HardcodedName: 'example'
Version: '3'
CustomMessage: 'Something went wrong...'
Notice that in this example, we have used the config key HardcodedName, which is not by default set to be parsed. Add the entry
HARDCODED_NAME = 'HardcodedName'
to the ParameterKeys class in the ConfigKeys.
You can change your scripts while keeping ecu.test running. Just reload the UserPyModules by pressing Ctrl + Shift + R
within ecu.test after each change.
Yo do not need to reload anything when simply modifying the config.yaml.
To report a bug or request an enhancement to this plugin please raise a new GitHub issue.
If you have any further questions, please contact us at [email protected].
This plugin is licensed under the terms of the MIT License.
Using the cyclonedx-bom tool and poetry, you can run
poetry export -f requirements.txt -o requirements.txt
cyclonedx-py -r
to get a bill of materials.