-
Notifications
You must be signed in to change notification settings - Fork 22
Home
Cuke4Nuke is a project to allow Cucumber to support step definitions written in .NET. It uses a simple wire protocol for Cucumber to issue commands to a .NET server that knows how to read and invoke .NET step definitions.
The goal: to make Cucumber usable for .NET teams who can’t or don’t want to work directly with Ruby.
See https://rspec.lighthouseapp.com/projects/16211-cucumber/tickets/428-wire-protocol for more information about the development of the Cucumber wire protocol.
The native Java step definition support for Cucumber is called Cuke4Duke, after the Java mascot, Duke. When we started work on this project at AA-FTT 2009, we had to call it something. .NET versions of Java tools tend to get an N added somewhere, hence Cuke4Nuke. That’s all.
As of version 0.2.0, Cuke4Nuke is distributed as a Ruby Gem. To install it, do the following…
1. Download and install the .NET Framework 3.5 (comes with Visual Studio 2008).
2. Download and run the Ruby one click installer with the default options.
3. Update RubyGems:
gem update -y --system --include-dependencies
4. Add Gemcutter as a gem source:
gem sources -a http://gemcutter.org/
5. Install win32console to get color in the command window:
gem install win32console
6. Install cuke4nuke:
gem install cuke4nuke --no-ri --no-rdoc
There are two sides to using Cuke4Nuke: the Cucumber side and the .NET side.
The Cucumber side has a directory structure like this:
MyProject/features/
MyProject/features/some_feature.feature
MyProject/features/another_feature.feature
MyProject/features/step_definitions/
MyProject/features/step_definitions/cucumber.wire
The directories and the *.feature files are normal Cucumber stuff. The interesting bit is the .wire file. This tells Cucumber to go look for step definitions using the wire protocol instead of using the normal Ruby files. The .wire file has only two lines:
host: localhost
port: 3901
which specify the host and port to find the wire server with the Cucumber step definitions. (The default port for the Cuke4Nuke server is 3901. Use this in your .wire file unless you have a really good reason not to.)
You’ll need a version of Cucumber that includes the wire protocol language, which means 0.4.3 or newer (installed automatically with the cuke4nuke gem).
On the .NET side you need a DLL with step definitions. Step definitions are methods with one of the Cuke4Nuke.Framework step definition attributes (Given, When, or Then). For example:
[Then("^it should pass.$")]
public void ItShouldPass()
{
Assert.Pass();
}
(The assertion is from NUnit, which is the only test framework Cuke4Nuke currently understands how to work with.)
Assuming you have a feature file that includes
Then it should pass.
when you run
> cuke4nuke MyProject\bin\Debug\MyStepDefinitionLibrary.dll
it should run Cucumber and cause Cuke4Nuke to invoke your ItShouldPass() method and show a successful test. You no longer have to run the Cuke4Nuke server and Cucumber separately. The cuke4nuke command does both for you.
Capture groups in a step definition regular expression are passed as parameters to the step definition method. Parameters can be any type with a built-in TypeConverter (e.g. int, decimal, bool, etc.). For example:
[Given(@"^(\d+) cukes$")]
public void GivenSomeCukes(int cukes)
{
_cukeCount = cukes;
}
Most tests require some kind of shared state between steps. When a scenario is run, the same instance of a step definition class is used for each step invocation, so you can share state in private fields. For example:
public class CukeSteps
{
int _cukeCount = 0;
[Given(@"^(\d+) cukes$")]
public void GivenSomeCukes(int cukes)
{
_cukeCount = cukes;
}
[When(@"^I add (\d+) more cukes$")]
public void AddCukes(int cukes)
{
_cukeCount += cukes;
}
}
As you get more step definitions, you’ll probably want to organize them into separate classes. You can share state between multiple step definition classes by requiring the same type of object in each class’s constructor. Cuke4Nuke will ensure that each step definition object gets the same instance of the dependency class. For example:
public class CukeJar
{
public int CukeCount { get; set; }
}
public class Steps1
{
CukeJar _cukeJar;
public Steps1(CukeJar cukeJar)
{
_cukeJar = cukeJar;
}
[Given(@"^(\d+) cukes$")]
public void GivenSomeCukes(int cukes)
{
_cukeJar.CukeCount = cukes;
}
[When(@"^I add (\d+) more cukes$")]
public void AddCukes(int cukes)
{
_cukeJar.CukeCount += cukes;
}
}
public class Steps2
{
CukeJar _cukeJar;
public Steps2(CukeJar cukeJar)
{
_cukeJar = cukeJar;
}
[Then(@"^I should have (\d+) cukes$")]
public void ExpectCukes(int cukes)
{
Assert.That(_cukeJar.CukeCount, Is.EqualTo(cukes));
}
}
Of course, you’re welcome to fork this repository, make changes, and send a pull request. But if you want to coordinate your work with the others contributing to the project, speak up on the cukes mailing list with what you intend to work on, and I’ll update the Issues list accordingly.