Skip to content

Commit abaeae5

Browse files
committed
Handle FuncCall nodes that represent first class callables
1 parent ccd3d55 commit abaeae5

File tree

3 files changed

+25
-25
lines changed

3 files changed

+25
-25
lines changed

composer.json

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
"thecodingmachine/safe": "^1.0 || ^2.0"
1616
},
1717
"require-dev": {
18+
"nikic/php-parser": "^4",
1819
"phpunit/phpunit": "^9.6",
1920
"php-coveralls/php-coveralls": "^2.1",
2021
"squizlabs/php_codesniffer": "^3.4"

src/Rules/UseSafeFunctionsRule.php

+17-13
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,11 @@
1919
*/
2020
class UseSafeFunctionsRule implements Rule
2121
{
22+
/**
23+
* @see JSON_THROW_ON_ERROR
24+
*/
25+
const JSON_THROW_ON_ERROR = 4194304;
26+
2227
public function getNodeType(): string
2328
{
2429
return Node\Expr\FuncCall::class;
@@ -33,18 +38,18 @@ public function processNode(Node $node, Scope $scope): array
3338
$unsafeFunctions = FunctionListLoader::getFunctionList();
3439

3540
if (isset($unsafeFunctions[$functionName])) {
36-
if (
37-
$functionName === "json_decode"
38-
&& $this->argValueIncludeJSONTHROWONERROR($node->getArgs()[3] ?? null)
39-
) {
40-
return [];
41-
}
41+
if (! $node->isFirstClassCallable()) {
42+
if ($functionName === "json_decode"
43+
&& $this->argValueIncludeJSONTHROWONERROR($node->getArgs()[3] ?? null)
44+
) {
45+
return [];
46+
}
4247

43-
if (
44-
$functionName === "json_encode"
45-
&& $this->argValueIncludeJSONTHROWONERROR($node->getArgs()[1] ?? null)
46-
) {
47-
return [];
48+
if ($functionName === "json_encode"
49+
&& $this->argValueIncludeJSONTHROWONERROR($node->getArgs()[1] ?? null)
50+
) {
51+
return [];
52+
}
4853
}
4954

5055
return [new SafeFunctionRuleError($node->name, $node->getStartLine())];
@@ -77,8 +82,7 @@ private function argValueIncludeJSONTHROWONERROR(?Arg $arg): bool
7782
}
7883

7984
return in_array(true, array_map(function ($element) {
80-
// JSON_THROW_ON_ERROR == 4194304
81-
return ($element & 4194304) == 4194304;
85+
return ($element & self::JSON_THROW_ON_ERROR) == self::JSON_THROW_ON_ERROR;
8286
}, array_filter($options, function ($element) {
8387
return is_int($element);
8488
})), true);

src/Type/Php/ReplaceSafeFunctionsDynamicReturnTypeExtension.php

+7-12
Original file line numberDiff line numberDiff line change
@@ -56,22 +56,17 @@ private function getPreliminarilyResolvedTypeFromFunctionCall(
5656
Scope $scope
5757
): Type {
5858
$argumentPosition = $this->functions[$functionReflection->getName()];
59-
$defaultReturnType = ParametersAcceptorSelector::selectFromArgs(
60-
$scope,
61-
$functionCall->getArgs(),
62-
$functionReflection->getVariants()
63-
)
59+
60+
$args = $functionCall->getArgs();
61+
$variants = $functionReflection->getVariants();
62+
$defaultReturnType = ParametersAcceptorSelector::selectFromArgs($scope, $args, $variants)
6463
->getReturnType();
65-
66-
if (count($functionCall->args) <= $argumentPosition) {
67-
return $defaultReturnType;
68-
}
6964

70-
$subjectArgument = $functionCall->args[$argumentPosition];
71-
if (!$subjectArgument instanceof Arg) {
65+
if (count($args) <= $argumentPosition) {
7266
return $defaultReturnType;
7367
}
74-
68+
69+
$subjectArgument = $args[$argumentPosition];
7570
$subjectArgumentType = $scope->getType($subjectArgument->value);
7671
$mixedType = new MixedType();
7772
if ($subjectArgumentType->isSuperTypeOf($mixedType)->yes()) {

0 commit comments

Comments
 (0)