1: | <?php declare(strict_types = 1); |
2: | |
3: | namespace PHPStan\Type; |
4: | |
5: | use PHPStan\TrinaryLogic; |
6: | use PHPStan\Type\Generic\TemplateTypeMap; |
7: | use function count; |
8: | |
9: | |
10: | class BenevolentUnionType extends UnionType |
11: | { |
12: | |
13: | |
14: | |
15: | |
16: | |
17: | public function __construct(array $types, bool $normalized = false) |
18: | { |
19: | parent::__construct($types, $normalized); |
20: | } |
21: | |
22: | public function filterTypes(callable $filterCb): Type |
23: | { |
24: | $result = parent::filterTypes($filterCb); |
25: | if (!$result instanceof self && $result instanceof UnionType) { |
26: | return TypeUtils::toBenevolentUnion($result); |
27: | } |
28: | |
29: | return $result; |
30: | } |
31: | |
32: | public function describe(VerbosityLevel $level): string |
33: | { |
34: | return '(' . parent::describe($level) . ')'; |
35: | } |
36: | |
37: | protected function unionTypes(callable $getType): Type |
38: | { |
39: | $resultTypes = []; |
40: | foreach ($this->getTypes() as $type) { |
41: | $result = $getType($type); |
42: | if ($result instanceof ErrorType) { |
43: | continue; |
44: | } |
45: | |
46: | $resultTypes[] = $result; |
47: | } |
48: | |
49: | if (count($resultTypes) === 0) { |
50: | return new ErrorType(); |
51: | } |
52: | |
53: | return TypeUtils::toBenevolentUnion(TypeCombinator::union(...$resultTypes)); |
54: | } |
55: | |
56: | protected function pickFromTypes( |
57: | callable $getValues, |
58: | callable $criteria, |
59: | ): array |
60: | { |
61: | $values = []; |
62: | foreach ($this->getTypes() as $type) { |
63: | $innerValues = $getValues($type); |
64: | if ($innerValues === [] && $criteria($type)) { |
65: | return []; |
66: | } |
67: | |
68: | foreach ($innerValues as $innerType) { |
69: | $values[] = $innerType; |
70: | } |
71: | } |
72: | |
73: | return $values; |
74: | } |
75: | |
76: | public function getOffsetValueType(Type $offsetType): Type |
77: | { |
78: | $types = []; |
79: | foreach ($this->getTypes() as $innerType) { |
80: | $valueType = $innerType->getOffsetValueType($offsetType); |
81: | if ($valueType instanceof ErrorType) { |
82: | continue; |
83: | } |
84: | |
85: | $types[] = $valueType; |
86: | } |
87: | |
88: | if (count($types) === 0) { |
89: | return new ErrorType(); |
90: | } |
91: | |
92: | return TypeUtils::toBenevolentUnion(TypeCombinator::union(...$types)); |
93: | } |
94: | |
95: | protected function unionResults(callable $getResult): TrinaryLogic |
96: | { |
97: | return TrinaryLogic::createNo()->lazyOr($this->getTypes(), $getResult); |
98: | } |
99: | |
100: | public function isAcceptedBy(Type $acceptingType, bool $strictTypes): AcceptsResult |
101: | { |
102: | $result = AcceptsResult::createNo(); |
103: | foreach ($this->getTypes() as $innerType) { |
104: | $result = $result->or($acceptingType->accepts($innerType, $strictTypes)); |
105: | } |
106: | |
107: | return $result; |
108: | } |
109: | |
110: | public function inferTemplateTypes(Type $receivedType): TemplateTypeMap |
111: | { |
112: | $types = TemplateTypeMap::createEmpty(); |
113: | |
114: | foreach ($this->getTypes() as $type) { |
115: | $types = $types->benevolentUnion($type->inferTemplateTypes($receivedType)); |
116: | } |
117: | |
118: | return $types; |
119: | } |
120: | |
121: | public function inferTemplateTypesOn(Type $templateType): TemplateTypeMap |
122: | { |
123: | $types = TemplateTypeMap::createEmpty(); |
124: | |
125: | foreach ($this->getTypes() as $type) { |
126: | $types = $types->benevolentUnion($templateType->inferTemplateTypes($type)); |
127: | } |
128: | |
129: | return $types; |
130: | } |
131: | |
132: | public function traverse(callable $cb): Type |
133: | { |
134: | $types = []; |
135: | $changed = false; |
136: | |
137: | foreach ($this->getTypes() as $type) { |
138: | $newType = $cb($type); |
139: | if ($type !== $newType) { |
140: | $changed = true; |
141: | } |
142: | $types[] = $newType; |
143: | } |
144: | |
145: | if ($changed) { |
146: | return TypeUtils::toBenevolentUnion(TypeCombinator::union(...$types)); |
147: | } |
148: | |
149: | return $this; |
150: | } |
151: | |
152: | public function traverseSimultaneously(Type $right, callable $cb): Type |
153: | { |
154: | $types = []; |
155: | $changed = false; |
156: | |
157: | if (!$right instanceof UnionType) { |
158: | return $this; |
159: | } |
160: | |
161: | if (count($this->getTypes()) !== count($right->getTypes())) { |
162: | return $this; |
163: | } |
164: | |
165: | foreach ($this->getSortedTypes() as $i => $leftType) { |
166: | $rightType = $right->getSortedTypes()[$i]; |
167: | $newType = $cb($leftType, $rightType); |
168: | if ($leftType !== $newType) { |
169: | $changed = true; |
170: | } |
171: | $types[] = $newType; |
172: | } |
173: | |
174: | if ($changed) { |
175: | return TypeUtils::toBenevolentUnion(TypeCombinator::union(...$types)); |
176: | } |
177: | |
178: | return $this; |
179: | } |
180: | |
181: | } |
182: | |