NAME
cucumber.pl - A Perl implementation of Aslak Hellesøy's Cucumber BDD framework.
SYNOPSIS
Assuming you have your features in './features/' and your steps in './features/steps/', you can run your Cucumber tests like this:
cucumber.pl
If your features and steps are in another directory, you can manually define them:
cucumber.pl --features=/dir/with/features --steps=/dir/with/steps
Should you wish to run a particular feature file, you can pass a single file:
cucumber.pl --feature_file=/some/file.feature
If you want to run only tagged scenarios, you can define the tag:
cucumber.pl --tag=some_tag
To run steps from a particular location and only given scenarios in a given file...
cucumber.pl --steps=/dir/with/steps --tag=some_tag --feature_file=/some/file.feature
The directories defined with '--steps' and '--features' are searched recursively for filenames that have '_steps.pl' and '.feature' on the end respectively. All other files are ignored. For example...
using_my_website_steps.pl
my_tests.feature
CUCUMBER DESCRIPTION
Having been impressed by the Cucumber Behavior Driven Development (BDD) suite for Rails, I thought it would be nice to have some basic implementation in Perl.
I won't go into much detail regarding Cucumber itself as there are websites that explain it well but the gist of it is this (see "SEE ALSO"): For BDD, you have non-developers (ideally) writing your tests. Not only does this cut down on development effort but also means no more 80 page func specs!
The structure is split down as features (the human readable test) and your steps, which will execute the actual code. Features are split into "Given", "When" and "Then". "Given" is your context (e.g. "Given I am on the home page"). "When" is the action performed (e.g. "When I click on 'blog'") and "Then" is your result check (e.g. "Then I should see 'my blogs'"). Wicked, init?
So lets see what our feature file for this will look like:
Scenario: I want to see my blog
Given I am on the home page
When I click on "blogs"
Then I will see "my blogs"
A feature file will contain many related "scenarios". The scenario is meta data and is used as an overall description of the task.
The feature file must have the extension of ".feature" and for simplicity we'll stuff it in "./features/".
So now lets have this actually do something. To do this, we need to create our step file. We'll call it "homepage_steps.pl" and it should live in "./features/steps/".
use Test::More qw(no_plan); # we'll use this for our tests
use HTTP::Request;
Given qr/^I am on the home page$/, sub {
$html = HTTP::Request->new('GET' => 'http://www.advancethinking.com');
# see "MR T" for details on the "%T" hash.
ok($html, $T{-name});
};
# any matches are passed in as parameters to the callback given below.
When qr/^I click on "(.+)"$/, sub {
my $urlLabel = shift;
# $urlLabel #=> blogs
my $content = $html->content(); # $html remains persistant
# yeah yeah, hacky
$content =~ /href="(.+)">$urlLabel</;
$html = HTTP::Request->new('GET' => $1);
ok($html, $T{-name});
};
Then qr/^I will see "(.+)"$/, sub {
my $pageTitle = shift;
like(
$html->content,
qr/title>$pageTitle</
);
};
Easy, init? Note: I've not actually tested the above! Have a look in 't/features/' of the distro to see simple examples.
What if you want to do multiple "Givens", "Whens" or "Thens"?? Well, that's easy, use an "And" or "But". For example
Scenario: When I was born
Given my mother is pregnant
And she is going in to labor
When she visits hospital
And so does my father
Then I will be born
And I will be a man
But I will not be a woman
"But" and "And" will repeat the previous step type. They are also technically identical but look pretty.
MR T
%T is an ambiguously but short named hash table that is global and available within your steps. It contains information about the current feature and step that is being run. It has the following keys...
-name => this is a string for convenience in your tests. It contains the current scenario and step (given, when, then string).
-step => the string of the step (e.g. "Given I am on the home page").
-stepType => the "type" of step ("Given", "Then" or "When").
-lineNumber => the line number being executed in the .feature file.
-featureFile => the filename of the currently executing feature.
-tableArray => a two dimentional array of a table (if there is one defined).
-tableHash => actually an array of hashes. each array element represents a row and each hash key contains the column header and the value is the content of the column.
SEE ALSO
Just to scratch the surface, look in the 't/features' directory for some simple examples. Also look at the following webpages (mostly Ruby & Java):
Cucumber home page: http://cukes.info/
BDD & Dan North: http://dannorth.net/introducing-bdd
Wikipedia Page: http://en.wikipedia.org/wiki/Behavior_Driven_Development
Test::More (not mandatory but certainly useful): http://search.cpan.org/~mschwern/Test-Simple-0.86/lib/Test/More.pm
TODO
- Add support for "Feature:" and other such Cucumberisms.
- Case insensitivity.
- Better, more modular code and make into distributable module.
- Grab the usefully named "Test::Cucumber" and maybe do something useful with it.
THANKS
To Dan North for BDD and Aslak Hellesøy for Cucumber.
I better get out of bed an enjoy my birthday now!
LICENCE AND COPYRIGHT
Copyright (c) 2009 Stephen Hardisty <moowahaha@hotmail.com>
This software is Free software and may be used and redistributed under the same terms as Perl itself.
1 POD Error
The following errors were encountered while parsing the POD:
- Around line 308:
Non-ASCII character seen before =encoding in 'Hellesøy's'. Assuming UTF-8