rails_log_parser 0.0.15 → 0.0.17

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 7075d8f72e2f41875ca8e7d7b96cc9895a09cf3561b1f7a8ce140decd8172f0c
4
- data.tar.gz: 0c497f4bcb5a4802d0a2933f5af11b95498a60fadb57619fe13719441e8f49d8
3
+ metadata.gz: 96ae5461bc6323f83c7ed166c728e1c0d161b9a0a3fe921d723b7f13bac11db9
4
+ data.tar.gz: 6b9d0276cdef26834994ea8001d1942fd025d14ed6c59f2bab25cbbcfcd13d22
5
5
  SHA512:
6
- metadata.gz: 1e9ec59884a21ac46292369dea9d3ff94426bff3d67cb4962cf220f5412448b8d366515837b12ea319cdfa2c7f34320819c9bdaf23ee4584bf3ebfb169478807
7
- data.tar.gz: 1ede0e6934f9e56d2fe0e88b93835d08da872395fb5678ba4eabdd729ba4e167bbea90dda06aa6565e485f1e2ec48082607d30d37df11c7245a78c979d8f21f5
6
+ metadata.gz: 92d7e4cc9808c00d9d3a2ad13c0885c1b540d589cccf4188f77d9ada5b042d092f125898eb32d6dd13c57ff460c2f391b1cd7bbb9661d99b2e17ad58db7261f0
7
+ data.tar.gz: '08257c0b4050e94b2ccff09d55dc391444da9c92c1d090e9e6459b23a683f686c7bf2d942f7a7b62987f67b2767e3501ab1c36abddbadbb54cb29c0a35670417'
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- rails_log_parser (0.0.15)
4
+ rails_log_parser (0.0.16)
5
5
  enumerize (~> 2.4)
6
6
  json (>= 2.0)
7
7
  rails (>= 2.1)
@@ -71,11 +71,13 @@ GEM
71
71
  marcel (1.0.1)
72
72
  method_source (1.0.0)
73
73
  mini_mime (1.1.0)
74
- mini_portile2 (2.4.0)
74
+ mini_portile2 (2.8.5)
75
75
  minitest (5.18.0)
76
76
  nio4r (2.5.7)
77
- nokogiri (1.10.10)
78
- mini_portile2 (~> 2.4.0)
77
+ nokogiri (1.16.0)
78
+ mini_portile2 (~> 2.8.2)
79
+ racc (~> 1.4)
80
+ racc (1.7.3)
79
81
  rack (2.2.3)
80
82
  rack-test (1.1.0)
81
83
  rack (>= 1.0, < 3)
data/README.md CHANGED
@@ -31,7 +31,6 @@ Call the rake tasks in cronjobs:
31
31
  ```
32
32
  LOG_PATH=/srv/rails/log/production.log
33
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
35
34
  ```
36
35
 
37
36
  Or use it in your code:
@@ -43,7 +42,6 @@ puts parser.actions.select(&:fatal?).map(&:headline)
43
42
 
44
43
  ```ruby
45
44
  parser = RailsLogParser::Parser.from_file(log_path)
46
- parser.enable_heuristic(File.dirname(log_path)) # path to save heuristic stats
47
45
  print parser.summary(last_minutes: 22) # print summary for the last 22 minutes
48
46
  ```
49
47
 
@@ -62,6 +60,16 @@ end
62
60
 
63
61
  ## Changelog
64
62
 
63
+ ### 0.0.17
64
+
65
+ * Adding `ActionDispatch::Http::MimeNegotiation::InvalidType` as known exceptions
66
+ * Update dependencies
67
+
68
+ ### 0.0.16
69
+
70
+ * Remove heuristic
71
+ * Fix last action handling
72
+
65
73
  ### 0.0.15
66
74
 
67
75
  * Filter lines by config file
@@ -4,10 +4,6 @@ require 'time'
4
4
  require 'securerandom'
5
5
 
6
6
  class RailsLogParser::Action
7
- class << self
8
- attr_accessor :last
9
- end
10
-
11
7
  SEVERITIES = %i[debug info warn error fatal].freeze
12
8
  KNOWN_EXCEPTIONS = {
13
9
  "Can't verify CSRF token authenticity." => :warn,
@@ -16,6 +12,7 @@ class RailsLogParser::Action
16
12
  'ActionController::UnfilteredParameters' => :fatal,
17
13
  'ActionController::UnknownFormat' => :fatal,
18
14
  'ActiveRecord::RecordNotFound' => :fatal,
15
+ 'ActionDispatch::Http::MimeNegotiation::InvalidType' => :fatal,
19
16
  }.freeze
20
17
 
21
18
  extend Enumerize
@@ -28,7 +25,6 @@ class RailsLogParser::Action
28
25
  @id = id
29
26
  @messages = []
30
27
  @stacktrace = []
31
- self.class.last = self
32
28
  end
33
29
 
34
30
  def severity=(value)
@@ -5,7 +5,7 @@ class RailsLogParser::Parser
5
5
  attr_writer :log_path
6
6
 
7
7
  def log_path
8
- @log_path || ENV['LOG_PATH']
8
+ @log_path || ENV['LOG_PATH'] || raise('no log_path given')
9
9
  end
10
10
 
11
11
  def from_file(path)
@@ -19,7 +19,7 @@ class RailsLogParser::Parser
19
19
  end
20
20
  end
21
21
 
22
- attr_reader :not_parseable_lines
22
+ attr_reader :not_parseable_lines, :last_action
23
23
 
24
24
  def initialize
25
25
  config_file = File.join(Dir.pwd,'config/rails_log_parser.rb')
@@ -27,13 +27,8 @@ class RailsLogParser::Parser
27
27
 
28
28
  @actions = {}
29
29
  @not_parseable_lines = RailsLogParser::NotParseableLines.new
30
- @heuristic = nil
31
30
  end
32
31
 
33
- def enable_heuristic(path)
34
- @heuristic = path
35
- @heuristic_today = RailsLogParser::HeuristicStatFile.new(@heuristic, Date.today).tap { |p| p.write_stats(actions) }
36
- end
37
32
 
38
33
  def summary(last_minutes: nil)
39
34
  relevant = actions
@@ -58,15 +53,6 @@ class RailsLogParser::Parser
58
53
  summary_output.push("\n\n")
59
54
  end
60
55
 
61
- unless @heuristic.nil?
62
- stats = RailsLogParser::HeuristicStatFile.build_heuristic(@heuristic, @heuristic_today)
63
- if stats.present?
64
- summary_output.push("Heuristic match! (threshold: #{RailsLogParser::HeuristicStatFile.heuristic_threshold})")
65
- stats.each { |k, v| summary_output.push("- #{k}: #{v.round(4)}") }
66
- summary_output.push("\n\n")
67
- end
68
- end
69
-
70
56
  summary_output.join("\n")
71
57
  end
72
58
 
@@ -83,6 +69,7 @@ class RailsLogParser::Parser
83
69
  @actions[params['id']].severity = params['severity_label']
84
70
  @actions[params['id']].datetime = params['datetime']
85
71
  @actions[params['id']].add_message(params['message']) unless params['message'].nil?
72
+ @last_action = @actions[params['id']]
86
73
  end
87
74
 
88
75
  def request(params)
@@ -113,8 +100,4 @@ class RailsLogParser::Parser
113
100
  @actions[params['id']] ||= RailsLogParser::Action.new(type, params['id'])
114
101
  @actions[params['id']].add_message(params['message'])
115
102
  end
116
-
117
- def last_action
118
- RailsLogParser::Action.last
119
- end
120
103
  end
@@ -2,9 +2,8 @@
2
2
 
3
3
  namespace :rails_log_parser do
4
4
  desc 'notify about found problems in production.log'
5
- task :parse, [:from_minutes, :heuristic] do |_t, args|
5
+ task :parse, [:from_minutes] do |_t, args|
6
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
7
  print parser.summary(last_minutes: args[:from_minutes])
9
8
  end
10
9
  end
@@ -4,9 +4,6 @@ require 'enumerize'
4
4
  require 'active_support/core_ext/module/attribute_accessors'
5
5
 
6
6
  module RailsLogParser
7
- THRESHOLD_HEURISTIC = 0.02
8
- MIN_ACTIONS_HEURISTIC = 100000 # sum of last 10 days
9
-
10
7
  mattr_accessor :ignore_lines, default: []
11
8
 
12
9
  def self.configure
@@ -17,7 +14,6 @@ end
17
14
  require_relative 'rails_log_parser/parser'
18
15
  require_relative 'rails_log_parser/action'
19
16
  require_relative 'rails_log_parser/line'
20
- require_relative 'rails_log_parser/heuristic_stat_file'
21
17
  require_relative 'rails_log_parser/not_parseable_lines'
22
18
 
23
19
  require 'rails_log_parser/railtie' if defined?(Rails::Railtie)
@@ -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.15'
5
+ spec.version = '0.0.17'
6
6
  spec.authors = ['Georg Limbach']
7
7
  spec.email = ['georg.limbach@lichtbit.com']
8
8
 
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.15
4
+ version: 0.0.17
5
5
  platform: ruby
6
6
  authors:
7
7
  - Georg Limbach
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2023-05-24 00:00:00.000000000 Z
11
+ date: 2024-01-11 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: enumerize
@@ -83,7 +83,6 @@ files:
83
83
  - Rakefile
84
84
  - lib/rails_log_parser.rb
85
85
  - lib/rails_log_parser/action.rb
86
- - lib/rails_log_parser/heuristic_stat_file.rb
87
86
  - lib/rails_log_parser/line.rb
88
87
  - lib/rails_log_parser/not_parseable_lines.rb
89
88
  - lib/rails_log_parser/parser.rb
@@ -111,7 +110,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
111
110
  - !ruby/object:Gem::Version
112
111
  version: '0'
113
112
  requirements: []
114
- rubygems_version: 3.0.8
113
+ rubygems_version: 3.3.26
115
114
  signing_key:
116
115
  specification_version: 4
117
116
  summary: Simple and fast analysing of default rails logs
@@ -1,85 +0,0 @@
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
- 10.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] < heuristic_min_actions
24
-
25
- quota = sums[exception.to_sym].to_f / sums[:actions]
26
- next if quota == 0
27
- today_quota = today.rate(exception)
28
- next if today_quota == 0
29
-
30
- rate = ((today_quota - quota) / quota) / Math.sqrt(sums[:actions].to_f)
31
- output[exception] = rate if rate > heuristic_threshold
32
- end
33
- output
34
- end
35
-
36
- def heuristic_threshold
37
- @heuristic_threshold ||= ENV['RAILS_LOG_PARSER_THRESHOLD_HEURISTIC'] || RailsLogParser::THRESHOLD_HEURISTIC
38
- end
39
-
40
- def heuristic_min_actions
41
- @heuristic_min_actions ||= ENV['RAILS_LOG_PARSER_MIN_ACTIONS_HEURISTIC'] || RailsLogParser::MIN_ACTIONS_HEURISTIC
42
- end
43
- end
44
-
45
- def write_stats(actions)
46
- actions = actions.select { |action| action.datetime.to_date == date }.sort_by(&:datetime)
47
- @stats = {
48
- actions: actions.count,
49
- known_exceptions: {},
50
- starts_at: actions.first&.datetime,
51
- ends_at: actions.last&.datetime,
52
- }
53
-
54
- RailsLogParser::Action::KNOWN_EXCEPTIONS.each_key do |exception|
55
- @stats[:known_exceptions][exception.to_sym] = actions.count { |action| action.known_exception?(exception) }
56
- end
57
-
58
- delete_old_stats
59
- File.write(heuristic_file_path, @stats.to_json)
60
- end
61
-
62
- def delete_old_stats
63
- last_20_days = (0..19).map { |i| (Date.today - i) }.map { |date| File.join(path, "heuristic_stats_#{date}.json") }
64
- Dir[File.join(path, 'heuristic_stats_*.json')].reject { |file| last_20_days.include?(file) }.each do |file|
65
- File.unlink(file)
66
- end
67
- end
68
-
69
- def load_stats
70
- @stats = JSON.parse(File.read(heuristic_file_path), symbolize_names: true) if File.file?(heuristic_file_path)
71
- @stats ||= {}
72
- rescue JSON::ParserError
73
- @stats = {}
74
- end
75
-
76
- def heuristic_file_path
77
- @heuristic_file_path ||= File.join(path, "heuristic_stats_#{date}.json")
78
- end
79
-
80
- def rate(exception)
81
- return 0 if stats[:actions] == 0
82
-
83
- stats.dig(:known_exceptions, exception.to_sym).to_f / stats[:actions]
84
- end
85
- end