-
-
Notifications
You must be signed in to change notification settings - Fork 404
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
Add helper to handle method decorators #773
Conversation
If we're going to add this, I'd like to see it added consistently across all visibility methods. That said, is this actually a common idiom? |
Method definition decorators were a major feature made possible in Ruby 2.1.0, so in general yes, that part is a common idiom (a project I’m a developer for makes significant use of this feature). Specifically pertaining to Based on what I learned creating this patch, there are two distinct tasks that the handler for
Ideally the class DecoratorExample
# Add functionality to a method.
def self.my_decorator method
imeth = instance_method method
define_method method do |*args, &block|
"does stuff to " +
imeth.bind(self).call(*args, &block)
end
end
# Returns a description of itself.
my_decorator def foo
'the foo method'
end
end
DecoratorExample.new.foo
# => “does stuff to the foo method” In this case class method So yes, I would agree with you that all the visibility methods that can behave like this should be updated. However, without a mechanism for handling decorators in general, I could see a fair amount of code duplication happening, and I think part of the problem would be getting solved in the wrong place. What’s the best way to proceed at this point? Should I open a ticket for discussion on implementing a way to handle decorators in general? |
YARD has a generalized mechanism for handling "decorators" through handlers / plugins. In other words, it should be done the same way we are doing it now. Duplication can be dealt with by putting common behavior in mixins that are shared across handlers. YARD should be supporting the builtin methods out of the box, but not generically handling all "decorators". That's the job of third-party plugins to provide handlers for custom DSL code. Basically, what we need to do is update YARD to handle this new "methods can be expressions passed to visibility declarations" syntax, and let third-party plugins handle the cases that are not part of the core language. |
I did some more research and experimentation, and I think we’re on the same page. I was using the term “decorator” a little too liberally before; the problem I saw was specifically with decorators in front of a method definition. I thought there was a more general problem caused by the method definition being shadowed in the AST by the decorator, but I may be wrong about this. I can bring that problem back up later if it does turn out to be an issue. Regarding handling the built-in visibility methods, yes, I think that makes sense. Regarding custom decorators being handled through the existing mechanisms by third-party plugins, yes, I agree. The implementation of "methods can be expressions passed to visibility declarations" (or passed to any method definition decorator in general?), sounds like it may fall into a whole different scope of work that involves some design decisions. Is this something I should actually try to do myself, or should I wait for you or another major maintainer to implement the functionality, then piggyback the visibility methods off that? |
I looked at this from a different perspective and started working on an implementation. It's easier than I thought. |
I've created a DecoratorHandlerMethods mixin at your recommendation and it seems to have solved a lot of the problems. YARD's built-in visibility handlers that can decorate method definitions have been refactored using this mixin. DSL handlers are completely left up to third-parties to implement, with an opt-in to include the mixin so they don't have to roll their own support for decorator chaining, method defintion parsing, and such. This PR should be ready to go if you approve. |
Any plans to merge this? |
I can update this branch if there is acceptance to it being merged in. |
I realize that the new composable "syntax" (driven by methods returning symbols) is something that a lot of developers want to begin using with YARD. I have no objections to merging this in theory, but there is quite a lot of churn in the commits that make it hard to follow. Specifically I want to ensure that:
If you can confirm these things I'm okay with taking a deeper dive on this and getting it in. It would also help to re-summarize what this PR is doing so we're all clear what the functionality is (re: point 3). There was some confusion mid-way through because the original patch was for something very different, and the original PR title / description is (or should be) obsolete. |
It's been so long that I need to review the PR as well, but I believe that all three points are ok. I can also try to squash some of the commits while I'm at it. |
1ed60a2
to
534e132
Compare
@lsegal I've fixed the merge conflicts and squashed the commits. Summary: This PR adds the opt-in module For example, given the decorator class Test
# The foo decorator.
def self.foo(method)
"foo " + new.send(method)
end
# The bar method.
# @return [String] bar modified by {foo}
foo def bar
"bar"
end
end A handler can be created: class Handler < YARD::Handlers::Ruby::Base
include YARD::Handlers::Ruby::DecoratorHandlerMethods
handles method_call(:foo)
def process
process_decorator
end
end And the docstring is passed through to
process do
# Set visibility of a class method to private.
process_decorator :scope => :class do |method|
method.visibility = :private if method.respond_to? :visibility=
end
end |
# | ||
# @!method process_decorator(*nodes, opts={}, &block) | ||
def process_decorator(*nodes, &block) | ||
opts = nodes.last.is_a?(Hash) ? nodes.pop : {} |
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.
Would it make more sense here to use .respond_to?(:[])
instead of .is_a?(Hash)
? What if the implementation changes to use a Hash-like object not descended from Hash?
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.
That seems unlikely and would break the API in all sorts of other ways. The is-a check is sufficient, and []
may not be the only thing our implementation expects from the node argument, so validating that it's a Hash seems correct to me.
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.
Fair enough.
I finally had time to look through this PR (sorry for the delay!). In general I think it's good. I threw in some brief comments and questions inline for most of these, but I have some generalized comments: The More importantly though, I'd like to understand what transferring the docstring actually does. I found this to be the hardest part of the code to understand. I see that you are copying over the unparsed docstring data from the statement object, which I understand to be necessary (since the underlying method declaration will not have picked up the docstring data), but you then copy over the Those were my two main issues with this. With the outstanding comments and questions resolved, I think it's getting close to being merged. Thanks for working through all of this, @amclain! |
cb8b8ff
to
7dcb92e
Compare
The comments disappeared after I pushed some changes. Here's the link to the changeset with them: |
For context I'm going to add follow ups to the original commit, please follow along there. |
Added extra comments, please follow up on any of the comment threads that have not yet been responded to. Thanks! Another high-level design point I forgot to bring up: How does your chaining algorithm handle evaluation order of decorators? Specifically, the evaluation order with which decorators are used is important to get right. To be correct, your helper would have to handle decorators from right to left, not left to right, since Ruby evaluation would happen in RTL. Consider two decorators, cpub cpriv def foo; end The above code should actually cause For YARD to be correct, it would have to emulate this order of evaluation. In my glance at the test suite, I only see one indirect test for order and it looks like it's validating LTR behavior. To summarize: the behavior of chaining should evaluate right-to-left, and we should add explicit test cases for this scenario. |
This took a while, but it's in. Thanks for the hard work on this, @amclain! |
Awesome, thanks @lsegal! |
Expanding on the work of PR #767 for issue #760, I've added the ability for private_class_method to be handled correctly when used as a method definition decorator, which is valid syntax as of Ruby 2.1.0.