diff --git a/src/Psalm/Config.php b/src/Psalm/Config.php index a8a08366e..069fd2c7a 100644 --- a/src/Psalm/Config.php +++ b/src/Psalm/Config.php @@ -2781,8 +2781,22 @@ class Config $version_parser = new VersionParser(); $constraint = $version_parser->parseConstraints($php_version); + $php_versions = [ + '5.4', + '5.5', + '5.6', + '7.0', + '7.1', + '7.2', + '7.3', + '7.4', + '8.0', + '8.1', + '8.2', + '8.3', + ]; - foreach (['5.4', '5.5', '5.6', '7.0', '7.1', '7.2', '7.3', '7.4', '8.0', '8.1'] as $candidate) { + foreach ($php_versions as $candidate) { if ($constraint->matches(new Constraint('<=', "$candidate.0.0-dev")) || $constraint->matches(new Constraint('<=', "$candidate.999")) ) { diff --git a/src/Psalm/Internal/PhpVisitor/Reflector/FunctionLikeDocblockParser.php b/src/Psalm/Internal/PhpVisitor/Reflector/FunctionLikeDocblockParser.php index 52106d389..25fcb44bf 100644 --- a/src/Psalm/Internal/PhpVisitor/Reflector/FunctionLikeDocblockParser.php +++ b/src/Psalm/Internal/PhpVisitor/Reflector/FunctionLikeDocblockParser.php @@ -23,6 +23,7 @@ use function count; use function explode; use function implode; use function in_array; +use function pathinfo; use function preg_last_error_msg; use function preg_match; use function preg_replace; @@ -37,6 +38,8 @@ use function substr; use function substr_count; use function trim; +use const PATHINFO_EXTENSION; + /** * @internal */ @@ -417,11 +420,15 @@ final class FunctionLikeDocblockParser if (isset($parsed_docblock->tags['since'])) { $since = trim(reset($parsed_docblock->tags['since'])); - if (preg_match('/^[4578]\.\d(\.\d+)?$/', $since)) { - $since_parts = explode('.', $since); - - $info->since_php_major_version = (int)$since_parts[0]; - $info->since_php_minor_version = (int)$since_parts[1]; + // only for phpstub files or @since 8.0.0 PHP + // since @since is commonly used with the project version, not the PHP version + // https://docs.phpdoc.org/3.0/guide/references/phpdoc/tags/since.html + // https://github.com/vimeo/psalm/issues/10761 + if (preg_match('/^([4578])\.(\d)(\.\d+)?(\s+PHP)?$/i', $since, $since_match) + && isset($since_match[1])&& isset($since_match[2]) + && (!empty($since_match[4]) || pathinfo($code_location->file_name, PATHINFO_EXTENSION) === 'phpstub')) { + $info->since_php_major_version = (int)$since_match[1]; + $info->since_php_minor_version = (int)$since_match[2]; } } diff --git a/tests/AnnotationTest.php b/tests/AnnotationTest.php index 1626c0bcc..7a7345bd5 100644 --- a/tests/AnnotationTest.php +++ b/tests/AnnotationTest.php @@ -1384,6 +1384,16 @@ class AnnotationTest extends TestCase class Foo {}', 'assertions' => [], ], + 'sinceTagNonPhpVersion' => [ + 'code' => '