binding_of_caller 0.2.0 → 0.6.8

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/.gemtest ADDED
File without changes
data/.gitignore ADDED
@@ -0,0 +1,7 @@
1
+ Makefile
2
+ *.so
3
+ *.o
4
+ *.def
5
+ doc/
6
+ pkg/
7
+ .yardoc/
data/.travis.yml ADDED
@@ -0,0 +1,23 @@
1
+ rvm:
2
+ - 1.9.2
3
+ - 1.9.3
4
+ - rbx-18mode
5
+ - rbx-19mode
6
+
7
+ notifications:
8
+ irc: "irc.freenode.org#pry"
9
+ recipients:
10
+ - jrmair@gmail.com
11
+
12
+ branches:
13
+ only:
14
+ - master
15
+
16
+ #script: rake test --trace
17
+ #
18
+ #before_install:
19
+ # - gem update --system
20
+ # - gem --version
21
+ # - gem install rake bacon
22
+ # - rake gem
23
+ # - gem install pkg/*.gem
data/.yardopts ADDED
@@ -0,0 +1 @@
1
+ --markup markdown
data/Gemfile ADDED
@@ -0,0 +1,2 @@
1
+ source :rubygems
2
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,25 @@
1
+ License
2
+ -------
3
+
4
+ (The MIT License)
5
+
6
+ Copyright (c) 2011 John Mair (banisterfiend)
7
+
8
+ Permission is hereby granted, free of charge, to any person obtaining
9
+ a copy of this software and associated documentation files (the
10
+ 'Software'), to deal in the Software without restriction, including
11
+ without limitation the rights to use, copy, modify, merge, publish,
12
+ distribute, sublicense, and/or sell copies of the Software, and to
13
+ permit persons to whom the Software is furnished to do so, subject to
14
+ the following conditions:
15
+
16
+ The above copyright notice and this permission notice shall be
17
+ included in all copies or substantial portions of the Software.
18
+
19
+ THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND,
20
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
21
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
22
+ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
23
+ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
24
+ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
25
+ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md CHANGED
@@ -1,27 +1,59 @@
1
+ [![Build Status](https://secure.travis-ci.org/banister/binding_of_caller.png)](http://travis-ci.org/banister/binding_of_caller)
2
+
1
3
  binding_of_caller
2
4
  ===========
3
5
 
4
6
  (C) John Mair (banisterfiend) 2011
5
7
 
6
- FIXME: _tagline_
8
+ _Retrieve the binding of a method's caller in MRI 1.9.2+, and RBX (Rubinius)_
9
+
10
+ The `binding_of_caller` gem provides the `Binding#of_caller` method.
11
+
12
+ Using `binding_of_caller` we can grab bindings from higher up the call
13
+ stack and evaluate code in that context. Allows access to bindings arbitrarily far up the
14
+ call stack, not limited to just the immediate caller.
7
15
 
8
- FIXME: _description goes here_
16
+ **Recommended for use only in debugging situations. Do not use this in production apps.**
17
+
18
+ **Only works in MRI Ruby 1.9.2, 1.9.3 and RBX (Rubinius)**
9
19
 
10
20
  * Install the [gem](https://rubygems.org/gems/binding_of_caller): `gem install binding_of_caller`
11
- * Read the [documentation](http://rdoc.info/github/banister/binding_of_caller/master/file/README.markdown)
12
21
  * See the [source code](http://github.com/banister/binding_of_caller)
13
22
 
14
- Example: Example description
23
+ Example: Modifying a local inside the caller of a caller
15
24
  --------
16
25
 
17
- Example preamble
26
+ ```ruby
27
+ def a
28
+ var = 10
29
+ b
30
+ puts var
31
+ end
32
+
33
+ def b
34
+ c
35
+ end
36
+
37
+ def c
38
+ binding.of_caller(2).eval('var = :hello')
39
+ end
40
+
41
+ a()
42
+
43
+ # OUTPUT
44
+ # => hello
45
+ ```
46
+
47
+ Spinoff project
48
+ -------
18
49
 
19
- puts "example code"
50
+ This project is a spinoff from the [Pry REPL project.](http://pry.github.com)
20
51
 
21
52
  Features and limitations
22
53
  -------------------------
23
54
 
24
- Feature List Preamble
55
+ * Only works with MRI 1.9.2, 1.9.3 and RBX (Rubinius)
56
+ * Does not work in 1.8.7, but there is a well known (continuation-based) hack to get a `Binding#of_caller` there.
25
57
 
26
58
  Contact
27
59
  -------
@@ -32,7 +64,7 @@ Problems or questions contact me at [github](http://github.com/banister)
32
64
  License
33
65
  -------
34
66
 
35
- (The MIT License)
67
+ (The MIT License)
36
68
 
37
69
  Copyright (c) 2011 (John Mair)
38
70
 
data/Rakefile CHANGED
@@ -1,99 +1,125 @@
1
- dlext = Config::CONFIG['DLEXT']
2
- direc = File.dirname(__FILE__)
3
-
4
- PROJECT_NAME = "binding_of_caller"
5
-
6
- require 'rake/clean'
7
- require 'rake/gempackagetask'
8
- require "#{direc}/lib/#{PROJECT_NAME}/version"
9
-
10
- CLOBBER.include("**/*.#{dlext}", "**/*~", "**/*#*", "**/*.log", "**/*.o")
11
- CLEAN.include("ext/**/*.#{dlext}", "ext/**/*.log", "ext/**/*.o",
12
- "ext/**/*~", "ext/**/*#*", "ext/**/*.obj", "**/*#*", "**/*#*.*",
13
- "ext/**/*.def", "ext/**/*.pdb", "**/*_flymake*.*", "**/*_flymake")
14
-
15
- def apply_spec_defaults(s)
16
- s.name = PROJECT_NAME
17
- s.summary = "FIX ME"
18
- s.version = BindingOfCaller::VERSION
19
- s.date = Time.now.strftime '%Y-%m-%d'
20
- s.author = "John Mair (banisterfiend)"
21
- s.email = 'jrmair@gmail.com'
22
- s.description = s.summary
23
- s.require_path = 'lib'
24
- s.add_development_dependency("bacon",">=1.1.0")
25
- s.homepage = "http://banisterfiend.wordpress.com"
26
- s.has_rdoc = 'yard'
27
- s.files = Dir["ext/**/extconf.rb", "ext/**/*.h", "ext/**/*.c", "lib/**/*.rb",
28
- "test/*.rb", "HISTORY", "README.md", "Rakefile"]
29
- end
30
-
31
- desc "run tests"
32
- task :test do
33
- sh "bacon -k #{direc}/test/test.rb"
34
- end
35
-
36
- [:mingw32, :mswin32].each do |v|
37
- namespace v do
38
- spec = Gem::Specification.new do |s|
39
- apply_spec_defaults(s)
40
- s.platform = "i386-#{v}"
41
- s.files += FileList["lib/**/*.#{dlext}"].to_a
42
- end
43
-
44
- Rake::GemPackageTask.new(spec) do |pkg|
45
- pkg.need_zip = false
46
- pkg.need_tar = false
47
- end
48
- end
49
- end
50
-
51
- namespace :ruby do
52
- spec = Gem::Specification.new do |s|
53
- apply_spec_defaults(s)
54
- s.platform = Gem::Platform::RUBY
55
- s.extensions = ["ext/#{PROJECT_NAME}/extconf.rb"]
56
- end
57
-
58
- Rake::GemPackageTask.new(spec) do |pkg|
59
- pkg.need_zip = false
60
- pkg.need_tar = false
61
- end
62
- end
63
-
64
- directories = ["#{direc}/lib/1.8", "#{direc}/lib/1.9"]
65
- directories.each { |d| directory d }
66
-
67
- desc "build the 1.8 and 1.9 binaries from source and copy to lib/"
68
- task :compile => directories do
69
- build_for = proc do |pik_ver, ver|
70
- sh %{ \
71
- c:\\devkit\\devkitvars.bat && \
72
- pik #{pik_ver} && \
73
- ruby extconf.rb && \
74
- make clean && \
75
- make && \
76
- cp *.so #{direc}/lib/#{ver} \
77
- }
78
- end
79
-
80
- chdir("#{direc}/ext/#{PROJECT_NAME}") do
81
- build_for.call("187", "1.8")
82
- build_for.call("192", "1.9")
83
- end
84
- end
85
-
86
- desc "build all platform gems at once"
87
- task :gems => [:clean, :rmgems, "mingw32:gem", "mswin32:gem", "ruby:gem"]
88
-
89
- desc "remove all platform gems"
90
- task :rmgems => ["ruby:clobber_package"]
91
-
92
- desc "build and push latest gems"
93
- task :pushgems => :gems do
94
- chdir("#{direc}/pkg") do
95
- Dir["*.gem"].each do |gemfile|
96
- sh "gem push #{gemfile}"
97
- end
98
- end
99
- end
1
+ dlext = RbConfig::CONFIG['DLEXT']
2
+ direc = File.dirname(__FILE__)
3
+
4
+ $:.unshift 'lib'
5
+
6
+ PROJECT_NAME = "binding_of_caller"
7
+
8
+ require 'rake/clean'
9
+ require 'rubygems/package_task'
10
+
11
+ require "#{PROJECT_NAME}/version"
12
+
13
+ CLOBBER.include("**/*.#{dlext}", "**/*~", "**/*#*", "**/*.log", "**/*.o")
14
+ CLEAN.include("ext/**/*.#{dlext}", "ext/**/*.log", "ext/**/*.o",
15
+ "ext/**/*~", "ext/**/*#*", "ext/**/*.obj", "**/*#*", "**/*#*.*",
16
+ "ext/**/*.def", "ext/**/*.pdb", "**/*_flymake*.*", "**/*_flymake", "**/*.rbc")
17
+
18
+ def apply_spec_defaults(s)
19
+ s.name = PROJECT_NAME
20
+ s.summary = "Retrieve the binding of a method's caller. Can also retrieve bindings even further up the stack."
21
+ s.version = BindingOfCaller::VERSION
22
+ s.date = Time.now.strftime '%Y-%m-%d'
23
+ s.author = "John Mair (banisterfiend)"
24
+ s.email = 'jrmair@gmail.com'
25
+ s.description = s.summary
26
+ s.require_path = 'lib'
27
+ s.add_development_dependency('bacon')
28
+ s.add_development_dependency('rake')
29
+ s.homepage = "http://github.com/banister/binding_of_caller"
30
+ s.has_rdoc = 'yard'
31
+ s.files = `git ls-files`.split("\n")
32
+ s.test_files = `git ls-files -- test/*`.split("\n")
33
+ end
34
+
35
+ desc "Show version"
36
+ task :version do
37
+ puts "BindingOfCaller version: #{BindingOfCaller::VERSION}"
38
+ end
39
+
40
+ desc "run tests"
41
+ task :default => [:test]
42
+
43
+ desc "Run tests"
44
+ task :test do
45
+ unless defined?(Rubinius)
46
+ Rake::Task['compile'].execute
47
+ end
48
+
49
+ $stdout.puts("\033[33m")
50
+ sh "bacon -Itest -rubygems -a -q"
51
+ $stdout.puts("\033[0m")
52
+
53
+ unless defined?(Rubinius)
54
+ Rake::Task['cleanup'].execute
55
+ end
56
+ end
57
+
58
+ task :pry do
59
+ puts "loading binding_of_caller into pry"
60
+ sh "pry -r ./lib/binding_of_caller"
61
+ end
62
+
63
+ desc "generate gemspec"
64
+ task :gemspec => "ruby:gemspec"
65
+
66
+ namespace :ruby do
67
+ spec = Gem::Specification.new do |s|
68
+ apply_spec_defaults(s)
69
+ s.platform = Gem::Platform::RUBY
70
+ s.extensions = ["ext/#{PROJECT_NAME}/extconf.rb"]
71
+ end
72
+
73
+ Gem::PackageTask.new(spec) do |pkg|
74
+ pkg.need_zip = false
75
+ pkg.need_tar = false
76
+ end
77
+
78
+ desc "Generate gemspec file"
79
+ task :gemspec do
80
+ File.open("#{spec.name}.gemspec", "w") do |f|
81
+ f << spec.to_ruby
82
+ end
83
+ end
84
+ end
85
+
86
+ desc "build the binaries"
87
+ task :compile do
88
+ chdir "./ext/#{PROJECT_NAME}/" do
89
+ sh "ruby extconf.rb"
90
+ sh "make clean"
91
+ sh "make"
92
+ sh "cp *.#{dlext} ../../lib/"
93
+ end
94
+ end
95
+
96
+ desc 'cleanup the extensions'
97
+ task :cleanup do
98
+ sh 'rm -rf lib/binding_of_caller.so'
99
+ chdir "./ext/#{PROJECT_NAME}/" do
100
+ sh 'make clean'
101
+ end
102
+ end
103
+
104
+ desc "reinstall gem"
105
+ task :reinstall => :gems do
106
+ sh "gem uninstall binding_of_caller" rescue nil
107
+ sh "gem install #{direc}/pkg/#{PROJECT_NAME}-#{BindingOfCaller::VERSION}.gem"
108
+ end
109
+
110
+ desc "build all platform gems at once"
111
+ task :gems => [:clean, :rmgems, "ruby:gem"]
112
+
113
+ task :gem => [:gems]
114
+
115
+ desc "remove all platform gems"
116
+ task :rmgems => ["ruby:clobber_package"]
117
+
118
+ desc "build and push latest gems"
119
+ task :pushgems => :gems do
120
+ chdir("./pkg") do
121
+ Dir["*.gem"].each do |gemfile|
122
+ sh "gem push #{gemfile}"
123
+ end
124
+ end
125
+ end
@@ -0,0 +1,34 @@
1
+ # -*- encoding: utf-8 -*-
2
+
3
+ Gem::Specification.new do |s|
4
+ s.name = "binding_of_caller"
5
+ s.version = "0.6.8"
6
+
7
+ s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
8
+ s.authors = ["John Mair (banisterfiend)"]
9
+ s.date = "2012-09-12"
10
+ s.description = "Retrieve the binding of a method's caller. Can also retrieve bindings even further up the stack."
11
+ s.email = "jrmair@gmail.com"
12
+ s.extensions = ["ext/binding_of_caller/extconf.rb"]
13
+ s.files = [".gemtest", ".gitignore", ".travis.yml", ".yardopts", "Gemfile", "HISTORY", "LICENSE", "README.md", "Rakefile", "binding_of_caller.gemspec", "examples/example.rb", "ext/binding_of_caller/binding_of_caller.c", "ext/binding_of_caller/extconf.rb", "ext/binding_of_caller/ruby_headers/192/debug.h", "ext/binding_of_caller/ruby_headers/192/dln.h", "ext/binding_of_caller/ruby_headers/192/eval_intern.h", "ext/binding_of_caller/ruby_headers/192/gc.h", "ext/binding_of_caller/ruby_headers/192/id.h", "ext/binding_of_caller/ruby_headers/192/iseq.h", "ext/binding_of_caller/ruby_headers/192/method.h", "ext/binding_of_caller/ruby_headers/192/node.h", "ext/binding_of_caller/ruby_headers/192/regenc.h", "ext/binding_of_caller/ruby_headers/192/regint.h", "ext/binding_of_caller/ruby_headers/192/regparse.h", "ext/binding_of_caller/ruby_headers/192/thread_pthread.h", "ext/binding_of_caller/ruby_headers/192/thread_win32.h", "ext/binding_of_caller/ruby_headers/192/timev.h", "ext/binding_of_caller/ruby_headers/192/transcode_data.h", "ext/binding_of_caller/ruby_headers/192/version.h", "ext/binding_of_caller/ruby_headers/192/vm_core.h", "ext/binding_of_caller/ruby_headers/192/vm_exec.h", "ext/binding_of_caller/ruby_headers/192/vm_insnhelper.h", "ext/binding_of_caller/ruby_headers/192/vm_opts.h", "ext/binding_of_caller/ruby_headers/193/addr2line.h", "ext/binding_of_caller/ruby_headers/193/atomic.h", "ext/binding_of_caller/ruby_headers/193/constant.h", "ext/binding_of_caller/ruby_headers/193/debug.h", "ext/binding_of_caller/ruby_headers/193/dln.h", "ext/binding_of_caller/ruby_headers/193/encdb.h", "ext/binding_of_caller/ruby_headers/193/eval_intern.h", "ext/binding_of_caller/ruby_headers/193/gc.h", "ext/binding_of_caller/ruby_headers/193/id.h", "ext/binding_of_caller/ruby_headers/193/internal.h", "ext/binding_of_caller/ruby_headers/193/iseq.h", "ext/binding_of_caller/ruby_headers/193/method.h", "ext/binding_of_caller/ruby_headers/193/node.h", "ext/binding_of_caller/ruby_headers/193/parse.h", "ext/binding_of_caller/ruby_headers/193/regenc.h", "ext/binding_of_caller/ruby_headers/193/regint.h", "ext/binding_of_caller/ruby_headers/193/regparse.h", "ext/binding_of_caller/ruby_headers/193/revision.h", "ext/binding_of_caller/ruby_headers/193/thread_pthread.h", "ext/binding_of_caller/ruby_headers/193/thread_win32.h", "ext/binding_of_caller/ruby_headers/193/timev.h", "ext/binding_of_caller/ruby_headers/193/transcode_data.h", "ext/binding_of_caller/ruby_headers/193/transdb.h", "ext/binding_of_caller/ruby_headers/193/version.h", "ext/binding_of_caller/ruby_headers/193/vm_core.h", "ext/binding_of_caller/ruby_headers/193/vm_exec.h", "ext/binding_of_caller/ruby_headers/193/vm_insnhelper.h", "ext/binding_of_caller/ruby_headers/193/vm_opts.h", "lib/binding_of_caller.rb", "lib/binding_of_caller/version.rb", "test/test_binding_of_caller.rb"]
14
+ s.homepage = "http://github.com/banister/binding_of_caller"
15
+ s.require_paths = ["lib"]
16
+ s.rubygems_version = "1.8.11"
17
+ s.summary = "Retrieve the binding of a method's caller. Can also retrieve bindings even further up the stack."
18
+ s.test_files = ["test/test_binding_of_caller.rb"]
19
+
20
+ if s.respond_to? :specification_version then
21
+ s.specification_version = 3
22
+
23
+ if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then
24
+ s.add_development_dependency(%q<bacon>, [">= 0"])
25
+ s.add_development_dependency(%q<rake>, [">= 0"])
26
+ else
27
+ s.add_dependency(%q<bacon>, [">= 0"])
28
+ s.add_dependency(%q<rake>, [">= 0"])
29
+ end
30
+ else
31
+ s.add_dependency(%q<bacon>, [">= 0"])
32
+ s.add_dependency(%q<rake>, [">= 0"])
33
+ end
34
+ end
@@ -0,0 +1,41 @@
1
+ unless Object.const_defined? :BindingOfCaller
2
+ $:.unshift File.expand_path '../../lib', __FILE__
3
+ require 'binding_of_caller'
4
+ require 'binding_of_caller/version'
5
+ end
6
+
7
+ outer = 10
8
+
9
+ class Z
10
+ def z
11
+ u = 10
12
+ A.new.a
13
+ end
14
+ end
15
+
16
+ class A
17
+ def a
18
+ y = 10
19
+ B.new.b
20
+ end
21
+ end
22
+
23
+ class B
24
+ def b
25
+ x = 10
26
+ puts binding.of_caller(0).eval('local_variables')
27
+ puts binding.of_caller(1).eval('local_variables')
28
+ puts binding.of_caller(2).eval('local_variables')
29
+ puts binding.of_caller(3).eval('local_variables')
30
+ puts binding.of_caller(400).eval('local_variables')
31
+ end
32
+ end
33
+
34
+ Z.new.z
35
+
36
+ # output:
37
+ # => x
38
+ # => y
39
+ # => u
40
+ # => outer
41
+ # Exception
@@ -1,22 +1,31 @@
1
1
  /* (c) 2011 John Mair (banisterfiend), MIT license */
2
2
 
3
3
  #include <ruby.h>
4
- //#include "compat.h"
5
-
6
- //#ifdef RUBY_19
7
- # include <ruby/io.h>
8
- # include <ruby/re.h>
9
- # include "vm_core.h"
10
- # include "gc.h"
11
- /* #else */
12
- /* # include "re.h" */
13
- /* # include "env.h" */
14
- /* # include "node.h" */
15
- /* # include "rubysig.h" */
16
- /* # include "rubyio.h" */
17
- /* #endif */
18
-
19
- extern rb_thread_t *ruby_current_thread;
4
+ #include "vm_core.h"
5
+ #include "gc.h"
6
+
7
+ typedef enum { false, true } bool;
8
+
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()))
20
29
 
21
30
  static size_t
22
31
  binding_memsize(const void *ptr)
@@ -44,7 +53,11 @@ binding_mark(void *ptr)
44
53
  if (ptr) {
45
54
  bind = ptr;
46
55
  RUBY_MARK_UNLESS_NULL(bind->env);
56
+
57
+ #ifdef RUBY_192
47
58
  RUBY_MARK_UNLESS_NULL(bind->filename);
59
+ #endif
60
+
48
61
  }
49
62
  RUBY_MARK_LEAVE("binding");
50
63
  }
@@ -65,60 +78,147 @@ binding_alloc(VALUE klass)
65
78
  return obj;
66
79
  }
67
80
 
68
- typedef enum { false, true } bool;
69
-
70
- static bool valid_frame_p(rb_control_frame_t * cfp) {
71
- return cfp->iseq && !NIL_P(cfp->self);
81
+ static bool ifunc_p(rb_control_frame_t * cfp) {
82
+ return (cfp->flag & VM_FRAME_MAGIC_MASK) == VM_FRAME_MAGIC_IFUNC;
72
83
  }
73
84
 
74
- static rb_control_frame_t * find_valid_frame(rb_control_frame_t * cfp) {
75
- int error_count = 0;
85
+ static bool valid_frame_p(rb_control_frame_t * cfp, rb_control_frame_t * limit_cfp) {
86
+ return cfp->iseq && !ifunc_p(cfp) && !NIL_P(cfp->self);
87
+ }
76
88
 
77
- while (error_count <= 4) {
89
+ static rb_control_frame_t * find_valid_frame(rb_control_frame_t * cfp, rb_control_frame_t * limit_cfp) {
90
+ while (cfp < limit_cfp) {
78
91
  cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
79
92
 
80
- if (valid_frame_p(cfp))
93
+ if (cfp >= limit_cfp)
94
+ return NULL;
95
+
96
+ if (valid_frame_p(cfp, limit_cfp))
81
97
  return cfp;
82
- else
83
- error_count += 1;
84
98
  }
85
99
 
86
- rb_raise(rb_eRuntimeError, "No valid stack frame found.");
100
+ // beyond end of stack
101
+ return NULL;
102
+ }
87
103
 
88
- // never reached
89
- return 0;
104
+ static VALUE
105
+ frametype_name(VALUE flag)
106
+ {
107
+ switch (flag & VM_FRAME_MAGIC_MASK) {
108
+ case VM_FRAME_MAGIC_METHOD: return string2sym("method");
109
+ case VM_FRAME_MAGIC_BLOCK: return string2sym("block");
110
+ case VM_FRAME_MAGIC_CLASS: return string2sym("class");
111
+ case VM_FRAME_MAGIC_TOP: return string2sym("top");
112
+ case VM_FRAME_MAGIC_FINISH: return string2sym("finish");
113
+ case VM_FRAME_MAGIC_CFUNC: return string2sym("cfunc");
114
+ case VM_FRAME_MAGIC_PROC: return string2sym("proc");
115
+ case VM_FRAME_MAGIC_IFUNC: return string2sym("ifunc");
116
+ case VM_FRAME_MAGIC_EVAL: return string2sym("eval");
117
+ case VM_FRAME_MAGIC_LAMBDA: return string2sym("lambda");
118
+ default:
119
+ rb_raise(rb_eRuntimeError, "Unknown frame type! got flag: %d", FIX2INT(flag));
120
+ }
90
121
  }
91
122
 
92
123
  static VALUE binding_of_caller(VALUE self, VALUE rb_level)
93
124
  {
94
- rb_thread_t *th = GET_THREAD();
125
+ rb_thread_t *th;
126
+ GetThreadPtr(rb_thread_current(), th);
127
+
95
128
  rb_control_frame_t *cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(th->cfp);
129
+ rb_control_frame_t *limit_cfp = (void *)(th->stack + th->stack_size);
96
130
  int level = FIX2INT(rb_level);
97
131
 
98
- for (int i = 0; i < level; i++)
132
+ // attempt to locate the nth parent control frame
133
+ for (int i = 0; i < level; i++) {
99
134
  cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
100
135
 
101
- if (!valid_frame_p(cfp))
102
- cfp = find_valid_frame(cfp);
136
+ if (cfp >= limit_cfp)
137
+ rb_raise(rb_eRuntimeError, "Invalid frame, gone beyond end of stack!");
138
+
139
+ // skip invalid frames
140
+ if (!valid_frame_p(cfp, limit_cfp))
141
+ cfp = find_valid_frame(cfp, limit_cfp);
142
+ }
103
143
 
104
144
  VALUE bindval = binding_alloc(rb_cBinding);
105
145
  rb_binding_t *bind;
106
146
 
107
- if (cfp == 0) {
147
+ if (cfp == 0)
108
148
  rb_raise(rb_eRuntimeError, "Can't create Binding Object on top of Fiber.");
109
- }
110
149
 
111
150
  GetBindingPtr(bindval, bind);
112
151
  bind->env = rb_vm_make_env_object(th, cfp);
113
152
  bind->filename = cfp->iseq->filename;
114
153
  bind->line_no = rb_vm_get_sourceline(cfp);
154
+
155
+ rb_iv_set(bindval, "@frame_type", frametype_name(cfp->flag));
156
+ rb_iv_set(bindval, "@frame_description", cfp->iseq->name);
157
+
115
158
  return bindval;
116
159
  }
117
160
 
161
+ static VALUE
162
+ frame_type(VALUE self)
163
+ {
164
+ return rb_iv_get(self, "@frame_type");
165
+ }
166
+
167
+ static VALUE
168
+ frame_description(VALUE self)
169
+ {
170
+ return rb_iv_get(self, "@frame_description");
171
+ }
172
+
173
+ static VALUE frame_count(VALUE self)
174
+ {
175
+ rb_thread_t *th;
176
+ GetThreadPtr(rb_thread_current(), th);
177
+
178
+ rb_control_frame_t *cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(th->cfp);
179
+ rb_control_frame_t *limit_cfp = (void *)(th->stack + th->stack_size);
180
+
181
+ int i = 1;
182
+ while (cfp < limit_cfp) {
183
+ cfp = RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp);
184
+
185
+ if (cfp >= limit_cfp)
186
+ return INT2FIX(i);
187
+
188
+ // skip invalid frames
189
+ if (!valid_frame_p(cfp, limit_cfp))
190
+ cfp = find_valid_frame(cfp, limit_cfp);
191
+
192
+ if (!cfp)
193
+ break;
194
+
195
+ i++;
196
+ }
197
+
198
+ return INT2FIX(i);
199
+ }
200
+
201
+ static VALUE
202
+ callers(VALUE self)
203
+ {
204
+ VALUE ary = rb_ary_new();
205
+
206
+ for (int i = 0; i < FIX2INT(frame_count(self)); i++)
207
+ rb_ary_push(ary, binding_of_caller(self, INT2FIX(i)));
208
+
209
+ return ary;
210
+ }
211
+
118
212
  void
119
213
  Init_binding_of_caller()
120
214
  {
121
- rb_define_method(rb_cObject, "binding_of_caller", binding_of_caller, 1);
122
-
215
+ VALUE mBindingOfCaller = rb_define_module("BindingOfCaller");
216
+
217
+ rb_define_method(mBindingOfCaller, "of_caller", binding_of_caller, 1);
218
+ rb_define_method(mBindingOfCaller, "frame_count", frame_count, 0);
219
+ rb_define_method(mBindingOfCaller, "frame_type", frame_type, 0);
220
+ rb_define_method(mBindingOfCaller, "frame_description", frame_description, 0);
221
+ rb_define_method(mBindingOfCaller, "callers", callers, 0);
222
+ rb_include_module(rb_cBinding, mBindingOfCaller);
123
223
  }
124
224
 
@@ -1,14 +1,24 @@
1
- require 'mkmf'
1
+ def fake_makefile
2
+ File.open(File.join(File.dirname(__FILE__), "Makefile"), "w") {|f|
3
+ f.puts %[install:\n\techo "Nada."]
4
+ }
5
+ end
6
+
7
+ if RUBY_ENGINE && RUBY_ENGINE =~ /rbx/
8
+ fake_makefile
9
+ else
10
+ require 'mkmf'
11
+
12
+ $CFLAGS += " -O0"
13
+ $CFLAGS += " -std=c99"
2
14
 
3
- $CFLAGS += " -O0"
4
- $CFLAGS += " -std=c99"
15
+ case RUBY_VERSION
16
+ when /1.9.2/
17
+ $CFLAGS += " -I./ruby_headers/192/ -DRUBY_192"
18
+ when /1.9.3/
19
+ $CFLAGS += " -I./ruby_headers/193/ -DRUBY_193"
20
+ end
5
21
 
6
- case RUBY_VERSION
7
- when /1.9.2/
8
- $CFLAGS += " -I./ruby_headers/192/"
9
- when /1.9.3/
10
- puts "hit 193"
11
- $CFLAGS += " -I./ruby_headers/193/"
22
+ create_makefile('binding_of_caller')
12
23
  end
13
24
 
14
- create_makefile('binding_of_caller')
@@ -1,3 +1,3 @@
1
- module BindingOfCaller
2
- VERSION = "0.2.0"
3
- end
1
+ module BindingOfCaller
2
+ VERSION = "0.6.8"
3
+ end
@@ -1,22 +1,85 @@
1
- # binding_of_caller.rb
2
- # (C) John Mair (banisterfiend); MIT license
3
-
1
+ dlext = RbConfig::CONFIG['DLEXT']
4
2
  direc = File.dirname(__FILE__)
5
3
 
6
- require "#{direc}/binding_of_caller/version"
4
+ if defined?(RUBY_ENGINE) && RUBY_ENGINE == "ruby"
5
+ require "binding_of_caller.#{dlext}"
7
6
 
8
- begin
9
- if RUBY_VERSION =~ /1.9/
10
- require "#{direc}/1.9/binding_of_caller"
11
- else
12
- require "#{direc}/1.8/binding_of_caller"
13
- end
14
- rescue LoadError => e
15
- require "rbconfig"
16
- dlext = Config::CONFIG['DLEXT']
17
- require "#{direc}/binding_of_caller.#{dlext}"
18
- end
7
+ elsif defined?(Rubinius)
8
+ module BindingOfCaller
9
+ module BindingExtensions
10
+
11
+ # Retrieve the binding of the nth caller of the current frame.
12
+ # @return [Binding]
13
+ def of_caller(n)
14
+ bt = Rubinius::VM.backtrace(1 + n, true).first
15
+
16
+ b = Binding.setup(
17
+ bt.variables,
18
+ bt.variables.method,
19
+ bt.static_scope,
20
+ bt.variables.self,
21
+ bt
22
+ )
23
+
24
+ b.instance_variable_set(:@frame_description, bt.describe)
25
+
26
+ b
27
+ rescue
28
+ raise RuntimeError, "Invalid frame, gone beyond end of stack!"
29
+ end
19
30
 
20
- module BindingOfCaller
21
- end
31
+ # The description of the frame.
32
+ # @return [String]
33
+ def frame_description
34
+ @frame_description
35
+ end
22
36
 
37
+ # Return bindings for all caller frames.
38
+ # @return [Array<Binding>]
39
+ def callers
40
+ ary = []
41
+ n = 0
42
+ loop do
43
+ begin
44
+ ary << Binding.of_caller(n)
45
+ rescue
46
+ break
47
+ end
48
+ n += 1
49
+ end
50
+ ary.drop_while do |v|
51
+ !(v.frame_type == :method && v.eval("__method__") == :callers)
52
+ end.drop(1)
53
+ end
54
+
55
+ # Number of parent frames available at the point of call.
56
+ # @return [Fixnum]
57
+ def frame_count
58
+ callers.size - 1
59
+ end
60
+
61
+ # The type of the frame.
62
+ # @return [Symbol]
63
+ def frame_type
64
+ case self.variables.method.metadata.to_a.first.to_s
65
+ when /block/
66
+ :block
67
+ when /eval/
68
+ :eval
69
+ else
70
+ if frame_description =~ /__(?:class|module)_init__/
71
+ :class
72
+ else
73
+ :method
74
+ end
75
+ end
76
+ end
77
+
78
+ end
79
+ end
80
+
81
+ class ::Binding
82
+ include BindingOfCaller::BindingExtensions
83
+ extend BindingOfCaller::BindingExtensions
84
+ end
85
+ end
@@ -0,0 +1,108 @@
1
+ unless Object.const_defined? :BindingOfCaller
2
+ $:.unshift File.expand_path '../../lib', __FILE__
3
+ require 'binding_of_caller'
4
+ require 'binding_of_caller/version'
5
+ end
6
+
7
+ puts "Testing binding_of_caller version #{BindingOfCaller::VERSION}..."
8
+ puts "Ruby version: #{RUBY_VERSION}"
9
+
10
+ describe BindingOfCaller do
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
20
+ end
21
+
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
28
+
29
+ def o.b
30
+ binding.of_caller(1).eval('var')
31
+ end
32
+
33
+ o.a.should == 1
34
+ end
35
+
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
47
+
48
+ o.a.should == 20
49
+ end
50
+
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
58
+ end
59
+ end
60
+
61
+ describe "callers" do
62
+ before do
63
+ @o = Object.new
64
+ end
65
+
66
+ it 'should return the first non-internal binding when using callers.first' do
67
+ def @o.meth
68
+ x = :a_local
69
+ [binding.callers.first, binding.of_caller(0)]
70
+ end
71
+
72
+ b1, b2 = @o.meth
73
+ b1.eval("x").should == :a_local
74
+ b2.eval("x").should == :a_local
75
+ end
76
+ end
77
+
78
+ describe "frame_count" do
79
+ it 'frame_count should equal callers.count' do
80
+ binding.frame_count.should == binding.callers.count
81
+ end
82
+ end
83
+
84
+ describe "frame_type" do
85
+ it 'should return the correct frame types' do
86
+ o = Object.new
87
+
88
+ # method frame
89
+ def o.a
90
+ b
91
+ end
92
+
93
+ # method frame
94
+ def o.b
95
+ # block frame
96
+ proc do
97
+ binding.callers
98
+ end.call
99
+ end
100
+ caller_bindings = o.a
101
+ caller_bindings[0].frame_type.should == :block
102
+ caller_bindings[1].frame_type.should == :method
103
+ caller_bindings[2].frame_type.should == :method
104
+ end
105
+
106
+ end
107
+ end
108
+
metadata CHANGED
@@ -1,39 +1,59 @@
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.8
4
5
  prerelease:
5
- version: 0.2.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-17 00:00:00 Z
14
- dependencies:
15
- - !ruby/object:Gem::Dependency
12
+ date: 2012-09-12 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
16
15
  name: bacon
16
+ requirement: &70130738847880 !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ! '>='
20
+ - !ruby/object:Gem::Version
21
+ version: '0'
22
+ type: :development
17
23
  prerelease: false
18
- requirement: &id001 !ruby/object:Gem::Requirement
24
+ version_requirements: *70130738847880
25
+ - !ruby/object:Gem::Dependency
26
+ name: rake
27
+ requirement: &70130738846740 !ruby/object:Gem::Requirement
19
28
  none: false
20
- requirements:
21
- - - ">="
22
- - !ruby/object:Gem::Version
23
- version: 1.1.0
29
+ requirements:
30
+ - - ! '>='
31
+ - !ruby/object:Gem::Version
32
+ version: '0'
24
33
  type: :development
25
- version_requirements: *id001
26
- description: FIX ME
34
+ prerelease: false
35
+ version_requirements: *70130738846740
36
+ description: Retrieve the binding of a method's caller. Can also retrieve bindings
37
+ even further up the stack.
27
38
  email: jrmair@gmail.com
28
39
  executables: []
29
-
30
- extensions:
40
+ extensions:
31
41
  - ext/binding_of_caller/extconf.rb
32
42
  extra_rdoc_files: []
33
-
34
- files:
43
+ files:
44
+ - .gemtest
45
+ - .gitignore
46
+ - .travis.yml
47
+ - .yardopts
48
+ - Gemfile
49
+ - HISTORY
50
+ - LICENSE
51
+ - README.md
52
+ - Rakefile
53
+ - binding_of_caller.gemspec
54
+ - examples/example.rb
55
+ - ext/binding_of_caller/binding_of_caller.c
35
56
  - ext/binding_of_caller/extconf.rb
36
- - ext/binding_of_caller/compat.h
37
57
  - ext/binding_of_caller/ruby_headers/192/debug.h
38
58
  - ext/binding_of_caller/ruby_headers/192/dln.h
39
59
  - ext/binding_of_caller/ruby_headers/192/eval_intern.h
@@ -82,39 +102,33 @@ files:
82
102
  - ext/binding_of_caller/ruby_headers/193/vm_exec.h
83
103
  - ext/binding_of_caller/ruby_headers/193/vm_insnhelper.h
84
104
  - ext/binding_of_caller/ruby_headers/193/vm_opts.h
85
- - ext/binding_of_caller/binding_of_caller.c
86
- - lib/binding_of_caller/version.rb
87
105
  - lib/binding_of_caller.rb
88
- - test/test.rb
89
- - HISTORY
90
- - README.md
91
- - Rakefile
92
- homepage: http://banisterfiend.wordpress.com
106
+ - lib/binding_of_caller/version.rb
107
+ - test/test_binding_of_caller.rb
108
+ homepage: http://github.com/banister/binding_of_caller
93
109
  licenses: []
94
-
95
110
  post_install_message:
96
111
  rdoc_options: []
97
-
98
- require_paths:
112
+ require_paths:
99
113
  - lib
100
- required_ruby_version: !ruby/object:Gem::Requirement
114
+ required_ruby_version: !ruby/object:Gem::Requirement
101
115
  none: false
102
- requirements:
103
- - - ">="
104
- - !ruby/object:Gem::Version
105
- version: "0"
106
- required_rubygems_version: !ruby/object:Gem::Requirement
116
+ requirements:
117
+ - - ! '>='
118
+ - !ruby/object:Gem::Version
119
+ version: '0'
120
+ required_rubygems_version: !ruby/object:Gem::Requirement
107
121
  none: false
108
- requirements:
109
- - - ">="
110
- - !ruby/object:Gem::Version
111
- version: "0"
122
+ requirements:
123
+ - - ! '>='
124
+ - !ruby/object:Gem::Version
125
+ version: '0'
112
126
  requirements: []
113
-
114
127
  rubyforge_project:
115
128
  rubygems_version: 1.8.11
116
129
  signing_key:
117
130
  specification_version: 3
118
- summary: FIX ME
119
- test_files: []
120
-
131
+ summary: Retrieve the binding of a method's caller. Can also retrieve bindings even
132
+ further up the stack.
133
+ test_files:
134
+ - test/test_binding_of_caller.rb
@@ -1,57 +0,0 @@
1
- /* contains basic macros to facilitate ruby 1.8 and ruby 1.9 compatibility */
2
-
3
- #ifndef GUARD_COMPAT_H
4
- #define GUARD_COMPAT_H
5
-
6
- #include <ruby.h>
7
-
8
- /* test for 1.9 */
9
- #if !defined(RUBY_19) && defined(ROBJECT_EMBED_LEN_MAX)
10
- # define RUBY_19
11
- #endif
12
-
13
- /* macros for backwards compatibility with 1.8 */
14
- #ifndef RUBY_19
15
- # define RCLASS_M_TBL(c) (RCLASS(c)->m_tbl)
16
- # define RCLASS_SUPER(c) (RCLASS(c)->super)
17
- # define RCLASS_IV_TBL(c) (RCLASS(c)->iv_tbl)
18
- # define OBJ_UNTRUSTED OBJ_TAINTED
19
- # include "st.h"
20
- #endif
21
-
22
- #ifdef RUBY_19
23
- inline static VALUE
24
- class_alloc(VALUE flags, VALUE klass)
25
- {
26
- rb_classext_t *ext = ALLOC(rb_classext_t);
27
- NEWOBJ(obj, struct RClass);
28
- OBJSETUP(obj, klass, flags);
29
- obj->ptr = ext;
30
- RCLASS_IV_TBL(obj) = 0;
31
- RCLASS_M_TBL(obj) = 0;
32
- RCLASS_SUPER(obj) = 0;
33
- RCLASS_IV_INDEX_TBL(obj) = 0;
34
- return (VALUE)obj;
35
- }
36
- #endif
37
-
38
- inline static VALUE
39
- create_class(VALUE flags, VALUE klass)
40
- {
41
- #ifdef RUBY_19
42
- VALUE new_klass = class_alloc(flags, klass);
43
- #else
44
- NEWOBJ(new_klass, struct RClass);
45
- OBJSETUP(new_klass, klass, flags);
46
- #endif
47
-
48
- return (VALUE)new_klass;
49
- }
50
-
51
- # define FALSE 0
52
- # define TRUE 1
53
-
54
- /* a useful macro. cannot use ordinary CLASS_OF as it does not return an lvalue */
55
- #define KLASS_OF(c) (RBASIC(c)->klass)
56
-
57
- #endif
data/test/test.rb DELETED
@@ -1,12 +0,0 @@
1
- direc = File.dirname(__FILE__)
2
-
3
- require 'rubygems'
4
- require "#{direc}/../lib/binding_of_caller"
5
- require 'bacon'
6
-
7
- puts "Testing binding_of_caller version #{BindingOfCaller::VERSION}..."
8
- puts "Ruby version: #{RUBY_VERSION}"
9
-
10
- describe BindingOfCaller do
11
- end
12
-