In this post I will describe how to create asynchronous unit tests for Silverlight applications. First of all, I’d like to point out that any communication with a server component in Silverlight must be asynchronous. Reason being is that if communication was synchronous and the remote service was not responding, browser which host Silverlight application would lock up. As you can see, asynchronous communication with a RIA Services WCF service is very important in our case.
Let’s take a look on how to write a test that would assert that we can successfully get a list of companies from the server. To do so, we must first take a look on how to write such a test. This can be accomplished using asynchronous constructs available in Silverlight unit testing framework. First of all, we have to change inheritance for our test class. When you create a new test class, it does not inherit from any class, and we have to change this first of all. Here is what old test class declaration look like:
[TestClass]
public class Tests
Here is what we are going to change it to:
[TestClass]
public class Tests : Microsoft.Silverlight.Testing.SilverlightTest
Now we must decorate our test method with appropriate attribute to signal to test running harness that it is about to execute asynchronous test:
[TestMethod]
[Asynchronous]
[Description("Asynch Get Companies")]
[Tag("Asynch")]
public void TestGetCompanies()
Now, let take a deeper look into framework itself to see the key methods that we can use in base class – SilverlightTest.
First one is EnqueueConditional. This method is designed to let unit test wait until condition specified as an argument to EnqueueConditional is met. It is very useful when you want to wait until a method call to the service is completed. Here is how I would use this method:
[TestMethod]
[Asynchronous]
[Description("Asynch Get Companies")]
[Tag("Asynch")]
public void TestGetCompanies()
{
bool isLoaded = false;
Exception error = null;
IEnumerable<Company> results = null;
CompanyDomainContext context = new CompanyDomainContext(new Uri(@"http://localhost:4988/SilverlightTestingDemo-Web-CompanyDomainService.svc", UriKind.Absolute));
var query = context.GetCompaniesQuery();
context.Load<Company>(query, (result) =>
{
isLoaded = true;
results = result.Entities;
error = result.Error;
}, null);
EnqueueConditional(() => isLoaded);
What I am doing above is declaring a Boolean variable isLoaded that would signal to the framework to continue the test once data has been loaded. I am setting this variable to true in the lambda expression callback that is called one server method’s execution is completed – (result)=> expression.
Next interesting method is EnqueueCallback. This method simply declares multiple or in Action delegates that will be executed one they are popped off execution stack when harness processes test method calls. What I mean is that the harness will execute all Action delegates that EnqueueCallback declares when it proceeds with running a test method once condition that EnqueueConditional declares is met. If there are no such calls in the method body and there are no calls to EnqueueDelay that would also pause execution, then EnqueueCallback has no meaning, and asynchronous test essentially becomes synchronous. Here is what we have so far:
[TestMethod]
[Asynchronous]
[Description("Asynch Get Companies")]
[Tag("Asynch")]
public void TestGetCompanies()
{
bool isLoaded = false;
Exception error = null;
IEnumerable<Company> results = null;
CompanyDomainContext context = new CompanyDomainContext(new Uri(@"http://localhost:4988/SilverlightTestingDemo-Web-CompanyDomainService.svc", UriKind.Absolute));
var query = context.GetCompaniesQuery();
context.Load<Company>(query, (result) =>
{
isLoaded = true;
results = result.Entities;
error = result.Error;
}, null);
EnqueueConditional(() => isLoaded);
EnqueueCallback(
() => Assert.IsTrue(results.Count() > 0, "Should have data"),
() => Assert.IsNull(error, "Should have no errors"));
The last important method is EnqueueTestComplete. This method puts Test Completed signal onto the execution stack. This method is typically the very last statement of the test method body. Here is our final version of the method:
[TestMethod]
[Asynchronous]
[Description("Asynch Get Companies")]
[Tag("Asynch")]
public void TestGetCompanies()
{
bool isLoaded = false;
Exception error = null;
IEnumerable<Company> results = null;
CompanyDomainContext context = new CompanyDomainContext(new Uri(@"http://localhost:4988/SilverlightTestingDemo-Web-CompanyDomainService.svc", UriKind.Absolute));
var query = context.GetCompaniesQuery();
context.Load<Company>(query, (result) =>
{
isLoaded = true;
results = result.Entities;
error = result.Error;
}, null);
EnqueueConditional(() => isLoaded);
EnqueueCallback(
() => Assert.IsTrue(results.Count() > 0, "Should have data"),
() => Assert.IsNull(error, "Should have no errors"));
EnqueueTestComplete();
}
You can download the complete code here. In the next post I will cover mocking for synchronous and asynchronous unit tests.
Thanks..
Just what I needed! 🙂
I am getting an error GetCompanies Not Found. Please help.
This typically refers to url in SL applicaiton not beeing correct. Could you double-check your setup, inlcuding virtual directory creation?
Great post!