blob: c5b296f6e2a7b088209e1aebb2c035ee0d570de1 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
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
115
116
117
118
119
120
|
require_relative '../../spec_helper'
require_relative 'fixtures/classes'
describe "Method#source_location" do
before :each do
@method = MethodSpecs::SourceLocation.method(:location)
end
it "returns an Array" do
@method.source_location.should be_an_instance_of(Array)
end
it "sets the first value to the path of the file in which the method was defined" do
file = @method.source_location.first
file.should be_an_instance_of(String)
file.should == File.realpath('fixtures/classes.rb', __dir__)
end
it "sets the last value to an Integer representing the line on which the method was defined" do
line = @method.source_location.last
line.should be_an_instance_of(Integer)
line.should == 5
end
it "returns the last place the method was defined" do
MethodSpecs::SourceLocation.method(:redefined).source_location.last.should == 13
end
it "returns the location of the original method even if it was aliased" do
MethodSpecs::SourceLocation.new.method(:aka).source_location.last.should == 17
end
it "works for methods defined with a block" do
line = nil
klass = Class.new do
line = __LINE__ + 1
define_method(:f) { }
end
method = klass.new.method(:f)
method.source_location[0].should =~ /#{__FILE__}/
method.source_location[1].should == line
end
it "works for methods defined with a Method" do
line = nil
klass = Class.new do
line = __LINE__ + 1
def f
end
define_method :g, new.method(:f)
end
method = klass.new.method(:g)
method.source_location[0].should =~ /#{__FILE__}/
method.source_location[1].should == line
end
it "works for methods defined with an UnboundMethod" do
line = nil
klass = Class.new do
line = __LINE__ + 1
def f
end
define_method :g, instance_method(:f)
end
method = klass.new.method(:g)
method.source_location[0].should =~ /#{__FILE__}/
method.source_location[1].should == line
end
it "works for methods whose visibility has been overridden in a subclass" do
line = nil
superclass = Class.new do
line = __LINE__ + 1
def f
end
end
subclass = Class.new(superclass) do
private :f
end
method = subclass.new.method(:f)
method.source_location[0].should =~ /#{__FILE__}/
method.source_location[1].should == line
end
it "works for core methods where it returns nil or <internal:" do
loc = method(:__id__).source_location
if loc == nil
loc.should == nil
else
loc[0].should.start_with?('<internal:')
loc[1].should be_kind_of(Integer)
end
loc = method(:tap).source_location
if loc == nil
loc.should == nil
else
loc[0].should.start_with?('<internal:')
loc[1].should be_kind_of(Integer)
end
end
it "works for eval with a given line" do
c = Class.new do
eval('def self.m; end', nil, "foo", 100)
end
c.method(:m).source_location.should == ["foo", 100]
end
describe "for a Method generated by respond_to_missing?" do
it "returns nil" do
m = MethodSpecs::Methods.new
m.method(:handled_via_method_missing).source_location.should be_nil
end
end
end
|