rails_log_parser 0.0.2 → 0.0.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/Gemfile.lock +3 -1
- data/README.md +24 -1
- data/lib/rails_log_parser/action.rb +5 -3
- data/lib/rails_log_parser/heuristic_stat_file.rb +71 -0
- data/lib/rails_log_parser/line.rb +1 -1
- data/lib/rails_log_parser/parser.rb +23 -8
- data/lib/rails_log_parser/tasks.rb +4 -2
- data/lib/rails_log_parser.rb +2 -0
- data/rails_log_parser.gemspec +2 -1
- metadata +17 -2
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: 2f608985d7c53f5ac09850df6c94d63437c5c1ecec18390072e83b4751a57e16
         | 
| 4 | 
            +
              data.tar.gz: 68606ff5cea0948ad95cdd6f0ae583374b73654c013e95807dc0242070291879
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: 7351dc20aa2fc84268825e955ef6b8ffe924571013c27e6b0728555442aa205c10600a5266c153387109fb9fe7f6bc03d2657fecc50956622a2ac7eb9e6339bc
         | 
| 7 | 
            +
              data.tar.gz: eb0689bc7193fc130aae3cc40dd68ad5fc865add71696bd9a044442ef3094e61a26966f0044967cf8939365f0b03cf64d6b9b96ee5dc559236f0c80eb5770634
         | 
    
        data/Gemfile.lock
    CHANGED
    
    | @@ -1,8 +1,9 @@ | |
| 1 1 | 
             
            PATH
         | 
| 2 2 | 
             
              remote: .
         | 
| 3 3 | 
             
              specs:
         | 
| 4 | 
            -
                rails_log_parser (0.0. | 
| 4 | 
            +
                rails_log_parser (0.0.6)
         | 
| 5 5 | 
             
                  enumerize (~> 2.4)
         | 
| 6 | 
            +
                  json
         | 
| 6 7 |  | 
| 7 8 | 
             
            GEM
         | 
| 8 9 | 
             
              remote: https://rubygems.org/
         | 
| @@ -19,6 +20,7 @@ GEM | |
| 19 20 | 
             
                  activesupport (>= 3.2)
         | 
| 20 21 | 
             
                i18n (1.8.11)
         | 
| 21 22 | 
             
                  concurrent-ruby (~> 1.0)
         | 
| 23 | 
            +
                json (2.3.1)
         | 
| 22 24 | 
             
                minitest (5.14.4)
         | 
| 23 25 | 
             
                rake (12.3.3)
         | 
| 24 26 | 
             
                rspec (3.10.0)
         | 
    
        data/README.md
    CHANGED
    
    | @@ -30,7 +30,8 @@ Call the rake tasks in cronjobs: | |
| 30 30 |  | 
| 31 31 | 
             
            ```
         | 
| 32 32 | 
             
            LOG_PATH=/srv/rails/log/production.log
         | 
| 33 | 
            -
            0,20,40 * * * * rake rails_log_parser:parse[22]' | 
| 33 | 
            +
            0,20,40 * * * * rake rails_log_parser:parse[22]'      # summary of the last 22 minutes
         | 
| 34 | 
            +
            59     23 * * * rake rails_log_parser:parse[22,true]' # summary of the last 22 minutes and save and analyse heuristic
         | 
| 34 35 | 
             
            ```
         | 
| 35 36 |  | 
| 36 37 | 
             
            Or use it in your code:
         | 
| @@ -40,8 +41,30 @@ parser = RailsLogParser::Parser.from_file(log_path) | |
| 40 41 | 
             
            puts parser.actions.select(&:fatal?).map(&:headline)
         | 
| 41 42 | 
             
            ```
         | 
| 42 43 |  | 
| 44 | 
            +
            ```ruby
         | 
| 45 | 
            +
            parser = RailsLogParser::Parser.from_file(log_path)
         | 
| 46 | 
            +
            parser.enable_heuristic(File.dirname(log_path)) # path to save heuristic stats
         | 
| 47 | 
            +
            print parser.summary(last_minutes: 22)          # print summary for the last 22 minutes
         | 
| 48 | 
            +
            ```
         | 
| 49 | 
            +
             | 
| 43 50 | 
             
            ## Changelog
         | 
| 44 51 |  | 
| 52 | 
            +
            ### 0.0.6
         | 
| 53 | 
            +
             | 
| 54 | 
            +
            * Adding heuristic to rate known exceptions
         | 
| 55 | 
            +
             | 
| 56 | 
            +
            ### 0.0.5
         | 
| 57 | 
            +
             | 
| 58 | 
            +
            * Removing `URI::InvalidURIError` as known exceptions
         | 
| 59 | 
            +
             | 
| 60 | 
            +
            ### 0.0.4
         | 
| 61 | 
            +
             | 
| 62 | 
            +
            * Handle stacktrace of fatals too
         | 
| 63 | 
            +
             | 
| 64 | 
            +
            ### 0.0.3
         | 
| 65 | 
            +
             | 
| 66 | 
            +
            * Adding `URI::InvalidURIError` as known exceptions
         | 
| 67 | 
            +
             | 
| 45 68 | 
             
            ### 0.0.2
         | 
| 46 69 |  | 
| 47 70 | 
             
            * Adding `ActionController::InvalidAuthenticityToken` as known exceptions
         | 
| @@ -1,4 +1,5 @@ | |
| 1 1 | 
             
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 2 3 | 
             
            require 'time'
         | 
| 3 4 | 
             
            require 'securerandom'
         | 
| 4 5 |  | 
| @@ -12,12 +13,13 @@ class RailsLogParser::Action | |
| 12 13 | 
             
                'ActiveRecord::RecordNotFound' => :fatal,
         | 
| 13 14 | 
             
                'ActionController::RoutingError' => :fatal,
         | 
| 14 15 | 
             
                "Can't verify CSRF token authenticity." => :warn,
         | 
| 15 | 
            -
                 | 
| 16 | 
            +
                'ActionController::InvalidAuthenticityToken' => :fatal,
         | 
| 16 17 | 
             
              }.freeze
         | 
| 17 18 |  | 
| 18 19 | 
             
              extend Enumerize
         | 
| 19 20 | 
             
              enumerize :severity, in: SEVERITIES, predicates: true
         | 
| 20 21 | 
             
              enumerize :type, in: %i[request without_request delayed_job active_job], predicates: true
         | 
| 22 | 
            +
              attr_reader :datetime
         | 
| 21 23 |  | 
| 22 24 | 
             
              def initialize(type, id)
         | 
| 23 25 | 
             
                self.type = type
         | 
| @@ -35,9 +37,9 @@ class RailsLogParser::Action | |
| 35 37 | 
             
                @headline = nil
         | 
| 36 38 | 
             
              end
         | 
| 37 39 |  | 
| 38 | 
            -
              def known_exception?
         | 
| 40 | 
            +
              def known_exception?(key = nil)
         | 
| 39 41 | 
             
                @messages.any? do |message|
         | 
| 40 | 
            -
                  KNOWN_EXCEPTIONS.any? { |e, s| message.include?(e) && severity == s }
         | 
| 42 | 
            +
                  KNOWN_EXCEPTIONS.any? { |e, s| message.include?(e) && severity == s && (key.nil? || key == e) }
         | 
| 41 43 | 
             
                end
         | 
| 42 44 | 
             
              end
         | 
| 43 45 |  | 
| @@ -0,0 +1,71 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require 'json'
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            RailsLogParser::HeuristicStatFile = Struct.new(:path, :date) do
         | 
| 6 | 
            +
              attr_reader :stats
         | 
| 7 | 
            +
             | 
| 8 | 
            +
              class << self
         | 
| 9 | 
            +
                def build_heuristic(path, today)
         | 
| 10 | 
            +
                  sums = { actions: 0 }
         | 
| 11 | 
            +
                  RailsLogParser::Action::KNOWN_EXCEPTIONS.each_key do |exception|
         | 
| 12 | 
            +
                    sums[exception.to_sym] = 0
         | 
| 13 | 
            +
                  end
         | 
| 14 | 
            +
                  7.times do |i|
         | 
| 15 | 
            +
                    stats = RailsLogParser::HeuristicStatFile.new(path, today.date - (i + 1)).load_stats
         | 
| 16 | 
            +
                    sums[:actions] += stats[:actions].to_i
         | 
| 17 | 
            +
                    RailsLogParser::Action::KNOWN_EXCEPTIONS.each_key do |exception|
         | 
| 18 | 
            +
                      sums[exception.to_sym] += stats.dig(:known_exceptions, exception.to_sym).to_i
         | 
| 19 | 
            +
                    end
         | 
| 20 | 
            +
                  end
         | 
| 21 | 
            +
                  output = {}
         | 
| 22 | 
            +
                  RailsLogParser::Action::KNOWN_EXCEPTIONS.each_key do |exception|
         | 
| 23 | 
            +
                    next if sums[:actions] == 0
         | 
| 24 | 
            +
             | 
| 25 | 
            +
                    quota = sums[exception.to_sym].to_f / sums[:actions]
         | 
| 26 | 
            +
                    today_quota = today.rate(exception)
         | 
| 27 | 
            +
                    next if today_quota == 0
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                    rate = ((today_quota - quota) / quota) / Math.sqrt(sums[:actions].to_f)
         | 
| 30 | 
            +
                    output[exception] = rate if rate > heuristic_threshold
         | 
| 31 | 
            +
                  end
         | 
| 32 | 
            +
                  output
         | 
| 33 | 
            +
                end
         | 
| 34 | 
            +
             | 
| 35 | 
            +
                def heuristic_threshold
         | 
| 36 | 
            +
                  @heuristic_threshold ||= ENV['RAILS_LOG_PARSER_THRESHOLD_HEURISTIC'] || RailsLogParser::THRESHOLD_HEURISTIC
         | 
| 37 | 
            +
                end
         | 
| 38 | 
            +
              end
         | 
| 39 | 
            +
             | 
| 40 | 
            +
              def write_stats(actions)
         | 
| 41 | 
            +
                actions = actions.select { |action| action.datetime.to_date == date }.sort_by(&:datetime)
         | 
| 42 | 
            +
                @stats = {
         | 
| 43 | 
            +
                  actions: actions.count,
         | 
| 44 | 
            +
                  known_exceptions: {},
         | 
| 45 | 
            +
                  starts_at: actions.first&.datetime,
         | 
| 46 | 
            +
                  ends_at: actions.last&.datetime,
         | 
| 47 | 
            +
                }
         | 
| 48 | 
            +
             | 
| 49 | 
            +
                RailsLogParser::Action::KNOWN_EXCEPTIONS.each_key do |exception|
         | 
| 50 | 
            +
                  @stats[:known_exceptions][exception.to_sym] = actions.count { |action| action.known_exception?(exception) }
         | 
| 51 | 
            +
                end
         | 
| 52 | 
            +
                File.write(heuristic_file_path, @stats.to_json)
         | 
| 53 | 
            +
              end
         | 
| 54 | 
            +
             | 
| 55 | 
            +
              def load_stats
         | 
| 56 | 
            +
                @stats = JSON.parse(File.read(heuristic_file_path), symbolize_names: true) if File.file?(heuristic_file_path)
         | 
| 57 | 
            +
                @stats ||= {}
         | 
| 58 | 
            +
              rescue JSON::ParserError
         | 
| 59 | 
            +
                @stats = {}
         | 
| 60 | 
            +
              end
         | 
| 61 | 
            +
             | 
| 62 | 
            +
              def heuristic_file_path
         | 
| 63 | 
            +
                @heuristic_file_path ||= File.join(path, "heuristic_stats_#{date}.json")
         | 
| 64 | 
            +
              end
         | 
| 65 | 
            +
             | 
| 66 | 
            +
              def rate(exception)
         | 
| 67 | 
            +
                return 0 if stats[:actions] == 0
         | 
| 68 | 
            +
             | 
| 69 | 
            +
                stats.dig(:known_exceptions, exception.to_sym).to_f / stats[:actions]
         | 
| 70 | 
            +
              end
         | 
| 71 | 
            +
            end
         | 
| @@ -24,6 +24,12 @@ class RailsLogParser::Parser | |
| 24 24 | 
             
              def initialize
         | 
| 25 25 | 
             
                @actions = {}
         | 
| 26 26 | 
             
                @not_parseable_lines = []
         | 
| 27 | 
            +
                @heuristic = nil
         | 
| 28 | 
            +
              end
         | 
| 29 | 
            +
             | 
| 30 | 
            +
              def enable_heuristic(path)
         | 
| 31 | 
            +
                @heuristic = path
         | 
| 32 | 
            +
                @heuristic_today = RailsLogParser::HeuristicStatFile.new(@heuristic, Date.today).tap { |p| p.write_stats(actions) }
         | 
| 27 33 | 
             
              end
         | 
| 28 34 |  | 
| 29 35 | 
             
              def summary(last_minutes: nil)
         | 
| @@ -32,23 +38,32 @@ class RailsLogParser::Parser | |
| 32 38 | 
             
                  from = last_minutes.to_i.minutes.ago
         | 
| 33 39 | 
             
                  relevant = relevant.select { |a| a.after?(from) }
         | 
| 34 40 | 
             
                end
         | 
| 35 | 
            -
                 | 
| 41 | 
            +
                summary_output = []
         | 
| 36 42 | 
             
                if @not_parseable_lines.present?
         | 
| 37 | 
            -
                   | 
| 38 | 
            -
                   | 
| 39 | 
            -
                   | 
| 43 | 
            +
                  summary_output.push('Not parseable lines:')
         | 
| 44 | 
            +
                  summary_output += @not_parseable_lines.map { |line| "  #{line}" }
         | 
| 45 | 
            +
                  summary_output.push("\n\n")
         | 
| 40 46 | 
             
                end
         | 
| 41 47 |  | 
| 42 48 | 
             
                %i[warn error fatal].each do |severity|
         | 
| 43 49 | 
             
                  selected = relevant.select { |a| a.public_send("#{severity}?") }.reject(&:known_exception?)
         | 
| 44 50 | 
             
                  next if selected.blank?
         | 
| 45 51 |  | 
| 46 | 
            -
                   | 
| 47 | 
            -
                   | 
| 48 | 
            -
                   | 
| 52 | 
            +
                  summary_output.push("#{selected.count} lines with #{severity}:")
         | 
| 53 | 
            +
                  summary_output += selected.map(&:headline).map { |line| "  #{line}" }
         | 
| 54 | 
            +
                  summary_output.push("\n\n")
         | 
| 55 | 
            +
                end
         | 
| 56 | 
            +
             | 
| 57 | 
            +
                unless @heuristic.nil?
         | 
| 58 | 
            +
                  stats = RailsLogParser::HeuristicStatFile.build_heuristic(@heuristic, @heuristic_today)
         | 
| 59 | 
            +
                  if stats.present?
         | 
| 60 | 
            +
                    summary_output.push('Heuristic match!')
         | 
| 61 | 
            +
                    stats.each { |k, v| summary_output.push("- #{k}: #{v}") }
         | 
| 62 | 
            +
                  end
         | 
| 63 | 
            +
                  summary_output.push("\n\n")
         | 
| 49 64 | 
             
                end
         | 
| 50 65 |  | 
| 51 | 
            -
                 | 
| 66 | 
            +
                summary_output.join("\n")
         | 
| 52 67 | 
             
              end
         | 
| 53 68 |  | 
| 54 69 | 
             
              def actions
         | 
| @@ -2,7 +2,9 @@ | |
| 2 2 |  | 
| 3 3 | 
             
            namespace :rails_log_parser do
         | 
| 4 4 | 
             
              desc 'notify about found problems in production.log'
         | 
| 5 | 
            -
              task :parse, [:from_minutes] do |_t, args|
         | 
| 6 | 
            -
                 | 
| 5 | 
            +
              task :parse, [:from_minutes, :heuristic] do |_t, args|
         | 
| 6 | 
            +
                parser = RailsLogParser::Parser.from_file(RailsLogParser::Parser.log_path)
         | 
| 7 | 
            +
                parser.enable_heuristic(File.dirname(RailsLogParser::Parser.log_path)) if args[:heuristic] == 'true'
         | 
| 8 | 
            +
                print parser.summary(last_minutes: args[:from_minutes])
         | 
| 7 9 | 
             
              end
         | 
| 8 10 | 
             
            end
         | 
    
        data/lib/rails_log_parser.rb
    CHANGED
    
    | @@ -3,10 +3,12 @@ | |
| 3 3 | 
             
            require 'enumerize'
         | 
| 4 4 |  | 
| 5 5 | 
             
            module RailsLogParser
         | 
| 6 | 
            +
              THRESHOLD_HEURISTIC = 0.01
         | 
| 6 7 | 
             
            end
         | 
| 7 8 |  | 
| 8 9 | 
             
            require_relative 'rails_log_parser/parser'
         | 
| 9 10 | 
             
            require_relative 'rails_log_parser/action'
         | 
| 10 11 | 
             
            require_relative 'rails_log_parser/line'
         | 
| 12 | 
            +
            require_relative 'rails_log_parser/heuristic_stat_file'
         | 
| 11 13 |  | 
| 12 14 | 
             
            require 'rails_log_parser/railtie' if defined?(Rails::Railtie)
         | 
    
        data/rails_log_parser.gemspec
    CHANGED
    
    | @@ -2,7 +2,7 @@ | |
| 2 2 |  | 
| 3 3 | 
             
            Gem::Specification.new do |spec|
         | 
| 4 4 | 
             
              spec.name          = 'rails_log_parser'
         | 
| 5 | 
            -
              spec.version       = '0.0. | 
| 5 | 
            +
              spec.version       = '0.0.6'
         | 
| 6 6 | 
             
              spec.authors       = ['Georg Limbach']
         | 
| 7 7 | 
             
              spec.email         = ['georg.limbach@lichtbit.com']
         | 
| 8 8 |  | 
| @@ -23,5 +23,6 @@ Gem::Specification.new do |spec| | |
| 23 23 | 
             
              spec.require_paths = ['lib']
         | 
| 24 24 |  | 
| 25 25 | 
             
              spec.add_dependency 'enumerize', '~> 2.4'
         | 
| 26 | 
            +
              spec.add_dependency 'json'
         | 
| 26 27 | 
             
              spec.add_development_dependency 'rspec'
         | 
| 27 28 | 
             
            end
         | 
    
        metadata
    CHANGED
    
    | @@ -1,14 +1,14 @@ | |
| 1 1 | 
             
            --- !ruby/object:Gem::Specification
         | 
| 2 2 | 
             
            name: rails_log_parser
         | 
| 3 3 | 
             
            version: !ruby/object:Gem::Version
         | 
| 4 | 
            -
              version: 0.0. | 
| 4 | 
            +
              version: 0.0.6
         | 
| 5 5 | 
             
            platform: ruby
         | 
| 6 6 | 
             
            authors:
         | 
| 7 7 | 
             
            - Georg Limbach
         | 
| 8 8 | 
             
            autorequire: 
         | 
| 9 9 | 
             
            bindir: bin
         | 
| 10 10 | 
             
            cert_chain: []
         | 
| 11 | 
            -
            date: 2021- | 
| 11 | 
            +
            date: 2021-12-15 00:00:00.000000000 Z
         | 
| 12 12 | 
             
            dependencies:
         | 
| 13 13 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 14 14 | 
             
              name: enumerize
         | 
| @@ -24,6 +24,20 @@ dependencies: | |
| 24 24 | 
             
                - - "~>"
         | 
| 25 25 | 
             
                  - !ruby/object:Gem::Version
         | 
| 26 26 | 
             
                    version: '2.4'
         | 
| 27 | 
            +
            - !ruby/object:Gem::Dependency
         | 
| 28 | 
            +
              name: json
         | 
| 29 | 
            +
              requirement: !ruby/object:Gem::Requirement
         | 
| 30 | 
            +
                requirements:
         | 
| 31 | 
            +
                - - ">="
         | 
| 32 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 33 | 
            +
                    version: '0'
         | 
| 34 | 
            +
              type: :runtime
         | 
| 35 | 
            +
              prerelease: false
         | 
| 36 | 
            +
              version_requirements: !ruby/object:Gem::Requirement
         | 
| 37 | 
            +
                requirements:
         | 
| 38 | 
            +
                - - ">="
         | 
| 39 | 
            +
                  - !ruby/object:Gem::Version
         | 
| 40 | 
            +
                    version: '0'
         | 
| 27 41 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 28 42 | 
             
              name: rspec
         | 
| 29 43 | 
             
              requirement: !ruby/object:Gem::Requirement
         | 
| @@ -55,6 +69,7 @@ files: | |
| 55 69 | 
             
            - Rakefile
         | 
| 56 70 | 
             
            - lib/rails_log_parser.rb
         | 
| 57 71 | 
             
            - lib/rails_log_parser/action.rb
         | 
| 72 | 
            +
            - lib/rails_log_parser/heuristic_stat_file.rb
         | 
| 58 73 | 
             
            - lib/rails_log_parser/line.rb
         | 
| 59 74 | 
             
            - lib/rails_log_parser/parser.rb
         | 
| 60 75 | 
             
            - lib/rails_log_parser/railtie.rb
         |