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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: ea15a1a3ec5a45569f3242ec132c7ad247d545c827cc8583439990095612099e
4
- data.tar.gz: 4ad9958d1b198976eeb78cde4b8527d9402d90cdfe50ee3737de803de7a43cc6
3
+ metadata.gz: 2f608985d7c53f5ac09850df6c94d63437c5c1ecec18390072e83b4751a57e16
4
+ data.tar.gz: 68606ff5cea0948ad95cdd6f0ae583374b73654c013e95807dc0242070291879
5
5
  SHA512:
6
- metadata.gz: ce42b4516aceb6284d2c51f551cadbb46b1ae53a2cc313a3cec003f423d095db6de3b269cb013b14797e9a5b5949f1051d25ae06a0c12f4a8fa4d9e63c75e3ef
7
- data.tar.gz: edd3514b50cf68188ae011230d73c692695d6abaff9f3146436f58f1b0f39621221bc0ba760be337ad8e2d375438a71de7f367ee53f584564282dc2acbf0d0d5
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.1)
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]' # summary of the last 22 minutes
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
- "ActionController::InvalidAuthenticityToken" => :fatal,
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
@@ -96,7 +96,7 @@ class RailsLogParser::Line
96
96
  return
97
97
  end
98
98
 
99
- if parser.last_action&.error?
99
+ if parser.last_action&.error? || parser.last_action&.fatal?
100
100
  parser.last_action.add_stacktrace(line)
101
101
  return
102
102
  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
- @summary_output = []
41
+ summary_output = []
36
42
  if @not_parseable_lines.present?
37
- @summary_output.push('Not parseable lines:')
38
- @summary_output += @not_parseable_lines.map { |line| " #{line}" }
39
- @summary_output.push("\n\n")
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
- @summary_output.push("#{selected.count} lines with #{severity}:")
47
- @summary_output += selected.map(&:headline).map { |line| " #{line}" }
48
- @summary_output.push("\n\n")
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
- @summary_output.join("\n")
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
- print RailsLogParser::Parser.from_file(RailsLogParser::Parser.log_path).summary(last_minutes: args[:from_minutes])
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
@@ -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)
@@ -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.2'
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.2
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-29 00:00:00.000000000 Z
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