rails_log_parser 0.0.3 → 0.0.7

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: 14be05b30152b307fd2bfc5aa398e2381fc3849c130ad21313aa74c1d5e533f2
4
- data.tar.gz: b8ab4f231d38089960ce3dd8d04451fcf04ef53d94c381abc11eb6c35f874b97
3
+ metadata.gz: c223d9d32b464e7247934f4324ed4f4e38c64a9f6714737703b01578de4b3fe2
4
+ data.tar.gz: 540d452e4d79f74725ef369920f1fbb2a11cb8a5392316708779fe253d1ebb14
5
5
  SHA512:
6
- metadata.gz: f49c2178c57daa4c470421d617d78460c5ff6fdebf28dc8141a9830655e1a6b6c3984690dce8ab602e70caf432b72e825e713846a46ba74ede5cd3eba1baf447
7
- data.tar.gz: 21f1c5026a46d19fc6f6577dc3927788108df79b430a1ce6c3549c494e4845b184a4ce1a3ad730e16faf1f95e16e35d73374d91cd546fb9e983322b1d98d649c
6
+ metadata.gz: 65053ab7d6218dbddff70035b00bdb719efea68f617a1d109a178e24e3cd86afbe734245a6326a6ee305330a81dd7831c059075e741953bbac07c116c3c8dc05
7
+ data.tar.gz: 47e107f64fa6ffbe6780268960884ffe19206de3cbfe74a2255fb6194e0ede28be2a00aa456c0edbbc7d0a52cc0f6f92284b54a7c48a793a5b75509a2b441daa
data/Gemfile.lock CHANGED
@@ -1,8 +1,9 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- rails_log_parser (0.0.3)
4
+ rails_log_parser (0.0.7)
5
5
  enumerize (~> 2.4)
6
+ json (>= 2.0)
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.6.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.7
53
+
54
+ * Remove empty lines on summary without report
55
+
56
+ ### 0.0.6
57
+
58
+ * Adding heuristic to rate known exceptions
59
+
60
+ ### 0.0.5
61
+
62
+ * Removing `URI::InvalidURIError` as known exceptions
63
+
64
+ ### 0.0.4
65
+
66
+ * Handle stacktrace of fatals too
67
+
45
68
  ### 0.0.3
46
69
 
47
70
  * Adding `URI::InvalidURIError` as known exceptions
@@ -1,4 +1,5 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  require 'time'
3
4
  require 'securerandom'
4
5
 
@@ -13,12 +14,12 @@ 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
17
  }.freeze
18
18
 
19
19
  extend Enumerize
20
20
  enumerize :severity, in: SEVERITIES, predicates: true
21
21
  enumerize :type, in: %i[request without_request delayed_job active_job], predicates: true
22
+ attr_reader :datetime
22
23
 
23
24
  def initialize(type, id)
24
25
  self.type = type
@@ -36,9 +37,9 @@ class RailsLogParser::Action
36
37
  @headline = nil
37
38
  end
38
39
 
39
- def known_exception?
40
+ def known_exception?(key = nil)
40
41
  @messages.any? do |message|
41
- 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) }
42
43
  end
43
44
  end
44
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
+ 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.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.3'
5
+ spec.version = '0.0.7'
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.3
4
+ version: 0.0.7
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,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
@@ -81,7 +96,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
81
96
  - !ruby/object:Gem::Version
82
97
  version: '0'
83
98
  requirements: []
84
- rubygems_version: 3.0.8
99
+ rubygems_version: 3.1.6
85
100
  signing_key:
86
101
  specification_version: 4
87
102
  summary: Simple and fast analysing of default rails logs