Ronald Bosma
Ronald Bosma
Software Architect
May 31, 2021 8 min read

Handling exceptions in SpecFlow

thumbnail for this post

I use Gherkin scenarios to describe the functional specifications of my software and SpecFlow to automate these scenarios as tests. Usually there will be a couple of scenarios describing the happy path of the feature I’m building but also some scenarios concerning failures. Depending on how the application code works, these failures are represented by exceptions. In this post I explain how I handle these exceptions.

Table of contents

Retrieve existing person successfully

Let’s start with the following happy path scenario to retrieve a person.

Scenario: Retrieve existing person successfully

Given the person 'Buffy Summers' is registered
When I retrieve the person 'Buffy Summers'
Then the person 'Buffy Summers' is returned

In the Given step we make sure the person exists in our system. We then retrieve the person and verify that the retrieval is successful.

The following PersonSteps class implements this scenario.

[Binding]
class PersonSteps
{
    private readonly PersonRepository _people = new PersonRepository();
    private string _actualName;

    [Given(@"the person '(.*)' is registered")]
    public void GivenThePersonIsRegistered(string name)
    {
        _people.AddPerson(name);
    }
        
    [When(@"I retrieve the person '(.*)'")]
    public void WhenIRetrieveThePerson(string name)
    {
        _actualName = _people.GetPersonByName(name);
    }

    [Then(@"the person '(.*)' is returned")]
    public void ThenThePersonIsReturned(string expectedName)
    {
        Assert.IsNotNull(_actualName, "No person retrieved");
        Assert.AreEqual(expectedName, _actualName);
    }
}

It uses a simple in-memory PersonRepository to store people. The _actualName instance field is used to store the person that is retrieved so we can check if the retrieval was successful in the Then step. (For demo purposes we only store and retrieve the name of the person.)

And here’s the implementation of PersonRepository.

class PersonRepository
{
    private readonly HashSet<string> _people = new HashSet<string>();

    public void AddPerson(string name)
    {
        // For demo purposes we only store the name.
        _people.Add(name);
    }

    public string GetPersonByName(string name)
    {
        // For demo purposes we only check if de name is stored and return the name if it is stored.
        if (_people.Contains(name))
        {
            return name;
        }
        throw new PersonNotFoundException(name);
    }
}

As you can see a PersonNotFoundException is raised when the person can’t be found.

Retrieve unknown person and expect an error

To verify that an error is raised when a person can’t be found, I’ve added a second scenario.

Scenario: Retrieve unknown person and expect an error

Given no person is registered
When I retrieve the person 'Buffy Summers'
Then the error 'Person with name Buffy Summers not found' should be raised

This scenario makes sure no person is registered. It then tries to retrieve a person and validates that an error is raised with the correct message.

If you execute this scenario with the current implementation of our steps the scenario will fail on the When step because we’re not handling the exception. As the code below shows, you can fix this by adding a try catch block in the When step that stores the raised exception in an instance field and check the exception message in the Then step.

private Exception _actualException;

[When(@"I retrieve the person '(.*)'")]
public void WhenIRetrieveThePerson(string name)
{
    try
    {
        _actualName = _people.GetPersonByName(name);
    }
    catch (Exception ex)
    {
        _actualException = ex;
    }
}

[Then(@"the error '(.*)' should be raised")]
public void ThenTheErrorShouldBeRaised(string expectedErrorMessage)
{
    Assert.IsNotNull(_actualException, "No error was raised");
    Assert.AreEqual(expectedErrorMessage, _actualException.Message);
}

With this implementation both scenarios will succeed as expected.

Note that I’m only checking the message of the exception and not the type. This is on purpose because the business is not familiar with or interested in exception types and I want to keep my scenarios as functional as possible. A unit test can be used to check the actual exception type if necessary.

Expected error was not raised

It’s also important that my scenarios fail when something goes wrong. Either because my implementation is wrong or the scenario has an error. Take the following two scenarios for example. I expect a specific error to be raised but this does not happen.

Scenario: Should fail: retrieve person that exists but expect error

Given the person 'Buffy Summers' is registered
When I retrieve the person 'Buffy Summers'
Then the error 'Person with name Buffy Summers not found' should be raised


Scenario: Should fail: different error message expected

Given no person is registered
When I retrieve the person 'Buffy Summers'
Then the error 'Something went wrong' should be raised

Both scenarios should and will fail. The first fails because I’m retrieving a person that exists but I expect the error that the person does not exist. The second scenario fails because I’m expecting an error with the wrong message.

Check for unexpected errors

One case that is often forgotten with this solution is to check for unexpected errors. Take the following scenario.

Scenario: Should fail: retrieve unknown person but don't check error

Given no person is registered
When I retrieve the person 'Buffy Summers'

I’m retrieving a person that is not registered. In the initial implementation of our When step without the try catch block this scenario would fail because an exception is raised in the When step. But now that I catch exceptions the scenario succeeds when it should fail.

Note that this scenario is missing a Then step so it’s not the greatest real-life example. I have seen this issue however in past projects with scenarios that succeeded when an unexpected error was raised even with a Then step. So, a bug in our production code or test automation code was flying under the radar.

To fix this issue we can use an AfterScenario hook to check if an unexpected error has occurred after a scenario has been executed. (See the SpecFlow documentation for more information on hooks.)

[AfterScenario]
public void CheckForUnexpectedExceptionsAfterEachScenario()
{
    Assert.IsNull(_actualException, $"No exception was expected to be raised but found exception: {_actualException}");
}

If _actualException has a value when the AfterScenario hook is executed then the exception was unexpected and the scenario will fail.

To make sure that scenarios where I do expect an error don’t fail I clear _actualException after checking the error. See the altered implementation of the Then step below.

[Then(@"the error '(.*)' should be raised")]
public void ThenTheErrorShouldBeRaised(string expectedErrorMessage)
{
    Assert.IsNotNull(_actualException, "No error was raised");
    Assert.AreEqual(expectedErrorMessage, _actualException.Message);

    // Clear the caught exception so it's not marked as unexpected in the AfterScenario hook
    _actualException = null;
}

A full example of the implementation so far can be found in this project.

Refactor to reusable code

The current solution works great when I’m retrieving a person but I usually have more features and When steps that need this kind of error handling logic. Also, the Then the error '<message>' should be raised step text is generic but can’t be reused over multiple step classes right now because of the use of the _actualException instance field.

To fix this I’ve created a generic ErrorDriver class following the Driver pattern described in the SpecFlow documentation. This class can catch and track exceptions and has a few helper methods for validation.

Refactored PersonSteps class

Before showing the ErrorDriver implementation I’ll first show how it’s used in the refactored PersonSteps class.

[Binding]
class PersonSteps
{
    private readonly PersonRepository _people = new PersonRepository();
    private string _actualName;

    private readonly ErrorDriver _errorDriver;

    public PersonSteps(ErrorDriver errorDriver)
    {
        _errorDriver = errorDriver;
    }

    /* Given steps omitted */

    [When(@"I retrieve the person '(.*)'")]
    public void WhenIRetrieveThePerson(string name)
    {
        _errorDriver.TryExecute(() =>
            _actualName = _people.GetPersonByName(name)
        );
    }

    /* Then step omitted */
}

As you can see the new ErrorDriver class is injected into the PersonSteps class via context injection. The person specific Given and Then steps are unaltered. The When step no longer has a try catch block. Instead the action of retrieving a person is passed into the TryExecute method of the ErrorDriver class as a lambda. The TryExecute method will catch any exception as you’ll see in moment. The _actualException private field is no longer used and has been removed.

New generic ErrorSteps

The Then the error '<message>' should be raised step and the AfterScenario hook are generic methods that can be reused for other features. I’ve moved these to a separate ErrorSteps class as shown below.

[Binding]
class ErrorSteps
{
    private readonly ErrorDriver _errorDriver;

    public ErrorSteps(ErrorDriver errorDriver)
    {
        _errorDriver = errorDriver;
    }

    [Then(@"the error '(.*)' should be raised")]
    public void ThenTheErrorShouldBeRaised(string expectedErrorMessage)
    {
        _errorDriver.AssertExceptionWasRaisedWithMessage(expectedErrorMessage);
    }

    [AfterScenario]
    public void CheckForUnexpectedExceptionsAfterEachScenario()
    {
        _errorDriver.AssertNoUnexpectedExceptionsRaised();
    }
}

This class also receives the ErrorDriver class via context injection. It uses the available Assert methods on ErrorDriver to verify if an expected or unexpected error has occurred.

The ErrorDriver class

And here’s the implementation of the ErrorDriver class.

class ErrorDriver
{
    private readonly Queue<Exception> _exceptions = new Queue<Exception>();

    public void TryExecute(Action act)
    {
        try
        {
            act();
        }
        catch (Exception ex)
        {
            Trace.WriteLine($"The following exception was caught while executing {act.Method.Name}: {ex}");
            _exceptions.Enqueue(ex);
        }
    }

    public void AssertExceptionWasRaisedWithMessage(string expectedErrorMessage)
    {
        Assert.IsTrue(_exceptions.Any(), $"No exception was raised but expected exception with message: {expectedErrorMessage}");

        var actualException = _exceptions.Dequeue();
        Assert.AreEqual(expectedErrorMessage, actualException.Message);
    }

    public void AssertNoUnexpectedExceptionsRaised()
    {
        if (_exceptions.Any())
        {
            var unexpectedException = _exceptions.Dequeue();
            Assert.Fail($"No exception was expected to be raised but found exception: {unexpectedException}");
        }
    }
}

As mentioned earlier the TryExecute method contains the try catch block and catches any exception raised by the action. When an exception is caught it will be written to a trace for troubleshooting and added to the _exceptions queue.

I’m using a queue so I’m able to handle the exceptions in the order they have occurred. Although there will usually only be 0 or 1 exception in the queue.

The AssertExceptionWasRaisedWithMessage method is used in the ErrorSteps class to verify if an expected error has occurred.

Lastly, the AssertNoUnexpectedExceptionsRaised method is used in the AfterScenario hook to check for any unexpected errors.

Conclusion

With the generic ErrorDriver and ErrorSteps classes we can quickly create scenarios that both support the happy flow and failures. This solution also protects against unexpected errors that have occurred but are not explicitly checked in a scenario. A case that is often forgotten when using this solution.

A full code example can be found here which also contains extra examples and code for dealing with asynchronous methods.