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

assertCount with remembered values causing false positives #208

Closed
janedbal opened this issue May 21, 2024 · 7 comments
Closed

assertCount with remembered values causing false positives #208

janedbal opened this issue May 21, 2024 · 7 comments

Comments

@janedbal
Copy link
Contributor

Simplified reproducible example:

class AssertCountProblemTest extends TestCase
{

    /**
     * @var list<string>
     */
    private static array $database = [];

    public function testFoo(): void
    {
        self::assertCount(0, $this->getDataFromDatabase());

        self::editDataInDatabase();

        $newData = $this->getDataFromDatabase();
        self::assertCount(1, $newData); // error: Call to static method PHPUnit\Framework\Assert::assertCount() with 1 and array{} will always evaluate to false.
    }

    /**
     * @return list<string>
     */
    private function getDataFromDatabase(): array
    {
        return self::$database;
    }

    private static function editDataInDatabase(): void
    {
        self::$database[] = 'new data';
    }

}

This poped up while upgrading phpstan-phpunit from 1.3.15 to 1.4.0

@janedbal
Copy link
Contributor Author

1.3.16 is the version where this breaks.

@Ciloe
Copy link

Ciloe commented Nov 22, 2024

Same here at version 1.4.1.

I solved temporarly by creating a variable, and using it in the Count function :

// Before :
self::assertCount(1, $this->getDataFromDatabase());

// After :
$newData = $this->getDataFromDatabase();
self::assertCount(1, $newData);

@thePanz
Copy link

thePanz commented Mar 24, 2025

Just upgraded to PHPStan v2.11.1 and got this issue.
The same code worked fine with v2.10.7 (this was the previous version I had installed)

@ondrejmirtes
Copy link
Member

@thePanz Your problem probably stems from a much smarter type narrowing when count() is involved. This isn't a bug but a feature. It's not specific to PHPUnit either.

You can either:

  1. Mark the function as @phpstan-impure: https://phpstan.org/r/e621db72-ce3f-4833-a967-79c455f25542
  2. or you can set rememberPossiblyImpureFunctionValues: false: https://phpstan.org/config-reference#rememberpossiblyimpurefunctionvalues

Detailed here: https://phpstan.org/blog/remembering-and-forgetting-returned-values

Also I talk about it for 10 minutes here: https://www.youtube.com/watch?v=AFjr3RlDOZQ&t=1728s

@ondrejmirtes
Copy link
Member

As this isn't specific to PHPUnit either, @janedbal please re-report this in PHPStan: https://phpstan.org/r/b610c6cb-c572-49b1-a78c-1a1ad5bd9100

@janedbal
Copy link
Contributor Author

Please re-report this in PHPStan: https://phpstan.org/r/b610c6cb-c572-49b1-a78c-1a1ad5bd9100

Does that actually make sense? It works when properly annotated. I believe the problem is more about DX. Nobody wants to mark all similar methods with @phpstan-impure.

@ondrejmirtes
Copy link
Member

I'm not sure what you're suggesting. As I'm describing the recording of my talk, these defaults make sense to me. If you don't want to use @phpstan-impure, there's rememberPossiblyImpureFunctionValues: false.

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

No branches or pull requests

4 participants