-
Notifications
You must be signed in to change notification settings - Fork 11
Cucumber
Cucumber is a Behavior-Driven Development (BDD) framework. Currently Neodymium only supports Cucumber framework in combination with JUnit4. Please follow these instructions Cucumber to migrate or set up your Cucumber project with Neodymium 5
BDD is a technique in agile software development used to achieve close collaboration between the quality assurance and the business analysis parts of the team. The team focuses on the targets of the stakeholders. The user stories are written in a natural language that follows a known structure and utilizes certain keywords to provide some kind of API that can be used by the developers later on to implement the test automation.
Cucumber itself is a popular BDD framework that provides some structural elements to orchestrate the test and uses Gherkin a defined set of keywords to differentiate and organize the steps.
We prepared a demo project: Neodymium Cucumber Example which gives an idea how cucumber could be used in a test automation suite. Further information can be obtained directly from the project.
- Feature is equivalent to a JUnit test class
- Scenario is equivalent a JUnit test method
- Scenario Outline is equivalent a JUnit test method with an attached test data table (Examples see below)
- Background could be seen as a JUnit @Before annotated method to set up the environment for each test method in the same way
- Examples is used to set up test data for a Scenario
It is good to know that the keywords are just used to build up a more natural language but the implementation it self doesn't differentiate between. This means you could use the And keyword all the time and it wouldn't make a difference for the execution of the script.
-
Given
: mostly the start of a scenario, but can be used for prerequisites -
When
: a start to describe the interaction with the page -
Then
: the keyword to bring up validation phrases -
And
: is used to chain all kinds of phrases -
But
: can be used to start and chain any kind of negative phrases
The following example show some basic structure of a test case and demonstrates the usage of the keywords.
Feature: Browse
Scenario Outline: Browsing the catalog
Given The browser "<browser>" is open
And I am on the homepage of the Posters shop
When I hover over "<categoryName>" and click on "<subCategoryName>"
Then I want to be on a category page and see the "<subCategoryName>" as headline
When I click on the product "<productName>"
Then I want to be on a product detail page and see the "<productName>" as headline
Examples:
| browser | categoryName | subCategoryName | productName |
| Chrome_1024x768 | World of Nature | Animals | Grizzly Bear |
| Chrome_1024x768 | Dining | Main Dishes | Tuna Steak |
| Chrome_1024x768 | Dining | Sweets | Colored Sprinkles |
Neodymium provides an utility class that can be called statically from you Cucumber support (hook) classes. It's called Driver and provides the functionality to setup and tear down WebDrivers. You can choose between the following two ways to setup a WebDriver via Neodymium. The tear down will stay the same for both. Please find a hands on example here: DriverHooks.java.
By calling a Cucumber Given
step and passing the browser profile name via the standard test data mechanism you can call the Neodymium WebDriverUtils
class to setup the browser.
@Given("^\"([^\"]*)\" is open$")
public static void setUp(final String browserProfileName)
{
Driver.setUp(browserProfileName);
}
The corresponding feature file could look like this. Each row of the example data leads to a WebDriver setup.
@Browse
Feature: Browse
Scenario Outline: Browsing the catalog
Given "<browser>" is open
And homepage is loaded
When I choose main category "<categoryName>" and sub category "<subCategoryName>"
Then I see category page with "<subCategoryName>" headline
Examples:
| browser | categoryName | subCategoryName |
| Chrome_1024x768 | Transportation | Air Travel |
| Chrome_1024x768 | World of Nature | Animals |
| FF_1024x768 | Dining | Main Dishes |
| FF_1024x768 | Dining | Sweets |
Please see the following example for more hand on experience: Browse.feature
If you don't need or want the extra step, you can create a general before hook that does the browser setup if it detects a configured browser profile name within the tags of the scenario. Beware that tagging a test execution with more than one browser profile name will lead to an exception, since it can't be decided which was meant to run.
@Before()
public static void setUp(Scenario scenario)
{
Driver.setUpWithBrowserTag(scenario);
}
The corresponding feature file could look like this. Each row of the example data leads to a WebDriver setup.
@Search
Feature: Searching for products
Scenario Outline: Searching for existing products
Given homepage is loaded
When I search for "<searchTerm>"
Then I see category page
And result page contains searchterm "<searchTerm>" and shows "<expectedCount>" products
And product "<productName>" is visible
@Chrome_1024x768
Examples:
| searchTerm | expectedCount | productName |
| bear | 3 | Grizzly Bear |
@FF_1024x768
Examples:
| searchTerm | expectedCount | productName |
| bee | 9 | Indian Summer: Orange Beech |
Please see the following example for more hands on experience: Search.feature
Note: Since we demonstrate both ways of setting up the WebDriver in the same demo project we had to add a special tag to the hook and the test case to avoid interferences. Probably you won't need this in your project.
To tear down the WebDriver you just need to add a general hook. Be aware that you need to pass an order number below 10000 to run it after all other After
hooks. Doing so, you make sure that the browser will be torn down once all standard After
steps have been executed.
@After(order = 100)
public void tearDown(Scenario scenario)
{
Driver.tearDown(scenario);
}
Sometimes test cases need a special setup on execution. In a perfect test world you could just start your test with a new prepared environment but since this can't be realized in every case you need to clean up after the test execution. Cucumber supports hooks to add an After
step after a scenario but you can't share data between different steps without further programming. We present an easy solution to the problem below.
We demonstrate how to share data between steps using Cucumber PicoContainer. This technique is used to implement clean up steps for Register.feature.
In Neodymium version 3.3.0 we moved the PicoContainer dependency into the library to avoid incompatibilities in child projects when we update the Cucumber version.
- Implement a global storage object that can hold the data that is supposed to be shared. Please see GlobalStorage.java example.
- This should ideally be a POJO that just holds data fields that can be other POJOs.
- Use dependency injection to initialize a GlobalStorage object within the class that will use it. Please check the following file for a hands on example RegisterSupport.java.
private GlobalStorage storage;
public RegisterSupport(GlobalStorage storage)
{
// The storage is passed via dependency injection
this.storage = storage;
}
- Now you can set the GlobalStorage with the passed data from the scenario.
@Given("^user setup: \"([^\"]*)\", \"([^\"]*)\", \"([^\"]*)\", \"([^\"]*)\"$")
public void setUpUser(String firstName, String lastName, String eMail, String password)
{
// set up user for the clean up steps
storage.user = new User(firstName, lastName, eMail, password);
};
- Afterwards you can use the data in every step that has a need for it.
@Given("^login page is opened after registration$")
public void registerUserSetup()
{
// use the user coming from dependency injection
registerUser(storage.user);
}
NOTE: Don't forget to delete the user after the test. To do this, mark the needed scenario with @DeleteUserAfterwards
and annotate the function implementing this with the same tag using a Cucumber After
annotation. See RegisterSupport.java class for a hands on example.
@After("@DeleteUserAfterwards")
Overview
Neodymium features
- Neodymium configuration properties
- Neodymium context
- Utility classes
- Test data provider
- Test Environments
- Multi browser support
- Applitools Plugin
- Localization
- Highlight and Wait
- Advanced Screenshots
- Seperate Browser Sessions for Setup and Cleanup
Best practices and used frameworks
Special