rails_log_parser 0.0.4 → 0.0.8

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: 587c6ce2164707314caa439dbf465b895661df0e3d08de7ed3b10e0dcb9859a0
4
- data.tar.gz: c4679547c0518a154aadf33ab33cc4cbac86abf15baec01767f975d4e7c48fd2
3
+ metadata.gz: c63fac78d3fda42cab291bcd30cc7abb739ed75ca82de98a72efb0587fd35954
4
+ data.tar.gz: d04eb23a0d4a37af92ce87a74ea900ef6b35ecb502e612ea559a5158afa76baf
5
5
  SHA512:
6
- metadata.gz: 1f27b7ed6feb6677e0c70c05ac437c768823cfe7ba1daa967e26027e65b0810f3e391b55dc19bcf88a0590d5000c46d6178195f17946c05bff4fabed934be6a8
7
- data.tar.gz: 69d6e3b2398d018236f6d5a5ec660011a9dc5b0c321b7ad5a0d8cec5a28c4a75f94306129ec2e909ff7a02ad0b36664b92aba9628c0c6fdb600cf3594c15f17b
6
+ metadata.gz: a5de5ab1e2853550540f58538743c1394332fd32895a061675cf4d39c0efda4a21c5d42a13045a9e78477350054403423b1b38264c2920e576aceb1658ac594f
7
+ data.tar.gz: 614f54e398d1f113cdd7345b05b51e402795dc1e2cc363ba9e1209f828cc3d928a008a6fd30ea7684382b2eb8affe59727f5d808fe8fb49df9bacd53a30e5ef2
data/Gemfile.lock CHANGED
@@ -1,13 +1,14 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- rails_log_parser (0.0.3)
4
+ rails_log_parser (0.0.8)
5
5
  enumerize (~> 2.4)
6
+ json (>= 2.0)
6
7
 
7
8
  GEM
8
9
  remote: https://rubygems.org/
9
10
  specs:
10
- activesupport (6.1.4.1)
11
+ activesupport (6.1.4.4)
11
12
  concurrent-ruby (~> 1.0, >= 1.0.2)
12
13
  i18n (>= 1.6, < 2)
13
14
  minitest (>= 5.1)
@@ -15,11 +16,12 @@ GEM
15
16
  zeitwerk (~> 2.3)
16
17
  concurrent-ruby (1.1.9)
17
18
  diff-lcs (1.4.4)
18
- enumerize (2.4.0)
19
+ enumerize (2.5.0)
19
20
  activesupport (>= 3.2)
20
21
  i18n (1.8.11)
21
22
  concurrent-ruby (~> 1.0)
22
- minitest (5.14.4)
23
+ json (2.6.1)
24
+ minitest (5.15.0)
23
25
  rake (12.3.3)
24
26
  rspec (3.10.0)
25
27
  rspec-core (~> 3.10.0)
@@ -36,7 +38,7 @@ GEM
36
38
  rspec-support (3.10.3)
37
39
  tzinfo (2.0.4)
38
40
  concurrent-ruby (~> 1.0)
39
- zeitwerk (2.5.1)
41
+ zeitwerk (2.5.3)
40
42
 
41
43
  PLATFORMS
42
44
  ruby
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,31 @@ 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.8
53
+
54
+ * Adding `ActionController::UnfilteredParameters` as known exceptions
55
+ * Adjust heuristic rate for better matching
56
+
57
+ ### 0.0.7
58
+
59
+ * Remove empty lines on summary without report
60
+
61
+ ### 0.0.6
62
+
63
+ * Adding heuristic to rate known exceptions
64
+
65
+ ### 0.0.5
66
+
67
+ * Removing `URI::InvalidURIError` as known exceptions
68
+
45
69
  ### 0.0.4
46
70
 
47
71
  * Handle stacktrace of fatals too
@@ -1,4 +1,5 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  require 'time'
3
4
  require 'securerandom'
4
5
 
@@ -13,12 +14,13 @@ class RailsLogParser::Action
13
14
  'ActionController::RoutingError' => :fatal,
14
15
  "Can't verify CSRF token authenticity." => :warn,
15
16
  'ActionController::InvalidAuthenticityToken' => :fatal,
16
- 'URI::InvalidURIError' => :fatal,
17
+ 'ActionController::UnfilteredParameters' => :fatal,
17
18
  }.freeze
18
19
 
19
20
  extend Enumerize
20
21
  enumerize :severity, in: SEVERITIES, predicates: true
21
22
  enumerize :type, in: %i[request without_request delayed_job active_job], predicates: true
23
+ attr_reader :datetime
22
24
 
23
25
  def initialize(type, id)
24
26
  self.type = type
@@ -36,9 +38,9 @@ class RailsLogParser::Action
36
38
  @headline = nil
37
39
  end
38
40
 
39
- def known_exception?
41
+ def known_exception?(key = nil)
40
42
  @messages.any? do |message|
41
- KNOWN_EXCEPTIONS.any? { |e, s| message.include?(e) && severity == s }
43
+ KNOWN_EXCEPTIONS.any? { |e, s| message.include?(e) && severity == s && (key.nil? || key == e) }
42
44
  end
43
45
  end
44
46
 
@@ -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
+ 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] == 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
- @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! (threshold: #{RailsLogParser::HeuristicStatFile.heuristic_threshold})")
61
+ stats.each { |k, v| summary_output.push("- #{k}: #{v.round(4)}") }
62
+ summary_output.push("\n\n")
63
+ end
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.02
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.4'
5
+ spec.version = '0.0.8'
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_development_dependency 'rspec'
26
+ spec.add_dependency 'json', '>= 2.0'
27
+ spec.add_development_dependency 'rspec', '>= 3.0'
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
4
+ version: 0.0.8
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-30 00:00:00.000000000 Z
11
+ date: 2022-01-10 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: enumerize
@@ -24,20 +24,34 @@ 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: '2.0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '2.0'
27
41
  - !ruby/object:Gem::Dependency
28
42
  name: rspec
29
43
  requirement: !ruby/object:Gem::Requirement
30
44
  requirements:
31
45
  - - ">="
32
46
  - !ruby/object:Gem::Version
33
- version: '0'
47
+ version: '3.0'
34
48
  type: :development
35
49
  prerelease: false
36
50
  version_requirements: !ruby/object:Gem::Requirement
37
51
  requirements:
38
52
  - - ">="
39
53
  - !ruby/object:Gem::Version
40
- version: '0'
54
+ version: '3.0'
41
55
  description: If you want to get an email with errors and fatal log lines you can use
42
56
  this gem.
43
57
  email:
@@ -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