fluentd 1.11.0-x86-mingw32 → 1.11.5-x86-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/CHANGELOG.md +92 -1
- data/example/copy_roundrobin.conf +3 -3
- data/example/counter.conf +1 -1
- data/example/filter_stdout.conf +2 -2
- data/example/{in_dummy_blocks.conf → in_sample_blocks.conf} +4 -4
- data/example/{in_dummy_with_compression.conf → in_sample_with_compression.conf} +3 -3
- data/example/logevents.conf +5 -5
- data/example/multi_filters.conf +1 -1
- data/example/out_exec_filter.conf +2 -2
- data/example/out_forward.conf +1 -1
- data/example/out_forward_buf_file.conf +1 -1
- data/example/out_forward_client.conf +5 -5
- data/example/out_forward_heartbeat_none.conf +1 -1
- data/example/out_forward_sd.conf +1 -1
- data/example/out_forward_shared_key.conf +2 -2
- data/example/out_forward_tls.conf +1 -1
- data/example/out_forward_users.conf +3 -3
- data/example/out_null.conf +4 -4
- data/example/secondary_file.conf +1 -1
- data/fluentd.gemspec +6 -6
- data/lib/fluent/command/fluentd.rb +11 -0
- data/lib/fluent/log.rb +33 -3
- data/lib/fluent/match.rb +9 -0
- data/lib/fluent/plugin/buffer.rb +49 -40
- data/lib/fluent/plugin/buffer/chunk.rb +2 -1
- data/lib/fluent/plugin/formatter.rb +24 -0
- data/lib/fluent/plugin/formatter_hash.rb +3 -1
- data/lib/fluent/plugin/formatter_json.rb +3 -1
- data/lib/fluent/plugin/formatter_ltsv.rb +3 -1
- data/lib/fluent/plugin/formatter_out_file.rb +3 -1
- data/lib/fluent/plugin/formatter_single_value.rb +3 -1
- data/lib/fluent/plugin/formatter_tsv.rb +3 -1
- data/lib/fluent/plugin/in_dummy.rb +2 -123
- data/lib/fluent/plugin/in_exec.rb +4 -2
- data/lib/fluent/plugin/in_http.rb +148 -77
- data/lib/fluent/plugin/in_sample.rb +141 -0
- data/lib/fluent/plugin/in_tail.rb +2 -2
- data/lib/fluent/plugin/out_http.rb +20 -2
- data/lib/fluent/plugin/output.rb +8 -5
- data/lib/fluent/plugin/parser_json.rb +5 -2
- data/lib/fluent/plugin_helper/cert_option.rb +5 -8
- data/lib/fluent/plugin_helper/child_process.rb +3 -2
- data/lib/fluent/plugin_helper/inject.rb +2 -1
- data/lib/fluent/plugin_helper/socket.rb +1 -1
- data/lib/fluent/supervisor.rb +11 -6
- data/lib/fluent/system_config.rb +2 -1
- data/lib/fluent/version.rb +1 -1
- data/test/command/test_binlog_reader.rb +22 -6
- data/test/plugin/test_buffer.rb +4 -0
- data/test/plugin/test_filter_stdout.rb +6 -1
- data/test/plugin/test_formatter_hash.rb +6 -3
- data/test/plugin/test_formatter_json.rb +14 -4
- data/test/plugin/test_formatter_ltsv.rb +13 -5
- data/test/plugin/test_formatter_out_file.rb +35 -14
- data/test/plugin/test_formatter_single_value.rb +12 -6
- data/test/plugin/test_formatter_tsv.rb +12 -4
- data/test/plugin/test_in_exec.rb +18 -0
- data/test/plugin/test_in_http.rb +57 -0
- data/test/plugin/{test_in_dummy.rb → test_in_sample.rb} +25 -25
- data/test/plugin/test_in_tail.rb +3 -0
- data/test/plugin/test_out_file.rb +23 -18
- data/test/plugin/test_output.rb +12 -0
- data/test/plugin_helper/data/cert/empty.pem +0 -0
- data/test/plugin_helper/test_cert_option.rb +7 -0
- data/test/plugin_helper/test_child_process.rb +15 -0
- data/test/plugin_helper/test_compat_parameters.rb +7 -2
- data/test/plugin_helper/test_http_server_helper.rb +5 -0
- data/test/plugin_helper/test_inject.rb +13 -0
- data/test/plugin_helper/test_server.rb +34 -0
- data/test/plugin_helper/test_socket.rb +8 -0
- data/test/test_formatter.rb +34 -10
- data/test/test_log.rb +44 -0
- data/test/test_match.rb +11 -0
- data/test/test_output.rb +6 -1
- data/test/test_static_config_analysis.rb +2 -2
- data/test/test_supervisor.rb +26 -0
- metadata +21 -18
@@ -25,6 +25,8 @@ module Fluent::Plugin
|
|
25
25
|
|
26
26
|
desc 'The command (program) to execute.'
|
27
27
|
config_param :command, :string
|
28
|
+
desc 'Specify connect mode to executed process'
|
29
|
+
config_param :connect_mode, :enum, list: [:read, :read_with_stderr], default: :read
|
28
30
|
|
29
31
|
config_section :parse do
|
30
32
|
config_set_default :@type, 'tsv'
|
@@ -72,9 +74,9 @@ module Fluent::Plugin
|
|
72
74
|
super
|
73
75
|
|
74
76
|
if @run_interval
|
75
|
-
child_process_execute(:exec_input, @command, interval: @run_interval, mode: [
|
77
|
+
child_process_execute(:exec_input, @command, interval: @run_interval, mode: [@connect_mode], &method(:run))
|
76
78
|
else
|
77
|
-
child_process_execute(:exec_input, @command, immediate: true, mode: [
|
79
|
+
child_process_execute(:exec_input, @command, immediate: true, mode: [@connect_mode], &method(:run))
|
78
80
|
end
|
79
81
|
end
|
80
82
|
|
@@ -27,10 +27,28 @@ require 'json'
|
|
27
27
|
module Fluent::Plugin
|
28
28
|
class InHttpParser < Parser
|
29
29
|
Fluent::Plugin.register_parser('in_http', self)
|
30
|
+
|
31
|
+
config_set_default :time_key, 'time'
|
32
|
+
|
33
|
+
def configure(conf)
|
34
|
+
super
|
35
|
+
|
36
|
+
# if no time parser related parameters, use in_http's time convert rule
|
37
|
+
@time_parser = if conf.has_key?('time_type') || conf.has_key?('time_format')
|
38
|
+
time_parser_create
|
39
|
+
else
|
40
|
+
nil
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
30
44
|
def parse(text)
|
31
45
|
# this plugin is dummy implementation not to raise error
|
32
46
|
yield nil, nil
|
33
47
|
end
|
48
|
+
|
49
|
+
def get_time_parser
|
50
|
+
@time_parser
|
51
|
+
end
|
34
52
|
end
|
35
53
|
|
36
54
|
class HttpInput < Input
|
@@ -54,12 +72,14 @@ module Fluent::Plugin
|
|
54
72
|
desc 'Add REMOTE_ADDR header to the record.'
|
55
73
|
config_param :add_remote_addr, :bool, default: false
|
56
74
|
config_param :blocking_timeout, :time, default: 0.5
|
57
|
-
desc 'Set a
|
75
|
+
desc 'Set a allow list of domains that can do CORS (Cross-Origin Resource Sharing)'
|
58
76
|
config_param :cors_allow_origins, :array, default: nil
|
59
77
|
desc 'Respond with empty gif image of 1x1 pixel.'
|
60
78
|
config_param :respond_with_empty_img, :bool, default: false
|
61
79
|
desc 'Respond status code with 204.'
|
62
80
|
config_param :use_204_response, :bool, default: false
|
81
|
+
desc 'Dump error log or not'
|
82
|
+
config_param :dump_error_log, :bool, default: true
|
63
83
|
|
64
84
|
config_section :parse do
|
65
85
|
config_set_default :@type, 'in_http'
|
@@ -67,6 +87,24 @@ module Fluent::Plugin
|
|
67
87
|
|
68
88
|
EVENT_RECORD_PARAMETER = '_event_record'
|
69
89
|
|
90
|
+
def initialize
|
91
|
+
super
|
92
|
+
|
93
|
+
@km = nil
|
94
|
+
@format_name = nil
|
95
|
+
@parser_time_key = nil
|
96
|
+
|
97
|
+
# default parsers
|
98
|
+
@parser_msgpack = nil
|
99
|
+
@parser_json = nil
|
100
|
+
@default_time_parser = nil
|
101
|
+
@default_keep_time_key = nil
|
102
|
+
@float_time_parser = nil
|
103
|
+
|
104
|
+
# <parse> configured parser
|
105
|
+
@custom_parser = nil
|
106
|
+
end
|
107
|
+
|
70
108
|
def configure(conf)
|
71
109
|
compat_parameters_convert(conf, :parser)
|
72
110
|
|
@@ -74,20 +112,22 @@ module Fluent::Plugin
|
|
74
112
|
|
75
113
|
m = if @parser_configs.first['@type'] == 'in_http'
|
76
114
|
@parser_msgpack = parser_create(usage: 'parser_in_http_msgpack', type: 'msgpack')
|
115
|
+
@parser_msgpack.time_key = nil
|
77
116
|
@parser_msgpack.estimate_current_event = false
|
78
117
|
@parser_json = parser_create(usage: 'parser_in_http_json', type: 'json')
|
118
|
+
@parser_json.time_key = nil
|
79
119
|
@parser_json.estimate_current_event = false
|
120
|
+
|
121
|
+
default_parser = parser_create(usage: '')
|
80
122
|
@format_name = 'default'
|
81
|
-
@parser_time_key =
|
82
|
-
|
83
|
-
|
84
|
-
'time'
|
85
|
-
end
|
123
|
+
@parser_time_key = default_parser.time_key
|
124
|
+
@default_time_parser = default_parser.get_time_parser
|
125
|
+
@default_keep_time_key = default_parser.keep_time_key
|
86
126
|
method(:parse_params_default)
|
87
127
|
else
|
88
|
-
@
|
128
|
+
@custom_parser = parser_create
|
89
129
|
@format_name = @parser_configs.first['@type']
|
90
|
-
@parser_time_key = @
|
130
|
+
@parser_time_key = @custom_parser.time_key
|
91
131
|
method(:parse_params_with_parser)
|
92
132
|
end
|
93
133
|
self.singleton_class.module_eval do
|
@@ -142,6 +182,13 @@ module Fluent::Plugin
|
|
142
182
|
super
|
143
183
|
end
|
144
184
|
|
185
|
+
RES_TEXT_HEADER = {'Content-Type' => 'text/plain'}.freeze
|
186
|
+
RESPONSE_200 = ["200 OK".freeze, RES_TEXT_HEADER, "".freeze].freeze
|
187
|
+
RESPONSE_204 = ["204 No Content".freeze, {}.freeze].freeze
|
188
|
+
RESPONSE_IMG = ["200 OK".freeze, {'Content-Type'=>'image/gif; charset=utf-8'}.freeze, EMPTY_GIF_IMAGE].freeze
|
189
|
+
RES_400_STATUS = "400 Bad Request".freeze
|
190
|
+
RES_500_STATUS = "500 Internal Server Error".freeze
|
191
|
+
|
145
192
|
def on_request(path_info, params)
|
146
193
|
begin
|
147
194
|
path = path_info[1..-1] # remove /
|
@@ -152,83 +199,77 @@ module Fluent::Plugin
|
|
152
199
|
if record.nil?
|
153
200
|
log.debug { "incoming event is invalid: path=#{path_info} params=#{params.to_json}" }
|
154
201
|
if @respond_with_empty_img
|
155
|
-
return
|
202
|
+
return RESPONSE_IMG
|
156
203
|
else
|
157
204
|
if @use_204_response
|
158
|
-
return
|
205
|
+
return RESPONSE_204
|
159
206
|
else
|
160
|
-
return
|
207
|
+
return RESPONSE_200
|
161
208
|
end
|
162
209
|
end
|
163
210
|
end
|
164
211
|
|
165
|
-
|
166
|
-
if @add_http_headers
|
167
|
-
params.each_pair { |k,v|
|
168
|
-
if k.start_with?("HTTP_")
|
169
|
-
record[k] = v
|
170
|
-
end
|
171
|
-
}
|
172
|
-
end
|
173
|
-
if @add_remote_addr
|
174
|
-
record['REMOTE_ADDR'] = params['REMOTE_ADDR']
|
175
|
-
end
|
176
|
-
end
|
177
|
-
time = if param_time = params['time']
|
178
|
-
param_time = param_time.to_f
|
179
|
-
param_time.zero? ? Fluent::EventTime.now : @float_time_parser.parse(param_time)
|
180
|
-
else
|
181
|
-
record_time.nil? ? Fluent::EventTime.now : record_time
|
182
|
-
end
|
183
|
-
rescue
|
184
|
-
return ["400 Bad Request", {'Content-Type'=>'text/plain'}, "400 Bad Request\n#{$!}\n"]
|
185
|
-
end
|
186
|
-
|
187
|
-
# TODO server error
|
188
|
-
begin
|
212
|
+
mes = nil
|
189
213
|
# Support batched requests
|
190
214
|
if record.is_a?(Array)
|
191
215
|
mes = Fluent::MultiEventStream.new
|
192
216
|
record.each do |single_record|
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
single_record['REMOTE_ADDR'] = params['REMOTE_ADDR']
|
202
|
-
end
|
203
|
-
|
204
|
-
if defined? @parser
|
205
|
-
single_time = @parser.parse_time(single_record)
|
206
|
-
single_time, single_record = @parser.convert_values(single_time, single_record)
|
217
|
+
add_params_to_record(single_record, params)
|
218
|
+
|
219
|
+
if param_time = params['time']
|
220
|
+
param_time = param_time.to_f
|
221
|
+
single_time = param_time.zero? ? Fluent::EventTime.now : @float_time_parser.parse(param_time)
|
222
|
+
elsif @custom_parser
|
223
|
+
single_time = @custom_parser.parse_time(single_record)
|
224
|
+
single_time, single_record = @custom_parser.convert_values(single_time, single_record)
|
207
225
|
else
|
208
|
-
single_time =
|
209
|
-
Fluent::EventTime.from_time(Time.at(t))
|
210
|
-
else
|
211
|
-
time
|
212
|
-
end
|
226
|
+
single_time = convert_time_field(single_record)
|
213
227
|
end
|
214
228
|
|
215
229
|
mes.add(single_time, single_record)
|
216
230
|
end
|
231
|
+
else
|
232
|
+
add_params_to_record(record, params)
|
233
|
+
|
234
|
+
time = if param_time = params['time']
|
235
|
+
param_time = param_time.to_f
|
236
|
+
param_time.zero? ? Fluent::EventTime.now : @float_time_parser.parse(param_time)
|
237
|
+
else
|
238
|
+
if record_time.nil?
|
239
|
+
convert_time_field(record)
|
240
|
+
else
|
241
|
+
record_time
|
242
|
+
end
|
243
|
+
end
|
244
|
+
end
|
245
|
+
rescue => e
|
246
|
+
if @dump_error_log
|
247
|
+
log.error "failed to process request", error: e
|
248
|
+
end
|
249
|
+
return [RES_400_STATUS, RES_TEXT_HEADER, "400 Bad Request\n#{e}\n"]
|
250
|
+
end
|
251
|
+
|
252
|
+
# TODO server error
|
253
|
+
begin
|
254
|
+
if mes
|
217
255
|
router.emit_stream(tag, mes)
|
218
256
|
else
|
219
257
|
router.emit(tag, time, record)
|
220
258
|
end
|
221
|
-
rescue
|
222
|
-
|
259
|
+
rescue => e
|
260
|
+
if @dump_error_log
|
261
|
+
log.error "failed to emit data", error: e
|
262
|
+
end
|
263
|
+
return [RES_500_STATUS, RES_TEXT_HEADER, "500 Internal Server Error\n#{e}\n"]
|
223
264
|
end
|
224
265
|
|
225
266
|
if @respond_with_empty_img
|
226
|
-
return
|
267
|
+
return RESPONSE_IMG
|
227
268
|
else
|
228
269
|
if @use_204_response
|
229
|
-
return
|
270
|
+
return RESPONSE_204
|
230
271
|
else
|
231
|
-
return
|
272
|
+
return RESPONSE_200
|
232
273
|
end
|
233
274
|
end
|
234
275
|
end
|
@@ -267,7 +308,7 @@ module Fluent::Plugin
|
|
267
308
|
|
268
309
|
def parse_params_with_parser(params)
|
269
310
|
if content = params[EVENT_RECORD_PARAMETER]
|
270
|
-
@
|
311
|
+
@custom_parser.parse(content) { |time, record|
|
271
312
|
raise "Received event is not #{@format_name}: #{content}" if record.nil?
|
272
313
|
return time, record
|
273
314
|
}
|
@@ -276,6 +317,32 @@ module Fluent::Plugin
|
|
276
317
|
end
|
277
318
|
end
|
278
319
|
|
320
|
+
def add_params_to_record(record, params)
|
321
|
+
if @add_http_headers
|
322
|
+
params.each_pair { |k, v|
|
323
|
+
if k.start_with?("HTTP_".freeze)
|
324
|
+
record[k] = v
|
325
|
+
end
|
326
|
+
}
|
327
|
+
end
|
328
|
+
|
329
|
+
if @add_remote_addr
|
330
|
+
record['REMOTE_ADDR'] = params['REMOTE_ADDR']
|
331
|
+
end
|
332
|
+
end
|
333
|
+
|
334
|
+
def convert_time_field(record)
|
335
|
+
if t = @default_keep_time_key ? record[@parser_time_key] : record.delete(@parser_time_key)
|
336
|
+
if @default_time_parser
|
337
|
+
@default_time_parser.parse(t)
|
338
|
+
else
|
339
|
+
Fluent::EventTime.from_time(Time.at(t))
|
340
|
+
end
|
341
|
+
else
|
342
|
+
Fluent::EventTime.now
|
343
|
+
end
|
344
|
+
end
|
345
|
+
|
279
346
|
class Handler
|
280
347
|
attr_reader :content_type
|
281
348
|
|
@@ -358,7 +425,7 @@ module Fluent::Plugin
|
|
358
425
|
end
|
359
426
|
}
|
360
427
|
if expect
|
361
|
-
if expect == '100-continue'
|
428
|
+
if expect == '100-continue'.freeze
|
362
429
|
if !size || size < @body_size_limit
|
363
430
|
send_response_nobody("100 Continue", {})
|
364
431
|
else
|
@@ -380,17 +447,20 @@ module Fluent::Plugin
|
|
380
447
|
@body << chunk
|
381
448
|
end
|
382
449
|
|
450
|
+
RES_200_STATUS = "200 OK".freeze
|
451
|
+
RES_403_STATUS = "403 Forbidden".freeze
|
452
|
+
|
383
453
|
# Web browsers can send an OPTIONS request before performing POST
|
384
454
|
# to check if cross-origin requests are supported.
|
385
455
|
def handle_options_request
|
386
456
|
# Is CORS enabled in the first place?
|
387
457
|
if @cors_allow_origins.nil?
|
388
|
-
return send_response_and_close(
|
458
|
+
return send_response_and_close(RES_403_STATUS, {}, "")
|
389
459
|
end
|
390
460
|
|
391
461
|
# in_http does not support HTTP methods except POST
|
392
462
|
if @access_control_request_method != 'POST'
|
393
|
-
return send_response_and_close(
|
463
|
+
return send_response_and_close(RES_403_STATUS, {}, "")
|
394
464
|
end
|
395
465
|
|
396
466
|
header = {
|
@@ -401,29 +471,29 @@ module Fluent::Plugin
|
|
401
471
|
# Check the origin and send back a CORS response
|
402
472
|
if @cors_allow_origins.include?('*')
|
403
473
|
header["Access-Control-Allow-Origin"] = "*"
|
404
|
-
send_response_and_close(
|
474
|
+
send_response_and_close(RES_200_STATUS, header, "")
|
405
475
|
elsif include_cors_allow_origin
|
406
476
|
header["Access-Control-Allow-Origin"] = @origin
|
407
|
-
send_response_and_close(
|
477
|
+
send_response_and_close(RES_200_STATUS, header, "")
|
408
478
|
else
|
409
|
-
send_response_and_close(
|
479
|
+
send_response_and_close(RES_403_STATUS, {}, "")
|
410
480
|
end
|
411
481
|
end
|
412
482
|
|
413
483
|
def on_message_complete
|
414
484
|
return if closing?
|
415
485
|
|
416
|
-
if @parser.http_method == 'OPTIONS'
|
486
|
+
if @parser.http_method == 'OPTIONS'.freeze
|
417
487
|
return handle_options_request()
|
418
488
|
end
|
419
489
|
|
420
490
|
# CORS check
|
421
491
|
# ==========
|
422
492
|
# For every incoming request, we check if we have some CORS
|
423
|
-
# restrictions and
|
493
|
+
# restrictions and allow listed origins through @cors_allow_origins.
|
424
494
|
unless @cors_allow_origins.nil?
|
425
495
|
unless @cors_allow_origins.include?('*') or include_cors_allow_origin
|
426
|
-
send_response_and_close(
|
496
|
+
send_response_and_close(RES_403_STATUS, {'Connection' => 'close'}, "")
|
427
497
|
return
|
428
498
|
end
|
429
499
|
end
|
@@ -433,14 +503,14 @@ module Fluent::Plugin
|
|
433
503
|
# Decode payload according to the "Content-Encoding" header.
|
434
504
|
# For now, we only support 'gzip' and 'deflate'.
|
435
505
|
begin
|
436
|
-
if @content_encoding == 'gzip'
|
506
|
+
if @content_encoding == 'gzip'.freeze
|
437
507
|
@body = Zlib::GzipReader.new(StringIO.new(@body)).read
|
438
|
-
elsif @content_encoding == 'deflate'
|
508
|
+
elsif @content_encoding == 'deflate'.freeze
|
439
509
|
@body = Zlib::Inflate.inflate(@body)
|
440
510
|
end
|
441
511
|
rescue
|
442
512
|
@log.warn 'fails to decode payload', error: $!.to_s
|
443
|
-
send_response_and_close(
|
513
|
+
send_response_and_close(RES_400_STATUS, {}, "")
|
444
514
|
return
|
445
515
|
end
|
446
516
|
|
@@ -466,8 +536,9 @@ module Fluent::Plugin
|
|
466
536
|
params.merge!(@env)
|
467
537
|
@env.clear
|
468
538
|
|
469
|
-
code, header, body =
|
539
|
+
code, header, body = @callback.call(path_info, params)
|
470
540
|
body = body.to_s
|
541
|
+
header = header.dup if header.frozen?
|
471
542
|
|
472
543
|
unless @cors_allow_origins.nil?
|
473
544
|
if @cors_allow_origins.include?('*')
|
@@ -478,7 +549,7 @@ module Fluent::Plugin
|
|
478
549
|
end
|
479
550
|
|
480
551
|
if @keep_alive
|
481
|
-
header['Connection'] = 'Keep-Alive'
|
552
|
+
header['Connection'] = 'Keep-Alive'.freeze
|
482
553
|
send_response(code, header, body)
|
483
554
|
else
|
484
555
|
send_response_and_close(code, header, body)
|
@@ -504,13 +575,13 @@ module Fluent::Plugin
|
|
504
575
|
|
505
576
|
def send_response(code, header, body)
|
506
577
|
header['Content-Length'] ||= body.bytesize
|
507
|
-
header['Content-Type'] ||= 'text/plain'
|
578
|
+
header['Content-Type'] ||= 'text/plain'.freeze
|
508
579
|
|
509
580
|
data = %[HTTP/1.1 #{code}\r\n]
|
510
581
|
header.each_pair {|k,v|
|
511
582
|
data << "#{k}: #{v}\r\n"
|
512
583
|
}
|
513
|
-
data << "\r\n"
|
584
|
+
data << "\r\n".freeze
|
514
585
|
@io.write(data)
|
515
586
|
|
516
587
|
@io.write(body)
|
@@ -521,7 +592,7 @@ module Fluent::Plugin
|
|
521
592
|
header.each_pair {|k,v|
|
522
593
|
data << "#{k}: #{v}\r\n"
|
523
594
|
}
|
524
|
-
data << "\r\n"
|
595
|
+
data << "\r\n".freeze
|
525
596
|
@io.write(data)
|
526
597
|
end
|
527
598
|
|
@@ -0,0 +1,141 @@
|
|
1
|
+
#
|
2
|
+
# Fluentd
|
3
|
+
#
|
4
|
+
# Licensed under the Apache License, Version 2.0 (the "License");
|
5
|
+
# you may not use this file except in compliance with the License.
|
6
|
+
# You may obtain a copy of the License at
|
7
|
+
#
|
8
|
+
# http://www.apache.org/licenses/LICENSE-2.0
|
9
|
+
#
|
10
|
+
# Unless required by applicable law or agreed to in writing, software
|
11
|
+
# distributed under the License is distributed on an "AS IS" BASIS,
|
12
|
+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
13
|
+
# See the License for the specific language governing permissions and
|
14
|
+
# limitations under the License.
|
15
|
+
#
|
16
|
+
|
17
|
+
require 'json'
|
18
|
+
|
19
|
+
require 'fluent/plugin/input'
|
20
|
+
require 'fluent/config/error'
|
21
|
+
|
22
|
+
module Fluent::Plugin
|
23
|
+
class SampleInput < Input
|
24
|
+
Fluent::Plugin.register_input('sample', self)
|
25
|
+
Fluent::Plugin.register_input('dummy', self)
|
26
|
+
|
27
|
+
helpers :thread, :storage
|
28
|
+
|
29
|
+
BIN_NUM = 10
|
30
|
+
DEFAULT_STORAGE_TYPE = 'local'
|
31
|
+
|
32
|
+
desc "The value is the tag assigned to the generated events."
|
33
|
+
config_param :tag, :string
|
34
|
+
desc "The number of events in event stream of each emits."
|
35
|
+
config_param :size, :integer, default: 1
|
36
|
+
desc "It configures how many events to generate per second."
|
37
|
+
config_param :rate, :integer, default: 1
|
38
|
+
desc "If specified, each generated event has an auto-incremented key field."
|
39
|
+
config_param :auto_increment_key, :string, default: nil
|
40
|
+
desc "The boolean to suspend-and-resume incremental value after restart"
|
41
|
+
config_param :suspend, :bool, default: false,deprecated: 'This parameters is ignored'
|
42
|
+
desc "The sample data to be generated. An array of JSON hashes or a single JSON hash."
|
43
|
+
config_param :sample, alias: :dummy, default: [{"message" => "sample"}] do |val|
|
44
|
+
begin
|
45
|
+
parsed = JSON.parse(val)
|
46
|
+
rescue JSON::ParserError => ex
|
47
|
+
# Fluent::ConfigParseError, "got incomplete JSON" will be raised
|
48
|
+
# at literal_parser.rb with --use-v1-config, but I had to
|
49
|
+
# take care at here for the case of --use-v0-config.
|
50
|
+
raise Fluent::ConfigError, "#{ex.class}: #{ex.message}"
|
51
|
+
end
|
52
|
+
sample = parsed.is_a?(Array) ? parsed : [parsed]
|
53
|
+
sample.each_with_index do |e, i|
|
54
|
+
raise Fluent::ConfigError, "#{i}th element of sample, #{e}, is not a hash" unless e.is_a?(Hash)
|
55
|
+
end
|
56
|
+
sample
|
57
|
+
end
|
58
|
+
|
59
|
+
def initialize
|
60
|
+
super
|
61
|
+
@storage = nil
|
62
|
+
end
|
63
|
+
|
64
|
+
def configure(conf)
|
65
|
+
super
|
66
|
+
@sample_index = 0
|
67
|
+
config = conf.elements.select{|e| e.name == 'storage' }.first
|
68
|
+
@storage = storage_create(usage: 'suspend', conf: config, default_type: DEFAULT_STORAGE_TYPE)
|
69
|
+
end
|
70
|
+
|
71
|
+
def multi_workers_ready?
|
72
|
+
true
|
73
|
+
end
|
74
|
+
|
75
|
+
def start
|
76
|
+
super
|
77
|
+
|
78
|
+
@storage.put(:increment_value, 0) unless @storage.get(:increment_value)
|
79
|
+
# keep 'dummy' to avoid breaking changes for existing environment. Change it in fluentd v2
|
80
|
+
@storage.put(:dummy_index, 0) unless @storage.get(:dummy_index)
|
81
|
+
|
82
|
+
if @auto_increment_key && !@storage.get(:auto_increment_value)
|
83
|
+
@storage.put(:auto_increment_value, -1)
|
84
|
+
end
|
85
|
+
|
86
|
+
thread_create(:sample_input, &method(:run))
|
87
|
+
end
|
88
|
+
|
89
|
+
def run
|
90
|
+
batch_num = (@rate / BIN_NUM).to_i
|
91
|
+
residual_num = (@rate % BIN_NUM)
|
92
|
+
while thread_current_running?
|
93
|
+
current_time = Time.now.to_i
|
94
|
+
BIN_NUM.times do
|
95
|
+
break unless (thread_current_running? && Time.now.to_i <= current_time)
|
96
|
+
wait(0.1) { emit(batch_num) }
|
97
|
+
end
|
98
|
+
emit(residual_num) if thread_current_running?
|
99
|
+
# wait for next second
|
100
|
+
while thread_current_running? && Time.now.to_i <= current_time
|
101
|
+
sleep 0.01
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
105
|
+
|
106
|
+
def emit(num)
|
107
|
+
begin
|
108
|
+
if @size > 1
|
109
|
+
num.times do
|
110
|
+
router.emit_array(@tag, Array.new(@size) { [Fluent::EventTime.now, generate] })
|
111
|
+
end
|
112
|
+
else
|
113
|
+
num.times { router.emit(@tag, Fluent::EventTime.now, generate) }
|
114
|
+
end
|
115
|
+
rescue => _
|
116
|
+
# ignore all errors not to stop emits by emit errors
|
117
|
+
end
|
118
|
+
end
|
119
|
+
|
120
|
+
def generate
|
121
|
+
d = @sample[@sample_index]
|
122
|
+
unless d
|
123
|
+
@sample_index = 0
|
124
|
+
d = @sample[@sample_index]
|
125
|
+
end
|
126
|
+
@sample_index += 1
|
127
|
+
if @auto_increment_key
|
128
|
+
d = d.dup
|
129
|
+
d[@auto_increment_key] = @storage.update(:auto_increment_value){|v| v + 1 }
|
130
|
+
end
|
131
|
+
d
|
132
|
+
end
|
133
|
+
|
134
|
+
def wait(time)
|
135
|
+
start_time = Time.now
|
136
|
+
yield
|
137
|
+
sleep_time = time - (Time.now - start_time)
|
138
|
+
sleep sleep_time if sleep_time > 0
|
139
|
+
end
|
140
|
+
end
|
141
|
+
end
|