NEW FixtureContext (#2)

This commit is contained in:
Ingo Schommer 2013-06-04 01:11:07 +02:00
parent de60775b98
commit 30ece1347f
2 changed files with 278 additions and 38 deletions

View File

@ -141,53 +141,67 @@ To find out all available steps (and the files they are defined in), run the fol
Note: There are more specific step definitions in the SilverStripe `framework` module Note: There are more specific step definitions in the SilverStripe `framework` module
for interacting with the CMS interfaces (see `framework/tests/behat/features/bootstrap`). for interacting with the CMS interfaces (see `framework/tests/behat/features/bootstrap`).
### Fixtures ## Fixtures
Fixtures should be provided in YAML format (standard SilverStripe fixture format) Since each test run creates a new database, you can't rely on existing state unless
as [PyStrings](http://docs.behat.org/guides/1.gherkin.html#pystrings) you explicitly define it.
Take a look at the sample fixture logic first: ### Database Defaults
Given there are the following Permission records The easiest way to get default data is through `DataObject->requireDefaultRecords()`.
""" Many modules already have this method defined, e.g. the `blog` module automatically
admin: creates a default `BlogHolder` entry in the page tree. Sometimes these defaults can
Code: ADMIN be counterproductive though, so you need to "opt-in" to them, via the `@database-defaults`
""" tag placed at the top of your feature definition. The defaults are reset after each
And there are the following Group records scenario automatically.
"""
admingroup:
Title: Admin Group
Code: admin
Permissions: =>Permission.admin
"""
And there are the following Member records
"""
admin:
FirstName: Admin
Email: admin@test.com
Groups: =>Group.admingroup
"""
In this example, the fixture is used to create Admin member with admin permissions. ### Inline Definition
As you can see, there are special Gherkin steps that take care of loading If you need more flexibility and transparency about which records are being created,
fixtures into database. They use the following format: use the inline definition syntax. The following example shows some syntax variations:
Given there are the following TableName records Feature: Do something with pages
""" As an site owner
RowIdentifier: I want to manage pages
ColumnName: Value
"""
Fixtures may also use a `=>` symbol to indicate relationships between records. Background:
In the example above `=>Permission.admin` will be replaced with row `ID` of a # Creates a new page without data. Can be accessed later under this identifier
`Permission` record that has `RowIdentifier` set as `admin`. Given a page "Page 1"
# Uses a custom RegistrationPage type
And a registration page "Register"
# Creates a page with inline properties
And a page "Page 2" with "URLSegment"="page-1" and "Content"="my page 1"
# Field names can be tabular, and based on DataObject::$field_labels
And the page "Page 3" has the following data
| Content | <blink> |
| My Property | foo |
| My Boolean | bar |
# Pages are published by default, can be explicitly unpublished
And the page "Page 1" is not published
# Create a hierarchy, and reference a record created earlier
And the page "Page 1.1" is a child of a page "Page 1"
# Specific page type step
And a page "My Redirect" which redirects to a page "Page 1"
And a member "Website User" with "FavouritePage"="=>Page.Page 1"
Fixtures are created where you defined them. If you want the fixtures to be created @javascript
before every scenario, define them in [Background](http://docs.behat.org/guides/1.gherkin.html#backgrounds). If you want them to be created only when a particular scenario runs, define them there. Scenario: View a page in the tree
Given I am logged in with "ADMIN" permissions
And I go to "/admin/pages"
Then I should see "Page 1" in CMS Tree
Fixtures are usually not cleared between scenarios. You can alter this behaviour * Fixtures are created where you defined them. If you want the fixtures to be created
before every scenario, define them in [Background](http://docs.behat.org/guides/1.gherkin.html#backgrounds).
If you want them to be created only when a particular scenario runs, define them there.
* The basic syntax works for all `DataObject` subclasses, but some specific
notations like "is not published" requires extensions like `Hierarchy` to be applied to the class
* Record identifiers, property names and property values need to be quoted
* Record types shouldn't be quoted, and can use more natural notation ("registration page" instead of "Registration Page")
* Fixtures are usually not cleared between scenarios. You can alter this behaviour
by tagging the feature or scenario with `@database-defaults` tag. by tagging the feature or scenario with `@database-defaults` tag.
* Property values may also use a `=>` symbol to indicate relationships between records.
The notation is `=><classname>.<identifier>`. For `has_many` or `many_many` relationships,
multiple relationships can be separated by a comma.
The module runner empties the database before each scenario tagged with The module runner empties the database before each scenario tagged with
`@database-defaults` and populates it with default records (usually a set of `@database-defaults` and populates it with default records (usually a set of

View File

@ -0,0 +1,226 @@
<?php
namespace SilverStripe\BehatExtension\Context;
use Behat\Behat\Context\ClosuredContextInterface,
Behat\Behat\Context\TranslatedContextInterface,
Behat\Behat\Context\BehatContext,
Behat\Behat\Context\Step,
Behat\Behat\Event\StepEvent,
Behat\Behat\Exception\PendingException;
use Behat\Mink\Driver\Selenium2Driver;
use Behat\Gherkin\Node\PyStringNode,
Behat\Gherkin\Node\TableNode;
// PHPUnit
require_once 'PHPUnit/Autoload.php';
require_once 'PHPUnit/Framework/Assert/Functions.php';
/**
* Context used to create fixtures in the SilverStripe ORM.
*/
class FixtureContext extends BehatContext
{
protected $context;
/**
* @var \FixtureFactory
*/
protected $fixtureFactory;
protected $filesPath;
protected $createdFilesPaths;
public function __construct(array $parameters)
{
$this->context = $parameters;
}
public function getSession($name = null)
{
return $this->getMainContext()->getSession($name);
}
/**
* @return \FixtureFactory
*/
public function getFixtureFactory() {
if(!$this->fixtureFactory) {
$this->fixtureFactory = \Injector::inst()->create('FixtureFactory', 'FixtureContextFactory');
}
return $this->fixtureFactory;
}
/**
* @param \FixtureFactory $factory
*/
public function setFixtureFactory(\FixtureFactory $factory) {
$this->fixtureFactory = $factory;
}
/**
* Example: Given a page "Page 1"
*
* @Given /^(?:(an|a|the) )(?<type>[^"]+)"(?<id>[^"]+)"$/
*/
public function stepCreateRecord($type, $id)
{
$class = $this->convertTypeToClass($type);
$this->fixtureFactory->createObject($class, $id);
}
/**
* Example: Given a page "Page 1" with "URL"="page-1" and "Content"="my page 1"
*
* @Given /^(?:(an|a|the) )(?<type>[^"]+)"(?<id>[^"]+)" with (?<data>.*)$/
*/
public function stepCreateRecordWithData($type, $id, $data)
{
$class = $this->convertTypeToClass($type);
preg_match_all(
'/"(?<key>[^"]+)"\s*=\s*"(?<value>[^"]+)"/',
$data,
$matches
);
$fields = $this->convertFields(
$class,
array_combine($matches['key'], $matches['value'])
);
$this->fixtureFactory->createObject($class, $id, $fields);
}
/**
* Example: And the page "Page 2" has the following data
* | Content | <blink> |
* | My Property | foo |
* | My Boolean | bar |
*
* @Given /^(?:(an|a|the) )(?<type>[^"]+)"(?<id>[^"]+)" has the following data$/
*/
public function stepCreateRecordWithTable($type, $id, $null, TableNode $fieldsTable)
{
$class = $this->convertTypeToClass($type);
// TODO Support more than one record
$fields = $this->convertFields($class, $fieldsTable->getRowsHash());
$this->fixtureFactory->createObject($class, $id, $fields);
}
/**
* Example: Given the page "Page 1.1" is a child of the page "Page1"
*
* @Given /^(?:(an|a|the) )(?<type>[^"]+)"(?<id>[^"]+)" is a (?<relation>[^\s]*) of (?:(an|a|the) )(?<relationType>[^"]+)"(?<relationId>[^"]+)"/
*/
public function stepUpdateRecordRelation($type, $id, $relation, $relationType, $relationId)
{
$class = $this->convertTypeToClass($type);
$relationClass = $this->convertTypeToClass($relationType);
$obj = $this->fixtureFactory->get($class, $id);
if(!$obj) $obj = $this->fixtureFactory->createObject($class, $id);
$relationObj = $this->fixtureFactory->get($relationClass, $relationId);
if(!$relationObj) $relationObj = $this->fixtureFactory->createObject($relationClass, $relationId);
switch($relation) {
case 'parent':
$relationObj->ParentID = $obj->ID;
$relationObj->write();
break;
case 'child':
$obj->ParentID = $relationObj->ID;
$obj->write();
break;
default:
throw new \InvalidArgumentException(sprintf(
'Invalid relation "%s"', $relation
));
}
}
/**
* Example: Given the page "Page 1" is not published
*
* @Given /^(?:(an|a|the) )(?<type>[^"]+)"(?<id>[^"]+)" is (?<state>[^"]*)$/
*/
public function stepUpdateRecordState($type, $id, $state)
{
$class = $this->convertTypeToClass($type);
$obj = $this->fixtureFactory->get($class, $id);
if(!$obj) {
throw new \InvalidArgumentException(sprintf(
'Can not find record "%s" with identifier "%s"',
$type,
$id
));
}
switch($state) {
case 'published':
$obj->publish('Stage', 'Live');
break;
case 'not published':
case 'unpublished':
$oldMode = \Versioned::get_reading_mode();
\Versioned::reading_stage('Live');
$clone = clone $obj;
$clone->delete();
\Versioned::reading_stage($oldMode);
break;
default:
throw new \InvalidArgumentException(sprintf(
'Invalid state: "%s"', $state
));
}
}
/**
* Converts a natural language class description to an actual class name.
* Respects {@link DataObject::$singular_name} variations.
* Example: "redirector page" -> "RedirectorPage"
*
* @param String
* @return String Class name
*/
protected function convertTypeToClass($type)
{
$type = trim($type);
// Try direct mapping
$class = str_replace(' ', '', ucfirst($type));
if(class_exists($class) || !is_subclass_of($class, 'DataObject')) {
return $class;
}
// Fall back to singular names
foreach(array_values(\ClassInfo::subclassesFor('DataObject')) as $candidate) {
if(singleton($candidate)->singular_name() == $type) return $candidate;
}
throw new \InvalidArgumentException(sprintf(
'Class "%s" does not exist, or is not a subclass of DataObjet',
$class
));
}
/**
* Updates an object with values, resolving aliases set through
* {@link DataObject->fieldLabels()}.
*
* @param String Class name
* @param Array Map of field names or aliases to their values.
* @return Array Map of actual object properties to their values.
*/
protected function convertFields($class, $fields) {
$labels = singleton($class)->fieldLabels();
foreach($fields as $fieldName => $fieldVal) {
if(array_key_exists($fieldName, $labels)) {
unset($fields[$fieldName]);
$fields[$labels[$fieldName]] = $fieldVal;
}
}
return $fields;
}
}