rails_log_parser 0.0.5 → 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 +12 -1
- data/lib/rails_log_parser/action.rb +4 -2
- data/lib/rails_log_parser/heuristic_stat_file.rb +71 -0
- 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,18 @@ 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
|
+
|
45
56
|
### 0.0.5
|
46
57
|
|
47
58
|
* Removing `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
|
|
@@ -18,6 +19,7 @@ class RailsLogParser::Action
|
|
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
|