diff --git a/src/Psalm/Internal/Analyzer/Statements/Expression/BinaryOp/ConcatAnalyzer.php b/src/Psalm/Internal/Analyzer/Statements/Expression/BinaryOp/ConcatAnalyzer.php index e5f6d515f..79b298c51 100644 --- a/src/Psalm/Internal/Analyzer/Statements/Expression/BinaryOp/ConcatAnalyzer.php +++ b/src/Psalm/Internal/Analyzer/Statements/Expression/BinaryOp/ConcatAnalyzer.php @@ -216,7 +216,8 @@ class ConcatAnalyzer $lowercase_type ); - $non_empty_string = Type::getNonEmptyString(); + $non_empty_string = clone $numeric_type; + $non_empty_string->addType(new Type\Atomic\TNonEmptyString()); $has_non_empty = UnionTypeComparator::isContainedBy( $codebase, diff --git a/src/Psalm/Internal/Type/Comparator/ScalarTypeComparator.php b/src/Psalm/Internal/Type/Comparator/ScalarTypeComparator.php index f3dc6a63c..9df93389b 100644 --- a/src/Psalm/Internal/Type/Comparator/ScalarTypeComparator.php +++ b/src/Psalm/Internal/Type/Comparator/ScalarTypeComparator.php @@ -585,6 +585,11 @@ class ScalarTypeComparator return true; } + if ($input_type_part instanceof TLowercaseString + && get_class($container_type_part) === TNonEmptyString::class) { + return false; + } + if ($input_type_part->getKey() === $container_type_part->getKey()) { return true; } diff --git a/tests/Template/ConditionalReturnTypeTest.php b/tests/Template/ConditionalReturnTypeTest.php index 0cbab4dc0..211dc000e 100644 --- a/tests/Template/ConditionalReturnTypeTest.php +++ b/tests/Template/ConditionalReturnTypeTest.php @@ -841,6 +841,24 @@ class ConditionalReturnTypeTest extends TestCase } ' ], + 'strlenReturnsIntForLowercaseString' => [ + '