in tdd, the term “acceptance tests” is misleading. acceptance tests actually represent the expected behavior of the system. in agile practices, collaboration of the whole team and interactions with the customer and other stakeholders is emphasized. this has given rise to the necessity of usage of terms that are easily understood by everyone involved in the project.
tdd makes you think about the required behavior and hence the term ‘behavior’ is more useful than the term ‘test’. bdd is test driven development with a vocabulary that focuses on behavior and not tests.
in the words of dan north, “i found the shift from thinking in tests to thinking in behavior so profound that i started to refer to tdd as bdd, or behavior driven development.” tdd focuses on how something will work, bdd focuses on why we build it at all.
bdd answers the following questions often faced by the developers −
question | answer |
---|---|
where to start? | outside-in |
what to test? | user stories |
what not to test? | anything else |
these answers result in the story framework as follows −
story framework
as a [role]
i want [feature]
so that [benefit]
this means, ‘when a feature is executed, the resulting benefit is to the person playing the role.’
bdd further answers the following questions −
question | answer |
---|---|
how much to test in one go? | very little-focused |
what to call their tests? | sentence template |
how to understand why a test fails | documentation |
these answers result in the example framework as follows −
example framework
given some initial context,
when an event occurs,
then ensure some outcomes.
this means, ‘starting with the initial context, when a particular event happens, we know what the outcomes should be.’
thus, the example shows the expected behavior of the system. the examples are used to illustrate different scenarios of the system.
story and scenarios
let us consider the following illustration by dan north about an atm system.
story
as a customer,
i want to withdraw cash from an atm,
so that i do not have to wait in line at the bank.
scenarios
there are two possible scenarios for this story.
scenario 1 − account is in credit
given the account is in credit
and the card is valid
and the dispenser contains cash
when the customer requests cash
then ensure the account is debited
and ensure cash is dispensed
and ensure the card is returned
scenario 2 − account is overdrawn past the overdraft limit
given the account is overdrawn
and the card is valid
when the customer requests cash
then ensure a rejection message is displayed
and ensure cash is not dispensed
and ensure the card is returned
the event is same in both the scenarios, but the context is different. hence, the outcomes are different.
development cycle
the development cycle for bdd is an outside-in approach.
step 1 − write a high-level (outside) business value example (using cucumber or rspec/capybara) that goes red. (rspec produces a bdd framework in the ruby language)
step 2 − write a lower-level (inside) rspec example for the first step of implementation that goes red.
step 3 − implement the minimum code to pass that lower-level example, see it go green.
step 4 − write the next lower-level rspec example pushing towards passing step 1 that goes red.
step 5 − repeat steps step 3 and step 4 until the high-level example in step 1 goes green.
note − the following points should be kept in mind −
red/green state is a permission status.
when your low-level tests are green, you have the permission to write new examples or refactor existing implementation. you must not, in the context of refactoring, add new functionality/flexibility.
when your low-level tests are red, you have permission to write or change implementation code only for making the existing tests go green. you must resist the urge to write the code to pass your next test, which does not exist, or implement features you may think are good (customer would not have asked).