2016-11-01 19:14:35 +01:00
|
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
|
|
|
|
|
use PhpParser;
|
2021-12-04 03:37:19 +01:00
|
|
|
|
use PhpParser\Node\Identifier;
|
|
|
|
|
use PhpParser\Node\Name;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
use Psalm\CodeLocation;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Codebase;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
use Psalm\Context;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\FileSource;
|
|
|
|
|
use Psalm\Internal\Algebra;
|
2020-11-03 22:15:44 +01:00
|
|
|
|
use Psalm\Internal\Algebra\FormulaGenerator;
|
2018-11-06 03:57:36 +01:00
|
|
|
|
use Psalm\Internal\Analyzer\ClassLikeAnalyzer;
|
|
|
|
|
use Psalm\Internal\Analyzer\FunctionLikeAnalyzer;
|
2021-12-14 00:31:46 +01:00
|
|
|
|
use Psalm\Internal\Analyzer\Statements\Expression\Call\ArgumentsAnalyzer;
|
2018-11-06 03:57:36 +01:00
|
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Internal\Analyzer\TraitAnalyzer;
|
|
|
|
|
use Psalm\Internal\MethodIdentifier;
|
|
|
|
|
use Psalm\Internal\Type\Comparator\TypeComparisonResult;
|
2020-07-22 01:40:35 +02:00
|
|
|
|
use Psalm\Internal\Type\Comparator\UnionTypeComparator;
|
2020-11-27 17:43:23 +01:00
|
|
|
|
use Psalm\Internal\Type\TemplateBound;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Internal\Type\TemplateInferredTypeReplacer;
|
2019-11-04 03:27:40 +01:00
|
|
|
|
use Psalm\Internal\Type\TemplateResult;
|
2021-07-11 18:03:21 +02:00
|
|
|
|
use Psalm\Internal\Type\TemplateStandinTypeReplacer;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Internal\Type\TypeExpander;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
use Psalm\Issue\ArgumentTypeCoercion;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
use Psalm\Issue\InvalidArgument;
|
2021-10-17 10:29:25 +02:00
|
|
|
|
use Psalm\Issue\InvalidDocblock;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
use Psalm\Issue\InvalidScalarArgument;
|
2019-04-26 00:02:19 +02:00
|
|
|
|
use Psalm\Issue\MixedArgumentTypeCoercion;
|
2022-01-20 23:33:06 +01:00
|
|
|
|
use Psalm\Issue\TypeDoesNotContainType;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
use Psalm\Issue\UndefinedFunction;
|
|
|
|
|
use Psalm\IssueBuffer;
|
2021-02-15 22:18:41 +01:00
|
|
|
|
use Psalm\Node\Expr\BinaryOp\VirtualIdentical;
|
|
|
|
|
use Psalm\Node\Expr\VirtualConstFetch;
|
|
|
|
|
use Psalm\Node\VirtualName;
|
2022-01-20 23:33:06 +01:00
|
|
|
|
use Psalm\Storage\Assertion\Falsy;
|
|
|
|
|
use Psalm\Storage\Assertion\IsIdentical;
|
|
|
|
|
use Psalm\Storage\Assertion\IsType;
|
|
|
|
|
use Psalm\Storage\Assertion\Truthy;
|
2017-02-10 02:35:17 +01:00
|
|
|
|
use Psalm\Storage\ClassLikeStorage;
|
2022-01-20 23:33:06 +01:00
|
|
|
|
use Psalm\Storage\Possibilities;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
use Psalm\Type;
|
2017-05-19 06:48:26 +02:00
|
|
|
|
use Psalm\Type\Atomic\TNamedObject;
|
2021-12-13 04:45:57 +01:00
|
|
|
|
use Psalm\Type\Atomic\TObjectWithProperties;
|
|
|
|
|
use Psalm\Type\Atomic\TTemplateParam;
|
2022-01-20 23:33:06 +01:00
|
|
|
|
use Psalm\Type\Atomic\TTrue;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Type\Reconciler;
|
2021-12-13 16:28:14 +01:00
|
|
|
|
use Psalm\Type\Union;
|
2021-12-03 21:40:18 +01:00
|
|
|
|
use UnexpectedValueException;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
|
2021-07-11 18:03:21 +02:00
|
|
|
|
use function array_filter;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
use function array_map;
|
|
|
|
|
use function array_merge;
|
2021-07-11 18:03:21 +02:00
|
|
|
|
use function array_unique;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use function count;
|
2021-10-17 10:29:25 +02:00
|
|
|
|
use function explode;
|
2021-10-09 15:01:13 +02:00
|
|
|
|
use function implode;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use function in_array;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
use function is_int;
|
2021-10-17 10:29:25 +02:00
|
|
|
|
use function is_numeric;
|
2021-12-03 21:07:25 +01:00
|
|
|
|
use function mt_rand;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use function preg_match;
|
|
|
|
|
use function preg_replace;
|
2021-12-03 21:07:25 +01:00
|
|
|
|
use function spl_object_id;
|
2019-09-06 03:00:02 +02:00
|
|
|
|
use function str_replace;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
use function strpos;
|
|
|
|
|
use function strtolower;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
|
2018-12-02 00:37:49 +01:00
|
|
|
|
/**
|
|
|
|
|
* @internal
|
|
|
|
|
*/
|
2018-11-06 03:57:36 +01:00
|
|
|
|
class CallAnalyzer
|
2016-11-01 19:14:35 +01:00
|
|
|
|
{
|
2017-02-10 02:35:17 +01:00
|
|
|
|
public static function collectSpecialInformation(
|
2018-11-06 03:57:36 +01:00
|
|
|
|
FunctionLikeAnalyzer $source,
|
2020-07-26 20:46:52 +02:00
|
|
|
|
string $method_name,
|
2017-02-10 02:35:17 +01:00
|
|
|
|
Context $context
|
2020-10-12 21:02:52 +02:00
|
|
|
|
): void {
|
2021-10-13 18:29:25 +02:00
|
|
|
|
$method_name_lc = strtolower($method_name);
|
2017-02-10 02:35:17 +01:00
|
|
|
|
$fq_class_name = (string)$source->getFQCLN();
|
2016-11-02 07:29:00 +01:00
|
|
|
|
|
2018-11-11 18:01:14 +01:00
|
|
|
|
$project_analyzer = $source->getFileAnalyzer()->project_analyzer;
|
2018-11-06 03:57:36 +01:00
|
|
|
|
$codebase = $source->getCodebase();
|
2017-07-29 21:05:06 +02:00
|
|
|
|
|
2017-02-10 02:35:17 +01:00
|
|
|
|
if ($context->collect_mutations &&
|
|
|
|
|
$context->self &&
|
|
|
|
|
(
|
|
|
|
|
$context->self === $fq_class_name ||
|
2018-02-01 06:50:01 +01:00
|
|
|
|
$codebase->classExtends(
|
2017-02-10 02:35:17 +01:00
|
|
|
|
$context->self,
|
|
|
|
|
$fq_class_name
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
) {
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$method_id = new MethodIdentifier(
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$fq_class_name,
|
2021-10-13 18:29:25 +02:00
|
|
|
|
$method_name_lc
|
2020-02-15 02:54:26 +01:00
|
|
|
|
);
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
if ((string) $method_id !== $source->getId()) {
|
2018-01-24 19:38:53 +01:00
|
|
|
|
if ($context->collect_initializations) {
|
2020-02-18 18:53:54 +01:00
|
|
|
|
if (isset($context->initialized_methods[(string) $method_id])) {
|
2018-01-24 19:38:53 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($context->initialized_methods === null) {
|
|
|
|
|
$context->initialized_methods = [];
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-18 18:53:54 +01:00
|
|
|
|
$context->initialized_methods[(string) $method_id] = true;
|
2018-01-24 19:38:53 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-03-02 21:07:26 +01:00
|
|
|
|
$project_analyzer->getMethodMutations(
|
|
|
|
|
$method_id,
|
|
|
|
|
$context,
|
|
|
|
|
$source->getRootFilePath(),
|
|
|
|
|
$source->getRootFileName()
|
|
|
|
|
);
|
2017-12-23 01:01:59 +01:00
|
|
|
|
}
|
2017-02-10 02:35:17 +01:00
|
|
|
|
} elseif ($context->collect_initializations &&
|
|
|
|
|
$context->self &&
|
|
|
|
|
(
|
2019-03-20 00:43:12 +01:00
|
|
|
|
$context->self === $fq_class_name
|
|
|
|
|
|| $codebase->classlikes->classExtends(
|
2017-02-10 02:35:17 +01:00
|
|
|
|
$context->self,
|
|
|
|
|
$fq_class_name
|
|
|
|
|
)
|
2017-02-22 06:51:34 +01:00
|
|
|
|
) &&
|
|
|
|
|
$source->getMethodName() !== $method_name
|
2017-02-10 02:35:17 +01:00
|
|
|
|
) {
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$method_id = new MethodIdentifier($fq_class_name, $method_name_lc);
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2019-01-09 14:35:53 +01:00
|
|
|
|
$declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id);
|
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
if (isset($context->vars_in_scope['$this'])) {
|
2020-01-04 18:20:26 +01:00
|
|
|
|
foreach ($context->vars_in_scope['$this']->getAtomicTypes() as $atomic_type) {
|
2019-03-01 23:30:55 +01:00
|
|
|
|
if ($atomic_type instanceof TNamedObject) {
|
|
|
|
|
if ($fq_class_name === $atomic_type->value) {
|
|
|
|
|
$alt_declaring_method_id = $declaring_method_id;
|
|
|
|
|
} else {
|
2019-01-09 14:35:53 +01:00
|
|
|
|
$fq_class_name = $atomic_type->value;
|
2019-03-01 23:30:55 +01:00
|
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$method_id = new MethodIdentifier(
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$fq_class_name,
|
2021-10-13 18:29:25 +02:00
|
|
|
|
$method_name_lc
|
2020-02-15 02:54:26 +01:00
|
|
|
|
);
|
2019-01-09 14:35:53 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
$alt_declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id);
|
|
|
|
|
}
|
2019-01-09 14:35:53 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
if ($alt_declaring_method_id) {
|
|
|
|
|
$declaring_method_id = $alt_declaring_method_id;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2019-01-09 14:35:53 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
if (!$atomic_type->extra_types) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2019-01-09 14:35:53 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
foreach ($atomic_type->extra_types as $intersection_type) {
|
|
|
|
|
if ($intersection_type instanceof TNamedObject) {
|
|
|
|
|
$fq_class_name = $intersection_type->value;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$method_id = new MethodIdentifier(
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$fq_class_name,
|
2021-10-13 18:29:25 +02:00
|
|
|
|
$method_name_lc
|
2020-02-15 02:54:26 +01:00
|
|
|
|
);
|
2019-01-09 14:35:53 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
$alt_declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id);
|
2019-01-09 14:35:53 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
if ($alt_declaring_method_id) {
|
|
|
|
|
$declaring_method_id = $alt_declaring_method_id;
|
|
|
|
|
break 2;
|
2019-01-09 14:35:53 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-03-01 23:30:55 +01:00
|
|
|
|
}
|
2019-01-09 14:35:53 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
if (!$declaring_method_id) {
|
|
|
|
|
// can happen for __call
|
|
|
|
|
return;
|
2019-01-09 14:35:53 +01:00
|
|
|
|
}
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
if (isset($context->initialized_methods[(string) $declaring_method_id])) {
|
2018-01-24 19:11:23 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($context->initialized_methods === null) {
|
|
|
|
|
$context->initialized_methods = [];
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$context->initialized_methods[(string) $declaring_method_id] = true;
|
2018-01-24 19:11:23 +01:00
|
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
|
$method_storage = $codebase->methods->getStorage($declaring_method_id);
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2018-11-11 18:01:14 +01:00
|
|
|
|
$class_analyzer = $source->getSource();
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2020-07-26 20:46:52 +02:00
|
|
|
|
$is_final = $method_storage->final;
|
|
|
|
|
|
|
|
|
|
if ($method_name !== $declaring_method_id->method_name) {
|
|
|
|
|
$appearing_method_id = $codebase->methods->getAppearingMethodId($method_id);
|
|
|
|
|
|
|
|
|
|
if ($appearing_method_id) {
|
|
|
|
|
$appearing_class_storage = $codebase->classlike_storage_provider->get(
|
|
|
|
|
$appearing_method_id->fq_class_name
|
|
|
|
|
);
|
|
|
|
|
|
2021-10-13 18:29:25 +02:00
|
|
|
|
if (isset($appearing_class_storage->trait_final_map[$method_name_lc])) {
|
2020-07-26 20:46:52 +02:00
|
|
|
|
$is_final = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-21 06:04:47 +02:00
|
|
|
|
if ($class_analyzer instanceof ClassLikeAnalyzer
|
|
|
|
|
&& !$method_storage->is_static
|
|
|
|
|
&& ($context->collect_nonprivate_initializations
|
|
|
|
|
|| $method_storage->visibility === ClassLikeAnalyzer::VISIBILITY_PRIVATE
|
2020-07-26 20:46:52 +02:00
|
|
|
|
|| $is_final)
|
2020-04-21 06:04:47 +02:00
|
|
|
|
) {
|
2017-02-10 02:35:17 +01:00
|
|
|
|
$local_vars_in_scope = [];
|
|
|
|
|
|
2020-12-06 17:07:59 +01:00
|
|
|
|
foreach ($context->vars_in_scope as $var_id => $type) {
|
|
|
|
|
if (strpos($var_id, '$this->') === 0) {
|
|
|
|
|
if ($type->initialized) {
|
|
|
|
|
$local_vars_in_scope[$var_id] = $context->vars_in_scope[$var_id];
|
|
|
|
|
|
2022-01-07 23:16:36 +01:00
|
|
|
|
$context->remove($var_id, false);
|
2020-12-06 17:07:59 +01:00
|
|
|
|
}
|
|
|
|
|
} elseif ($var_id !== '$this') {
|
|
|
|
|
$local_vars_in_scope[$var_id] = $context->vars_in_scope[$var_id];
|
2017-02-10 02:35:17 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-06 17:07:59 +01:00
|
|
|
|
$local_vars_possibly_in_scope = $context->vars_possibly_in_scope;
|
|
|
|
|
|
2020-03-26 17:35:27 +01:00
|
|
|
|
$old_calling_method_id = $context->calling_method_id;
|
2020-01-03 17:29:45 +01:00
|
|
|
|
|
2019-03-01 23:30:55 +01:00
|
|
|
|
if ($fq_class_name === $source->getFQCLN()) {
|
2020-07-26 21:21:05 +02:00
|
|
|
|
$class_analyzer->getMethodMutations($declaring_method_id->method_name, $context);
|
2019-03-01 23:30:55 +01:00
|
|
|
|
} else {
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$declaring_fq_class_name = $declaring_method_id->fq_class_name;
|
2019-03-01 23:30:55 +01:00
|
|
|
|
|
|
|
|
|
$old_self = $context->self;
|
|
|
|
|
$context->self = $declaring_fq_class_name;
|
2019-03-02 21:07:26 +01:00
|
|
|
|
$project_analyzer->getMethodMutations(
|
|
|
|
|
$declaring_method_id,
|
|
|
|
|
$context,
|
|
|
|
|
$source->getRootFilePath(),
|
|
|
|
|
$source->getRootFileName()
|
|
|
|
|
);
|
2019-03-01 23:30:55 +01:00
|
|
|
|
$context->self = $old_self;
|
|
|
|
|
}
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2020-03-26 17:35:27 +01:00
|
|
|
|
$context->calling_method_id = $old_calling_method_id;
|
2020-01-03 17:29:45 +01:00
|
|
|
|
|
2017-02-10 02:35:17 +01:00
|
|
|
|
foreach ($local_vars_in_scope as $var => $type) {
|
|
|
|
|
$context->vars_in_scope[$var] = $type;
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-28 18:43:19 +01:00
|
|
|
|
foreach ($local_vars_possibly_in_scope as $var => $_) {
|
2017-12-03 00:28:18 +01:00
|
|
|
|
$context->vars_possibly_in_scope[$var] = true;
|
2017-02-10 02:35:17 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-11-01 19:14:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-10 02:35:17 +01:00
|
|
|
|
/**
|
2020-10-28 17:45:26 +01:00
|
|
|
|
* @param list<PhpParser\Node\Arg> $args
|
2017-02-10 02:35:17 +01:00
|
|
|
|
*/
|
2020-09-01 00:56:21 +02:00
|
|
|
|
public static function checkMethodArgs(
|
2021-12-03 20:11:20 +01:00
|
|
|
|
?MethodIdentifier $method_id,
|
2017-02-10 02:35:17 +01:00
|
|
|
|
array $args,
|
2022-01-31 15:36:16 +01:00
|
|
|
|
TemplateResult $template_result,
|
2017-02-10 02:35:17 +01:00
|
|
|
|
Context $context,
|
|
|
|
|
CodeLocation $code_location,
|
2018-11-11 18:01:14 +01:00
|
|
|
|
StatementsAnalyzer $statements_analyzer
|
2021-12-05 18:51:26 +01:00
|
|
|
|
): bool {
|
2018-11-11 18:19:53 +01:00
|
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
2017-01-02 02:10:28 +01:00
|
|
|
|
|
2020-10-03 02:58:51 +02:00
|
|
|
|
if (!$method_id) {
|
2021-12-14 00:31:46 +01:00
|
|
|
|
return ArgumentsAnalyzer::analyze(
|
2020-10-03 02:58:51 +02:00
|
|
|
|
$statements_analyzer,
|
|
|
|
|
$args,
|
|
|
|
|
null,
|
|
|
|
|
null,
|
|
|
|
|
true,
|
|
|
|
|
$context,
|
2022-01-31 15:36:16 +01:00
|
|
|
|
$template_result
|
2020-10-03 02:58:51 +02:00
|
|
|
|
) !== false;
|
2017-02-10 02:35:17 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-10-03 02:58:51 +02:00
|
|
|
|
$method_params = $codebase->methods->getMethodParams($method_id, $statements_analyzer, $args, $context);
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$fq_class_name = $method_id->fq_class_name;
|
|
|
|
|
$method_name = $method_id->method_name;
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$fq_class_name = strtolower($codebase->classlikes->getUnAliasedName($fq_class_name));
|
2018-12-21 17:32:44 +01:00
|
|
|
|
|
2018-11-11 18:19:53 +01:00
|
|
|
|
$class_storage = $codebase->classlike_storage_provider->get($fq_class_name);
|
2017-11-02 20:07:39 +01:00
|
|
|
|
|
|
|
|
|
$method_storage = null;
|
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
if (isset($class_storage->declaring_method_ids[$method_name])) {
|
|
|
|
|
$declaring_method_id = $class_storage->declaring_method_ids[$method_name];
|
2017-11-02 20:07:39 +01:00
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$declaring_fq_class_name = $declaring_method_id->fq_class_name;
|
|
|
|
|
$declaring_method_name = $declaring_method_id->method_name;
|
2017-11-02 20:07:39 +01:00
|
|
|
|
|
|
|
|
|
if ($declaring_fq_class_name !== $fq_class_name) {
|
2018-11-11 18:19:53 +01:00
|
|
|
|
$declaring_class_storage = $codebase->classlike_storage_provider->get($declaring_fq_class_name);
|
2017-11-02 20:07:39 +01:00
|
|
|
|
} else {
|
|
|
|
|
$declaring_class_storage = $class_storage;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
if (!isset($declaring_class_storage->methods[$declaring_method_name])) {
|
2021-12-03 21:40:18 +01:00
|
|
|
|
throw new UnexpectedValueException('Storage should not be empty here');
|
2017-11-02 20:07:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
|
$method_storage = $declaring_class_storage->methods[$declaring_method_name];
|
2018-06-22 07:13:49 +02:00
|
|
|
|
|
2019-03-25 16:25:43 +01:00
|
|
|
|
if ($declaring_class_storage->user_defined
|
|
|
|
|
&& !$method_storage->has_docblock_param_types
|
|
|
|
|
&& isset($declaring_class_storage->documenting_method_ids[$method_name])
|
|
|
|
|
) {
|
|
|
|
|
$documenting_method_id = $declaring_class_storage->documenting_method_ids[$method_name];
|
|
|
|
|
|
|
|
|
|
$documenting_method_storage = $codebase->methods->getStorage($documenting_method_id);
|
|
|
|
|
|
|
|
|
|
if ($documenting_method_storage->template_types) {
|
|
|
|
|
$method_storage = $documenting_method_storage;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-29 00:50:29 +01:00
|
|
|
|
if (!$context->isSuppressingExceptions($statements_analyzer)) {
|
2019-03-29 00:43:14 +01:00
|
|
|
|
$context->mergeFunctionExceptions($method_storage, $code_location);
|
2018-06-22 07:13:49 +02:00
|
|
|
|
}
|
2017-11-02 20:07:39 +01:00
|
|
|
|
}
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
2021-12-14 00:31:46 +01:00
|
|
|
|
if (ArgumentsAnalyzer::analyze(
|
2020-10-03 02:58:51 +02:00
|
|
|
|
$statements_analyzer,
|
|
|
|
|
$args,
|
|
|
|
|
$method_params,
|
|
|
|
|
(string) $method_id,
|
2021-09-26 23:41:26 +02:00
|
|
|
|
$method_storage->allow_named_arg_calls ?? true,
|
2020-10-03 02:58:51 +02:00
|
|
|
|
$context,
|
2022-01-31 15:36:16 +01:00
|
|
|
|
$template_result
|
2020-10-03 02:58:51 +02:00
|
|
|
|
) === false) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-14 00:31:46 +01:00
|
|
|
|
if (ArgumentsAnalyzer::checkArgumentsMatch(
|
2018-11-11 18:01:14 +01:00
|
|
|
|
$statements_analyzer,
|
2017-02-27 17:07:44 +01:00
|
|
|
|
$args,
|
|
|
|
|
$method_id,
|
|
|
|
|
$method_params,
|
|
|
|
|
$method_storage,
|
|
|
|
|
$class_storage,
|
2022-01-31 15:36:16 +01:00
|
|
|
|
$template_result,
|
2017-02-27 17:07:44 +01:00
|
|
|
|
$code_location,
|
2017-09-03 00:15:52 +02:00
|
|
|
|
$context
|
2017-02-27 17:07:44 +01:00
|
|
|
|
) === false) {
|
|
|
|
|
return false;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2022-01-31 15:36:16 +01:00
|
|
|
|
if ($template_result->template_types) {
|
2020-04-07 06:13:56 +02:00
|
|
|
|
self::checkTemplateResult(
|
|
|
|
|
$statements_analyzer,
|
2022-01-31 15:36:16 +01:00
|
|
|
|
$template_result,
|
2020-04-07 06:13:56 +02:00
|
|
|
|
$code_location,
|
|
|
|
|
strtolower((string) $method_id)
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-18 21:13:27 +02:00
|
|
|
|
return true;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2020-11-30 03:36:50 +01:00
|
|
|
|
* This gets all the template params (and their types) that we think
|
|
|
|
|
* we'll need to know about
|
|
|
|
|
*
|
2021-12-13 16:28:14 +01:00
|
|
|
|
* @return array<string, array<string, Union>>
|
|
|
|
|
* @param array<string, non-empty-array<string, Union>> $existing_template_types
|
|
|
|
|
* @param array<string, array<string, Union>> $class_template_params
|
2020-05-19 04:57:00 +02:00
|
|
|
|
*/
|
|
|
|
|
public static function getTemplateTypesForCall(
|
2021-12-03 20:11:20 +01:00
|
|
|
|
Codebase $codebase,
|
2020-07-05 18:05:25 +02:00
|
|
|
|
?ClassLikeStorage $declaring_class_storage,
|
|
|
|
|
?string $appearing_class_name,
|
|
|
|
|
?ClassLikeStorage $calling_class_storage,
|
2020-11-30 03:36:50 +01:00
|
|
|
|
array $existing_template_types = [],
|
|
|
|
|
array $class_template_params = []
|
2021-12-05 18:51:26 +01:00
|
|
|
|
): array {
|
2020-05-19 04:57:00 +02:00
|
|
|
|
$template_types = $existing_template_types;
|
2020-02-15 02:54:26 +01:00
|
|
|
|
|
2020-05-19 04:57:00 +02:00
|
|
|
|
if ($declaring_class_storage) {
|
|
|
|
|
if ($calling_class_storage
|
|
|
|
|
&& $declaring_class_storage !== $calling_class_storage
|
2020-11-29 21:05:32 +01:00
|
|
|
|
&& $calling_class_storage->template_extended_params
|
2020-05-19 04:57:00 +02:00
|
|
|
|
) {
|
2020-11-29 21:05:32 +01:00
|
|
|
|
foreach ($calling_class_storage->template_extended_params as $class_name => $type_map) {
|
2020-05-19 04:57:00 +02:00
|
|
|
|
foreach ($type_map as $template_name => $type) {
|
2020-11-29 21:05:32 +01:00
|
|
|
|
if ($class_name === $declaring_class_storage->name) {
|
2020-05-19 04:57:00 +02:00
|
|
|
|
$output_type = null;
|
2018-03-19 01:29:41 +01:00
|
|
|
|
|
2020-05-19 04:57:00 +02:00
|
|
|
|
foreach ($type->getAtomicTypes() as $atomic_type) {
|
2021-12-13 04:45:57 +01:00
|
|
|
|
if ($atomic_type instanceof TTemplateParam) {
|
2020-11-11 19:25:17 +01:00
|
|
|
|
$output_type_candidate = self::getGenericParamForOffset(
|
|
|
|
|
$atomic_type->defining_class,
|
|
|
|
|
$atomic_type->param_name,
|
2020-11-29 21:05:32 +01:00
|
|
|
|
$calling_class_storage->template_extended_params,
|
2020-11-30 03:36:50 +01:00
|
|
|
|
$class_template_params + $template_types
|
2020-11-11 19:25:17 +01:00
|
|
|
|
);
|
2018-03-19 01:29:41 +01:00
|
|
|
|
} else {
|
2021-12-13 16:28:14 +01:00
|
|
|
|
$output_type_candidate = new Union([$atomic_type]);
|
2018-03-19 01:29:41 +01:00
|
|
|
|
}
|
2018-03-02 05:43:52 +01:00
|
|
|
|
|
2021-09-25 04:30:19 +02:00
|
|
|
|
$output_type = Type::combineUnionTypes(
|
|
|
|
|
$output_type_candidate,
|
|
|
|
|
$output_type
|
|
|
|
|
);
|
2017-08-12 00:48:58 +02:00
|
|
|
|
}
|
2019-08-06 16:33:21 +02:00
|
|
|
|
|
2020-11-27 17:43:23 +01:00
|
|
|
|
$template_types[$template_name][$declaring_class_storage->name] = $output_type;
|
2019-08-06 16:33:21 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-05-19 04:57:00 +02:00
|
|
|
|
} elseif ($declaring_class_storage->template_types) {
|
|
|
|
|
foreach ($declaring_class_storage->template_types as $template_name => $type_map) {
|
2020-11-27 17:43:23 +01:00
|
|
|
|
foreach ($type_map as $key => $type) {
|
2020-11-30 03:36:50 +01:00
|
|
|
|
$template_types[$template_name][$key]
|
|
|
|
|
= $class_template_params[$template_name][$key] ?? $type;
|
2019-08-14 06:47:57 +02:00
|
|
|
|
}
|
2019-08-06 16:33:21 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-05-21 17:51:41 +02:00
|
|
|
|
}
|
2018-02-08 19:01:39 +01:00
|
|
|
|
|
2020-05-19 04:57:00 +02:00
|
|
|
|
foreach ($template_types as $key => $type_map) {
|
|
|
|
|
foreach ($type_map as $class => $type) {
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$template_types[$key][$class] = TypeExpander::expandUnion(
|
2020-07-05 18:05:25 +02:00
|
|
|
|
$codebase,
|
2020-11-27 17:43:23 +01:00
|
|
|
|
$type,
|
2020-07-05 18:05:25 +02:00
|
|
|
|
$appearing_class_name,
|
2021-09-26 22:57:04 +02:00
|
|
|
|
$calling_class_storage->name ?? null,
|
2020-07-05 18:05:25 +02:00
|
|
|
|
null,
|
|
|
|
|
true,
|
|
|
|
|
false,
|
2021-09-26 22:57:04 +02:00
|
|
|
|
$calling_class_storage->final ?? false
|
2020-07-05 18:05:25 +02:00
|
|
|
|
);
|
2019-05-21 18:11:17 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-19 04:57:00 +02:00
|
|
|
|
return $template_types;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-11-11 19:25:17 +01:00
|
|
|
|
/**
|
2021-12-13 16:28:14 +01:00
|
|
|
|
* @param array<string, array<string, Union>> $template_extended_params
|
|
|
|
|
* @param array<string, array<string, Union>> $found_generic_params
|
2020-11-11 19:25:17 +01:00
|
|
|
|
*/
|
|
|
|
|
public static function getGenericParamForOffset(
|
|
|
|
|
string $fq_class_name,
|
|
|
|
|
string $template_name,
|
2020-11-29 21:05:32 +01:00
|
|
|
|
array $template_extended_params,
|
2021-02-13 22:23:11 +01:00
|
|
|
|
array $found_generic_params
|
2021-12-13 16:28:14 +01:00
|
|
|
|
): Union {
|
2020-11-11 19:25:17 +01:00
|
|
|
|
if (isset($found_generic_params[$template_name][$fq_class_name])) {
|
2020-11-27 17:43:23 +01:00
|
|
|
|
return $found_generic_params[$template_name][$fq_class_name];
|
2020-11-11 19:25:17 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-02-13 22:16:58 +01:00
|
|
|
|
foreach ($template_extended_params as $extended_class_name => $type_map) {
|
2020-11-11 19:25:17 +01:00
|
|
|
|
foreach ($type_map as $extended_template_name => $extended_type) {
|
|
|
|
|
foreach ($extended_type->getAtomicTypes() as $extended_atomic_type) {
|
2021-12-13 04:45:57 +01:00
|
|
|
|
if ($extended_atomic_type instanceof TTemplateParam
|
2020-11-11 19:25:17 +01:00
|
|
|
|
&& $extended_atomic_type->param_name === $template_name
|
2021-02-13 22:16:58 +01:00
|
|
|
|
&& $extended_atomic_type->defining_class === $fq_class_name
|
2020-11-11 19:25:17 +01:00
|
|
|
|
) {
|
|
|
|
|
return self::getGenericParamForOffset(
|
2021-02-13 22:16:58 +01:00
|
|
|
|
$extended_class_name,
|
2020-11-11 19:25:17 +01:00
|
|
|
|
$extended_template_name,
|
2020-11-29 21:05:32 +01:00
|
|
|
|
$template_extended_params,
|
2021-02-13 22:23:11 +01:00
|
|
|
|
$found_generic_params
|
2020-11-11 19:25:17 +01:00
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Type::getMixed();
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-15 01:30:11 +02:00
|
|
|
|
/**
|
2021-09-25 17:14:10 +02:00
|
|
|
|
* @param PhpParser\Node\Scalar\String_|PhpParser\Node\Expr\Array_|PhpParser\Node\Expr\BinaryOp\Concat $callable_arg
|
2017-08-15 01:30:11 +02:00
|
|
|
|
*
|
2020-10-17 18:36:44 +02:00
|
|
|
|
* @return list<non-empty-string>
|
2020-05-15 16:18:05 +02:00
|
|
|
|
*
|
|
|
|
|
* @psalm-suppress LessSpecificReturnStatement
|
|
|
|
|
* @psalm-suppress MoreSpecificReturnType
|
2017-08-15 01:30:11 +02:00
|
|
|
|
*/
|
|
|
|
|
public static function getFunctionIdsFromCallableArg(
|
2021-12-03 20:11:20 +01:00
|
|
|
|
FileSource $file_source,
|
2020-10-12 21:46:47 +02:00
|
|
|
|
PhpParser\Node\Expr $callable_arg
|
2020-09-04 22:26:33 +02:00
|
|
|
|
): array {
|
2018-10-17 19:22:57 +02:00
|
|
|
|
if ($callable_arg instanceof PhpParser\Node\Expr\BinaryOp\Concat) {
|
|
|
|
|
if ($callable_arg->left instanceof PhpParser\Node\Expr\ClassConstFetch
|
2021-12-04 03:37:19 +01:00
|
|
|
|
&& $callable_arg->left->class instanceof Name
|
|
|
|
|
&& $callable_arg->left->name instanceof Identifier
|
2018-10-17 19:22:57 +02:00
|
|
|
|
&& strtolower($callable_arg->left->name->name) === 'class'
|
|
|
|
|
&& !in_array(strtolower($callable_arg->left->class->parts[0]), ['self', 'static', 'parent'])
|
|
|
|
|
&& $callable_arg->right instanceof PhpParser\Node\Scalar\String_
|
|
|
|
|
&& preg_match('/^::[A-Za-z0-9]+$/', $callable_arg->right->value)
|
|
|
|
|
) {
|
2018-10-17 20:37:32 +02:00
|
|
|
|
return [
|
|
|
|
|
(string) $callable_arg->left->class->getAttribute('resolvedName') . $callable_arg->right->value
|
|
|
|
|
];
|
2018-10-17 19:22:57 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-15 01:30:11 +02:00
|
|
|
|
if ($callable_arg instanceof PhpParser\Node\Scalar\String_) {
|
2019-08-12 21:04:43 +02:00
|
|
|
|
$potential_id = preg_replace('/^\\\/', '', $callable_arg->value);
|
|
|
|
|
|
|
|
|
|
if (preg_match('/^[A-Za-z0-9_]+(\\\[A-Za-z0-9_]+)*(::[A-Za-z0-9_]+)?$/', $potential_id)) {
|
|
|
|
|
return [$potential_id];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return [];
|
2017-08-15 01:30:11 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (count($callable_arg->items) !== 2) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-24 22:22:50 +01:00
|
|
|
|
/** @psalm-suppress PossiblyNullPropertyFetch */
|
2020-02-24 22:19:33 +01:00
|
|
|
|
if ($callable_arg->items[0]->key || $callable_arg->items[1]->key) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-14 00:33:32 +01:00
|
|
|
|
if (!isset($callable_arg->items[0]) || !isset($callable_arg->items[1])) {
|
2021-12-03 21:40:18 +01:00
|
|
|
|
throw new UnexpectedValueException('These should never be unset');
|
2018-01-14 00:33:32 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-15 01:30:11 +02:00
|
|
|
|
$class_arg = $callable_arg->items[0]->value;
|
|
|
|
|
$method_name_arg = $callable_arg->items[1]->value;
|
|
|
|
|
|
|
|
|
|
if (!$method_name_arg instanceof PhpParser\Node\Scalar\String_) {
|
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($class_arg instanceof PhpParser\Node\Scalar\String_) {
|
2017-08-30 19:45:41 +02:00
|
|
|
|
return [preg_replace('/^\\\/', '', $class_arg->value) . '::' . $method_name_arg->value];
|
2017-08-15 01:30:11 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($class_arg instanceof PhpParser\Node\Expr\ClassConstFetch
|
2021-12-04 03:37:19 +01:00
|
|
|
|
&& $class_arg->name instanceof Identifier
|
2018-04-17 18:16:25 +02:00
|
|
|
|
&& strtolower($class_arg->name->name) === 'class'
|
2021-12-04 03:37:19 +01:00
|
|
|
|
&& $class_arg->class instanceof Name
|
2017-08-15 01:30:11 +02:00
|
|
|
|
) {
|
2018-11-06 03:57:36 +01:00
|
|
|
|
$fq_class_name = ClassLikeAnalyzer::getFQCLNFromNameObject(
|
2017-08-15 01:30:11 +02:00
|
|
|
|
$class_arg->class,
|
2018-03-02 05:33:21 +01:00
|
|
|
|
$file_source->getAliases()
|
2017-08-15 01:30:11 +02:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return [$fq_class_name . '::' . $method_name_arg->value];
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-25 17:44:54 +01:00
|
|
|
|
if (!$file_source instanceof StatementsAnalyzer
|
|
|
|
|
|| !($class_arg_type = $file_source->node_data->getType($class_arg))
|
|
|
|
|
) {
|
2017-08-15 01:30:11 +02:00
|
|
|
|
return [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$method_ids = [];
|
|
|
|
|
|
2020-01-04 18:20:26 +01:00
|
|
|
|
foreach ($class_arg_type->getAtomicTypes() as $type_part) {
|
2017-08-15 01:30:11 +02:00
|
|
|
|
if ($type_part instanceof TNamedObject) {
|
2018-03-19 01:29:41 +01:00
|
|
|
|
$method_id = $type_part->value . '::' . $method_name_arg->value;
|
|
|
|
|
|
|
|
|
|
if ($type_part->extra_types) {
|
|
|
|
|
foreach ($type_part->extra_types as $extra_type) {
|
2021-12-13 04:45:57 +01:00
|
|
|
|
if ($extra_type instanceof TTemplateParam
|
|
|
|
|
|| $extra_type instanceof TObjectWithProperties
|
2019-06-19 18:00:07 +02:00
|
|
|
|
) {
|
2021-12-03 21:40:18 +01:00
|
|
|
|
throw new UnexpectedValueException('Shouldn’t get a generic param here');
|
2018-11-02 18:08:56 +01:00
|
|
|
|
}
|
2019-06-19 18:00:07 +02:00
|
|
|
|
|
2018-03-19 01:29:41 +01:00
|
|
|
|
$method_id .= '&' . $extra_type->value . '::' . $method_name_arg->value;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-25 19:05:34 +02:00
|
|
|
|
$method_ids[] = '$' . $method_id;
|
2017-08-15 01:30:11 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $method_ids;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-01 19:14:35 +01:00
|
|
|
|
/**
|
2020-05-15 16:18:05 +02:00
|
|
|
|
* @param non-empty-string $function_id
|
2018-02-25 17:13:00 +01:00
|
|
|
|
* @param bool $can_be_in_root_scope if true, the function can be shortened to the root version
|
2017-05-27 02:16:18 +02:00
|
|
|
|
*
|
2016-11-01 19:14:35 +01:00
|
|
|
|
*/
|
2020-05-19 04:57:00 +02:00
|
|
|
|
public static function checkFunctionExists(
|
2018-11-11 18:01:14 +01:00
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
2020-10-12 21:46:47 +02:00
|
|
|
|
string &$function_id,
|
2018-02-25 17:13:00 +01:00
|
|
|
|
CodeLocation $code_location,
|
2020-09-07 01:36:47 +02:00
|
|
|
|
bool $can_be_in_root_scope
|
2020-09-04 22:26:33 +02:00
|
|
|
|
): bool {
|
2016-11-01 19:14:35 +01:00
|
|
|
|
$cased_function_id = $function_id;
|
|
|
|
|
$function_id = strtolower($function_id);
|
|
|
|
|
|
2018-11-11 18:01:14 +01:00
|
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
2018-11-11 18:01:14 +01:00
|
|
|
|
if (!$codebase->functions->functionExists($statements_analyzer, $function_id)) {
|
2020-05-15 16:18:05 +02:00
|
|
|
|
/** @var non-empty-lowercase-string */
|
2017-01-15 18:34:23 +01:00
|
|
|
|
$root_function_id = preg_replace('/.*\\\/', '', $function_id);
|
|
|
|
|
|
2018-02-25 17:13:00 +01:00
|
|
|
|
if ($can_be_in_root_scope
|
|
|
|
|
&& $function_id !== $root_function_id
|
2018-11-11 18:01:14 +01:00
|
|
|
|
&& $codebase->functions->functionExists($statements_analyzer, $root_function_id)
|
2017-01-15 18:34:23 +01:00
|
|
|
|
) {
|
|
|
|
|
$function_id = $root_function_id;
|
|
|
|
|
} else {
|
2021-11-29 20:54:17 +01:00
|
|
|
|
IssueBuffer::maybeAdd(
|
2017-08-15 01:30:11 +02:00
|
|
|
|
new UndefinedFunction(
|
|
|
|
|
'Function ' . $cased_function_id . ' does not exist',
|
2018-11-30 05:19:22 +01:00
|
|
|
|
$code_location,
|
|
|
|
|
$function_id
|
2017-08-15 01:30:11 +02:00
|
|
|
|
),
|
2018-11-11 18:01:14 +01:00
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
|
);
|
2017-08-15 01:30:11 +02:00
|
|
|
|
|
|
|
|
|
return false;
|
2016-11-01 19:14:35 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2018-02-21 17:32:52 +01:00
|
|
|
|
|
2018-02-23 21:39:33 +01:00
|
|
|
|
/**
|
2021-12-04 03:37:19 +01:00
|
|
|
|
* @param Identifier|Name $expr
|
2022-01-20 23:33:06 +01:00
|
|
|
|
* @param Possibilities[] $var_assertions
|
2020-10-28 17:45:26 +01:00
|
|
|
|
* @param list<PhpParser\Node\Arg> $args
|
2018-02-23 21:39:33 +01:00
|
|
|
|
*
|
|
|
|
|
*/
|
2020-11-09 06:58:45 +01:00
|
|
|
|
public static function applyAssertionsToContext(
|
2020-10-12 21:46:47 +02:00
|
|
|
|
PhpParser\NodeAbstract $expr,
|
2019-09-06 03:00:02 +02:00
|
|
|
|
?string $thisName,
|
2022-01-20 23:33:06 +01:00
|
|
|
|
array $var_assertions,
|
2018-02-23 21:39:33 +01:00
|
|
|
|
array $args,
|
2022-01-20 23:33:06 +01:00
|
|
|
|
TemplateResult $template_result,
|
2018-02-23 21:39:33 +01:00
|
|
|
|
Context $context,
|
2018-11-11 18:01:14 +01:00
|
|
|
|
StatementsAnalyzer $statements_analyzer
|
2020-09-12 17:24:05 +02:00
|
|
|
|
): void {
|
2018-02-23 21:39:33 +01:00
|
|
|
|
$type_assertions = [];
|
|
|
|
|
|
2018-11-14 19:12:31 +01:00
|
|
|
|
$asserted_keys = [];
|
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
|
foreach ($var_assertions as $var_possibilities) {
|
2018-10-30 14:20:34 +01:00
|
|
|
|
$assertion_var_id = null;
|
|
|
|
|
|
2019-01-05 20:50:11 +01:00
|
|
|
|
$arg_value = null;
|
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
|
if (is_int($var_possibilities->var_id)) {
|
|
|
|
|
if (!isset($args[$var_possibilities->var_id])) {
|
2018-02-23 21:39:33 +01:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
|
$arg_value = $args[$var_possibilities->var_id]->value;
|
2018-02-23 21:39:33 +01:00
|
|
|
|
|
2022-02-04 18:49:12 +01:00
|
|
|
|
$arg_var_id = ExpressionIdentifier::getExtendedVarId($arg_value, null, $statements_analyzer);
|
2018-02-23 21:39:33 +01:00
|
|
|
|
|
|
|
|
|
if ($arg_var_id) {
|
2018-10-30 14:20:34 +01:00
|
|
|
|
$assertion_var_id = $arg_var_id;
|
2018-02-23 21:39:33 +01:00
|
|
|
|
}
|
2022-01-20 23:33:06 +01:00
|
|
|
|
} elseif ($var_possibilities->var_id === '$this' && $thisName !== null) {
|
2020-04-09 14:15:07 +02:00
|
|
|
|
$assertion_var_id = $thisName;
|
2022-01-20 23:33:06 +01:00
|
|
|
|
} elseif (strpos($var_possibilities->var_id, '$this->') === 0 && $thisName !== null) {
|
|
|
|
|
$assertion_var_id = $thisName . str_replace('$this->', '->', $var_possibilities->var_id);
|
|
|
|
|
} elseif (strpos($var_possibilities->var_id, 'self::') === 0 && $context->self) {
|
|
|
|
|
$assertion_var_id = $context->self . str_replace('self::', '::', $var_possibilities->var_id);
|
|
|
|
|
} elseif (strpos($var_possibilities->var_id, '::$') !== false) {
|
2020-12-21 18:05:44 +01:00
|
|
|
|
// allow assertions to bring external static props into scope
|
2022-01-20 23:33:06 +01:00
|
|
|
|
$assertion_var_id = $var_possibilities->var_id;
|
|
|
|
|
} elseif (isset($context->vars_in_scope[$var_possibilities->var_id])) {
|
|
|
|
|
$assertion_var_id = $var_possibilities->var_id;
|
|
|
|
|
} elseif (strpos($var_possibilities->var_id, '->') !== false) {
|
|
|
|
|
$exploded = explode('->', $var_possibilities->var_id);
|
2021-10-17 10:29:25 +02:00
|
|
|
|
|
|
|
|
|
if (count($exploded) < 2) {
|
|
|
|
|
IssueBuffer::add(
|
|
|
|
|
new InvalidDocblock(
|
|
|
|
|
'Assert notation is malformed',
|
|
|
|
|
new CodeLocation($statements_analyzer, $expr)
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[$var_id, $property] = $exploded;
|
|
|
|
|
|
|
|
|
|
$var_id = is_numeric($var_id) ? (int) $var_id : $var_id;
|
|
|
|
|
|
|
|
|
|
if (!is_int($var_id) || !isset($args[$var_id])) {
|
|
|
|
|
IssueBuffer::add(
|
|
|
|
|
new InvalidDocblock(
|
|
|
|
|
'Variable ' . $var_id . ' is not an argument so cannot be asserted',
|
|
|
|
|
new CodeLocation($statements_analyzer, $expr)
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @var PhpParser\Node\Expr\Variable $arg_value */
|
|
|
|
|
$arg_value = $args[$var_id]->value;
|
|
|
|
|
|
2022-02-04 18:49:12 +01:00
|
|
|
|
$arg_var_id = ExpressionIdentifier::getExtendedVarId($arg_value, null, $statements_analyzer);
|
2021-10-17 10:29:25 +02:00
|
|
|
|
|
|
|
|
|
if (!$arg_var_id) {
|
|
|
|
|
IssueBuffer::add(
|
|
|
|
|
new InvalidDocblock(
|
|
|
|
|
'Variable being asserted as argument ' . ($var_id+1) . ' cannot be found in local scope',
|
|
|
|
|
new CodeLocation($statements_analyzer, $expr)
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (count($exploded) === 2) {
|
|
|
|
|
$failedMessage = AssertionFinder::isPropertyImmutableOnArgument(
|
|
|
|
|
$property,
|
|
|
|
|
$statements_analyzer->getNodeTypeProvider(),
|
|
|
|
|
$statements_analyzer->getCodebase()->classlike_storage_provider,
|
|
|
|
|
$arg_value
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (null !== $failedMessage) {
|
|
|
|
|
IssueBuffer::add(
|
|
|
|
|
new InvalidDocblock($failedMessage, new CodeLocation($statements_analyzer, $expr))
|
|
|
|
|
);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
|
$assertion_var_id = str_replace((string) $var_id, $arg_var_id, $var_possibilities->var_id);
|
2018-10-30 14:20:34 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-11 18:03:21 +02:00
|
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
|
|
2018-10-30 14:20:34 +01:00
|
|
|
|
if ($assertion_var_id) {
|
2022-01-20 23:33:06 +01:00
|
|
|
|
$orred_rules = [];
|
2018-10-30 14:20:34 +01:00
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
|
foreach ($var_possibilities->rule as $assertion_rule) {
|
|
|
|
|
$assertion_type = $assertion_rule->getAtomicType();
|
2018-11-16 00:50:08 +01:00
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
|
if ($assertion_type) {
|
|
|
|
|
$union = new Union([clone $assertion_type]);
|
|
|
|
|
TemplateInferredTypeReplacer::replace(
|
|
|
|
|
$union,
|
|
|
|
|
$template_result,
|
2021-07-11 18:03:21 +02:00
|
|
|
|
$codebase
|
|
|
|
|
);
|
|
|
|
|
|
2022-06-12 01:15:46 +02:00
|
|
|
|
if ($union->isSingle()) {
|
|
|
|
|
$atomic_type = $union->getSingleAtomic();
|
|
|
|
|
if (!$assertion_type instanceof TTemplateParam
|
|
|
|
|
|| $assertion_type->as->getId() !== $atomic_type->getId()
|
|
|
|
|
) {
|
2022-01-20 23:33:06 +01:00
|
|
|
|
$assertion_rule = clone $assertion_rule;
|
|
|
|
|
$assertion_rule->setAtomicType($atomic_type);
|
|
|
|
|
$orred_rules[] = $assertion_rule;
|
2019-07-21 07:40:19 +02:00
|
|
|
|
}
|
2022-01-20 23:33:06 +01:00
|
|
|
|
} elseif (isset($context->vars_in_scope[$var_possibilities->var_id])) {
|
|
|
|
|
$other_type = $context->vars_in_scope[$var_possibilities->var_id];
|
2019-07-21 07:40:19 +02:00
|
|
|
|
|
2022-01-20 23:33:06 +01:00
|
|
|
|
if ($assertion_rule instanceof IsIdentical
|
|
|
|
|
|| $assertion_rule instanceof IsType
|
2019-07-21 07:40:19 +02:00
|
|
|
|
) {
|
2022-01-20 23:33:06 +01:00
|
|
|
|
if (!UnionTypeComparator::canExpressionTypesBeIdentical(
|
|
|
|
|
$codebase,
|
|
|
|
|
$union,
|
|
|
|
|
$context->vars_in_scope[$var_possibilities->var_id]
|
|
|
|
|
)) {
|
|
|
|
|
IssueBuffer::maybeAdd(
|
|
|
|
|
new TypeDoesNotContainType(
|
|
|
|
|
$union->getId() . ' cannot be identical to ' . $other_type->getId(),
|
|
|
|
|
new CodeLocation($statements_analyzer->getSource(), $expr),
|
|
|
|
|
$union->getId() . ' ' . $other_type->getId()
|
|
|
|
|
),
|
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
|
|
|
|
);
|
|
|
|
|
}
|
2019-07-21 07:40:19 +02:00
|
|
|
|
}
|
2022-06-12 01:33:13 +02:00
|
|
|
|
} elseif (isset($context->vars_in_scope[$assertion_var_id])) {
|
|
|
|
|
$other_type = $context->vars_in_scope[$assertion_var_id];
|
2022-06-12 01:15:46 +02:00
|
|
|
|
if (self::isNewTypeNarrowingDownOldType($other_type, $union)) {
|
2022-06-12 01:29:30 +02:00
|
|
|
|
foreach ($union->getAtomicTypes() as $atomic_type) {
|
2022-06-12 01:15:46 +02:00
|
|
|
|
if ($assertion_type instanceof TTemplateParam
|
|
|
|
|
&& $assertion_type->as->getId() === $atomic_type->getId()
|
|
|
|
|
) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$assertion_rule = clone $assertion_rule;
|
|
|
|
|
$assertion_rule->setAtomicType($atomic_type);
|
|
|
|
|
$orred_rules[] = $assertion_rule;
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-10-30 14:20:34 +01:00
|
|
|
|
}
|
2022-01-20 23:33:06 +01:00
|
|
|
|
} else {
|
|
|
|
|
$orred_rules[] = $assertion_rule;
|
2018-11-16 17:50:07 +01:00
|
|
|
|
}
|
2022-01-20 23:33:06 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($orred_rules) {
|
2019-12-09 23:21:58 +01:00
|
|
|
|
if (isset($type_assertions[$assertion_var_id])) {
|
|
|
|
|
$type_assertions[$assertion_var_id] = array_merge(
|
|
|
|
|
$type_assertions[$assertion_var_id],
|
2022-01-20 23:33:06 +01:00
|
|
|
|
[$orred_rules]
|
2019-12-09 23:21:58 +01:00
|
|
|
|
);
|
|
|
|
|
} else {
|
2022-01-20 23:33:06 +01:00
|
|
|
|
$type_assertions[$assertion_var_id] = [$orred_rules];
|
2019-12-09 23:21:58 +01:00
|
|
|
|
}
|
2018-10-30 14:20:34 +01:00
|
|
|
|
}
|
2022-01-20 23:33:06 +01:00
|
|
|
|
} elseif ($arg_value
|
|
|
|
|
&& count($var_possibilities->rule) === 1
|
|
|
|
|
) {
|
|
|
|
|
$assert_clauses = [];
|
|
|
|
|
|
|
|
|
|
$single_rule = $var_possibilities->rule[0];
|
|
|
|
|
|
|
|
|
|
if ($single_rule instanceof Truthy) {
|
|
|
|
|
$assert_clauses = FormulaGenerator::getFormula(
|
|
|
|
|
spl_object_id($arg_value),
|
|
|
|
|
spl_object_id($arg_value),
|
|
|
|
|
$arg_value,
|
|
|
|
|
$context->self,
|
|
|
|
|
$statements_analyzer,
|
|
|
|
|
$statements_analyzer->getCodebase()
|
|
|
|
|
);
|
|
|
|
|
} elseif ($single_rule instanceof Falsy) {
|
|
|
|
|
$assert_clauses = Algebra::negateFormula(
|
|
|
|
|
FormulaGenerator::getFormula(
|
|
|
|
|
spl_object_id($arg_value),
|
|
|
|
|
spl_object_id($arg_value),
|
|
|
|
|
$arg_value,
|
|
|
|
|
$context->self,
|
|
|
|
|
$statements_analyzer,
|
|
|
|
|
$codebase
|
|
|
|
|
)
|
|
|
|
|
);
|
|
|
|
|
} elseif ($single_rule instanceof IsType
|
|
|
|
|
&& $single_rule->type instanceof TTrue
|
|
|
|
|
) {
|
2021-02-15 22:18:41 +01:00
|
|
|
|
$conditional = new VirtualIdentical(
|
2019-12-08 22:35:56 +01:00
|
|
|
|
$arg_value,
|
2021-02-15 22:18:41 +01:00
|
|
|
|
new VirtualConstFetch(new VirtualName('true'))
|
2019-12-08 22:35:56 +01:00
|
|
|
|
);
|
|
|
|
|
|
2020-11-03 22:15:44 +01:00
|
|
|
|
$assert_clauses = FormulaGenerator::getFormula(
|
2022-01-05 00:00:05 +01:00
|
|
|
|
mt_rand(0, 1_000_000),
|
|
|
|
|
mt_rand(0, 1_000_000),
|
2019-12-08 22:35:56 +01:00
|
|
|
|
$conditional,
|
2019-12-28 21:56:19 +01:00
|
|
|
|
$context->self,
|
2019-08-29 17:53:36 +02:00
|
|
|
|
$statements_analyzer,
|
2021-07-11 18:03:21 +02:00
|
|
|
|
$codebase
|
2019-08-29 17:53:36 +02:00
|
|
|
|
);
|
|
|
|
|
}
|
2019-01-05 20:50:11 +01:00
|
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$simplified_clauses = Algebra::simplifyCNF(
|
2019-01-05 20:50:11 +01:00
|
|
|
|
array_merge($context->clauses, $assert_clauses)
|
|
|
|
|
);
|
|
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$assert_type_assertions = Algebra::getTruthsFromFormula(
|
2019-01-05 20:50:11 +01:00
|
|
|
|
$simplified_clauses
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$type_assertions = array_merge($type_assertions, $assert_type_assertions);
|
2018-02-23 21:39:33 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-07 07:23:35 +01:00
|
|
|
|
$changed_var_ids = [];
|
2018-02-23 21:39:33 +01:00
|
|
|
|
|
2018-11-14 18:25:17 +01:00
|
|
|
|
foreach ($type_assertions as $var_id => $_) {
|
|
|
|
|
$asserted_keys[$var_id] = true;
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-11 18:03:21 +02:00
|
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
|
|
2018-12-17 21:23:56 +01:00
|
|
|
|
if ($type_assertions) {
|
2022-01-20 23:33:06 +01:00
|
|
|
|
$template_type_map = [];
|
2020-05-31 01:11:41 +02:00
|
|
|
|
|
2018-12-17 21:23:56 +01:00
|
|
|
|
// while in an and, we allow scope to boil over to support
|
|
|
|
|
// statements of the form if ($x && $x->foo())
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$op_vars_in_scope = Reconciler::reconcileKeyedTypes(
|
2019-12-08 22:35:56 +01:00
|
|
|
|
$type_assertions,
|
2018-12-17 21:23:56 +01:00
|
|
|
|
$type_assertions,
|
|
|
|
|
$context->vars_in_scope,
|
2022-01-11 00:45:29 +01:00
|
|
|
|
$context->references_in_scope,
|
2019-12-07 07:23:35 +01:00
|
|
|
|
$changed_var_ids,
|
2018-12-17 21:23:56 +01:00
|
|
|
|
$asserted_keys,
|
|
|
|
|
$statements_analyzer,
|
2020-05-31 01:11:41 +02:00
|
|
|
|
$template_type_map,
|
2018-12-17 21:23:56 +01:00
|
|
|
|
$context->inside_loop,
|
|
|
|
|
new CodeLocation($statements_analyzer->getSource(), $expr)
|
|
|
|
|
);
|
2018-02-23 21:39:33 +01:00
|
|
|
|
|
2019-12-07 07:23:35 +01:00
|
|
|
|
foreach ($changed_var_ids as $var_id => $_) {
|
|
|
|
|
if (isset($op_vars_in_scope[$var_id])) {
|
2020-05-02 20:55:21 +02:00
|
|
|
|
$first_appearance = $statements_analyzer->getFirstAppearance($var_id);
|
|
|
|
|
|
2020-05-11 03:09:48 +02:00
|
|
|
|
if ($first_appearance
|
|
|
|
|
&& isset($context->vars_in_scope[$var_id])
|
|
|
|
|
&& $context->vars_in_scope[$var_id]->hasMixed()
|
|
|
|
|
) {
|
2020-05-02 20:55:21 +02:00
|
|
|
|
if (!$context->collect_initializations
|
|
|
|
|
&& !$context->collect_mutations
|
|
|
|
|
&& $statements_analyzer->getFilePath() === $statements_analyzer->getRootFilePath()
|
|
|
|
|
&& (!(($parent_source = $statements_analyzer->getSource())
|
2021-12-03 20:11:20 +01:00
|
|
|
|
instanceof FunctionLikeAnalyzer)
|
|
|
|
|
|| !$parent_source->getSource() instanceof TraitAnalyzer)
|
2020-05-02 20:55:21 +02:00
|
|
|
|
) {
|
|
|
|
|
$codebase->analyzer->decrementMixedCount($statements_analyzer->getFilePath());
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-09 05:01:45 +01:00
|
|
|
|
IssueBuffer::remove(
|
|
|
|
|
$statements_analyzer->getFilePath(),
|
|
|
|
|
'MixedAssignment',
|
|
|
|
|
$first_appearance->raw_file_start
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-07 07:23:35 +01:00
|
|
|
|
$op_vars_in_scope[$var_id]->from_docblock = true;
|
2019-05-16 00:41:26 +02:00
|
|
|
|
|
2020-01-04 18:20:26 +01:00
|
|
|
|
foreach ($op_vars_in_scope[$var_id]->getAtomicTypes() as $changed_atomic_type) {
|
2019-05-16 00:41:26 +02:00
|
|
|
|
$changed_atomic_type->from_docblock = true;
|
|
|
|
|
|
2021-12-13 04:45:57 +01:00
|
|
|
|
if ($changed_atomic_type instanceof TNamedObject
|
2019-05-16 00:41:26 +02:00
|
|
|
|
&& $changed_atomic_type->extra_types
|
|
|
|
|
) {
|
|
|
|
|
foreach ($changed_atomic_type->extra_types as $extra_type) {
|
|
|
|
|
$extra_type->from_docblock = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-12-17 21:23:56 +01:00
|
|
|
|
}
|
2018-02-23 21:39:33 +01:00
|
|
|
|
}
|
|
|
|
|
|
2018-12-17 21:23:56 +01:00
|
|
|
|
$context->vars_in_scope = $op_vars_in_scope;
|
|
|
|
|
}
|
2018-02-23 21:39:33 +01:00
|
|
|
|
}
|
2020-04-07 06:13:56 +02:00
|
|
|
|
|
2021-07-11 18:03:21 +02:00
|
|
|
|
/**
|
|
|
|
|
* This method looks for problems with a generated TemplateResult.
|
|
|
|
|
*
|
|
|
|
|
* The TemplateResult object contains upper bounds and lower bounds for each template param.
|
|
|
|
|
*
|
|
|
|
|
* Those upper bounds represent a series of constraints like
|
|
|
|
|
*
|
|
|
|
|
* Lower bound:
|
|
|
|
|
* T >: X (the type param T matches X, or is a supertype of X)
|
|
|
|
|
* Upper bound:
|
|
|
|
|
* T <: Y (the type param T matches Y, or is a subtype of Y)
|
|
|
|
|
* Equality (currently represented as an upper bound with a special flag)
|
|
|
|
|
* T = Z (the template T must match Z)
|
|
|
|
|
*
|
|
|
|
|
* This method attempts to reconcile those constraints.
|
|
|
|
|
*
|
|
|
|
|
* Valid constraints:
|
|
|
|
|
*
|
|
|
|
|
* T <: int|float, T >: int --- implies T is an int
|
|
|
|
|
* T = int --- implies T is an int
|
|
|
|
|
*
|
|
|
|
|
* Invalid constraints:
|
|
|
|
|
*
|
|
|
|
|
* T <: int|string, T >: string|float --- implies T <: int and T >: float, which is impossible
|
|
|
|
|
* T = int, T = string --- implies T is a string _and_ and int, which is impossible
|
|
|
|
|
*/
|
2020-04-07 06:13:56 +02:00
|
|
|
|
public static function checkTemplateResult(
|
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
|
TemplateResult $template_result,
|
|
|
|
|
CodeLocation $code_location,
|
|
|
|
|
?string $function_id
|
2021-12-05 18:51:26 +01:00
|
|
|
|
): void {
|
2021-07-10 20:08:09 +02:00
|
|
|
|
if ($template_result->lower_bounds && $template_result->upper_bounds) {
|
|
|
|
|
foreach ($template_result->upper_bounds as $template_name => $defining_map) {
|
|
|
|
|
foreach ($defining_map as $defining_id => $upper_bound) {
|
|
|
|
|
if (isset($template_result->lower_bounds[$template_name][$defining_id])) {
|
2021-07-11 18:03:21 +02:00
|
|
|
|
$lower_bound_type = TemplateStandinTypeReplacer::getMostSpecificTypeFromBounds(
|
|
|
|
|
$template_result->lower_bounds[$template_name][$defining_id],
|
|
|
|
|
$statements_analyzer->getCodebase()
|
|
|
|
|
);
|
|
|
|
|
|
2021-07-10 20:08:09 +02:00
|
|
|
|
$upper_bound_type = $upper_bound->type;
|
2020-04-07 06:13:56 +02:00
|
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$union_comparison_result = new TypeComparisonResult();
|
2020-04-08 18:37:38 +02:00
|
|
|
|
|
2021-07-10 20:08:09 +02:00
|
|
|
|
if (count($template_result->upper_bounds_unintersectable_types) > 1) {
|
|
|
|
|
[$lower_bound_type, $upper_bound_type]
|
|
|
|
|
= $template_result->upper_bounds_unintersectable_types;
|
2020-04-29 20:57:57 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-07-22 01:40:35 +02:00
|
|
|
|
if (!UnionTypeComparator::isContainedBy(
|
2020-04-07 06:13:56 +02:00
|
|
|
|
$statements_analyzer->getCodebase(),
|
2020-11-27 17:47:12 +01:00
|
|
|
|
$lower_bound_type,
|
2021-07-10 20:08:09 +02:00
|
|
|
|
$upper_bound_type,
|
2020-04-08 18:37:38 +02:00
|
|
|
|
false,
|
|
|
|
|
false,
|
|
|
|
|
$union_comparison_result
|
2020-04-07 06:13:56 +02:00
|
|
|
|
)) {
|
2020-04-08 18:37:38 +02:00
|
|
|
|
if ($union_comparison_result->type_coerced) {
|
|
|
|
|
if ($union_comparison_result->type_coerced_from_mixed) {
|
2021-11-29 20:54:17 +01:00
|
|
|
|
IssueBuffer::maybeAdd(
|
2020-04-08 18:37:38 +02:00
|
|
|
|
new MixedArgumentTypeCoercion(
|
2021-07-10 20:08:09 +02:00
|
|
|
|
'Type ' . $lower_bound_type->getId() . ' should be a subtype of '
|
|
|
|
|
. $upper_bound_type->getId(),
|
2020-04-08 18:37:38 +02:00
|
|
|
|
$code_location,
|
|
|
|
|
$function_id
|
|
|
|
|
),
|
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
|
);
|
2020-04-08 18:37:38 +02:00
|
|
|
|
} else {
|
2021-11-29 20:54:17 +01:00
|
|
|
|
IssueBuffer::maybeAdd(
|
2020-04-08 18:37:38 +02:00
|
|
|
|
new ArgumentTypeCoercion(
|
2021-07-10 20:08:09 +02:00
|
|
|
|
'Type ' . $lower_bound_type->getId() . ' should be a subtype of '
|
|
|
|
|
. $upper_bound_type->getId(),
|
2020-04-08 18:37:38 +02:00
|
|
|
|
$code_location,
|
|
|
|
|
$function_id
|
|
|
|
|
),
|
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
|
);
|
2020-04-08 18:37:38 +02:00
|
|
|
|
}
|
|
|
|
|
} elseif ($union_comparison_result->scalar_type_match_found) {
|
2021-11-29 20:54:17 +01:00
|
|
|
|
IssueBuffer::maybeAdd(
|
2020-04-08 18:37:38 +02:00
|
|
|
|
new InvalidScalarArgument(
|
2021-07-10 20:08:09 +02:00
|
|
|
|
'Type ' . $lower_bound_type->getId() . ' should be a subtype of '
|
|
|
|
|
. $upper_bound_type->getId(),
|
2020-04-08 18:37:38 +02:00
|
|
|
|
$code_location,
|
|
|
|
|
$function_id
|
|
|
|
|
),
|
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
|
);
|
2020-04-08 18:37:38 +02:00
|
|
|
|
} else {
|
2021-11-29 20:54:17 +01:00
|
|
|
|
IssueBuffer::maybeAdd(
|
2020-04-08 18:37:38 +02:00
|
|
|
|
new InvalidArgument(
|
2021-07-10 20:08:09 +02:00
|
|
|
|
'Type ' . $lower_bound_type->getId() . ' should be a subtype of '
|
|
|
|
|
. $upper_bound_type->getId(),
|
2020-04-08 18:37:38 +02:00
|
|
|
|
$code_location,
|
|
|
|
|
$function_id
|
|
|
|
|
),
|
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
|
);
|
2020-04-07 06:13:56 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2021-07-11 18:03:21 +02:00
|
|
|
|
$template_result->lower_bounds[$template_name][$defining_id] = [
|
|
|
|
|
new TemplateBound(
|
|
|
|
|
clone $upper_bound->type
|
|
|
|
|
)
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Attempt to identify invalid lower bounds
|
|
|
|
|
foreach ($template_result->lower_bounds as $template_name => $lower_bounds) {
|
|
|
|
|
foreach ($lower_bounds as $lower_bounds) {
|
|
|
|
|
if (count($lower_bounds) > 1) {
|
|
|
|
|
$bounds_with_equality = array_filter(
|
|
|
|
|
$lower_bounds,
|
2022-04-09 11:58:26 +02:00
|
|
|
|
static fn($lower_bound): bool => (bool)$lower_bound->equality_bound_classlike
|
2021-07-11 18:03:21 +02:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (!$bounds_with_equality) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-01 01:50:56 +02:00
|
|
|
|
$equality_types = array_unique(
|
|
|
|
|
array_map(
|
2022-04-09 11:58:26 +02:00
|
|
|
|
static fn($bound_with_equality) => $bound_with_equality->type->getId(),
|
2021-08-01 01:50:56 +02:00
|
|
|
|
$bounds_with_equality
|
|
|
|
|
)
|
|
|
|
|
);
|
2021-07-11 18:03:21 +02:00
|
|
|
|
|
2021-08-01 01:50:56 +02:00
|
|
|
|
if (count($equality_types) > 1) {
|
2021-11-29 20:54:17 +01:00
|
|
|
|
IssueBuffer::maybeAdd(
|
2021-07-11 18:03:21 +02:00
|
|
|
|
new InvalidArgument(
|
2021-10-09 15:02:49 +02:00
|
|
|
|
'Incompatible types found for ' . $template_name . ' (must have only one of ' .
|
2021-10-09 15:01:13 +02:00
|
|
|
|
implode(', ', $equality_types) . ')',
|
2021-07-11 18:03:21 +02:00
|
|
|
|
$code_location,
|
|
|
|
|
$function_id
|
|
|
|
|
),
|
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
2021-11-29 20:54:17 +01:00
|
|
|
|
);
|
2021-07-11 18:03:21 +02:00
|
|
|
|
} else {
|
|
|
|
|
foreach ($lower_bounds as $lower_bound) {
|
|
|
|
|
if ($lower_bound->equality_bound_classlike === null) {
|
2022-04-27 07:47:02 +02:00
|
|
|
|
foreach ($bounds_with_equality as $bound_with_equality) {
|
|
|
|
|
if (UnionTypeComparator::isContainedBy(
|
|
|
|
|
$statements_analyzer->getCodebase(),
|
|
|
|
|
$lower_bound->type,
|
|
|
|
|
$bound_with_equality->type
|
|
|
|
|
) && UnionTypeComparator::isContainedBy(
|
|
|
|
|
$statements_analyzer->getCodebase(),
|
|
|
|
|
$bound_with_equality->type,
|
|
|
|
|
$lower_bound->type
|
|
|
|
|
)) {
|
|
|
|
|
continue 2;
|
|
|
|
|
}
|
2021-07-11 18:03:21 +02:00
|
|
|
|
}
|
2022-04-27 07:47:02 +02:00
|
|
|
|
|
|
|
|
|
IssueBuffer::maybeAdd(
|
|
|
|
|
new InvalidArgument(
|
|
|
|
|
'Incompatible types found for ' . $template_name . ' (' .
|
|
|
|
|
$lower_bound->type->getId() . ' is not in ' .
|
|
|
|
|
implode(', ', $equality_types) . ')',
|
|
|
|
|
$code_location,
|
|
|
|
|
$function_id
|
|
|
|
|
),
|
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
|
|
|
|
);
|
2021-07-11 18:03:21 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-04-07 06:13:56 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-06-12 00:32:36 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* This method should detect if the new type narrows down the old type.
|
|
|
|
|
*/
|
|
|
|
|
private static function isNewTypeNarrowingDownOldType(Union $old_type, Union $new_type): bool
|
|
|
|
|
{
|
2022-06-12 03:35:59 +02:00
|
|
|
|
if ($new_type->isSingle()) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-12 02:51:36 +02:00
|
|
|
|
// non-mixed is always better than mixed
|
2022-06-12 01:15:46 +02:00
|
|
|
|
if ($old_type->isMixed() && !$new_type->hasMixed()) {
|
2022-06-12 00:32:36 +02:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-12 02:51:36 +02:00
|
|
|
|
// non-nullable is always better than nullable
|
|
|
|
|
if ($old_type->isNullable() && !$new_type->isNullable()) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-12 03:35:59 +02:00
|
|
|
|
// Do not hassle around with non-single old types if they are not nullable
|
|
|
|
|
if (!$old_type->isSingle()) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$old_atomic_type = $old_type->getSingleAtomic();
|
|
|
|
|
foreach ($new_type->getAtomicTypes() as $new_atomic_type) {
|
|
|
|
|
if ($new_atomic_type->equals($old_atomic_type, false)) {
|
|
|
|
|
// Old type is one of the new types and thus, the old type must not be modified
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Literals should always replace non-literals
|
|
|
|
|
if (!$old_type->containsAnyLiteral() &&
|
|
|
|
|
(
|
|
|
|
|
($old_type->isString() && $new_type->allStringLiterals())
|
|
|
|
|
|| ($old_type->isInt() && $new_type->allIntLiterals())
|
|
|
|
|
|| ($old_type->isFloat() && $new_type->allFloatLiterals())
|
|
|
|
|
)
|
|
|
|
|
) {
|
2022-06-12 02:51:36 +02:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
2022-06-12 00:32:36 +02:00
|
|
|
|
}
|
2016-11-01 19:14:35 +01:00
|
|
|
|
}
|