Skip to content
/ befta-fw Public

A Back End Test Automation Framework Built on top of Cucumber and RestAssured Frameworks

License

Notifications You must be signed in to change notification settings

hmcts/befta-fw

Repository files navigation

BACK END FUNCTIONAL TEST AUTOMATION FRAMEWORK

LICENSE

This project is licensed under the MIT License - see the LICENSE file for details.

1) WHAT IS BEFTA FRAMEWORK?

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.

2) FEATURES AND CONVENIENCES PROVIDED

It provides the following functionalities and conveniences:

  1. 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.
  2. Underlying programmatic logic to make the DSL an executable script language to execute functional tests.
  3. 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.
  4. Fast-performing, full-range response verification, robust mechanism to assert full details of responses including response statuses, reason phrases, headers and response bodies.
  5. Dynamic data configuration means allowing dynamic & run-time-specific test data to be identified without any programming at all
  6. Custom-programmable dynamicity to allow for programmable injections into test logic
  7. Clearly designed way of adapting and integrating the framework into various API suites for any new functional test suit.
  8. Reporting, and for-diagnostic outputting
  9. Custom extensibility of out-of-the box DSL (Cucumber feature)
  10. Other features coming with Cucumber Framework wrapped.

3) HOW TO SETUP & INTEGRATE

3.1) System Requirements

3.2) Software Requirements

  • Java SE Development Kit 21 (JDK 21)
  • Your Favourite IDE
  • Gradle 8.10+ (included as wrapper)

3.3) Setting Up Environment

  1. Install JDK 21 or higher
  2. Install a command line terminal application

3.4) Common Environment Variables

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.

3.5) Domain Specific Environment Variables

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'

3.6) Run BEFTA Framework Without a Build Tool

  1. 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
  2. Open your preferred terminal and change current directory to the root directory of your test automation project.
  3. 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.

3.7) Run BEFTA Framework With Gradle

  1. Install Gradle 8.7 or higher. You can simply copy a gradle wrapper from https://github.com/hmcts/befta-fw.

  2. Add the following dependency to your build.gradle file:
    testCompile group: 'com.github.hmcts', name: 'befta-fw', version: '9.0.2'

  3. 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"]
        }
    

3.8) Observe Cucumber Report

  1. Open in your web browser the local Cucumber report file:
    ./⁨target⁩/cucumber⁩/⁨cucumber-html-reports⁩/overview-features.html

3.9) Integrate Functional Test Suite Executions into Pipelines

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'

3.10) Setting up a Local API Application under Test

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.

3.11) Importing test data configuration into CCD

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!");
            }

        };
    }

3.11.1) Importing CCD roles

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.

3.11.2) Importing definition files

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.

⚠️ Any changes made to XLSX files in the directory 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 files

    The definitionsToJson gradle task will run the DefinitionConverter against each of the XLSX files in the directory src/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 file

    This 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
    

3.11.3) Create Role Assignment data

For information on how to configure Role Assignments for import see Domain Specific Environment Variables.

4) SAMPLE REPOSITORIES USING BEFTA FRAMEWORK

Below are a few github repositories into which BEFTA Framework has been successfully integrated:

Some Spring Boot Applications:

Some Node.js Applications:

5) HOW TO DEVELOP A SIMPLE AUTOMATED SCENARIO

Development of an automated test scenario takes, at a high level, the below main steps:

  1. Introduce a .feature file of any good name, and add a test scenario composed of the DSL elements available with BEFTA framework.

  2. 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.

  3. 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 class uk.gov.hmcts.befta.data.HttpTestData. Make sure that the 'guid' field in the file contains exactly the same text as the S- 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.

  4. 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.

5.1) Json Inheritance Mechanism - How it Works & How to Use

Attributes in an object represented by a JSON object in BEFTA-way are collated in the order or calculation shown in the below diagram:

5.2) Dynamic Value Place-holders

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:

  1. 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 the id 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 as a '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 as a 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 the id 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]} \

  1. ANY_NULLABLE: This is used to accept any data of any type including null or missing ones as the value of the response detail specified.

  2. ANY_STRING_NULLABLE: This is used to accept any text data including null or missing ones as the value of the response detail specified.

  3. ANY_NUMBER_NULLABLE: This is used to accept any numeric data including null or missing ones as the value of the response detail specified.

  4. ANY_INTEGER_NULLABLE: This is used to accept any integral data including null or missing ones as the value of the response detail specified.

  5. ANY_FLOATING_NULLABLE: This is used to accept any floating point data including null or missing ones as the value of the response detail specified.

  6. ANY_DATE_NULLABLE: This is used to accept any date data including null or missing ones as the value of the response detail specified.

  7. ANY_TIMESTAMP_NULLABLE: This is used to accept any time-stamp data including null or missing ones as the value of the response detail specified.

  8. ANY_OBJECT_NULLABLE: This is used to accept any Object data of any type including null or missing ones as the value of the response detail specified.

  9. ANYTHING_PRESENT: This is used to accept any data of any type but reject null or missing ones as the value of the response detail specified. It is an alias of ANY_NOT_NULLABLE.

  10. ANY_NOT_NULLABLE: This is used to accept any data of any type but reject null or missing ones as the value of the response detail specified.

  11. ANY_STRING_NOT_NULLABLE: This is used to accept any text data but reject null or missing ones as the value of the response detail specified.

  12. ANY_NUMBER_NOT_NULLABLE: This is used to accept any numeric data but reject null or missing ones as the value of the response detail specified.

  13. ANY_INTEGER_NOT_NULLABLE: This is used to accept any integer data but reject null or missing ones as the value of the response detail specified.

  14. ANY_FLOATING_NOT_NULLABLE: This is used to accept any floating point data but reject null or missing ones as the value of the response detail specified.

  15. ANY_DATE_NOT_NULLABLE: This is used to accept any date data but reject null or missing ones as the value of the response detail specified.

  16. ANY_TIMESTAMP_NOT_NULLABLE: This is used to accept any time-stamp data but reject null or missing ones as the value of the response detail specified.

  17. ANY_OBJECT_NOT_NULLABLE: This is used to accept any Object data of any type but reject null or missing ones as the value of the response detail specified.

5.3) Injecting Values of Environment Variables:

Environment variables can be referenced between double curly braces like in {{SOME_ENV_VAR}}.

5.4) Injecting Values from Scenario Context:

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:

5.5) Introducing Programmable Custom Dynamic Values

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:

  1. request: shortcut for [testData][request]
  2. requestBody: shortcut for [testData][request][body]
  3. requestHeaders: shortcut for [testData][request][headers]
  4. requestPathVars: shortcut for [testData][request][pathVariables]
  5. requestQueryParams: shortcut for [testData][request][queryParams]
  6. expectedResponse: shortcut for [testData][expectedResponse]
  7. expectedResponseHeaders: shortcut for [testData][expectedResponse][headers]
  8. expectedResponseBody: shortcut for [testData][expectedResponse][body]
  9. actualResponse: shortcut for [testData][actualResponse]
  10. actualResponseHeaders: shortcut for [testData][actualResponse][headers]
  11. actualResponseBody: shortcut for [testData][actualResponse][body]
  12. today: short cut for the value of the current date in yyyy-MM-dd format.
  13. today(<any-date-format>): short cut for the value of the current date in the specified format.
  14. now: short cut for the value of the current system time in yyyy-MM-dd'T'HH:mm:ss.SSS format.
  15. 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.

5.6) Special Conventions for Test Data

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.

5.6.1) Arrays at Top Level in Request and Response Bodies

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.

5.6.2) Files in Request Bodies

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"
          }
      ]
   }

5.6.3) Files in Response Bodies

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.

5.6.4) Conventions for Expected Collections

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 of equivalent, subset and superset. Default is equivalent 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 of ordered, and unordered. Default is ordered 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 or UNORDERED.

    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 to UNORDERED, either explicitly in test data or by setting DEFAULT_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.

6) INTRODUCING PROGRAMMATIC LOGIC INTO TEST SUITES

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.

7) HOW TO DEBUG TEST SUITES

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.

8) LOW-LEVEL DESIGN

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:

9) RETRY POLICY

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.

Default Policy

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).

Sample Configuration:

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

Service Level Policy

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.

Usage

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.").

Examples

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.

10) DELAYED HTTP CALLS

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.

Example Usage:

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.

About

A Back End Test Automation Framework Built on top of Cucumber and RestAssured Frameworks

Resources

License

Stars

Watchers

Forks

Packages

No packages published