map/src/Olivebbs/Map/GenericMap.php

230 lines
6.5 KiB
PHP
Raw Normal View History

2021-03-03 13:59:55 -08:00
<?php declare(strict_types=1);
/*
* Copyright (c) 2020 https://rewiv.com sikofitt@gmail.com
*
* This file is a part of Olive BBS
*
* This Source Code Form is subject to the
* terms of the Mozilla Public License, v. 2.0.
*
* If a copy of the MPL was not distributed with this file,
* You can obtain one at https://mozilla.org/MPL/2.0/.
*
* ___ ___ ___
* ( ).-. ( ) ( )
* .--. | |( __)___ ___ .--. | |.-. | |.-. .--.
* / \| |(''"( )( / \| / \| / \ / _ \
* | .-. | | | | | | | | .-. | .-. | .-. |. .' `. ;
* | | | | | | | | | | | | | | | | | | | || ' | |
* | | | | | | | | | | | |/ | | | | | | |_\_`.(___)
* | | | | | | | | | | | ' _.| | | | | | ( ). '.
* | ' | | | | | ' ' ; | .'.-| ' | | ' | || | `\ |
* ' `-' | | | | \ `' /' `-' ' `-' ;' `-' ; ; '._,' '
* `.__.(___(___) '_.' `.__.' `.__. `.__. '.___.'
*
*/
namespace Olivebbs\Map;
use ArrayAccess;
use function class_exists;
2021-03-03 13:59:55 -08:00
use Countable;
use Ds\Map;
use function enum_exists;
use function in_array;
2021-03-03 13:59:55 -08:00
use function is_array;
use function is_callable;
2021-03-03 13:59:55 -08:00
use function is_int;
use function is_object;
use function is_string;
use function mb_strlen;
2021-03-03 13:59:55 -08:00
use Olivebbs\Map\Exception\InvalidArgumentException;
use function sprintf;
use function strtolower;
2021-03-03 13:59:55 -08:00
use TypeError;
use ValueError;
class GenericMap implements ArrayAccess, Countable
{
public const OBJECT = 'object';
public const CALLABLE = 'callable';
2021-03-03 13:59:55 -08:00
public const ARRAY = 'array';
public const INT = 'int';
public const INTEGER = 'integer';
public const STRING = 'string';
public const CHAR = 'char';
public const ENUM = 'enum';
2021-03-03 13:59:55 -08:00
public const ANY = 'any';
protected Map $map;
public function __construct(protected string $keyType = self::ANY, protected string $valueType = self::ANY)
2021-03-03 13:59:55 -08:00
{
$this->keyType = strtolower(string: $keyType);
$this->valueType = class_exists(class: $valueType) ? $valueType : strtolower(string: $valueType);
2021-03-03 13:59:55 -08:00
if (!$this->isValidKeyType(type: $this->keyType)) {
throw new InvalidArgumentException(message: sprintf('Invalid key type (%s)', $this->keyType));
2021-03-03 13:59:55 -08:00
}
if (!$this->isValidValueType(type: $this->valueType)) {
throw new InvalidArgumentException(message: sprintf('Invalid value type (%s)', $this->valueType));
2021-03-03 13:59:55 -08:00
}
$this->map = new Map();
}
public function getKeyType(): string
{
return $this->keyType;
}
public function getValueType(): string
{
return $this->valueType;
}
/**
* @inheritDoc
*/
public function offsetExists(mixed $offset): bool
2021-03-03 13:59:55 -08:00
{
$this->checkTypes(offset: $offset);
2021-03-03 13:59:55 -08:00
return $this->map->offsetExists(offset: $offset);
2021-03-03 13:59:55 -08:00
}
/**
* @inheritDoc
*/
public function offsetGet(mixed $offset): mixed
2021-03-03 13:59:55 -08:00
{
$this->checkTypes(offset: $offset);
2021-03-03 13:59:55 -08:00
if (!$this->map->offsetExists(offset: $offset)) {
2021-03-03 13:59:55 -08:00
return null;
}
return $this->map->offsetGet(offset: $offset);
2021-03-03 13:59:55 -08:00
}
/**
* @inheritDoc
*/
public function offsetSet(mixed $offset, mixed $value): void
2021-03-03 13:59:55 -08:00
{
$this->checkTypes($offset, $value);
2021-03-03 13:59:55 -08:00
$this->map->offsetSet(offset: $offset, value: $value);
2021-03-03 13:59:55 -08:00
}
/**
* @inheritDoc
*/
public function offsetUnset(mixed $offset): void
2021-03-03 13:59:55 -08:00
{
$this->checkTypes(offset: $offset);
2021-03-03 13:59:55 -08:00
$this->map->offsetUnset(offset: $offset);
2021-03-03 13:59:55 -08:00
}
/**
* Count elements of an object
*
* @link https://php.net/manual/en/countable.count.php
* @return int The custom count as an integer.
* </p>
* <p>
* The return value is cast to an integer.
*/
public function count(): int
{
return $this->map->count();
}
public function toArray(): array
{
return $this->map->toArray();
}
protected function assertInitialValues(array $initialValues): bool
{
foreach ($initialValues as $key => $value) {
if (!$this->checkType(type: $this->keyType, var: $key) || !$this->checkType(type: $this->valueType, var: $value)) {
throw new InvalidArgumentException(message: sprintf('Invalid types for map [%s => %s], they should be [%s => %s]', get_debug_type(value: $key), get_debug_type(value: $value), $this->keyType, $this->valueType));
2021-03-03 13:59:55 -08:00
}
}
return true;
}
private function checkTypes(mixed $offset, mixed $value = null): void
{
if (is_array(value: $offset) || is_object(value: $offset)) {
throw new InvalidArgumentException(message: 'Map keys cannot be objects or arrays');
}
if (!$this->checkType(type: $this->keyType, var: $offset)) {
throw new TypeError(message: sprintf('Key should be of value %s.', $this->keyType));
}
if (null !== $value && !$this->checkType(type: $this->valueType, var: $value)) {
throw new ValueError(message: sprintf('Value should be of type %s.', $this->valueType));
}
}
2021-03-03 13:59:55 -08:00
/**
* @codeCoverageIgnore
* @param string $type
* @param mixed $var
2021-03-03 13:59:55 -08:00
*
* @return bool
*/
private function checkType(string $type, mixed $var): bool
2021-03-03 13:59:55 -08:00
{
return match ($type) {
self::OBJECT => is_object(value: $var),
self::CALLABLE => is_callable(value: $var),
self::ARRAY => is_array(value: $var),
self::INT, self::INTEGER => is_int(value: $var),
self::STRING => is_string(value: $var),
self::CHAR => is_string(value: $var) && (function_exists('mb_strlen') ? mb_strlen(string: $var) === 1 : strlen(string: $var) === 1),
self::ENUM => is_object(value: $var) && enum_exists(enum: $var::class),
self::ANY => true,
default => $var instanceof $type,
};
2021-03-03 13:59:55 -08:00
}
2021-03-03 13:59:55 -08:00
private function isValidKeyType(string $type): bool
{
return in_array(needle: $type, haystack: [
2021-03-03 13:59:55 -08:00
self::INT,
self::INTEGER,
self::STRING,
self::CHAR,
self::ANY,
], strict: true);
2021-03-03 13:59:55 -08:00
}
private function isValidValueType(string $type): bool
{
if (class_exists(class: $type)) {
return true;
}
return in_array(needle: $type, haystack: [
2021-03-03 13:59:55 -08:00
self::OBJECT,
self::CALLABLE,
2021-03-03 13:59:55 -08:00
self::ARRAY,
self::INT,
self::INTEGER,
self::STRING,
self::CHAR,
self::ENUM,
2021-03-03 13:59:55 -08:00
self::ANY,
], strict: true);
2021-03-03 13:59:55 -08:00
}
}