rails_log_parser 0.0.15 → 0.0.16
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 +1 -1
- data/README.md +5 -2
- data/lib/rails_log_parser/action.rb +0 -5
- data/lib/rails_log_parser/parser.rb +3 -20
- data/lib/rails_log_parser/tasks.rb +1 -2
- data/lib/rails_log_parser.rb +0 -4
- data/rails_log_parser.gemspec +1 -1
- metadata +2 -3
- data/lib/rails_log_parser/heuristic_stat_file.rb +0 -85
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: fe25438fe1aa9e1fe506de50b0d337e26c387648f01aefa8fc558db5e1914bab
|
4
|
+
data.tar.gz: 9ec5eb524baba92586f71dff16fdd31243b5b4a7e68d767a11fb01ec2f40bd3f
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 6cb369807f23be8de3ad39d3a616e000d374cd5bb7591b48394c12b38f61b5685fe464bb5712e35d20523089bbd0e69f8e090df7581985df1f30124ab5ba685b
|
7
|
+
data.tar.gz: 5386f801caf9f7a259c5dbc6061793300c6d0347528511f4e0974fe1f64450fd90b92d900730d136f2ce92b8285f9e9c20cdefd9885884e355eae8c0b2ebfcce
|
data/Gemfile.lock
CHANGED
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,11 @@ end
|
|
62
60
|
|
63
61
|
## Changelog
|
64
62
|
|
63
|
+
### 0.0.16
|
64
|
+
|
65
|
+
* Remove heuristic
|
66
|
+
* Fix last action handling
|
67
|
+
|
65
68
|
### 0.0.15
|
66
69
|
|
67
70
|
* 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,
|
@@ -28,7 +24,6 @@ class RailsLogParser::Action
|
|
28
24
|
@id = id
|
29
25
|
@messages = []
|
30
26
|
@stacktrace = []
|
31
|
-
self.class.last = self
|
32
27
|
end
|
33
28
|
|
34
29
|
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
|
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
|
data/lib/rails_log_parser.rb
CHANGED
@@ -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)
|
data/rails_log_parser.gemspec
CHANGED
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.16
|
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-
|
11
|
+
date: 2023-06-16 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
|
@@ -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
|