This library introduces a set of methods to check if a variable or variables obey some restrictions. These methods are especially useful for checking method parameters. All methods throw AssertionExceptionInterface should one or more assertions fail.
Imagine you write a function that adds two numbers and returns the result, like this:
function add($x, $y){
Assert::isNumeric($x); // throws exception if $x is not a number
Assert::isNumeric($y); // throws exception if $y is not a number
return $x + $y;
}
Here you can just write some assertions to make sure your function gets valid parameters, or do not run at all otherwise.
The main benefit of using this library is to get meaningful exception messages. Consider below:
function addToArray($key, $value){
Assert::isValidArrayIndex($key, 'key');
$this->array[$key] = $value;
}
Imagine addToArray
function gets passed new stdClass()
as its first parameter. Without assertion, PHP will generate
following message:
PHP Fatal error: Illegal offset type
but with assertion, you will get this message:
Variable with name "key" is expected to be valid array index, given stdClass.
which is way more meaningful.
By using assertion methods, you can prevent getting Trying to get property of non object
messages almost %100 of the
time, thus debug a lot easier.
Note: PHP 7.1 or higher is required.
- Install composer.
- Run
composer require fabstract/assert
.
- General operations
- isObject($value, [$name])
- isNotNull($value, [$name])
- isEqualTo($value, $expected, [$name])
- isNotEqualTo($value, $expected, [$name])
- isTypeExists($value, [$name])
- isClassExists($value, [$name])
- isInterfaceExists($value, [$name])
- isMethodExists($object_or_class_name, $method, [$name])
- isInArray($value, $allowed_value_list, [$type_strict], [$name])
- Type checkers
- isCallable($value, [$name])
- isString($value, [$name])
- isBoolean($value, [$name])
- isInt($value, [$name])
- isStringOrInt($value, [$name])
- isIntOrFloat($value, [$name])
- isFloat($value, [$name])
- isValidArrayIndex($value, [$name])
- isArray($value, [$name])
- isNumeric($value, [$name])
- isType($value, $type, [$name])
- isInstanceOf($value, $type, [$name])
- isOneOfTypes($value, $type_list, [$name])
- isImplements($value, $interface, [$name])
- isChildOf($value, $parent, [$name])
- String operations
- Array operations
- Int operations
- Number operations
Checks if given $value
is object. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isObject(new stdClass(), 'variable name'); // no exception
Assert::isObject(1, 'variable name'); // exception!
Checks if given $value
is not null. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isNotNull(5, 'variable name'); // no exception
Assert::isNotNull(null, 'variable name'); // exception!
Checks if given $value
is equal to $expected
. Throws exception if fails.
This method is type safe, meaning that it will fail when $value
is integer 1
and $expected
is string '1'
.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isEqualTo(1, 1, 'variable name'); // no exception
Assert::isEqualTo(null, null, 'variable name'); // no exception
Assert::isEqualTo(1, 1.0, 'variable name'); // exception!
Assert::isEqualTo(1, '1', 'variable name'); // exception!
Checks if given $value
is NOT equal to $expected
. Throws exception if fails.
This method is type safe, meaning that it will NOT fail when $value
is integer 1
and $expected
is string '1'
.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isEqualTo(1, 1.0, 'variable name'); // no exception
Assert::isEqualTo(1, '1', 'variable name'); // no exception
Assert::isEqualTo(1, 1, 'variable name'); // exception!
Assert::isEqualTo(null, null, 'variable name'); // exception!
Checks if there is a class or interface named $value
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
class SomeClass {}
interface SomeInterface {}
Assert::isTypeExists('someclass', 'variable name'); // no exception
Assert::isTypeExists('someinterface', 'variable name'); // no exception
Assert::isTypeExists('someclass2', 'variable name'); // exception!
Checks if there is a class named $value
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
class SomeClass {}
interface SomeInterface {}
Assert::isClassExists('someclass', 'variable name'); // no exception
Assert::isClassExists('someinterface', 'variable name'); // exception!
Assert::isClassExists('someclass2', 'variable name'); // exception!
Checks if there is an interface named $value
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
class SomeClass {}
interface SomeInterface {}
Assert::isInterfaceExists('someinterface', 'variable name'); // no exception
Assert::isInterfaceExists('someclass', 'variable name'); // exception!
Checks if there is a method called $method
inside $object_or_class_name
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
trait SomeTrait {
public function somePublicTraitMethod() {}
private function somePrivateTraitMethod() {}
}
trait SomeOtherTrait {
public function someOtherPublicTraitMethod() {}
}
class SomeClass {
use SomeTrait;
public function somePublicMethod(){}
private function somePrivateMethod(){}
}
Assert::isMethodExists('someclass', 'somePublicMethod', 'variable name'); // no exception
Assert::isMethodExists('someclass', 'somePrivateMethod', 'variable name'); // no exception
Assert::isMethodExists('someclass', 'somePublicTraitMethod', 'variable name'); // no exception
Assert::isMethodExists('someclass', 'somePrivateTraitMethod', 'variable name'); // no exception
Assert::isMethodExists('someclass', 'someOtherPublicTraitMethod', 'variable name'); // exception!
Assert::isMethodExists('someclass', 'someMethodThatDoesNotExist', 'variable name'); // exception!
Works with instances as well:
$instance = new SomeClass();
Assert::isMethodExists($instance, 'somePublicMethod', 'variable name'); // no exception
Assert::isMethodExists($instance, 'somePrivateMethod', 'variable name'); // no exception
Assert::isMethodExists($instance, 'somePublicTraitMethod', 'variable name'); // no exception
Assert::isMethodExists($instance, 'somePrivateTraitMethod', 'variable name'); // no exception
Assert::isMethodExists($instance, 'someOtherPublicTraitMethod', 'variable name'); // exception!
Assert::isMethodExists($instance, 'someMethodThatDoesNotExist', 'variable name'); // exception!
Checks if given $value
is in given array $allowed_value_list
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
$allowed_value_list = ['1', '2', '3'];
Assert::isInArray('1', $allowed_value_list, true, 'variable name'); // no exception
Assert::isInArray('1', $allowed_value_list, false, 'variable name'); // no exception
Assert::isInArray(1, $allowed_value_list, false, 'variable name'); // no exception
Assert::isInArray(1, $allowed_value_list, true, 'variable name'); // exception!
Checks if given $value
is callable. Throws exception if fails.
This does not throw for closures, built-in php functions, public instance methods and public static methods. It will throw exception for protected and private instance and static methods.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isCallable(function(){}, 'variable name'); // no exception
Assert::isCallable('str_replace', 'variable_name'); // no exception
class SomeClass {
public function somePublicMethod(){}
public static function somePublicStaticMethod(){}
protected function someProtectedMethod(){}
public function somePrivateMethod(){}
}
Assert::isCallable(['someclass', 'somePublicMethod'], 'variable name'); // no exception
Assert::isCallable(['someclass', 'somePublicStaticMethod'], 'variable name'); // no exception
Assert::isCallable(['someclass', 'someProtectedMethod'], 'variable name'); // exception!
Assert::isCallable(['someclass', 'somePrivateMethod'], 'variable name'); // exception!
Works with instances as well:
$instance = new SomeClass();
Assert::isCallable([$instance, 'somePublicMethod'], 'variable name'); // no exception
Assert::isCallable([$instance, 'somePublicStaticMethod'], 'variable name'); // no exception
Assert::isCallable([$instance, 'someProtectedMethod'], 'variable name'); // exception!
Assert::isCallable([$instance, 'somePrivateMethod'], 'variable name'); // exception!
Checks if given $value
is string. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isString('some string', 'variable name'); // no exception
Assert::isString('', 'variable name'); // no exception
Assert::isString(new stdClass(), 'variable name'); // exception!
Assert::isString(null, 'variable name'); // exception!
Checks if given $value
is boolean, that is true
or false
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isBoolean(true, 'variable name'); // no exception
Assert::isBoolean(false, 'variable name'); // no exception
Assert::isBoolean('true', 'variable name'); // exception!
Assert::isBoolean(1, 'variable name'); // exception!
Checks if given $value
is int. Throws exception if fails.
This method is type safe, meaning that it will fail when $value
is string '1'
.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isInt(1, 'variable name'); // no exception
Assert::isInt(0, 'variable name'); // no exception
Assert::isInt('1', 'variable name'); // exception!
Checks if given $value
is string or int. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isStringOrInt(0, 'variable name'); // no exception
Assert::isStringOrInt('0', 'variable name'); // no exception
Assert::isStringOrInt(null, 'variable name'); // exception!
Assert::isStringOrInt([], 'variable name'); // exception!
Checks if given $value
is int or float. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isIntOrFloat(0, 'variable name'); // no exception
Assert::isIntOrFloat(0.5, 'variable name'); // no exception
Assert::isIntOrFloat('0.5', 'variable name'); // exception!
Assert::isIntOrFloat('1', 'variable name'); // exception!
Checks if given $value
is float. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isFloat(0.5, 'variable name'); // no exception
Assert::isFloat('0.5', 'variable name'); // exception!
Checks if given $value
can be used as array index. Throws exception if fails.
Note: Remember that in PHP, only strings and integers can be index. Floats and null
are also valid indexes but they get converted
to integer or string first.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isValidArrayIndex(1, 'variable name'); // no exception
Assert::isValidArrayIndex('some string', 'variable name'); // no exception
Assert::isValidArrayIndex(null, 'variable name'); // no exception
Assert::isValidArrayIndex(0.5, 'variable name'); // no exception
Assert::isValidArrayIndex(-1, 'variable name'); // no exception
Assert::isValidArrayIndex(true, 'variable name'); // no exception
Assert::isValidArrayIndex(false, 'variable name'); // no exception
Assert::isValidArrayIndex([], 'variable name'); // exception!
Assert::isValidArrayIndex(new stdClass(), 'variable name'); // exception!
Checks if given $value
is array. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info
Assert::isArray([], 'variable name'); // no exception
Assert::isArray(1, 'variable name'); // exception!
Checks if given $value
is numeric value. Throws exception if fails.
Valid values are integers, floats and numeric strings.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isNumeric(1, 'variable name'); // no exception
Assert::isNumeric(1.5, 'variable name'); // no exception
Assert::isNumeric('1.5', 'variable name'); // no exception
Assert::isNumeric(INF, 'variable name'); // no exception
Assert::isNumeric([], 'variable name'); // exception!
Assert::isNumeric(true, 'variable name'); // exception!
Checks if given $value
is instance of $type
. Throws exception if fails.
This method works for classes and interfaces, and their children.
Note that null
values always throw.Traits also fails since traits are not types.
Optional parameter $name
is used for exceptions. See exceptions for more info.
class SomeClass {}
Assert::isType(new SomeClass(), SomeClass::class, 'variable name'); // no exception
Assert::isType(null, SomeClass::class, 'variable name'); // exception!
See isType($value, $type, [$name]).
Checks if given $value
is instance of one of types from $type_list
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
class SomeClass {}
Assert::isOneOfTypes(new SomeClass(), [SomeClass::class], 'variable_name'); // no exception
Assert::isOneOfTypes(new stdClass(), [SomeClass::class], 'variable_name'); // exception!
Checks if given $value
implements given $interface
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
interface SomeInterface {}
class SomeClass implements SomeInterface {}
Assert::isImplements(new SomeClass(), 'someinterface'); // no exception
Assert::isImplements('someclass', 'someinterface'); // no exception
Assert::isImplements(new stdClass(), 'someinterface'); // exception!
Assert::isImplements(new stdClass(), 'someclass'); // exception!
Checks if given $value
is child of $parent
. Throws exception if fails.
Note that $value
and $parent
are the same, this method still throws.
Optional parameter $name
is used for exceptions. See exceptions for more info.
class SomeClass {}
class ChildClass extends SomeClass {}
Assert::isChildOf('childclass', 'someclass', 'variable name'); // no exception
Assert::isChildOf(new ChildClass(), 'someclass', 'variable name'); // no exception
Assert::isChildOf(new ChildClass(), SomeClass::class, 'variable name'); // no exception
Assert::isChildOf(new SomeClass(), SomeClass::class, 'variable name'); // exception!
Works with interfaces and child interfaces too.
interface SomeInterface {}
interface ChildInterface {}
class SomeClass implements SomeInterface {}
Assert::isChildOf('someclass', 'someinterface', 'variable name'); // no exception
Assert::isChildOf('childinterface', 'someinterface', 'variable name'); // no exception
Assert::isChildOf('someinterface', 'someinterface', 'variable name'); // exception!
Checks if given $value
is empty string or not. Throws exception if fails.
Optional parameter $accept_blanks
determines whether blank strings are allowed or not.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isNotEmptyString('some string', false, 'variable name'); // no exception
Assert::isNotEmptyString(' ', true, 'variable name'); // no exception
Assert::isNotEmptyString(' ', false, 'variable name'); // exception!
Assert::isNotEmptyString('', true, 'variable name'); // exception!
Assert::isNotEmptyString('', false, 'variable name'); // exception!
Assert::isNotEmptyString(0, false, 'variable name'); // exception!
Checks if given $value
starts with given $starts_with
. Throws exception if fails.
Note that $starts_with
can be empty string, and if it is then no string value can generate an exception.
Also note that this method is case-sensitive.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::startsWith('string', 's', 'variable name'); // no exception
Assert::startsWith('string', 'str', 'variable name'); // no exception
Assert::startsWith(' string', ' ', 'variable name'); // no exception
Assert::startsWith('string', '', 'variable name'); // no exception (empty string never throws)
Assert::startsWith('string', 'a', 'variable name'); // exception!
Assert::startsWith('string', 'S', 'variable name'); // exception!
Checks if given $value
matches given $regex_pattern
. Throws exception if fails.
Note that if $regex_pattern
is not a valid regex pattern, again exception will be thrown.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isRegexMatches('string', '/[w]+/', 'variable name'); // no exception
Assert::isRegexMatches('string', '/[d]+/', 'variable name'); // exception!
Assert::isRegexMatches('string', 'string', 'variable name'); // exception! (invalid regex pattern)
Checks if given $value
is a valid regex pattern. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isRegexPattern('/regex/', 'variable name'); // no exception
Assert::isRegexPattern('/\w/', 'variable name'); // no exception
Assert::isRegexPattern('//', 'variable name'); // no exception
Assert::isRegexPattern('string', 'variable name'); // exception!
See isNotEmptyString($value, [$accept_blanks], [$name]).
Checks if given $value
is in $allowed_string_list
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isInStringArray('string', ['abcd', 'string']); // no exception
Assert::isInStringArray('', ['']); // no exception
Assert::isInStringArray('abcd', ['string']); // exception!
Checks if given $value
is not an empty array. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isNotEmptyArray(['1'], 'variable name'); // no exception
Assert::isNotEmptyArray([], 'variable name'); // exception!
Checks if given $value
is an array of given $type
. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
class SomeClass() {}
Assert::isArrayOfType([new SomeClass()], 'someclass', 'variable name'); // no exception
Assert::isArrayOfType([], 'someclass', 'variable name'); // no exception
Assert::isArrayOfType([new SomeClass(), 'string'], 'someclass', 'variable name'); // exception!
Checks if given $value
is an array of string. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isArrayOfType([], 'variable name'); // no exception
Assert::isArrayOfType(['a', 'b', 'c'], 'variable name'); // no exception
Assert::isArrayOfType(['string', null], 'someclass', 'variable name'); // exception!
Checks if given $value
is a sequential array. Throws exception if fails.
Sequential array is an array whose keys start from 0, and increments by 1.
Optional parameter $accept_empty
is used to decide whether to accept empty arrays or not.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isSequentialArray(['a', 'z', 99], false, 'variable name'); // no exception
Assert::isSequentialArray(['a', 'z', 99], true, 'variable name'); // no exception
Assert::isSequentialArray([], true, 'variable name'); // no exception
Assert::isSequentialArray(['a' => 0, 'b' => 1], true, 'variable name'); // exception!
Checks if given $value
is a positive integer. Throws exception if fails.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isPositiveInt(1, 'variable name'); // no exception
Assert::isPositiveInt(-1, 'variable name'); // exception!
Assert::isPositiveInt(INF, 'variable name'); // exception!
Checks if given $value
is an integer, but not a negative integer. Throws exception if fails.
Note that this method will throw exception if $value
is not an integer.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isNotNegativeInt(1, 'variable name'); // no exception
Assert::isNotNegativeInt(0, 'variable name'); // no exception
Assert::isPositiveInt(-1, 'variable name'); // exception!
Assert::isPositiveInt(INF, 'variable name'); // exception!
Assert::isPositiveInt('string', 'variable name'); // exception!
Checks if given $value
is a positive number. Throws exception if fails.
Optional parameter $allow_string
determines whether strings are treated as numbers.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isPositiveNumber(1, false, 'variable name'); // no exception
Assert::isPositiveNumber(1, true, 'variable name'); // no exception
Assert::isPositiveNumber('1', true, 'variable name'); // no exception
Assert::isPositiveNumber('1', false, 'variable name'); // exception!
Assert::isPositiveNumber(-1, false, 'variable name'); // exception!
Checks if given $value
is a number, but not a negative number. Throws exception if fails.
Note that this method will throw exception if $value
is not a number.
Optional parameter $allow_string
determines whether strings are treated as numbers.
Optional parameter $name
is used for exceptions. See exceptions for more info.
Assert::isNotNegativeNumber(1, false, 'variable name'); // no exception
Assert::isNotNegativeNumber(1, true, 'variable name'); // no exception
Assert::isNotNegativeNumber('1', true, 'variable name'); // no exception
Assert::isNotNegativeNumber(0, false, 'variable name'); // no exception
Assert::isNotNegativeNumber('0', true, 'variable name'); // no exception
Assert::isNotNegativeNumber('0', false, 'variable name'); // exception!
Assert::isNotNegativeNumber('1', false, 'variable name'); // exception!
Assert::isNotNegativeNumber(-1, false, 'variable name'); // exception!
Assert::isNotNegativeNumber('string', false, 'variable name'); // exception!
The optional parameter $name
at the end of every method is used for better exception messages. Consider following:
function divideIntegers($dividend, $divisor) {
Assert::isInt($dividend);
Assert::isInt($divisor);
return intdiv($dividend, $divisor);
}
Now imagine running this function with following parameters:
divideIntegers(5, '2');
This code, when executed, will produce the following:
Fabstract\Component\Assert\AssertionException: Variable is expected to be int, given string (no name was provided).
However, if $name
parameter is provided like this:
function divideIntegers($dividend, $divisor) {
Assert::isInt($dividend, 'dividend'); // <- name is provided
Assert::isInt($divisor, 'divisor'); // <- name is provided
return intdiv($dividend, $divisor);
}
divideIntegers(5, '2');
Then exception message will be more helpful:
Fabstract\Component\Assert\AssertionException: Variable with name "divisor" is expected to be int, given string.
Another usage of exceptions is extending them.
By default, every failed assertion will cause Assert::generateException()
method to be called. Note that this method
is protected, and all methods that throw exception do it like this:
static::throwException($name, $expected, $given);
This means that if a custom Assert class is created by extending Fabstract's Assert class, it is possible to make all
methods throw a custom exception, by overriding generateException
method alone:
class MyCustomAssertionException extends \Exception implements AssertionExceptionInterface
{
}
class MyCustomAssert extends \Fabstract\Component\Assert\Assert
{
protected static function generateException($name, $expected, $given)
{
$exception = parent::generateException($name, $expected, $given);
return new MyCustomAssertionException(
$exception->getMessage(),
$exception->getCode(),
$exception
);
}
}
function divideIntegers($dividend, $divisor) {
MyCustomAssert::isInt($dividend, 'dividend');
MyCustomAssert::isInt($divisor, 'divisor');
return intdiv($dividend, $divisor);
}
divideIntegers(5, '2');
Now this will produce following:
MyCustomAssertionException: Variable with name "divisor" is expected to be int, given string.
You can comfortably create classes that extend Assert for your libraries, and use try-catch
blocks to find which library
throws assertion exceptions by separating them by their classes.
try {
$app->run();
} catch (LoggerLibraryAssertionException $exception) {
// do something related to logger library
} catch (DateTimeLibraryAssertionException $exception) {
// do something related to datetime library
} catch (AssertionExceptionInterface $exception) {
// none of above
}