fluentd 1.10.0-x64-mingw32 → 1.11.0-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/CHANGELOG.md +117 -1
- data/CONTRIBUTING.md +1 -1
- data/README.md +4 -0
- data/docs/SECURITY_AUDIT.pdf +0 -0
- data/lib/fluent/command/debug.rb +1 -0
- data/lib/fluent/command/fluentd.rb +14 -1
- data/lib/fluent/config.rb +1 -0
- data/lib/fluent/daemonizer.rb +88 -0
- data/lib/fluent/log.rb +45 -6
- data/lib/fluent/match.rb +1 -1
- data/lib/fluent/plugin/in_dummy.rb +2 -2
- data/lib/fluent/plugin/in_forward.rb +2 -2
- data/lib/fluent/plugin/in_gc_stat.rb +16 -0
- data/lib/fluent/plugin/in_http.rb +2 -2
- data/lib/fluent/plugin/in_monitor_agent.rb +1 -1
- data/lib/fluent/plugin/in_syslog.rb +4 -4
- data/lib/fluent/plugin/in_tail.rb +3 -3
- data/lib/fluent/plugin/in_tail/position_file.rb +23 -6
- data/lib/fluent/plugin/in_unix.rb +77 -77
- data/lib/fluent/plugin/out_copy.rb +1 -1
- data/lib/fluent/plugin/out_file.rb +1 -1
- data/lib/fluent/plugin/out_forward.rb +23 -18
- data/lib/fluent/plugin/out_forward/load_balancer.rb +1 -1
- data/lib/fluent/plugin/out_http.rb +15 -2
- data/lib/fluent/plugin/parser_multiline.rb +1 -1
- data/lib/fluent/plugin/parser_syslog.rb +216 -55
- data/lib/fluent/plugin_helper/record_accessor.rb +14 -0
- data/lib/fluent/plugin_helper/service_discovery.rb +7 -0
- data/lib/fluent/plugin_helper/service_discovery/manager.rb +8 -0
- data/lib/fluent/plugin_helper/socket.rb +20 -2
- data/lib/fluent/plugin_helper/socket_option.rb +2 -2
- data/lib/fluent/supervisor.rb +21 -9
- data/lib/fluent/system_config.rb +2 -1
- data/lib/fluent/test/filter_test.rb +2 -2
- data/lib/fluent/test/output_test.rb +3 -3
- data/lib/fluent/version.rb +1 -1
- data/test/command/test_fluentd.rb +57 -10
- data/test/config/test_system_config.rb +2 -0
- data/test/plugin/in_tail/test_position_file.rb +24 -0
- data/test/plugin/out_forward/test_load_balancer.rb +46 -0
- data/test/plugin/test_in_gc_stat.rb +24 -1
- data/test/plugin/test_in_syslog.rb +16 -1
- data/test/plugin/test_in_tail.rb +39 -16
- data/test/plugin/test_in_unix.rb +128 -73
- data/test/plugin/test_out_forward.rb +11 -2
- data/test/plugin/test_out_http.rb +38 -0
- data/test/plugin/test_out_null.rb +1 -1
- data/test/plugin/test_output_as_buffered_retries.rb +12 -4
- data/test/plugin/test_output_as_buffered_secondary.rb +9 -1
- data/test/plugin/test_parser_syslog.rb +77 -29
- data/test/plugin_helper/data/cert/cert_chains/ca-cert-key.pem +27 -0
- data/test/plugin_helper/data/cert/cert_chains/ca-cert.pem +20 -0
- data/test/plugin_helper/data/cert/cert_chains/cert-key.pem +27 -0
- data/test/plugin_helper/data/cert/cert_chains/cert.pem +40 -0
- data/test/plugin_helper/data/cert/generate_cert.rb +38 -0
- data/test/plugin_helper/http_server/test_app.rb +1 -1
- data/test/plugin_helper/http_server/test_route.rb +1 -1
- data/test/plugin_helper/test_http_server_helper.rb +2 -2
- data/test/plugin_helper/test_record_accessor.rb +41 -0
- data/test/plugin_helper/test_server.rb +1 -1
- data/test/plugin_helper/test_service_discovery.rb +37 -4
- data/test/plugin_helper/test_socket.rb +131 -0
- data/test/test_daemonizer.rb +91 -0
- data/test/test_log.rb +44 -0
- metadata +16 -2
@@ -105,10 +105,10 @@ module Fluent::Plugin
|
|
105
105
|
begin
|
106
106
|
if @size > 1
|
107
107
|
num.times do
|
108
|
-
router.emit_array(@tag, Array.new(@size) { [Fluent::
|
108
|
+
router.emit_array(@tag, Array.new(@size) { [Fluent::EventTime.now, generate] })
|
109
109
|
end
|
110
110
|
else
|
111
|
-
num.times { router.emit(@tag, Fluent::
|
111
|
+
num.times { router.emit(@tag, Fluent::EventTime.now, generate) }
|
112
112
|
end
|
113
113
|
rescue => _
|
114
114
|
# ignore all errors not to stop emits by emit errors
|
@@ -327,7 +327,7 @@ module Fluent::Plugin
|
|
327
327
|
record = e[1]
|
328
328
|
next if record.nil?
|
329
329
|
time = e[0]
|
330
|
-
time = Fluent::
|
330
|
+
time = Fluent::EventTime.now if time.nil? || time.to_i == 0 # `to_i == 0` for empty EventTime
|
331
331
|
es.add(time, record)
|
332
332
|
}
|
333
333
|
es
|
@@ -347,7 +347,7 @@ module Fluent::Plugin
|
|
347
347
|
return msg[3] # retry never succeeded so return ack and drop incoming event.
|
348
348
|
end
|
349
349
|
return if record.nil?
|
350
|
-
time = Fluent::
|
350
|
+
time = Fluent::EventTime.now if time.to_i == 0
|
351
351
|
if @enable_field_injection
|
352
352
|
record[@source_address_key] = conn.remote_addr if @source_address_key
|
353
353
|
record[@source_hostname_key] = conn.remote_host if @source_hostname_key
|
@@ -24,13 +24,22 @@ module Fluent::Plugin
|
|
24
24
|
|
25
25
|
def initialize
|
26
26
|
super
|
27
|
+
@key_map = nil
|
27
28
|
end
|
28
29
|
|
29
30
|
config_param :emit_interval, :time, default: 60
|
31
|
+
config_param :use_symbol_keys, :bool, default: true
|
30
32
|
config_param :tag, :string
|
31
33
|
|
32
34
|
def configure(conf)
|
33
35
|
super
|
36
|
+
|
37
|
+
unless @use_symbol_keys
|
38
|
+
@key_map = {}
|
39
|
+
GC.stat.each_key { |key|
|
40
|
+
@key_map[key] = key.to_s
|
41
|
+
}
|
42
|
+
end
|
34
43
|
end
|
35
44
|
|
36
45
|
def multi_workers_ready?
|
@@ -50,6 +59,13 @@ module Fluent::Plugin
|
|
50
59
|
def on_timer
|
51
60
|
now = Fluent::EventTime.now
|
52
61
|
record = GC.stat
|
62
|
+
unless @use_symbol_keys
|
63
|
+
new_record = {}
|
64
|
+
record.each_pair { |k, v|
|
65
|
+
new_record[@key_map[k]] = v
|
66
|
+
}
|
67
|
+
record = new_record
|
68
|
+
end
|
53
69
|
router.emit(@tag, now, record)
|
54
70
|
end
|
55
71
|
end
|
@@ -176,9 +176,9 @@ module Fluent::Plugin
|
|
176
176
|
end
|
177
177
|
time = if param_time = params['time']
|
178
178
|
param_time = param_time.to_f
|
179
|
-
param_time.zero? ? Fluent::
|
179
|
+
param_time.zero? ? Fluent::EventTime.now : @float_time_parser.parse(param_time)
|
180
180
|
else
|
181
|
-
record_time.nil? ? Fluent::
|
181
|
+
record_time.nil? ? Fluent::EventTime.now : record_time
|
182
182
|
end
|
183
183
|
rescue
|
184
184
|
return ["400 Bad Request", {'Content-Type'=>'text/plain'}, "400 Bad Request\n#{$!}\n"]
|
@@ -223,7 +223,7 @@ module Fluent::Plugin
|
|
223
223
|
opts = {with_config: false, with_retry: false}
|
224
224
|
timer_execute(:in_monitor_agent_emit, @emit_interval, repeat: true) {
|
225
225
|
es = Fluent::MultiEventStream.new
|
226
|
-
now = Fluent::
|
226
|
+
now = Fluent::EventTime.now
|
227
227
|
plugins_info_all(opts).each { |record|
|
228
228
|
es.add(now, record)
|
229
229
|
}
|
@@ -181,12 +181,12 @@ module Fluent::Plugin
|
|
181
181
|
if octet_count_frame
|
182
182
|
while idx = buffer.index(delimiter, pos)
|
183
183
|
num = Integer(buffer[pos..idx])
|
184
|
-
|
185
|
-
msg
|
186
|
-
if msg.size < num - 1
|
187
|
-
pos = pos - num - num.to_s.size
|
184
|
+
msg = buffer[idx + delimiter_size, num]
|
185
|
+
if msg.size != num
|
188
186
|
break
|
189
187
|
end
|
188
|
+
|
189
|
+
pos = idx + delimiter_size + num
|
190
190
|
message_handler(msg, conn)
|
191
191
|
end
|
192
192
|
else
|
@@ -70,7 +70,7 @@ module Fluent::Plugin
|
|
70
70
|
desc 'Fluentd will record the position it last read into this file.'
|
71
71
|
config_param :pos_file, :string, default: nil
|
72
72
|
desc 'The cleanup interval of pos file'
|
73
|
-
config_param :pos_file_compaction_interval, :
|
73
|
+
config_param :pos_file_compaction_interval, :time, default: nil
|
74
74
|
desc 'Start to read the logs from the head of file, not bottom.'
|
75
75
|
config_param :read_from_head, :bool, default: false
|
76
76
|
# When the program deletes log file and re-creates log file with same filename after passed refresh_interval,
|
@@ -136,7 +136,7 @@ module Fluent::Plugin
|
|
136
136
|
raise Fluent::ConfigError, "#{rc} are reserved words: #{@path_delimiter}"
|
137
137
|
end
|
138
138
|
|
139
|
-
@paths = @path.split(@path_delimiter).map(&:strip)
|
139
|
+
@paths = @path.split(@path_delimiter).map(&:strip).uniq
|
140
140
|
if @paths.empty?
|
141
141
|
raise Fluent::ConfigError, "tail: 'path' parameter is required on tail input"
|
142
142
|
end
|
@@ -296,7 +296,7 @@ module Fluent::Plugin
|
|
296
296
|
end
|
297
297
|
path.include?('*') ? Dir.glob(path) : path
|
298
298
|
}.flatten.uniq
|
299
|
-
paths - excluded
|
299
|
+
paths.uniq - excluded
|
300
300
|
end
|
301
301
|
|
302
302
|
# in_tail with '*' path doesn't check rotation file equality at refresh phase.
|
@@ -21,7 +21,6 @@ module Fluent::Plugin
|
|
21
21
|
class PositionFile
|
22
22
|
UNWATCHED_POSITION = 0xffffffffffffffff
|
23
23
|
POSITION_FILE_ENTRY_REGEX = /^([^\t]+)\t([0-9a-fA-F]+)\t([0-9a-fA-F]+)/.freeze
|
24
|
-
POSITION_FILE_ENTRY_FORMAT = "%s\t%016x\t%016x\n".freeze
|
25
24
|
|
26
25
|
def self.load(file, logger:)
|
27
26
|
pf = new(file, logger: logger)
|
@@ -83,8 +82,8 @@ module Fluent::Plugin
|
|
83
82
|
size = nil
|
84
83
|
|
85
84
|
@file_mutex.synchronize do
|
86
|
-
last_modified = @file.mtime
|
87
85
|
size = @file.size
|
86
|
+
last_modified = @file.mtime
|
88
87
|
end
|
89
88
|
|
90
89
|
entries = fetch_compacted_entries
|
@@ -93,7 +92,13 @@ module Fluent::Plugin
|
|
93
92
|
if last_modified == @file.mtime && size == @file.size
|
94
93
|
@file.pos = 0
|
95
94
|
@file.truncate(0)
|
96
|
-
@file.write(entries.join)
|
95
|
+
@file.write(entries.values.map(&:to_entry_fmt).join)
|
96
|
+
|
97
|
+
entries.each do |path, val|
|
98
|
+
if (m = @map[path])
|
99
|
+
m.seek = val.seek
|
100
|
+
end
|
101
|
+
end
|
97
102
|
else
|
98
103
|
# skip
|
99
104
|
end
|
@@ -104,7 +109,7 @@ module Fluent::Plugin
|
|
104
109
|
|
105
110
|
def compact
|
106
111
|
@file_mutex.synchronize do
|
107
|
-
entries = fetch_compacted_entries
|
112
|
+
entries = fetch_compacted_entries.values.map(&:to_entry_fmt)
|
108
113
|
|
109
114
|
@file.pos = 0
|
110
115
|
@file.truncate(0)
|
@@ -116,6 +121,7 @@ module Fluent::Plugin
|
|
116
121
|
entries = {}
|
117
122
|
|
118
123
|
@file.pos = 0
|
124
|
+
file_pos = 0
|
119
125
|
@file.each_line do |line|
|
120
126
|
m = POSITION_FILE_ENTRY_REGEX.match(line)
|
121
127
|
if m.nil?
|
@@ -133,11 +139,20 @@ module Fluent::Plugin
|
|
133
139
|
@logger.warn("#{path} already exists. use latest one: deleted #{entries[path]}") if @logger
|
134
140
|
end
|
135
141
|
|
136
|
-
entries[path] = (
|
142
|
+
entries[path] = Entry.new(path, pos, ino, file_pos + path.size + 1)
|
143
|
+
file_pos += line.size
|
137
144
|
end
|
138
145
|
end
|
139
146
|
|
140
|
-
entries
|
147
|
+
entries
|
148
|
+
end
|
149
|
+
end
|
150
|
+
|
151
|
+
Entry = Struct.new(:path, :pos, :ino, :seek) do
|
152
|
+
POSITION_FILE_ENTRY_FORMAT = "%s\t%016x\t%016x\n".freeze
|
153
|
+
|
154
|
+
def to_entry_fmt
|
155
|
+
POSITION_FILE_ENTRY_FORMAT % [path, pos, ino]
|
141
156
|
end
|
142
157
|
end
|
143
158
|
|
@@ -158,6 +173,8 @@ module Fluent::Plugin
|
|
158
173
|
@inode = inode
|
159
174
|
end
|
160
175
|
|
176
|
+
attr_writer :seek
|
177
|
+
|
161
178
|
def update(ino, pos)
|
162
179
|
@file_mutex.synchronize {
|
163
180
|
@file.pos = @seek
|
@@ -14,50 +14,68 @@
|
|
14
14
|
# limitations under the License.
|
15
15
|
#
|
16
16
|
|
17
|
-
require '
|
18
|
-
require '
|
17
|
+
require 'fluent/env'
|
18
|
+
require 'fluent/plugin/input'
|
19
|
+
require 'fluent/msgpack_factory'
|
19
20
|
|
20
21
|
require 'cool.io'
|
21
22
|
require 'yajl'
|
23
|
+
require 'fileutils'
|
24
|
+
require 'socket'
|
25
|
+
|
26
|
+
module Fluent::Plugin
|
27
|
+
# TODO: This plugin will be 3rd party plugin
|
28
|
+
class UnixInput < Input
|
29
|
+
Fluent::Plugin.register_input('unix', self)
|
30
|
+
|
31
|
+
helpers :event_loop
|
32
|
+
|
33
|
+
def initialize
|
34
|
+
super
|
22
35
|
|
23
|
-
|
24
|
-
|
36
|
+
@lsock = nil
|
37
|
+
end
|
25
38
|
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
config_param :
|
39
|
+
desc 'The path to your Unix Domain Socket.'
|
40
|
+
config_param :path, :string, default: Fluent::DEFAULT_SOCKET_PATH
|
41
|
+
desc 'The backlog of Unix Domain Socket.'
|
42
|
+
config_param :backlog, :integer, default: nil
|
43
|
+
desc "New tag instead of incoming tag"
|
44
|
+
config_param :tag, :string, default: nil
|
45
|
+
|
46
|
+
def configure(conf)
|
47
|
+
super
|
48
|
+
end
|
30
49
|
|
31
50
|
def start
|
32
51
|
super
|
33
52
|
|
34
|
-
@loop = Coolio::Loop.new
|
35
53
|
@lsock = listen
|
36
|
-
|
37
|
-
@thread = Thread.new(&method(:run))
|
54
|
+
event_loop_attach(@lsock)
|
38
55
|
end
|
39
56
|
|
40
57
|
def shutdown
|
41
|
-
@
|
42
|
-
|
43
|
-
|
44
|
-
|
58
|
+
if @lsock
|
59
|
+
event_loop_detach(@lsock)
|
60
|
+
@lsock.close
|
61
|
+
end
|
45
62
|
|
46
63
|
super
|
47
64
|
end
|
48
65
|
|
49
|
-
|
50
|
-
|
66
|
+
def listen
|
67
|
+
if File.exist?(@path)
|
68
|
+
log.warn "Found existing '#{@path}'. Remove this file for in_unix plugin"
|
69
|
+
File.unlink(@path)
|
70
|
+
end
|
71
|
+
FileUtils.mkdir_p(File.dirname(@path))
|
51
72
|
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
log.error_backtrace
|
73
|
+
log.info "listening fluent socket on #{@path}"
|
74
|
+
s = Coolio::UNIXServer.new(@path, Handler, log, method(:on_message))
|
75
|
+
s.listen(@backlog) unless @backlog.nil?
|
76
|
+
s
|
57
77
|
end
|
58
78
|
|
59
|
-
private
|
60
|
-
|
61
79
|
# message Entry {
|
62
80
|
# 1: long time
|
63
81
|
# 2: object record
|
@@ -79,23 +97,27 @@ module Fluent
|
|
79
97
|
# 3: object record
|
80
98
|
# }
|
81
99
|
def on_message(msg)
|
82
|
-
|
83
|
-
|
100
|
+
unless msg.is_a?(Array)
|
101
|
+
log.warn "incoming data is broken:", msg: msg
|
102
|
+
return
|
103
|
+
end
|
104
|
+
|
105
|
+
tag = @tag || (msg[0].to_s)
|
84
106
|
entries = msg[1]
|
85
107
|
|
86
|
-
|
108
|
+
case entries
|
109
|
+
when String
|
87
110
|
# PackedForward
|
88
|
-
es = MessagePackEventStream.new(entries)
|
111
|
+
es = Fluent::MessagePackEventStream.new(entries)
|
89
112
|
router.emit_stream(tag, es)
|
90
113
|
|
91
|
-
|
114
|
+
when Array
|
92
115
|
# Forward
|
93
|
-
es = MultiEventStream.new
|
116
|
+
es = Fluent::MultiEventStream.new
|
94
117
|
entries.each {|e|
|
95
118
|
record = e[1]
|
96
119
|
next if record.nil?
|
97
|
-
time = e[0]
|
98
|
-
time = (now ||= Engine.now) if time.to_i == 0
|
120
|
+
time = convert_time(e[0])
|
99
121
|
es.add(time, record)
|
100
122
|
}
|
101
123
|
router.emit_stream(tag, es)
|
@@ -105,25 +127,28 @@ module Fluent
|
|
105
127
|
record = msg[2]
|
106
128
|
return if record.nil?
|
107
129
|
|
108
|
-
time = msg[1]
|
109
|
-
time = Engine.now if time.to_i == 0
|
130
|
+
time = convert_time(msg[1])
|
110
131
|
router.emit(tag, time, record)
|
111
132
|
end
|
112
133
|
end
|
113
134
|
|
135
|
+
def convert_time(time)
|
136
|
+
case time
|
137
|
+
when nil, 0
|
138
|
+
Fluent::EventTime.now
|
139
|
+
when Fluent::EventTime
|
140
|
+
time
|
141
|
+
else
|
142
|
+
Fluent::EventTime.from_time(Time.at(time))
|
143
|
+
end
|
144
|
+
end
|
145
|
+
|
114
146
|
class Handler < Coolio::Socket
|
115
147
|
def initialize(io, log, on_message)
|
116
148
|
super(io)
|
117
|
-
|
118
|
-
opt = [1, @timeout.to_i].pack('I!I!') # { int l_onoff; int l_linger; }
|
119
|
-
io.setsockopt(Socket::SOL_SOCKET, Socket::SO_LINGER, opt)
|
120
|
-
end
|
149
|
+
|
121
150
|
@on_message = on_message
|
122
151
|
@log = log
|
123
|
-
@log.trace {
|
124
|
-
remote_port, remote_addr = *Socket.unpack_sockaddr_in(@_io.getpeername) rescue nil
|
125
|
-
"accepted fluent socket from '#{remote_addr}:#{remote_port}': object_id=#{self.object_id}"
|
126
|
-
}
|
127
152
|
end
|
128
153
|
|
129
154
|
def on_connect
|
@@ -131,13 +156,13 @@ module Fluent
|
|
131
156
|
|
132
157
|
def on_read(data)
|
133
158
|
first = data[0]
|
134
|
-
if first == '{' || first == '['
|
159
|
+
if first == '{'.freeze || first == '['.freeze
|
135
160
|
m = method(:on_read_json)
|
136
|
-
@
|
137
|
-
@
|
161
|
+
@parser = Yajl::Parser.new
|
162
|
+
@parser.on_parse_complete = @on_message
|
138
163
|
else
|
139
164
|
m = method(:on_read_msgpack)
|
140
|
-
@
|
165
|
+
@parser = Fluent::MessagePackFactory.msgpack_unpacker
|
141
166
|
end
|
142
167
|
|
143
168
|
singleton_class.module_eval do
|
@@ -147,17 +172,17 @@ module Fluent
|
|
147
172
|
end
|
148
173
|
|
149
174
|
def on_read_json(data)
|
150
|
-
@
|
151
|
-
rescue
|
152
|
-
@log.error "unexpected error", error:
|
175
|
+
@parser << data
|
176
|
+
rescue => e
|
177
|
+
@log.error "unexpected error in json payload", error: e.to_s
|
153
178
|
@log.error_backtrace
|
154
179
|
close
|
155
180
|
end
|
156
181
|
|
157
182
|
def on_read_msgpack(data)
|
158
|
-
@
|
159
|
-
rescue
|
160
|
-
@log.error "unexpected error", error:
|
183
|
+
@parser.feed_each(data, &@on_message)
|
184
|
+
rescue => e
|
185
|
+
@log.error "unexpected error in msgpack payload", error: e.to_s
|
161
186
|
@log.error_backtrace
|
162
187
|
close
|
163
188
|
end
|
@@ -167,29 +192,4 @@ module Fluent
|
|
167
192
|
end
|
168
193
|
end
|
169
194
|
end
|
170
|
-
|
171
|
-
class UnixInput < StreamInput
|
172
|
-
Plugin.register_input('unix', self)
|
173
|
-
|
174
|
-
desc 'The path to your Unix Domain Socket.'
|
175
|
-
config_param :path, :string, default: DEFAULT_SOCKET_PATH
|
176
|
-
desc 'The backlog of Unix Domain Socket.'
|
177
|
-
config_param :backlog, :integer, default: nil
|
178
|
-
|
179
|
-
def configure(conf)
|
180
|
-
super
|
181
|
-
#log.warn "'unix' input is obsoleted and will be removed. Use 'forward' instead."
|
182
|
-
end
|
183
|
-
|
184
|
-
def listen
|
185
|
-
if File.exist?(@path)
|
186
|
-
File.unlink(@path)
|
187
|
-
end
|
188
|
-
FileUtils.mkdir_p File.dirname(@path)
|
189
|
-
log.info "listening fluent socket on #{@path}"
|
190
|
-
s = Coolio::UNIXServer.new(@path, Handler, log, method(:on_message))
|
191
|
-
s.listen(@backlog) unless @backlog.nil?
|
192
|
-
s
|
193
|
-
end
|
194
|
-
end
|
195
195
|
end
|