我们最近遇到了一个问题,在一系列提交之后,一个后端进程无法运行。现在,我们都很乖,每次签入后都会运行rake测试,但是,由于Rails库加载中的一些奇怪之处,它只在我们直接从Mongrel在生产模式下运行时才会发生。

我追踪了这个bug,它是由于一个新的Rails gem覆盖了String类中的一个方法,这种方法破坏了运行时Rails代码中的一个狭窄使用。

总之,长话短说,有没有一种方法,在运行时,问Ruby在哪里定义了一个方法?类似于whereami(:foo),返回/path/to/some/file。Rb线45号?在这种情况下,告诉我它是在类String中定义的是没有帮助的,因为它被一些库重载了。

我不能保证源生活在我的项目,所以grepping 'def foo'不一定给我我需要的,更不用说,如果我有很多def foo's,有时我不知道,直到运行时,我可能会使用哪一个。


当前回答

很晚才回答:)但之前的回答对我没有帮助

set_trace_func proc{ |event, file, line, id, binding, classname|
  printf "%8s %s:%-2d %10s %8s\n", event, file, line, id, classname
}
# call your method
set_trace_func nil

其他回答

这可能会有帮助,但您必须自己编写代码。贴自博客:

Ruby提供了method_added() 调用的回调函数 方法中添加或重新定义 类。它是Module类的一部分, 每个类都是一个模块。有 也调用了两个相关的回调 method_removed()和 method_undefined()。

http://scie.nti.st/2008/9/17/making-methods-immutable-in-ruby

如果你能让这个方法崩溃,你会得到一个回溯,它会告诉你它的确切位置。

不幸的是,如果不能使它崩溃,那么就无法找到它的定义位置。如果您试图通过重写或重写方法来操纵该方法,那么任何崩溃都将来自您所重写或重写的方法,并且它将没有任何用处。

崩溃方法的有用方法:

在禁止的地方传递nil——很多时候该方法会在nil类上引发ArgumentError或始终存在的NoMethodError。 如果您有方法的内部知识,并且知道该方法反过来调用其他方法,那么您可以重写其他方法,并在其中引发。

你可以这样做:

foo_finder.rb:

 class String
   def String.method_added(name)
     if (name==:foo)
        puts "defining #{name} in:\n\t"
        puts caller.join("\n\t")
     end
   end
 end

然后确保foo_finder首先加载如下内容

ruby -r foo_finder.rb railsapp

(我只摆弄过rails,所以我不知道确切的情况,但我想有一种方法可以像这样开始。)

这将显示String#foo的所有重新定义。通过一点元编程,您可以将其泛化到任何您想要的函数。但它确实需要在实际执行重定义的文件之前加载。

你总是可以通过使用caller()得到你所在位置的回溯。

也许#source_location可以帮助查找方法的来源。

ex:

ModelName.method(:has_one).source_location

返回

[project_path/vendor/ruby/version_number/gems/activerecord-number/lib/active_record/associations.rb", line_number_of_where_method_is]

OR

ModelName.new.method(:valid?).source_location

返回

[project_path/vendor/ruby/version_number/gems/activerecord-number/lib/active_record/validations.rb", line_number_of_where_method_is]