Testing Your Database
The Test Class
In order to take advantage of the built-in database tools that CodeIgniter provides for testing, your
tests must extend CIUnitTestCase
and use the DatabaseTestTrait
:
<?php
namespace App\Database;
use CodeIgniter\Test\CIUnitTestCase;
use CodeIgniter\Test\DatabaseTestTrait;
class MyTests extends CIUnitTestCase
{
use DatabaseTestTrait;
// ...
}
Because special functionality executed during the setUp()
and tearDown()
phases, you must ensure
that you call the parent’s methods if you need to use those methods, otherwise you will lose much
of the functionality described here:
<?php
namespace App\Database;
use CodeIgniter\Test\CIUnitTestCase;
use CodeIgniter\Test\DatabaseTestTrait;
class MyTests extends CIUnitTestCase
{
use DatabaseTestTrait;
protected function setUp(): void
{
parent::setUp();
// Do something here....
}
protected function tearDown(): void
{
parent::tearDown();
// Do something here....
}
}
Setting Up a Test Database
When running database tests, you need to provide a database that can be used during testing. Instead of
using the PHPUnit built-in database features, the framework provides tools specific to CodeIgniter. The first
step is to ensure that you have set up a tests
database group in app/Config/Database.php.
This specifies a database connection that is only used while running tests, to keep your other data safe.
If you have multiple developers on your team, you will likely want to keep your credentials stored in the .env file. To do so, edit the file to ensure the following lines are present and have the correct information:
database.tests.hostname = localhost
database.tests.database = ci4_test
database.tests.username = root
database.tests.password = root
database.tests.DBDriver = MySQLi
database.tests.DBPrefix =
database.tests.port = 3306
Migrations and Seeds
When running tests, you need to ensure that your database has the correct schema set up and that it is in a known state for every test. You can use migrations and seeds to set up your database, by adding a couple of class properties to your test.
<?php
namespace App\Database;
use CodeIgniter\Test\CIUnitTestCase;
use CodeIgniter\Test\DatabaseTestTrait;
class MyTests extends CIUnitTestCase
{
use DatabaseTestTrait;
// For Migrations
protected $migrate = true;
protected $migrateOnce = false;
protected $refresh = true;
protected $namespace = 'Tests\Support';
// For Seeds
protected $seedOnce = false;
protected $seed = 'TestSeeder';
protected $basePath = 'path/to/database/files';
// ...
}
Migrations
$migrate
This boolean value determines whether the database migration runs before test.
By default, the database is always migrated to the latest available state as defined by $namespace
.
If false
, migration never runs. If you want to disable migration, set false
.
$migrateOnce
This boolean value determines whether the database migration runs only once. If you want
to run migration once before the first test, set true
. If not present or false
, migration
runs before each test.
$refresh
This boolean value determines whether the database is completely refreshed before test. If true
,
all migrations are rolled back to version 0.
$namespace
By default, CodeIgniter will look in tests/_support/Database/Migrations to locate the migrations
that it should run during testing. You can change this location by specifying a new namespace in the $namespace
properties.
This should not include the Database\Migrations sub-namespace but just the base namespace.
Important
If you set this property to null
, it runs migrations from all available namespaces like php spark migrate --all
.
Seeds
$seed
If present and not empty, this specifies the name of a Seed file that is used to populate the database with test data prior to test running.
$seedOnce
This boolean value determines whether the database seeding runs only once. If you want
to run database seeding once before the first test, set true
. If not present or false
, database seeding
runs before each test.
$basePath
By default, CodeIgniter will look in tests/_support/Database/Seeds to locate the seeds that it should run during testing.
You can change this directory by specifying the $basePath
property. This should not include the Seeds directory,
but the path to the single directory that holds the sub-directory.
Helper Methods
The DatabaseTestTrait class provides several helper methods to aid in testing your database.
Changing Database State
regressDatabase()
Called during $refresh
described above, this method is available if you need to reset the database manually.
migrateDatabase()
Called during setUp()
, this method is available if you need to run migrations manually.
seed($name)
Allows you to manually load a Seed into the database. The only parameter is the name of the seed to run. The seed
must be present within the path specified in $basePath
.
hasInDatabase($table, $data)
Inserts a new row into the database. This row is removed after the current test runs. $data
is an associative
array with the data to insert into the table.
<?php
$data = [
'email' => 'joe@example.com',
'name' => 'Joe Cool',
];
$this->hasInDatabase('users', $data);
Getting Data from Database
grabFromDatabase($table, $column, $criteria)
Returns the value of $column
from the specified table where the row matches $criteria
. If more than one
row is found, it will only return the first one.
<?php
$username = $this->grabFromDatabase('users', 'username', ['email' => 'joe@example.com']);
Assertions
dontSeeInDatabase($table, $criteria)
Asserts that a row with criteria matching the key/value pairs in $criteria
DOES NOT exist in the database.
<?php
$criteria = [
'email' => 'joe@example.com',
'active' => 1,
];
$this->dontSeeInDatabase('users', $criteria);
seeInDatabase($table, $criteria)
Asserts that a row with criteria matching the key/value pairs in $criteria
DOES exist in the database.
<?php
$criteria = [
'email' => 'joe@example.com',
'active' => 1,
];
$this->seeInDatabase('users', $criteria);
seeNumRecords($expected, $table, $criteria)
Asserts that a number of matching rows are found in the database that match $criteria
.
<?php
$criteria = [
'active' => 1,
];
$this->seeNumRecords(2, 'users', $criteria);