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

Fix GH-12895: Incorrect ReflectionClass::getNamespaceName for anonymous class #12907

Open
wants to merge 1 commit into
base: PHP-8.2
Choose a base branch
from

Conversation

nielsdos
Copy link
Member

@nielsdos nielsdos commented Dec 8, 2023

Because paths can contain backslashes, searching for the first backslash starting on the right for detecting namespaces yields the wrong result. Furthermore, because the namespace of the class is not actually in the name (only the interface or parent class is) the namespace can be wrong.

This fix consists of two parts:

  • Fix how a prefix is chosen such that the correct namespace is included in the name.
  • If ZEND_ACC_ANON_CLASS is set, check the position of @ and search for the backslash before that.

While the issue seems Windows-specific, I could even reproduce this on Linux because Linux allows directories to contain backslashes.

Important I'm not so sure we can fully fix this in stable branches. The previous time the name scheme for anonymous classes changed was targeted to the master branch. However, the check for the ZEND_ACC_ANON_CLASS flag is safe for stable branches I think.

…ymous class

Because paths can contain backslashes, searching for the first backslash
starting on the right for detecting namespaces yields the wrong result.
Furthermore, because the namespace of the class is not actually in the
name (only the interface or parent class is) the namespace can be wrong.

This fix consists of two parts:
 - Fix how a prefix is chosen such that the correct namespace is included in
   the name.
 - If ZEND_ACC_ANON_CLASS is set, check the position of @ and search for the
   backslash before that.

While the issue seems Windows-specific, I could even reproduce this on Linux
because Linux allows directories to contain backslashes.
--EXPECTF--
object(class@%s)#1 (0) {
--EXPECT--
object(lone\@anonymous)#1 (0) {
Copy link
Contributor

Choose a reason for hiding this comment

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

I would be happy if the "1st parent non-anonymous class" can be kept dumped, as from anonymous class it is often not easy to tell quickly what the anonymous class is about and the "1st parent non-anonymous class" was helpful, compare:

DateTime@anonymous vs. class@anonymous (or Ns\@anonymous) - https://3v4l.org/Bj2qg

Copy link
Member

@nikic nikic left a comment

Choose a reason for hiding this comment

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

This should be a reflection-only change, not changing the generated name for anonymous classes.

@nikic
Copy link
Member

nikic commented Dec 9, 2023

It's not super clear what the behavior for anonymous classes should actually be -- I could see either returning the namespace portion before the @, or just always saying that the namespace name for anon classes is empty. The concept is not really well-defined for anon classes.

@nielsdos
Copy link
Member Author

nielsdos commented Dec 9, 2023

Returning the namespace portion before @ without changing the generated name seems definitely wrong to me because then it'll return the namespace portion of the first interface or parent class.
Returning an empty string could make some sense and is consistent with the result for getNamespaceName for an anonymous class without interfaces/parent class.

@mvorisek
Copy link
Contributor

mvorisek commented Dec 9, 2023

IMHO NS should be returned exactly the same as for non-anonymous class - https://3v4l.org/5RLAY and https://3v4l.org/bal7C

I belive so as NS (ReflectionClass::getNamespaceName()) is at least used to resolve relative names in phpdocs.

@iluuu1994
Copy link
Member

Also worth noting that __NAMESPACE__ currently refers to the lexical namespace in an anonymous class. Having (new \ReflectionClass($this))->getNamespaceName() diverge is a bit odd.

@nielsdos
Copy link
Member Author

nielsdos commented Dec 9, 2023

Right, it should be consistent with at least __NAMESPACE__ in which case I can't avoid changing the generated name.

@nikic
Copy link
Member

nikic commented Dec 9, 2023

Also worth noting that __NAMESPACE__ currently refers to the lexical namespace in an anonymous class. Having (new \ReflectionClass($this))->getNamespaceName() diverge is a bit odd.

It's a bit odd, but it's not necessarily wrong. The lexical and logical scope are not necessarily the same. A very similar example would be the behavior of __CLASS__ vs self::class inside a closure:

class A {}

class B {
    public function test() {
        $fn = function() {
            var_dump(__CLASS__);
            var_dump(self::class);
        };
        $fn->bindTo(new A, 'A')();
    }
}
(new B)->test();

Here __CLASS__ always refers to the lexical scope, which will be "B", as the closure was declared lexically within class B. However, this does not match the self scope of the closure. self::class will return "A" in this example.

@derickr
Copy link
Member

derickr commented Dec 18, 2023

Are anonymous classes really part of a namespace? Shouldn't they be "anonymous" and hence not even have any name logically?

@mvorisek
Copy link
Contributor

The namespace is not about only naming, but also about how names not startwing with \ are resolved. In anonymous classes, there are resolved (correctly) using the actual namespace. The same for PHPDoc.

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

Successfully merging this pull request may close these issues.

Incorrect ReflectionClass::getNamespaceName for anonymous class
5 participants