Using Fixtures

Fixtures establish data that’s predictable in each test run for making assertions.

They can be defined in the fixturesMethod defined in the codeception.yml file. You can read more in the Yii2 docs (opens new window) about fixture classes and data and how these can be used for testing.

To setup fixtures Create a folder called fixtures in your tests folder. In this folder we will put our fixture classes and fixture data.

# Craft-Specific Data

Regular fixtures will suffice for traditional database rows. Because Craft utilizes complex linked data structures, it introduces some of its own fixtures for an improved testing experience.

# Element Fixtures

Craft’s powerful element types are a key selling point for developers, but their under-the-hood complexity can be a challenge for testing. For this reason, Craft’s test suite provides support for setting up various element types.

Craft’s element fixtures are based on the robuust (opens new window) team’s Fixtures plugin (opens new window).

# Working with your own custom element type?

You can extend craft\test\fixtures\elements\ElementFixture (opens new window) for your own testing and to offer other developers the ability to use your element type as a fixture testing their code.

# Asset Fixtures

You can add Asset fixtures by extending craft\test\fixtures\elements\AssetFixture (opens new window).

The fixture data file could look like this:

<?php

return [
    [
        'tempFilePath' => 'path/to/_craft/storage/runtime/temp/product.jpg',
        'filename' => 'product.jpg',
        'volumeId' => $this->volumeIds['products'],
        'folderId' => $this->folderIds['clothes'],
    ],
];

product.jpg should live alongside other testing assets in tests/_craft/assets.

This will upload and link product.jpg as an Asset.

AssetFixture will define $this->volumeIds and $this->folderIds with their handles as key.

The primary keys are: volumeId, folderId, filename and title.

The AssetFixture class will automatically copy your assets into the tests/_craft/storage/runtime/temp folder. Please ensure the tempFilePath points to a filename this directory.

# Category Fixtures

You can add Category fixtures by extending craft\test\fixtures\elements\CategoryFixture (opens new window).

The fixture data file could look like this:

<?php

return [
    [
        'groupId' => $this->groupIds['categories'],
        'title' => 'Category',
    ],
];

CategoryFixture will define $this->groupIds with all category group handles as key.

The primary keys are: siteId, groupId and title.

# Entry Fixtures

You can add Entry fixtures by extending craft\test\fixtures\elements\EntryFixture (opens new window).

The fixture data file could look like this:

<?php

return [
    [
        'sectionId' => $this->sectionIds['home'],
        'typeId' => $this->typeIds['home']['home'],
        'title' => 'Home',
    ],
];

EntryFixture will define $this->sectionIds with all section handles as key. It will also define $this->typeIds with all section handles as the first key and the entry type handles as the second key.

The primary keys are: siteId, sectionId, typeId and title.

# Global Set Fixture

You can add global set fixtures by extending craft\test\fixtures\elements\GlobalSetFixture (opens new window).

The fixture data file could look like this:

<?php

return [
    [
        'handle' => 'aHandle',
    ],
];

The primary keys are: handle.

By default, a global set doesn’t get its own database row. If you need it to, you can set $useActiveRecord to true.

# Tag Fixtures

You can add Tag fixtures by extending craft\test\fixtures\elements\TagFixture (opens new window).

The fixture data file could look like this:

<?php

return [
    [
        'groupId' => $this->groupIds['tags'],
        'title' => 'Tag',
    ],
];

TagFixture will define $this->groupIds with all tag group handles as key.

The primary keys are: siteId, groupId and title.

# User Fixtures

You can add User fixtures by extending craft\test\fixtures\elements\UserFixture (opens new window).

The fixture data file could look like this:

<?php

return [
    [
        'username' => 'User',
        'email' => 'info@domain.tld',
    ],
];

The primary keys are: siteId, username and email.

# Element Fixture Field Layout & Content

If you pass a fieldLayoutType into any class that extends the base ElementFixture class, Craft will automatically try to find the associated field layout and link it to the new Element you’re creating.

If you want to set custom field values you can simply include those into your fixture data file, prefixing the custom field handle with field:.

Here’s an example of a fixture data file that creates an entry with a title and custom fields:

<?php
return [
    [
        // Standard `craft\elements\Entry` fields.
        'authorId' => '1',
        'sectionId' => '1000',
        'typeId' => '1000',
        'title' => 'Theories of matrix',

        // Set a field layout
        'fieldLayoutType' => 'field_layout_with_matrix_and_normal_fields',

        // Set custom field values
        'field:myPlainTextFieldHandle' => 'value of text field',
        'field:myOtherPlainTextFieldHandle' => 'another value',

        // Set custom Matrix field values
        'field:matrixFirst' => [
            'new1' => [
                'type' => 'myBlockType',
                'fields:mySubfield' => 'Some text'
            ],
            'new2' => [
                'type' => 'myBlockType',
                'fields:mySubfield' => 'Some text'
            ],
        ],
    ]
];

# Field Layout Fixtures

Field Layouts are another Craft-specific concept. They consist of:

  • the layouts themselves
  • Tabs
  • Fields

These are linked to one another and would be difficult to manage in a test environment with ordinary fixtures. Craft provides a special FieldLayoutFixture (opens new window) class that provides all the required support for field layouts along with their tabs and the underlying fields—this includes creating the fields in the {{%content}} table.

To use this, first create an ordinary fixture class extending craft\test\fixtures\FieldLayoutFixture (opens new window).

Then, add the public $dataFile = 'path/to/datafile/ property that points to a datafile containing at least the following keys (including the nested position).

<?php
use craft\fields\Matrix;
use craft\fields\PlainText;

return [
    [
        'type' => 'craft\test\Craft', // Required - can be set to whatever you want.
        'tabs' => [ // Required - Value can be set to an empty array[]
            [
                'name' => 'My First Tab', // Required
                'fields' => [ // Required - Value can be set to an empty array[]
                    [
                        'layout-link' => [ // Required
                            'required' => true // Required
                        ],
                        'field' => [
                            'name' => 'Test field', // Required
                            'handle' => 'myTestField', // Required
                            'fieldType' => PlainText::class, // Required
                        ]
                    ],
                    // Matrix fields are supported in the following config:
                    [
                        'layout-link' => [
                            'required' => false
                        ],
                        'field' => [
                            'name' => 'Matrix Field',
                            'handle' => 'myMatrixField',
                            'fieldType' => Matrix::class,
                            'blockTypes' => [
                                'new1' => [
                                    'name' => 'A Block',
                                    'handle' => 'myMatrixBlock',
                                    'fields' => [
                                        'new1' => [
                                            'type' => PlainText::class,
                                            'name' => 'First Subfield',
                                            'handle' => 'myBlockField',
                                            'instructions' => '',
                                            'required' => false,
                                            'typesettings' => [
                                                'multiline' => ''
                                            ]
                                        ]
                                    ]
                                ],
                                'new2' => [
                                    'name' => 'Another Block',
                                    'handle' => 'myOtherMatrixBlock',
                                    'fields' => [
                                        'new1' => [
                                            'type' => PlainText::class,
                                            'name' => 'Another Subfield',
                                            'handle' => 'myOtherBlockField',
                                            'instructions' => '',
                                            'required' => false,
                                            'typesettings' => [
                                                'multiline' => ''
                                            ]
                                        ]
                                    ]
                                ]
                            ]
                        ]
                    ],
                ]
            ]
        ]
    ]
];

The value part of the key-value pairs can be set to whatever is required for your project. It’s crucial that the array keys are set with any string value. You can add your own parameters to the array value of the field key, as long as they correspond to public properties in the class defined in the fieldType key.