-
Notifications
You must be signed in to change notification settings - Fork 7.8k
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
base: PHP-8.2
Are you sure you want to change the base?
Conversation
…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) { |
There was a problem hiding this comment.
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
There was a problem hiding this 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.
It's not super clear what the behavior for anonymous classes should actually be -- I could see either returning the namespace portion before the |
Returning the namespace portion before |
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 ( |
Also worth noting that |
Right, it should be consistent with at least |
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 A {}
class B {
public function test() {
$fn = function() {
var_dump(__CLASS__);
var_dump(self::class);
};
$fn->bindTo(new A, 'A')();
}
}
(new B)->test(); Here |
Are anonymous classes really part of a namespace? Shouldn't they be "anonymous" and hence not even have any name logically? |
The namespace is not about only naming, but also about how names not startwing with |
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:
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.