binding_of_caller 0.4.0 → 0.6.0

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.
data/README.md CHANGED
@@ -13,7 +13,7 @@ call stack, not limited to just the immediate caller.
13
13
 
14
14
  **Recommended for use only in debugging situations. Do not use this in production apps.**
15
15
 
16
- **Only works in MRI Ruby 1.9.2**
16
+ **Only works in MRI Ruby 1.9.2 and 1.9.3**
17
17
 
18
18
  * Install the [gem](https://rubygems.org/gems/binding_of_caller): `gem install binding_of_caller`
19
19
  * See the [source code](http://github.com/banister/binding_of_caller)
@@ -50,8 +50,7 @@ This project is a spinoff from the [Pry REPL project.](http://pry.github.com)
50
50
  Features and limitations
51
51
  -------------------------
52
52
 
53
- * Only works with MRI 1.9.2
54
- * Broken in 1.9.3, support will hopefully be provided in the near future.
53
+ * Only works with MRI 1.9.2 and 1.9.3
55
54
  * Does not work in 1.8.7, but there is a well known (continuation-based) hack to get a `Binding#of_caller` there.
56
55
 
57
56
  Contact
@@ -1,15 +1,31 @@
1
1
  /* (c) 2011 John Mair (banisterfiend), MIT license */
2
2
 
3
3
  #include <ruby.h>
4
-
5
- #include <ruby/io.h>
6
- #include <ruby/re.h>
7
4
  #include "vm_core.h"
8
5
  #include "gc.h"
9
6
 
10
7
  typedef enum { false, true } bool;
11
8
 
12
- const int max_frame_errors = 4;
9
+ static VALUE
10
+ string2sym(const char * string)
11
+ {
12
+ return ID2SYM(rb_intern(string));
13
+ }
14
+
15
+ static inline const rb_data_type_t *
16
+ threadptr_data_type(void)
17
+ {
18
+ static const rb_data_type_t *thread_data_type;
19
+ if (!thread_data_type) {
20
+ VALUE current_thread = rb_thread_current();
21
+ thread_data_type = RTYPEDDATA_TYPE(current_thread);
22
+ }
23
+ return thread_data_type;
24
+ }
25
+
26
+ #define ruby_thread_data_type *threadptr_data_type()
27
+
28
+ #define ruby_current_thread ((rb_thread_t *)RTYPEDDATA_DATA(rb_thread_current()))
13
29
 
14
30
  static size_t
15
31
  binding_memsize(const void *ptr)
@@ -59,31 +75,48 @@ binding_alloc(VALUE klass)
59
75
  }
60
76
 
61
77
  static bool valid_frame_p(rb_control_frame_t * cfp, rb_control_frame_t * limit_cfp) {
62
- if (cfp > limit_cfp)
63
- rb_raise(rb_eRuntimeError, "Invalid frame, gone beyond end of stack!");
64
-
65
78
  return cfp->iseq && !NIL_P(cfp->self);
66
79
  }
67
80
 
68
81
  static rb_control_frame_t * find_valid_frame(rb_control_frame_t * cfp, rb_control_frame_t * limit_cfp) {
69
- int error_count = 0;
70
-
71
- while (error_count <= max_frame_errors) {
82
+ while (cfp < limit_cfp) {
72
83
  cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
73
84
 
85
+ if (cfp >= limit_cfp)
86
+ return NULL;
87
+
74
88
  if (valid_frame_p(cfp, limit_cfp))
75
89
  return cfp;
76
- else
77
- error_count += 1;
78
90
  }
79
91
 
80
- // never reached
81
- return 0;
92
+ // beyond end of stack
93
+ return NULL;
94
+ }
95
+
96
+ static VALUE
97
+ frametype_name(VALUE flag)
98
+ {
99
+ switch (flag & VM_FRAME_MAGIC_MASK) {
100
+ case VM_FRAME_MAGIC_METHOD: return string2sym("method");
101
+ case VM_FRAME_MAGIC_BLOCK: return string2sym("block");
102
+ case VM_FRAME_MAGIC_CLASS: return string2sym("class");
103
+ case VM_FRAME_MAGIC_TOP: return string2sym("top");
104
+ case VM_FRAME_MAGIC_FINISH: return string2sym("finish");
105
+ case VM_FRAME_MAGIC_CFUNC: return string2sym("cfunc");
106
+ case VM_FRAME_MAGIC_PROC: return string2sym("proc");
107
+ case VM_FRAME_MAGIC_IFUNC: return string2sym("ifunc");
108
+ case VM_FRAME_MAGIC_EVAL: return string2sym("eval");
109
+ case VM_FRAME_MAGIC_LAMBDA: return string2sym("lambda");
110
+ default:
111
+ rb_raise(rb_eRuntimeError, "Unknown frame type! got flag: %d", FIX2INT(flag));
112
+ }
82
113
  }
83
114
 
84
115
  static VALUE binding_of_caller(VALUE self, VALUE rb_level)
85
116
  {
86
- rb_thread_t *th = GET_THREAD();
117
+ rb_thread_t *th;
118
+ GetThreadPtr(rb_thread_current(), th);
119
+
87
120
  rb_control_frame_t *cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(th->cfp);
88
121
  rb_control_frame_t *limit_cfp = (void *)(th->stack + th->stack_size);
89
122
  int level = FIX2INT(rb_level);
@@ -92,6 +125,9 @@ static VALUE binding_of_caller(VALUE self, VALUE rb_level)
92
125
  for (int i = 0; i < level; i++) {
93
126
  cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
94
127
 
128
+ if (cfp >= limit_cfp)
129
+ rb_raise(rb_eRuntimeError, "Invalid frame, gone beyond end of stack!");
130
+
95
131
  // skip invalid frames
96
132
  if (!valid_frame_p(cfp, limit_cfp))
97
133
  cfp = find_valid_frame(cfp, limit_cfp);
@@ -100,23 +136,78 @@ static VALUE binding_of_caller(VALUE self, VALUE rb_level)
100
136
  VALUE bindval = binding_alloc(rb_cBinding);
101
137
  rb_binding_t *bind;
102
138
 
103
- if (cfp == 0) {
139
+ if (cfp == 0)
104
140
  rb_raise(rb_eRuntimeError, "Can't create Binding Object on top of Fiber.");
105
- }
106
141
 
107
142
  GetBindingPtr(bindval, bind);
108
143
  bind->env = rb_vm_make_env_object(th, cfp);
109
144
  bind->filename = cfp->iseq->filename;
110
145
  bind->line_no = rb_vm_get_sourceline(cfp);
146
+
147
+ rb_iv_set(bindval, "@frame_type", frametype_name(cfp->flag));
148
+ rb_iv_set(bindval, "@frame_description", cfp->iseq->name);
149
+
111
150
  return bindval;
112
151
  }
113
152
 
153
+ static VALUE
154
+ frame_type(VALUE self)
155
+ {
156
+ return rb_iv_get(self, "@frame_type");
157
+ }
158
+
159
+ static VALUE
160
+ frame_description(VALUE self)
161
+ {
162
+ return rb_iv_get(self, "@frame_description");
163
+ }
164
+
165
+ static VALUE frame_count(VALUE self)
166
+ {
167
+ rb_thread_t *th;
168
+ GetThreadPtr(rb_thread_current(), th);
169
+
170
+ rb_control_frame_t *cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(th->cfp);
171
+ rb_control_frame_t *limit_cfp = (void *)(th->stack + th->stack_size);
172
+
173
+ int i = 1;
174
+ while (cfp < limit_cfp) {
175
+ cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
176
+
177
+ if (cfp >= limit_cfp)
178
+ return INT2FIX(i);
179
+
180
+ // skip invalid frames
181
+ if (!valid_frame_p(cfp, limit_cfp))
182
+ cfp = find_valid_frame(cfp, limit_cfp);
183
+
184
+ i++;
185
+ }
186
+
187
+ return INT2FIX(i);
188
+ }
189
+
190
+ static VALUE
191
+ callers(VALUE self)
192
+ {
193
+ VALUE ary = rb_ary_new();
194
+
195
+ for (int i = 0; i < FIX2INT(frame_count(self)); i++)
196
+ rb_ary_push(ary, binding_of_caller(self, INT2FIX(i)));
197
+
198
+ return ary;
199
+ }
200
+
114
201
  void
115
202
  Init_binding_of_caller()
116
203
  {
117
204
  VALUE mBindingOfCaller = rb_define_module("BindingOfCaller");
118
205
 
119
206
  rb_define_method(mBindingOfCaller, "of_caller", binding_of_caller, 1);
207
+ rb_define_method(mBindingOfCaller, "frame_count", frame_count, 0);
208
+ rb_define_method(mBindingOfCaller, "frame_type", frame_type, 0);
209
+ rb_define_method(mBindingOfCaller, "frame_description", frame_description, 0);
210
+ rb_define_method(mBindingOfCaller, "callers", callers, 0);
120
211
  rb_include_module(rb_cBinding, mBindingOfCaller);
121
212
  }
122
213
 
@@ -1,3 +1,3 @@
1
1
  module BindingOfCaller
2
- VERSION = "0.4.0"
2
+ VERSION = "0.6.0"
3
3
  end
data/test/test.rb CHANGED
@@ -8,52 +8,87 @@ puts "Testing binding_of_caller version #{BindingOfCaller::VERSION}..."
8
8
  puts "Ruby version: #{RUBY_VERSION}"
9
9
 
10
10
  describe BindingOfCaller do
11
- it "should fetch immediate caller's binding when 0 is passed" do
12
- o = Object.new
13
- def o.a
14
- var = 1
15
- binding.of_caller(0).eval('var')
11
+ describe "of_caller" do
12
+ it "should fetch immediate caller's binding when 0 is passed" do
13
+ o = Object.new
14
+ def o.a
15
+ var = 1
16
+ binding.of_caller(0).eval('var')
17
+ end
18
+
19
+ o. a.should == 1
16
20
  end
17
21
 
18
- o. a.should == 1
19
- end
22
+ it "should fetch parent of caller's binding when 1 is passed" do
23
+ o = Object.new
24
+ def o.a
25
+ var = 1
26
+ b
27
+ end
20
28
 
21
- it "should fetch parent of caller's binding when 1 is passed" do
22
- o = Object.new
23
- def o.a
24
- var = 1
25
- b
26
- end
29
+ def o.b
30
+ binding.of_caller(1).eval('var')
31
+ end
27
32
 
28
- def o.b
29
- binding.of_caller(1).eval('var')
33
+ o.a.should == 1
30
34
  end
31
35
 
32
- o.a.should == 1
33
- end
36
+ it "should modify locals in parent of caller's binding" do
37
+ o = Object.new
38
+ def o.a
39
+ var = 1
40
+ b
41
+ var
42
+ end
43
+
44
+ def o.b
45
+ binding.of_caller(1).eval('var = 20')
46
+ end
34
47
 
35
- it "should modify locals in parent of caller's binding" do
36
- o = Object.new
37
- def o.a
38
- var = 1
39
- b
40
- var
48
+ o.a.should == 20
41
49
  end
42
50
 
43
- def o.b
44
- binding.of_caller(1).eval('var = 20')
51
+ it "should raise an exception when retrieving an out of band binding" do
52
+ o = Object.new
53
+ def o.a
54
+ binding.of_caller(100)
55
+ end
56
+
57
+ lambda { o.a }.should.raise RuntimeError
45
58
  end
59
+ end
46
60
 
47
- o.a.should == 20
61
+ describe "frame_count" do
62
+ it 'frame_count should equal callers.count' do
63
+ binding.frame_count.should == binding.callers.count
64
+ end
48
65
  end
49
66
 
50
- it "should raise an exception when retrieving an out of band binding" do
51
- o = Object.new
52
- def o.a
53
- binding.of_caller(100)
67
+ describe "frame_type" do
68
+ it 'should return the correct frame types' do
69
+ o = Object.new
70
+
71
+ # method frame
72
+ def o.a
73
+ b
74
+ end
75
+
76
+ # method frame
77
+ def o.b
78
+ # block frame
79
+ proc do
80
+ binding.callers
81
+ end.call
82
+ end
83
+ caller_bindings = o.a
84
+ caller_bindings[0].frame_type.should == :block
85
+ caller_bindings[1].frame_type.should == :method
86
+ caller_bindings[2].frame_type.should == :method
54
87
  end
55
88
 
56
- lambda { o.a }.should.raise RuntimeError
89
+ it 'should raise when invoked on an ordinary binding (i.e one not generated through Binding#of_caller)' do
90
+ lambda { binding.frame_type }.should.raise RuntimeError
91
+ end
57
92
  end
58
93
  end
59
94
 
metadata CHANGED
@@ -1,37 +1,35 @@
1
- --- !ruby/object:Gem::Specification
1
+ --- !ruby/object:Gem::Specification
2
2
  name: binding_of_caller
3
- version: !ruby/object:Gem::Version
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.6.0
4
5
  prerelease:
5
- version: 0.4.0
6
6
  platform: ruby
7
- authors:
7
+ authors:
8
8
  - John Mair (banisterfiend)
9
9
  autorequire:
10
10
  bindir: bin
11
11
  cert_chain: []
12
-
13
- date: 2011-10-20 00:00:00 Z
14
- dependencies:
15
- - !ruby/object:Gem::Dependency
12
+ date: 2011-11-28 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
16
15
  name: bacon
17
- prerelease: false
18
- requirement: &id001 !ruby/object:Gem::Requirement
16
+ requirement: &70166101101640 !ruby/object:Gem::Requirement
19
17
  none: false
20
- requirements:
18
+ requirements:
21
19
  - - ~>
22
- - !ruby/object:Gem::Version
20
+ - !ruby/object:Gem::Version
23
21
  version: 1.1.0
24
22
  type: :development
25
- version_requirements: *id001
26
- description: Retrieve the binding of a method's caller. Can also retrieve bindings even further up the stack. Currently only works for MRI 1.9.2.
23
+ prerelease: false
24
+ version_requirements: *70166101101640
25
+ description: Retrieve the binding of a method's caller. Can also retrieve bindings
26
+ even further up the stack. Currently only works for MRI 1.9.2.
27
27
  email: jrmair@gmail.com
28
28
  executables: []
29
-
30
- extensions:
29
+ extensions:
31
30
  - ext/binding_of_caller/extconf.rb
32
31
  extra_rdoc_files: []
33
-
34
- files:
32
+ files:
35
33
  - .gemtest
36
34
  - .gitignore
37
35
  - .yardopts
@@ -94,30 +92,28 @@ files:
94
92
  - test/test.rb
95
93
  homepage: http://github.com/banister/binding_of_caller
96
94
  licenses: []
97
-
98
95
  post_install_message:
99
96
  rdoc_options: []
100
-
101
- require_paths:
97
+ require_paths:
102
98
  - lib
103
- required_ruby_version: !ruby/object:Gem::Requirement
99
+ required_ruby_version: !ruby/object:Gem::Requirement
104
100
  none: false
105
- requirements:
106
- - - ">="
107
- - !ruby/object:Gem::Version
108
- version: "0"
109
- required_rubygems_version: !ruby/object:Gem::Requirement
101
+ requirements:
102
+ - - ! '>='
103
+ - !ruby/object:Gem::Version
104
+ version: '0'
105
+ required_rubygems_version: !ruby/object:Gem::Requirement
110
106
  none: false
111
- requirements:
112
- - - ">="
113
- - !ruby/object:Gem::Version
114
- version: "0"
107
+ requirements:
108
+ - - ! '>='
109
+ - !ruby/object:Gem::Version
110
+ version: '0'
115
111
  requirements: []
116
-
117
112
  rubyforge_project:
118
- rubygems_version: 1.8.11
113
+ rubygems_version: 1.8.10
119
114
  signing_key:
120
115
  specification_version: 3
121
- summary: Retrieve the binding of a method's caller. Can also retrieve bindings even further up the stack. Currently only works for MRI 1.9.2.
122
- test_files:
116
+ summary: Retrieve the binding of a method's caller. Can also retrieve bindings even
117
+ further up the stack. Currently only works for MRI 1.9.2.
118
+ test_files:
123
119
  - test/test.rb