tags:

views:

100

answers:

1

Hi, I would like to add some debugs for my simple ruby functions and I wrote a function as below,

def debug(&block)
  varname = block.call.to_s
  puts "#{varname} = #{eval(varname,block)}"
end

debug {:x} #prints x = 5
debug {:y} #prints y = 5

I understand that eval is evil. So I have two questions.

  1. Is there any way to write that debug method without using eval? If NO is there a preferred way to do this?
  2. Is there any way to pass a list of arguments to this method? I would ideally prefer debug {:x, :y. :anynumOfvariables}. I could not quite figure out how to factor that into the debug method (i.e, to take a list of arguments)
+8  A: 

Just use arrays. You can use the Array method to ensure that you will always have an array, even if someone passes in only a single value:

def debug(&block)
  Array(block[]).each do |var| puts "#{var} = #{eval var.to_s, block}" end
end

x, y = 3, 5

debug {:x} # => "x = 3"
debug {[:x, :y]} # => "x = 3" "y = 5"

BTW: passing a block as the binding no longer works in Ruby 1.9. (Despite the fact that the documentation says it does work.) You have to explicitly call Proc#binding to get a Binding object for that Proc:

def debug(&block)
  Array(block.()).flatten.each do |var|
    puts "#{var} = #{eval var.to_s, block.binding}"
  end
end

Fortunately, this already works in Ruby 1.8, so you can futureproof your code by including it.

An alternative would be to forgo the block altogether. I mean, you already force the user of the debug to use the unfamiliar idiom of passing arguments in the block instead of in parentheses. Why not force them to just pass the binding instead?

def debug(*vars, bnd)
  vars.each do |var|
    puts "#{var} = #{eval var.to_s, bnd}"
  end
end

x, y = 3, 5

debug :x, binding # => "x = 3"
debug :x, :y, binding # => "x = 3" "y = 5"

This has the added flexibility that they can actually pass a different binding than the one at the callsite, e.g. if they want to actually debug a piece of code in a different piece of the application.


BTW: here's some fun with Ruby 1.9.2's parameter introspection (Proc#parameters):

def debug(&block)
  block.parameters.map(&:last).each do |var|
    puts "#{var} = #{eval var.to_s, block.binding}"
  end
end

x, y = 3, 5

debug {|x|} # => "x = 3"
debug {|x, y|} # => "x = 3" "y = 5"
Jörg W Mittag
@Jörg, nicely done. Do you think that `Array(stuff)` would read better than `[*stuff].flatten?`
Wayne Conrad
@Wayne Conrad: Probably. Frankly, I just didn't think of it, because the splat-flatten idiom is so deeply burnt into my brain. `Array()` breaks if you pass more than one argument, which is why I avoid it, but in this case the language specification guarantees that rest arguments will *always* be an array.
Jörg W Mittag
Very nice write-up, Jörg. This helped me understand some stuff about blocks that confused me for a long time. +1
macek
Jörg, can you explain what `binding` does for the `def debug(*vars, bnd)` example?
macek
@smotchkiss: `binding` is a method that returns a `Binding` object representing the variable bindings at the point where the method is called. To put it in less technical terms: a `Binding` object contains information about which variables exist at a particular point in the code. Normally, when you call `eval`, it uses the variables that exist at the point where you *call* `eval`. But you can also pass in a `Binding`, so that `eval` uses the variables that existed at the place where the `Binding` was created. We want the `debug` method to print the variables from where it was called, not from
Jörg W Mittag
Jörg W Mittag
@Jörg W Mittag, 17K reputation and you haven't asked even a single question. I've been reading your responses all afternoon and they're by far the best I've seen on stackoverflow. You are my hero <3
macek
@smotchkiss: I *tried* to ask questions, but when I started to type them in, the autosuggestion feature popped up, and ... they were all *already* answered :-) That's the true power of StackOverflow: you get your questions answered *before* you even ask them. It's like what the Python community calls the "Guido Time Machine": for every piece of code you want to write, there's already a module in the standard library that solves your problem.
Jörg W Mittag
@Jörg, I'm still reading your answers! Even if they're unrelated to things I'm learning about or interested in! What path did you take to acquire all of this knowledge? I aspire to have mastery such as yours. Perhaps I can contact you personally? You can find my contact details in my profile :)
macek