Module: StateMachines::EvalHelpers
- Included in:
- Branch, Callback, Machine, StateContext
- Defined in:
- lib/state_machines/eval_helpers.rb
Overview
Provides a set of helper methods for evaluating methods within the context of an object.
Instance Method Summary collapse
-
#evaluate_method(object, method, *args, **kwargs, &block) ⇒ Object
Evaluates one of several different types of methods within the context of the given object.
Instance Method Details
#evaluate_method(object, method, *args, **kwargs, &block) ⇒ Object
Evaluates one of several different types of methods within the context of the given object. Methods can be one of the following types:
-
Symbol
-
Method / Proc
-
String
Examples
Below are examples of the various ways that a method can be evaluated on an object:
class Person
def initialize(name)
@name = name
end
def name
@name
end
end
class PersonCallback
def self.run(person)
person.name
end
end
person = Person.new('John Smith')
evaluate_method(person, :name) # => "John Smith"
evaluate_method(person, PersonCallback.method(:run)) # => "John Smith"
evaluate_method(person, Proc.new {|person| person.name}) # => "John Smith"
evaluate_method(person, lambda {|person| person.name}) # => "John Smith"
evaluate_method(person, '@name') # => "John Smith"
Additional arguments
Additional arguments can be passed to the methods being evaluated. If the method defines additional arguments other than the object context, then all arguments are required.
For example,
person = Person.new('John Smith')
evaluate_method(person, lambda {|person| person.name}, 21) # => "John Smith"
evaluate_method(person, lambda {|person, age| "#{person.name} is #{age}"}, 21) # => "John Smith is 21"
evaluate_method(person, lambda {|person, age| "#{person.name} is #{age}"}, 21, 'male') # => ArgumentError: wrong number of arguments (3 for 2)
55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 |
# File 'lib/state_machines/eval_helpers.rb', line 55 def evaluate_method(object, method, *args, **kwargs, &block) case method when Symbol klass = (class << object; self; end) args = [] if (klass.method_defined?(method) || klass.private_method_defined?(method)) && object.method(method).arity == 0 object.send(method, *args, **kwargs, &block) when Proc args.unshift(object) arity = method.arity # Handle blocks for Procs if block_given? && arity != 0 if [1, 2].include?(arity) # Force the block to be either the only argument or the 2nd one # after the object (may mean additional arguments get discarded) args = args[0, arity - 1] + [block] else # Tack the block to the end of the args args << block end else # These method types are only called with 0, 1, or n arguments args = args[0, arity] if [0, 1].include?(arity) end # Call the Proc with the arguments method.call(*args, **kwargs) when Method args.unshift(object) arity = method.arity # Methods handle blocks via &block, not as arguments # Only limit arguments if necessary based on arity args = args[0, arity] if [0, 1].include?(arity) # Call the Method with the arguments and pass the block method.call(*args, **kwargs, &block) when String if block_given? if StateMachines::Transition.pause_supported? eval(method, object.instance_eval { binding }, &block) else # Support for JRuby and Truffle Ruby, which don't support binding blocks eigen = class << object; self; end eigen.class_eval <<-RUBY, __FILE__, __LINE__ + 1 def __temp_eval_method__(*args, &b) #{method} end RUBY result = object.__temp_eval_method__(*args, &block) eigen.send(:remove_method, :__temp_eval_method__) result end else eval(method, object.instance_eval { binding }) end else raise ArgumentError, 'Methods must be a symbol denoting the method to call, a block to be invoked, or a string to be evaluated' end end |