Skip to content

Commit 31362eb

Browse files
authored
Fix sprintf() inference for constant values with format-width in pattern
1 parent 18196e9 commit 31362eb

File tree

2 files changed

+15
-6
lines changed

2 files changed

+15
-6
lines changed

src/Type/Php/SprintfFunctionDynamicReturnTypeExtension.php

+8-6
Original file line numberDiff line numberDiff line change
@@ -84,13 +84,13 @@ public function getTypeFromFunctionCall(
8484
}
8585

8686
// The printf format is %[argnum$][flags][width][.precision]specifier.
87-
if (preg_match('/^%([0-9]*\$)?[0-9]*\.?[0-9]*([sbdeEfFgGhHouxX])$/', $constantString->getValue(), $matches) === 1) {
88-
if ($matches[1] !== '') {
87+
if (preg_match('/^%(?P<argnum>[0-9]*\$)?(?P<width>[0-9]*)\.?[0-9]*(?P<specifier>[sbdeEfFgGhHouxX])$/', $constantString->getValue(), $matches) === 1) {
88+
if ($matches['argnum'] !== '') {
8989
// invalid positional argument
90-
if ($matches[1] === '0$') {
90+
if ($matches['argnum'] === '0$') {
9191
return null;
9292
}
93-
$checkArg = intval(substr($matches[1], 0, -1));
93+
$checkArg = intval(substr($matches['argnum'], 0, -1));
9494
} else {
9595
$checkArg = 1;
9696
}
@@ -103,11 +103,13 @@ public function getTypeFromFunctionCall(
103103
// if the format string is just a placeholder and specified an argument
104104
// of stringy type, then the return value will be of the same type
105105
$checkArgType = $scope->getType($args[$checkArg]->value);
106-
if ($matches[2] === 's'
106+
if (
107+
$matches['specifier'] === 's'
108+
&& ($checkArgType->isConstantValue()->no() || $matches['width'] === '')
107109
&& ($checkArgType->isString()->yes() || $checkArgType->isInteger()->yes())
108110
) {
109111
$singlePlaceholderEarlyReturn = $checkArgType->toString();
110-
} elseif ($matches[2] !== 's') {
112+
} elseif ($matches['specifier'] !== 's') {
111113
$singlePlaceholderEarlyReturn = new IntersectionType([
112114
new StringType(),
113115
new AccessoryNumericStringType(),

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

+7
Original file line numberDiff line numberDiff line change
@@ -59,6 +59,13 @@ public function positionalArgs($mixed, int $i, float $f, string $s, int $posInt,
5959
assertType('numeric-string', sprintf('%2$14s', $mixed, $intRange));
6060
assertType('non-falsy-string&numeric-string', sprintf('%2$14s', $mixed, $nonZeroIntRange));
6161

62+
assertType("non-falsy-string", sprintf('%2$14s', $mixed, 1));
63+
assertType("non-falsy-string", sprintf('%2$14s', $mixed, '1'));
64+
assertType("non-falsy-string", sprintf('%2$14s', $mixed, 'abc'));
65+
assertType("'1'", sprintf('%2$s', $mixed, 1));
66+
assertType("'1'", sprintf('%2$s', $mixed, '1'));
67+
assertType("'abc'", sprintf('%2$s', $mixed, 'abc'));
68+
6269
assertType('numeric-string', sprintf('%2$.14F', $mixed, $i));
6370
assertType('numeric-string', sprintf('%2$.14F', $mixed, $f));
6471
assertType('numeric-string', sprintf('%2$.14F', $mixed, $s));

0 commit comments

Comments
 (0)