- 
                Notifications
    You must be signed in to change notification settings 
- Fork 8k
Add #[NonpublicConstructor] attribute #17846
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
base: master
Are you sure you want to change the base?
Conversation
| @DanielEScherzer Thank you for this PR! I have a question though, how hard would it be to just generalize this to  | 
| Do we really need this at all? Can we not simply error in  | 
| @iluuu1994 Sure, if your goal is to only cover internal, and not provide information about the right way. But it makes also for a nice attribute for userland. | 
|  | ||
| uint32_t zend_attribute_attribute_get_flags(zend_attribute *attr, zend_class_entry *scope); | ||
|  | ||
| ZEND_API ZEND_COLD zend_result ZEND_FASTCALL zend_attribute_get_nonpublic_suffix(HashTable *attributes, zend_string **message_suffix); | 
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.
Does it actually need to be a fastcall? As far as I saw, it's not on any hotpath
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.
ZEND_FASTCALL changes the calling convention, rather than improving performance. It should be present only on functions called from the JIT.
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.
OK, I thought that passing arguments in registers is useful for improving performance.
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.
ZEND_FASTCALL doesn't have an effect on x86-64 anymore, where register arg passing has become the default. But yes, I suppose ZEND_FASTCALL could still be faster on 32-bit.
| zend_throw_exception_ex( | ||
| reflection_exception_ptr, | ||
| 0, | ||
| "Access to non-public constructor of class %s%s", | 
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.
Nit: I'd probably prefer if this error message was similar to what is used elsewhere:
Cannot call %s constructor of class %s%s
There's also another variant of the same error message which is  Call to private ..., so I think the best would be to choose only one.
|  | ||
| /* The assignment might fail due to 'readonly'. */ | ||
| if (UNEXPECTED(EG(exception))) { | ||
| RETURN_THROWS(); | 
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.
A test for this path would be nice to have
| zend_string *message_suffix = ZSTR_EMPTY_ALLOC(); | ||
|  | ||
| if (zend_attribute_get_nonpublic_suffix(constructor->common.attributes, &message_suffix) == FAILURE) { | ||
| return; | 
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.
could you please add the same assertion here that you added everywhere else when the above operation fails?
| 
 Accessing private members through reflection was never considered public API, so is this really a problem worth solving? Preventing memory corruption for internal classes is a different story. | 
| @iluuu1994 I'm not talking about reflection here. I'm talking about  This whole feature is, IMHO, not mainly about reflection. That we can use it for the specific case of private internal constructors is nice, but not the main thing which this feature must solve for me. | 
| This was originally just intended for the internal constructors so that we kept a helpful error message, but I see how it might be useful in userland too - let me work on this for a bit,  | 
| 
 That seems like a rather niche scenario. There are also solutions to this. For example, instead of making the method private, it can be internally renamed and the original method can throw an exception. Normally, such methods would be deprecated first anyway, in which case you can already provide a message for the user. /cc @TimWolla Pinging you as it's related to  | 
| I agree with Ilija. For public methods there is  And for the private constructor case, if you use Reflection to touch private members, you already need to check the code of the library in question to find out how to do safely - which is power-user territory. Adding an entirely new feature just to emit nicer error messages in a power-user API doesn't feel worthwhile. | 
| How is this related to  The deprecated attribute is something you apply before a breaking change like this. Not after. This is a different use case. | 
| Before you outright remove or hide (through  | 
| Yep, you'll Deprecate first, then change the annotation to NonpublicMethod when you actually remove it. | 
No description provided.