2
0
mirror of https://github.com/opencultureconsulting/php-basics.git synced 2025-03-30 00:00:15 +01:00
php-basics/src/DataStructures/StrictList.php

339 lines
8.9 KiB
PHP

<?php
/**
* Useful PHP Basics
* Copyright (C) 2023 Sebastian Meyer <sebastian.meyer@opencultureconsulting.com>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
declare(strict_types=1);
namespace OCC\Basics\DataStructures;
use InvalidArgumentException;
use SplDoublyLinkedList;
use OCC\Basics\Traits\Getter;
/**
* A type-sensitive, taversable List.
*
* @author Sebastian Meyer <sebastian.meyer@opencultureconsulting.com>
* @package opencultureconsulting/basics
*/
class StrictList extends SplDoublyLinkedList
{
use Getter;
/**
* Defines the allowed types for items.
*
* If empty, all types are allowed. Possible values are:
* - "array"
* - "bool"
* - "callable"
* - "countable"
* - "float" or "double"
* - "int" or "integer" or "long"
* - "iterable"
* - "null"
* - "numeric"
* - "object" or FQCN
* - "resource"
* - "scalar"
* - "string"
*
* Fully qualified class names (FQCN) can be specified instead of the
* generic type "object".
*/
protected array $allowedTypes = [];
/**
* Add/insert a new item at the specified index.
* @see SplDoublyLinkedList::add()
*
* @param int $index The index where the new item is to be inserted
* @param mixed $item The new item for the index
*
* @return void
*
* @throws InvalidArgumentException
*/
public function add(int $index, mixed $item): void
{
if (!$this->isAllowedType($item)) {
throw new InvalidArgumentException(
sprintf(
'Parameter 2 must be an allowed type, %s given.',
get_debug_type($item)
)
);
}
parent::add($index, $item);
}
/**
* Append items at the end of the list.
*
* @param mixed ...$items One or more items to append
*
* @return void
*
* @throws InvalidArgumentException
*/
public function append(mixed ...$items): void
{
foreach ($items as $count => $item) {
if (!$this->isAllowedType($item)) {
throw new InvalidArgumentException(
sprintf(
'Parameter %d must be an allowed type, %s given.',
$count + 1,
get_debug_type($item)
)
);
}
}
foreach ($items as $item) {
parent::push($item);
}
}
/**
* Get the allowed item types.
*
* @return array The list of allowed item types
*/
public function getAllowedTypes(): array
{
return $this->allowedTypes;
}
/**
* Check if item is an allowed type.
*
* @param mixed $item The item to check
*
* @return bool Whether the item is an allowed type
*/
public function isAllowedType(mixed $item): bool
{
if (count($this->allowedTypes) === 0) {
return true;
}
foreach ($this->allowedTypes as $type) {
$function = 'is_' . $type;
if (function_exists($function) && $function($item)) {
return true;
}
$fqcn = '\\' . ltrim($type, '\\');
if (is_object($item) && is_a($item, $fqcn)) {
return true;
}
}
return false;
}
/**
* Magic getter method for $this->allowedTypes.
* @see Getter
*
* @return array The list of allowed item types
*/
protected function magicGetAllowedTypes(): array
{
return $this->allowedTypes;
}
/**
* Set the item at the specified index.
* @see \ArrayAccess::offsetSet()
*
* @param ?int $index The index being set or NULL to append
* @param mixed $item The new item for the index
*
* @return void
*
* @throws InvalidArgumentException
*/
public function offsetSet(mixed $index, mixed $item): void
{
if (!$this->isAllowedType($item)) {
throw new InvalidArgumentException(
sprintf(
'Parameter 2 must be an allowed type, %s given.',
get_debug_type($item)
)
);
}
parent::offsetSet($index, $item);
}
/**
* Prepend items at the start of the list.
*
* @param mixed ...$items One or more items to prepend
*
* @return void
*
* @throws InvalidArgumentException
*/
public function prepend(mixed ...$items): void
{
foreach ($items as $count => $item) {
if (!$this->isAllowedType($item)) {
throw new InvalidArgumentException(
sprintf(
'Parameter %d must be an allowed type, %s given.',
$count + 1,
get_debug_type($item)
)
);
}
}
foreach ($items as $item) {
parent::unshift($item);
}
}
/**
* Push an item at the end of the list.
* @see SplDoublyLinkedList::push()
*
* @param mixed $item The item to push
*
* @return void
*
* @throws InvalidArgumentException
*/
public function push(mixed $item): void
{
if (!$this->isAllowedType($item)) {
throw new InvalidArgumentException(
sprintf(
'Parameter 1 must be an allowed type, %s given.',
get_debug_type($item)
)
);
}
parent::push($item);
}
/**
* Get string representation of $this.
* @see \Serializable::serialize()
*
* @return string String representation
*/
public function serialize(): string
{
return serialize($this->__serialize());
}
/**
* Restore $this from string representation.
* @see \Serializable::unserialize()
*
* @param string $data String representation
*
* @return void
*/
public function unserialize($data): void
{
$this->__unserialize(unserialize($data));
}
/**
* Prepend the list with an item.
* @see SplDoublyLinkedList::unshift()
*
* @param mixed $item The item to unshift
*
* @return void
*
* @throws InvalidArgumentException
*/
public function unshift(mixed $item): void
{
if (!$this->isAllowedType($item)) {
throw new InvalidArgumentException(
sprintf(
'Parameter 1 must be an allowed type, %s given.',
get_debug_type($item)
)
);
}
parent::unshift($item);
}
/**
* Create a type-sensitive, traversable list of items.
*
* @param iterable $items Initial set of items
* @param string[] $allowedTypes Allowed types of items (optional)
*
* @throws InvalidArgumentException
*/
public function __construct(iterable $items = [], array $allowedTypes = [])
{
if (array_sum(array_map('is_string', $allowedTypes)) !== count($allowedTypes)) {
throw new InvalidArgumentException(
'Allowed types must be array of strings or empty array.'
);
}
$this->allowedTypes = $allowedTypes;
$this->append(...$items);
}
/**
* Get debug information for $this.
*
* @return array Array of debug information
*/
public function __debugInfo(): array
{
return $this->__serialize();
}
/**
* Get array representation of $this.
*
* @return array Array representation
*/
public function __serialize(): array
{
return [
'StrictList::allowedTypes' => $this->allowedTypes,
'SplDoublyLinkedList::flags' => $this->getIteratorMode(),
'SplDoublyLinkedList::dllist' => iterator_to_array($this)
];
}
/**
* Restore $this from array representation.
*
* @param array $data Array representation
*
* @return void
*/
public function __unserialize(array $data): void
{
$this->__construct(
$data['SplDoublyLinkedList::dllist'],
$data['StrictList::allowedTypes']
);
$this->setIteratorMode($data['SplDoublyLinkedList::flags']);
}
}