Skip to content

BDD style code blocks for PHPUnit / Codeception

License

Notifications You must be signed in to change notification settings

501st-alpha1/Specify

 
 

Repository files navigation

Specify

BDD style code blocks for PHPUnit / Codeception

Specify allows you to write your tests in more readable BDD style, the same way you might have experienced with Jasmine. Inspired by MiniTest of Ruby now you combine BDD and classical TDD style in one test.

Build Status Latest Stable Version

Additionaly, we recommend to combine this with Codeception/Verify library, to get BDD style assertions.

<?
class UserTest extends PHPUnit_Framework_TestCase {

	use Codeception\Specify;

	public function setUp()
	{		
		$this->user = new User;
	}

	public function testValidation()
	{
		$this->assertInstanceOf('Model', $this->user);

		$this->specify("username is required", function() {
			$this->user->username = null;
			verify($user->validate(['username'])->false());	
		});

		$this->specify("username is too long", function() {
			$this->user->username = 'toolooooongnaaaaaaameeee',
			verify($user->validate(['username'])->false());			
		});

		// alternative, TDD assertions can be used too.
		$this->specify("username is ok", function() {
			$this->user->username = 'davert',
			$this->assertTrue($user->validate(['username']));			
		});				
	}
}
?>

Purpose

This tiny library makes your tests a bit readable, by orginizing test in well described code blocks. Each code block is isolated.

This means call to $this->specify does not affect any instance variable of a test class.

<?php
$this->user->name = 'davert';
$this->specify("i can change my name", function() {
   $this->user->name = 'jon';
   $this->assertEquals('jon', $this->user->name);
});
       
$this->assertEquals('davert', $this->user->name);
?>        

Failure in specify block won't get your test stopped.

<?php
$this->specify("failing but test goes on", function() {
	$this->fail('bye');
});
$this->assertTrue(true);

// Assertions: 2, Failures: 1
?>

If a test fails you will specification text in the result.

Isolation

Isolation is achieved by cloning object properties for each specify block. By default objects are cloned using deep cloning method. This behavior can be customized in order to speed up test execution by preventing some objects from cloning or switching to shallow cloning using clone operator. Some properties can be ignored from cloning using either global or local config settings.

Global Configuration

Cloning configuration can be set globally

<?php
// globally disabling cloning of properties
Codeception\Specify\Config::setIgnoredProperties(['user', 'repository']);
?>

See complete reference.

Local Configuration

Configuring can be done locally per test case

<?php
class UserTest extends \PHPUnit_Framework_TestCase
{
    use Codeception\Specify;

    function testUser()
    {
        // do not deep clone user property
        $this->specifyConfig()
            ->shallowClone('user');
    }
}

Only specific properties can be preserved in specify blocks:

<?php
class UserTest extends \PHPUnit_Framework_TestCase
{
    use Codeception\Specify;
    protected $user;
    protected $post;

    function testUser()
    {
        $this->user = 'davert';
        $this->post = 'hello world';

        $this->specifyConfig()
            ->cloneOnly('user');

        $this->specify('post is not cloned', function() {
            $this->user = 'john';
            $this->post = 'bye world';
        });
        $this->assertEquals('davert', $this->user); // user is restored
        $this->assertEquals('bye world', $this->post); // post was not stored
    }
}

Reference

Exceptions

You can wait for exception thrown inside a block.

<?php

$this->specify('404 if user does not exist', function() {
	$this->userController->show(999);
}, ['throws' => 'NotFoundException']);

// alternatively
$this->specify('404 if user does not exist', function() {
	$this->userController->show(999);
}, ['throws' => new NotFoundException]);
?>

Also you can handle fails inside a block.

<?php

$this->specify('this assertion is failing', function() {
	$this->assertEquals(2, 3+5);
}, ['throws' => 'fail']);
?>

In both cases, you can optionally test the exception message

<?php

$this->specify('some exception with a message', function() {
	throw new NotFoundException("my error message');
}, ['throws' => ['NotFoundException', 'my error message']]);
?>

Examples

DataProviders alternative. Quite useful for basic data providers.

<?php
$this->specify("should calculate square numbers", function($number, $square) {
	$this->assertEquals($square, $number*$number);
}, ['examples' => [
		[2,4],
		[3,9]
]]);
?>

You can also use DataProvider functions in examples param.

<?php
$this->specify("should calculate square numbers", function($number, $square) {
	$this->assertEquals($square, $number*$number);
}, ['examples' => $this->provider()]);
?>

Before/After

There are also before and after callbacks, which act as setUp/tearDown but only for specify.

<?php
$this->beforeSpecify(function() {
	// prepare something;	
});
$this->afterSpecify(function() {
	// reset something
});
$this->cleanSpecify(); // removes before/after callbacks
?>

Installation

Requires PHP >= 5.4.

Install with Composer:

"require-dev": {
    "codeception/specify": "*",
    "codeception/verify": "*"

}

Include Codeception\Specifiy trait into your test.

License: MIT

About

BDD style code blocks for PHPUnit / Codeception

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • PHP 100.0%