pwm / tc
Type safe collections.
This package's canonical repository appears to be gone and the package has been frozen as a result.
Requires
- php: >=7.1.0
Requires (Dev)
- phpstan/phpstan: ^0.7.0
- phpunit/phpunit: ^6.1
- squizlabs/php_codesniffer: ^3.0
This package is auto-updated.
Last update: 2023-04-28 22:59:13 UTC
README
TypedCollection
is a simple abstract collection class that type checks its elements upon creation ensuring a typed collection.
Collections created using TypedCollection
are immutable, iterable, countable and listable. If you need a mutable container you can simply subtype it and add the relevant mutators.
The library comes with a few concrete implementations, namely: Bool, Callable, Float, Int and String collections, but it's trivial to implement your own custom typed collection using any custom type.
Table of Contents
Why
With the advent of PHP7 the language started supporting type declarations to a degree, so we can finally write code like:
function isEven(int $x): bool { return $x % 2 === 0; }
where it is guaranteed that $x
is an int
as the return type is bool
. On the other hand we still can't say things like we want to get a list of integers, ie.:
function sum(int[] $xs): int { // can't write int[], only array return array_sum($xs); }
is not valid php syntax. We can only use array
as type declaration here and thus something nonsensical like sum([1, 'a', false])
can't be caught by the type checker.
Taking the above one step further it's perfectly valid to only want to guarantee that elements of a list are of a specific type, regardless of what the actual type is. This might sounds strange at first, so here is an example:
$addOne = function (int $x): int { return $x + 1; }; $toUpper = function (string $s): string { return strtoupper($s); }; array_map($addOne, [1, 2, 3]); // [2, 3, 4] array_map($toUpper, ['a', 'b', 'c']); // ['A', 'B', 'C']
Map cares neither about its function's argument type nor about the type of elements of its list. It only cares about 2 things:
- That the list's elements are of all the same type
- That this type is the same as its function's argument type
With PHP's array
we can't guarantee either. With TypedCollection
we can guarantee the first:
function map(callable $f, TypedCollection $c) { return array_map($f, $c->toList()); }
guarantees that elements of $c
are of the same type, ie.:
map($addOne, new IntCollection(1, 2, 3)); // [2, 3, 4] map($toUpper, new StringCollection('a', 'b', 'c')); // ['A', 'B', 'C']
but
map($addOne, new IntCollection(1, 'a', false)); // Fatal error: Uncaught TypeError
A small but important step towards more type safe and robust code.
Requirements
PHP 7.1+
Installation
$ composer require pwm/tc
Usage
To create a concrete typed collection we extend TypedCollection
and supply a function with a type declaration to it.
As an example here is the integer collection:
class IntCollection extends TypedCollection { public function __construct(...$ints) { parent::__construct(function (int $int) { return $int; }, $ints); } }
To use it:
$intCollection = new IntCollection(1, 2, 3); // iterable foreach ($intCollection as $element) { assert(is_int($element) === true); } // countable assert(count($intCollection) === 3); // listable assert($intCollection->toList() === [1, 2, 3]);
Basic scalar typed collections, ie. bool, callable, float, int and string is part of the lib for convenience.
How it works
The trick is to supply a function to TypedCollection
that requires its argument to be of a particular type. For this we just use PHP's built in function type declaration. This works with built in types as well as user defined types (aka. classes). TypedCollection
also checks that the supplied type checker function really does type check, ie. it does throw if it gets an untyped function.
TypedCollection
implements the IteratorAggregate
and Countable
interfaces for ease of use. getIterator()
returns a Generator
which aligns well with the immutable semantics of the collection, ie. you can iterate it accessing its elements, without having access to the container itself.
To get the underlying mutable array representation use toList()
. This is useful for using map/filter/reduce.
Tests
$ vendor/bin/phpunit
$ composer phpcs
$ composer phpstan