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

Difference between Instance and Static properties #3906

Draft
wants to merge 12 commits into
base: 2.1.x
Choose a base branch
from

Conversation

VincentLanglet
Copy link
Contributor

@VincentLanglet VincentLanglet commented Mar 28, 2025

@VincentLanglet
Copy link
Contributor Author

VincentLanglet commented Mar 28, 2025

I encounter some issues/questions for you @ondrejmirtes.

First, HasPropertyType is tricky

  • since it's used for both static and non static properties.
  • and property_exists returns true either it's a static or non static properties

So we have

class Foo
{
	public $foo;
	public static $staticFoo;
}


var_dump(property_exists(new Foo(), 'foo')); // TRUE
var_dump(property_exists(new Foo(), 'staticFoo')); // TRUE
Foo::$foo; // CRASH
(new Foo())->foo; // Works
Foo::$staticFoo; // Works
(new Foo())->staticFoo; // CRASH

Second, and kinda related, how should be considered properties from ObjectShapeType ?
Are they only instance properties ?

Also if you already have some comment on the existing work.

@VincentLanglet VincentLanglet force-pushed the instanceVsStatic branch 3 times, most recently from d78425a to 15cda8f Compare March 29, 2025 10:09
@VincentLanglet VincentLanglet force-pushed the instanceVsStatic branch 2 times, most recently from 6d7cb92 to 0291c75 Compare March 29, 2025 10:48
@@ -9118,10 +9118,6 @@ public function dataUnionProperties(): array
'UnionProperties\Bar|UnionProperties\Foo',
'$something->doSomething',
],
[
'UnionProperties\Bar|UnionProperties\Foo',
'$something::$doSomething',
Copy link
Contributor Author

Choose a reason for hiding this comment

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

This throw an error in PHP, a static access to a instance property is not valid.

@@ -26,5 +26,4 @@ function foo(?A $a): void
\PHPStan\Testing\assertType('string|null', $a?->b->get());

\PHPStan\Testing\assertType('int|null', $a?->b::$value);
\PHPStan\Testing\assertType('int|null', $a?->b->value);
Copy link
Contributor Author

Choose a reason for hiding this comment

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

An instance property to a static property is not valid

@@ -43,10 +43,6 @@ public function testTypesAssignedToProperties(): void
'Static property PropertiesAssignedTypes\Foo::$staticStringProperty (string) does not accept int.',
37,
],
[
'Property PropertiesAssignedTypes\Ipsum::$parentStringProperty (string) does not accept int.',
Copy link
Contributor Author

Choose a reason for hiding this comment

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

parent::$parentStringProperty is not valid since the property is an instance one.

$this->parentStringProperty is correctly reported

@ondrejmirtes
Copy link
Member

Yeah, these kinds of questions are great and generally are a positive effect of deprecating old methods - it forces us to do stuff correctly and fix existing problems along the way.

My thoughts:

  • HasPropertyType is used alongside property_exists so representing both instance and static properties is fine. Not sure what kind of problem you have with that now. If you don't know what to do about this we can leave it for later (it's fine to leave some failing tests here for now).
  • ObjectShapeType - should work only for instance properties.

@VincentLanglet
Copy link
Contributor Author

VincentLanglet commented Mar 29, 2025

In the new list of todo to see with you @ondrejmirtes I have:

  • PHPClassReflectionExtension::createProperty to maybe split into createInstanceProperty and createStaticProperty (?) ; I'm not familiar with the implementation yet.
  • HasPropertyType::isSuperTypeOf => remplace usage of hasProperty
  • HasProperty::isSubTypeOf => remplace usage of hasProperty
  • Having or not a difference between HasProperty::hasInstanceProperty and HasProperty::hasStaticProperty and updating the new HasProperty() eventually.
  • Deciding if the LevelsIntegrationTest change is ok. (It's UnionType related, when one of the class has not the property and the other has the static property of the same name)

HasPropertyType is used alongside property_exists so representing both instance and static properties is fine. Not sure what kind of problem you have with that now. If you don't know what to do about this we can leave it for later (it's fine to leave some failing tests here for now).

Currently the TypeSpecifier does

if (
					$var instanceof PropertyFetch
					&& $var->name instanceof Node\Identifier
				) {
					$types = $types->unionWith(
						$this->create($var->var, new IntersectionType([
							new ObjectWithoutClassType(),
							new HasPropertyType($var->name->toString()),
						]), TypeSpecifierContext::createTruthy(), $scope)->setRootExpr($expr),
					);
				} elseif (
					$var instanceof StaticPropertyFetch
					&& $var->class instanceof Expr
					&& $var->name instanceof Node\VarLikeIdentifier
				) {
					$types = $types->unionWith(
						$this->create($var->class, new IntersectionType([
							new ObjectWithoutClassType(),
							new HasPropertyType($var->name->toString()),
						]), TypeSpecifierContext::createTruthy(), $scope)->setRootExpr($expr),
					);
				}

Which I think, means that accessing to Foo::$bar makes PHPStan thinks that foo->bar exists. (and the opposite)

So should we add a param to the HasPropertyType construct which would be STATIC|INSTANCE|UNKNOWN

@VincentLanglet VincentLanglet force-pushed the instanceVsStatic branch 3 times, most recently from 73439f3 to 739d02c Compare March 29, 2025 13:32
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.

2 participants