hi-media/pdo-tools

Simplifies the build of a database for testing purposes and allows you to easily verify the contents of a database after operations.

v1.1.1 2015-03-20 23:22 UTC

This package is not auto-updated.

Last update: 2024-03-26 01:04:22 UTC


README

pdo-tools simplifies the build of a database for testing purposes and allows you to easily verify the contents of a database after operations.

Specifically pdo-tools is a small PHP class extending PHPUnit_Framework_TestCase (see PHPUnit) allowing to dynamically build entirely new test databases, with:

  • no database, schema or privileges prerequisites,
  • keep the last N built databases (customizable, for monitoring and debugging),
  • easy fixtures,
  • additional specific assertions,
  • powerful combination of PDO/PDOStatement mock objects and callbacks, e.g. to mock a third-party database,
  • ready-to-use with Jenkins and local PHPUnit.

Table of Contents

Installation

  1. Class autoloading and dependencies are managed by Composer so install it following the instructions on Composer: Installation - *nix or just run the following command:

    $ curl -sS https://getcomposer.org/installer | php
  2. Add dependency to Himedia\PDOTools into require section of your composer.json:

    {
        "require": {
            "hi-media/pdo-tools": "1.*"
        }
    }

    and run php composer.phar install from the terminal into the root folder of your project.

  3. Include Composer's autoloader:

    <?php
    
    require_once 'vendor/autoload.php';
    …

Usage

Instantiation

Example:

class MyTestCase extends DbTestCase
{
    public function __construct($sName = null, array $aData = array(), $sDataName = '')
    {
        // BUILD_NUMBER environment variable is handled by Jenkins:
        $iSuffix = isset($_SERVER['BUILD_NUMBER']) ? $_SERVER['BUILD_NUMBER'] : floor(microtime(true));
        $sTestDbName = "tests_$iSuffix";
        $aDbBuilderDsn = array(
            'driver'   => 'pgsql',
            'hostname' => 'localhost',
            'port'     => 5432,
            'dbname'   => $sTestDbName,
            'username' => 'user',
            'password' => ''
        );
        
        $sDbBuildFile = '/path/to/buildfile.php';
        parent::__construct($aDbBuilderDsn, array(), $sDbBuildFile);
    }
}

where /path/to/buildfile.php (example here) is a build file describing how to create a fresh database, roles/users and schemas, with listing of fixtures to load

Simple test

Example including:

  • fixture loading,
  • comparison between result of executed SQL query, converted into CSV, and a CSV file,
  • clean up
public function testSimple ()
{
    // Load SQL dump file, possibly gzipped (.gz):
    $this->loadSqlDumpFile('/path/to/dump.sql');

    // calls to tested program…

    // Asserts that SQL query result is equal to CSV file content:
    $this->assertQueryResultEqualsCsv('SELECT … FROM A', '/path/to/expected.csv');
    
    // Asserts that SQL query doesn't return any rows:
    $this->assertQueryReturnsNoRows('SELECT * FROM B');
    
    // Optional clean up:
    $this->loadSqlDumpFile('/path/to/clean_up.sql');
}

where /path/to/fixture.sql is a typical SQL dump file, gzipped or not.

Note that in CSV files, following field's values are converted:

  • '∅'null
  • 't'true
  • 'f'false

Mocking a third-party database

In this example, successives calls to PDOStatement::fetch() method on the third-party database…

  • will return content of /path/to/fixture.csv file like calls to PDOStatement::fetch(PDO::FETCH_ASSOC) if initial statement of PDO::query() was 'SELECT … FROM A', and false if no more lines in specified CSV,
  • will return results of successives calls to the user callback if initial statement of PDO::query() was 'SELECT … FROM B'

All queries are internally normalized to simplify matching.

public function testWithMockDb ()
{
    /* @var $oMockPDO \Himedia\DW\Tests\Mocks\PDO|\PHPUnit_Framework_MockObject_MockObject */
    $oMockPDO = $this->getMock('Himedia\PDOTools\Mocks\PDO', array('query'));
    $that = $this;
    $oMockPDO->expects($this->any())->method('query')->will(
        $this->returnCallback(
            function ($sQuery) use ($that, $sResourcePath) {
                return $that->getPdoStmtMock(
                    $sQuery,
                    array(
                        'SELECT … FROM A' => '/path/to/fixture.csv',
                        'SELECT … FROM B' => function () {
                            static $i = 0;
                            return ++$i > 10 ? false : array('id' => $i, 'name' => md5(rand()));
                        }
                    )
                );
            }
        )
    );
    
    // injection of $oMockPDO, to mock third-party database…
    
    // calls to tested program…
    
    // assertions…
}

Documentation

API documentation is generated by ApiGen in the doc/api folder.

$ vendor/bin/apigen -c apigen.neon

Change log

See CHANGELOG file for details.

Contributions

All suggestions for improvement or direct contributions are very welcome. Please report bugs or ask questions using the Issue Tracker.

Versioning & Git branching model

For transparency into our release cycle and in striving to maintain backward compatibility, Padocc's engine is maintained under the Semantic Versioning guidelines. We'll adhere to these rules whenever possible.

The git branching model used for development is the one described and assisted by twgit tool: https://github.com/Twenga/twgit.

Copyrights & licensing

Licensed under the GNU Lesser General Public License v3 (LGPL version 3). See LICENSE file for details.