Before implementing and establishing automated UI tests in a project lots of questions should be answered.
For the use case I'm testing for, these decisions were made
- A "real" browser is needed, no emulation
- The Selenium web driver should be used, cause a selenium grid is already in place
- A framework needed, because using Selenium will lead to messy and not maintainable code. The question is, if an existent one is good enough to work with quite a long time horizon. Also the framwork should not be dead when starting this project
- At least some patterns should be used to develop maintainable code and supports quick creation of unit tests
- MS Test should be used
- The UI tests must be stable and should not lead to lots of false positives
The Atata Framework sounds promising, so a VS test solution shall be created which uses Atata Framework. Then unit tests for Amazon will be written to test the framework and the reliability of both, Selenium and Atata Framework.
This document contains a step by step guide of what was done. The goal of this solution is that it is reproducible if it's worth it.
A new solution was created with one Unit Test Project based on .NET Framework 4.7
These packages have been installed:
- https://www.nuget.org/packages/Atata/0.15.0/
- https://www.nuget.org/packages/Atata.WebDriverExtras/0.13.0/
Automatically these packages were installed because of dependencies
- https://www.nuget.org/packages/Newtonsoft.Json/
- https://www.nuget.org/packages/Selenium.Support/
- https://www.nuget.org/packages/Selenium.WebDriver/
Then a chrome web driver was installed, as well as [NLog] (https://www.nuget.org/packages/NLog/).
To get started, a new app.config was added which will contain the settings, like a base url, accounts and other settings. A static configuration class holds these values
A base class for the unit tests is used to setup the web driver and the framework:
[TestClass]
public class TestBase
{
[TestInitialize]
public void TestInitialize()
{
AtataContext.Configure()
.UseChrome().WithArguments("disable-extensions", "no sandbox", "start-maximized")
.UseBaseUrl(Configuration.BaseUrl)
.UseTestName(TestContext.TestName)
.AddNLogLogging()
.UseCulture("de-de")
.AddTraceLogging()
.Build();
}
[TestCleanup]
public void TestCleanup()
{
AtataContext.Current.CleanUp();
}
}
That's the basic setup, one can now directly start writing unit tests.
For home a new page object has been created which can then be used for navigation etc.
public class HomePage : Page<HomePage>
{
}
Now some tests needs to be written to test how fast one can write tests with this framework, how stable is it and what happens when some test fails.
Go.To<HomePage>().PageTitle.Should
.Equal(
"Amazon.de: Günstige Preise für Elektronik & Foto, Filme, Musik, Bücher, Games, Spielzeug & mehr");
A page object can derive from other page obejcts and can contain controls to group features.
public class HomePage : CommonPage<_> {}
public class CommonPage<TPage> : Page<TPage> where TPage : Page<TPage>
{
public SearchBar<TPage> SearchBar { get; set; }
public SortBar<TPage> SortBar { get; set; }
}
Controls which can contain multiple other UI components:
[ControlDefinition("div", ContainingClass = "searchTemplate", ComponentTypeName = "search template")]
public class SortBar<TPage> : Control<TPage> where TPage : PageObject<TPage>
{
[FindById("s-result-count")]
public H1<TPage> ResultsCount { get; set; }
[FindByClass("a-color-state a-text-bold")]
public Text<TPage> SearchTerm { get; set; }
}
[ControlDefinition("form", ContainingClass = "nav-searchbar", ComponentTypeName = "nav searchbar")]
public class SearchBar<TPage> : Control<TPage> where TPage : PageObject<TPage>
{
[FindById("twotabsearchtextbox")]
public TextInput<TPage> SearchText { get; set; }
[FindByClass("nav-input")]
public Button<TPage> SearchButton { get; set; }
}
The actual unit tests looks like this:
Go.To<HomePage>().SearchBar.Set(searchTerm).SearchButton.Click()
.SortBar.ResultsCount.Should.Not.BeNull()
.SortBar.SearchTerm.Should.Contain(searchTerm);
Because of the AddNLogLogging()
in the test initialization and the nlog config log files are automatically generated.
When all unit tests are executed in VS these logs were generated:
- Home_ShouldHaveCorrectTitle_WhenOpened.txt
- SortBar_ShouldContainSearchTerm_WhenSearchIsUsedWithSearchTerm.txt
The second file contains than this log:
2017-11-30 19:33:45.5688 INFO Starting test: SortBar_ShouldContainSearchTerm_WhenSearchIsUsedWithSearchTerm
2017-11-30 19:33:45.6278 TRACE Starting: Set up AtataContext
2017-11-30 19:33:45.6323 TRACE Set: BaseUrl=https://www.amazon.de/
2017-11-30 19:33:45.6358 TRACE Set: RetryTimeout=5.000s; RetryInterval=0.500s
2017-11-30 19:33:45.6393 TRACE Set: Culture=de-DE
2017-11-30 19:33:48.1906 TRACE Set: Driver=ChromeDriver (alias=chrome)
2017-11-30 19:33:48.1981 TRACE Finished: Set up AtataContext (2.570s)
2017-11-30 19:33:48.2681 INFO Go to "Home" page
2017-11-30 19:33:48.3416 INFO Go to URL "https://www.amazon.de/"
2017-11-30 19:33:50.6276 INFO Starting: Set "ui testing" to "Search Bar" text input
2017-11-30 19:33:52.9064 INFO Finished: Set "ui testing" to "Search Bar" text input (2.278s)
2017-11-30 19:33:52.9159 INFO Starting: Click "Search" button
2017-11-30 19:33:54.7788 INFO Finished: Click "Search" button (1.862s)
2017-11-30 19:33:54.7978 INFO Starting: Verify "Sort Bar" search template's "Results Count" <h1> heading content should not be null
2017-11-30 19:33:54.9603 INFO Finished: Verify "Sort Bar" search template's "Results Count" <h1> heading content should not be null (0.162s)
2017-11-30 19:33:54.9683 INFO Starting: Verify "Sort Bar" search template's "Search Term" element content should contain "ui testing"
2017-11-30 19:33:55.4425 INFO Finished: Verify "Sort Bar" search template's "Search Term" element content should contain "ui testing" (0.474s)
2017-11-30 19:33:55.4740 INFO Starting: Clean up test context
2017-11-30 19:33:55.6400 INFO Finished: Clean up test context (0.165s)
2017-11-30 19:33:55.6480 INFO Finished test (10.081s)
2017-11-30 19:33:55.6560 INFO Pure test execution time: 7.271s
An ordered list can be tested using the classes ListItem and OrderedList
[FindByClass("a-carousel")]
public OrderedList<CarouselWidgetItem<TPage>, TPage> Items { get; set; }
public class CarouselWidgetItem<TPage> : ListItem<TPage> where TPage : PageObject<TPage> {}
A link delegate makes clicking easier, because the property itself is a function - no need to call Click(). When the Screenshot-attribute is used, a screenshot is automatically taken, even though the purpose of this is unclear (maybe for testing purposes during UI test development).
[Screenshot("BeforeClick")]
[Screenshot("AfterClick", TriggerEvents.AfterClick)]
[FindByClass("a-carousel-goto-nextpage")]
public LinkDelegate<TPage> Next { get; set; }
///Can be used like this
Go.To<HomePage>().CarouselTeaser.Next();
Go.To<HomePage>().CarouselTeaser.Items.Items.Count.Should.Equal(7);
The Atata Framework page objects and controls seems more useful when using interactive elements and they do not support querying html elements. It seems that an extension package can do the job, also it's possible to directly using selenium as a fallback.
To also use Should
for the selenium queries, the package Fluent Assertions have been installed.
Go.To<HomePage>().Header.LanguageSwitch.Hover();
var driver = AtataContext.Current.Driver;
var element = driver.Get(By.Id("nav-flyout-icp").OfAnyVisibility().AtOnce());
element.Displayed.Should().BeTrue();
- How to get a screenshot from a specific UI component like a div