magic-help 0.20170204
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/lib/magic/help.rb +104 -0
- data/lib/magic/help/irb.rb +28 -0
- data/lib/magic/help/set_trace_func.rb +109 -0
- data/lib/magic/help/tracepoint.rb +79 -0
- data/lib/magic_help.rb +1 -0
- data/test/mass_test.rb +206 -0
- data/test/tc_magic_help.rb +240 -0
- metadata +79 -0
    
        checksums.yaml
    ADDED
    
    | @@ -0,0 +1,7 @@ | |
| 1 | 
            +
            ---
         | 
| 2 | 
            +
            SHA1:
         | 
| 3 | 
            +
              metadata.gz: 59db211c23ba22dab20b3e5b3e16e3303dcfe1dd
         | 
| 4 | 
            +
              data.tar.gz: b50b3dae0566f1807bd297df5d3113c0830f7ef6
         | 
| 5 | 
            +
            SHA512:
         | 
| 6 | 
            +
              metadata.gz: 5523804800dc7e7733e12e775900ce47fb7a568608a27abc3c78ae43157f6f285cb2f1133e0d302360e20da31a21a946b4861328f1047603c0d244679ce2e602
         | 
| 7 | 
            +
              data.tar.gz: 101f7ccdfd34ddc90287b470f295c16dcbb2de5fc1041fa6bf861576e5e60d1306a0951f2bc57201455a14e4c38383d8b4cc18ef9b2310a2cba38e2356a75400
         | 
    
        data/lib/magic/help.rb
    ADDED
    
    | @@ -0,0 +1,104 @@ | |
| 1 | 
            +
            require_relative "help/irb"
         | 
| 2 | 
            +
            require_relative "help/tracepoint"
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            module Magic
         | 
| 5 | 
            +
              module Help
         | 
| 6 | 
            +
                def self.help_method_extract(m)
         | 
| 7 | 
            +
                  unless m.inspect =~ %r[\A#<(?:Unbound)?Method: (.*?)>\Z]
         | 
| 8 | 
            +
                    raise "Cannot parse result of #{m.class}#inspect: #{m.inspect}"
         | 
| 9 | 
            +
                  end
         | 
| 10 | 
            +
                  $1.sub(/\A.*?\((.*?)\)(.*)\Z/){ "#{$1}#{$2}" }.sub(/\./, "::").sub(/#<Class:(.*?)>#/) { "#{$1}::" }
         | 
| 11 | 
            +
                end
         | 
| 12 | 
            +
             | 
| 13 | 
            +
                # Magic::Help.postprocess is used to postprocess queries in two cases:
         | 
| 14 | 
            +
                # * help "Foo.bar" queries - res defined, more hacks
         | 
| 15 | 
            +
                # * help { Foo.bar } queries - res not defined, fewer hacks
         | 
| 16 | 
            +
                def self.postprocess(m, res=nil)
         | 
| 17 | 
            +
                  # Kernel#method_missing here means class was found but method wasn't
         | 
| 18 | 
            +
                  # It is possible that such method exists, it was simply not included.
         | 
| 19 | 
            +
                  # Example - Time::rfc2822 from time.rb.
         | 
| 20 | 
            +
                  #
         | 
| 21 | 
            +
                  # Do not correct it if actual method_missing was called.
         | 
| 22 | 
            +
                  if res and m == "Kernel#method_missing"
         | 
| 23 | 
            +
                      m = res unless res =~ /\A(?:.*)(?:\#|::|\.)method_missing\Z/
         | 
| 24 | 
            +
                  # Most classes do not override Foo::new, but provide new documentation anyway !
         | 
| 25 | 
            +
                  # Two cases are possible
         | 
| 26 | 
            +
                  # * Class#new is used
         | 
| 27 | 
            +
                  # * Bar::new is used, for Bar being some ancestor of Foo
         | 
| 28 | 
            +
                  elsif res and (m =~ /\A(.*)\#new\Z/ or m =~ /\A(.*)::new\Z/)
         | 
| 29 | 
            +
                    cls = $1
         | 
| 30 | 
            +
                    # Do not correct requests for Foo#new
         | 
| 31 | 
            +
                    # If Foo#new became Class#new, it must have been
         | 
| 32 | 
            +
                    # by some evil metaclass hackery.
         | 
| 33 | 
            +
                    #
         | 
| 34 | 
            +
                    # Foo.new or Foo::new both become Foo::new
         | 
| 35 | 
            +
                    if res =~ /\A(.*)(::|\.)new\Z/
         | 
| 36 | 
            +
                      cls_requested, k = $1, $2
         | 
| 37 | 
            +
                      # Condition just to get "Class#new" working correctly
         | 
| 38 | 
            +
                      # Otherwise it would be changed to "Class::new"
         | 
| 39 | 
            +
                      m = "#{cls_requested}::new" unless cls == cls_requested
         | 
| 40 | 
            +
                    end
         | 
| 41 | 
            +
                  end
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                  m
         | 
| 44 | 
            +
                end
         | 
| 45 | 
            +
             | 
| 46 | 
            +
                def self.resolve_help_res(res)
         | 
| 47 | 
            +
                  query = case res
         | 
| 48 | 
            +
                  when Module
         | 
| 49 | 
            +
                    res.to_s
         | 
| 50 | 
            +
                  when UnboundMethod, Method
         | 
| 51 | 
            +
                    help_method_extract(res)
         | 
| 52 | 
            +
                  when /\A(.*)(#|::|\.)(.*)\Z/
         | 
| 53 | 
            +
                    cp, k, m = $1, $2, $3
         | 
| 54 | 
            +
                    begin
         | 
| 55 | 
            +
                      # For multielement paths like File::Stat const_get must be
         | 
| 56 | 
            +
                      # called multiple times, that is:
         | 
| 57 | 
            +
                      # Object.const_get("File").const_get("Stat")
         | 
| 58 | 
            +
                      cls = cp.split(/::/).inject(Object){|c, path_elem| c.const_get(path_elem) }
         | 
| 59 | 
            +
                      case k
         | 
| 60 | 
            +
                      when "#"
         | 
| 61 | 
            +
                        m = cls.instance_method(m)
         | 
| 62 | 
            +
                        m = help_method_extract(m)
         | 
| 63 | 
            +
                      when "::"
         | 
| 64 | 
            +
                        m = cls.method(m)
         | 
| 65 | 
            +
                        # Make sure a module method is returned
         | 
| 66 | 
            +
                        # It fixes `Class::new' resolving to `Class#new'
         | 
| 67 | 
            +
                        # (Class::new and Class#new are the same thing,
         | 
| 68 | 
            +
                        # but their documentations are different)
         | 
| 69 | 
            +
                        m = help_method_extract(m)
         | 
| 70 | 
            +
                        m = m.sub(/\#/, "::") if cls == Class && m == "Class#new"
         | 
| 71 | 
            +
                      when "."
         | 
| 72 | 
            +
                        begin
         | 
| 73 | 
            +
                          m = cls.instance_method(m)
         | 
| 74 | 
            +
                        rescue NameError
         | 
| 75 | 
            +
                          m = cls.method(m)
         | 
| 76 | 
            +
                        end
         | 
| 77 | 
            +
                        m = help_method_extract(m)
         | 
| 78 | 
            +
                      end
         | 
| 79 | 
            +
                      postprocess(m, res)
         | 
| 80 | 
            +
                    rescue NameError
         | 
| 81 | 
            +
                      res
         | 
| 82 | 
            +
                    end
         | 
| 83 | 
            +
                  when String
         | 
| 84 | 
            +
                    res
         | 
| 85 | 
            +
                  else
         | 
| 86 | 
            +
                    res.class.to_s
         | 
| 87 | 
            +
                  end
         | 
| 88 | 
            +
                  query
         | 
| 89 | 
            +
                end
         | 
| 90 | 
            +
             | 
| 91 | 
            +
                def self.resolve_help_query(*args, &block)
         | 
| 92 | 
            +
                  raise ArgumentError, "wrong number of arguments (#{args.size} for 1)" if args.size > 1
         | 
| 93 | 
            +
                  raise ArgumentError, "help cannot take both arguments and block" if args.size > 0 and block_given?
         | 
| 94 | 
            +
                  if block_given?
         | 
| 95 | 
            +
                    resolve_help_block(&block)
         | 
| 96 | 
            +
                  elsif args.empty?
         | 
| 97 | 
            +
                    # No block, no arguments
         | 
| 98 | 
            +
                    nil
         | 
| 99 | 
            +
                  else
         | 
| 100 | 
            +
                    resolve_help_res(args[0])
         | 
| 101 | 
            +
                  end
         | 
| 102 | 
            +
                end
         | 
| 103 | 
            +
              end
         | 
| 104 | 
            +
            end
         | 
| @@ -0,0 +1,28 @@ | |
| 1 | 
            +
            require "irb"
         | 
| 2 | 
            +
            require "irb/extend-command"
         | 
| 3 | 
            +
             | 
| 4 | 
            +
            # Now let's hack irb not to alias irb_help -> help
         | 
| 5 | 
            +
            # It saves us a silly warning at startup:
         | 
| 6 | 
            +
            #     irb: warn: can't alias help from irb_help.
         | 
| 7 | 
            +
            module IRB::ExtendCommandBundle # :nodoc:
         | 
| 8 | 
            +
              @ALIASES.delete_if{|a| a == [:help, :irb_help, NO_OVERRIDE]}
         | 
| 9 | 
            +
            end
         | 
| 10 | 
            +
             | 
| 11 | 
            +
            # help is a Do-What-I-Mean help function.
         | 
| 12 | 
            +
            # It can be called with either a block or a single argument.
         | 
| 13 | 
            +
            # When called with single argument, it behaves like normal
         | 
| 14 | 
            +
            # help function, except for being much smarter:
         | 
| 15 | 
            +
            #
         | 
| 16 | 
            +
            #  help "Array"         - help on Array
         | 
| 17 | 
            +
            #  help "Array#sort"    - help on Array#sort
         | 
| 18 | 
            +
            #  help "File#sync="    - help on IO#sync=
         | 
| 19 | 
            +
            #
         | 
| 20 | 
            +
            #  help { [].sort }     - help on Array#sort
         | 
| 21 | 
            +
            #  help { obj.foo = 1 } - help on obj.foo=
         | 
| 22 | 
            +
            #  help { Array }       - help on Array
         | 
| 23 | 
            +
            #  help { [] }          - help on Array
         | 
| 24 | 
            +
            #  help { Dir["*"] }    - help on Dir::[]
         | 
| 25 | 
            +
            def help(*args, &block)
         | 
| 26 | 
            +
              query = Magic::Help.resolve_help_query(*args, &block)
         | 
| 27 | 
            +
              irb_help(query) if query
         | 
| 28 | 
            +
            end
         | 
| @@ -0,0 +1,109 @@ | |
| 1 | 
            +
            # This is 1.8 API, it's still there in 2.0 but they broke it a bit
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module Magic
         | 
| 4 | 
            +
              module Help
         | 
| 5 | 
            +
                def self.resolve_help_block(&block)
         | 
| 6 | 
            +
                  call_event = nil
         | 
| 7 | 
            +
                  res = nil
         | 
| 8 | 
            +
                  done = false
         | 
| 9 | 
            +
                  res_mm = nil
         | 
| 10 | 
            +
                  argument_error = false
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  # We want to capture calls to method_missing too
         | 
| 13 | 
            +
                  original_method_missing = BasicObject.instance_method(:method_missing)
         | 
| 14 | 
            +
                  BasicObject.class_eval {
         | 
| 15 | 
            +
                    def method_missing(*args)
         | 
| 16 | 
            +
                      throw :magically_irb_helped, [self, *args]
         | 
| 17 | 
            +
                    end
         | 
| 18 | 
            +
                  }
         | 
| 19 | 
            +
             | 
| 20 | 
            +
                  tf = lambda{|*xargs|
         | 
| 21 | 
            +
                    return if done
         | 
| 22 | 
            +
                    event = xargs[0]
         | 
| 23 | 
            +
                    if argument_error
         | 
| 24 | 
            +
                      if event == "return"
         | 
| 25 | 
            +
                        done = true
         | 
| 26 | 
            +
                        # For functions called with wrong number of arguments,
         | 
| 27 | 
            +
                        # call event is not generated (function is never called),
         | 
| 28 | 
            +
                        # but the return event is !
         | 
| 29 | 
            +
                        call_event = xargs
         | 
| 30 | 
            +
                        throw :magically_irb_helped
         | 
| 31 | 
            +
                      end
         | 
| 32 | 
            +
                    elsif event == "call" or event == "c-call"
         | 
| 33 | 
            +
                      call_event = xargs
         | 
| 34 | 
            +
                      if call_event.values_at(0, 3, 5) == ["c-call", :new, Class] and
         | 
| 35 | 
            +
                        eval("self", call_event[4]) == ArgumentError
         | 
| 36 | 
            +
                        argument_error = true
         | 
| 37 | 
            +
                      else
         | 
| 38 | 
            +
                        done = true
         | 
| 39 | 
            +
                        # Let Kernel#method_missing run, otherwise throw
         | 
| 40 | 
            +
                        unless call_event.values_at(0, 3, 5) == ["call", :method_missing, BasicObject]
         | 
| 41 | 
            +
                          throw :magically_irb_helped
         | 
| 42 | 
            +
                        end
         | 
| 43 | 
            +
                      end
         | 
| 44 | 
            +
                    end
         | 
| 45 | 
            +
                  }
         | 
| 46 | 
            +
                  res_mm = catch(:magically_irb_helped) {
         | 
| 47 | 
            +
                    set_trace_func tf
         | 
| 48 | 
            +
                    res = yield()
         | 
| 49 | 
            +
                    done = true
         | 
| 50 | 
            +
                    nil
         | 
| 51 | 
            +
                  }
         | 
| 52 | 
            +
                  done = true
         | 
| 53 | 
            +
                  set_trace_func nil
         | 
| 54 | 
            +
                  BasicObject.instance_eval {
         | 
| 55 | 
            +
                    define_method(:method_missing, original_method_missing)
         | 
| 56 | 
            +
                    # It was originally private, restore it as such
         | 
| 57 | 
            +
                    private :method_missing
         | 
| 58 | 
            +
                  }
         | 
| 59 | 
            +
                  # Handle captured method_missing
         | 
| 60 | 
            +
             | 
| 61 | 
            +
                  if res_mm
         | 
| 62 | 
            +
                    # This is explicit Foo#method_missing:
         | 
| 63 | 
            +
                    # It shouldn't really be allowed as it's private.
         | 
| 64 | 
            +
                    if res_mm.size == 1
         | 
| 65 | 
            +
                      return "Kernel#method_missing"
         | 
| 66 | 
            +
                    else
         | 
| 67 | 
            +
                      bound_self, meth = *res_mm
         | 
| 68 | 
            +
                      # method_missing is called if:
         | 
| 69 | 
            +
                      # * there was no such method
         | 
| 70 | 
            +
                      # * there was a method, but a private one only
         | 
| 71 | 
            +
                      begin
         | 
| 72 | 
            +
                        # Surprise ! A private method !
         | 
| 73 | 
            +
                        m = bound_self.method(meth)
         | 
| 74 | 
            +
                        query = help_method_extract(m)
         | 
| 75 | 
            +
                      rescue NameError
         | 
| 76 | 
            +
                        # No such method
         | 
| 77 | 
            +
                        if bound_self.is_a? Class
         | 
| 78 | 
            +
                          query = "#{bound_self}::#{meth}"
         | 
| 79 | 
            +
                        else
         | 
| 80 | 
            +
                          query = "#{bound_self.class}.#{meth}"
         | 
| 81 | 
            +
                        end
         | 
| 82 | 
            +
                      end
         | 
| 83 | 
            +
                    end
         | 
| 84 | 
            +
                    postprocess(query)
         | 
| 85 | 
            +
                  # Handle normal call events
         | 
| 86 | 
            +
                  elsif call_event
         | 
| 87 | 
            +
                    meth, bind, cls = call_event[3], call_event[4], call_event[5]
         | 
| 88 | 
            +
                    bound_self = bind.receiver
         | 
| 89 | 
            +
                    if meth == :new
         | 
| 90 | 
            +
                      cls = bound_self
         | 
| 91 | 
            +
                    end
         | 
| 92 | 
            +
             | 
| 93 | 
            +
                    # Foo::bar and Foo#bar look the same
         | 
| 94 | 
            +
                    # Check whether self == Foo to tell them apart
         | 
| 95 | 
            +
                    #
         | 
| 96 | 
            +
                    # Only Class::new is documented as such,
         | 
| 97 | 
            +
                    # all other Class::foo are really Class#foo
         | 
| 98 | 
            +
                    if bound_self == cls && (bound_self != Class || meth == :new)
         | 
| 99 | 
            +
                      query = "#{cls}::#{meth}"
         | 
| 100 | 
            +
                    else
         | 
| 101 | 
            +
                      query = "#{cls}##{meth}"
         | 
| 102 | 
            +
                    end
         | 
| 103 | 
            +
                    postprocess(query)
         | 
| 104 | 
            +
                  else
         | 
| 105 | 
            +
                    resolve_help_res(res)
         | 
| 106 | 
            +
                  end
         | 
| 107 | 
            +
                end
         | 
| 108 | 
            +
              end
         | 
| 109 | 
            +
            end
         | 
| @@ -0,0 +1,79 @@ | |
| 1 | 
            +
            module Magic
         | 
| 2 | 
            +
              module Help
         | 
| 3 | 
            +
                def self.resolve_help_block(&block)
         | 
| 4 | 
            +
                  call_event = nil
         | 
| 5 | 
            +
                  done = false
         | 
| 6 | 
            +
                  res = nil
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  # We want to capture calls to method_missing too
         | 
| 9 | 
            +
                  # This should be available via TracePoint but isn't
         | 
| 10 | 
            +
                  original_method_missing = BasicObject.instance_method(:method_missing)
         | 
| 11 | 
            +
                  BasicObject.class_eval do
         | 
| 12 | 
            +
                    define_method(:method_missing) do |*args|
         | 
| 13 | 
            +
                      if args.empty?
         | 
| 14 | 
            +
                        # This is presumably called on self, and without arguments
         | 
| 15 | 
            +
                        throw :done, {cls: method(:method_missing).owner, meth: :method_missing, self: self}
         | 
| 16 | 
            +
                      end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                      if self.is_a?(Class)
         | 
| 19 | 
            +
                        throw :done, {cls: self.singleton_class, meth: args[0], self: self}
         | 
| 20 | 
            +
                      else
         | 
| 21 | 
            +
                        throw :done, {cls: self.class, meth: args[0], self: self}
         | 
| 22 | 
            +
                      end
         | 
| 23 | 
            +
                    end
         | 
| 24 | 
            +
                  end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                  trace = TracePoint.new do |ev|
         | 
| 27 | 
            +
                    next if done
         | 
| 28 | 
            +
                    case ev.event
         | 
| 29 | 
            +
                    when :call, :c_call
         | 
| 30 | 
            +
                      if ev.defined_class == BasicObject and ev.method_id == :method_missing
         | 
| 31 | 
            +
                        done = true
         | 
| 32 | 
            +
                        # Let it reach our special handler
         | 
| 33 | 
            +
                      # elsif ev.self == ArgumentError and ev.method_id == :new
         | 
| 34 | 
            +
                        # Function was called without full number of arguments
         | 
| 35 | 
            +
                        # There doesn't seem to be any way to recover from this in ruby 2.x
         | 
| 36 | 
            +
                        # In 1.8 we'd get extra return event
         | 
| 37 | 
            +
                        #
         | 
| 38 | 
            +
                        # It's possible to hack argument name from stack trace,
         | 
| 39 | 
            +
                        # (with massive hacking)
         | 
| 40 | 
            +
                        # but not self/class, so it's not most useful
         | 
| 41 | 
            +
                      else
         | 
| 42 | 
            +
                        done = true
         | 
| 43 | 
            +
                        throw :done, {cls: ev.defined_class, meth: ev.method_id, self: ev.self}
         | 
| 44 | 
            +
                      end
         | 
| 45 | 
            +
                    else
         | 
| 46 | 
            +
                      # Ignore everything eles
         | 
| 47 | 
            +
                    end
         | 
| 48 | 
            +
                  end
         | 
| 49 | 
            +
                  call_event = catch(:done) do
         | 
| 50 | 
            +
                    trace.enable
         | 
| 51 | 
            +
                    res = yield
         | 
| 52 | 
            +
                    nil
         | 
| 53 | 
            +
                  end
         | 
| 54 | 
            +
                  done = true
         | 
| 55 | 
            +
                  trace.disable
         | 
| 56 | 
            +
             | 
| 57 | 
            +
                  BasicObject.instance_eval do
         | 
| 58 | 
            +
                    define_method(:method_missing, original_method_missing)
         | 
| 59 | 
            +
                    # It was originally private, restore it as such
         | 
| 60 | 
            +
                    private :method_missing
         | 
| 61 | 
            +
                  end
         | 
| 62 | 
            +
             | 
| 63 | 
            +
                  if call_event
         | 
| 64 | 
            +
                    cls = call_event[:cls]
         | 
| 65 | 
            +
                    meth = call_event[:meth]
         | 
| 66 | 
            +
                    bound_self = call_event[:self]
         | 
| 67 | 
            +
                    is_singleton = (cls.is_a?(Class) and cls.singleton_class?)
         | 
| 68 | 
            +
                    if is_singleton or meth == :new
         | 
| 69 | 
            +
                      query = "#{bound_self}::#{meth}"
         | 
| 70 | 
            +
                    else
         | 
| 71 | 
            +
                      query = "#{cls}##{meth}"
         | 
| 72 | 
            +
                    end
         | 
| 73 | 
            +
                    postprocess(query)
         | 
| 74 | 
            +
                  else
         | 
| 75 | 
            +
                    resolve_help_res(res)
         | 
| 76 | 
            +
                  end
         | 
| 77 | 
            +
                end
         | 
| 78 | 
            +
              end
         | 
| 79 | 
            +
            end
         | 
    
        data/lib/magic_help.rb
    ADDED
    
    | @@ -0,0 +1 @@ | |
| 1 | 
            +
            require_relative "magic/help"
         | 
    
        data/test/mass_test.rb
    ADDED
    
    | @@ -0,0 +1,206 @@ | |
| 1 | 
            +
            #!/usr/bin/env ruby
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            # This only tests how many items magic/help BREAKS,
         | 
| 4 | 
            +
            # that is - how many would work otherwise.
         | 
| 5 | 
            +
             | 
| 6 | 
            +
            require_relative "../lib/magic_help"
         | 
| 7 | 
            +
            require "rdoc/ri"
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            # Use fake irb_help for testing
         | 
| 10 | 
            +
            $irb_help = nil
         | 
| 11 | 
            +
             | 
| 12 | 
            +
            class Object
         | 
| 13 | 
            +
              def irb_help(arg)
         | 
| 14 | 
            +
                $irb_help = arg
         | 
| 15 | 
            +
              end
         | 
| 16 | 
            +
            end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            # A convenience method
         | 
| 19 | 
            +
            def try_help(*args, &blk)
         | 
| 20 | 
            +
              $irb_help = nil
         | 
| 21 | 
            +
              help(*args, &blk)
         | 
| 22 | 
            +
              rv, $irb_help = $irb_help, nil
         | 
| 23 | 
            +
              rv
         | 
| 24 | 
            +
            end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
            # Extract documentation
         | 
| 27 | 
            +
             | 
| 28 | 
            +
            # FIXME: Hardcoded paths are not cross-platform compatible
         | 
| 29 | 
            +
            default_ri_root_path = "#{RDoc::RI::Paths::BASE}/system"
         | 
| 30 | 
            +
            ri_root_path = ARGV.shift || default_ri_root_path
         | 
| 31 | 
            +
             | 
| 32 | 
            +
            docs = Dir["#{ri_root_path}/**/*"]
         | 
| 33 | 
            +
             | 
| 34 | 
            +
            docs_class = []
         | 
| 35 | 
            +
            docs_imeth = []
         | 
| 36 | 
            +
            docs_cmeth = []
         | 
| 37 | 
            +
             | 
| 38 | 
            +
            docs.each{|fn|
         | 
| 39 | 
            +
              next if File.directory?(fn)
         | 
| 40 | 
            +
              raise "Weird path: #{fn}" unless fn[0, ri_root_path.size] == ri_root_path
         | 
| 41 | 
            +
              fn = fn[ri_root_path.size..-1].sub(/^\/*/, "")
         | 
| 42 | 
            +
              # gsub after split to deal with Fixnum#/ etc.
         | 
| 43 | 
            +
              path = fn.split(/\//).map{|x| x.gsub(/%(..)/){$1.hex.chr}}
         | 
| 44 | 
            +
              case path[-1]
         | 
| 45 | 
            +
              when /\A(.*)-i\.(yaml|ri)\Z/
         | 
| 46 | 
            +
                docs_imeth << path[0..-2] + [$1]
         | 
| 47 | 
            +
              when /\A(.*)-c\.(yaml|ri)\Z/
         | 
| 48 | 
            +
                docs_cmeth << path[0..-2] + [$1]
         | 
| 49 | 
            +
              when /\Acdesc-(.*)\.(yaml|ri)\Z/
         | 
| 50 | 
            +
                raise "Malformatted file name: #{fn}" unless $1 == path[-2]
         | 
| 51 | 
            +
                docs_class << path[0..-2]
         | 
| 52 | 
            +
              else
         | 
| 53 | 
            +
                # Ignore
         | 
| 54 | 
            +
              end
         | 
| 55 | 
            +
            }
         | 
| 56 | 
            +
             | 
| 57 | 
            +
            # Go over documentation
         | 
| 58 | 
            +
             | 
| 59 | 
            +
            cl_nc = []
         | 
| 60 | 
            +
            cl_ni = []
         | 
| 61 | 
            +
            cl_fc = []
         | 
| 62 | 
            +
            cl_fi = []
         | 
| 63 | 
            +
             | 
| 64 | 
            +
            docs_class.each{|class_path|
         | 
| 65 | 
            +
              class_name = class_path.join("::")
         | 
| 66 | 
            +
              begin
         | 
| 67 | 
            +
                cls = class_path.inject(Object){|c,path_elem| c.const_get(path_elem)}
         | 
| 68 | 
            +
              rescue NameError
         | 
| 69 | 
            +
                rv = try_help class_name
         | 
| 70 | 
            +
                if rv == class_name
         | 
| 71 | 
            +
                  cl_nc << class_name
         | 
| 72 | 
            +
                else
         | 
| 73 | 
            +
                  cl_ni << [class_name, rv]
         | 
| 74 | 
            +
                end
         | 
| 75 | 
            +
                next
         | 
| 76 | 
            +
              end
         | 
| 77 | 
            +
              rv1  = try_help class_name
         | 
| 78 | 
            +
              rv2 = try_help cls
         | 
| 79 | 
            +
              if rv1 == class_name && rv2 == class_name
         | 
| 80 | 
            +
                cl_fc << class_name
         | 
| 81 | 
            +
              else
         | 
| 82 | 
            +
                cl_fi << [class_name, rv1, rv2]
         | 
| 83 | 
            +
              end
         | 
| 84 | 
            +
            }
         | 
| 85 | 
            +
             | 
| 86 | 
            +
            print <<EOS
         | 
| 87 | 
            +
            Class documentation:
         | 
| 88 | 
            +
            * #{docs_class.size} classes
         | 
| 89 | 
            +
            * #{cl_fc.size} correct
         | 
| 90 | 
            +
            * #{cl_fi.size} incorrect
         | 
| 91 | 
            +
            * #{cl_nc.size} could not be verified (seem ok)
         | 
| 92 | 
            +
            * #{cl_ni.size} could not be verified (seem bad)
         | 
| 93 | 
            +
            EOS
         | 
| 94 | 
            +
            if cl_fi.size != 0
         | 
| 95 | 
            +
              puts "\nIncorrect:"
         | 
| 96 | 
            +
              cl_fi.each{|ex,rv1,rv2|
         | 
| 97 | 
            +
                puts "* #{ex} - #{rv1}/#{rv2}"
         | 
| 98 | 
            +
              }
         | 
| 99 | 
            +
            end
         | 
| 100 | 
            +
             | 
| 101 | 
            +
            cm_nc = [] # Class not found
         | 
| 102 | 
            +
            cm_ni = []
         | 
| 103 | 
            +
            cm_fc = []
         | 
| 104 | 
            +
            cm_fi = []
         | 
| 105 | 
            +
             | 
| 106 | 
            +
            docs_cmeth.each{|path|
         | 
| 107 | 
            +
              class_path, method_name = path[0..-2], path[-1]
         | 
| 108 | 
            +
              class_name = class_path.join("::")
         | 
| 109 | 
            +
              begin
         | 
| 110 | 
            +
                cls = class_path.inject(Object){|c,path_elem| c.const_get(path_elem)}
         | 
| 111 | 
            +
              rescue NameError
         | 
| 112 | 
            +
                expected = "#{class_name}::#{method_name}"
         | 
| 113 | 
            +
                rv = try_help expected
         | 
| 114 | 
            +
                if rv == expected
         | 
| 115 | 
            +
                  cm_nc << expected
         | 
| 116 | 
            +
                else
         | 
| 117 | 
            +
                  cm_ni << [expected, rv]
         | 
| 118 | 
            +
                end
         | 
| 119 | 
            +
                next
         | 
| 120 | 
            +
              end
         | 
| 121 | 
            +
              expected = "#{class_name}::#{method_name}"
         | 
| 122 | 
            +
              rv1 = try_help "#{class_name}::#{method_name}"
         | 
| 123 | 
            +
              # cls.send(:method_name) would find help for Object#send ...
         | 
| 124 | 
            +
              rv2 = eval "try_help { cls.#{method_name}() }"
         | 
| 125 | 
            +
              if rv1 == expected && rv2 == expected
         | 
| 126 | 
            +
                cm_fc << expected
         | 
| 127 | 
            +
              else
         | 
| 128 | 
            +
                cm_fi << [expected, rv1, rv2]
         | 
| 129 | 
            +
              end
         | 
| 130 | 
            +
            }
         | 
| 131 | 
            +
             | 
| 132 | 
            +
            print <<EOS
         | 
| 133 | 
            +
             | 
| 134 | 
            +
            Class method documentation:
         | 
| 135 | 
            +
            * #{docs_cmeth.size} class methods
         | 
| 136 | 
            +
            * #{cm_fc.size} correct
         | 
| 137 | 
            +
            * #{cm_fi.size} incorrect
         | 
| 138 | 
            +
            * #{cm_nc.size} could not be verified (seem ok)
         | 
| 139 | 
            +
            * #{cm_ni.size} could not be verified (seem bad)
         | 
| 140 | 
            +
            EOS
         | 
| 141 | 
            +
            if cm_fi.size != 0
         | 
| 142 | 
            +
              puts "\nIncorrect:"
         | 
| 143 | 
            +
              cm_fi.each{|ex,rv1,rv2|
         | 
| 144 | 
            +
                puts "* #{ex} - #{rv1}/#{rv2}"
         | 
| 145 | 
            +
              }
         | 
| 146 | 
            +
            end
         | 
| 147 | 
            +
             | 
| 148 | 
            +
            # And instance methods
         | 
| 149 | 
            +
             | 
| 150 | 
            +
            im_nc = [] # Class not found
         | 
| 151 | 
            +
            im_ni = []
         | 
| 152 | 
            +
            im_fc = []
         | 
| 153 | 
            +
            im_fi = []
         | 
| 154 | 
            +
             | 
| 155 | 
            +
            docs_imeth.each{|path|
         | 
| 156 | 
            +
              class_path, method_name = path[0..-2], path[-1]
         | 
| 157 | 
            +
              class_name = class_path.join("::")
         | 
| 158 | 
            +
              begin
         | 
| 159 | 
            +
                cls = class_path.inject(Object){|c,path_elem| c.const_get(path_elem)}
         | 
| 160 | 
            +
              rescue NameError
         | 
| 161 | 
            +
                expected = "#{class_name}.#{method_name}"
         | 
| 162 | 
            +
                rv = try_help expected
         | 
| 163 | 
            +
                if rv == expected
         | 
| 164 | 
            +
                  im_nc << expected
         | 
| 165 | 
            +
                else
         | 
| 166 | 
            +
                  im_ni << [expected, rv]
         | 
| 167 | 
            +
                end
         | 
| 168 | 
            +
                next
         | 
| 169 | 
            +
              end
         | 
| 170 | 
            +
              expected = "#{class_name}##{method_name}"
         | 
| 171 | 
            +
              rv1 = try_help "#{class_name}##{method_name}"
         | 
| 172 | 
            +
              # We don't know how to create a real cls object.
         | 
| 173 | 
            +
              # We could try some hacks or mock objects later.
         | 
| 174 | 
            +
             | 
| 175 | 
            +
              if rv1 == expected
         | 
| 176 | 
            +
                im_fc << expected
         | 
| 177 | 
            +
              else
         | 
| 178 | 
            +
                im_fi << [expected, rv1]
         | 
| 179 | 
            +
              end
         | 
| 180 | 
            +
            }
         | 
| 181 | 
            +
             | 
| 182 | 
            +
            print <<EOS
         | 
| 183 | 
            +
             | 
| 184 | 
            +
            Instance method documentation:
         | 
| 185 | 
            +
            * #{docs_imeth.size} instance methods
         | 
| 186 | 
            +
            * #{im_fc.size} correct
         | 
| 187 | 
            +
            * #{im_fi.size} incorrect
         | 
| 188 | 
            +
            * #{im_nc.size} could not be verified (seem ok)
         | 
| 189 | 
            +
            * #{im_ni.size} could not be verified (seem bad)
         | 
| 190 | 
            +
            EOS
         | 
| 191 | 
            +
            if im_fi.size != 0
         | 
| 192 | 
            +
              puts "\nIncorrect:"
         | 
| 193 | 
            +
              im_fi.each{|ex,rv1|
         | 
| 194 | 
            +
                puts "* #{ex} - #{rv1}"
         | 
| 195 | 
            +
              }
         | 
| 196 | 
            +
            end
         | 
| 197 | 
            +
             | 
| 198 | 
            +
            print <<EOS
         | 
| 199 | 
            +
             | 
| 200 | 
            +
            Summary:
         | 
| 201 | 
            +
            * #{docs_class.size + docs_cmeth.size  + docs_imeth.size} documentation items
         | 
| 202 | 
            +
            * #{cl_fc.size + cm_fc.size + im_fc.size} correct
         | 
| 203 | 
            +
            * #{cl_fi.size + cm_fi.size + im_fi.size} incorrect
         | 
| 204 | 
            +
            * #{cl_nc.size + cm_nc.size + im_nc.size} could not be verified (seem ok)
         | 
| 205 | 
            +
            * #{cl_ni.size + cm_ni.size + im_ni.size} could not be verified (seem bad)
         | 
| 206 | 
            +
            EOS
         | 
| @@ -0,0 +1,240 @@ | |
| 1 | 
            +
            #!/usr/bin/env ruby
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require "minitest/autorun"
         | 
| 4 | 
            +
            require_relative "../lib/magic_help"
         | 
| 5 | 
            +
            require "fileutils"
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            class Test_Magic_Help < Minitest::Test
         | 
| 8 | 
            +
              def self.test_order
         | 
| 9 | 
            +
                :alpha
         | 
| 10 | 
            +
              end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
              def assert_irb_help(expected, *args, &block)
         | 
| 13 | 
            +
                got = Magic::Help.resolve_help_query(*args, &block)
         | 
| 14 | 
            +
                if expected == nil
         | 
| 15 | 
            +
                  assert_nil(got)
         | 
| 16 | 
            +
                else
         | 
| 17 | 
            +
                  assert_equal(expected, got)
         | 
| 18 | 
            +
                end
         | 
| 19 | 
            +
              end
         | 
| 20 | 
            +
             | 
| 21 | 
            +
              # There were ways to make it work in 1.8 set_trace_func,
         | 
| 22 | 
            +
              # but TracePoint just doesn't seem to allow it
         | 
| 23 | 
            +
              def test_argument_number_mismatch
         | 
| 24 | 
            +
                # Correct number of arguments
         | 
| 25 | 
            +
                assert_irb_help("FileUtils::compare_file"){ FileUtils.compare_file 1, 2 }
         | 
| 26 | 
            +
                # Various incorrect argument counts
         | 
| 27 | 
            +
                # assert_irb_help("FileUtils::compare_file"){ FileUtils.compare_file }
         | 
| 28 | 
            +
                # assert_irb_help("FileUtils::compare_file"){ FileUtils.compare_file 1 }
         | 
| 29 | 
            +
                # assert_irb_help("FileUtils::compare_file"){ FileUtils.compare_file 1, 2, 3 }
         | 
| 30 | 
            +
              end
         | 
| 31 | 
            +
             | 
| 32 | 
            +
              def test_argumenterror_new
         | 
| 33 | 
            +
                assert_irb_help("ArgumentError::new"){ ArgumentError.new }
         | 
| 34 | 
            +
              end
         | 
| 35 | 
            +
             | 
| 36 | 
            +
              def test_class
         | 
| 37 | 
            +
                assert_irb_help("Array"){ Array }
         | 
| 38 | 
            +
                assert_irb_help("Array"){ [] }
         | 
| 39 | 
            +
                x = [1, 2, 3]
         | 
| 40 | 
            +
                assert_irb_help("Array"){ x }
         | 
| 41 | 
            +
              end
         | 
| 42 | 
            +
             | 
| 43 | 
            +
              def test_method
         | 
| 44 | 
            +
                assert_irb_help("Array#sort"){ [].sort }
         | 
| 45 | 
            +
                x = [1, 2, 3]
         | 
| 46 | 
            +
                assert_irb_help("Array#sort"){ x.sort }
         | 
| 47 | 
            +
                im  = Array.instance_method(:sort)
         | 
| 48 | 
            +
                assert_irb_help("Array#sort"){ im }
         | 
| 49 | 
            +
                m = [].method(:sort)
         | 
| 50 | 
            +
                assert_irb_help("Array#sort"){ m }
         | 
| 51 | 
            +
                um = [].method(:sort).unbind
         | 
| 52 | 
            +
                assert_irb_help("Array#sort"){ um }
         | 
| 53 | 
            +
              end
         | 
| 54 | 
            +
             | 
| 55 | 
            +
              def test_class_method
         | 
| 56 | 
            +
                assert_irb_help("Dir::[]"){ Dir[""] }
         | 
| 57 | 
            +
                m  = Dir.method(:[])
         | 
| 58 | 
            +
                assert_irb_help("Dir::[]"){ m }
         | 
| 59 | 
            +
                um = Dir.method(:[]).unbind
         | 
| 60 | 
            +
                assert_irb_help("Dir::[]"){ um }
         | 
| 61 | 
            +
              end
         | 
| 62 | 
            +
             | 
| 63 | 
            +
              def test_module
         | 
| 64 | 
            +
                assert_irb_help("Enumerable", Enumerable)
         | 
| 65 | 
            +
                assert_irb_help("Enumerable"){ Enumerable }
         | 
| 66 | 
            +
                assert_irb_help("Enumerable", "Enumerable")
         | 
| 67 | 
            +
                um = Enumerable.instance_method(:map)
         | 
| 68 | 
            +
                assert_irb_help("Enumerable#map"){ um }
         | 
| 69 | 
            +
                um2 = Range.instance_method(:any?)
         | 
| 70 | 
            +
                assert_irb_help("Enumerable#any?"){ um2 }
         | 
| 71 | 
            +
                m = (0..1).method(:any?)
         | 
| 72 | 
            +
                assert_irb_help("Enumerable#any?"){ m }
         | 
| 73 | 
            +
              end
         | 
| 74 | 
            +
             | 
| 75 | 
            +
              def test_method_inherited
         | 
| 76 | 
            +
                f = File.open(__FILE__)
         | 
| 77 | 
            +
                assert_irb_help("IO#sync"){ f.sync }
         | 
| 78 | 
            +
                im = File.instance_method(:sync)
         | 
| 79 | 
            +
                assert_irb_help("IO#sync"){ im }
         | 
| 80 | 
            +
                m  = f.method(:sync)
         | 
| 81 | 
            +
                assert_irb_help("IO#sync"){ m }
         | 
| 82 | 
            +
                um = f.method(:sync).unbind
         | 
| 83 | 
            +
                assert_irb_help("IO#sync"){ um }
         | 
| 84 | 
            +
              end
         | 
| 85 | 
            +
             | 
| 86 | 
            +
              def test_string
         | 
| 87 | 
            +
                assert_irb_help("Array",    "Array" )
         | 
| 88 | 
            +
                assert_irb_help("Array#[]", "Array#[]")
         | 
| 89 | 
            +
                assert_irb_help("Dir::[]",  "Dir::[]" )
         | 
| 90 | 
            +
                assert_irb_help("Array#[]", "Array.[]")
         | 
| 91 | 
            +
                assert_irb_help("Dir::[]",  "Dir.[]" )
         | 
| 92 | 
            +
                assert_irb_help("IO#sync",  "File#sync" )
         | 
| 93 | 
            +
              end
         | 
| 94 | 
            +
             | 
| 95 | 
            +
              def test_string_bogus
         | 
| 96 | 
            +
                assert_irb_help("Xyzzy#foo",  "Xyzzy#foo")
         | 
| 97 | 
            +
                assert_irb_help("Xyzzy::foo", "Xyzzy::foo")
         | 
| 98 | 
            +
                assert_irb_help("Xyzzy.foo",  "Xyzzy.foo")
         | 
| 99 | 
            +
             | 
| 100 | 
            +
                assert_irb_help("Array#xyzzy",  "Array#xyzzy")
         | 
| 101 | 
            +
                assert_irb_help("Array::xyzzy", "Array::xyzzy")
         | 
| 102 | 
            +
                assert_irb_help("Array.xyzzy",  "Array.xyzzy")
         | 
| 103 | 
            +
              end
         | 
| 104 | 
            +
             | 
| 105 | 
            +
              def test_operators
         | 
| 106 | 
            +
                # Ruby apparently optimizes away 2+2, that's awkward...
         | 
| 107 | 
            +
                assert_irb_help("Fixnum#**"){ 2 ** 2 }
         | 
| 108 | 
            +
                assert_irb_help("Float#**"){ 2.0 ** 2.0 }
         | 
| 109 | 
            +
                assert_irb_help("Array#[]"){ [][] }
         | 
| 110 | 
            +
                # =~ is instance method of Kernel, but is documented as instance method of Object
         | 
| 111 | 
            +
                # assert_irb_help("Kernel#=~"){ [] =~ [] }
         | 
| 112 | 
            +
                assert_irb_help("Kernel#=~"){ [] =~ [] }
         | 
| 113 | 
            +
              end
         | 
| 114 | 
            +
             | 
| 115 | 
            +
              def test_nil
         | 
| 116 | 
            +
                assert_irb_help(nil)
         | 
| 117 | 
            +
                assert_irb_help("NilClass"){ nil }
         | 
| 118 | 
            +
                assert_irb_help("NilClass"){ }
         | 
| 119 | 
            +
              end
         | 
| 120 | 
            +
             | 
| 121 | 
            +
              def test_superclass
         | 
| 122 | 
            +
                # superclass is a method of Class
         | 
| 123 | 
            +
                # So Foo::superclass should find Class#superclass
         | 
| 124 | 
            +
             | 
| 125 | 
            +
                assert_irb_help("Class#superclass"){ Float.superclass }
         | 
| 126 | 
            +
                assert_irb_help("Class#superclass", "Float::superclass")
         | 
| 127 | 
            +
                assert_irb_help("Class#superclass", "Float.superclass")
         | 
| 128 | 
            +
             | 
| 129 | 
            +
                assert_irb_help("Class#superclass"){ Class.superclass }
         | 
| 130 | 
            +
                assert_irb_help("Class#superclass"){ "Class.superclass" }
         | 
| 131 | 
            +
                assert_irb_help("Class#superclass"){ "Class::superclass" }
         | 
| 132 | 
            +
              end
         | 
| 133 | 
            +
             | 
| 134 | 
            +
              def test_class_new
         | 
| 135 | 
            +
                # Most classes do not override Class#new, but have
         | 
| 136 | 
            +
                # Documentation for Foo.new anyway (it actually documents Foo#initialize)
         | 
| 137 | 
            +
             | 
| 138 | 
            +
                # First, handling of Class#new (default creator of instances)
         | 
| 139 | 
            +
                # and Class::new (creator of new classses)
         | 
| 140 | 
            +
                assert_irb_help("Class::new"){ Class.new }
         | 
| 141 | 
            +
                assert_irb_help("Class::new"){ Class::new }
         | 
| 142 | 
            +
                assert_irb_help("Class#new",  "Class#new")
         | 
| 143 | 
            +
                assert_irb_help("Class::new", "Class::new")
         | 
| 144 | 
            +
                assert_irb_help("Class#new",  "Class.new")
         | 
| 145 | 
            +
             | 
| 146 | 
            +
                # Module::new is documented and it uses default Class#new
         | 
| 147 | 
            +
                assert_irb_help("Module::new"){ Module.new }
         | 
| 148 | 
            +
                assert_irb_help("Module::new", "Module.new")
         | 
| 149 | 
            +
                assert_irb_help("Module::new", "Module::new")
         | 
| 150 | 
            +
             | 
| 151 | 
            +
                # IO::new is documented and it has separate implementation
         | 
| 152 | 
            +
                assert_irb_help("IO::new"){ IO.new }
         | 
| 153 | 
            +
                assert_irb_help("IO::new", "IO.new")
         | 
| 154 | 
            +
                assert_irb_help("IO::new", "IO::new")
         | 
| 155 | 
            +
             | 
| 156 | 
            +
                # File::new is documented and it uses IO::new
         | 
| 157 | 
            +
                assert_irb_help("File::new"){ File.new }
         | 
| 158 | 
            +
                assert_irb_help("File::new", "File.new")
         | 
| 159 | 
            +
                assert_irb_help("File::new", "File::new")
         | 
| 160 | 
            +
              end
         | 
| 161 | 
            +
             | 
| 162 | 
            +
              # This tests work-arounds for bugs in Ruby documentation !!!
         | 
| 163 | 
            +
              # In the perfect world it should totally fail !!!
         | 
| 164 | 
            +
              def test_object_methods
         | 
| 165 | 
            +
                # Documentation mixes some Kernel and Object methods
         | 
| 166 | 
            +
             | 
| 167 | 
            +
                # Ruby has Kernel#__id__ but documentation has Object#__id__
         | 
| 168 | 
            +
                assert_irb_help("BasicObject#__id__"){ __id__ }
         | 
| 169 | 
            +
                assert_irb_help("BasicObject#__id__"){ 42.__id__ }
         | 
| 170 | 
            +
                assert_irb_help("BasicObject#__id__", "Object#__id__")
         | 
| 171 | 
            +
                assert_irb_help("BasicObject#__id__", "Object.__id__")
         | 
| 172 | 
            +
                assert_irb_help("Kernel#__id__", "Kernel#__id__")
         | 
| 173 | 
            +
                assert_irb_help("BasicObject#__id__", "Kernel.__id__")
         | 
| 174 | 
            +
             | 
| 175 | 
            +
                # Ruby has Kernel#sprintf and documentation has Kernel#sprintf
         | 
| 176 | 
            +
                assert_irb_help("Kernel#sprintf"){ sprintf }
         | 
| 177 | 
            +
                assert_irb_help("Kernel#sprintf", "Object#sprintf")
         | 
| 178 | 
            +
                assert_irb_help("Kernel#sprintf", "Object.sprintf")
         | 
| 179 | 
            +
                assert_irb_help("Kernel#sprintf", "Kernel#sprintf")
         | 
| 180 | 
            +
                assert_irb_help("Kernel#sprintf", "Kernel.sprintf")
         | 
| 181 | 
            +
             | 
| 182 | 
            +
                # TODO: For completion - Object method documented in Object
         | 
| 183 | 
            +
                # TODO: For completion - Object method documented in Kernel
         | 
| 184 | 
            +
                # TODO: For completion - class methods of both
         | 
| 185 | 
            +
              end
         | 
| 186 | 
            +
             | 
| 187 | 
            +
              def test_method_missing
         | 
| 188 | 
            +
                # Some fake method
         | 
| 189 | 
            +
                assert_irb_help("Time::rfc999"){ Time::rfc999 }
         | 
| 190 | 
            +
                assert_irb_help("Time::rfc999"){ Time.rfc999 } # is that really best ?
         | 
| 191 | 
            +
                assert_irb_help("Time::rfc999", "Time::rfc999")
         | 
| 192 | 
            +
                assert_irb_help("Time.rfc999", "Time.rfc999")
         | 
| 193 | 
            +
              end
         | 
| 194 | 
            +
             | 
| 195 | 
            +
              def test_method_missing_explicit
         | 
| 196 | 
            +
                assert_irb_help("Kernel#method_missing", "Kernel#method_missing")
         | 
| 197 | 
            +
                assert_irb_help("BasicObject#method_missing", "Kernel.method_missing")
         | 
| 198 | 
            +
                assert_irb_help("BasicObject#method_missing", "Float#method_missing")
         | 
| 199 | 
            +
                assert_irb_help("BasicObject#method_missing", "Float.method_missing")
         | 
| 200 | 
            +
                assert_irb_help("BasicObject#method_missing"){ 42.method_missing }
         | 
| 201 | 
            +
                assert_irb_help("BasicObject#method_missing"){ method_missing }
         | 
| 202 | 
            +
              end
         | 
| 203 | 
            +
             | 
| 204 | 
            +
              def test_longpath
         | 
| 205 | 
            +
                assert_irb_help("File::Stat::new", "File::Stat.new")
         | 
| 206 | 
            +
                assert_irb_help("File::Stat::new"){ File::Stat.new }
         | 
| 207 | 
            +
                assert_irb_help("File::Stat::new"){ File::Stat::new }
         | 
| 208 | 
            +
                fs = File::Stat.new(__FILE__)
         | 
| 209 | 
            +
                assert_irb_help("File::Stat#size"){ fs.size }
         | 
| 210 | 
            +
                assert_irb_help("File::Stat#size", "File::Stat#size")
         | 
| 211 | 
            +
                assert_irb_help("File::Stat#size", "File::Stat.size")
         | 
| 212 | 
            +
              end
         | 
| 213 | 
            +
             | 
| 214 | 
            +
              def test_private
         | 
| 215 | 
            +
                # help should ignore public/protected/private
         | 
| 216 | 
            +
                # private is a private function of Module
         | 
| 217 | 
            +
                assert_irb_help("Module#private", "Module#private")
         | 
| 218 | 
            +
                assert_irb_help("Module#private", "Module.private")
         | 
| 219 | 
            +
                assert_irb_help("Module#private", "Module::private")
         | 
| 220 | 
            +
             | 
| 221 | 
            +
                assert_irb_help("Module#private", "Class#private")
         | 
| 222 | 
            +
                assert_irb_help("Module#private", "Class.private")
         | 
| 223 | 
            +
                assert_irb_help("Module#private", "Class::private")
         | 
| 224 | 
            +
             | 
| 225 | 
            +
                assert_irb_help("Module#private", "Float.private")
         | 
| 226 | 
            +
                assert_irb_help("Module#private", "Float::private")
         | 
| 227 | 
            +
             | 
| 228 | 
            +
                assert_irb_help("Module::private"){ Module::private }
         | 
| 229 | 
            +
                assert_irb_help("Module::private"){ Module.private }
         | 
| 230 | 
            +
             | 
| 231 | 
            +
                assert_irb_help("Class::private"){ Class::private }
         | 
| 232 | 
            +
                assert_irb_help("Class::private"){ Class.private }
         | 
| 233 | 
            +
             | 
| 234 | 
            +
                assert_irb_help("Float::private"){ Float::private }
         | 
| 235 | 
            +
                assert_irb_help("Float::private"){ Float.private }
         | 
| 236 | 
            +
             | 
| 237 | 
            +
                assert_irb_help("String#singleton_method_added"){ "".singleton_method_added }
         | 
| 238 | 
            +
                assert_irb_help("BasicObject#singleton_method_added"){ singleton_method_added }
         | 
| 239 | 
            +
              end
         | 
| 240 | 
            +
            end
         | 
    
        metadata
    ADDED
    
    | @@ -0,0 +1,79 @@ | |
| 1 | 
            +
            --- !ruby/object:Gem::Specification
         | 
| 2 | 
            +
            name: magic-help
         | 
| 3 | 
            +
            version: !ruby/object:Gem::Version
         | 
| 4 | 
            +
              version: '0.20170204'
         | 
| 5 | 
            +
            platform: ruby
         | 
| 6 | 
            +
            authors:
         | 
| 7 | 
            +
            - Tomasz Wegrzanowski
         | 
| 8 | 
            +
            autorequire: 
         | 
| 9 | 
            +
            bindir: bin
         | 
| 10 | 
            +
            cert_chain: []
         | 
| 11 | 
            +
            date: 2017-02-04 00:00:00.000000000 Z
         | 
| 12 | 
            +
            dependencies:
         | 
| 13 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 14 | 
            +
              name: rake
         | 
| 15 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 16 | 
            +
                requirements:
         | 
| 17 | 
            +
                - - ">="
         | 
| 18 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 19 | 
            +
                    version: '0'
         | 
| 20 | 
            +
              type: :development
         | 
| 21 | 
            +
              prerelease: false
         | 
| 22 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 23 | 
            +
                requirements:
         | 
| 24 | 
            +
                - - ">="
         | 
| 25 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 26 | 
            +
                    version: '0'
         | 
| 27 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 28 | 
            +
              name: minitest
         | 
| 29 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 30 | 
            +
                requirements:
         | 
| 31 | 
            +
                - - ">="
         | 
| 32 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 33 | 
            +
                    version: '0'
         | 
| 34 | 
            +
              type: :development
         | 
| 35 | 
            +
              prerelease: false
         | 
| 36 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 37 | 
            +
                requirements:
         | 
| 38 | 
            +
                - - ">="
         | 
| 39 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 40 | 
            +
                    version: '0'
         | 
| 41 | 
            +
            description: 
         | 
| 42 | 
            +
            email: Tomasz.Wegrzanowski@gmail.com
         | 
| 43 | 
            +
            executables: []
         | 
| 44 | 
            +
            extensions: []
         | 
| 45 | 
            +
            extra_rdoc_files: []
         | 
| 46 | 
            +
            files:
         | 
| 47 | 
            +
            - lib/magic/help.rb
         | 
| 48 | 
            +
            - lib/magic/help/irb.rb
         | 
| 49 | 
            +
            - lib/magic/help/set_trace_func.rb
         | 
| 50 | 
            +
            - lib/magic/help/tracepoint.rb
         | 
| 51 | 
            +
            - lib/magic_help.rb
         | 
| 52 | 
            +
            - test/mass_test.rb
         | 
| 53 | 
            +
            - test/tc_magic_help.rb
         | 
| 54 | 
            +
            homepage: https://github.com/taw/magic-help
         | 
| 55 | 
            +
            licenses: []
         | 
| 56 | 
            +
            metadata: {}
         | 
| 57 | 
            +
            post_install_message: 
         | 
| 58 | 
            +
            rdoc_options: []
         | 
| 59 | 
            +
            require_paths:
         | 
| 60 | 
            +
            - lib
         | 
| 61 | 
            +
            required_ruby_version: !ruby/object:Gem::Requirement
         | 
| 62 | 
            +
              requirements:
         | 
| 63 | 
            +
              - - ">="
         | 
| 64 | 
            +
                - !ruby/object:Gem::Version
         | 
| 65 | 
            +
                  version: '0'
         | 
| 66 | 
            +
            required_rubygems_version: !ruby/object:Gem::Requirement
         | 
| 67 | 
            +
              requirements:
         | 
| 68 | 
            +
              - - ">="
         | 
| 69 | 
            +
                - !ruby/object:Gem::Version
         | 
| 70 | 
            +
                  version: '0'
         | 
| 71 | 
            +
            requirements: []
         | 
| 72 | 
            +
            rubyforge_project: 
         | 
| 73 | 
            +
            rubygems_version: 2.5.2
         | 
| 74 | 
            +
            signing_key: 
         | 
| 75 | 
            +
            specification_version: 4
         | 
| 76 | 
            +
            summary: Plugin for irb providing more intuitive documentation access.
         | 
| 77 | 
            +
            test_files:
         | 
| 78 | 
            +
            - test/mass_test.rb
         | 
| 79 | 
            +
            - test/tc_magic_help.rb
         |