Skip to content

Commit 0305a41

Browse files
Fix mb_convert_encoding signature
1 parent 8a6f7e9 commit 0305a41

File tree

4 files changed

+89
-13
lines changed

4 files changed

+89
-13
lines changed

resources/functionMap.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -6312,7 +6312,7 @@
63126312
'mb_check_encoding' => ['bool', 'var='=>'string|array<string>', 'encoding='=>'string'],
63136313
'mb_chr' => ['string|false', 'cp'=>'int', 'encoding='=>'string'],
63146314
'mb_convert_case' => ['string', 'sourcestring'=>'string', 'mode'=>'int', 'encoding='=>'string'],
6315-
'mb_convert_encoding' => ['string|array<int, string>|false', 'val'=>'string|array<int, string>', 'to_encoding'=>'string', 'from_encoding='=>'mixed'],
6315+
'mb_convert_encoding' => ['string|array<mixed>|false', 'val'=>'string|array<mixed>', 'to_encoding'=>'string', 'from_encoding='=>'mixed'],
63166316
'mb_convert_kana' => ['string', 'str'=>'string', 'option='=>'string', 'encoding='=>'string'],
63176317
'mb_convert_variables' => ['string|false', 'to_encoding'=>'string', 'from_encoding'=>'array|string', '&rw_vars'=>'string|array|object', '&...rw_vars='=>'string|array|object'],
63186318
'mb_decode_mimeheader' => ['string', 'string'=>'string'],

src/Type/Php/MbConvertEncodingFunctionReturnTypeExtension.php

+52-8
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,17 @@
55
use PhpParser\Node\Expr\FuncCall;
66
use PHPStan\Analyser\Scope;
77
use PHPStan\Reflection\FunctionReflection;
8+
use PHPStan\Reflection\ParametersAcceptorSelector;
9+
use PHPStan\Type\Accessory\AccessoryArrayListType;
10+
use PHPStan\Type\Accessory\NonEmptyArrayType;
811
use PHPStan\Type\ArrayType;
12+
use PHPStan\Type\Constant\ConstantBooleanType;
913
use PHPStan\Type\DynamicFunctionReturnTypeExtension;
10-
use PHPStan\Type\IntegerType;
14+
use PHPStan\Type\NeverType;
1115
use PHPStan\Type\StringType;
1216
use PHPStan\Type\Type;
17+
use PHPStan\Type\TypeCombinator;
18+
use PHPStan\Type\UnionType;
1319

1420
final class MbConvertEncodingFunctionReturnTypeExtension implements DynamicFunctionReturnTypeExtension
1521
{
@@ -30,16 +36,54 @@ public function getTypeFromFunctionCall(
3036
}
3137

3238
$argType = $scope->getType($functionCall->getArgs()[0]->value);
33-
$isString = $argType->isString();
34-
$isArray = $argType->isArray();
35-
$compare = $isString->compareTo($isArray);
36-
if ($compare === $isString) {
39+
40+
$initialReturnType = ParametersAcceptorSelector::selectFromArgs(
41+
$scope,
42+
$functionCall->getArgs(),
43+
$functionReflection->getVariants(),
44+
)->getReturnType();
45+
46+
$result = TypeCombinator::intersect($initialReturnType, $this->generalizeStringType($argType));
47+
if ($result instanceof NeverType) {
48+
return null;
49+
}
50+
51+
return TypeCombinator::union($result, new ConstantBooleanType(false));
52+
}
53+
54+
public function generalizeStringType(Type $type): Type
55+
{
56+
if ($type instanceof UnionType) {
57+
return $type->traverse([$this, 'generalizeStringType']);
58+
}
59+
60+
if ($type->isString()->yes()) {
3761
return new StringType();
38-
} elseif ($compare === $isArray) {
39-
return new ArrayType(new IntegerType(), new StringType());
4062
}
4163

42-
return null;
64+
$constantArrays = $type->getConstantArrays();
65+
if (count($constantArrays) > 0) {
66+
$types = [];
67+
foreach ($constantArrays as $constantArray) {
68+
$types[] = $constantArray->traverse([$this, 'generalizeStringType']);
69+
}
70+
71+
return TypeCombinator::union(...$types);
72+
}
73+
74+
if ($type->isArray()->yes()) {
75+
$newArrayType = new ArrayType($type->getIterableKeyType(), $this->generalizeStringType($type->getIterableValueType()));
76+
if ($type->isIterableAtLeastOnce()->yes()) {
77+
$newArrayType = TypeCombinator::intersect($newArrayType, new NonEmptyArrayType());
78+
}
79+
if ($type->isList()->yes()) {
80+
$newArrayType = TypeCombinator::intersect($newArrayType, new AccessoryArrayListType());
81+
}
82+
83+
return $newArrayType;
84+
}
85+
86+
return $type;
4387
}
4488

4589
}

tests/PHPStan/Analyser/nsrt/bug-3336.php

+4-4
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,8 @@
33
namespace Bug3336;
44

55
function (array $arr, string $str, $mixed): void {
6-
\PHPStan\Testing\assertType('array<int, string>', mb_convert_encoding($arr));
7-
\PHPStan\Testing\assertType('string', mb_convert_encoding($str));
8-
\PHPStan\Testing\assertType('array<int, string>|string|false', mb_convert_encoding($mixed));
9-
\PHPStan\Testing\assertType('array<int, string>|string|false', mb_convert_encoding());
6+
\PHPStan\Testing\assertType('array<mixed>|false', mb_convert_encoding($arr));
7+
\PHPStan\Testing\assertType('string|false', mb_convert_encoding($str));
8+
\PHPStan\Testing\assertType('array<mixed>|string|false', mb_convert_encoding($mixed));
9+
\PHPStan\Testing\assertType('array<mixed>|string|false', mb_convert_encoding());
1010
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
<?php
2+
3+
namespace MbConvertEncoding;
4+
5+
/**
6+
* @param 'foo'|'bar' $constantString
7+
* @param array{foo: string, bar: int, baz: 'foo'} $structuredArray
8+
* @param list<string> $stringList
9+
* @param list<int> $intList
10+
* @param 'foo'|'bar'|array{foo: string, bar: int, baz: 'foo'}|bool $union
11+
*/
12+
function test_mb_convert_encoding(
13+
mixed $mixed,
14+
string $constantString,
15+
string $string,
16+
array $mixedArray,
17+
array $structuredArray,
18+
array $stringList,
19+
array $intList,
20+
string|array|bool $union,
21+
int $int,
22+
): void {
23+
\PHPStan\Testing\assertType('array|string|false', mb_convert_encoding($mixed, 'UTF-8'));
24+
\PHPStan\Testing\assertType('string|false', mb_convert_encoding($constantString, 'UTF-8'));
25+
\PHPStan\Testing\assertType('string|false', mb_convert_encoding($string, 'UTF-8'));
26+
\PHPStan\Testing\assertType('array|false', mb_convert_encoding($mixedArray, 'UTF-8'));
27+
\PHPStan\Testing\assertType('array{foo: string, bar: int, baz: string}|false', mb_convert_encoding($structuredArray, 'UTF-8'));
28+
\PHPStan\Testing\assertType('list<string>|false', mb_convert_encoding($stringList, 'UTF-8'));
29+
\PHPStan\Testing\assertType('list<int>|false', mb_convert_encoding($intList, 'UTF-8'));
30+
\PHPStan\Testing\assertType('array{foo: string, bar: int, baz: string}|string|false', mb_convert_encoding($union, 'UTF-8'));
31+
\PHPStan\Testing\assertType('array|string|false', mb_convert_encoding($int, 'UTF-8'));
32+
};

0 commit comments

Comments
 (0)