# encoding: utf-8 # This file is distributed under New Relic's license terms. # See https://github.com/newrelic/rpm/blob/master/LICENSE for complete details. require 'new_relic/agent/rules_engine/replacement_rule' require 'new_relic/agent/rules_engine/segment_terms_rule' require 'new_relic/language_support' module NewRelic module Agent class RulesEngine SEGMENT_SEPARATOR = '/'.freeze LEADING_SLASH_REGEX = %r{^/}.freeze include Enumerable extend Forwardable def_delegators :@rules, :size, :inspect, :each, :clear def self.create_metric_rules(connect_response) specs = connect_response['metric_name_rules'] || [] rules = specs.map { |spec| ReplacementRule.new(spec) } self.new(rules) end def self.create_transaction_rules(connect_response) txn_name_specs = connect_response['transaction_name_rules'] || [] segment_rule_specs = connect_response['transaction_segment_terms'] || [] txn_name_rules = txn_name_specs.map { |s| ReplacementRule.new(s) } segment_rules = [] segment_rule_specs.each do |spec| if spec[SegmentTermsRule::PREFIX_KEY] && SegmentTermsRule.valid?(spec) # Build segment_rules in reverse order from which they're provided, # so that when we eliminate duplicates with #uniq!, we retain the last # instances of repeated rules. segment_rules.unshift SegmentTermsRule.new(spec) end end reject_rules_with_duplicate_prefixes!(segment_rules) segment_rules.reverse! # Reset the rules to their original order. self.new(txn_name_rules, segment_rules) end # When multiple rules share the same prefix, # only apply the rule with the last instance of the prefix. # Note that the incoming rules are in reverse order to facilitate this. if NewRelic::LanguageSupport.uniq_accepts_block? def self.reject_rules_with_duplicate_prefixes!(rules) rules.uniq! { |rule| rule.prefix } end else def self.reject_rules_with_duplicate_prefixes!(rules) unique_rules = {} rules.reject! do |rule| if unique_rules[rule.prefix] true else unique_rules[rule.prefix] = rule false end end end end def initialize(rules=[], segment_term_rules=[]) @rules = rules.sort @segment_term_rules = segment_term_rules end def apply_rules(rules, string) rules.each do |rule| if rule.matches?(string) string = rule.apply(string) break if rule.terminal? end end string end def rename(original_string) renamed = apply_rules(@rules, original_string) return nil unless renamed renamed = apply_rules(@segment_term_rules, renamed) renamed end end end end