mirror of
https://github.com/danog/psalm.git
synced 2025-01-10 15:09:04 +01:00
1515 lines
56 KiB
PHP
1515 lines
56 KiB
PHP
<?php
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression\Call;
|
|
|
|
use PhpParser;
|
|
use Psalm\CodeLocation;
|
|
use Psalm\Codebase;
|
|
use Psalm\Context;
|
|
use Psalm\Internal\Analyzer\Statements\Expression\AssignmentAnalyzer;
|
|
use Psalm\Internal\Analyzer\Statements\Expression\CallAnalyzer;
|
|
use Psalm\Internal\Analyzer\Statements\Expression\ExpressionIdentifier;
|
|
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\ArrayFetchAnalyzer;
|
|
use Psalm\Internal\Analyzer\Statements\ExpressionAnalyzer;
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
|
use Psalm\Internal\Codebase\Functions;
|
|
use Psalm\Internal\Codebase\InternalCallMapHandler;
|
|
use Psalm\Internal\Codebase\TaintFlowGraph;
|
|
use Psalm\Internal\DataFlow\TaintSink;
|
|
use Psalm\Internal\MethodIdentifier;
|
|
use Psalm\Internal\Stubs\Generator\StubsGenerator;
|
|
use Psalm\Internal\Type\Comparator\UnionTypeComparator;
|
|
use Psalm\Internal\Type\TemplateInferredTypeReplacer;
|
|
use Psalm\Internal\Type\TemplateResult;
|
|
use Psalm\Internal\Type\TemplateStandinTypeReplacer;
|
|
use Psalm\Issue\InvalidNamedArgument;
|
|
use Psalm\Issue\InvalidPassByReference;
|
|
use Psalm\Issue\PossiblyUndefinedVariable;
|
|
use Psalm\Issue\TooFewArguments;
|
|
use Psalm\Issue\TooManyArguments;
|
|
use Psalm\IssueBuffer;
|
|
use Psalm\Node\VirtualArg;
|
|
use Psalm\Storage\ClassLikeStorage;
|
|
use Psalm\Storage\FunctionLikeParameter;
|
|
use Psalm\Storage\FunctionLikeStorage;
|
|
use Psalm\Type;
|
|
use Psalm\Type\Atomic\TArray;
|
|
use Psalm\Type\Atomic\TKeyedArray;
|
|
use Psalm\Type\Atomic\TList;
|
|
|
|
use function array_map;
|
|
use function array_reverse;
|
|
use function count;
|
|
use function in_array;
|
|
use function is_string;
|
|
use function reset;
|
|
use function strpos;
|
|
use function strtolower;
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
class ArgumentsAnalyzer
|
|
{
|
|
/**
|
|
* @param list<PhpParser\Node\Arg> $args
|
|
* @param array<int, FunctionLikeParameter>|null $function_params
|
|
*
|
|
* @return false|null
|
|
*/
|
|
public static function analyze(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
array $args,
|
|
?array $function_params,
|
|
?string $method_id,
|
|
bool $allow_named_args,
|
|
Context $context,
|
|
?TemplateResult $template_result = null
|
|
): ?bool {
|
|
$last_param = $function_params
|
|
? $function_params[count($function_params) - 1]
|
|
: null;
|
|
|
|
// if this modifies the array type based on further args
|
|
if (in_array($method_id, ['array_push', 'array_unshift'], true)
|
|
&& $function_params
|
|
&& isset($args[0])
|
|
&& isset($args[1])
|
|
) {
|
|
if (ArrayFunctionArgumentsAnalyzer::handleAddition(
|
|
$statements_analyzer,
|
|
$args,
|
|
$context,
|
|
$method_id
|
|
) === false
|
|
) {
|
|
return false;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
if ($method_id === 'array_splice' && $function_params && count($args) > 1) {
|
|
if (ArrayFunctionArgumentsAnalyzer::handleSplice($statements_analyzer, $args, $context) === false) {
|
|
return false;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
if ($method_id === 'array_map') {
|
|
$args = array_reverse($args, true);
|
|
}
|
|
|
|
foreach ($args as $argument_offset => $arg) {
|
|
if ($function_params === null) {
|
|
if (self::evaluateArbitraryParam(
|
|
$statements_analyzer,
|
|
$arg,
|
|
$context
|
|
) === false) {
|
|
return false;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
$param = null;
|
|
|
|
if ($arg->name && $allow_named_args) {
|
|
foreach ($function_params as $candidate_param) {
|
|
if ($candidate_param->name === $arg->name->name) {
|
|
$param = $candidate_param;
|
|
break;
|
|
}
|
|
}
|
|
} elseif ($argument_offset < count($function_params)) {
|
|
$param = $function_params[$argument_offset];
|
|
} elseif ($last_param && $last_param->is_variadic) {
|
|
$param = $last_param;
|
|
}
|
|
|
|
$by_ref = $param && $param->by_ref;
|
|
|
|
$by_ref_type = null;
|
|
|
|
if ($by_ref) {
|
|
$by_ref_type = $param->type ? clone $param->type : Type::getMixed();
|
|
}
|
|
|
|
if ($by_ref
|
|
&& $by_ref_type
|
|
&& !($arg->value instanceof PhpParser\Node\Expr\Closure
|
|
|| $arg->value instanceof PhpParser\Node\Expr\ConstFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\ClassConstFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\FuncCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\MethodCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\StaticCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\New_
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Assign
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Array_
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Ternary
|
|
|| $arg->value instanceof PhpParser\Node\Expr\BinaryOp
|
|
)
|
|
) {
|
|
if (self::handleByRefFunctionArg(
|
|
$statements_analyzer,
|
|
$method_id,
|
|
$argument_offset,
|
|
$arg,
|
|
$context
|
|
) === false) {
|
|
return false;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
$toggled_class_exists = false;
|
|
|
|
if ($method_id === 'class_exists'
|
|
&& $argument_offset === 0
|
|
&& !$context->inside_class_exists
|
|
) {
|
|
$context->inside_class_exists = true;
|
|
$toggled_class_exists = true;
|
|
}
|
|
|
|
if (($arg->value instanceof PhpParser\Node\Expr\Closure
|
|
|| $arg->value instanceof PhpParser\Node\Expr\ArrowFunction)
|
|
&& $template_result
|
|
&& $template_result->lower_bounds
|
|
&& $param
|
|
&& !$arg->value->getDocComment()
|
|
) {
|
|
self::handleClosureArg(
|
|
$statements_analyzer,
|
|
$args,
|
|
$method_id,
|
|
$context,
|
|
$template_result,
|
|
$argument_offset,
|
|
$arg,
|
|
$param
|
|
);
|
|
}
|
|
|
|
$was_inside_call = $context->inside_call;
|
|
|
|
$context->inside_call = true;
|
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $arg->value, $context) === false) {
|
|
return false;
|
|
}
|
|
|
|
if (!$was_inside_call) {
|
|
$context->inside_call = false;
|
|
}
|
|
|
|
if (($argument_offset === 0 && $method_id === 'array_filter' && count($args) === 2)
|
|
|| ($argument_offset > 0 && $method_id === 'array_map' && count($args) >= 2)
|
|
) {
|
|
self::handleArrayMapFilterArrayArg(
|
|
$statements_analyzer,
|
|
$method_id,
|
|
$argument_offset,
|
|
$arg,
|
|
$context,
|
|
$template_result
|
|
);
|
|
}
|
|
|
|
if ($toggled_class_exists) {
|
|
$context->inside_class_exists = false;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private static function handleArrayMapFilterArrayArg(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
string $method_id,
|
|
int $argument_offset,
|
|
PhpParser\Node\Arg $arg,
|
|
Context $context,
|
|
?TemplateResult &$template_result
|
|
) : void {
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
$generic_param_type = new Type\Union([
|
|
new Type\Atomic\TArray([
|
|
Type::getArrayKey(),
|
|
new Type\Union([
|
|
new Type\Atomic\TTemplateParam(
|
|
'ArrayValue' . $argument_offset,
|
|
Type::getMixed(),
|
|
$method_id
|
|
)
|
|
])
|
|
])
|
|
]);
|
|
|
|
$template_types = ['ArrayValue' . $argument_offset => [$method_id => Type::getMixed()]];
|
|
|
|
$replace_template_result = new \Psalm\Internal\Type\TemplateResult(
|
|
$template_types,
|
|
[]
|
|
);
|
|
|
|
$existing_type = $statements_analyzer->node_data->getType($arg->value);
|
|
|
|
\Psalm\Internal\Type\TemplateStandinTypeReplacer::replace(
|
|
$generic_param_type,
|
|
$replace_template_result,
|
|
$codebase,
|
|
$statements_analyzer,
|
|
$existing_type,
|
|
$argument_offset,
|
|
$context->self,
|
|
$context->calling_method_id ?: $context->calling_function_id
|
|
);
|
|
|
|
if ($replace_template_result->lower_bounds) {
|
|
if (!$template_result) {
|
|
$template_result = new TemplateResult([], []);
|
|
}
|
|
|
|
$template_result->lower_bounds += $replace_template_result->lower_bounds;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param array<int, PhpParser\Node\Arg> $args
|
|
*/
|
|
private static function handleClosureArg(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
array $args,
|
|
?string $method_id,
|
|
Context $context,
|
|
TemplateResult $template_result,
|
|
int $argument_offset,
|
|
PhpParser\Node\Arg $arg,
|
|
FunctionLikeParameter $param
|
|
) : void {
|
|
if (!$param->type) {
|
|
return;
|
|
}
|
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
if (($argument_offset === 1 && $method_id === 'array_filter' && count($args) === 2)
|
|
|| ($argument_offset === 0 && $method_id === 'array_map' && count($args) >= 2)
|
|
) {
|
|
$function_like_params = [];
|
|
|
|
foreach ($template_result->lower_bounds as $template_name => $_) {
|
|
$function_like_params[] = new \Psalm\Storage\FunctionLikeParameter(
|
|
'function',
|
|
false,
|
|
new Type\Union([
|
|
new Type\Atomic\TTemplateParam(
|
|
$template_name,
|
|
Type::getMixed(),
|
|
$method_id
|
|
)
|
|
])
|
|
);
|
|
}
|
|
|
|
$replaced_type = new Type\Union([
|
|
new Type\Atomic\TCallable(
|
|
'callable',
|
|
array_reverse($function_like_params)
|
|
)
|
|
]);
|
|
} else {
|
|
$replaced_type = clone $param->type;
|
|
}
|
|
|
|
$replace_template_result = new \Psalm\Internal\Type\TemplateResult(
|
|
array_map(
|
|
function ($template_map) use ($codebase) {
|
|
return array_map(
|
|
function ($lower_bounds) use ($codebase) {
|
|
return \Psalm\Internal\Type\TemplateStandinTypeReplacer::getMostSpecificTypeFromBounds(
|
|
$lower_bounds,
|
|
$codebase
|
|
);
|
|
},
|
|
$template_map
|
|
);
|
|
},
|
|
$template_result->lower_bounds
|
|
),
|
|
[]
|
|
);
|
|
|
|
$replaced_type = \Psalm\Internal\Type\TemplateStandinTypeReplacer::replace(
|
|
$replaced_type,
|
|
$replace_template_result,
|
|
$codebase,
|
|
$statements_analyzer,
|
|
null,
|
|
null,
|
|
null,
|
|
$context->calling_method_id ?: $context->calling_function_id
|
|
);
|
|
|
|
TemplateInferredTypeReplacer::replace(
|
|
$replaced_type,
|
|
$replace_template_result,
|
|
$codebase
|
|
);
|
|
|
|
$closure_id = strtolower($statements_analyzer->getFilePath())
|
|
. ':' . $arg->value->getLine()
|
|
. ':' . (int)$arg->value->getAttribute('startFilePos')
|
|
. ':-:closure';
|
|
|
|
try {
|
|
$closure_storage = $codebase->getClosureStorage(
|
|
$statements_analyzer->getFilePath(),
|
|
$closure_id
|
|
);
|
|
} catch (\UnexpectedValueException $e) {
|
|
return;
|
|
}
|
|
|
|
foreach ($closure_storage->params as $closure_param_offset => $param_storage) {
|
|
$param_type_inferred = $param_storage->type_inferred;
|
|
|
|
$newly_inferred_type = null;
|
|
$has_different_docblock_type = false;
|
|
|
|
if ($param_storage->type && !$param_type_inferred) {
|
|
if ($param_storage->type !== $param_storage->signature_type) {
|
|
$has_different_docblock_type = true;
|
|
}
|
|
}
|
|
|
|
if (!$has_different_docblock_type) {
|
|
foreach ($replaced_type->getAtomicTypes() as $replaced_type_part) {
|
|
if ($replaced_type_part instanceof Type\Atomic\TCallable
|
|
|| $replaced_type_part instanceof Type\Atomic\TClosure
|
|
) {
|
|
if (isset($replaced_type_part->params[$closure_param_offset]->type)
|
|
&& !$replaced_type_part->params[$closure_param_offset]->type->hasTemplate()
|
|
) {
|
|
if ($param_storage->type && !$param_type_inferred) {
|
|
$type_match_found = UnionTypeComparator::isContainedBy(
|
|
$codebase,
|
|
$replaced_type_part->params[$closure_param_offset]->type,
|
|
$param_storage->type
|
|
);
|
|
|
|
if (!$type_match_found) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
$newly_inferred_type = Type::combineUnionTypes(
|
|
$newly_inferred_type,
|
|
$replaced_type_part->params[$closure_param_offset]->type,
|
|
$codebase
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($newly_inferred_type) {
|
|
$param_storage->type = $newly_inferred_type;
|
|
$param_storage->type_inferred = true;
|
|
}
|
|
|
|
if ($param_storage->type && ($method_id === 'array_map' || $method_id === 'array_filter')) {
|
|
ArrayFetchAnalyzer::taintArrayFetch(
|
|
$statements_analyzer,
|
|
$args[1 - $argument_offset]->value,
|
|
null,
|
|
$param_storage->type,
|
|
Type::getMixed()
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param list<PhpParser\Node\Arg> $args
|
|
* @param string|MethodIdentifier|null $method_id
|
|
* @param array<int,FunctionLikeParameter> $function_params
|
|
*
|
|
* @return false|null
|
|
*
|
|
* @psalm-suppress ComplexMethod there's just not much that can be done about this
|
|
*/
|
|
public static function checkArgumentsMatch(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
array $args,
|
|
$method_id,
|
|
array $function_params,
|
|
?FunctionLikeStorage $function_storage,
|
|
?ClassLikeStorage $class_storage,
|
|
?TemplateResult $class_template_result,
|
|
CodeLocation $code_location,
|
|
Context $context
|
|
): ?bool {
|
|
$in_call_map = $method_id ? InternalCallMapHandler::inCallMap((string) $method_id) : false;
|
|
|
|
$cased_method_id = (string) $method_id;
|
|
|
|
$is_variadic = false;
|
|
|
|
$fq_class_name = null;
|
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
if ($method_id) {
|
|
if (!$in_call_map && $method_id instanceof MethodIdentifier) {
|
|
$fq_class_name = $method_id->fq_class_name;
|
|
}
|
|
|
|
if ($function_storage) {
|
|
$is_variadic = $function_storage->variadic;
|
|
} elseif (is_string($method_id)) {
|
|
$is_variadic = Functions::isVariadic(
|
|
$codebase,
|
|
strtolower($method_id),
|
|
$statements_analyzer->getRootFilePath()
|
|
);
|
|
} else {
|
|
$is_variadic = $codebase->methods->isVariadic($method_id);
|
|
}
|
|
}
|
|
|
|
if ($method_id instanceof MethodIdentifier) {
|
|
$cased_method_id = $codebase->methods->getCasedMethodId($method_id);
|
|
} elseif ($function_storage) {
|
|
$cased_method_id = $function_storage->cased_name;
|
|
}
|
|
|
|
$calling_class_storage = $class_storage;
|
|
|
|
$static_fq_class_name = $fq_class_name;
|
|
$self_fq_class_name = $fq_class_name;
|
|
|
|
if ($method_id instanceof MethodIdentifier) {
|
|
$declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id);
|
|
|
|
if ($declaring_method_id && (string)$declaring_method_id !== (string)$method_id) {
|
|
$self_fq_class_name = $declaring_method_id->fq_class_name;
|
|
$class_storage = $codebase->classlike_storage_provider->get($self_fq_class_name);
|
|
}
|
|
|
|
$appearing_method_id = $codebase->methods->getAppearingMethodId($method_id);
|
|
|
|
if ($appearing_method_id && $declaring_method_id !== $appearing_method_id) {
|
|
$self_fq_class_name = $appearing_method_id->fq_class_name;
|
|
}
|
|
}
|
|
|
|
if ($function_params) {
|
|
foreach ($function_params as $function_param) {
|
|
$is_variadic = $is_variadic || $function_param->is_variadic;
|
|
}
|
|
}
|
|
|
|
$has_packed_var = false;
|
|
|
|
$packed_var_definite_args = 0;
|
|
|
|
foreach ($args as $arg) {
|
|
if ($arg->unpack) {
|
|
$arg_value_type = $statements_analyzer->node_data->getType($arg->value);
|
|
|
|
if (!$arg_value_type
|
|
|| !$arg_value_type->isSingle()
|
|
|| !$arg_value_type->hasArray()
|
|
) {
|
|
$has_packed_var = true;
|
|
break;
|
|
}
|
|
|
|
foreach ($arg_value_type->getAtomicTypes() as $atomic_arg_type) {
|
|
if (!$atomic_arg_type instanceof TKeyedArray) {
|
|
$has_packed_var = true;
|
|
break 2;
|
|
}
|
|
|
|
$packed_var_definite_args = 0;
|
|
|
|
foreach ($atomic_arg_type->properties as $property_type) {
|
|
if ($property_type->possibly_undefined) {
|
|
$has_packed_var = true;
|
|
} else {
|
|
$packed_var_definite_args++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!$has_packed_var) {
|
|
$packed_var_definite_args = \max(0, $packed_var_definite_args - 1);
|
|
}
|
|
|
|
$last_param = $function_params
|
|
? $function_params[count($function_params) - 1]
|
|
: null;
|
|
|
|
$template_result = null;
|
|
|
|
$class_generic_params = [];
|
|
|
|
if ($class_template_result) {
|
|
foreach ($class_template_result->lower_bounds as $template_name => $type_map) {
|
|
foreach ($type_map as $class => $lower_bounds) {
|
|
if (count($lower_bounds) === 1) {
|
|
$class_generic_params[$template_name][$class] = clone reset($lower_bounds)->type;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($function_storage) {
|
|
$template_result = self::getProvisionalTemplateResultForFunctionLike(
|
|
$statements_analyzer,
|
|
$codebase,
|
|
$context,
|
|
$class_storage,
|
|
$self_fq_class_name,
|
|
$calling_class_storage,
|
|
$function_storage,
|
|
$class_generic_params,
|
|
$class_template_result,
|
|
$args,
|
|
$function_params,
|
|
$last_param
|
|
);
|
|
}
|
|
|
|
$function_param_count = count($function_params);
|
|
|
|
if (count($function_params) > count($args) && !$has_packed_var) {
|
|
for ($i = count($args), $iMax = count($function_params); $i < $iMax; $i++) {
|
|
if ($function_params[$i]->default_type
|
|
&& $function_params[$i]->type
|
|
&& $function_params[$i]->type->hasTemplate()
|
|
) {
|
|
if ($function_params[$i]->default_type instanceof Type\Union) {
|
|
$default_type = $function_params[$i]->default_type;
|
|
} else {
|
|
$default_type_atomic = \Psalm\Internal\Codebase\ConstantTypeResolver::resolve(
|
|
$codebase->classlikes,
|
|
$function_params[$i]->default_type,
|
|
$statements_analyzer
|
|
);
|
|
|
|
$default_type = new Type\Union([$default_type_atomic]);
|
|
}
|
|
|
|
if ($default_type->hasLiteralValue()) {
|
|
ArgumentAnalyzer::checkArgumentMatches(
|
|
$statements_analyzer,
|
|
$cased_method_id,
|
|
$method_id instanceof MethodIdentifier ? $method_id : null,
|
|
$self_fq_class_name,
|
|
$static_fq_class_name,
|
|
$code_location,
|
|
$function_params[$i],
|
|
$i,
|
|
$i,
|
|
$function_storage->allow_named_arg_calls ?? true,
|
|
new VirtualArg(
|
|
StubsGenerator::getExpressionFromType($default_type)
|
|
),
|
|
$default_type,
|
|
$context,
|
|
$class_generic_params,
|
|
$template_result,
|
|
$function_storage->specialize_call ?? true,
|
|
$in_call_map
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($method_id === 'preg_match_all' && count($args) > 3) {
|
|
$args = array_reverse($args, true);
|
|
}
|
|
|
|
$arg_function_params = [];
|
|
$matched_args = [];
|
|
|
|
foreach ($args as $argument_offset => $arg) {
|
|
if ($arg->unpack) {
|
|
if ($function_param_count > $argument_offset) {
|
|
for ($i = $argument_offset; $i < $function_param_count; $i++) {
|
|
$arg_function_params[$argument_offset][] = $function_params[$i];
|
|
}
|
|
}
|
|
|
|
if (($arg_value_type = $statements_analyzer->node_data->getType($arg->value))
|
|
&& $arg_value_type->hasArray()) {
|
|
/**
|
|
* @psalm-suppress PossiblyUndefinedStringArrayOffset
|
|
* @var TArray|TList|TKeyedArray
|
|
*/
|
|
$array_type = $arg_value_type->getAtomicTypes()['array'];
|
|
|
|
if ($array_type instanceof TKeyedArray) {
|
|
$key_types = $array_type->getGenericArrayType()->getChildNodes()[0]->getChildNodes();
|
|
|
|
foreach ($key_types as $key_type) {
|
|
if (!$key_type instanceof Type\Atomic\TLiteralString
|
|
|| ($function_storage && !$function_storage->allow_named_arg_calls)) {
|
|
continue;
|
|
}
|
|
|
|
$param_found = false;
|
|
|
|
foreach ($function_params as $candidate_param) {
|
|
if ($candidate_param->name === $key_type->value || $candidate_param->is_variadic) {
|
|
if ($candidate_param->name === $key_type->value) {
|
|
if (isset($matched_args[$candidate_param->name])) {
|
|
if (IssueBuffer::accepts(
|
|
new InvalidNamedArgument(
|
|
'Parameter $' . $key_type->value . ' has already been used in '
|
|
. ($cased_method_id ?: $method_id),
|
|
new CodeLocation($statements_analyzer, $arg),
|
|
(string)$method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
}
|
|
|
|
$matched_args[$candidate_param->name] = true;
|
|
}
|
|
|
|
$param_found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!$param_found) {
|
|
if (IssueBuffer::accepts(
|
|
new InvalidNamedArgument(
|
|
'Parameter $' . $key_type->value . ' does not exist on function '
|
|
. ($cased_method_id ?: $method_id),
|
|
new CodeLocation($statements_analyzer, $arg),
|
|
(string)$method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} elseif ($arg->name && (!$function_storage || $function_storage->allow_named_arg_calls)) {
|
|
foreach ($function_params as $candidate_param) {
|
|
if ($candidate_param->name === $arg->name->name || $candidate_param->is_variadic) {
|
|
if ($candidate_param->name === $arg->name->name) {
|
|
if (isset($matched_args[$candidate_param->name])) {
|
|
if (IssueBuffer::accepts(
|
|
new InvalidNamedArgument(
|
|
'Parameter $' . $arg->name->name . ' has already been used in '
|
|
. ($cased_method_id ?: $method_id),
|
|
new CodeLocation($statements_analyzer, $arg->name),
|
|
(string) $method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
}
|
|
|
|
$matched_args[$candidate_param->name] = true;
|
|
}
|
|
|
|
$arg_function_params[$argument_offset] = [$candidate_param];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!isset($arg_function_params[$argument_offset])) {
|
|
if (IssueBuffer::accepts(
|
|
new InvalidNamedArgument(
|
|
'Parameter $' . $arg->name->name . ' does not exist on function '
|
|
. ($cased_method_id ?: $method_id),
|
|
new CodeLocation($statements_analyzer, $arg->name),
|
|
(string) $method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
}
|
|
} elseif ($function_param_count > $argument_offset) {
|
|
$arg_function_params[$argument_offset] = [$function_params[$argument_offset]];
|
|
$matched_args[$function_params[$argument_offset]->name] = true;
|
|
} elseif ($last_param && $last_param->is_variadic) {
|
|
$arg_function_params[$argument_offset] = [$last_param];
|
|
$matched_args[$last_param->name] = true;
|
|
}
|
|
}
|
|
|
|
foreach ($args as $argument_offset => $arg) {
|
|
if (!isset($arg_function_params[$argument_offset])) {
|
|
continue;
|
|
}
|
|
|
|
if ($arg_function_params[$argument_offset][0]->by_ref
|
|
&& $method_id !== 'extract'
|
|
) {
|
|
if (self::handlePossiblyMatchingByRefParam(
|
|
$statements_analyzer,
|
|
$codebase,
|
|
(string) $method_id,
|
|
$cased_method_id,
|
|
$last_param,
|
|
$function_params,
|
|
$argument_offset,
|
|
$arg,
|
|
$context,
|
|
$template_result
|
|
) === false) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
$arg_value_type = $statements_analyzer->node_data->getType($arg->value);
|
|
|
|
foreach ($arg_function_params[$argument_offset] as $i => $function_param) {
|
|
if (ArgumentAnalyzer::checkArgumentMatches(
|
|
$statements_analyzer,
|
|
$cased_method_id,
|
|
$method_id instanceof MethodIdentifier ? $method_id : null,
|
|
$self_fq_class_name,
|
|
$static_fq_class_name,
|
|
$code_location,
|
|
$function_param,
|
|
$argument_offset + $i,
|
|
$i,
|
|
$function_storage->allow_named_arg_calls ?? true,
|
|
$arg,
|
|
$arg_value_type,
|
|
$context,
|
|
$class_generic_params,
|
|
$template_result,
|
|
$function_storage->specialize_call ?? true,
|
|
$in_call_map
|
|
) === false) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($statements_analyzer->data_flow_graph instanceof TaintFlowGraph
|
|
&& $cased_method_id
|
|
) {
|
|
foreach ($args as $argument_offset => $_) {
|
|
if (!isset($arg_function_params[$argument_offset])) {
|
|
continue;
|
|
}
|
|
|
|
foreach ($arg_function_params[$argument_offset] as $function_param) {
|
|
if ($function_param->sinks) {
|
|
if (!$function_storage || $function_storage->specialize_call) {
|
|
$sink = TaintSink::getForMethodArgument(
|
|
$cased_method_id,
|
|
$cased_method_id,
|
|
$argument_offset,
|
|
$function_param->location,
|
|
$code_location
|
|
);
|
|
} else {
|
|
$sink = TaintSink::getForMethodArgument(
|
|
$cased_method_id,
|
|
$cased_method_id,
|
|
$argument_offset,
|
|
$function_param->location
|
|
);
|
|
}
|
|
|
|
$sink->taints = $function_param->sinks;
|
|
|
|
$statements_analyzer->data_flow_graph->addSink($sink);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ($method_id === 'array_map' || $method_id === 'array_filter') {
|
|
if ($method_id === 'array_map' && count($args) < 2) {
|
|
if (IssueBuffer::accepts(
|
|
new TooFewArguments(
|
|
'Too few arguments for ' . $method_id,
|
|
$code_location,
|
|
$method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
} elseif ($method_id === 'array_filter' && count($args) < 1) {
|
|
if (IssueBuffer::accepts(
|
|
new TooFewArguments(
|
|
'Too few arguments for ' . $method_id,
|
|
$code_location,
|
|
$method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
}
|
|
|
|
ArrayFunctionArgumentsAnalyzer::checkArgumentsMatch(
|
|
$statements_analyzer,
|
|
$context,
|
|
$args,
|
|
$method_id,
|
|
$context->check_functions
|
|
);
|
|
|
|
return null;
|
|
}
|
|
|
|
self::checkArgCount(
|
|
$statements_analyzer,
|
|
$codebase,
|
|
$function_storage,
|
|
$context,
|
|
$template_result,
|
|
$is_variadic,
|
|
$args,
|
|
$function_params,
|
|
$in_call_map,
|
|
$method_id,
|
|
$cased_method_id,
|
|
$code_location,
|
|
$has_packed_var,
|
|
$packed_var_definite_args
|
|
);
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param array<int, FunctionLikeParameter> $function_params
|
|
* @return false|null
|
|
*/
|
|
private static function handlePossiblyMatchingByRefParam(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
Codebase $codebase,
|
|
string $method_id,
|
|
?string $cased_method_id,
|
|
?FunctionLikeParameter $last_param,
|
|
array $function_params,
|
|
int $argument_offset,
|
|
PhpParser\Node\Arg $arg,
|
|
Context $context,
|
|
?TemplateResult $template_result
|
|
): ?bool {
|
|
if ($arg->value instanceof PhpParser\Node\Scalar
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Cast
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Array_
|
|
|| $arg->value instanceof PhpParser\Node\Expr\ClassConstFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\BinaryOp
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Ternary
|
|
|| (
|
|
(
|
|
$arg->value instanceof PhpParser\Node\Expr\ConstFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\FuncCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\MethodCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\StaticCall
|
|
) && (
|
|
!($arg_value_type = $statements_analyzer->node_data->getType($arg->value))
|
|
|| !$arg_value_type->by_ref
|
|
)
|
|
)
|
|
) {
|
|
if (IssueBuffer::accepts(
|
|
new InvalidPassByReference(
|
|
'Parameter ' . ($argument_offset + 1) . ' of ' . $cased_method_id . ' expects a variable',
|
|
new CodeLocation($statements_analyzer->getSource(), $arg->value)
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
if (!in_array(
|
|
$method_id,
|
|
[
|
|
'ksort', 'asort', 'krsort', 'arsort', 'natcasesort', 'natsort',
|
|
'reset', 'end', 'next', 'prev', 'array_pop', 'array_shift',
|
|
'array_push', 'array_unshift', 'socket_select', 'array_splice',
|
|
],
|
|
true
|
|
)) {
|
|
$by_ref_type = null;
|
|
$by_ref_out_type = null;
|
|
|
|
$check_null_ref = true;
|
|
|
|
if ($last_param) {
|
|
if ($argument_offset < count($function_params)) {
|
|
$function_param = $function_params[$argument_offset];
|
|
} else {
|
|
$function_param = $last_param;
|
|
}
|
|
|
|
if ($function_param->type) {
|
|
$by_ref_type = clone $function_param->type;
|
|
}
|
|
if ($function_param->out_type) {
|
|
$by_ref_out_type = clone $function_param->out_type;
|
|
}
|
|
|
|
if ($by_ref_type && $by_ref_type->isNullable()) {
|
|
$check_null_ref = false;
|
|
}
|
|
|
|
if ($template_result && $by_ref_type) {
|
|
$original_by_ref_type = clone $by_ref_type;
|
|
|
|
$by_ref_type = TemplateStandinTypeReplacer::replace(
|
|
clone $by_ref_type,
|
|
$template_result,
|
|
$codebase,
|
|
$statements_analyzer,
|
|
$statements_analyzer->node_data->getType($arg->value),
|
|
$argument_offset,
|
|
$context->self,
|
|
$context->calling_method_id ?: $context->calling_function_id
|
|
);
|
|
|
|
if ($template_result->lower_bounds) {
|
|
TemplateInferredTypeReplacer::replace(
|
|
$original_by_ref_type,
|
|
$template_result,
|
|
$codebase
|
|
);
|
|
|
|
$by_ref_type = $original_by_ref_type;
|
|
}
|
|
}
|
|
|
|
if ($template_result && $by_ref_out_type) {
|
|
$original_by_ref_out_type = clone $by_ref_out_type;
|
|
|
|
$by_ref_out_type = TemplateStandinTypeReplacer::replace(
|
|
clone $by_ref_out_type,
|
|
$template_result,
|
|
$codebase,
|
|
$statements_analyzer,
|
|
$statements_analyzer->node_data->getType($arg->value),
|
|
$argument_offset,
|
|
$context->self,
|
|
$context->calling_method_id ?: $context->calling_function_id
|
|
);
|
|
|
|
if ($template_result->lower_bounds) {
|
|
TemplateInferredTypeReplacer::replace(
|
|
$original_by_ref_out_type,
|
|
$template_result,
|
|
$codebase
|
|
);
|
|
|
|
$by_ref_out_type = $original_by_ref_out_type;
|
|
}
|
|
}
|
|
|
|
if ($by_ref_type && $function_param->is_variadic && $arg->unpack) {
|
|
$by_ref_type = new Type\Union([
|
|
new Type\Atomic\TArray([
|
|
Type::getInt(),
|
|
$by_ref_type,
|
|
]),
|
|
]);
|
|
}
|
|
}
|
|
|
|
$by_ref_type = $by_ref_type ?: Type::getMixed();
|
|
|
|
AssignmentAnalyzer::assignByRefParam(
|
|
$statements_analyzer,
|
|
$arg->value,
|
|
$by_ref_type,
|
|
$by_ref_out_type ?: $by_ref_type,
|
|
$context,
|
|
$method_id && (strpos($method_id, '::') !== false || !InternalCallMapHandler::inCallMap($method_id)),
|
|
$check_null_ref
|
|
);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @return false|null
|
|
*/
|
|
private static function evaluateArbitraryParam(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
PhpParser\Node\Arg $arg,
|
|
Context $context
|
|
): ?bool {
|
|
// there are a bunch of things we want to evaluate even when we don't
|
|
// know what function/method is being called
|
|
if ($arg->value instanceof PhpParser\Node\Expr\Closure
|
|
|| $arg->value instanceof PhpParser\Node\Expr\ConstFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\ClassConstFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\FuncCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\MethodCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\StaticCall
|
|
|| $arg->value instanceof PhpParser\Node\Expr\New_
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Cast
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Assign
|
|
|| $arg->value instanceof PhpParser\Node\Expr\ArrayDimFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\PropertyFetch
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Array_
|
|
|| $arg->value instanceof PhpParser\Node\Expr\BinaryOp
|
|
|| $arg->value instanceof PhpParser\Node\Expr\Ternary
|
|
|| $arg->value instanceof PhpParser\Node\Scalar\Encapsed
|
|
|| $arg->value instanceof PhpParser\Node\Expr\PostInc
|
|
|| $arg->value instanceof PhpParser\Node\Expr\PostDec
|
|
|| $arg->value instanceof PhpParser\Node\Expr\PreInc
|
|
|| $arg->value instanceof PhpParser\Node\Expr\PreDec
|
|
) {
|
|
$was_inside_call = $context->inside_call;
|
|
$context->inside_call = true;
|
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $arg->value, $context) === false) {
|
|
return false;
|
|
}
|
|
|
|
if (!$was_inside_call) {
|
|
$context->inside_call = false;
|
|
}
|
|
}
|
|
|
|
if ($arg->value instanceof PhpParser\Node\Expr\PropertyFetch
|
|
&& $arg->value->name instanceof PhpParser\Node\Identifier
|
|
) {
|
|
$var_id = '$' . $arg->value->name->name;
|
|
} else {
|
|
$var_id = ExpressionIdentifier::getVarId(
|
|
$arg->value,
|
|
$statements_analyzer->getFQCLN(),
|
|
$statements_analyzer
|
|
);
|
|
}
|
|
|
|
if ($var_id) {
|
|
if ($arg->value instanceof PhpParser\Node\Expr\Variable) {
|
|
$statements_analyzer->registerPossiblyUndefinedVariable($var_id, $arg->value);
|
|
}
|
|
|
|
if (!$context->hasVariable($var_id)
|
|
|| $context->vars_in_scope[$var_id]->isNull()
|
|
) {
|
|
if (!isset($context->vars_in_scope[$var_id])
|
|
&& $arg->value instanceof PhpParser\Node\Expr\Variable
|
|
) {
|
|
if (IssueBuffer::accepts(
|
|
new PossiblyUndefinedVariable(
|
|
'Variable ' . $var_id
|
|
. ' must be defined prior to use within an unknown function or method',
|
|
new CodeLocation($statements_analyzer->getSource(), $arg->value)
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
}
|
|
|
|
// we don't know if it exists, assume it's passed by reference
|
|
$context->vars_in_scope[$var_id] = Type::getMixed();
|
|
$context->vars_possibly_in_scope[$var_id] = true;
|
|
} else {
|
|
$was_inside_call = $context->inside_call;
|
|
$context->inside_call = true;
|
|
ExpressionAnalyzer::analyze($statements_analyzer, $arg->value, $context);
|
|
$context->inside_call = $was_inside_call;
|
|
|
|
$context->removeVarFromConflictingClauses(
|
|
$var_id,
|
|
$context->vars_in_scope[$var_id],
|
|
$statements_analyzer
|
|
);
|
|
|
|
foreach ($context->vars_in_scope[$var_id]->getAtomicTypes() as $type) {
|
|
if ($type instanceof TArray && $type->type_params[1]->isEmpty()) {
|
|
$context->vars_in_scope[$var_id]->removeType('array');
|
|
$context->vars_in_scope[$var_id]->addType(
|
|
new TArray(
|
|
[Type::getArrayKey(), Type::getMixed()]
|
|
)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @return false|null
|
|
*/
|
|
private static function handleByRefFunctionArg(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
?string $method_id,
|
|
int $argument_offset,
|
|
PhpParser\Node\Arg $arg,
|
|
Context $context
|
|
): ?bool {
|
|
$var_id = ExpressionIdentifier::getVarId(
|
|
$arg->value,
|
|
$statements_analyzer->getFQCLN(),
|
|
$statements_analyzer
|
|
);
|
|
|
|
$builtin_array_functions = [
|
|
'ksort', 'asort', 'krsort', 'arsort', 'natcasesort', 'natsort',
|
|
'reset', 'end', 'next', 'prev', 'array_pop', 'array_shift',
|
|
];
|
|
|
|
if (($var_id && isset($context->vars_in_scope[$var_id]))
|
|
|| ($method_id
|
|
&& in_array(
|
|
$method_id,
|
|
$builtin_array_functions,
|
|
true
|
|
))
|
|
) {
|
|
$was_inside_assignment = $context->inside_assignment;
|
|
$context->inside_assignment = true;
|
|
|
|
// if the variable is in scope, get or we're in a special array function,
|
|
// figure out its type before proceeding
|
|
if (ExpressionAnalyzer::analyze(
|
|
$statements_analyzer,
|
|
$arg->value,
|
|
$context
|
|
) === false) {
|
|
return false;
|
|
}
|
|
|
|
$context->inside_assignment = $was_inside_assignment;
|
|
}
|
|
|
|
// special handling for array sort
|
|
if ($argument_offset === 0
|
|
&& $method_id
|
|
&& in_array(
|
|
$method_id,
|
|
$builtin_array_functions,
|
|
true
|
|
)
|
|
) {
|
|
if (in_array($method_id, ['array_pop', 'array_shift'], true)) {
|
|
ArrayFunctionArgumentsAnalyzer::handleByRefArrayAdjustment(
|
|
$statements_analyzer,
|
|
$arg,
|
|
$context,
|
|
$method_id === 'array_shift'
|
|
);
|
|
|
|
return null;
|
|
}
|
|
|
|
// noops
|
|
if (in_array($method_id, ['reset', 'end', 'next', 'prev', 'ksort'], true)) {
|
|
return null;
|
|
}
|
|
|
|
if (($arg_value_type = $statements_analyzer->node_data->getType($arg->value))
|
|
&& $arg_value_type->hasArray()
|
|
) {
|
|
/**
|
|
* @psalm-suppress PossiblyUndefinedStringArrayOffset
|
|
* @var TArray|TList|TKeyedArray
|
|
*/
|
|
$array_type = $arg_value_type->getAtomicTypes()['array'];
|
|
|
|
if ($array_type instanceof TKeyedArray) {
|
|
$array_type = $array_type->getGenericArrayType();
|
|
}
|
|
|
|
if ($array_type instanceof TList) {
|
|
$array_type = new TArray([Type::getInt(), $array_type->type_param]);
|
|
}
|
|
|
|
$by_ref_type = new Type\Union([clone $array_type]);
|
|
|
|
AssignmentAnalyzer::assignByRefParam(
|
|
$statements_analyzer,
|
|
$arg->value,
|
|
$by_ref_type,
|
|
$by_ref_type,
|
|
$context,
|
|
false
|
|
);
|
|
|
|
return null;
|
|
}
|
|
}
|
|
|
|
if ($method_id === 'socket_select') {
|
|
if (ExpressionAnalyzer::analyze(
|
|
$statements_analyzer,
|
|
$arg->value,
|
|
$context
|
|
) === false) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (!$arg->value instanceof PhpParser\Node\Expr\Variable) {
|
|
$suppressed_issues = $statements_analyzer->getSuppressedIssues();
|
|
|
|
if (!in_array('EmptyArrayAccess', $suppressed_issues, true)) {
|
|
$statements_analyzer->addSuppressedIssues(['EmptyArrayAccess']);
|
|
}
|
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $arg->value, $context) === false) {
|
|
return false;
|
|
}
|
|
|
|
if (!in_array('EmptyArrayAccess', $suppressed_issues, true)) {
|
|
$statements_analyzer->removeSuppressedIssues(['EmptyArrayAccess']);
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* @param list<PhpParser\Node\Arg> $args
|
|
* @param array<int,FunctionLikeParameter> $function_params
|
|
* @param array<string, array<string, Type\Union>> $class_generic_params
|
|
*/
|
|
private static function getProvisionalTemplateResultForFunctionLike(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
Codebase $codebase,
|
|
Context $context,
|
|
?ClassLikeStorage $class_storage,
|
|
?string $self_fq_class_name,
|
|
?ClassLikeStorage $calling_class_storage,
|
|
FunctionLikeStorage $function_storage,
|
|
array $class_generic_params,
|
|
?TemplateResult $class_template_result,
|
|
array $args,
|
|
array $function_params,
|
|
?FunctionLikeParameter $last_param
|
|
): ?TemplateResult {
|
|
$template_types = CallAnalyzer::getTemplateTypesForCall(
|
|
$codebase,
|
|
$class_storage,
|
|
$self_fq_class_name,
|
|
$calling_class_storage,
|
|
$function_storage->template_types ?: [],
|
|
$class_generic_params
|
|
);
|
|
|
|
if (!$template_types) {
|
|
return null;
|
|
}
|
|
|
|
if (!$class_template_result) {
|
|
return new TemplateResult($template_types, []);
|
|
}
|
|
|
|
$template_result = $class_template_result;
|
|
|
|
if (!$template_result->template_types) {
|
|
$template_result->template_types = $template_types;
|
|
}
|
|
|
|
foreach ($args as $argument_offset => $arg) {
|
|
$function_param = null;
|
|
|
|
if ($arg->name && $function_storage->allow_named_arg_calls) {
|
|
foreach ($function_params as $candidate_param) {
|
|
if ($candidate_param->name === $arg->name->name) {
|
|
$function_param = $candidate_param;
|
|
break;
|
|
}
|
|
}
|
|
} elseif ($argument_offset < count($function_params)) {
|
|
$function_param = $function_params[$argument_offset];
|
|
} elseif ($last_param && $last_param->is_variadic) {
|
|
$function_param = $last_param;
|
|
}
|
|
|
|
if (!$function_param
|
|
|| !$function_param->type
|
|
) {
|
|
continue;
|
|
}
|
|
|
|
$arg_value_type = $statements_analyzer->node_data->getType($arg->value);
|
|
|
|
if (!$arg_value_type) {
|
|
continue;
|
|
}
|
|
|
|
$fleshed_out_param_type = \Psalm\Internal\Type\TypeExpander::expandUnion(
|
|
$codebase,
|
|
$function_param->type,
|
|
$class_storage->name ?? null,
|
|
$calling_class_storage->name ?? null,
|
|
null,
|
|
true,
|
|
false,
|
|
$calling_class_storage->final ?? false
|
|
);
|
|
|
|
TemplateStandinTypeReplacer::replace(
|
|
$fleshed_out_param_type,
|
|
$template_result,
|
|
$codebase,
|
|
$statements_analyzer,
|
|
$arg_value_type,
|
|
$argument_offset,
|
|
$context->self,
|
|
$context->calling_method_id ?: $context->calling_function_id,
|
|
false
|
|
);
|
|
}
|
|
|
|
return $template_result;
|
|
}
|
|
|
|
/**
|
|
* @param array<int, PhpParser\Node\Arg> $args
|
|
* @param string|MethodIdentifier|null $method_id
|
|
* @param array<int,FunctionLikeParameter> $function_params
|
|
*/
|
|
private static function checkArgCount(
|
|
StatementsAnalyzer $statements_analyzer,
|
|
Codebase $codebase,
|
|
?FunctionLikeStorage $function_storage,
|
|
Context $context,
|
|
?TemplateResult $template_result,
|
|
bool $is_variadic,
|
|
array $args,
|
|
array $function_params,
|
|
bool $in_call_map,
|
|
$method_id,
|
|
?string $cased_method_id,
|
|
CodeLocation $code_location,
|
|
bool $has_packed_var,
|
|
int $packed_var_definite_args
|
|
): void {
|
|
if (!$is_variadic
|
|
&& count($args) > count($function_params)
|
|
&& (!count($function_params) || $function_params[count($function_params) - 1]->name !== '...=')
|
|
&& ($in_call_map
|
|
|| !$function_storage instanceof \Psalm\Storage\MethodStorage
|
|
|| $function_storage->is_static
|
|
|| ($method_id instanceof MethodIdentifier
|
|
&& $method_id->method_name === '__construct'))
|
|
) {
|
|
if (IssueBuffer::accepts(
|
|
new TooManyArguments(
|
|
'Too many arguments for ' . ($cased_method_id ?: $method_id)
|
|
. ' - expecting ' . count($function_params) . ' but saw ' . count($args),
|
|
$code_location,
|
|
(string)$method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (!$has_packed_var && count($args) < count($function_params)) {
|
|
if ($function_storage) {
|
|
$expected_param_count = $function_storage->required_param_count;
|
|
} else {
|
|
for ($i = 0, $j = count($function_params); $i < $j; ++$i) {
|
|
$param = $function_params[$i];
|
|
|
|
if ($param->is_optional || $param->is_variadic) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
$expected_param_count = $i;
|
|
}
|
|
|
|
for ($i = count($args) + $packed_var_definite_args, $j = count($function_params); $i < $j; ++$i) {
|
|
$param = $function_params[$i];
|
|
|
|
if (!$param->is_optional
|
|
&& !$param->is_variadic
|
|
&& ($in_call_map
|
|
|| !$function_storage instanceof \Psalm\Storage\MethodStorage
|
|
|| $function_storage->is_static
|
|
|| ($method_id instanceof MethodIdentifier
|
|
&& $method_id->method_name === '__construct'))
|
|
) {
|
|
if (IssueBuffer::accepts(
|
|
new TooFewArguments(
|
|
'Too few arguments for ' . $cased_method_id
|
|
. ' - expecting ' . $expected_param_count
|
|
. ' but saw ' . (count($args) + $packed_var_definite_args),
|
|
$code_location,
|
|
(string)$method_id
|
|
),
|
|
$statements_analyzer->getSuppressedIssues()
|
|
)) {
|
|
// fall through
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
if ($param->is_optional
|
|
&& $param->type
|
|
&& $param->default_type
|
|
&& !$param->is_variadic
|
|
&& $template_result
|
|
) {
|
|
if ($param->default_type instanceof Type\Union) {
|
|
$default_type = clone $param->default_type;
|
|
} else {
|
|
$default_type_atomic = \Psalm\Internal\Codebase\ConstantTypeResolver::resolve(
|
|
$codebase->classlikes,
|
|
$param->default_type,
|
|
$statements_analyzer
|
|
);
|
|
|
|
$default_type = new Type\Union([$default_type_atomic]);
|
|
}
|
|
|
|
TemplateStandinTypeReplacer::replace(
|
|
$param->type,
|
|
$template_result,
|
|
$codebase,
|
|
$statements_analyzer,
|
|
$default_type,
|
|
$i,
|
|
$context->self,
|
|
$context->calling_method_id ?: $context->calling_function_id,
|
|
true
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|