Sha256: 416dc629dac19eaee567de7b389c8c9f2e8b9e2b3f13ad1305c54432b96cceb8

Contents?: true

Size: 1.59 KB

Versions: 21

Compression:

Stored size: 1.59 KB

Contents

require 'backports/tools/alias_method_chain'

module Backports
  # Metaprogramming utility to make block optional.
  # Tests first if block is already optional when given options
  def self.make_block_optional(mod, *methods)
    mod = class << mod; self; end unless mod.is_a? Module
    options = methods.last.is_a?(Hash) ? methods.pop : {}
    methods.each do |selector|
      unless mod.method_defined? selector
        warn "#{mod}##{selector} is not defined, so block can't be made optional"
        next
      end
      unless options[:force]
        # Check if needed
        test_on = options.fetch(:test_on)
        result =  begin
                    test_on.send(selector, *options.fetch(:arg, []))
                  rescue LocalJumpError
                    false
                  end
        next if result.class.name =~ /Enumerator$/
      end
      require 'enumerator'
      arity = mod.instance_method(selector).arity
      last_arg = []
      if arity < 0
        last_arg = ["*rest"]
        arity = -1-arity
      end
      arg_sequence = ((0...arity).map{|i| "arg_#{i}"} + last_arg + ["&block"]).join(", ")

      alias_method_chain(mod, selector, :optional_block) do |aliased_target, punctuation|
        mod.module_eval <<-end_eval, __FILE__, __LINE__ + 1
          def #{aliased_target}_with_optional_block#{punctuation}(#{arg_sequence})
            return to_enum(:#{aliased_target}_without_optional_block#{punctuation}, #{arg_sequence}) unless block_given?
            #{aliased_target}_without_optional_block#{punctuation}(#{arg_sequence})
          end
        end_eval
      end
    end
  end
end

Version data entries

21 entries across 21 versions & 2 rubygems

Version Path
backports-3.11.3 lib/backports/tools/make_block_optional.rb
backports-3.11.2 lib/backports/tools/make_block_optional.rb
backports-3.11.1 lib/backports/tools/make_block_optional.rb
backports-3.11.0 lib/backports/tools/make_block_optional.rb
backports-3.10.3 lib/backports/tools/make_block_optional.rb
backports-3.10.2 lib/backports/tools/make_block_optional.rb
backports-3.10.1 lib/backports/tools/make_block_optional.rb
backports-3.10.0 lib/backports/tools/make_block_optional.rb
backports-3.9.1 lib/backports/tools/make_block_optional.rb
backports-3.9.0 lib/backports/tools/make_block_optional.rb
backports-3.8.0 lib/backports/tools/make_block_optional.rb
backports-3.7.0 lib/backports/tools/make_block_optional.rb
mdg-1.0.1 vendor/bundle/ruby/2.3.0/gems/backports-3.6.8/lib/backports/tools/make_block_optional.rb
backports-3.6.8 lib/backports/tools/make_block_optional.rb
backports-3.6.7 lib/backports/tools/make_block_optional.rb
backports-3.6.6 lib/backports/tools/make_block_optional.rb
backports-3.6.5 lib/backports/tools/make_block_optional.rb
backports-3.6.4 lib/backports/tools/make_block_optional.rb
backports-3.6.3 lib/backports/tools/make_block_optional.rb
backports-3.6.2 lib/backports/tools/make_block_optional.rb