Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Treat #[Pure(true)] in PhpStorm stubs as hasSideEffects => true #3880

Conversation

zonuexe
Copy link
Contributor

@zonuexe zonuexe commented Mar 13, 2025

PhpStorm marks functions that have side effects but whose return value is important as #[Pure(true)], which in current PHPStan functionality means [hasSideEffects => true].

refs #3867 (comment)
refs phpstan/phpstan#12738

@zonuexe zonuexe marked this pull request as draft March 13, 2025 16:59
@zonuexe zonuexe force-pushed the feature/phpstorm-stub-pure-true-as-sideeffect branch 2 times, most recently from 7444b55 to 7753667 Compare March 14, 2025 06:12
@zonuexe zonuexe force-pushed the feature/phpstorm-stub-pure-true-as-sideeffect branch from 7753667 to 37d1861 Compare March 14, 2025 06:14
@zonuexe zonuexe marked this pull request as ready for review March 14, 2025 06:33
@phpstan-bot
Copy link
Collaborator

This pull request has been marked as ready for review.

Copy link
Contributor

@staabm staabm left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

before this PR phpstan errored
https://phpstan.org/r/57c356b1-b4f6-4a36-94f8-01a21b8996d4

Call to function checkdate() on a separate line has no effect.

after this change it no longer does

➜  phpstan-src git:(pr/3880) ✗ cat test.php
<?php declare(strict_types = 1);

class HelloWorld
{
        public function sayHello($m): void
        {
                checkdate($m, $m, $m);
        }
}                                                                           
                                                                                                                        

➜  phpstan-src git:(pr/3880) ✗ php bin/phpstan analyze test.php --debug
Note: Using configuration file /Users/staabm/workspace/phpstan-src/phpstan.neon.dist.
/Users/staabm/workspace/phpstan-src/test.php
 ------ --------------------------------------------- 
  Line   test.php                                     
 ------ --------------------------------------------- 
  3      Class HelloWorld must be abstract or final.  
         🪪  phpstan.finalClass                       
 ------ --------------------------------------------- 


                                                                                                                        
 [ERROR] Found 1 error                                                                                                  
                                                                                                                        

➜  phpstan-src git:(2.1.x) ✗ php bin/phpstan analyze test.php --debug
Note: Using configuration file /Users/staabm/workspace/phpstan-src/phpstan.neon.dist.
/Users/staabm/workspace/phpstan-src/test.php
 ------ ---------------------------------------------------------------- 
  Line   test.php                                                        
 ------ ---------------------------------------------------------------- 
  3      Class HelloWorld must be abstract or final.                     
         🪪  phpstan.finalClass                                          
  7      Call to function checkdate() on a separate line has no effect.  
         🪪  function.resultUnused                                       
 ------ ---------------------------------------------------------------- 


                                                                                                                        
 [ERROR] Found 2 errors                                                                                                 
                                                                                                                        

➜  phpstan-src git:(2.1.x) ✗ 

@zonuexe
Copy link
Contributor Author

zonuexe commented Mar 16, 2025

@staabm Thanks, I've sent a PR upstream since the checkdate() function is clearly a pure function.
https://github.com/JetBrains/phpstorm-stubs/pull/1726/files#r1997508160

Comment on lines +814 to +815
'curl_errno' => ['hasSideEffects' => true],
'curl_error' => ['hasSideEffects' => true],
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This also looks wrong. I guess you should review all lines which changed from false to true

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Although they appear to be pure, they actually rely on some external state associated with the resource, so it makes sense that PhpStorm marks them as #[Pure(true)].

While it may be convenient to make curl_errno() pure for convenience in common use cases, it is inconsistent with impure functions like ob_get_level()(#12577).

To solve this problem, must_use/#[NoDiscard] needs to be supported separately from @pure/@impure.

@claudepache
Copy link
Contributor

As I understand, both #[Pure(false)] and #[Pure(true)] mean that the function has no side-effect, the difference between the two is that a function marked with #[Pure(true)] may depend on global state (and therefore is not pure).

In case "hasSideEffects" => (bool) doesn’t mean more than what it says (which I have not checked), those functions should be marked with "hasSideEffects" => false.

@ondrejmirtes ondrejmirtes merged commit c533a6e into phpstan:2.1.x Mar 17, 2025
416 of 418 checks passed
@ondrejmirtes
Copy link
Member

Thank you!

@zonuexe zonuexe deleted the feature/phpstorm-stub-pure-true-as-sideeffect branch March 17, 2025 09:00
@staabm
Copy link
Contributor

staabm commented Mar 17, 2025

nice, this fixed a few more bugs, see https://github.com/phpstan/phpstan-src/actions/runs/13851103733

regression tests would be awesome

@ondrejmirtes
Copy link
Member

I don't think we need them, they'd just verify "yeah, this function is in this array", not any real logic 😊

@ondrejmirtes
Copy link
Member

But yeah, three issues can be closed thanks to this :)

@@ -1444,8 +1455,8 @@
'posix_uname' => ['hasSideEffects' => false],
'pow' => ['hasSideEffects' => false],
'preg_grep' => ['hasSideEffects' => false],
'preg_last_error' => ['hasSideEffects' => false],
'preg_last_error_msg' => ['hasSideEffects' => false],
'preg_last_error' => ['hasSideEffects' => true],
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

https://3v4l.org/O43Rb

what side effects does it have?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks to this line we solved this issue phpstan/phpstan#10342

Copy link
Contributor

@mvorisek mvorisek Mar 17, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, I mean we should differentiate between pure and side effects. In my understanding "pure" is about return value, "side effects" are about effects on other functions. So I would expect this function to have no side effects, but be pure instead.

related with #118

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants