2020-03-12 04:15:15 +01:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression\Call;
|
|
|
|
|
|
|
|
use Psalm\Codebase;
|
2020-05-30 22:55:18 +02:00
|
|
|
use Psalm\Internal\Type\TypeExpander;
|
2020-03-12 04:15:15 +01:00
|
|
|
use Psalm\Storage\ClassLikeStorage;
|
|
|
|
use Psalm\Type;
|
2020-05-30 22:55:18 +02:00
|
|
|
use Psalm\Type\Atomic;
|
2020-03-12 04:15:15 +01:00
|
|
|
use Psalm\Type\Atomic\TGenericObject;
|
2020-05-30 22:55:18 +02:00
|
|
|
use Psalm\Type\Atomic\TScalarClassConstant;
|
|
|
|
|
2020-03-12 04:15:15 +01:00
|
|
|
use function array_merge;
|
|
|
|
use function array_search;
|
|
|
|
use function array_keys;
|
|
|
|
|
|
|
|
class ClassTemplateParamCollector
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @param lowercase-string $method_name
|
|
|
|
* @return array<string, array<string, array{Type\Union, 1?:int}>>|null
|
|
|
|
*/
|
|
|
|
public static function collect(
|
|
|
|
Codebase $codebase,
|
|
|
|
ClassLikeStorage $class_storage,
|
2020-05-27 05:29:37 +02:00
|
|
|
ClassLikeStorage $static_class_storage,
|
2020-03-12 04:15:15 +01:00
|
|
|
string $method_name = null,
|
|
|
|
Type\Atomic $lhs_type_part = null,
|
|
|
|
string $lhs_var_id = null
|
2020-09-04 22:26:33 +02:00
|
|
|
): ?array {
|
2020-05-27 05:29:37 +02:00
|
|
|
$static_fq_class_name = $static_class_storage->name;
|
2020-03-12 04:15:15 +01:00
|
|
|
|
|
|
|
$non_trait_class_storage = $class_storage->is_trait
|
|
|
|
? $static_class_storage
|
|
|
|
: $class_storage;
|
|
|
|
|
|
|
|
$template_types = $class_storage->template_types;
|
|
|
|
|
|
|
|
$candidate_class_storages = [$class_storage];
|
|
|
|
|
|
|
|
if ($static_class_storage->template_type_extends
|
|
|
|
&& $method_name
|
|
|
|
&& !empty($non_trait_class_storage->overridden_method_ids[$method_name])
|
|
|
|
&& isset($class_storage->methods[$method_name])
|
|
|
|
&& (!isset($non_trait_class_storage->methods[$method_name]->return_type)
|
|
|
|
|| $class_storage->methods[$method_name]->inherited_return_type)
|
|
|
|
) {
|
|
|
|
foreach ($non_trait_class_storage->overridden_method_ids[$method_name] as $overridden_method_id) {
|
|
|
|
$overridden_storage = $codebase->methods->getStorage($overridden_method_id);
|
|
|
|
|
|
|
|
if (!$overridden_storage->return_type) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($overridden_storage->return_type->isNull()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$fq_overridden_class = $overridden_method_id->fq_class_name;
|
|
|
|
|
|
|
|
$overridden_class_storage = $codebase->classlike_storage_provider->get($fq_overridden_class);
|
|
|
|
|
|
|
|
$overridden_template_types = $overridden_class_storage->template_types;
|
|
|
|
|
|
|
|
if (!$template_types) {
|
|
|
|
$template_types = $overridden_template_types;
|
|
|
|
} elseif ($overridden_template_types) {
|
|
|
|
foreach ($overridden_template_types as $template_name => $template_map) {
|
|
|
|
if (isset($template_types[$template_name])) {
|
|
|
|
$template_types[$template_name] = array_merge(
|
|
|
|
$template_types[$template_name],
|
|
|
|
$template_map
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$template_types[$template_name] = $template_map;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$candidate_class_storages[] = $overridden_class_storage;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$template_types) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$class_template_params = [];
|
|
|
|
$e = $static_class_storage->template_type_extends;
|
|
|
|
|
|
|
|
if ($lhs_type_part instanceof TGenericObject) {
|
|
|
|
if ($class_storage === $static_class_storage && $static_class_storage->template_types) {
|
|
|
|
$i = 0;
|
|
|
|
|
|
|
|
foreach ($static_class_storage->template_types as $type_name => $_) {
|
|
|
|
if (isset($lhs_type_part->type_params[$i])) {
|
|
|
|
if ($lhs_var_id !== '$this' || $static_fq_class_name !== $static_class_storage->name) {
|
|
|
|
$class_template_params[$type_name][$static_class_storage->name] = [
|
|
|
|
$lhs_type_part->type_params[$i]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$i = 0;
|
|
|
|
foreach ($template_types as $type_name => $_) {
|
|
|
|
if (isset($class_template_params[$type_name])) {
|
|
|
|
$i++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($class_storage !== $static_class_storage
|
|
|
|
&& isset($e[$class_storage->name][$type_name])
|
|
|
|
) {
|
|
|
|
$input_type_extends = $e[$class_storage->name][$type_name];
|
|
|
|
|
|
|
|
$output_type_extends = null;
|
|
|
|
|
|
|
|
foreach ($input_type_extends->getAtomicTypes() as $type_extends_atomic) {
|
|
|
|
if ($type_extends_atomic instanceof Type\Atomic\TTemplateParam) {
|
|
|
|
if (isset($static_class_storage->template_types[$type_extends_atomic->param_name])) {
|
|
|
|
$mapped_offset = array_search(
|
|
|
|
$type_extends_atomic->param_name,
|
|
|
|
array_keys($static_class_storage->template_types)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (isset($lhs_type_part->type_params[(int) $mapped_offset])) {
|
|
|
|
$candidate_type = $lhs_type_part->type_params[(int) $mapped_offset];
|
|
|
|
|
|
|
|
if (!$output_type_extends) {
|
|
|
|
$output_type_extends = $candidate_type;
|
|
|
|
} else {
|
|
|
|
$output_type_extends = Type::combineUnionTypes(
|
|
|
|
$candidate_type,
|
|
|
|
$output_type_extends
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} elseif (isset(
|
|
|
|
$static_class_storage
|
|
|
|
->template_type_extends
|
|
|
|
[$type_extends_atomic->defining_class]
|
|
|
|
[$type_extends_atomic->param_name]
|
|
|
|
)) {
|
|
|
|
$mapped_offset = array_search(
|
|
|
|
$type_extends_atomic->param_name,
|
|
|
|
array_keys($static_class_storage
|
|
|
|
->template_type_extends
|
|
|
|
[$type_extends_atomic->defining_class])
|
|
|
|
);
|
|
|
|
|
|
|
|
if (isset($lhs_type_part->type_params[(int) $mapped_offset])) {
|
|
|
|
$candidate_type = $lhs_type_part->type_params[(int) $mapped_offset];
|
|
|
|
|
|
|
|
if (!$output_type_extends) {
|
|
|
|
$output_type_extends = $candidate_type;
|
|
|
|
} else {
|
|
|
|
$output_type_extends = Type::combineUnionTypes(
|
|
|
|
$candidate_type,
|
|
|
|
$output_type_extends
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!$output_type_extends) {
|
|
|
|
$output_type_extends = new Type\Union([$type_extends_atomic]);
|
|
|
|
} else {
|
|
|
|
$output_type_extends = Type::combineUnionTypes(
|
|
|
|
new Type\Union([$type_extends_atomic]),
|
|
|
|
$output_type_extends
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($lhs_var_id !== '$this' || $static_fq_class_name !== $class_storage->name) {
|
|
|
|
$class_template_params[$type_name][$class_storage->name] = [
|
|
|
|
$output_type_extends ?: Type::getMixed()
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (($lhs_var_id !== '$this' || $static_fq_class_name !== $class_storage->name)
|
|
|
|
&& !isset($class_template_params[$type_name])
|
|
|
|
) {
|
|
|
|
$class_template_params[$type_name] = [
|
|
|
|
$class_storage->name => [Type::getMixed()]
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($template_types as $type_name => $type_map) {
|
2020-09-02 06:17:41 +02:00
|
|
|
foreach ($type_map as [$type]) {
|
2020-03-12 04:15:15 +01:00
|
|
|
foreach ($candidate_class_storages as $candidate_class_storage) {
|
|
|
|
if ($candidate_class_storage !== $static_class_storage
|
|
|
|
&& isset($e[$candidate_class_storage->name][$type_name])
|
|
|
|
&& !isset($class_template_params[$type_name][$candidate_class_storage->name])
|
|
|
|
) {
|
|
|
|
$class_template_params[$type_name][$candidate_class_storage->name] = [
|
|
|
|
new Type\Union(
|
|
|
|
self::expandType(
|
2020-05-30 22:55:18 +02:00
|
|
|
$codebase,
|
2020-03-12 04:15:15 +01:00
|
|
|
$e[$candidate_class_storage->name][$type_name],
|
|
|
|
$e,
|
|
|
|
$static_class_storage->name,
|
|
|
|
$static_class_storage->template_types
|
|
|
|
)
|
|
|
|
)
|
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($lhs_var_id !== '$this') {
|
|
|
|
if (!isset($class_template_params[$type_name])) {
|
|
|
|
$class_template_params[$type_name][$class_storage->name] = [$type];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $class_template_params;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array<string, array<int|string, Type\Union>> $e
|
|
|
|
* @return non-empty-list<Type\Atomic>
|
|
|
|
*/
|
|
|
|
private static function expandType(
|
2020-05-30 22:55:18 +02:00
|
|
|
Codebase $codebase,
|
2020-03-12 04:15:15 +01:00
|
|
|
Type\Union $input_type_extends,
|
|
|
|
array $e,
|
|
|
|
string $static_fq_class_name,
|
|
|
|
?array $static_template_types
|
|
|
|
) : array {
|
|
|
|
$output_type_extends = [];
|
|
|
|
|
|
|
|
foreach ($input_type_extends->getAtomicTypes() as $type_extends_atomic) {
|
|
|
|
if ($type_extends_atomic instanceof Type\Atomic\TTemplateParam
|
|
|
|
&& ($static_fq_class_name !== $type_extends_atomic->defining_class
|
|
|
|
|| !isset($static_template_types[$type_extends_atomic->param_name]))
|
|
|
|
&& isset($e[$type_extends_atomic->defining_class][$type_extends_atomic->param_name])
|
|
|
|
) {
|
|
|
|
$output_type_extends = array_merge(
|
|
|
|
$output_type_extends,
|
|
|
|
self::expandType(
|
2020-05-30 22:55:18 +02:00
|
|
|
$codebase,
|
2020-03-12 04:15:15 +01:00
|
|
|
$e[$type_extends_atomic->defining_class][$type_extends_atomic->param_name],
|
|
|
|
$e,
|
|
|
|
$static_fq_class_name,
|
|
|
|
$static_template_types
|
|
|
|
)
|
|
|
|
);
|
2020-05-30 22:55:18 +02:00
|
|
|
} elseif ($type_extends_atomic instanceof TScalarClassConstant) {
|
|
|
|
$expanded = TypeExpander::expandAtomic(
|
|
|
|
$codebase,
|
|
|
|
$type_extends_atomic,
|
|
|
|
$type_extends_atomic->fq_classlike_name,
|
|
|
|
$type_extends_atomic->fq_classlike_name,
|
|
|
|
null,
|
|
|
|
true,
|
|
|
|
true
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($expanded instanceof Atomic) {
|
|
|
|
$output_type_extends[] = $expanded;
|
|
|
|
} else {
|
|
|
|
foreach ($expanded as $type) {
|
|
|
|
$output_type_extends[] = $type;
|
|
|
|
}
|
|
|
|
}
|
2020-03-12 04:15:15 +01:00
|
|
|
} else {
|
|
|
|
$output_type_extends[] = $type_extends_atomic;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $output_type_extends;
|
|
|
|
}
|
|
|
|
}
|