This project is licensed under the MIT License - see the LICENSE file for details.
BEFTA Framework is a framework for automation of functional tests for http-based APIs. It uses Cucumber and Rest Assured frameworks and supports a BDD (Behaviour-Driven Development) approach to software development.
It provides the following functionalities and conveniences:
- A Domain Specific Language (DSL), which is specific to the domain of automation of http-based APIs, to describe the functional/behavioural requirements of APIs.
- Underlying programmatic logic to make the DSL an executable script language to execute functional tests.
- Flexible and efficient, JSON-based test data repository enabling test automators to define and maintain their test data with maximum re-use and minimum duplications.
- Fast-performing, full-range response verification, robust mechanism to assert full details of responses including response statuses, reason phrases, headers and response bodies.
- Dynamic data configuration means allowing dynamic & run-time-specific test data to be identified without any programming at all
- Custom-programmable dynamicity to allow for programmable injections into test logic
- Clearly designed way of adapting and integrating the framework into various API suites for any new functional test suit.
- Reporting, and for-diagnostic outputting
- Custom extensibility of out-of-the box DSL (Cucumber feature)
- Other features coming with Cucumber Framework wrapped.
- System Resources (Memory, Disk, CPU) - Same for a JDK 21 installation. Click here to see Oracle's reference for this.
- Java SE Development Kit 21 (JDK 21)
- Your Favourite IDE
- Gradle 8.10+ (included as wrapper)
- Install JDK 21 or higher
- Install a command line terminal application
Befta Framework uses the below environment variables:
- TEST_URL: This is the base URL of the APIs to be tested.
- IDAM_API_URL_BASE: This is the base URL of the API authenticating the users on behalf of which the API calls will be delivered.
- S2S_URL_BASE: This is the base URL of the API authenticating the client applications on behalf of which the API calls will be delivered.
- BEFTA_S2S_CLIENT_ID: Client ID of the application on behalf of which the API calls will be delivered.
- BEFTA_S2S_CLIENT_SECRET: Client Secret of the application on behalf of which the API calls will be delivered.
- BEFTA_RESPONSE_HEADER_CHECK_POLICY: This env var can be used optionally to switch the policy applied to mismatching response header values. It can be one of IGNORE, JUST_WARN and FAIL_TEST. Default behaviour is to assume FAIL_TEST.
- BEFTA_USER_AUTHENTICATION_RETRY_MAX_ATTEMPTS: This env var can be used to specify the number of retry attempts if token acquisition fails. Default to 3 attempts, if not specified.
- BEFTA_USER_AUTHENTICATION_RETRY_MAX_TIME_SECONDS: This specifies the max time to retry if token acquisition fails.
- BEFTA_USER_AUTHENTICATION_RETRY_MULTIPLIER_MILLISECONDS: This specifies the retry time after a given series of gaps after every trial.
- BEFTA_USER_TOKEN_CACHE_TTL_SECONDS: Configurable amount of time in seconds for user token.
- BEFTA_S2S_TOKEN_CACHE_TTL_SECONDS: Configurable amount of time in seconds for s2s toke.
Below are the environment needed specifically for CCD domain.
- DEFINITION_STORE_URL_BASE: Base URL of the Definition Store APIs.
- DEFINITION_IMPORTER_USERNAME: Email id of the user on behalf of which definitions will be imported to Definition Store, for automated test data preparation.
- DEFINITION_IMPORTER_PASSWORD: Password of the user on behalf of which definitions will be imported to Definition Store, for automated test data preparation.
Below are the environment needed specifically to Create Role Assignment data.
-
ROLE_ASSIGNMENT_API_GATEWAY_S2S_CLIENT_ID:S2S service token for Role Assignment service.
-
ROLE_ASSIGNMENT_API_GATEWAY_S2S_CLIENT_KEY:S2S key for Role Assignment service.
-
ROLE_ASSIGNMENT_HOST: Base URL of the Role Assignment APIs.
-
ROLE_ASSIGNMENT_USER_EMAIL: Email id of the user who can access role Assignment service.
-
ROLE_ASSIGNMENT_USER_PASSWORD: Password of the user who can access role Assignment service.
-
[[$ENV_VAR_FOR_USER]]: The ActorID, reference, assignerId fields in the json payload will have placeholders which will be parsed to get the email address of the user. Password placeholder for the user [[$ENV_VAR_FOR_USER]] should be in the format of [[$ENV_VAR_FOR_USER_PWD]] Ex: If json payload contains [[$ROLE_ASSIGNMENT_SUPER_USER]] for ActorId, reference , the corresponding environment Password variable should be ROLE_ASSIGNMENT_SUPER_USER_PWD
-
The json payload files should be under 'src/aat/resources/roleAssignments' with extension '.ras.json'
- Download a copy of BEFTA Framework (say, version 9.0.2) in a local folder, say the root directory of an IDE project in which you (will) have your automated functional tests. //TODO: fat jar release of framework
- Open your preferred terminal and change current directory to the root directory of your test automation project.
- java -cp befta-fw-9.0.2.jar uk.gov.hmcts.befta.BeftaMain 'my-feature-files/are/here, and/here, and-also/there'
This will run the test scenarios under the local folder you specify.
Test automation teams can write their simple, tiny custom Main classes to customise the the test suite launching logic.
-
Install Gradle 8.7 or higher. You can simply copy a gradle wrapper from
https://github.com/hmcts/befta-fw
. -
Add the following dependency to your build.gradle file:
testCompile group: 'com.github.hmcts', name: 'befta-fw', version: '9.0.2'
-
Add a javaExec section to wherever you want a functional test suit to be executed, like below:
javaexec { mainClass = "uk.gov.hmcts.befta.BeftaMain" classpath += configurations.cucumberRuntime + sourceSets.aat.runtimeClasspath + sourceSets.main.output + sourceSets.test.output args = ['--plugin', "json:${projectDir}/target/cucumber.json", '--tags', 'not @Ignore', '--glue', 'uk.gov.hmcts.befta.player', 'my-feature-files/are/here, and/here, and-also/there'] }
You can place this block inside the
task functional(type: Test) { ... }
of your test automation project.
Test automation teams can write their simple, tiny custom Main classes to customise the the test suite launching logic.If you see an error similar to
Failing to give add access to DeclaredFields in CucumberStepAnnotationUtils.replaceAnnotationClass. Exception in thread "main" java.lang.reflect.InaccessibleObjectException: Unable to make field private transient volatile java.util.Map java.lang.reflect.Executable.declaredAnnotations accessible: module java.base does not "opens java.lang.reflect" to unnamed module @1ca7bef1 at java.base/java.lang.reflect.AccessibleObject.checkCanSetAccessible(AccessibleObject.java:354) at java.base/java.lang.reflect.Field.setAccessible(Field.java:172) at uk.gov.hmcts.befta.util.CucumberStepAnnotationUtils.replaceAnnotationClass(CucumberStepAnnotationUtils.java:65) at uk.gov.hmcts.befta.util.CucumberStepAnnotationUtils.injectCommonSyntacticFlexibilitiesIntoStepDefinitions(CucumberStepAnnotationUtils.java:29) at uk.gov.hmcts.befta.BeftaMain.setUp(BeftaMain.java:85) at uk.gov.hmcts.befta.BeftaMain.main(BeftaMain.java:37)
you may need to add a line to the failing gradle task in youe build.gradle file allow opens via a jvmArg like in the below example
test { failFast = true testLogging.showStandardStreams = true jvmArgs = ["--add-opens=java.base/java.lang.reflect=ALL-UNNAMED"] }
- Open in your web browser the local Cucumber report file:
./target/cucumber/cucumber-html-reports/overview-features.html
We assume you will have build pipelines making gradle calls to run your automated functional tests which is the case for HMCTS Reform programme as seen in the open source repositories. However, with the simple means of test suite executions provided above, test suite executions can be integrated into build pipelines using Maven or any other tools as well. When it comes to BEFTA, test suite execution is a simple Java call to run a Main class with runtime arguments to specify where the features are, where the step implementations are and scenarios with which tags to pick up and run. You can skip all runtime arguments to this Main class, in which case the default arguments will be:
'--plugin', "json:${projectDir}/target/cucumber.json", '--tags', 'not @Ignore', '--glue', 'uk.gov.hmcts.befta.player', 'src/aat/resources/features'
The most typical use of BEFTA Framework will include running an application in local machine and executing a functional test suit against it. Running such applications in local will require application-specific setup instructions. Once such an application is setup correctly, BEFTA framework can then be used to functionally verify the behavioural requirements of any http-based API in local. Multiple applications from totally diverse domains can be setup in a local machine each having their respective base URLs, and BEFTA Framework can be used for each of them simply switching from one configuration to another to test a set of APIs.
In the case of HMCTS Reform CCD application suite, the local setup procedure is described here on the README of the ccd-docker repository. Obviously, to test a CCD application, for example, the proper setup procedure should have been completed as a prerequisite. An incomplete setup, like some users not fully and correctly configured, can cause some or all of the functional tests to fail as false positives.
The BEFTA Framework also contains the DataLoaderToDefinitionStore
as an alternative to the
DefaultBeftaTestDataLoader
. This extension can be registered using a custom TestAutomationAdapter
and will add
functionality to allow:
Example custom TestAutomationAdapter
registering the DataLoaderToDefinitionStore
but with createRoleAssignment
functionality disabled:
public class MyTestAutomationAdapter extends DefaultTestAutomationAdapter {
@Override
protected BeftaTestDataLoader buildTestDataLoader() {
return new DataLoaderToDefinitionStore(this,
DataLoaderToDefinitionStore.VALID_CCD_TEST_DEFINITIONS_PATH) {
@Override
protected void createRoleAssignment(String resource, String filename) {
// Do not create role assignments.
BeftaUtils.defaultLog("Will NOT create role assignments!");
}
};
}
If the DataLoaderToDefinitionStore
is used to load data: by default it will attempt to load CCD Roles from a file
named ccd-roles.json
stored alongside the configured test definitions folder. If this file does not exist or if it
fails to load, it will switch to use the list of CCD Roles hard coded into the DataLoaderToDefinitionStore
class.
For an example CCD Roles JSON file see the ccd-test-definitions repository.
BEFTA Framework contains definition files in both XLSX and JSON formats.
Note: The default CCD test definition files used by the
DataLoaderToDefinitionStore
are now stored in the ccd-test-definitions repository. The commands below can still be used to generate the required JSON template files when making changes inside the other repository.
Typically, CCD services under test will call BEFTA Framework code in order to load this definition data before running feature tests.
The BEFTA Framework will always load the JSON definitions in befta-fw
from the directory
src/main/resources/uk/gov/hmcts/befta/dse/ccd/definitions
, use them to create a XLSX file and import it to the
ccd definition store.
src/main/resources/uk/gov/hmcts/befta/dse/ccd/definitions/excel
will
NOT be imported to the definition store.
The XLSX may be updated, rather than the JSON directly, but for these changes to take effect, they must be applied to the
corresponding JSON files, by executing the DefinitionConverter
class provided by BEFTA Framework. This can be done by either:
-
Running
DefinitionConverter
against all XLSX filesThe
definitionsToJson
gradle task will run theDefinitionConverter
against each of the XLSX files in the directorysrc/main/resources/uk/gov/hmcts/befta/dse/ccd/definitions/excel
. This will replace the corresponding output in the following directory:src/main/resources/uk/gov/hmcts/befta/dse/ccd/definitions
../gradlew definitionsToJson
OR
-
Running
DefinitionConverter
against a single fileThis can be done by creating a run configuration within your IDE, providing the path to the main class
uk.gov.hmcts.befta.dse.ccd.DefinitionConverter
, as well as the necessary program arguments listed below.arg1: to-json | to-excel : key word to convert from excel to json or from json to excel arg2: input file path for excel document or parent jurisdiction folder for json version arg3: (Optional) output folder path for resulting json or excel file. By default will use parent folder from the input location arg4: (Optional) Boolean: true - use jurisdiction name to generate the parent folder name when converting from excel to JSON, false - use file name as the folder name
For information on how to configure Role Assignments for import see Domain Specific Environment Variables.
Below are a few github repositories into which BEFTA Framework has been successfully integrated:
Some Spring Boot Applications:
- https://github.com/hmcts/ccd-data-store-api
- https://github.com/hmcts/ccd-definition-store-api
- https://github.com/hmcts/ccd-user-profile-api
- https://github.com/hmcts/ccd-case-document-am-api
Some Node.js Applications:
- https://github.com/hmcts/ccd-case-print-service
- https://github.com/hmcts/ccd-case-activity-api
Development of an automated test scenario takes, at a high level, the below main steps:
-
Introduce a
.feature
file of any good name, and add a test scenario composed of the DSL elements available with BEFTA framework. -
Annotate the scenario with a tag starting with
S-
. This will identify the scenario uniquely to the framework. Make sure this is a unique scenario ID across the entire test suite being executed in a round. -
Introduce a test data file with a good name ending with
.td.json
. The test data file should be a marshaled json content of an object of classuk.gov.hmcts.befta.data.HttpTestData
. Make sure that the 'guid' field in the file contains exactly the same text as theS-
tag of the scenario in the.feature
file.
The json-based test data can be prepared making use of inheritance mechanism provided by the framework. The inheritance mechanism will be described under a separate heading.The test data can also contain dynamic content to be calculated in the runtime, in the form of pre-defined place-holders or json-path-like formulas to refer to the value of any request or response details, or any programmatically computed custom placeholder. The dynamic value features will be described under a separate heading.
-
Run your test scenario and correct any issues in the test data configuration until you are sure the test data is complete and correct to supply the framework with the full and precise data requirements of the scenario script in the
.feature
file. You can find more about how to debug your scenarios under the dedicated header below.
Attributes in an object represented by a JSON object in BEFTA-way are collated in the
order or calculation shown in the below diagram:
The below place-holders can be used to configure test data with request details dynamically calculated in runtime, or with response details to specify acceptable values in API responses as per behavioural requirements:
DEFAULT_AUTO_VALUE
: This is used to calculate a default automatic value in runtime for the below specific request fields:
-
uid
: When this is the attribute name, the value is dynamically injected as theid of the user
on behalf of which the test call is being delivered. -
Authorization
: When this is the attribute name, the value is dynamically injected asa 'Bearer' token issued to the name of the user
on behalf of which the test call is being delivered. -
ServiceAuthorization
: When this is the attribute name, the value is dynamically injected asa service to service token issued to the name of the client
on behalf of which the test call is being delivered. -
cid
: When this is the attribute name, the value is dynamically injected as theid of a case created just before this test call, through a default case creation call
. This is equivalent to hacing the below formulae in the value for the attribute:
${[scenarioContext][childContexts][Standard_Full_Case_Creation_Data][testData][actualResponse][body][id]}
\
-
ANY_NULLABLE
: This is used toaccept any data of any type including null or missing ones
as the value of the response detail specified. -
ANY_STRING_NULLABLE
: This is used toaccept any text data including null or missing ones
as the value of the response detail specified. -
ANY_NUMBER_NULLABLE
: This is used toaccept any numeric data including null or missing ones
as the value of the response detail specified. -
ANY_INTEGER_NULLABLE
: This is used toaccept any integral data including null or missing ones
as the value of the response detail specified. -
ANY_FLOATING_NULLABLE
: This is used toaccept any floating point data including null or missing ones
as the value of the response detail specified. -
ANY_DATE_NULLABLE
: This is used toaccept any date data including null or missing ones
as the value of the response detail specified. -
ANY_TIMESTAMP_NULLABLE
: This is used toaccept any time-stamp data including null or missing ones
as the value of the response detail specified. -
ANY_OBJECT_NULLABLE
: This is used toaccept any Object data of any type including null or missing ones
as the value of the response detail specified. -
ANYTHING_PRESENT
: This is used toaccept any data of any type but reject null or missing ones
as the value of the response detail specified. It is an alias ofANY_NOT_NULLABLE
. -
ANY_NOT_NULLABLE
: This is used toaccept any data of any type but reject null or missing ones
as the value of the response detail specified. -
ANY_STRING_NOT_NULLABLE
: This is used toaccept any text data but reject null or missing ones
as the value of the response detail specified. -
ANY_NUMBER_NOT_NULLABLE
: This is used toaccept any numeric data but reject null or missing ones
as the value of the response detail specified. -
ANY_INTEGER_NOT_NULLABLE
: This is used toaccept any integer data but reject null or missing ones
as the value of the response detail specified. -
ANY_FLOATING_NOT_NULLABLE
: This is used toaccept any floating point data but reject null or missing ones
as the value of the response detail specified. -
ANY_DATE_NOT_NULLABLE
: This is used toaccept any date data but reject null or missing ones
as the value of the response detail specified. -
ANY_TIMESTAMP_NOT_NULLABLE
: This is used toaccept any time-stamp data but reject null or missing ones
as the value of the response detail specified. -
ANY_OBJECT_NOT_NULLABLE
: This is used toaccept any Object data of any type but reject null or missing ones
as the value of the response detail specified.
Environment variables can be referenced between double curly braces like in {{SOME_ENV_VAR}}
.
Values in the runtime scenario context can be fetched and injected into designated
places by using the ${<field-reference-formal>}
notation. Users of this feature should
understand the object structure of a scenario context, which is shown in the below
diagram.
Field reference format is a square bracketed sequence [like][this][here]
of field
names, map keys or array/collection indexes.
Consider the following example:
${[scenarioContext][childContexts][S-212_Get_Case_Data][testData][actualResponse][body][events][0][id]}
This reference picks the id
field of the 0
-indexed element in the events
collection
in the body
of the actualResponse
of the testData
of the childContext
of the
api call S-212_Get_Case_Data
in the scenarioContext
of this test scenario being
worked on.
Some nice examples of use of this feature are available in the links below:
Each scenario context uses a custom value calculator function to translate certain
customised values into programmatically computed values. This function delegates its logic
to the TestAdapter object used by the framework instance executing the test suite. This
is done by calling the calculateCustomValue
method of the adapter. The default implementation
of the adapter already introduces the below custom values into the context of a scenario:
request
: shortcut for[testData][request]
requestBody
: shortcut for[testData][request][body]
requestHeaders
: shortcut for[testData][request][headers]
requestPathVars
: shortcut for[testData][request][pathVariables]
requestQueryParams
: shortcut for[testData][request][queryParams]
expectedResponse
: shortcut for[testData][expectedResponse]
expectedResponseHeaders
: shortcut for[testData][expectedResponse][headers]
expectedResponseBody
: shortcut for[testData][expectedResponse][body]
actualResponse
: shortcut for[testData][actualResponse]
actualResponseHeaders
: shortcut for[testData][actualResponse][headers]
actualResponseBody
: shortcut for[testData][actualResponse][body]
today
: short cut for the value of the current date inyyyy-MM-dd
format.today(<any-date-format>)
: short cut for the value of the current date in the specified format.now
: short cut for the value of the current system time inyyyy-MM-dd'T'HH:mm:ss.SSS
format.now(<any-time-stamp-format>)
: short cut for the value of the current system time in the specified time-stamp format.
Users of BEFTA framework can override this specific method to alter the calculation for the above custom values, or introduce new custom values needed in their specific tests data requirements.
Test data for a functional test scenario is designed to support various nice features, which rely on use of certain conventions on the way the data is configured.
Ideally, BEFTA framework identifies the expected value of a response body with a JSON payload in the form of a field-value or key-value structure. However, not all APIs will have a JSON-based response payload. APIs can return files, or plain text, or encoded binary content representing files being downloaded. What is a better payload format for an API response is obviously a matter of what kind of API it is and the general design guidelines adopted by API providers. However, the framework is designed to allow a variety of such response formats. This applies to request payload formats, as well.
Moreover, when the test data is edited by automation developers for an API, the logic of comparison of actual and expected response details may require certain switches specified per local elements of test data. For example, a test data can be configured to expect a collection of objects in a specific field of a response, but then the ordering of the response may or may not be predictable if such ordering is not part of the API contract. Or, in the same case, the automation developer may want to expect a subset of the objects she/he listed in the test data artefacts, or a superset of them. There has to be a way to provide such verification parameters to the framework per each of such points in the overall verification of an API response and functionality.
Below are some special conventions to make use of flexibilities described above in general terms.
If the response body of an API is meant to contain a JSON array at root level, the test data should follow the following convention for the body section of the expected response:
"expectedResponse": {
"body": {
"arrayInMap": [
{
"attribute1": "value1",
"attribute2": "value2",
"attribute3": "value3"
},
{
"attribute1": "...",
"attribute2": "...",
"attribute3": "..."
},
...
]
}
}
In this example, the value of the arrayInMap field is used as the actual expected value of the response body. Same convention is applicable for request bodies, as well.
Files can be specified to be contained in request bodies. This is useful for APIs uploading a file to a destination. Following is the structural convention for such a specification:
{
...
"request": {
"headers": {
"Content-Type": "multipart/form-data"
},
"body": {
"key": "file",
"filePath": "some/path/to/a/resource/in/classpath/of/the/test/suite/being/executed/name.extension"
}
}
...
}
The request content type should be multipart, and the body should contain the 2 fields as shown. While placing the actual API call, the framework will stream the file designated by the resource path, to the request body.
To specify a more detailed 'form-data' in a request payload, fields (controls) in the form along with their values can be specified as below. Note that this example specifies multiple files to be attached to the same request body.
"body": {
"arrayInMap": [
{
"key": "some-form-field",
"value": "some-value"
},
{
"key": "some-other-form-field",
"value": "some other value"
},
{
"key": "files",
"filePath": "SampleImage1.png"
},
{
"key": "files",
"filePath": "some/path/SampleImage2.png"
},
{
"key": "files",
"filePath": "some/path/SampleImage3.png"
}
]
}
If a file content is expected in a response body, the framework can be instructed to check if the actual size of the download file is equal to that of the expected one. Below is the convention for such instruction:
"__fileInBody__" : {
"fullPath" : "[[ANY_STRING_NOT_NULLABLE]]",
"size" : "679361",
"contentHash" : "[[ANY_STRING_NOT_NULLABLE]]"
}
The fullPath and contentHash fields are intended to be used in the future. So, they can for now be configured to accept any String except for null.
When a collection of objects are expected in any place in a response, that can be specified as an array of those objects in the test data. When so done in the simplest manner, the framework will compare every element in the array with the corresponding, same-indexed element in the actual response. Any extra element or missing element will be clearly identified. Any unacceptable value in any field in any element will fail the test clearly being reported. However, if the expected collection does not have to be returned in a particular order, the framework should be instructed of that with a simple object as the first element in the expected data array. That first element will be recognised by the framework as an instructive configuration and the array's second element will be treated as the first data object intended to be listed. The instructive configuration element should be as below:
"some-collection-field": [
{
"__operator__": "equivalent",
"__ordering__": "unordered",
"__elementId__": "id"
},
{
"id": "first object",
"field": "First"
},
{
"id": "second object",
"field": "Second"
},
{
"id": "third object",
"field": "First tab"
}
]
Here are the 3 fields in this convention:
-
__operator__
field: This one represents the mode of comparison of expected and actual content of the collection. It can be one ofequivalent
,subset
andsuperset
. Default isequivalent
and the field can be omitted if that's the preferred one. Subset and superset comparisons have not yet been implemented. -
__ordering__
field: This one represents whether the elements in the array are has to be compared in the order provided in the test data. It can be one ofordered
, andunordered
. Default isordered
and the field can be omitted if that's the preferred one.The ordering can be specified by setting the
DEFAULT_COLLECTION_ASSERTION_MODE
environment variable to either
ORDERED
orUNORDERED
.If not set, will revert to default of
ORDERED
. -
__elementId__
field:This one represents the field names in the data objects which form up a unique key.
In the case of unordered equivalence check, this field is of no effect.
In all other cases, the default or specified value of this instructive configuration is essential for the framework to be able to decide which object in the actual response should be compared to which one in the expected response.
If
__operator__
is set toUNORDERED
, either explicitly in test data or by settingDEFAULT_COLLECTION_ASSERTION_MODE
environment variable and the the__elementId__
field id(s) are specified, then these values are used by the framework.However, if the
__elementId__
id field is not specified, the framework automatically extracts best candidates for field(s) to uniquely identify the elements for matching static, expected test data.Any wildcard element values or data resolved at runtime are excluded as
__elementId__
candidates. Examples would be"id": "[[ANY_STRING_NOT_NULLABLE]]" "case_id": "${}${[scenarioContext][siblingContexts][F-105_Case_Data_Create_C1][testData][actualResponse][body][id]}"
An example test data file is here showing a nested use of this feature.
Befta framework allows programmatic extensibility in various ways.
- Custom main classes or runners: Automation developer can write their own main classes or runners to introduce customised logic to be executed before and after the execution of test suites.
- Custom automation adapter: A custom implementation of TestAutomationAdapter interface or an extension of DefaultTestAutomationAdapter class can be implemented and injected into the framework from a main class or runner. An example is here.
A custom implementation of Object calculateCustomValue(BackEndFunctionalTestScenarioContext scenarioContext, Object key)
in an adapter implementation can be particularly useful in introducing fully flexible verification
logic in any level of detail in responses. A good example is available here.
-
Adding customised step implementations: Automation developers can write their own step implementation classes and glue them to the feature files, just as a typical feature of Cucumber framework.
-
Other methods: Thanks to the broad range of features of Java language and platform, various other ways of introducing custom logic into test suites can be improvised. The framework doesn't bring any limitations to such flexibilities.
Test scenarios can be run in debug mode on any modern IDE. All it takes to do so is to have a simple Runner class like the example here. and run that runner class in debug mode of the IDE. Doing so, the automation developer can trace the execution of scenarios, which may many times be very convenient to quickly resolve any test issue. Runner classes will need environment variables set for them, and they can either be inherited automatically from the platform by the IDE, or can be copy-pasted into Run configurations of IDEs.
BEFTA Framework has been designed at a low level to contain components and their interactions as depicted in the below diagram.
Typical sequence of activities during the execution of test suite is as shown in the below Sequence Diagram:
The Retryable Feature is a new addition that allows you to execute tests multiple times until they pass or reach the maximum number of attempts. This is useful when you have flaky tests that fail randomly due to network issues, timeouts, or other intermittent failures.
The Default Retry Policy provides a baseline configuration for retrying scenarios in the absence of service-specific settings. This default behavior is configured using the following environment variables:
- BEFTA_RETRY_MAX_ATTEMPTS: The maximum number of retry attempts for each service (default: 1).
- BEFTA_RETRY_STATUS_CODES: A comma-separated list of HTTP status codes that trigger a retry (default: 500,502,503,504).
- BEFTA_RETRY_MAX_DELAY: The maximum delay (in milliseconds) between retry attempts (default: 1000).
- BEFTA_RETRY_RETRYABLE_EXCEPTIONS: A comma-separated list of Java exceptions that are considered retryable (default: java.net.SocketException, javax.net.ssl.SSLException, java.net.ConnectException).
- BEFTA_RETRY_NON_RETRYABLE_HTTP_METHODS: A comma-separated list of HTTP methods that should not be retried (default: ). Use "" to specify that no methods should be retried.
- BEFTA_RETRY_ENABLE_LISTENER: A boolean flag indicating whether the Retry Policy listener should be enabled (default: true).
export BEFTA_RETRY_MAX_ATTEMPTS=3
export BEFTA_RETRY_STATUS_CODES=500,502,503,504
export BEFTA_RETRY_MAX_DELAY=1000
export BEFTA_RETRY_RETRYABLE_EXCEPTIONS=java.net.SocketException,javax.net.ssl.SSLException,java.net.ConnectException
export BEFTA_RETRY_NON_RETRYABLE_HTTP_METHODS=POST,PUT
export BEFTA_RETRY_ENABLE_LISTENER=true
The feature can be defined with an annotation as follows: @Retryable(maxAttempts=3, delay=1000, statusCodes={400, 502}, match={".error."}). This annotation specifies:
- statusCodes: A mandatory list of HTTP status codes that trigger a retry. If not provided, the scenario will fail with a FunctionalTestException.
- maxAttempts: An optional parameter specifying the maximum number of retry attempts. Defaults to 3 if not provided.
- delay: An optional parameter specifying the delay between retry attempts in milliseconds. Defaults to 1000 milliseconds if not provided.
- match: An optional parameter that accepts an array of regex patterns. If any of these patterns match the response content, a retry will be triggered.
To use the Retryable Feature, you need to annotate your test scenarios with the @Retryable annotation in your feature file and provide the necessary parameters. Here's an example:
@S-096.1 @Retryable(maxAttempts=3, delay=500, statusCodes={409, 500}, match={".*error.*", ".*timeout.*"})
Scenario: Sample Scenario
Given given_context
When when_context
And and_context
In this example:
- The scenario @S-096.1 will be executed up to 3 times with a delay of 500 milliseconds between each attempt.
- The test will be retried if the HTTP response status code is either 409 or 500, or if the response content matches any of the provided regex patterns (".error." or ".timeout.").
Here are some examples of how you can use the Retryable Feature with the match attribute:
@S-096.1 @Retryable(statusCodes={500, 502}, match={".*server.*"})
Scenario: Sample Scenario
Given given_context
When when_context
And and_context
In this example:
- The scenario @S-096.1 will be executed up to 3 times with a delay of 1000 milliseconds between each attempt.
- The test will be retried if the HTTP response status code is either 500 or 502, or if the response content matches the regex pattern ".server.".
@S-096.1 @Retryable(statusCodes={404, 503}, maxAttempts=5, match={".*not found.*"})
Scenario: Sample Scenario
Given given_context
When when_context
And and_context
In this example:
- The scenario @S-096.1 will be executed up to 5 times with a delay of 1000 milliseconds between each attempt.
- The test will be retried if the HTTP response status code is either 404 or 503, or if the response content matches the regex pattern ".not found.".
@S-096.1 @Retryable(statusCodes={400,502}, delay=500)
Scenario: Sample Scenario
Given given_context
When when_context
And and_context
In this example, the scenario @S-096.1 will be executed up to 3 times with a delay of 500 milliseconds between each attempt. If the HTTP response status code is either 400 or 502, the test will be retried.
@S-096.1 @Retryable(maxAttempts=2, delay=500)
Scenario: Sample Scenario
Given given_context
When when_context
And and_context
In this example, the scenario @S-096.1 will fail with FunctionalTestException:Missing statusCode configuration in @Retryable because the statusCodes parameter is mandatory and was not provided.
The Befta framework now supports delaying the flow before or after an HTTP call using a new feature. This can be useful in scenarios where you need to wait for a specific amount of time before making an HTTP request or after receiving a response. The syntax for this feature is as follows:
it is submitted to call the [OPERATION_NAME] operation of [PRODUCT_NAME] with a delay of [DELAY_SECONDS] seconds [DELAY_POSITION] the call
Parameters:
- OPERATION_NAME: The name of the operation being called (e.g., Get Document Metadata by Document ID).
- PRODUCT_NAME: The name of the product or service that provides the operation (e.g., CCD Case Document AM API).
- DELAY_SECONDS: The number of seconds to delay.
- DELAY_POSITION: Specifies whether the delay is to be applied before or after the HTTP call. Possible values are before or after.
Delay Before the HTTP Call:
it is submitted to call the [Get User Details] operation of [User Management Service] with a delay of [10] seconds [before] the call
In this example, the framework will wait for 10 seconds before making the request to the "Get User Details" operation of the "User Management Service".
Delay After the HTTP Call:
it is submitted to call the [Get User Details] operation of [User Management Service] with a delay of [5] seconds [after] the call
Here, the framework will wait for 5 seconds after making the request to "Get User Details" operation of the "User Management Service" before proceeding with the next steps.