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

Improve count() narrowing of constant arrays #3709

Merged
merged 7 commits into from
Mar 10, 2025

Conversation

herndlm
Copy link
Contributor

@herndlm herndlm commented Dec 4, 2024

Generalizes the already existing solution (wow, this is way more complicated than I thought!) and uses it outside of union types as well. I kind of had to make the "isNormalCount" determination a bit more dry, which unfortunately made the diff worse..

I did do a performance comparison with hyperfine running make phpstan and didn't notice any changes, but my system is not the best.

Closes phpstan/phpstan#12190
Closes phpstan/phpstan#3631

@herndlm
Copy link
Contributor Author

herndlm commented Dec 4, 2024

I know, I changed that narrow* method to return Type instead of SpecifiesTypes. Maybe I'll change this back again tomorrow.

@herndlm herndlm force-pushed the constant-array-count-narrowing branch 2 times, most recently from 891412d to 22ef97b Compare December 5, 2024 09:47
@herndlm
Copy link
Contributor Author

herndlm commented Dec 5, 2024

I know, I changed that narrow* method to return Type instead of SpecifiesTypes. Maybe I'll change this back again tomorrow.

adapted. looks better now IMO. The master plan is to move more common things for both count() == xx and count > xx specification into the now renamed specifyTypesForCountFuncCall(). Would also be great if turnListIntoConstantArray() goes away completely or is only used internally once... But for here this is more than enough IMO.

@staabm what do you think about this?

@staabm
Copy link
Contributor

staabm commented Dec 5, 2024

what do you think about this?

this PR looks great, thanks for cleaning up after me :)

@herndlm herndlm force-pushed the constant-array-count-narrowing branch 2 times, most recently from 5964211 to 75991d1 Compare December 5, 2024 20:19
@herndlm
Copy link
Contributor Author

herndlm commented Dec 23, 2024

Should stuff like this be based on 1.12.x or latest dev branch? I sometimes have issues deciding 😅

@ondrejmirtes
Copy link
Member

Big changes to TypeSpecifier I prefer on 2.1.x, to avoid conflicts.

@herndlm herndlm changed the base branch from 1.12.x to 2.1.x December 24, 2024 17:25
@herndlm herndlm force-pushed the constant-array-count-narrowing branch 3 times, most recently from 8092d6b to f340663 Compare December 24, 2024 18:20
@herndlm
Copy link
Contributor Author

herndlm commented Dec 24, 2024

Thx, makes sense. And good call, there was a conflict.. Rebased this and also other of my PRs on 2.1.x and dealt with conflicts 😊

@herndlm herndlm force-pushed the constant-array-count-narrowing branch 2 times, most recently from fc54113 to d92b6f0 Compare December 26, 2024 01:16
@herndlm herndlm force-pushed the constant-array-count-narrowing branch 2 times, most recently from 4b53a7f to 66a0ccd Compare January 6, 2025 15:19
@herndlm
Copy link
Contributor Author

herndlm commented Jan 6, 2025

I think this is ready, only that first issue bot response is confusing me slightly: https://github.com/phpstan/phpstan-src/actions/runs/12635127480. but at the same time it looks like those errors are already there now (https://phpstan.org/r/3cb22712-7994-4b1c-9946-3946083ddb27) or am I missing something?

UPDATE: ah wait, it's line 109 of PHP 7.1 - 7.4 which seems to be loosing some generic info I think. I'll check again then..

@herndlm herndlm marked this pull request as draft January 6, 2025 18:17
@herndlm herndlm force-pushed the constant-array-count-narrowing branch from 3c0996a to ace9569 Compare January 6, 2025 20:06
@herndlm
Copy link
Contributor Author

herndlm commented Jan 6, 2025

ok done, this was even simplified further, is now behaving as I expect it to and should be save :)

@herndlm herndlm marked this pull request as ready for review January 6, 2025 20:24
@phpstan-bot
Copy link
Collaborator

This pull request has been marked as ready for review.

@herndlm herndlm force-pushed the constant-array-count-narrowing branch from ace9569 to ff89f34 Compare January 26, 2025 11:33
@herndlm herndlm force-pushed the constant-array-count-narrowing branch 2 times, most recently from db853e0 to d669212 Compare February 10, 2025 12:47
@herndlm herndlm force-pushed the constant-array-count-narrowing branch from f9fb3ef to 4c5363b Compare March 10, 2025 10:20
Copy link
Member

@ondrejmirtes ondrejmirtes left a comment

Choose a reason for hiding this comment

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

The levels test JSON needs updating: https://github.com/phpstan/phpstan-src/actions/runs/13762263415/job/38480859499?pr=3709

Just run it locally, it will autoupdate the JSON file, and commit that.

Also it might be pointing out an actual bug, it might be worth reviewing what changed.

@herndlm
Copy link
Contributor Author

herndlm commented Mar 10, 2025

@ondrejmirtes thx for checking this. it looks like though those levels test differences are not coming from here but were recently introduced in deb0911

UPDATE: but let me update those levels tests in a dedicated PR anyway, you can review the changes there.

--> #3869

@herndlm herndlm requested a review from ondrejmirtes March 10, 2025 20:15
{
if (count($arr) <= 1) {
assertType('1', count($arr));
return;
Copy link
Contributor

Choose a reason for hiding this comment

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

What about adding assertType('array{string}', $arr); ?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

good point, I already forgot. but quickly looked again and apparently it's currently

assertType('array{0: string, 1?: string}', $arr);

it ends up using a falsey scope here, because count($arr) <= 1 is transformed to 2 < count($arr) and then something in the loop still skips creating a result type because it can't fully deal with falsey scopes unfortunately. If I remove that continue, it would basically come up with "cannot be array{string, string}" which should make it then array{string} by some falsey scope filtering or so irrelevant of the new code. but that does also not work :/ and many tests for (list) counting start to fail :/ would be great to further improve this, but I was kind of stuck there..

@ondrejmirtes ondrejmirtes force-pushed the constant-array-count-narrowing branch from 4c5363b to 6153190 Compare March 10, 2025 20:52
@ondrejmirtes ondrejmirtes merged commit 60b29fa into phpstan:2.1.x Mar 10, 2025
43 checks passed
@ondrejmirtes
Copy link
Member

Thank you!

@herndlm herndlm deleted the constant-array-count-narrowing branch March 10, 2025 20:53
@ondrejmirtes
Copy link
Member

Hi, I found out about a regression because of this PR: https://phpstan.org/r/38051a80-e8ef-4f26-abaa-dac27791b444

The keys shouldn't be optional :)

@herndlm
Copy link
Contributor Author

herndlm commented Mar 11, 2025

It looks like this was already like that before but that case is not supported by the changes here? but I agree, they should be non-optional 🤔

@herndlm
Copy link
Contributor Author

herndlm commented Mar 11, 2025

aha, that one is not handled because the $arrayType->isList() returns maybe. interesting

@ondrejmirtes
Copy link
Member

Yeah but even if something is not a list, if the total number of keys is <= than the verified count, we know we can make all of them required.

@herndlm
Copy link
Contributor Author

herndlm commented Mar 11, 2025

I'll look into this. @staabm initially created all kind of list count tests (luckily!) and some start to fail, so those cases are dangerous. e.g. smth like where the keys have gaps I suppose.

if (count($row) === 2) {
	assertType('array{0: int, 3?: string|null}', $row); // is array{int, *ERROR*} instead
} else {
	assertType('array{0: int, 3?: string|null}|array{string}', $row);
}

or, in general, some new feature for this needs to be added or the existing one changed :) should be doable

@ondrejmirtes
Copy link
Member

FYI this PR broke this snippet https://phpstan.org/r/8b318eac-e881-46c4-a72d-8ae55f2953e5

There's nothing we can deduce for $this->importedDaySummaryRows so the type shouldn't be narrowed to Never.

I'm looking into this myself but I'm not sure how successful I'm going to be, so please look into it too. The problem goes away when I change the type $this->totalExpectedRows from int to positive-int so it's going to be something about that.

@ondrejmirtes
Copy link
Member

This diff fixes the problem but makes some tests fail:

diff --git a/src/Analyser/TypeSpecifier.php b/src/Analyser/TypeSpecifier.php
index c016b2869..e54de7d5e 100644
--- a/src/Analyser/TypeSpecifier.php
+++ b/src/Analyser/TypeSpecifier.php
@@ -1136,8 +1136,10 @@ final class TypeSpecifier
 				$resultTypes[] = $valueTypesBuilder->getArray();
 				continue;
 			}
+		}
 
-			$resultTypes[] = $arrayType;
+		if (count($resultTypes) === 0) {
+			return null;
 		}
 
 		return $this->create($countFuncCall->getArgs()[0]->value, TypeCombinator::union(...$resultTypes), $context, $scope)->setRootExpr($rootExpr);

Please look into this as this is currently blocking a release. Thanks.

@staabm
Copy link
Contributor

staabm commented Mar 20, 2025

I think I have a working fix. PR incoming

@staabm staabm mentioned this pull request Mar 20, 2025
@ondrejmirtes
Copy link
Member

I have one more blocking regression caused by this PR: https://phpstan.org/r/1fe15afb-62c0-4c3d-8df5-b240ccb36399

The type of $otherMinPrices should not degrade to mixed.

@herndlm
Copy link
Contributor Author

herndlm commented Mar 22, 2025

preparing a fix, but currently it "unfixes" phpstan/phpstan#1311 again unfortunately.
UPDATE: looks like it was not properly fixed really anyway, I'll open the PR soon with explanations

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