class Object # # Return a hash of local variables in the caller's scope: {:variable_name=>value} # def locals require 'binding_of_caller' caller = binding.of_caller(1) vars = caller.eval("local_variables").reject{|e| e[/^_/]} vals = caller.eval("[ #{vars.join(",")} ]") Hash[ vars.zip(vals) ] end # # Gives you a copy of the object with its attributes changed to whatever was # passed in the options hash. # # Example: # >> cookie = Cookie.new(:size=>10, :chips=>200) # => # # >> cookie.with(:chips=>50) # => # # # (All this method does is dup the object, then call "key=(value)" for each # key/value in the options hash.) # # == BONUS FEATURE! == # # If you supply a block, it just gives you the object, and # returns whatever your block returns. # # Example: # >> {a: 10, b: 2}.with { |hash| hash[:a] / hash[:b] } # => 5 # # Good for chaining lots of operations together in a REPL. # def with(options={}) if block_given? yield self else obj = dup options.each { |key, value| obj.send "#{key}=", value } obj end end # # Return a copy of the class with modules mixed into it. # def self.using(*args) if block_given? yield using(*args) else copy = self.dup args.each { |arg| copy.send(:include, arg) } copy end end # # Cache (memoize) the result of this method the first time it's called # and return this value for all subsequent calls. # def self.memoize(*methods) methods.each do |meth| old_method = instance_method(meth) if old_method.arity == 0 ivarname = "@#{meth}" define_method meth do instance_variable_get(ivarname) or instance_variable_set(ivarname, old_method.bind(self).call) end else raise "Can't memoize methods with arguments. (YET.)" end end end # # Serialize this object to a binary String, using Marshal.dump. # def marshal Marshal.dump self end alias_method :to_marshal, :marshal # # Serialize this object to YAML. # def to_yaml YAML::dump(self) end # # Serialize this object to JSON (defaults to "pretty" indented JSON). # def to_json(pretty=true) pretty ? JSON::pretty_generate(self) : JSON.dump(self) end # # Proper grammar. # alias_method :is_an?, :is_a? alias_method :responds_to?, :respond_to? # # Emit a quick debug message (only if $DEBUG is true) # def dmsg(msg) if $DEBUG case msg when String puts msg else puts msg.inspect end end end # # Time a block. # def time(message=nil) start = Time.now result = yield elapsed = Time.now - start print "[#{message}] " if message puts "elapsed time: %0.5fs" % elapsed result end # # Quick and easy benchmark. # # Examples: # bench { something } # bench(90000000) { something } # bench :fast => proc { fast_method }, :slow => proc { slow_method } # # In Ruby 1.9: # bench fast: ->{ fast_method }, slow: ->{ slow_method } # def bench(*args, &block) # Shitty perl-esque hack to let the method take all the different kind of arguments. opts = Hash === args.last ? args.pop : {} n = args.first || 100 if opts.any? raise "Error: Supply either a do/end block, or procs as options. Not both." if block_given? raise "Error: Options must be procs." unless opts.all? { |k, v| v.is_a?(Proc) } benchblock = proc do |bm| opts.each do |name, blk| bm.report(name.to_s) { n.times &blk } end end elsif block_given? benchblock = proc do |bm| bm.report { n.times &block } end else raise "Error: Must supply some code to benchmark." end puts "* Benchmarking #{n} iterations..." Benchmark.bmbm(&benchblock) end end