Class: Verbose
- Inherits:
-
Object
- Object
- Verbose
- Defined in:
- lib/verbose.rb
Overview
Decorator that logs method calls to any object, providing transparency into the execution flow by recording method names, parameters, and execution time. This class wraps any Ruby object and intercepts method calls, logging each invocation with its arguments and the time taken to execute.
The decorator uses method_missing to intercept all method calls on the wrapped object, logs them with parameter details, and then delegates to the original object. String parameters are truncated for readability, complex objects are shown as their class names, and primitive values are displayed as-is.
- Author
-
Yegor Bugayenko ([email protected])
- Copyright
-
Copyright © 2024-2025 Yegor Bugayenko
- License
-
MIT
Instance Method Summary collapse
-
#initialize(origin, log: nil) ⇒ Verbose
constructor
A new instance of Verbose.
- #method_missing(*args) ⇒ Object
- #respond_to?(method, include_private = false) ⇒ Boolean
- #respond_to_missing?(_method, _include_private = false) ⇒ Boolean
Constructor Details
#initialize(origin, log: nil) ⇒ Verbose
Returns a new instance of Verbose.
47 48 49 50 |
# File 'lib/verbose.rb', line 47 def initialize(origin, log: nil) @origin = origin @log = log end |
Dynamic Method Handling
This class handles dynamic methods through the method_missing method
#method_missing(*args) ⇒ Object
52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 |
# File 'lib/verbose.rb', line 52 def method_missing(*args) start = Time.now mtd = args.shift if @origin.respond_to?(mtd) params = @origin.method(mtd).parameters reqs = params.count { |p| p[0] == :req } if params.any? { |p| p[0] == :key } && args.size > reqs @origin.__send__(mtd, *args[0...-1], **args.last) do |*a| yield(*a) if block_given? end else @origin.__send__(mtd, *args) do |*a| yield(*a) if block_given? end end else super(mtd, *args) end ensure params = args.map do |a| if a.is_a?(String) max = 32 a = a.inspect if a.length > max "#{a[0..((max / 2) - 2)]}...#{a[((max / 2) + 1)..]}" else a end elsif [Integer, Float, TrueClass, FalseClass].include?(a.class) a else a.class end end msg = "#{@origin.class}.#{mtd}(#{params.join(', ')}) in #{start.ago}" if @log.respond_to?(:debug) @log.debug(msg) else puts(msg) end end |
Instance Method Details
#respond_to?(method, include_private = false) ⇒ Boolean
94 95 96 |
# File 'lib/verbose.rb', line 94 def respond_to?(method, include_private = false) @origin.respond_to?(method, include_private) end |
#respond_to_missing?(_method, _include_private = false) ⇒ Boolean
98 99 100 |
# File 'lib/verbose.rb', line 98 def respond_to_missing?(_method, _include_private = false) true end |