PHPackages                             droptica/codeception-drupal-content-types - PHPackages - PHPackages  [Skip to content](#main-content)[PHPackages](/)[Directory](/)[Categories](/categories)[Trending](/trending)[Leaderboard](/leaderboard)[Changelog](/changelog)[Analyze](/analyze)[Collections](/collections)[Log in](/login)[Sign up](/register)

1. [Directory](/)
2. /
3. [Testing &amp; Quality](/categories/testing)
4. /
5. droptica/codeception-drupal-content-types

ActiveLibrary[Testing &amp; Quality](/categories/testing)

droptica/codeception-drupal-content-types
=========================================

A Codeception module to provide Drupal content types support.

1.4.5(9y ago)22.3k↓100%MITPHP

Since Aug 25Pushed 7y ago17 watchersCompare

[ Source](https://github.com/droptica/codeception-drupal-content-types)[ Packagist](https://packagist.org/packages/droptica/codeception-drupal-content-types)[ Docs](https://github.com/ixis/codeception-drupal-content-types)[ RSS](/packages/droptica-codeception-drupal-content-types/feed)WikiDiscussions develop Synced 1mo ago

READMEChangelogDependencies (1)Versions (13)Used By (0)

Drupal Content Type Registry
============================

[](#drupal-content-type-registry)

A [Codeception](http://www.codeception.com) module to provide a set of classes that encapsulate [Drupal](http://drupal.org) content types. This makes it much easier to quickly test standard Drupal functionality relating to content types, taking into account how they exist on your site.

It will test many things such as the content types admin page, the 'manage fields' page for each content type, and provides a useful createNode() method that can be used to quickly create test nodes, where you can provide the test data using specific values, random values, or a range of values where one is picked at random.

Install
-------

[](#install)

Install using composer, as follows:

```
{
    "repositories": [
        {
            "type": "vcs",
            "url": "https://github.com/ixis/codeception-drupal-content-types.git"
        }
    ],
    "require": {
        "codeception/codeception": "2.*",
        "ixis/codeception-drupal-content-types": "~1.4",
    },
}
```

Add the module to the suite configuration:

```
modules:
    enabled:
        - DrupalContentTypeRegistry
```

Configuration
-------------

[](#configuration)

Put a **contentTypes.yml** at your test root (unless you want a specific contentTypes.yml for each suite, in which case, see below).

Here's an example file:

```
GlobalFields:
    body:
        machineName:    body
        label:          Body
        type:           Long text and summary
        selector:       "#edit-body-und-0-value"
        widget:         Text area with a summary
        required:       true
    title:
        machineName:    title
        label:          Title
        type:           Node module element
        selector:       "#edit-title"
ContentTypes:
    news:
        humanName:    News
        machineName:  news
        fields:
            globals:
                - title
                - body
            field_image:
                machineName:    field_image
                label:          Image
                type:           Image
                selector:       "#edit-field-image"
                widget:         Media file selector
                required:       true
                testData:       "image1.png"
            field_icon:
                machineName:    field_icon
                label:          Icon
                type:           Text
                selector:       "#edit-field-icon"
                widget:         Text field
                skipRoles:
                    - editor
                    - publisher
                testData:
                    - smiley
                    - grumpy
                    - happy
                    - wacky
                preSteps:
                    - ["click", ["#button"]]
                    - ["fillField", ["#the-field", "the-value"]]
                postSteps:
                    - ["waitForJs", ["return jQuery.active == 0;"]]
        submit: "#edit-submit-me-please"
```

### GlobalFields

[](#globalfields)

In the first section, you can define fields that will be used across all of the content types on the site. This is useful for things like title and body fields, to save you having to redefine the exact same field on every content type.

GlobalFields are keyed by machine name (this should be the same name as the machine name of the Drupal field) and the values are the same as they would be if they were declared for the fields of a content type (see below).

### GlobalExtras

[](#globalextras)

As with GlobalFields above, you can define extras that will be used across all of the content types on the site. This is useful if you always want to fill out the published status of a node no matter what type it is, for example. See the "extras" section below for more.

### ContentTypes

[](#contenttypes)

Each content type should be keyed according to its machine name (although this is just a hint as machineName takes care of the actual naming, so you could give the content type any key you like).

- **humanName** is the way that the content type is named in the UI (and is case-sensitive).
- **machineName** is the way that the content type is named to Drupal, and should match whatever is set in Drupal.
- **fields** is a list of all of the fields on the content type, with their properties.
    - **globalFields** is a simple list of the "reused" fields on this type. If your content type has a field that simply reuses exactly a field from another content type, set it up in GlobalFields (above) and just reference it here. An exception would be if it had a slight difference, such as when you set a title, but you change its label from "Title" to something else. In this case, it wouldn't be able to be a global field.
    - **properties**: fields can have the following properties...
        - **machineName** is the machine name of the field as seen by Drupal. In general these will start with field\_ but there might be exceptions such as for title and body fields.
        - **label** is the human name (label) for this field, and should match exactly what is set in the UI, including case sensitivity.
        - **type** is the field type as set in the Drupal UI, on the "manage fields" page. Case-sensitive.
        - **selector** is the CSS or XPath selector used to pick out this field's element where it appears on a node create or edit page. Note that this is usually optional and if omitted, will be derived from the field name, which is usually enough.
        - **widget** is the name of the widget for this field as set in the Druapl UI, on the "manage fields" page. Case- sensitive. Some fields don't have widgets (such as title) so just leave it out. There is a list of field types that are exempted from having a widget, so the ContentTypeRegistry will be aware of this.
        - **required** can be set to "true" if the field is required. If it's not, just leave this out altogether.
        - **pre** can be used to specify an XPath selector for an element that should be clicked before the field is filled. If this is set, this element will be clicked and then the field will be filled. If not set, nothing will be clicked before the field is filled. This can be useful for elements that are hidden behind vertical tabs and would not be visible to the user unless the vertical tab is selected first.
        - **skippedRoles** is an array of role names that will not be able to see this field and should not attempt to fill it in.
        - **testData** should contain the dummy data used to test this field. Each field can be instructed to fill itself with test data and this is the data that will be used. Note that unless the field is mandatory and Drupal provides no default value for the field, testData can be left out of the yaml. If an array of values if provided, one can be chosen at random by the Field class. Special values can also be used here. See 'special values' below.
        - **preSteps** optional steps to run before filling the field. This is an extension to the **pre** option, but any method can be run, not just click. The format is a two element array. The first element is the method name and the second element is an array of arguments to pass to the method. e.g. `["checkOption", ["#checkbox-name"]]` will call `$I->checkOption("#checkbox-name")` prior to filling the field.
        - **postSteps** optional steps to run after filling a field. For example, `["waitForJs", ["return jQuery.active == 0;"]]` will wait for ajax calls to complete after field is filled before continuing.
- **extras** is a list of all extras (elements to interact with on the node edit form) that aren't fields in their own right. See below for more.
    - **globalExtras** is a list of the "reused" extras on this type. This works the same as globalFields but with things that are on the node form that are not actually fields.
- **submit** is the CSS or XPath used to find the submit button on the node add or edit form for this content type. The Drupal default is "#edit-submit" and this can be omitted if you're using the default on your site.

#### Special values

[](#special-values)

You can use a special value that will be substituted each time the field is created for testing. This is useful, for example, if you want to insert a random value. All special values begin with the identifier **special::** and then are followed by the type of special value. The types are listed below:

- **randomText** uses eight random alphanumeric characters.

Example:

```
testData: "special::randomText"
```

Specific widget types
---------------------

[](#specific-widget-types)

### AddressWidget

[](#addresswidget)

Set the selector for this widget as the first part of the selector for each individual address field inside the widget. For example, if the machine name is **field\_address** then you would set the selector as follows:

```
selector:   "#edit-field-address-und-0"
```

You will need to define the individual elements that go into the address widget since these can be defined differently on a per-widget basis.

You can do this with the **elements** key. The array key is the label for each of the fields and the value is the end of the selector used, which is joined to the selector described above.

```
elements:
    Company:        "-organisation"
    Address 1:      "-thoroughfare"
    Address 2:      "-locality"
    "Town/City":    "-locality"
```

If you need to set up testData for this field, you will need to put a wrapper around each group of test elements as follows:

```
testData:
    address1:
        Company:        Test location
        Address 1:      Test location thoroughfare
        Address 2:      Test location locality
        "Town/City":    Test location city
        County:         Test location county
        Postcode:       Test location postal code
    # Then, if necessary:
    address2:
        Company:        Test location 2
        Address 1:      Test location thoroughfare 2
        Address 2:      Test location locality 2
        "Town/City":    Test location city 2
        County:         Test location county 2
        Postcode:       Test location postal code 2
```

### CheckboxesWidget

[](#checkboxeswidget)

Remember that you only need to set up testData if the value of each or any checkbox needs to be altered.

If you need to set up testData for a CheckboxesWidget, make sure you put each group of boxes in its own wrapper, as follows:

```
testData:
    values1:
        Grapefruit: true
        Melon:      false
        Avocado:    true
    values2:
        "Big Hairy Kiwi Fruit/Kiwi Fruits": true
```

### WysiwygWidget

[](#wysiwygwidget)

Use this for WYSIWYG fields. The selector should be the ID of the text area element for this field, but without the "-value" part at the end. For example, for a body field, you might use:

```
selector: "#edit-body-und-0"
```

Currently the widget will switch to plain text format to enter data.

Suite-specific contentTypes.yml
-------------------------------

[](#suite-specific-contenttypesyml)

You can put a separate contentTypes.yml in each suite folder if you prefer, and these files can override the main contentTypes.yml (or just don't create a main one in the test root folder).

If you do this, you will need to add the following into your *suite's* \_bootstrap.php:

```
\Codeception\Module\Drupal\ContentTypeRegistry\SuiteSettings::$suiteName = 'mysuite';
```

The suite name should match the name of the directory in which the suite lives. This is because Codeception has no other way of knowing what directory to look in when it's trying to find the contentTypes.yml file. It knows the location of the root tests directory and has a list of the suites it's supposed to run, but can't determine the directory in which the current suite is running. If some way of doing this within Codeception is developed in the future, this extra step can be dropped.

Extras
------

[](#extras)

Sometimes, you will want to simulate the user clicking things on the node edit form that are not actually fields. This is where extras come in. You can do things like set the sticky status or the publication status of a node in this way. Here is an example:

```
ContentTypes:
    news:
        humanName:    News
        machineName:  news
        fields:
            globals:
                - title
                - body
            field_image:
                machineName:    field_image
                label:          Image
                type:           Image
                selector:       "#edit-field-image"
                widget:         Media file selector
                required:       true
                testData:       "image1.png"
        extras:
            published:
                machineName:    published
                label:          Published
                type:           List (text)
                selector:       "#edit-published"
                widget:         Select list
                testData:       Published
        submit: "#edit-submit-me-please"
```

As you can see, these slot alongside fields. You will normally have to set a selector manually for these because the naming conventions that apply to fields do not apply here. You can still use the widget property to tell this module what type of form widget is being used.

Node creation/deletion
----------------------

[](#node-creationdeletion)

During createNode/deleteNode the success status is checked by looking for the standard Drupal messages in an element e.g. `.messages`

Some themes have different selectors or may not display those messages at all. If that is the case, you can implement `seeCreateNodeWasSuccessful()` and/or `seeDeleteNodeWasSuccessful()` in your suite helper.

e.g.

```
class AcceptanceHelper extends \Codeception\Module
{
    /**
     * Check a node creation was successful.
     *
     * This overrides the default since the css selectors are different in
     * this site's theme.
     *
     * @see DrupalContentTypeRegistry::seeCreateNodeWasSuccessful()
     *
     * @param WebInterface $I
     *   A reference to the Actor being used.
     * @param string $msg
     *   The success message that should be displayed by Drupal.
     * @param int $nid
     *   The created nid.
     */
    public function seeCreateNodeWasSuccessful($I, $msg, $nid)
    {
        $I->see($msg, ".messages");
        $I->dontSee(" ", ".messages.error");
    }

    /**
     * Check a node deletion was successful.
     *
     * This overrides the default since this site redirects to the
     * homepage on node deletion and does not show a message. We
     * therefore do a check by editing the node and make sure it's
     * not found.
     *
     * @see DrupalContentTypeRegistry::seeDeleteNodeWasSuccessful()
     *
     * @param AuthenticatedSteps $I
     *   A reference to the Actor being used.
     * @param int $nid
     *   The deleted nid.
     */
    public function seeDeleteNodeWasSuccessful($I, $nid)
    {
        $I->amOnPage(NodePage::route($nid, true));
        $I->see("we can't find this page", "h1");
    }
}
```

###  Health Score

34

—

LowBetter than 77% of packages

Maintenance20

Infrequent updates — may be unmaintained

Popularity19

Limited adoption so far

Community17

Small or concentrated contributor base

Maturity70

Established project with proven stability

 Bus Factor1

Top contributor holds 70.6% of commits — single point of failure

How is this calculated?**Maintenance (25%)** — Last commit recency, latest release date, and issue-to-star ratio. Uses a 2-year decay window.

**Popularity (30%)** — Total and monthly downloads, GitHub stars, and forks. Logarithmic scaling prevents top-heavy scores.

**Community (15%)** — Contributors, dependents, forks, watchers, and maintainers. Measures real ecosystem engagement.

**Maturity (30%)** — Project age, version count, PHP version support, and release stability.

###  Release Activity

Cadence

Every ~37 days

Recently: every ~30 days

Total

9

Last Release

3614d ago

### Community

Maintainers

![](https://www.gravatar.com/avatar/675aee3c25d9199cdbff3dd4624b2696ddbef744f668923927caf647bbee55ac?d=identicon)[droptica](/maintainers/droptica)

---

Top Contributors

[![ixis-chris](https://avatars.githubusercontent.com/u/5684064?v=4)](https://github.com/ixis-chris "ixis-chris (108 commits)")[![ixisandyr](https://avatars.githubusercontent.com/u/5859244?v=4)](https://github.com/ixisandyr "ixisandyr (31 commits)")[![pfaocle](https://avatars.githubusercontent.com/u/216767?v=4)](https://github.com/pfaocle "pfaocle (7 commits)")[![chriscohen](https://avatars.githubusercontent.com/u/205978?v=4)](https://github.com/chriscohen "chriscohen (5 commits)")[![pawel-gorski-droptica](https://avatars.githubusercontent.com/u/13447277?v=4)](https://github.com/pawel-gorski-droptica "pawel-gorski-droptica (1 commits)")[![PJnes](https://avatars.githubusercontent.com/u/5631?v=4)](https://github.com/PJnes "PJnes (1 commits)")

### Embed Badge

![Health badge](/badges/droptica-codeception-drupal-content-types/health.svg)

```
[![Health](https://phpackages.com/badges/droptica-codeception-drupal-content-types/health.svg)](https://phpackages.com/packages/droptica-codeception-drupal-content-types)
```

###  Alternatives

[phpspec/prophecy

Highly opinionated mocking framework for PHP 5.3+

8.5k551.7M678](/packages/phpspec-prophecy)[vimeo/psalm

A static analysis tool for finding errors in PHP applications

5.8k77.5M6.7k](/packages/vimeo-psalm)[brianium/paratest

Parallel testing for PHP

2.5k118.8M753](/packages/brianium-paratest)[beberlei/assert

Thin assertion library for input validation in business models.

2.4k96.9M571](/packages/beberlei-assert)[mikey179/vfsstream

Virtual file system to mock the real file system in unit tests.

1.4k108.0M2.7k](/packages/mikey179-vfsstream)[orchestra/testbench

Laravel Testing Helper for Packages Development

2.2k39.1M32.0k](/packages/orchestra-testbench)

PHPackages © 2026

[Directory](/)[Categories](/categories)[Trending](/trending)[Changelog](/changelog)[Analyze](/analyze)
