File: log_entry_matcher.rb

package info (click to toggle)
ruby-lumberjack 2.0.4-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 956 kB
  • sloc: ruby: 7,957; makefile: 2
file content (126 lines) | stat: -rw-r--r-- 5,648 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
# frozen_string_literal: true

module Lumberjack
  # A flexible matching utility for testing and filtering log entries based on
  # multiple criteria. This class provides pattern-based matching against log
  # entry components including message content, severity levels, program names,
  # and custom attributes with support for nested attribute structures.
  #
  # The matcher uses Ruby's case equality operator (===) for flexible matching,
  # supporting exact values, regular expressions, ranges, classes, and other
  # pattern matching constructs. It's primarily designed for use with the Test
  # device in testing scenarios but can be used anywhere log entry filtering
  # is needed.
  #
  # @see Lumberjack::Device::Test
  class LogEntryMatcher
    require_relative "log_entry_matcher/score"

    # Create a new log entry matcher with optional filtering criteria. All
    # parameters are optional and nil values indicate no filtering for that
    # component. The matcher uses case equality (===) for flexible pattern
    # matching against each specified criterion.
    #
    # @param message [Object, nil] Pattern to match against log entry messages.
    #   Supports strings, regular expressions, or any object responding to ===
    # @param severity [Integer, String, Symbol, nil] Severity level to match.
    #   Accepts numeric levels or symbolic names (:debug, :info, etc.)
    # @param progname [Object, nil] Pattern to match against program names.
    #   Supports strings, regular expressions, or any object responding to ===
    # @param attributes [Hash, nil] Hash of attribute patterns to match against
    #   log entry attributes. Supports nested attribute matching and dot notation
    def initialize(message: nil, severity: nil, progname: nil, attributes: nil)
      message = message.strip if message.is_a?(String)
      @message_filter = message
      @severity_filter = Severity.coerce(severity) if severity
      @progname_filter = progname
      @attributes_filter = Utils.expand_attributes(attributes) if attributes
    end

    # Test whether a log entry matches all specified criteria. The entry must
    # satisfy all non-nil filter conditions to be considered a match. Uses
    # case equality (===) for flexible pattern matching.
    #
    # @param entry [Lumberjack::LogEntry] The log entry to test against the matcher
    # @return [Boolean] True if the entry matches all specified criteria, false otherwise
    def match?(entry)
      return false unless match_filter?(entry.message, @message_filter)
      return false unless match_filter?(entry.severity, @severity_filter)
      return false unless match_filter?(entry.progname, @progname_filter)

      if @attributes_filter
        attributes = Utils.expand_attributes(entry.attributes)
        return false unless match_attributes?(attributes, @attributes_filter)
      end

      true
    end

    # Find the closest matching log entry from a list of candidates. This method
    # scores each entry based on how well it matches the specified criteria and
    # returns the entry with the highest score, provided it meets a minimum
    # threshold. If no entries meet the threshold, nil is returned.
    #
    # @param entries [Array<Lumberjack::LogEntry>] The list of log entries to evaluate
    # @return [Lumberjack::LogEntry, nil] The closest matching log entry or nil if none match
    def closest(entries)
      scored_entries = entries.map { |entry| [entry, entry_score(entry)] }
      best_score = scored_entries.max_by { |_, score| score }
      (best_score&.last.to_f >= Score::MIN_SCORE_THRESHOLD) ? best_score.first : nil
    end

    private

    def entry_score(entry)
      Score.calculate_match_score(
        entry,
        message: @message_filter,
        severity: @severity_filter,
        attributes: @attributes_filter,
        progname: @progname_filter
      )
    end

    # Apply a filter pattern against a value using case equality. Returns true
    # if no filter is specified (nil) or if the filter matches the value.
    #
    # @param value [Object] The value to test against the filter
    # @param filter [Object, nil] The filter pattern, nil means no filtering
    # @return [Boolean] True if the filter matches or is nil, false otherwise
    def match_filter?(value, filter)
      return true if filter.nil?

      filter === value
    end

    # Recursively match nested attribute structures against filter patterns.
    # Handles both simple attribute matching and complex nested hash structures
    # with support for partial matching and empty value detection.
    #
    # @param attributes [Hash] The expanded attributes hash from the log entry
    # @param filter [Hash] The filter patterns to match against attributes
    # @return [Boolean] True if all filter patterns match their corresponding attributes
    def match_attributes?(attributes, filter)
      return true unless filter
      return false unless attributes

      filter.all? do |name, value_filter|
        name = name.to_s
        attribute_values = attributes[name]
        if attribute_values.is_a?(Hash)
          if value_filter.is_a?(Hash)
            match_attributes?(attribute_values, value_filter)
          else
            match_filter?(attribute_values, value_filter)
          end
        elsif value_filter.nil? || (value_filter.is_a?(Enumerable) && value_filter.empty?)
          attribute_values.nil? || (attribute_values.is_a?(Array) && attribute_values.empty?)
        elsif attributes.include?(name)
          match_filter?(attribute_values, value_filter)
        else
          false
        end
      end
    end
  end
end