austp / peridot-gherkin-plugin
A Peridot plugin that adds a Gherkin style DSL.
Requires
- peridot-php/peridot: ^1.19
Requires (Dev)
- mockery/mockery: ^1.2
This package is auto-updated.
Last update: 2024-12-20 10:51:37 UTC
README
A Peridot plugin that adds a Gherkin style DSL.
Gherkin Style Tests
Feature: Some Feature
In order to ...
As a ...
I want to ...
Additional text...
Background:
Given something...
And something else...
Scenario: Some Scenario
Given something...
When something happens...
Then something else happens...
This is an example of a feature written in the Gherkin language. It looks nice but how do you convert this into a PHP test? This plugin will help you do that.
This plugin adds a DSL wrapper around Peridot so you can write your tests in a PHP-version of the Gherkin language. Here is an example for the above feature:
feature(
'Some Feature',
'',
'In order to...',
'As a ...',
'I want to...',
'',
'Additional text...',
background(
'Given something',
'And something else',
function () { /* setup function */ },
function () { /* optional teardown function */ }
),
scenario(
'Some scenario',
'',
'Given something',
'When something happens',
function () {},
'Then something else happens',
function () {}
)
);
This is the output for the above test:
Feature: Some Feature
In order to...
As a ...
I want to...
Additional text...
Background:
Given something
And something else
Scenario: Some scenario
✓ Given something
✓ When something happens
✓ Then something else happens
3 passing (2 ms)
DSL
This plugin adds 6 primary functions: feature
, background
, scenario
, isolatedScenario
, stories
, and isolatedStories
and 2 secondary functions: focusNextStory
and skipNextStory
.
feature
- The first argument is the name of the feature.
- Any
''
arguments are rendered as newlines. - Any other string arguments are rendered as-is.
- Can have the return value of any of the other 3 functions as an argument.
Other available methods:
xfeature
- Marks the feature as pending.
ffeature
- Focuses the feature.
background
- Any
''
arguments are rendered as newlines. - Any other string arguments are rendered as-is.
- The first callable argument encountered will be the setup function.
- The second callable argument encountered will be the teardown function.
Note: It is possible to call background
without supplying any strings to just add setup / teardown functions.
scenario
- The first argument is the title of the scenario.
- Any
''
arguments are ignored. - Any other string arguments will be used as test descriptions.
- Any callable arguments will be used as the test functions.
The scenario function is what determines what actually gets tested. Any string followed by a callable will be a "test". So in the above example, we had 2 tests. When something happens
and Then something else happens
because they were followed by a callable.
Notice that in the output, Given something
had a check by it. This plugin assumes that if no callable is given after a string, the logic for the string will be handled in the next callable. (You could have a callable after the Given something
argument, but oftentimes the logic is just a variable assignment which is untestable.)
If you fail to include a callable after including string arguments, they will be marked as pending tests.
Other available methods:
xscenario
- Marks the scenario as pending.
fscenario
- Focuses the scenario.
isolatedScenario
This behaves exactly as scenario
except for one key difference. This scenario will run in a separate blocking process. This allows the tests defined in the scenario to be isolated from the rest of the tests.
An example use case for using isolatedScenario
is when using Mockery to mock/spy a static method. The only way to accomplish this, is by making an alias. As long as the class hasn't been autoloaded yet, Mockery will load a fake class.
If you need access to the real class later on in your tests, there is no way to load it because Mockery has already loaded a class with the same name. But by running your tests in isolation, Mockery will load the fake class in a separate process. This leaves you free to load the real class when the time comes.
Here is an example to illustrate this use case. Assume we have a class named SomeClass
that has a static method named makeSomethingElseHappen
:
feature(
'Another Feature',
isolatedScenario(
'Some scenario',
'When something happens',
function () {
$this->spy = Mockery::spy('alias:SomeClass');
makeSomethingHappen();
},
'Then something else happens',
function () {
$this->spy->shouldHaveReceived('makeSomethingElseHappen');
}
)
);
feature(
'Yet another feature',
isolatedScenario(
'Another scenario',
'When something else happens',
function () {
$itHappened = SomeClass::makeSomethingElseHappen();
$this->itHappened = $itHappened;
},
'Then something else should have happened',
function () {
assert($this->itHappened === true);
}
)
);
In the above example, the first scenario is run in isolation. This allows us to test the makeSomethingElseHappen
method later on in our tests. (Note: Each feature
call should be in its own file).
Other available methods:
xisolatedScenario
- Marks the scenario as pending.
fisolatedScenario
- Focuses the scenario.
stories
- Any
''
arguments are ignored. - Any other string arguments will be used as test descriptions.
- Any callable arguments will be used as the test functions.
Although they don't fit in as nicely into the Gherkin language as scenarios do, sometimes it is helpful to define tests as stories rather than scenarios. This function will help you do that. It behaves identically to scenario
except for the fact that it doesn't require a title argument as it's first parameter.
feature(
'Stories Feature',
stories(
'I want it to do something',
function () {},
'I want it to do something else',
function () {}
)
);
Other available methods:
xstories
- Marks the stories as pending.
fstories
- Focuses the stories.
isolatedStories
This behaves exactly as stories
except it will run in a separate process. See isolatedScenario
for more details about running in a separate process.
Other available methods:
xisolatedStories
- Marks the stories as pending.
fisolatedStories
- Focuses the stories.
focusNextStory
Sometimes it's useful to focus a single story in a collection of stories.
feature(
'Focus next story feature',
stories(
// story definitions,
focusNextStory(),
'I want this test to be focused',
function () {}
)
);
skipNextStory()
This is similar to focusNextStory
, except it marks the next test to be skipped.
feature(
'Skip next story feature',
stories(
// story definitions,
skipNextStory(),
'I want this test to be skipped',
function () {}
)
);
Installation / Setup
To install:
composer require --dev austp/peridot-gherkin-plugin
In your peridot.php
file:
<?php
require('vendor/autoload.php');
use Peridot\Plugin\GherkinPlugin;
return function ($emitter) {
new GherkinPlugin($emitter);
};
This will configure Peridot to:
- Use the described DSL
- Look for tests in the
features/
directory - Look for tests in PHP files that end in
.feature.php
- Use a spec reporter built specifically for Gherkin