mirror of
https://github.com/Respect/Validation.git
synced 2026-03-14 22:35:45 +01:00
With PHP's promoted properties, there is little reason to have an abstract class that only provides a constructor and a proxy method. The greater part of the classes that extended Wrapper had their own implementation of evaluate(), hence it made very little sense to keep Wrapper in the codebase. Assisted-by: Claude Code (Opus 4.5)
102 lines
3.3 KiB
PHP
102 lines
3.3 KiB
PHP
<?php
|
|
|
|
/*
|
|
* SPDX-License-Identifier: MIT
|
|
* SPDX-FileCopyrightText: (c) Respect Project Contributors
|
|
* SPDX-FileContributor: Alexandre Gomes Gaigalas <alganet@gmail.com>
|
|
* SPDX-FileContributor: Danilo Correa <danilosilva87@gmail.com>
|
|
* SPDX-FileContributor: Henrique Moody <henriquemoody@gmail.com>
|
|
* SPDX-FileContributor: v0idpwn <v0idpwn@gmail.com>
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Respect\Validation\Validators;
|
|
|
|
use Attribute;
|
|
use Psr\Http\Message\StreamInterface;
|
|
use Psr\Http\Message\UploadedFileInterface;
|
|
use Respect\Validation\Exceptions\InvalidValidatorException;
|
|
use Respect\Validation\Message\Template;
|
|
use Respect\Validation\Result;
|
|
use Respect\Validation\Validator;
|
|
use SplFileInfo;
|
|
|
|
use function filesize;
|
|
use function is_string;
|
|
|
|
#[Attribute(Attribute::TARGET_PROPERTY | Attribute::IS_REPEATABLE)]
|
|
#[Template(
|
|
'The size in {{unit|trans}} of',
|
|
'The size in {{unit|trans}} of',
|
|
Size::TEMPLATE_STANDARD,
|
|
)]
|
|
#[Template(
|
|
'{{subject}} must be a filename or an instance of SplFileInfo or a PSR-7 interface',
|
|
'{{subject}} must not be a filename or an instance of SplFileInfo or a PSR-7 interface',
|
|
self::TEMPLATE_WRONG_TYPE,
|
|
)]
|
|
final readonly class Size implements Validator
|
|
{
|
|
public const string TEMPLATE_WRONG_TYPE = '__wrong_type__';
|
|
|
|
private const array DATA_STORAGE_UNITS = [
|
|
'B' => ['name' => 'bytes', 'bytes' => 1],
|
|
'KB' => ['name' => 'kilobytes', 'bytes' => 1024],
|
|
'MB' => ['name' => 'megabytes', 'bytes' => 1024 ** 2],
|
|
'GB' => ['name' => 'gigabytes', 'bytes' => 1024 ** 3],
|
|
'TB' => ['name' => 'terabytes', 'bytes' => 1024 ** 4],
|
|
'PB' => ['name' => 'petabytes', 'bytes' => 1024 ** 5],
|
|
'EB' => ['name' => 'exabytes', 'bytes' => 1024 ** 6],
|
|
'ZB' => ['name' => 'zettabytes', 'bytes' => 1024 ** 7],
|
|
'YB' => ['name' => 'yottabytes', 'bytes' => 1024 ** 8],
|
|
];
|
|
|
|
/** @param "B"|"KB"|"MB"|"GB"|"TB"|"PB"|"EB"|"ZB"|"YB" $unit */
|
|
public function __construct(
|
|
private string $unit,
|
|
private Validator $validator,
|
|
) {
|
|
if (!isset(self::DATA_STORAGE_UNITS[$unit])) {
|
|
throw new InvalidValidatorException('"%s" is not a recognized data storage unit.', $unit);
|
|
}
|
|
}
|
|
|
|
public function evaluate(mixed $input): Result
|
|
{
|
|
$size = $this->getSize($input);
|
|
if ($size === null) {
|
|
return Result::failed($input, $this, [], self::TEMPLATE_WRONG_TYPE)
|
|
->withId($this->validator->evaluate($input)->id->withPrefix('size'));
|
|
}
|
|
|
|
$result = $this->validator->evaluate($this->getSize($input) / self::DATA_STORAGE_UNITS[$this->unit]['bytes']);
|
|
$parameters = ['unit' => self::DATA_STORAGE_UNITS[$this->unit]['name']];
|
|
|
|
return $result->asAdjacentOf(
|
|
Result::of($result->hasPassed, $input, $this, $parameters),
|
|
'size',
|
|
);
|
|
}
|
|
|
|
private function getSize(mixed $input): int|null
|
|
{
|
|
if (is_string($input)) {
|
|
return (int) filesize($input);
|
|
}
|
|
|
|
if ($input instanceof SplFileInfo) {
|
|
return $input->getSize();
|
|
}
|
|
|
|
if ($input instanceof UploadedFileInterface) {
|
|
return $input->getSize();
|
|
}
|
|
|
|
if ($input instanceof StreamInterface) {
|
|
return $input->getSize();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|