Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/retailmenot/scaffold
A Java based Selenium WebDriver abstraction
https://github.com/retailmenot/scaffold
java selenium selenium-java selenium-webdriver
Last synced: 3 months ago
JSON representation
A Java based Selenium WebDriver abstraction
- Host: GitHub
- URL: https://github.com/retailmenot/scaffold
- Owner: RetailMeNot
- License: mit
- Created: 2019-03-20T15:57:21.000Z (almost 6 years ago)
- Default Branch: master
- Last Pushed: 2023-04-10T17:03:52.000Z (almost 2 years ago)
- Last Synced: 2024-09-29T02:05:56.736Z (3 months ago)
- Topics: java, selenium, selenium-java, selenium-webdriver
- Language: Java
- Homepage:
- Size: 201 KB
- Stars: 7
- Watchers: 10
- Forks: 9
- Open Issues: 11
-
Metadata Files:
- Readme: README.md
- Contributing: CONTRIBUTING.md
- License: LICENSE.txt
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# Scaffold
Scaffold is a Selenium WebDriver abstraction built in Java 11 with Spring Boot 2.x / Jersey / Jax Rs. Out of the box, it provides a myriad of additional features on top of the base WebDriver wrapper:* Bootstrapper - Starts and stops a new thread safe WebDriver instances (parallelizable).
* Provides an abstraction of WebElements to depict literal object representations of what exists in the DOM (e.g. InputWebElement or LinkWebElement).
* Implicitly waits for elements to be located during page object instantiation and element manipulation
* Automatically handles StaleElementException issues when the state of the DOM has changed.
* Provides a Navigation class to give users a separation of concerns between test setup and test assertions.
* Provides a Spring Boot auto-configuration for configuring a browser when creating Spring Boot profiles in an implementing project (DesiredCapabilities, e.g. browser type, environment type, etc).
* Provides a Spring Boot auto-configuration for managing configuring connections to SauceLabs.
* Configures Junit Jupiter's parallel testing when running the testing through an automated framework like Sauce or Grid
* Provides all dependencies to implementing projects# Current Version
To view the most current version, [visit the Central Repository](https://search.maven.org/search?q=g:com.retailmenot.scaffold).# Links
- [Contributing Guide](https://github.com/RetailMeNot/scaffold/blob/master/CONTRIBUTING.md)
- [Code of Conduct](https://github.com/RetailMeNot/scaffold/blob/master/CODE_OF_CONDUCT.md)
- [License](https://github.com/RetailMeNot/scaffold/blob/master/LICENSE.txt)
# Required Tools for Dev
* Java 11
* Maven 3.x# CI
The build can be found [on Travis CI](https://travis-ci.org/RetailMeNot/scaffold).# Setting Up Your Project
Use this section for setting up a new project using Scaffold. Setup follows a fairly standard Spring Boot application design by using modules for the code base's environment and main testing.Coming in a future update, we will provide an example implementation project and a maven archetype to easily start up new projects.
## Create the Project
Create a new Java Maven project.Follow the standard maven [naming conventions](https://maven.apache.org/guides/mini/guide-naming-conventions.html) when creating the new project.
## Add Maven Repo Dependencies
Add the following dependency to your parent POM's DependencyManagement section:```
com.retailmenot.scaffold
scaffold
current_versioncom.retailmenot.scaffold
framework
current_versioncom.retailmenot.scaffold
environment
current_version```
## Create Modules
Create two new maven modules in your project.* `environment`
* `core`The environment module's POM should contain Scaffold's environment dependency.
```com.retailmenot.scaffold
environment```
The core module's POM should contain Scaffold's framework dependency.
```com.retailmenot.scaffold
framework```
## Required Files
The following are files that should be created in your project.### Spring Configuration
A spring configuration file is necessary for handling component scanning and the usage of spring profiles when running testing. Component scanning is necessary for dependency injection and spring profiles are used for setting
`DesiredCapabilities` for testing. More detail regarding spring profiles are included in a later section.Create a new file under the environment module adhering to the naming convention standardization hierarchy set when creating your new project.
E.G: `environment > src > main > java > your > groupID > environment > YourProjectConfiguration.java`
The contents of this file should include:
```
@Configuration
@ComponentScan(value = "your.group.id")
@PropertySource("classpath:/application.properties")
public class YourProjectConfig {
}
```### Page Objects
Scaffold implements the paradigm of a page object. Page objects are simple representations of web pages as a Java Object. In order to create a representation of a web page, you'll need to create a new class that contains properties
of Scaffold's strongly typed elements.#### Strongly Typed Elements
Scaffold makes available particular types of WebElements for you to use, which narrow the scope of methods available for any given WebElement, and keep the user focused on the actions they should be performing on these elements.
For example, a user cannot `sendKeys()` to a Button or a Link. The `ButtonWebElement` and `LinkWebElement` objects take that into account and don't expose those methods.Another advantage of these elements is that they manage all interaction with the WebDriver internally. Most frameworks require the test write/page object maintainer to use the WebDriver to perform all their actions,
which requires a lot of Selenium knowledge, and can also lead to race conditions, thread-safety issues, and the exposure of unnecessary complexity to the testers.In the millions of test cases run by Scaffold over the years, there has never been a reported occurrence of a StaleElementReferenceException. This is possible because the framework manages the WebDriver and the WebElements
internally, in a thread-safe and careful manner.A strongly typed element in scaffold can be one of the following:
* ButtonWebElement
* CheckBoxWebElement
* DateWebElement
* DivWebElement
* DropDownWebElement
* ImageWebElement
* InputWebElement
* LinkWebElement
* RadioWebElement
* StaticTextWebElement#### Page Object Example
Page objects should only be written in a way that makes them agnostic to navigation, the web driver itself, or by any other external means outside the scope of its own representation. They should also never contain any
assertions. While the `WebDriver` is not directly injected into a page object, the underlying `WebDriver` is used for finding elements when a `getElement()` is performed and _not_ when an element is declared.In other words, simply initializing a strongly typed element does not perform an interaction with the underlying WebDriver; but instead, only creates a reference point with a By locator. This creates a dynamic use case when
performing navigation in that when a new page object is initialized, the elements will not be searched for at the time of the class being constructed. It isn't until you attempt to get the strongly typed element with a getter
that the underlying WebDriver will perform a `getWebElement()`, therefore performing the `findElement()` interaction.To create a page object, follow the same design as Java Beans, [found here](https://www.javatpoint.com/java-bean).
The page objects should live within the core module in a page module. E.G: `core > src > main > java > your > groupID > page`
An example page object:
```
public class LoginPage {private DivWebElement pageHeader = new DivWebElement(By.cssSelector("#someHeader"));
private InputWebElement emailInput = new InputWebElement(By.cssSelector("#emailInput"));
private InputWebElement passwordInput = new InputWebElement(By.cssSelector("#passwordInput"));
private ButtonWebElement loginButton = new ButtonWebElement(By.cssSelector("#loginButton"));public DivWebElement getPageHeader() {
return pageHeader;
}public DivWebElement getEmailInput() {
return profileGreeting;
}public LinkWebElement getPasswordInput() {
return editProfileLink;
}public DivWebElement getLoginButton() {
return emailList;
}public void clickLoginButton() {
getLoginButton().click();
}
public void login(String username, String password) {
getEmailInput().clearAndSendKeys(username);
getPasswordInput().clearAndSendKeys(password);
getLogInButton().click();
}
}
```Let's break down what we see in the example above.
1. Properties
These are the elements on the page that we wish to represent. They can be whatever you feel is necessary to have. They could be headers, inputs, buttons, images, or anything else on the list of Strong Typed Elements above
and are all merely references to be used later when you're getting the elements. They are located with the `By` class.
2. Getters
Just regular ol' getters. It was already mentioned above, but this is where the magic happens. For example, when you use the `clickLoginButton()` method in a test, the `AbstractWebElement` class will perform a find element
from the `WebDriverWrapper` to find the element `loginButton`, and then perform the click action from `AbstractClickable`.
3. Page actions
The page object is a good opportunity to include any page specific actions you'd like to abstract. This is yet another level of creating an additional layer that will allow us to maintain our testing a little easier as it scales.### Navigation
Your project should have a navigation file that extends `WebDriverNavigation`. This file should live within the core module in a navigation package. E.G: `core > src > main > java > your > groupID > page > Navigation.class`
The navigation file handles the navigation of the page objects. For example, using the example login page object above, let's say your implementing project has a login page and a profile page.
In order to navigate to the profile page, it first requires a login (since the profile is gated). Your page navigation file might contain the method `navigateToProfilePage`:
```
@Component
public class AutomationNavigation extends WebDriverNavigation {// Spring environment variable from a configuration file in environment module
private final String baseEnvironmentUrl;public NavigationImpl(@Value("${base-environment-url}") String baseEnvironmentUrl) {
this.baseEnvironmentUrl = baseEnvironmentUrl;
}public ProfilePage navigateToProfilePage(String username, String password) {
// Start the web test from a base environment URL pulled in from a Spring env variable
getWebDriverWrapper().get(baseEnvironmentUrl);
// Create a new instance of the LoginPage
var logInPage = new LoginPage();
// Use the login method from the LoginPage
logInPage.login(username, password);
// Return the new ProfilePage
return new ProfilePage();
}
}
```Let's break down what we see in this above example.
1. @Component
This is an important annotation to use here because we'll be injecting this class into the BaseTest file later on. Because we have a component scan configuration file (set up earlier), it will detect classes with the `@Component`
annotation and allow us to perform the injection.
2. WebDriverNavigation extension
Extending off of `WebDriverNavigation` is important here so we can get the `WebDriverWrapper` instance from the current thread. That allows us to perform the `.get` call.
3. Constructor dependency injection
Seen in the example above, we reference `@Value("${base-environment-url}") String baseEnvironmentUrl`. This is a spring environment variable defined in a spring profile (more on that later). This allows us have a configure what
we want our url to be based on what environment we're wanting to run the testing on. It's important to know that only Strings can be used for these variables.
4. Navigation method
Here is where the actual work happens. We get the WebDriverWrapper from the WebDriverNavigation class (because it's protected) and we perform the `get()` function to navigate to our base environment url set by a spring profile.
Then, we create a new instance of the LoginPage page object, we perform the login that is defined on the LoginPage class (since it's gated), and then return a new instance of a ProfilePage page object.The advantage here for separating out the navigation is that it gives us further abstraction in our test writing. Or, in other words, creating an additional layer that will allow us to maintain our testing a little easier as it scales.
### BaseTest
Your project should have a BaseTest file that extends `ScaffoldBaseTest`. This file should live within the core module's test package. E.G: `core > src > test > java > your > groupID > BaseTest.java`This BaseTest file should include any project specific spring wiring or configurations along with the Scaffold configuration. Make sure to also attach the annotation listed below, as well.
```
@Execution(ExecutionMode.CONCURRENT)
@ExtendWith(SpringExtension.class)
@SpringBootTest(
webEnvironment = SpringBootTest.WebEnvironment.NONE,
classes = { your_own_config.class, ScaffoldConfig.class }
)
public abstract class BaseTest extends ScaffoldBaseTest {@Autowired
protected Navigation navigation;
@BeforeEach
public void doSomethingAtStart() {
// Some code here that might do something before each test is started
}
@AfterEach
public void doSomethingAtEnd() {
// Some code here that might do something after each test is finished
}
}
```Let's break down what we see in this example above.
1. Annotations
These annotations are important for configuring what was once known in Junit4 as the test runner, the parallelization mode, and the spring configuration for loading the application context / spring boot test app. Under the
`@SpringBootTest` annotation, be sure to change `your_own_config.class` to your projects configuration class name.
2. Abstract class identifier
It's important to declare this class abstract so any Junit5 methods are not ran as tests.
3. ScaffoldBaseTest extension
Extending `ScaffoldBaseTest` gets you the driver initialization and tear down and will also get you the `WebDriverWrapper` instance from the current thread.
4. @Autowired
This is a means of dependency injection within Spring Boot known as composition. With this annotation, we can wire in an instance of a `@Component` with ease. Since it's protected, any class that extends off of BaseTest will
now be able to access it.
5. @BeforeEach and @AfterEach
These are just some examples of additional code you can write in your `BaseTest` file to perform any pre req or tear down actions outside of the driver initializing and closing.### Spring Profiles
Spring profiles are sets of configurations that can be used when running testing locally or through a test automation framework like Sauce. These configurations determine the `DesiredCapabilities` of the browser and can also
configure Sauce credentials. During a test run, you specify the spring profile to use. This will be explained a little bit later.The spring profiles should live under the resources package in the environment module that was set up earlier in this guide. E.G: `environment > src > main > resources > application-chrome_sauce.properties`. For more information on spring profiles, [check out this link here](https://www.springboottutorial.com/spring-boot-profiles).
#### Desired Capabilities
To set the `DesiredCapabilites`, include pre configured properties from the `DesiredCapabilitiesConfigurationProperties` file. A full list of these properties [can be found at the following link](https://github.com/RetailMeNot/scaffold/blob/master/environment/src/main/java/com/retailmenot/scaffold/environment/config/DesiredCapabilitiesConfigurationProperties.java).
All of these properties are preceded by the prefix of `desired-capabilities`. So, for example, if you wish to define the run type of sauce, you'd enter `desired-capabilities.run-type=sauce`. Because Scaffold includes an
auto configuration for these properties, you gain the benefit of auto complete, as well. Simply type the first few letters of the word `desired` will show you a list of capabilities that can be set.If you'd like to learn more about desired capabilities, details on the configuration [options that can be found here](https://wiki.saucelabs.com/display/DOCS/Test+Configuration+Options).
Below is the current list of potential desired capabilities to set.
```
# Browser/OS Config
desired-capabilities.run-type=where the browser testing is running, e.g. local, sauce, or grid
desired-capabilities.environment-type=the environment the testing is running on, e.g. test or stage.
desired-capabilities.browser-type=the browser type to be launched
desired-capabilities.browserVersion=the version of the browser to be launched
desired-capabilities.runPlatform=the operating system the browser is launching on
desired-capabilities.remote-url=The default grid URL to use
desired-capabilities.upload-screenshots=a boolean to determine if screenshots will be uploaded
```#### Local Chrome Example
One option of a test run could include a local execution. This type of configuration is good for a one off test to debug or for POC'ing a test. It's not recommended that you run a large suite of testing with a local
browser. Typically, this sort of local configuration is included in an overrides spring profile and is not used for CI. If using an overrides profile _do not_ include this file in your commit.
Ensure that this file is added to your .gitignore. Overrides files are also used to include hard coded secrets and should never be exposed publicly.*Important note: Since a local browser is being used, you must have the local driver installed. E.G: ChromeDriver. Otherwise, the testing will not work.*
For this example, let's assume we have created a new overrides profile. This overrides profile is called `application-overrides.properties`. In that file, we would include the following:
```
base-environment-url=some_base_urldesired-capabilities.run-type=local
desired-capabilities.browser-type=chrome
desired-capabilities.run-platform=mac
```
1. The base environment is in this file because it's a required environment variable from the `Navigation` class.
2. The run type is local because we're executing the test against our local browser
3. The browser type is chrome because chrome happens to be our main browser. This can be switched to other browsers but those drivers will be required
4. The run platform is mac because our OS is mac. This can be changed to whatever OS is being run.#### Sauce Chrome Example
Another option of a test run could include a test execution against Sauce Labs. Because of the auto configuration defined by Scaffold, it's easy to add the sauce credentials to the overrides profile (to run the testing from your machine but sending the testing to sauce labs)
or to a spring profile that is used in the CI/CD pipeline.*Important note: If adding the sauce credentials to an overrides file, ensure the overrides file is not checked in to your code base. These values should never be publicly exposed.*
For this example, let's create a new spring profile called `application-chrome_test.properties`. This configuration is going to be ran through sauce labs and will execute against a lower environment, TEST, with a windows browser.
This configuration file should be included would have the following:
```
base-environment-url=http://www.websitetest.com# Sauce Config
desired-capabilities.sauce.url=
desired-capabilities.sauce.user-name=
desired-capabilities.sauce.password=
desired-capabilities.sauce.access-key=
desired-capabilities.sauce.tunnel-identifier=# Base Desired Capabilities
desired-capabilities.run-type=sauce
desired-capabilities.browser-type=chrome
desired-capabilities.run-platform=windows
```1. The base environment is in this file because it's a required environment variable from the `Navigation` class.
2. The sauce url is the endpoint you'd like to hit for testing. For now, Scaffold does not have a hard requirement to force you to hit the Selenium endpoint.
3. The username is your sauce labs account username
4. The password is your sauce labs account password
5. The access key is your sauce labs account's access key
6. The tunnel identifier is the name of your tunnel that is started in your CI (or locally) to run against.
7. The run type is sauce because the testing is running against sauce labs
8. The browser type is chrome because we'd like to run the testing in chrome. This can be changed to any other browser supported by Sauce Labs.
9. The run platform is windows because we'd like to run the testing on a Windows OS.#### Configuring Constant Values for DesiredCapabilities
Because of the hierarchy of the spring profile system, it is possible to create constant environment variable values that all spring profiles can automatically include. This is useful for sauce credentials since you can include
the configuration in only one profile.These constants can be included in a file named `application.properties`. It's worth mentioning that any subsequent spring profile that sets the same environment variable will override the existing value in `application.properties`.
This will allow you to set default values for any additional environment variable you create but override them in child profiles if you so wish.For this example, let's say we'd like to include the sauce credentials in `application.properties`. Since we might have multiple configurations we'd like to have, with all of them requiring sauce credentials, it makes sense to only
include them in one file.That file will include the following:
```
desired-capabilities.sauce.url=
desired-capabilities.sauce.user-name=
desired-capabilities.sauce.password=
desired-capabilities.sauce.access-key=
desired-capabilities.sauce.tunnel-identifier=
```Taking the example from the Sauce Chrome Example section above, the new `application-chrome_test.properties` file would look like this:
```
base-environment-url=http://www.websitetest.com# Base Desired Capabilities
desired-capabilities.run-type=sauce
desired-capabilities.browser-type=chrome
desired-capabilities.run-platform=windows
```Since the sauce credentials are already included in `application.properties`, the `application-chrome_test.properties` file does not need the sauce credentials. Therefore, we cut back on a little bit of code!
## Running the Testing
### Locally
There are two potential methods of running the testing locally. The first is maven goal execution and the second is running the testing through the IDE.### Maven
During your maven goal execution step on your CI, specify the following system property: `-Dspring.profiles.active=your_spring_profile.properties`.This will pull the environment variables from the profile specified in the system property.
### IDE
If you don't already have one, create a file named `application.OVERRIDES` in `environment > src > main > resources > application-OVERRIDES.properties`. In this file, include the desired capabilities and any other secret values required to run your testing.*NOTE: Make sure to add `application.OVERRIDES` to your `.gitignore`*
Next, create a new JUNIT run configuration for the testing you'd like to run locally. In the new run configuration, add the new environment variable `SPRING.PROFILES.ACTIVE=OVERRIDES`. This setting will pull in the environment variables from the OVERRIDES profile when that test is run through the IDE.
*As a reminder, it's recommended to not use a spring configuration that contains a local browser desired capability for running a large amount of testing.*
Now your testing is ready to be executed.
### CI
During your maven goal execution step on your CI, specify the following system property: `-Dspring.profiles.active=your_spring_profile.properties`.This will pull the environment variables from the profile specified in the system property.