fluentd 0.14.17-x64-mingw32 → 1.3.1-x64-mingw32
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of fluentd might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/.travis.yml +16 -5
- data/ADOPTERS.md +5 -0
- data/{ChangeLog → CHANGELOG.md} +495 -6
- data/CONTRIBUTING.md +5 -2
- data/GOVERNANCE.md +55 -0
- data/LICENSE +202 -0
- data/MAINTAINERS.md +7 -5
- data/README.md +17 -10
- data/bin/fluent-ca-generate +6 -0
- data/example/counter.conf +18 -0
- data/example/secondary_file.conf +3 -2
- data/fluentd.gemspec +3 -3
- data/lib/fluent/agent.rb +1 -1
- data/lib/fluent/command/binlog_reader.rb +11 -2
- data/lib/fluent/command/ca_generate.rb +181 -0
- data/lib/fluent/command/cat.rb +28 -15
- data/lib/fluent/command/debug.rb +4 -4
- data/lib/fluent/command/fluentd.rb +2 -2
- data/lib/fluent/command/plugin_config_formatter.rb +24 -2
- data/lib/fluent/command/plugin_generator.rb +26 -8
- data/lib/fluent/config/configure_proxy.rb +7 -1
- data/lib/fluent/config/dsl.rb +8 -5
- data/lib/fluent/config/element.rb +5 -0
- data/lib/fluent/config/literal_parser.rb +7 -1
- data/lib/fluent/config/types.rb +28 -2
- data/lib/fluent/config/v1_parser.rb +1 -2
- data/lib/fluent/configurable.rb +1 -0
- data/lib/fluent/counter.rb +23 -0
- data/lib/fluent/counter/base_socket.rb +46 -0
- data/lib/fluent/counter/client.rb +297 -0
- data/lib/fluent/counter/error.rb +86 -0
- data/lib/fluent/counter/mutex_hash.rb +163 -0
- data/lib/fluent/counter/server.rb +273 -0
- data/lib/fluent/counter/store.rb +205 -0
- data/lib/fluent/counter/validator.rb +145 -0
- data/lib/fluent/env.rb +1 -0
- data/lib/fluent/event_router.rb +1 -1
- data/lib/fluent/log.rb +119 -29
- data/lib/fluent/plugin/base.rb +12 -0
- data/lib/fluent/plugin/buf_file.rb +20 -16
- data/lib/fluent/plugin/buffer.rb +130 -32
- data/lib/fluent/plugin/buffer/file_chunk.rb +23 -4
- data/lib/fluent/plugin/compressable.rb +1 -1
- data/lib/fluent/plugin/filter_grep.rb +135 -21
- data/lib/fluent/plugin/filter_parser.rb +13 -2
- data/lib/fluent/plugin/filter_record_transformer.rb +16 -14
- data/lib/fluent/plugin/formatter_stdout.rb +3 -2
- data/lib/fluent/plugin/formatter_tsv.rb +5 -1
- data/lib/fluent/plugin/in_debug_agent.rb +8 -1
- data/lib/fluent/plugin/in_forward.rb +1 -1
- data/lib/fluent/plugin/in_http.rb +84 -3
- data/lib/fluent/plugin/in_monitor_agent.rb +7 -1
- data/lib/fluent/plugin/in_syslog.rb +31 -10
- data/lib/fluent/plugin/in_tail.rb +142 -53
- data/lib/fluent/plugin/in_tcp.rb +5 -6
- data/lib/fluent/plugin/in_udp.rb +6 -2
- data/lib/fluent/plugin/in_unix.rb +1 -1
- data/lib/fluent/plugin/multi_output.rb +1 -0
- data/lib/fluent/plugin/out_copy.rb +25 -2
- data/lib/fluent/plugin/out_file.rb +26 -7
- data/lib/fluent/plugin/out_forward.rb +81 -42
- data/lib/fluent/plugin/out_secondary_file.rb +2 -2
- data/lib/fluent/plugin/out_stdout.rb +0 -1
- data/lib/fluent/plugin/out_stream.rb +1 -1
- data/lib/fluent/plugin/output.rb +221 -57
- data/lib/fluent/plugin/parser_apache.rb +1 -1
- data/lib/fluent/plugin/parser_apache2.rb +5 -1
- data/lib/fluent/plugin/parser_apache_error.rb +1 -1
- data/lib/fluent/plugin/parser_json.rb +10 -3
- data/lib/fluent/plugin/parser_ltsv.rb +7 -0
- data/lib/fluent/plugin/parser_multiline.rb +2 -1
- data/lib/fluent/plugin/parser_nginx.rb +1 -1
- data/lib/fluent/plugin/parser_none.rb +1 -0
- data/lib/fluent/plugin/parser_regexp.rb +15 -14
- data/lib/fluent/plugin/parser_syslog.rb +9 -5
- data/lib/fluent/plugin_helper.rb +2 -0
- data/lib/fluent/plugin_helper/cert_option.rb +28 -9
- data/lib/fluent/plugin_helper/compat_parameters.rb +3 -1
- data/lib/fluent/plugin_helper/counter.rb +51 -0
- data/lib/fluent/plugin_helper/event_loop.rb +9 -0
- data/lib/fluent/plugin_helper/record_accessor.rb +210 -0
- data/lib/fluent/plugin_helper/retry_state.rb +15 -7
- data/lib/fluent/plugin_helper/server.rb +87 -25
- data/lib/fluent/plugin_helper/socket_option.rb +5 -2
- data/lib/fluent/plugin_helper/timer.rb +8 -7
- data/lib/fluent/root_agent.rb +18 -9
- data/lib/fluent/supervisor.rb +63 -23
- data/lib/fluent/system_config.rb +30 -2
- data/lib/fluent/test/helpers.rb +1 -1
- data/lib/fluent/time.rb +15 -7
- data/lib/fluent/timezone.rb +26 -2
- data/lib/fluent/version.rb +1 -1
- data/templates/new_gem/README.md.erb +2 -2
- data/templates/new_gem/lib/fluent/plugin/filter.rb.erb +1 -1
- data/templates/new_gem/lib/fluent/plugin/input.rb.erb +1 -1
- data/templates/new_gem/lib/fluent/plugin/output.rb.erb +1 -1
- data/templates/new_gem/lib/fluent/plugin/parser.rb.erb +4 -4
- data/test/command/test_ca_generate.rb +70 -0
- data/test/command/test_fluentd.rb +2 -2
- data/test/command/test_plugin_config_formatter.rb +8 -7
- data/test/command/test_plugin_generator.rb +65 -39
- data/test/config/test_config_parser.rb +7 -2
- data/test/config/test_configurable.rb +7 -2
- data/test/config/test_configure_proxy.rb +41 -3
- data/test/config/test_dsl.rb +10 -10
- data/test/config/test_element.rb +10 -0
- data/test/config/test_literal_parser.rb +8 -0
- data/test/config/test_plugin_configuration.rb +56 -0
- data/test/config/test_system_config.rb +19 -1
- data/test/config/test_types.rb +37 -0
- data/test/counter/test_client.rb +559 -0
- data/test/counter/test_error.rb +44 -0
- data/test/counter/test_mutex_hash.rb +179 -0
- data/test/counter/test_server.rb +589 -0
- data/test/counter/test_store.rb +258 -0
- data/test/counter/test_validator.rb +137 -0
- data/test/plugin/test_buf_file.rb +124 -0
- data/test/plugin/test_buffer.rb +3 -2
- data/test/plugin/test_filter_grep.rb +580 -2
- data/test/plugin/test_filter_parser.rb +33 -2
- data/test/plugin/test_filter_record_transformer.rb +22 -1
- data/test/plugin/test_formatter_ltsv.rb +3 -0
- data/test/plugin/test_formatter_tsv.rb +68 -0
- data/test/plugin/test_in_debug_agent.rb +21 -0
- data/test/plugin/test_in_exec.rb +3 -5
- data/test/plugin/test_in_http.rb +178 -0
- data/test/plugin/test_in_monitor_agent.rb +1 -1
- data/test/plugin/test_in_syslog.rb +64 -0
- data/test/plugin/test_in_tail.rb +116 -6
- data/test/plugin/test_in_tcp.rb +21 -0
- data/test/plugin/test_in_udp.rb +78 -0
- data/test/plugin/test_metadata.rb +89 -0
- data/test/plugin/test_out_copy.rb +31 -0
- data/test/plugin/test_out_file.rb +108 -2
- data/test/plugin/test_out_forward.rb +195 -2
- data/test/plugin/test_out_secondary_file.rb +14 -0
- data/test/plugin/test_output.rb +159 -45
- data/test/plugin/test_output_as_buffered.rb +19 -0
- data/test/plugin/test_output_as_buffered_backup.rb +307 -0
- data/test/plugin/test_output_as_buffered_retries.rb +70 -0
- data/test/plugin/test_output_as_buffered_secondary.rb +1 -1
- data/test/plugin/test_parser_apache2.rb +1 -0
- data/test/plugin/test_parser_labeled_tsv.rb +17 -0
- data/test/plugin/test_parser_nginx.rb +40 -0
- data/test/plugin/test_parser_regexp.rb +6 -7
- data/test/plugin/test_parser_syslog.rb +155 -5
- data/test/plugin_helper/test_child_process.rb +4 -4
- data/test/plugin_helper/test_compat_parameters.rb +22 -0
- data/test/plugin_helper/test_record_accessor.rb +197 -0
- data/test/plugin_helper/test_retry_state.rb +20 -0
- data/test/plugin_helper/test_server.rb +30 -2
- data/test/test_config.rb +3 -3
- data/test/test_configdsl.rb +2 -2
- data/test/test_log.rb +51 -1
- data/test/test_root_agent.rb +33 -0
- data/test/test_supervisor.rb +105 -0
- metadata +68 -8
- data/COPYING +0 -14
@@ -22,54 +22,168 @@ module Fluent::Plugin
|
|
22
22
|
class GrepFilter < Filter
|
23
23
|
Fluent::Plugin.register_filter('grep', self)
|
24
24
|
|
25
|
-
|
25
|
+
def initialize
|
26
|
+
super
|
26
27
|
|
27
|
-
|
28
|
-
|
28
|
+
@_regexp_and_conditions = nil
|
29
|
+
@_exclude_and_conditions = nil
|
30
|
+
@_regexp_or_conditions = nil
|
31
|
+
@_exclude_or_conditions = nil
|
32
|
+
end
|
29
33
|
|
30
34
|
# for test
|
31
|
-
attr_reader :
|
32
|
-
|
35
|
+
attr_reader :_regexp_and_conditions, :_exclude_and_conditions, :_regexp_or_conditions, :_exclude_or_conditions
|
36
|
+
|
37
|
+
helpers :record_accessor
|
38
|
+
|
39
|
+
REGEXP_MAX_NUM = 20
|
40
|
+
|
41
|
+
(1..REGEXP_MAX_NUM).each {|i| config_param :"regexp#{i}", :string, default: nil, deprecated: "Use <regexp> section" }
|
42
|
+
(1..REGEXP_MAX_NUM).each {|i| config_param :"exclude#{i}", :string, default: nil, deprecated: "Use <exclude> section" }
|
43
|
+
|
44
|
+
config_section :regexp, param_name: :regexps, multi: true do
|
45
|
+
desc "The field name to which the regular expression is applied."
|
46
|
+
config_param :key, :string
|
47
|
+
desc "The regular expression."
|
48
|
+
config_param :pattern, :regexp
|
49
|
+
end
|
50
|
+
|
51
|
+
config_section :exclude, param_name: :excludes, multi: true do
|
52
|
+
desc "The field name to which the regular expression is applied."
|
53
|
+
config_param :key, :string
|
54
|
+
desc "The regular expression."
|
55
|
+
config_param :pattern, :regexp
|
56
|
+
end
|
57
|
+
|
58
|
+
config_section :and, param_name: :and_conditions, multi: true do
|
59
|
+
config_section :regexp, param_name: :regexps, multi: true do
|
60
|
+
desc "The field name to which the regular expression is applied."
|
61
|
+
config_param :key, :string
|
62
|
+
desc "The regular expression."
|
63
|
+
config_param :pattern, :regexp
|
64
|
+
end
|
65
|
+
config_section :exclude, param_name: :excludes, multi: true do
|
66
|
+
desc "The field name to which the regular expression is applied."
|
67
|
+
config_param :key, :string
|
68
|
+
desc "The regular expression."
|
69
|
+
config_param :pattern, :regexp
|
70
|
+
end
|
71
|
+
end
|
72
|
+
|
73
|
+
config_section :or, param_name: :or_conditions, multi: true do
|
74
|
+
config_section :regexp, param_name: :regexps, multi: true do
|
75
|
+
desc "The field name to which the regular expression is applied."
|
76
|
+
config_param :key, :string
|
77
|
+
desc "The regular expression."
|
78
|
+
config_param :pattern, :regexp
|
79
|
+
end
|
80
|
+
config_section :exclude, param_name: :excludes, multi: true do
|
81
|
+
desc "The field name to which the regular expression is applied."
|
82
|
+
config_param :key, :string
|
83
|
+
desc "The regular expression."
|
84
|
+
config_param :pattern, :regexp
|
85
|
+
end
|
86
|
+
end
|
33
87
|
|
34
88
|
def configure(conf)
|
35
89
|
super
|
36
90
|
|
37
|
-
|
91
|
+
regexp_and_conditions = {}
|
92
|
+
regexp_or_conditions = {}
|
93
|
+
exclude_and_conditions = {}
|
94
|
+
exclude_or_conditions = {}
|
95
|
+
|
38
96
|
(1..REGEXP_MAX_NUM).each do |i|
|
39
97
|
next unless conf["regexp#{i}"]
|
40
98
|
key, regexp = conf["regexp#{i}"].split(/ /, 2)
|
41
99
|
raise Fluent::ConfigError, "regexp#{i} does not contain 2 parameters" unless regexp
|
42
|
-
raise Fluent::ConfigError, "regexp#{i} contains a duplicated key, #{key}" if
|
43
|
-
|
100
|
+
raise Fluent::ConfigError, "regexp#{i} contains a duplicated key, #{key}" if regexp_and_conditions[key]
|
101
|
+
regexp_and_conditions[key] = Expression.new(record_accessor_create(key), Regexp.compile(regexp))
|
44
102
|
end
|
45
103
|
|
46
|
-
@excludes = {}
|
47
104
|
(1..REGEXP_MAX_NUM).each do |i|
|
48
105
|
next unless conf["exclude#{i}"]
|
49
106
|
key, exclude = conf["exclude#{i}"].split(/ /, 2)
|
50
107
|
raise Fluent::ConfigError, "exclude#{i} does not contain 2 parameters" unless exclude
|
51
|
-
raise Fluent::ConfigError, "exclude#{i} contains a duplicated key, #{key}" if
|
52
|
-
|
108
|
+
raise Fluent::ConfigError, "exclude#{i} contains a duplicated key, #{key}" if exclude_or_conditions[key]
|
109
|
+
exclude_or_conditions[key] = Expression.new(record_accessor_create(key), Regexp.compile(exclude))
|
110
|
+
end
|
111
|
+
|
112
|
+
if @regexps.size > 1
|
113
|
+
log.info "Top level multiple <regexp> is intepreted as 'and' condition"
|
53
114
|
end
|
115
|
+
@regexps.each do |e|
|
116
|
+
raise Fluent::ConfigError, "Duplicate key: #{e.key}" if regexp_and_conditions.key?(e.key)
|
117
|
+
regexp_and_conditions[e.key] = Expression.new(record_accessor_create(e.key), e.pattern)
|
118
|
+
end
|
119
|
+
|
120
|
+
if @excludes.size > 1
|
121
|
+
log.info "Top level multiple <exclude> is intepreted as 'or' condition"
|
122
|
+
end
|
123
|
+
@excludes.each do |e|
|
124
|
+
raise Fluent::ConfigError, "Duplicate key: #{e.key}" if exclude_or_conditions.key?(e.key)
|
125
|
+
exclude_or_conditions[e.key] = Expression.new(record_accessor_create(e.key), e.pattern)
|
126
|
+
end
|
127
|
+
|
128
|
+
@and_conditions.each do |and_condition|
|
129
|
+
if !and_condition.regexps.empty? && !and_condition.excludes.empty?
|
130
|
+
raise Fluent::ConfigError, "Do not specify both <regexp> and <exclude> in <and>"
|
131
|
+
end
|
132
|
+
and_condition.regexps.each do |e|
|
133
|
+
raise Fluent::ConfigError, "Duplicate key in <and>: #{e.key}" if regexp_and_conditions.key?(e.key)
|
134
|
+
regexp_and_conditions[e.key] = Expression.new(record_accessor_create(e.key), e.pattern)
|
135
|
+
end
|
136
|
+
and_condition.excludes.each do |e|
|
137
|
+
raise Fluent::ConfigError, "Duplicate key in <and>: #{e.key}" if exclude_and_conditions.key?(e.key)
|
138
|
+
exclude_and_conditions[e.key] = Expression.new(record_accessor_create(e.key), e.pattern)
|
139
|
+
end
|
140
|
+
end
|
141
|
+
|
142
|
+
@or_conditions.each do |or_condition|
|
143
|
+
if !or_condition.regexps.empty? && !or_condition.excludes.empty?
|
144
|
+
raise Fluent::ConfigError, "Do not specify both <regexp> and <exclude> in <or>"
|
145
|
+
end
|
146
|
+
or_condition.regexps.each do |e|
|
147
|
+
raise Fluent::ConfigError, "Duplicate key in <or>: #{e.key}" if regexp_or_conditions.key?(e.key)
|
148
|
+
regexp_or_conditions[e.key] = Expression.new(record_accessor_create(e.key), e.pattern)
|
149
|
+
end
|
150
|
+
or_condition.excludes.each do |e|
|
151
|
+
raise Fluent::ConfigError, "Duplicate key in <or>: #{e.key}" if exclude_or_conditions.key?(e.key)
|
152
|
+
exclude_or_conditions[e.key] = Expression.new(record_accessor_create(e.key), e.pattern)
|
153
|
+
end
|
154
|
+
end
|
155
|
+
|
156
|
+
@_regexp_and_conditions = regexp_and_conditions.values unless regexp_and_conditions.empty?
|
157
|
+
@_exclude_and_conditions = exclude_and_conditions.values unless exclude_and_conditions.empty?
|
158
|
+
@_regexp_or_conditions = regexp_or_conditions.values unless regexp_or_conditions.empty?
|
159
|
+
@_exclude_or_conditions = exclude_or_conditions.values unless exclude_or_conditions.empty?
|
54
160
|
end
|
55
161
|
|
56
162
|
def filter(tag, time, record)
|
57
|
-
result = nil
|
58
163
|
begin
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
164
|
+
if @_regexp_and_conditions && @_regexp_and_conditions.any? { |expression| !expression.match?(record) }
|
165
|
+
return nil
|
166
|
+
end
|
167
|
+
if @_regexp_or_conditions && @_regexp_or_conditions.none? { |expression| expression.match?(record) }
|
168
|
+
return nil
|
169
|
+
end
|
170
|
+
if @_exclude_and_conditions && @_exclude_and_conditions.all? { |expression| expression.match?(record) }
|
171
|
+
return nil
|
172
|
+
end
|
173
|
+
if @_exclude_or_conditions && @_exclude_or_conditions.any? { |expression| expression.match?(record) }
|
174
|
+
return nil
|
67
175
|
end
|
68
176
|
rescue => e
|
69
177
|
log.warn "failed to grep events", error: e
|
70
178
|
log.warn_backtrace
|
71
179
|
end
|
72
|
-
|
180
|
+
record
|
181
|
+
end
|
182
|
+
|
183
|
+
Expression = Struct.new(:key, :pattern) do
|
184
|
+
def match?(record)
|
185
|
+
::Fluent::StringUtil.match_regexp(pattern, key.call(record).to_s)
|
186
|
+
end
|
73
187
|
end
|
74
188
|
end
|
75
189
|
end
|
@@ -24,14 +24,23 @@ module Fluent::Plugin
|
|
24
24
|
class ParserFilter < Filter
|
25
25
|
Fluent::Plugin.register_filter('parser', self)
|
26
26
|
|
27
|
-
helpers :parser, :compat_parameters
|
27
|
+
helpers :parser, :record_accessor, :compat_parameters
|
28
28
|
|
29
|
+
desc 'Specify field name in the record to parse.'
|
29
30
|
config_param :key_name, :string
|
31
|
+
desc 'Keep original key-value pair in parsed result.'
|
30
32
|
config_param :reserve_data, :bool, default: false
|
33
|
+
desc 'Keep original event time in parsed result.'
|
31
34
|
config_param :reserve_time, :bool, default: false
|
35
|
+
desc 'Remove "key_name" field from the record when parsing is succeeded'
|
36
|
+
config_param :remove_key_name_field, :bool, default: false
|
37
|
+
desc 'Store parsed values with specified key name prefix.'
|
32
38
|
config_param :inject_key_prefix, :string, default: nil
|
39
|
+
desc 'If true, invalid string is replaced with safe characters and re-parse it.'
|
33
40
|
config_param :replace_invalid_sequence, :bool, default: false
|
41
|
+
desc 'Store parsed values as a hash value in a field.'
|
34
42
|
config_param :hash_value_field, :string, default: nil
|
43
|
+
desc 'Emit invalid record to @ERROR label'
|
35
44
|
config_param :emit_invalid_record_to_error, :bool, default: true
|
36
45
|
|
37
46
|
attr_reader :parser
|
@@ -41,6 +50,7 @@ module Fluent::Plugin
|
|
41
50
|
|
42
51
|
super
|
43
52
|
|
53
|
+
@accessor = record_accessor_create(@key_name)
|
44
54
|
@parser = parser_create
|
45
55
|
end
|
46
56
|
|
@@ -48,7 +58,7 @@ module Fluent::Plugin
|
|
48
58
|
REPLACE_CHAR = '?'.freeze
|
49
59
|
|
50
60
|
def filter_with_time(tag, time, record)
|
51
|
-
raw_value = record
|
61
|
+
raw_value = @accessor.call(record)
|
52
62
|
if raw_value.nil?
|
53
63
|
if @emit_invalid_record_to_error
|
54
64
|
router.emit_error_event(tag, time, record, ArgumentError.new("#{@key_name} does not exist"))
|
@@ -67,6 +77,7 @@ module Fluent::Plugin
|
|
67
77
|
else
|
68
78
|
t.nil? ? time : t
|
69
79
|
end
|
80
|
+
@accessor.delete(record) if @remove_key_name_field
|
70
81
|
r = handle_parsed(tag, record, t, values)
|
71
82
|
return t, r
|
72
83
|
else
|
@@ -27,10 +27,12 @@ module Fluent::Plugin
|
|
27
27
|
class RecordTransformerFilter < Fluent::Plugin::Filter
|
28
28
|
Fluent::Plugin.register_filter('record_transformer', self)
|
29
29
|
|
30
|
+
helpers :record_accessor
|
31
|
+
|
30
32
|
desc 'A comma-delimited list of keys to delete.'
|
31
|
-
config_param :remove_keys, :
|
33
|
+
config_param :remove_keys, :array, default: nil
|
32
34
|
desc 'A comma-delimited list of keys to keep.'
|
33
|
-
config_param :keep_keys, :
|
35
|
+
config_param :keep_keys, :array, default: nil
|
34
36
|
desc 'Create new Hash to transform incoming data'
|
35
37
|
config_param :renew_record, :bool, default: false
|
36
38
|
desc 'Specify field name of the record to overwrite the time of events. Its value must be unix time.'
|
@@ -52,15 +54,14 @@ module Fluent::Plugin
|
|
52
54
|
end
|
53
55
|
end
|
54
56
|
|
55
|
-
if @remove_keys
|
56
|
-
@remove_keys = @remove_keys.split(',')
|
57
|
-
end
|
58
|
-
|
59
57
|
if @keep_keys
|
60
58
|
raise Fluent::ConfigError, "`renew_record` must be true to use `keep_keys`" unless @renew_record
|
61
|
-
@keep_keys = @keep_keys.split(',')
|
62
59
|
end
|
63
60
|
|
61
|
+
@key_deleters = if @remove_keys
|
62
|
+
@remove_keys.map { |k| record_accessor_create(k) }
|
63
|
+
end
|
64
|
+
|
64
65
|
placeholder_expander_params = {
|
65
66
|
log: log,
|
66
67
|
auto_typecast: @auto_typecast,
|
@@ -101,8 +102,7 @@ module Fluent::Plugin
|
|
101
102
|
if @renew_time_key && new_record.has_key?(@renew_time_key)
|
102
103
|
time = Fluent::EventTime.from_time(Time.at(new_record[@renew_time_key].to_f))
|
103
104
|
end
|
104
|
-
@
|
105
|
-
|
105
|
+
@key_deleters.each { |deleter| deleter.delete(new_record) } if @key_deleters
|
106
106
|
new_es.add(time, new_record)
|
107
107
|
rescue => e
|
108
108
|
router.emit_error_event(tag, time, record, e)
|
@@ -129,7 +129,9 @@ module Fluent::Plugin
|
|
129
129
|
placeholders = @placeholder_expander.prepare_placeholders(placeholder_values)
|
130
130
|
|
131
131
|
new_record = @renew_record ? {} : record.dup
|
132
|
-
@keep_keys.each
|
132
|
+
@keep_keys.each do |k|
|
133
|
+
new_record[k] = record[k] if record.has_key?(k)
|
134
|
+
end if @keep_keys && @renew_record
|
133
135
|
new_record.merge!(expand_placeholders(@map, placeholders))
|
134
136
|
|
135
137
|
new_record
|
@@ -218,7 +220,7 @@ module Fluent::Plugin
|
|
218
220
|
# @param [String] str
|
219
221
|
# @param [Boolean] force_stringify the value must be string, used for hash key
|
220
222
|
def expand(str, placeholders, force_stringify = false)
|
221
|
-
if @auto_typecast
|
223
|
+
if @auto_typecast && !force_stringify
|
222
224
|
single_placeholder_matched = str.match(/\A(\${[^}]+}|__[A-Z_]+__)\z/)
|
223
225
|
if single_placeholder_matched
|
224
226
|
log_if_unknown_placeholder($1, placeholders)
|
@@ -261,9 +263,9 @@ module Fluent::Plugin
|
|
261
263
|
def preprocess_map(value, force_stringify = false)
|
262
264
|
new_value = nil
|
263
265
|
if value.is_a?(String)
|
264
|
-
if @auto_typecast
|
266
|
+
if @auto_typecast && !force_stringify
|
265
267
|
num_placeholders = value.scan('${').size
|
266
|
-
if num_placeholders == 1
|
268
|
+
if num_placeholders == 1 && value.start_with?('${') && value.end_with?('}')
|
267
269
|
new_value = value[2..-2] # ${..} => ..
|
268
270
|
end
|
269
271
|
end
|
@@ -314,7 +316,7 @@ module Fluent::Plugin
|
|
314
316
|
end
|
315
317
|
|
316
318
|
(Object.instance_methods).each do |m|
|
317
|
-
undef_method m unless m.to_s =~ /^__|respond_to_missing\?|object_id|public_methods|instance_eval|method_missing|define_singleton_method|respond_to\?|new_ostruct_member
|
319
|
+
undef_method m unless m.to_s =~ /^__|respond_to_missing\?|object_id|public_methods|instance_eval|method_missing|define_singleton_method|respond_to\?|new_ostruct_member|^class$/
|
318
320
|
end
|
319
321
|
end
|
320
322
|
end
|
@@ -21,13 +21,14 @@ module Fluent
|
|
21
21
|
class StdoutFormatter < Formatter
|
22
22
|
Plugin.register_formatter('stdout', self)
|
23
23
|
|
24
|
-
TIME_FORMAT = '%Y-%m-%d %H:%M:%S.%
|
24
|
+
TIME_FORMAT = '%Y-%m-%d %H:%M:%S.%N %z'
|
25
25
|
|
26
26
|
config_param :output_type, :string, default: 'json'
|
27
27
|
|
28
28
|
def configure(conf)
|
29
29
|
super
|
30
30
|
|
31
|
+
@time_formatter = Strftime.new(TIME_FORMAT)
|
31
32
|
@sub_formatter = Plugin.new_formatter(@output_type, parent: self.owner)
|
32
33
|
@sub_formatter.configure(conf)
|
33
34
|
end
|
@@ -38,7 +39,7 @@ module Fluent
|
|
38
39
|
end
|
39
40
|
|
40
41
|
def format(tag, time, record)
|
41
|
-
"#{Time.at(time).localtime
|
42
|
+
"#{@time_formatter.exec(Time.at(time).localtime)} #{tag}: #{@sub_formatter.format(tag, time, record).chomp}\n"
|
42
43
|
end
|
43
44
|
|
44
45
|
def stop
|
@@ -25,9 +25,13 @@ module Fluent
|
|
25
25
|
config_param :keys, :array, value_type: :string
|
26
26
|
desc 'The delimiter character (or string) of TSV values'
|
27
27
|
config_param :delimiter, :string, default: "\t"
|
28
|
+
desc 'The parameter to enable writing to new lines'
|
29
|
+
config_param :add_newline, :bool, default: true
|
28
30
|
|
29
31
|
def format(tag, time, record)
|
30
|
-
@keys.map{|k| record[k].to_s }.join(@delimiter)
|
32
|
+
formatted = @keys.map{|k| record[k].to_s }.join(@delimiter)
|
33
|
+
formatted << "\n".freeze if @add_newline
|
34
|
+
formatted
|
31
35
|
end
|
32
36
|
end
|
33
37
|
end
|
@@ -34,6 +34,9 @@ module Fluent::Plugin
|
|
34
34
|
|
35
35
|
def configure(conf)
|
36
36
|
super
|
37
|
+
if system_config.workers > 1
|
38
|
+
@port += fluentd_worker_id
|
39
|
+
end
|
37
40
|
if @unix_path
|
38
41
|
unless ::Fluent::FileUtil.writable?(@unix_path)
|
39
42
|
raise Fluent::ConfigError, "in_debug_agent: `#{@unix_path}` is not writable"
|
@@ -41,6 +44,10 @@ module Fluent::Plugin
|
|
41
44
|
end
|
42
45
|
end
|
43
46
|
|
47
|
+
def multi_workers_ready?
|
48
|
+
@unix_path.nil?
|
49
|
+
end
|
50
|
+
|
44
51
|
def start
|
45
52
|
super
|
46
53
|
|
@@ -50,7 +57,7 @@ module Fluent::Plugin
|
|
50
57
|
else
|
51
58
|
uri = "druby://#{@bind}:#{@port}"
|
52
59
|
end
|
53
|
-
log.info "listening dRuby", uri: uri, object: @object
|
60
|
+
log.info "listening dRuby", uri: uri, object: @object, worker: fluentd_worker_id
|
54
61
|
obj = eval(@object)
|
55
62
|
@server = DRb::DRbServer.new(uri, obj)
|
56
63
|
end
|
@@ -199,8 +199,8 @@ module Fluent::Plugin
|
|
199
199
|
when :pingpong
|
200
200
|
success, reason_or_salt, shared_key = check_ping(msg, conn.remote_addr, user_auth_salt, nonce)
|
201
201
|
unless success
|
202
|
+
conn.on(:write_complete) { |c| c.close_after_write_complete }
|
202
203
|
send_data.call(serializer, generate_pong(false, reason_or_salt, nonce, shared_key))
|
203
|
-
conn.close
|
204
204
|
next
|
205
205
|
end
|
206
206
|
send_data.call(serializer, generate_pong(true, reason_or_salt, nonce, shared_key))
|
@@ -78,10 +78,16 @@ module Fluent::Plugin
|
|
78
78
|
@parser_json = parser_create(usage: 'parser_in_http_json', type: 'json')
|
79
79
|
@parser_json.estimate_current_event = false
|
80
80
|
@format_name = 'default'
|
81
|
+
@parser_time_key = if parser_config = conf.elements('parse').first
|
82
|
+
parser_config['time_key'] || 'time'
|
83
|
+
else
|
84
|
+
'time'
|
85
|
+
end
|
81
86
|
method(:parse_params_default)
|
82
87
|
else
|
83
88
|
@parser = parser_create
|
84
89
|
@format_name = @parser_configs.first['@type']
|
90
|
+
@parser_time_key = @parser.time_key
|
85
91
|
method(:parse_params_with_parser)
|
86
92
|
end
|
87
93
|
self.singleton_class.module_eval do
|
@@ -202,7 +208,18 @@ module Fluent::Plugin
|
|
202
208
|
if @add_remote_addr
|
203
209
|
single_record['REMOTE_ADDR'] = params['REMOTE_ADDR']
|
204
210
|
end
|
205
|
-
|
211
|
+
|
212
|
+
if defined? @parser
|
213
|
+
single_time = @parser.parse_time(single_record)
|
214
|
+
single_time, single_record = @parser.convert_values(single_time, single_record)
|
215
|
+
else
|
216
|
+
single_time = if t = single_record.delete(@parser_time_key)
|
217
|
+
Fluent::EventTime.from_time(Time.at(t))
|
218
|
+
else
|
219
|
+
time
|
220
|
+
end
|
221
|
+
end
|
222
|
+
|
206
223
|
mes.add(single_time, single_record)
|
207
224
|
end
|
208
225
|
router.emit_stream(tag, mes)
|
@@ -301,6 +318,7 @@ module Fluent::Plugin
|
|
301
318
|
end
|
302
319
|
@env = {}
|
303
320
|
@content_type = ""
|
321
|
+
@content_encoding = ""
|
304
322
|
headers.each_pair {|k,v|
|
305
323
|
@env["HTTP_#{k.gsub('-','_').upcase}"] = v
|
306
324
|
case k
|
@@ -310,6 +328,8 @@ module Fluent::Plugin
|
|
310
328
|
size = v.to_i
|
311
329
|
when /Content-Type/i
|
312
330
|
@content_type = v
|
331
|
+
when /Content-Encoding/i
|
332
|
+
@content_encoding = v
|
313
333
|
when /Connection/i
|
314
334
|
if v =~ /close/i
|
315
335
|
@keep_alive = false
|
@@ -322,6 +342,10 @@ module Fluent::Plugin
|
|
322
342
|
# For multiple X-Forwarded-For headers. Use first header value.
|
323
343
|
v = v.first if v.is_a?(Array)
|
324
344
|
@remote_addr = v.split(",").first
|
345
|
+
when /Access-Control-Request-Method/i
|
346
|
+
@access_control_request_method = v
|
347
|
+
when /Access-Control-Request-Headers/i
|
348
|
+
@access_control_request_headers = v
|
325
349
|
end
|
326
350
|
}
|
327
351
|
if expect
|
@@ -347,20 +371,70 @@ module Fluent::Plugin
|
|
347
371
|
@body << chunk
|
348
372
|
end
|
349
373
|
|
374
|
+
# Web browsers can send an OPTIONS request before performing POST
|
375
|
+
# to check if cross-origin requests are supported.
|
376
|
+
def handle_options_request
|
377
|
+
# Is CORS enabled in the first place?
|
378
|
+
if @cors_allow_origins.nil?
|
379
|
+
return send_response_and_close("403 Forbidden", {}, "")
|
380
|
+
end
|
381
|
+
|
382
|
+
# in_http does not support HTTP methods except POST
|
383
|
+
if @access_control_request_method != 'POST'
|
384
|
+
return send_response_and_close("403 Forbidden", {}, "")
|
385
|
+
end
|
386
|
+
|
387
|
+
header = {
|
388
|
+
"Access-Control-Allow-Methods" => "POST",
|
389
|
+
"Access-Control-Allow-Headers" => @access_control_request_headers || "",
|
390
|
+
}
|
391
|
+
|
392
|
+
# Check the origin and send back a CORS response
|
393
|
+
if @cors_allow_origins.include?('*')
|
394
|
+
header["Access-Control-Allow-Origin"] = "*"
|
395
|
+
send_response_and_close("200 OK", header, "")
|
396
|
+
elsif @cors_allow_origins.include?(@origin)
|
397
|
+
header["Access-Control-Allow-Origin"] = @origin
|
398
|
+
send_response_and_close("200 OK", header, "")
|
399
|
+
else
|
400
|
+
send_response_and_close("403 Forbidden", {}, "")
|
401
|
+
end
|
402
|
+
end
|
403
|
+
|
350
404
|
def on_message_complete
|
351
405
|
return if closing?
|
352
406
|
|
407
|
+
if @parser.http_method == 'OPTIONS'
|
408
|
+
return handle_options_request()
|
409
|
+
end
|
410
|
+
|
353
411
|
# CORS check
|
354
412
|
# ==========
|
355
413
|
# For every incoming request, we check if we have some CORS
|
356
414
|
# restrictions and white listed origins through @cors_allow_origins.
|
357
415
|
unless @cors_allow_origins.nil?
|
358
|
-
unless @cors_allow_origins.include?(@origin)
|
416
|
+
unless @cors_allow_origins.include?('*') or @cors_allow_origins.include?(@origin)
|
359
417
|
send_response_and_close("403 Forbidden", {'Connection' => 'close'}, "")
|
360
418
|
return
|
361
419
|
end
|
362
420
|
end
|
363
421
|
|
422
|
+
# Content Encoding
|
423
|
+
# =================
|
424
|
+
# Decode payload according to the "Content-Encoding" header.
|
425
|
+
# For now, we only support 'gzip' and 'deflate'.
|
426
|
+
begin
|
427
|
+
if @content_encoding == 'gzip'
|
428
|
+
@body = Zlib::GzipReader.new(StringIO.new(@body)).read
|
429
|
+
elsif @content_encoding == 'deflate'
|
430
|
+
@body = Zlib::Inflate.inflate(@body)
|
431
|
+
end
|
432
|
+
rescue
|
433
|
+
@log.warn 'fails to decode payload', error: $!.to_s
|
434
|
+
send_response_and_close("400 Bad Request", {}, "")
|
435
|
+
return
|
436
|
+
end
|
437
|
+
|
364
438
|
@env['REMOTE_ADDR'] = @remote_addr if @remote_addr
|
365
439
|
|
366
440
|
uri = URI.parse(@parser.request_url)
|
@@ -386,7 +460,14 @@ module Fluent::Plugin
|
|
386
460
|
code, header, body = *@callback.call(path_info, params)
|
387
461
|
body = body.to_s
|
388
462
|
|
389
|
-
|
463
|
+
unless @cors_allow_origins.nil?
|
464
|
+
if @cors_allow_origins.include?('*')
|
465
|
+
header['Access-Control-Allow-Origin'] = '*'
|
466
|
+
elsif @cors_allow_origins.include?(@origin)
|
467
|
+
header['Access-Control-Allow-Origin'] = @origin
|
468
|
+
end
|
469
|
+
end
|
470
|
+
|
390
471
|
if @keep_alive
|
391
472
|
header['Connection'] = 'Keep-Alive'
|
392
473
|
send_response(code, header, body)
|