fluent-plugin-elastic-log 0.4.2 → 0.5.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.md +3 -1
- data/fluent-plugin-elastic-log.gemspec +1 -1
- data/lib/fluent/plugin/elastic_log/audit_log_to_metric_processor.rb +17 -0
- data/lib/fluent/plugin/elastic_log/failed_login_metric.rb +105 -0
- data/lib/fluent/plugin/elastic_log/granted_privileges_metric.rb +14 -7
- data/lib/fluent/plugin/out_elastic_audit_log_metric.rb +9 -3
- metadata +3 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: a09b3853194e612ab30572268a5fd930995f02fbc7a7f1430e3809222df3f093
|
4
|
+
data.tar.gz: be9066ab3630764364e51772a5ea8a71dee68d5e0cd6da4d332198f00ec804bc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 720cf4ed71f63792a55dec5e0fccdb601a14ee8119eb4c3264f8827c9625b3f9bb7142eaffc6db8b5d8c0214596ca2ceec47014f7e491ae882781ec9d2e98487
|
7
|
+
data.tar.gz: c5ca8575b0050568388685bf74222080799952394f884f0e568a996c05a5c190db697ad73529e8909dae81aabd8c11b519bef144fe4f5e41403bfa64393562a8
|
data/README.md
CHANGED
@@ -35,11 +35,13 @@ parameters for input record:
|
|
35
35
|
* r_indices_key: Resolved indices key in input record
|
36
36
|
* timestamp_key: Timestamp key in input record
|
37
37
|
* privilege_key: Request privilege key in input record
|
38
|
+
* rest_request_path_key: Rest request path key in input record
|
39
|
+
* request_body_key: Request body key in input record
|
38
40
|
|
39
41
|
parameters for output metric:
|
40
42
|
* timestamp_format: Timestamp format (iso, epochmillis, epochmillis_str)
|
41
43
|
* prefix: Attribute prefix for output metric
|
42
|
-
*
|
44
|
+
* aggregate_index: Aggregate index (remove ilm suffix, wildcard suffix)
|
43
45
|
|
44
46
|
More details from the
|
45
47
|
[elastic_audit_log_metric output plugin code](lib/fluent/plugin/out_elastic_audit_log_metric.rb#L49)
|
@@ -1,6 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require_relative 'granted_privileges_metric'
|
4
|
+
require_relative 'failed_login_metric'
|
4
5
|
|
5
6
|
module Fluent
|
6
7
|
module Plugin
|
@@ -48,6 +49,22 @@ module Fluent
|
|
48
49
|
).generate_metrics
|
49
50
|
end
|
50
51
|
# rubocop:enable Metrics/AbcSize
|
52
|
+
|
53
|
+
# rubocop:disable Metrics/AbcSize
|
54
|
+
def generate_failed_login_metrics_for(record)
|
55
|
+
FailedLoginMetric.new(
|
56
|
+
record: {
|
57
|
+
timestamp: record[conf.timestamp_key],
|
58
|
+
user: record[conf.user_key],
|
59
|
+
cluster: record[conf.cluster_key],
|
60
|
+
layer: record[conf.layer_key],
|
61
|
+
request_path: record[conf.rest_request_path_key],
|
62
|
+
request_body: record[conf.request_body_key]
|
63
|
+
},
|
64
|
+
conf: conf
|
65
|
+
).generate_metrics
|
66
|
+
end
|
67
|
+
# rubocop:enable Metrics/AbcSize
|
51
68
|
end
|
52
69
|
end
|
53
70
|
end
|
@@ -0,0 +1,105 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'set'
|
4
|
+
require 'time'
|
5
|
+
require 'json'
|
6
|
+
|
7
|
+
module Fluent
|
8
|
+
module Plugin
|
9
|
+
module ElasticLog
|
10
|
+
# record to metric converter
|
11
|
+
# for FAILED_LOGIN
|
12
|
+
class FailedLoginMetric
|
13
|
+
attr_reader :record, :conf
|
14
|
+
|
15
|
+
ELASTIC_URL_PATTERN = %r{(?:/(?<target>[^/]*))?/(?<action>_\w+)}.freeze
|
16
|
+
QUERY_TYPE_MAP = {
|
17
|
+
'_msearch' => 'msearch',
|
18
|
+
'_bulk' => 'bulk',
|
19
|
+
'_doc' => 'write',
|
20
|
+
'_create' => 'write',
|
21
|
+
'_search' => 'search'
|
22
|
+
}.freeze
|
23
|
+
|
24
|
+
INDEX_PATTERN = /-?\*$/.freeze
|
25
|
+
ILM_PATTERN = /-\d{6}$/.freeze
|
26
|
+
|
27
|
+
def initialize(record:, conf:)
|
28
|
+
@record = record
|
29
|
+
@conf = conf
|
30
|
+
end
|
31
|
+
|
32
|
+
def timestamp
|
33
|
+
timestamp = Time.parse(record[:timestamp])
|
34
|
+
|
35
|
+
return (timestamp.utc.to_f * 1000).to_i if conf.timestamp_format == :epochmillis
|
36
|
+
return timestamp.utc.strftime('%s%3N') if conf.timestamp_format == :epochmillis_str
|
37
|
+
|
38
|
+
timestamp.utc.iso8601(3)
|
39
|
+
rescue StandardError
|
40
|
+
nil
|
41
|
+
end
|
42
|
+
|
43
|
+
def query_details
|
44
|
+
if (match = ELASTIC_URL_PATTERN.match(record[:request_path]))
|
45
|
+
return [QUERY_TYPE_MAP.fetch(match[:action], 'other'),
|
46
|
+
match[:target]]
|
47
|
+
end
|
48
|
+
['other', nil]
|
49
|
+
end
|
50
|
+
|
51
|
+
def base
|
52
|
+
{
|
53
|
+
'timestamp' => timestamp,
|
54
|
+
'metric_name' => 'failed_login_count',
|
55
|
+
'metric_value' => 1,
|
56
|
+
"#{conf.prefix}user" => record[:user],
|
57
|
+
"#{conf.prefix}cluster" => record[:cluster]
|
58
|
+
}
|
59
|
+
end
|
60
|
+
|
61
|
+
def bulk_indices
|
62
|
+
req_body = record.fetch(:request_body, {})
|
63
|
+
return [] if req_body.empty?
|
64
|
+
|
65
|
+
req_body.each_line.each_slice(2).with_object(Set.new) do |(cmd_line, _data_line), acc|
|
66
|
+
cmd = JSON.parse(cmd_line)
|
67
|
+
acc << aggregate_index(cmd[cmd.keys.first]['_index'])
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
def msearch_indices
|
72
|
+
req_body = record.fetch(:request_body, {})
|
73
|
+
return [] if req_body.empty?
|
74
|
+
|
75
|
+
req_body.each_line.each_slice(2).with_object(Set.new) do |(cmd_line, _data_line), acc|
|
76
|
+
cmd = JSON.parse(cmd_line)
|
77
|
+
acc << aggregate_index(cmd['index'])
|
78
|
+
end
|
79
|
+
end
|
80
|
+
|
81
|
+
def aggregate_index(index)
|
82
|
+
return unless index
|
83
|
+
return index unless conf.aggregate_index
|
84
|
+
|
85
|
+
index.sub(INDEX_PATTERN, '').sub(ILM_PATTERN, '')
|
86
|
+
end
|
87
|
+
|
88
|
+
def generate_metrics
|
89
|
+
query_action, query_index = query_details
|
90
|
+
indices = case query_action
|
91
|
+
when 'bulk' then bulk_indices
|
92
|
+
when 'msearch' then msearch_indices
|
93
|
+
else []
|
94
|
+
end
|
95
|
+
indices << aggregate_index(query_index) if query_index || indices.empty?
|
96
|
+
|
97
|
+
indices.inject([]) do |metrics, index|
|
98
|
+
metrics << base.merge("#{conf.prefix}index" => index,
|
99
|
+
"#{conf.prefix}query_type" => query_action)
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
105
|
+
end
|
@@ -29,7 +29,8 @@ module Fluent
|
|
29
29
|
'indices:monitor/' => 'monitor'
|
30
30
|
}.freeze
|
31
31
|
|
32
|
-
|
32
|
+
INDEX_PATTERN = /-?\*$/.freeze
|
33
|
+
ILM_PATTERN = /-\d{6}$/.freeze
|
33
34
|
|
34
35
|
attr_reader :record, :conf
|
35
36
|
|
@@ -69,19 +70,25 @@ module Fluent
|
|
69
70
|
|
70
71
|
def indices
|
71
72
|
indices = record[:r_indices] || record[:indices] || [nil]
|
72
|
-
if conf.
|
73
|
+
if conf.aggregate_index
|
73
74
|
indices = indices.inject(Set.new) do |acc, index|
|
74
|
-
|
75
|
-
|
76
|
-
end.to_a
|
75
|
+
acc << aggregate_index(index)
|
76
|
+
end
|
77
77
|
end
|
78
78
|
indices
|
79
79
|
end
|
80
80
|
|
81
|
+
def aggregate_index(index)
|
82
|
+
return unless index
|
83
|
+
return index unless conf.aggregate_index
|
84
|
+
|
85
|
+
index.sub(INDEX_PATTERN, '').sub(ILM_PATTERN, '')
|
86
|
+
end
|
87
|
+
|
81
88
|
def generate_metrics
|
82
89
|
metrics = []
|
83
|
-
indices.each do |
|
84
|
-
metrics << base.merge("#{conf.prefix}
|
90
|
+
indices.each do |index|
|
91
|
+
metrics << base.merge("#{conf.prefix}index" => index)
|
85
92
|
end
|
86
93
|
metrics
|
87
94
|
end
|
@@ -31,7 +31,7 @@ module Fluent
|
|
31
31
|
|
32
32
|
helpers :event_emitter
|
33
33
|
|
34
|
-
ALLOWED_CATEGORIES = %w[GRANTED_PRIVILEGES].freeze
|
34
|
+
ALLOWED_CATEGORIES = %w[GRANTED_PRIVILEGES FAILED_LOGIN].freeze
|
35
35
|
# FAILED_LOGIN AUTHENTICATED MISSING_PRIVILEGES SSL_EXCEPTION
|
36
36
|
# OPENDISTRO_SECURITY_INDEX_ATTEMPT BAD_HEADERS
|
37
37
|
|
@@ -42,6 +42,8 @@ module Fluent
|
|
42
42
|
DEFAULT_USER_KEY = 'audit_request_effective_user'
|
43
43
|
DEFAULT_INDICES_KEY = 'audit_trace_indices'
|
44
44
|
DEFAULT_R_INDICES_KEY = 'audit_trace_resolved_indices'
|
45
|
+
DEFAULT_REST_REQUEST_PATH = 'audit_rest_request_path'
|
46
|
+
DEFAULT_REQUEST_BODY = 'audit_request_body'
|
45
47
|
DEFAULT_TIMESTAMP_KEY = '@timestamp'
|
46
48
|
DEFAULT_PRIVILEGE_KEY = 'audit_request_privilege'
|
47
49
|
DEFAULT_PREFIX = ''
|
@@ -69,13 +71,17 @@ module Fluent
|
|
69
71
|
config_param :timestamp_key, :string, default: DEFAULT_TIMESTAMP_KEY
|
70
72
|
desc 'Request privilege key'
|
71
73
|
config_param :privilege_key, :string, default: DEFAULT_PRIVILEGE_KEY
|
74
|
+
desc 'Rest request path key'
|
75
|
+
config_param :rest_request_path_key, :string, default: DEFAULT_REST_REQUEST_PATH
|
76
|
+
desc 'Request body key'
|
77
|
+
config_param :request_body_key, :string, default: DEFAULT_REQUEST_BODY
|
72
78
|
|
73
79
|
desc 'Timestamp format'
|
74
80
|
config_param :timestamp_format, :enum, list: %i[iso epochmillis epochmillis_str], default: :iso
|
75
81
|
desc 'Attribute prefix'
|
76
82
|
config_param :prefix, :string, default: DEFAULT_PREFIX
|
77
|
-
desc 'Aggregate
|
78
|
-
config_param :
|
83
|
+
desc 'Aggregate index'
|
84
|
+
config_param :aggregate_index, :bool, default: true
|
79
85
|
desc 'Events block size'
|
80
86
|
config_param :event_stream_size, :integer, default: 1000
|
81
87
|
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: fluent-plugin-elastic-log
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.5.1
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Thomas Tych
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2023-06-
|
11
|
+
date: 2023-06-27 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: bump
|
@@ -163,6 +163,7 @@ files:
|
|
163
163
|
- Rakefile
|
164
164
|
- fluent-plugin-elastic-log.gemspec
|
165
165
|
- lib/fluent/plugin/elastic_log/audit_log_to_metric_processor.rb
|
166
|
+
- lib/fluent/plugin/elastic_log/failed_login_metric.rb
|
166
167
|
- lib/fluent/plugin/elastic_log/granted_privileges_metric.rb
|
167
168
|
- lib/fluent/plugin/out_elastic_audit_log_metric.rb
|
168
169
|
homepage: https://gitlab.com/ttych/fluent-plugin-elastic-log
|