Not many people have talked about the "other side" of test modelling - what I call the user model, or the human proxy :-) And it's taken me some time to fall on this pattern to separate the input from the app model (decouple it) in a way that I'm happy with.
My Test projects typically have 3 main pieces:
My Test projects typically have 3 main pieces:
- A test fixture/s .. the test/s themselves - uses both the page and user model
- An app model .. whatever test framework drives the app (i.e. WatiN /UIAutomation etc...)
- and a user model
The user model is the mechanism I use to deliver data to the AUT. Think of the typical user view of the app - as a User of the app I have a username, password, email and dob etc. The user might need a shopping list which in itself might have many items, or only a few. They may need credit card with specific properites - Good, Expired, noAvailableCredit, weirdCardtype etc
So for my example app here, I have simple property bag classes of type User, PurchaseItem and CreditCard - with a User instance having a ShoppingList (made up of PurchaseItems) and a CreditCard. The nature of this model will depend on what your app does - a zoo management app would probably need animal, cage and refreshment-kiosk objects.
Broadly speaking I end up with model object per form in the UI, although things like registering for an account and logging in will both use the "User" object.
When I initialise a test I intantiate pieces of the user model and assign them values. Mostly the defaults will be fine, in other situations I construct specific data required for the test. By having mandatory data defaults built into the model (or even better dynamically retrieved from a known DB state and set in the fixture setup) - I'll keep clear of the "setup is good" vs "setup is bad" debate here...
It means that the intent of THIS TEST is more obvious, as only the data that really matters for the test gets set here. I quite often find I use random generators where uniqueness is required. I have a set of helper classes in the user model to do this kind of thing with guids text dates etc.
Once the user model is initialised I can start calling the page model and passing the user model components in as parameters.
Here is an example....in this case the existingUser object, their default shopping list and the expiredMasterCard would be set in the setup fixture.
When I initialise a test I intantiate pieces of the user model and assign them values. Mostly the defaults will be fine, in other situations I construct specific data required for the test. By having mandatory data defaults built into the model (or even better dynamically retrieved from a known DB state and set in the fixture setup) - I'll keep clear of the "setup is good" vs "setup is bad" debate here...
It means that the intent of THIS TEST is more obvious, as only the data that really matters for the test gets set here. I quite often find I use random generators where uniqueness is required. I have a set of helper classes in the user model to do this kind of thing with guids text dates etc.
Once the user model is initialised I can start calling the page model and passing the user model components in as parameters.
Here is an example....in this case the existingUser object, their default shopping list and the expiredMasterCard would be set in the setup fixture.
[Test, Description("Purchase fails when attempting to use an expired CreditCard ")]
public void CreateNewAccountVerifyDetails()
{
//Setup Test Context
user.CreditCard = expiredMasterCard
using (HomePage homePage = new HomePage(ConfigurationHandler.PreProdUrl))
{
LoginPage loginPage = homepage.GoToLogin()
ProfilePage profilePage = loginPage.Login(user)
StorePage storePage = profilePage.GoToStore()
CheckOutPage checkOutPage = storePage(user.shoppinglist)
PaymentPage paymentPage = checkOutPage(user.CreditCard)
Assert.IsTrue(enterPaymentDetails.ExpiredCardMessage())
}
This way the test code becomes pretty generic for the "login, select items and pay" flow. Only the expected result would change for different card info.Maybe you could even parameterise this further with something like the MBUnit rowtest pattern, one per card-type.
Anyway this seems to work for me at the moment, but I'm sure there are many more ways to skin this cat. Let me know what works for you.
No comments:
Post a Comment