From 878dfa225047cd600405bcc25a6d7e689b8088a5 Mon Sep 17 00:00:00 2001 From: adrew Date: Fri, 31 Dec 2021 17:34:06 +0300 Subject: [PATCH 1/4] Variable types inference in method context when method marked with psalm-if-this-is --- .../Analyzer/FunctionLikeAnalyzer.php | 28 ++++++++- tests/IfThisIsTest.php | 60 +++++++++++++++++++ 2 files changed, 86 insertions(+), 2 deletions(-) diff --git a/src/Psalm/Internal/Analyzer/FunctionLikeAnalyzer.php b/src/Psalm/Internal/Analyzer/FunctionLikeAnalyzer.php index 891cd1206..99a11cbb3 100644 --- a/src/Psalm/Internal/Analyzer/FunctionLikeAnalyzer.php +++ b/src/Psalm/Internal/Analyzer/FunctionLikeAnalyzer.php @@ -24,6 +24,9 @@ use Psalm\Internal\PhpVisitor\NodeCounterVisitor; use Psalm\Internal\Provider\NodeDataProvider; use Psalm\Internal\Type\Comparator\TypeComparisonResult; use Psalm\Internal\Type\Comparator\UnionTypeComparator; +use Psalm\Internal\Type\TemplateInferredTypeReplacer; +use Psalm\Internal\Type\TemplateResult; +use Psalm\Internal\Type\TemplateStandinTypeReplacer; use Psalm\Internal\Type\TypeExpander; use Psalm\Issue\InvalidDocblockParamName; use Psalm\Issue\InvalidParamDefault; @@ -64,6 +67,7 @@ use function count; use function end; use function in_array; use function is_string; +use function mb_strpos; use function md5; use function microtime; use function reset; @@ -1808,10 +1812,30 @@ abstract class FunctionLikeAnalyzer extends SourceAnalyzer } else { $this_object_type = new TNamedObject($context->self); } - + $this_object_type->was_static = !$storage->final; - $context->vars_in_scope['$this'] = new Union([$this_object_type]); + if ($this->storage instanceof MethodStorage && $this->storage->if_this_is_type) { + $template_result = new TemplateResult($this->getTemplateTypeMap() ?? [], []); + + TemplateStandinTypeReplacer::replace( + new Union([$this_object_type]), + $template_result, + $codebase, + null, + $this->storage->if_this_is_type + ); + + foreach ($context->vars_in_scope as $var_name => $var_type) { + if (0 === mb_strpos($var_name, '$this->')) { + TemplateInferredTypeReplacer::replace($var_type, $template_result, $codebase); + } + } + + $context->vars_in_scope['$this'] = $this->storage->if_this_is_type; + } else { + $context->vars_in_scope['$this'] = new Union([$this_object_type]); + } if ($codebase->taint_flow_graph && $storage->specialize_call diff --git a/tests/IfThisIsTest.php b/tests/IfThisIsTest.php index b8f4bcfdf..b10ba3e80 100644 --- a/tests/IfThisIsTest.php +++ b/tests/IfThisIsTest.php @@ -134,6 +134,66 @@ class IfThisIsTest extends TestCase $app->start(); ' ], + 'ifThisIsChangeThisTypeInsideMethod' => [ + ' */ + private $items; + + /** + * @param list $items + */ + public function __construct(array $items) + { + $this->items = $items; + } + + /** + * @psalm-if-this-is ArrayList> + * @return ArrayList + */ + public function compact(): ArrayList + { + $values = []; + + foreach ($this->items as $item) { + $value = $item->unwrap(); + + if (null !== $value) { + $values[] = $value; + } + } + + return new self($values); + } + } + + /** @var ArrayList> $list */ + $list = new ArrayList([]); + $numbers = $list->compact(); + ', + 'assertions' => [ + '$numbers' => 'ArrayList' + ], + ], ]; } From 79ea09443353c52baa10f4b75648b49cc9df432d Mon Sep 17 00:00:00 2001 From: adrew Date: Fri, 31 Dec 2021 19:57:49 +0300 Subject: [PATCH 2/4] Template resolving for psalm-if-this-is --- .../ExistingAtomicMethodCallAnalyzer.php | 62 ++++++++++++------- 1 file changed, 40 insertions(+), 22 deletions(-) diff --git a/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php b/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php index 12eeff498..3ba98a8a9 100644 --- a/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php +++ b/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php @@ -20,7 +20,9 @@ use Psalm\Internal\FileManipulation\FileManipulationBuffer; use Psalm\Internal\MethodIdentifier; use Psalm\Internal\Type\Comparator\TypeComparisonResult; use Psalm\Internal\Type\Comparator\UnionTypeComparator; +use Psalm\Internal\Type\TemplateInferredTypeReplacer; use Psalm\Internal\Type\TemplateResult; +use Psalm\Internal\Type\TemplateStandinTypeReplacer; use Psalm\Internal\Type\TypeExpander; use Psalm\Issue\IfThisIsMismatch; use Psalm\Issue\InvalidPropertyAssignmentValue; @@ -188,7 +190,30 @@ class ExistingAtomicMethodCallAnalyzer extends CallAnalyzer } } + try { + $method_storage = $codebase->methods->getStorage($declaring_method_id ?? $method_id); + } catch (UnexpectedValueException $e) { + $method_storage = null; + } + + $method_template_params = []; + + if ($method_storage && $method_storage->if_this_is_type) { + $method_template_result = new TemplateResult($method_storage->template_types ?: [], []); + + TemplateStandinTypeReplacer::replace( + clone $method_storage->if_this_is_type, + $method_template_result, + $codebase, + null, + new Union([$lhs_type_part]) + ); + + $method_template_params = $method_template_result->lower_bounds; + } + $template_result = new TemplateResult([], $class_template_params ?: []); + $template_result->lower_bounds += $method_template_params; if ($codebase->store_node_types && !$context->collect_initializations @@ -264,30 +289,23 @@ class ExistingAtomicMethodCallAnalyzer extends CallAnalyzer } } - try { - $method_storage = $codebase->methods->getStorage($declaring_method_id ?? $method_id); - } catch (UnexpectedValueException $e) { - $method_storage = null; - } - if ($method_storage) { - $class_type = new Union([$lhs_type_part]); + if ($method_storage->if_this_is_type) { + $class_type = new Union([$lhs_type_part]); + $if_this_is_type = clone $method_storage->if_this_is_type; - if ($method_storage->if_this_is_type - && !UnionTypeComparator::isContainedBy( - $codebase, - $class_type, - $method_storage->if_this_is_type - ) - ) { - IssueBuffer::maybeAdd( - new IfThisIsMismatch( - 'Class type must be ' . $method_storage->if_this_is_type->getId() - . ' current type ' . $class_type->getId(), - new CodeLocation($source, $stmt->name) - ), - $statements_analyzer->getSuppressedIssues() - ); + TemplateInferredTypeReplacer::replace($if_this_is_type, $template_result, $codebase); + + if (!UnionTypeComparator::isContainedBy($codebase, $class_type, $if_this_is_type)) { + IssueBuffer::maybeAdd( + new IfThisIsMismatch( + 'Class type must be ' . $method_storage->if_this_is_type->getId() + . ' current type ' . $class_type->getId(), + new CodeLocation($source, $stmt->name) + ), + $statements_analyzer->getSuppressedIssues() + ); + } } if ($method_storage->self_out_type && $lhs_var_id) { From 29af83bf61c6aa6f63739bbcc8059a6f58c3deba Mon Sep 17 00:00:00 2001 From: adrew Date: Fri, 31 Dec 2021 20:35:56 +0300 Subject: [PATCH 3/4] Add tests for psalm-if-this-is variables and template inference --- tests/IfThisIsTest.php | 96 ++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 96 insertions(+) diff --git a/tests/IfThisIsTest.php b/tests/IfThisIsTest.php index b10ba3e80..2c3c3b004 100644 --- a/tests/IfThisIsTest.php +++ b/tests/IfThisIsTest.php @@ -194,6 +194,76 @@ class IfThisIsTest extends TestCase '$numbers' => 'ArrayList' ], ], + 'ifThisIsResolveTemplateParams' => [ + ' */ + private $items; + + /** + * @param list $items + */ + public function __construct(array $items) + { + $this->items = $items; + } + + /** + * @template A + * @template B + * @template TOption of Option + * @template TEither of Either + * + * @psalm-if-this-is ArrayList + * @return ArrayList + */ + public function compact(): ArrayList + { + $values = []; + + foreach ($this->items as $item) { + $value = $item->unwrap(); + + if (null !== $value) { + $values[] = $value; + } + } + + return new self($values); + } + } + + /** @var ArrayList|Option> $list */ + $list = new ArrayList([]); + $numbers = $list->compact(); + ', + 'assertions' => [ + '$numbers' => 'ArrayList' + ], + ], ]; } @@ -281,6 +351,32 @@ class IfThisIsTest extends TestCase ', 'error_message' => 'IfThisIsMismatch' ], + 'failWithInvalidTemplateConstraint' => [ + '> + * @return ArrayList + */ + public function compact(): ArrayList + { + throw new RuntimeException("???"); + } + } + + /** @var ArrayList $list */ + $list = new ArrayList(); + $numbers = $list->compact();', + 'error_message' => 'IfThisIsMismatch' + ], ]; } } From 0f69483cc1d7da53a7ba403d9ef9a58a36b5dc01 Mon Sep 17 00:00:00 2001 From: adrew Date: Sat, 1 Jan 2022 13:33:00 +0300 Subject: [PATCH 4/4] Fix method storage fetching --- .../Call/Method/ExistingAtomicMethodCallAnalyzer.php | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php b/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php index 3ba98a8a9..1fda720d0 100644 --- a/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php +++ b/src/Psalm/Internal/Analyzer/Statements/Expression/Call/Method/ExistingAtomicMethodCallAnalyzer.php @@ -190,6 +190,8 @@ class ExistingAtomicMethodCallAnalyzer extends CallAnalyzer } } + $declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id); + try { $method_storage = $codebase->methods->getStorage($declaring_method_id ?? $method_id); } catch (UnexpectedValueException $e) { @@ -240,8 +242,6 @@ class ExistingAtomicMethodCallAnalyzer extends CallAnalyzer return Type::getMixed(); } - $declaring_method_id = $codebase->methods->getDeclaringMethodId($method_id); - $return_type_candidate = MethodCallReturnTypeFetcher::fetch( $statements_analyzer, $codebase,