forked from codeigniter4/CodeIgniter4
-
Notifications
You must be signed in to change notification settings - Fork 1
Expand file tree
/
Copy pathEnumCast.php
More file actions
120 lines (96 loc) · 3.23 KB
/
EnumCast.php
File metadata and controls
120 lines (96 loc) · 3.23 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
<?php
declare(strict_types=1);
/**
* This file is part of CodeIgniter 4 framework.
*
* (c) CodeIgniter Foundation <[email protected]>
*
* For the full copyright and license information, please view
* the LICENSE file that was distributed with this source code.
*/
namespace CodeIgniter\DataCaster\Cast;
use BackedEnum;
use CodeIgniter\DataCaster\Exceptions\CastException;
use ReflectionEnum;
use UnitEnum;
/**
* Class EnumCast
*
* Handles casting for PHP enums (both backed and unit enums)
*
* (PHP) [enum --> value/name] --> (DB driver) --> (DB column) int|string
* [ <-- value/name] <-- (DB driver) <-- (DB column) int|string
*/
class EnumCast extends BaseCast implements CastInterface
{
public static function get(
mixed $value,
array $params = [],
?object $helper = null,
): BackedEnum|UnitEnum {
if (! is_string($value) && ! is_int($value)) {
self::invalidTypeValueError($value);
}
$enumClass = $params[0] ?? null;
if ($enumClass === null) {
throw CastException::forMissingEnumClass();
}
if (! enum_exists($enumClass)) {
throw CastException::forNotEnum($enumClass);
}
$reflection = new ReflectionEnum($enumClass);
// Unit enum
if (! $reflection->isBacked()) {
// Unit enum - match by name
foreach ($enumClass::cases() as $case) {
if ($case->name === $value) {
return $case;
}
}
throw CastException::forInvalidEnumCaseName($enumClass, $value);
}
// Backed enum - validate and cast the value to proper type
$backingType = $reflection->getBackingType();
// Cast to proper type (int or string)
if ($backingType->getName() === 'int') {
$value = (int) $value;
} elseif ($backingType->getName() === 'string') {
$value = (string) $value;
}
$enum = $enumClass::tryFrom($value);
if ($enum === null) {
throw CastException::forInvalidEnumValue($enumClass, $value);
}
return $enum;
}
public static function set(
mixed $value,
array $params = [],
?object $helper = null,
): int|string {
if (! is_object($value) || ! enum_exists($value::class)) {
self::invalidTypeValueError($value);
}
// Get the expected enum class
$enumClass = $params[0] ?? null;
if ($enumClass === null) {
throw CastException::forMissingEnumClass();
}
if (! enum_exists($enumClass)) {
throw CastException::forNotEnum($enumClass);
}
// Validate that the enum is of the expected type
if (! $value instanceof $enumClass) {
throw CastException::forInvalidEnumType($enumClass, $value::class);
}
$reflection = new ReflectionEnum($value::class);
// Backed enum - return the properly typed backing value
if ($reflection->isBacked()) {
/** @var BackedEnum $value */
return $value->value;
}
// Unit enum - return the case name
/** @var UnitEnum $value */
return $value->name;
}
}