fluentd 1.15.2-x64-mingw32 → 1.16.0-x64-mingw32
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/.github/workflows/linux-test.yaml +2 -2
- data/.github/workflows/macos-test.yaml +2 -2
- data/.github/workflows/windows-test.yaml +2 -2
- data/CHANGELOG.md +96 -0
- data/MAINTAINERS.md +2 -0
- data/README.md +0 -1
- data/fluentd.gemspec +2 -2
- data/lib/fluent/command/fluentd.rb +55 -64
- data/lib/fluent/config/yaml_parser/loader.rb +18 -1
- data/lib/fluent/daemon.rb +2 -4
- data/lib/fluent/file_wrapper.rb +137 -0
- data/lib/fluent/log/console_adapter.rb +66 -0
- data/lib/fluent/log.rb +35 -5
- data/lib/fluent/oj_options.rb +1 -2
- data/lib/fluent/plugin/base.rb +5 -7
- data/lib/fluent/plugin/buf_file.rb +32 -3
- data/lib/fluent/plugin/buf_file_single.rb +32 -3
- data/lib/fluent/plugin/buffer/file_chunk.rb +1 -1
- data/lib/fluent/plugin/buffer.rb +21 -0
- data/lib/fluent/plugin/in_tail.rb +1 -6
- data/lib/fluent/plugin/in_tcp.rb +4 -2
- data/lib/fluent/plugin/out_file.rb +0 -4
- data/lib/fluent/plugin/out_forward/ack_handler.rb +19 -4
- data/lib/fluent/plugin/out_forward.rb +2 -2
- data/lib/fluent/plugin/out_secondary_file.rb +39 -22
- data/lib/fluent/plugin/output.rb +49 -12
- data/lib/fluent/plugin_helper/http_server/server.rb +2 -1
- data/lib/fluent/supervisor.rb +157 -232
- data/lib/fluent/test/driver/base.rb +11 -5
- data/lib/fluent/test/driver/filter.rb +4 -0
- data/lib/fluent/test/startup_shutdown.rb +6 -8
- data/lib/fluent/version.rb +1 -1
- data/test/command/test_ctl.rb +1 -1
- data/test/command/test_fluentd.rb +168 -22
- data/test/command/test_plugin_config_formatter.rb +0 -1
- data/test/compat/test_parser.rb +5 -5
- data/test/config/test_system_config.rb +0 -8
- data/test/log/test_console_adapter.rb +110 -0
- data/test/plugin/out_forward/test_ack_handler.rb +39 -0
- data/test/plugin/test_base.rb +98 -0
- data/test/plugin/test_buf_file.rb +62 -23
- data/test/plugin/test_buf_file_single.rb +65 -0
- data/test/plugin/test_in_http.rb +2 -3
- data/test/plugin/test_in_monitor_agent.rb +2 -3
- data/test/plugin/test_in_tail.rb +105 -103
- data/test/plugin/test_in_tcp.rb +15 -0
- data/test/plugin/test_out_file.rb +3 -2
- data/test/plugin/test_out_forward.rb +14 -18
- data/test/plugin/test_out_http.rb +1 -0
- data/test/plugin/test_output.rb +269 -0
- data/test/plugin/test_parser_regexp.rb +1 -6
- data/test/plugin_helper/test_http_server_helper.rb +1 -1
- data/test/plugin_helper/test_server.rb +10 -5
- data/test/test_config.rb +57 -21
- data/test/{plugin/test_file_wrapper.rb → test_file_wrapper.rb} +2 -2
- data/test/test_formatter.rb +23 -20
- data/test/test_log.rb +85 -40
- data/test/test_supervisor.rb +300 -283
- metadata +15 -24
- data/.drone.yml +0 -35
- data/.github/workflows/issue-auto-closer.yml +0 -12
- data/.github/workflows/stale-actions.yml +0 -22
- data/.gitlab-ci.yml +0 -103
- data/lib/fluent/plugin/file_wrapper.rb +0 -131
- data/test/test_logger_initializer.rb +0 -46
data/test/command/test_ctl.rb
CHANGED
@@ -5,20 +5,35 @@ require_relative '../helper'
|
|
5
5
|
|
6
6
|
require 'fileutils'
|
7
7
|
require 'timeout'
|
8
|
+
require 'securerandom'
|
9
|
+
require 'fluent/file_wrapper'
|
8
10
|
|
9
11
|
class TestFluentdCommand < ::Test::Unit::TestCase
|
10
|
-
TMP_DIR = File.expand_path(File.dirname(__FILE__) + "/../tmp/command/fluentd#{ENV['TEST_ENV_NUMBER']}")
|
11
12
|
SUPERVISOR_PID_PATTERN = /starting fluentd-[.0-9]+ pid=(\d+)/
|
12
13
|
WORKER_PID_PATTERN = /starting fluentd worker pid=(\d+) /
|
13
14
|
|
15
|
+
def tmp_dir
|
16
|
+
File.join(File.dirname(__FILE__), "..", "tmp", "command" "fluentd#{ENV['TEST_ENV_NUMBER']}", SecureRandom.hex(10))
|
17
|
+
end
|
18
|
+
|
14
19
|
setup do
|
15
|
-
|
16
|
-
FileUtils.mkdir_p(
|
20
|
+
@tmp_dir = tmp_dir
|
21
|
+
FileUtils.mkdir_p(@tmp_dir)
|
17
22
|
@supervisor_pid = nil
|
18
23
|
@worker_pids = []
|
19
24
|
ENV["TEST_RUBY_PATH"] = nil
|
20
25
|
end
|
21
26
|
|
27
|
+
teardown do
|
28
|
+
begin
|
29
|
+
FileUtils.rm_rf(@tmp_dir)
|
30
|
+
rescue Errno::EACCES
|
31
|
+
# It may occur on Windows because of delete pending state due to delayed GC.
|
32
|
+
# Ruby 3.2 or later doesn't ignore Errno::EACCES:
|
33
|
+
# https://github.com/ruby/ruby/commit/983115cf3c8f75b1afbe3274f02c1529e1ce3a81
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
22
37
|
def process_exist?(pid)
|
23
38
|
begin
|
24
39
|
r = Process.waitpid(pid, Process::WNOHANG)
|
@@ -30,17 +45,17 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
30
45
|
end
|
31
46
|
|
32
47
|
def create_conf_file(name, content, ext_enc = 'utf-8')
|
33
|
-
conf_path = File.join(
|
34
|
-
|
48
|
+
conf_path = File.join(@tmp_dir, name)
|
49
|
+
Fluent::FileWrapper.open(conf_path, "w:#{ext_enc}:utf-8") do |file|
|
35
50
|
file.write content
|
36
51
|
end
|
37
52
|
conf_path
|
38
53
|
end
|
39
54
|
|
40
55
|
def create_plugin_file(name, content)
|
41
|
-
file_path = File.join(
|
56
|
+
file_path = File.join(@tmp_dir, 'plugin', name)
|
42
57
|
FileUtils.mkdir_p(File.dirname(file_path))
|
43
|
-
|
58
|
+
Fluent::FileWrapper.open(file_path, 'w') do |file|
|
44
59
|
file.write content
|
45
60
|
end
|
46
61
|
file_path
|
@@ -56,8 +71,22 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
56
71
|
end
|
57
72
|
end
|
58
73
|
|
59
|
-
def
|
60
|
-
|
74
|
+
def process_kill(pid)
|
75
|
+
if Fluent.windows?
|
76
|
+
Process.kill(:KILL, pid) rescue nil
|
77
|
+
return
|
78
|
+
end
|
79
|
+
|
80
|
+
begin
|
81
|
+
Process.kill(:TERM, pid) rescue nil
|
82
|
+
Timeout.timeout(10){ sleep 0.1 while process_exist?(pid) }
|
83
|
+
rescue Timeout::Error
|
84
|
+
Process.kill(:KILL, pid) rescue nil
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
def execute_command(cmdline, chdir=@tmp_dir, env = {})
|
89
|
+
null_stream = Fluent::FileWrapper.open(File::NULL, 'w')
|
61
90
|
gemfile_path = File.expand_path(File.dirname(__FILE__) + "../../../Gemfile")
|
62
91
|
|
63
92
|
env = { "BUNDLE_GEMFILE" => gemfile_path }.merge(env)
|
@@ -70,12 +99,12 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
70
99
|
yield pid, io
|
71
100
|
# p(here: "execute command", pid: pid, worker_pids: @worker_pids)
|
72
101
|
ensure
|
73
|
-
|
102
|
+
process_kill(pid)
|
74
103
|
if @supervisor_pid
|
75
|
-
|
104
|
+
process_kill(@supervisor_pid)
|
76
105
|
end
|
77
106
|
@worker_pids.each do |cpid|
|
78
|
-
|
107
|
+
process_kill(cpid)
|
79
108
|
end
|
80
109
|
# p(here: "execute command", pid: pid, exist: process_exist?(pid), worker_pids: @worker_pids, exists: @worker_pids.map{|i| process_exist?(i) })
|
81
110
|
Timeout.timeout(10){ sleep 0.1 while process_exist?(pid) }
|
@@ -97,16 +126,18 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
97
126
|
end
|
98
127
|
end
|
99
128
|
|
100
|
-
|
129
|
+
# ATTENTION: This stops taking logs when all `pattern_list` match or timeout,
|
130
|
+
# so `patterns_not_match` can test only logs up to that point.
|
131
|
+
def assert_log_matches(cmdline, *pattern_list, patterns_not_match: [], timeout: 20, env: {})
|
101
132
|
matched = false
|
102
133
|
matched_wrongly = false
|
103
134
|
assert_error_msg = ""
|
104
135
|
stdio_buf = ""
|
105
136
|
begin
|
106
|
-
execute_command(cmdline,
|
137
|
+
execute_command(cmdline, @tmp_dir, env) do |pid, stdout|
|
107
138
|
begin
|
108
139
|
waiting(timeout) do
|
109
|
-
while process_exist?(pid)
|
140
|
+
while process_exist?(pid)
|
110
141
|
readables, _, _ = IO.select([stdout], nil, nil, 1)
|
111
142
|
next unless readables
|
112
143
|
break if readables.first.eof?
|
@@ -118,6 +149,18 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
118
149
|
if pattern_list.all?{|ptn| lines.any?{|line| ptn.is_a?(Regexp) ? ptn.match(line) : line.include?(ptn) } }
|
119
150
|
matched = true
|
120
151
|
end
|
152
|
+
|
153
|
+
if Fluent.windows?
|
154
|
+
# https://github.com/fluent/fluentd/issues/4095
|
155
|
+
# On Windows, the initial process is different from the supervisor process,
|
156
|
+
# so we need to wait until `SUPERVISOR_PID_PATTERN` appears in the logs to get the pid.
|
157
|
+
# (Worker processes will be killed by the supervisor process, so we don't need it-)
|
158
|
+
break if matched && SUPERVISOR_PID_PATTERN =~ stdio_buf
|
159
|
+
else
|
160
|
+
# On Non-Windows, the initial process is the supervisor process,
|
161
|
+
# so we don't need to wait `SUPERVISOR_PID_PATTERN`.
|
162
|
+
break if matched
|
163
|
+
end
|
121
164
|
end
|
122
165
|
end
|
123
166
|
ensure
|
@@ -131,6 +174,10 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
131
174
|
end
|
132
175
|
rescue Timeout::Error
|
133
176
|
assert_error_msg = "execution timeout"
|
177
|
+
# https://github.com/fluent/fluentd/issues/4095
|
178
|
+
# On Windows, timeout without `@supervisor_pid` means that the test is invalid,
|
179
|
+
# since the supervisor process will survive without being killed correctly.
|
180
|
+
flunk("Invalid test: The pid of supervisor could not be taken, which is necessary on Windows.") if Fluent.windows? && @supervisor_pid.nil?
|
134
181
|
rescue => e
|
135
182
|
assert_error_msg = "unexpected error in launching fluentd: #{e.inspect}"
|
136
183
|
else
|
@@ -162,7 +209,7 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
162
209
|
assert matched && !matched_wrongly, assert_error_msg
|
163
210
|
end
|
164
211
|
|
165
|
-
def assert_fluentd_fails_to_start(cmdline, *pattern_list, timeout:
|
212
|
+
def assert_fluentd_fails_to_start(cmdline, *pattern_list, timeout: 20)
|
166
213
|
# empty_list.all?{ ... } is always true
|
167
214
|
matched = false
|
168
215
|
running = false
|
@@ -198,6 +245,10 @@ class TestFluentdCommand < ::Test::Unit::TestCase
|
|
198
245
|
end
|
199
246
|
rescue Timeout::Error
|
200
247
|
assert_error_msg = "execution timeout with command out:\n" + stdio_buf
|
248
|
+
# https://github.com/fluent/fluentd/issues/4095
|
249
|
+
# On Windows, timeout without `@supervisor_pid` means that the test is invalid,
|
250
|
+
# since the supervisor process will survive without being killed correctly.
|
251
|
+
flunk("Invalid test: The pid of supervisor could not be taken, which is necessary on Windows.") if Fluent.windows? && @supervisor_pid.nil?
|
201
252
|
rescue => e
|
202
253
|
assert_error_msg = "unexpected error in launching fluentd: #{e.inspect}\n" + stdio_buf
|
203
254
|
assert false, assert_error_msg
|
@@ -269,7 +320,7 @@ CONF
|
|
269
320
|
|
270
321
|
sub_test_case 'with system configuration about root directory' do
|
271
322
|
setup do
|
272
|
-
@root_path = File.join(
|
323
|
+
@root_path = File.join(@tmp_dir, "rootpath")
|
273
324
|
FileUtils.rm_rf(@root_path)
|
274
325
|
@conf = <<CONF
|
275
326
|
<system>
|
@@ -308,7 +359,7 @@ CONF
|
|
308
359
|
end
|
309
360
|
|
310
361
|
test 'fails to launch fluentd if specified root path is invalid path for directory' do
|
311
|
-
|
362
|
+
Fluent::FileWrapper.open(@root_path, 'w') do |_|
|
312
363
|
# create file and close it
|
313
364
|
end
|
314
365
|
conf_path = create_conf_file('existing_root_dir.conf', @conf)
|
@@ -508,7 +559,7 @@ CONF
|
|
508
559
|
|
509
560
|
assert_fluentd_fails_to_start(
|
510
561
|
create_cmdline(conf_path, "-p", File.dirname(plugin_path)),
|
511
|
-
"in_buggy.rb:5: syntax error, unexpected end-of-input
|
562
|
+
"in_buggy.rb:5: syntax error, unexpected end-of-input"
|
512
563
|
)
|
513
564
|
end
|
514
565
|
end
|
@@ -554,7 +605,7 @@ CONF
|
|
554
605
|
|
555
606
|
sub_test_case 'configured to run 2 workers' do
|
556
607
|
setup do
|
557
|
-
@root_path = File.join(
|
608
|
+
@root_path = File.join(@tmp_dir, "rootpath")
|
558
609
|
FileUtils.rm_rf(@root_path)
|
559
610
|
FileUtils.mkdir_p(@root_path)
|
560
611
|
end
|
@@ -961,10 +1012,10 @@ CONF
|
|
961
1012
|
</match>
|
962
1013
|
CONF
|
963
1014
|
ruby_path = ServerEngine.ruby_bin_path
|
964
|
-
tmp_ruby_path = File.join(
|
1015
|
+
tmp_ruby_path = File.join(@tmp_dir, "ruby with spaces")
|
965
1016
|
if Fluent.windows?
|
966
1017
|
tmp_ruby_path << ".bat"
|
967
|
-
|
1018
|
+
Fluent::FileWrapper.open(tmp_ruby_path, "w") do |file|
|
968
1019
|
file.write "#{ruby_path} %*"
|
969
1020
|
end
|
970
1021
|
else
|
@@ -1136,4 +1187,99 @@ CONF
|
|
1136
1187
|
"shared socket for multiple workers is disabled",)
|
1137
1188
|
end
|
1138
1189
|
end
|
1190
|
+
|
1191
|
+
# TODO: `patterns_not_match` can test only logs up to `pattern_list`,
|
1192
|
+
# so we need to fix some meaningless `patterns_not_match` conditions.
|
1193
|
+
sub_test_case 'log_level by command line option' do
|
1194
|
+
test 'info' do
|
1195
|
+
conf = ""
|
1196
|
+
conf_path = create_conf_file('empty.conf', conf)
|
1197
|
+
assert File.exist?(conf_path)
|
1198
|
+
assert_log_matches(create_cmdline(conf_path),
|
1199
|
+
"[info]",
|
1200
|
+
patterns_not_match: ["[debug]"])
|
1201
|
+
end
|
1202
|
+
|
1203
|
+
test 'debug' do
|
1204
|
+
conf = ""
|
1205
|
+
conf_path = create_conf_file('empty.conf', conf)
|
1206
|
+
assert File.exist?(conf_path)
|
1207
|
+
assert_log_matches(create_cmdline(conf_path, "-v"),
|
1208
|
+
"[debug]",
|
1209
|
+
patterns_not_match: ["[trace]"])
|
1210
|
+
end
|
1211
|
+
|
1212
|
+
data("Trace" => "-vv")
|
1213
|
+
data("Invalid low level should be treated as Trace level": "-vvv")
|
1214
|
+
test 'trace' do |option|
|
1215
|
+
conf = <<CONF
|
1216
|
+
<source>
|
1217
|
+
@type sample
|
1218
|
+
tag test
|
1219
|
+
</source>
|
1220
|
+
CONF
|
1221
|
+
conf_path = create_conf_file('sample.conf', conf)
|
1222
|
+
assert File.exist?(conf_path)
|
1223
|
+
assert_log_matches(create_cmdline(conf_path, option),
|
1224
|
+
"[trace]",)
|
1225
|
+
end
|
1226
|
+
|
1227
|
+
test 'warn' do
|
1228
|
+
omit "Can't run on Windows since there is no way to take pid of the supervisor." if Fluent.windows?
|
1229
|
+
conf = <<CONF
|
1230
|
+
<source>
|
1231
|
+
@type sample
|
1232
|
+
tag test
|
1233
|
+
</source>
|
1234
|
+
CONF
|
1235
|
+
conf_path = create_conf_file('sample.conf', conf)
|
1236
|
+
assert File.exist?(conf_path)
|
1237
|
+
assert_log_matches(create_cmdline(conf_path, "-q"),
|
1238
|
+
"[warn]",
|
1239
|
+
patterns_not_match: ["[info]"])
|
1240
|
+
end
|
1241
|
+
|
1242
|
+
data("Error" => "-qq")
|
1243
|
+
data("Fatal should be treated as Error level" => "-qqq")
|
1244
|
+
data("Invalid high level should be treated as Error level": "-qqqq")
|
1245
|
+
test 'error' do |option|
|
1246
|
+
# This test can run on Windows correctly,
|
1247
|
+
# since the process will stop automatically with an error.
|
1248
|
+
conf = <<CONF
|
1249
|
+
<source>
|
1250
|
+
@type plugin_not_found
|
1251
|
+
tag test
|
1252
|
+
</source>
|
1253
|
+
CONF
|
1254
|
+
conf_path = create_conf_file('plugin_not_found.conf', conf)
|
1255
|
+
assert File.exist?(conf_path)
|
1256
|
+
assert_log_matches(create_cmdline(conf_path, option),
|
1257
|
+
"[error]",
|
1258
|
+
patterns_not_match: ["[warn]"])
|
1259
|
+
end
|
1260
|
+
|
1261
|
+
test 'system config one should not be overwritten when cmd line one is not specified' do
|
1262
|
+
conf = <<CONF
|
1263
|
+
<system>
|
1264
|
+
log_level debug
|
1265
|
+
</system>
|
1266
|
+
CONF
|
1267
|
+
conf_path = create_conf_file('debug.conf', conf)
|
1268
|
+
assert File.exist?(conf_path)
|
1269
|
+
assert_log_matches(create_cmdline(conf_path),
|
1270
|
+
"[debug]")
|
1271
|
+
end
|
1272
|
+
end
|
1273
|
+
|
1274
|
+
sub_test_case "inline_config" do
|
1275
|
+
test "can change log_level by --inline-config" do
|
1276
|
+
# Since we can't define multiple `<system>` directives, this use-case is not recommended.
|
1277
|
+
# This is just for this test.
|
1278
|
+
inline_conf = '<system>\nlog_level debug\n</system>'
|
1279
|
+
conf_path = create_conf_file('test.conf', "")
|
1280
|
+
assert File.exist?(conf_path)
|
1281
|
+
assert_log_matches(create_cmdline(conf_path, "--inline-config", inline_conf),
|
1282
|
+
"[debug]")
|
1283
|
+
end
|
1284
|
+
end
|
1139
1285
|
end
|
@@ -188,7 +188,6 @@ slow_flush_log_threshold: float: (20.0)
|
|
188
188
|
retry_exponential_backoff_base: float: (2)
|
189
189
|
retry_max_interval: time: (nil)
|
190
190
|
retry_randomize: bool: (true)
|
191
|
-
disable_chunk_backup: bool: (false)
|
192
191
|
<secondary>: optional, single
|
193
192
|
@type: string: (nil)
|
194
193
|
<buffer>: optional, single
|
data/test/compat/test_parser.rb
CHANGED
@@ -38,14 +38,14 @@ class TextParserTest < ::Test::Unit::TestCase
|
|
38
38
|
|
39
39
|
def test_parse_with_return
|
40
40
|
parser = Fluent::TextParser.new
|
41
|
-
parser.configure('format' => 'none')
|
41
|
+
parser.configure(config_element('test', '', 'format' => 'none'))
|
42
42
|
_time, record = parser.parse('log message!')
|
43
43
|
assert_equal({'message' => 'log message!'}, record)
|
44
44
|
end
|
45
45
|
|
46
46
|
def test_parse_with_block
|
47
47
|
parser = Fluent::TextParser.new
|
48
|
-
parser.configure('format' => 'none')
|
48
|
+
parser.configure(config_element('test', '', 'format' => 'none'))
|
49
49
|
parser.parse('log message!') { |time, record|
|
50
50
|
assert_equal({'message' => 'log message!'}, record)
|
51
51
|
}
|
@@ -53,7 +53,7 @@ class TextParserTest < ::Test::Unit::TestCase
|
|
53
53
|
|
54
54
|
def test_multi_event_parser
|
55
55
|
parser = Fluent::TextParser.new
|
56
|
-
parser.configure('format' => 'multi_event_test')
|
56
|
+
parser.configure(config_element('test', '', 'format' => 'multi_event_test'))
|
57
57
|
i = 0
|
58
58
|
parser.parse('log message!') { |time, record|
|
59
59
|
assert_equal('log message!', record['message'])
|
@@ -67,7 +67,7 @@ class TextParserTest < ::Test::Unit::TestCase
|
|
67
67
|
assert_nil p1.estimate_current_event
|
68
68
|
assert_nil p1.parser
|
69
69
|
|
70
|
-
p1.configure('format' => 'none')
|
70
|
+
p1.configure(config_element('test', '', 'format' => 'none'))
|
71
71
|
assert_equal true, p1.parser.estimate_current_event
|
72
72
|
|
73
73
|
p2 = Fluent::TextParser.new
|
@@ -76,7 +76,7 @@ class TextParserTest < ::Test::Unit::TestCase
|
|
76
76
|
|
77
77
|
p2.estimate_current_event = false
|
78
78
|
|
79
|
-
p2.configure('format' => 'none')
|
79
|
+
p2.configure(config_element('test', '', 'format' => 'none'))
|
80
80
|
assert_equal false, p2.parser.estimate_current_event
|
81
81
|
end
|
82
82
|
|
@@ -5,13 +5,6 @@ require 'fluent/config/section'
|
|
5
5
|
require 'fluent/system_config'
|
6
6
|
|
7
7
|
module Fluent::Config
|
8
|
-
class FakeLoggerInitializer
|
9
|
-
attr_accessor :level
|
10
|
-
def initialize
|
11
|
-
@level = nil
|
12
|
-
end
|
13
|
-
end
|
14
|
-
|
15
8
|
class FakeSupervisor
|
16
9
|
attr_writer :log_level
|
17
10
|
|
@@ -21,7 +14,6 @@ module Fluent::Config
|
|
21
14
|
wokers: nil,
|
22
15
|
restart_worker_interval: nil,
|
23
16
|
root_dir: nil,
|
24
|
-
log: FakeLoggerInitializer.new,
|
25
17
|
log_level: Fluent::Log::LEVEL_INFO,
|
26
18
|
suppress_interval: nil,
|
27
19
|
suppress_config_dump: nil,
|
@@ -0,0 +1,110 @@
|
|
1
|
+
require_relative '../helper'
|
2
|
+
|
3
|
+
require 'fluent/log'
|
4
|
+
require 'fluent/log/console_adapter'
|
5
|
+
|
6
|
+
class ConsoleAdapterTest < Test::Unit::TestCase
|
7
|
+
def setup
|
8
|
+
@timestamp = Time.parse("2023-01-01 15:32:41 +0000")
|
9
|
+
@timestamp_str = @timestamp.strftime("%Y-%m-%d %H:%M:%S %z")
|
10
|
+
Timecop.freeze(@timestamp)
|
11
|
+
|
12
|
+
@logdev = Fluent::Test::DummyLogDevice.new
|
13
|
+
@logger = ServerEngine::DaemonLogger.new(@logdev)
|
14
|
+
@fluent_log = Fluent::Log.new(@logger)
|
15
|
+
@console_logger = Fluent::Log::ConsoleAdapter.wrap(@fluent_log)
|
16
|
+
end
|
17
|
+
|
18
|
+
def teardown
|
19
|
+
Timecop.return
|
20
|
+
end
|
21
|
+
|
22
|
+
def test_expected_log_levels
|
23
|
+
assert_equal({debug: 0, info: 1, warn: 2, error: 3, fatal: 4},
|
24
|
+
Console::Logger::LEVELS)
|
25
|
+
end
|
26
|
+
|
27
|
+
data(trace: [Fluent::Log::LEVEL_TRACE, :debug],
|
28
|
+
debug: [Fluent::Log::LEVEL_DEBUG, :debug],
|
29
|
+
info: [Fluent::Log::LEVEL_INFO, :info],
|
30
|
+
warn: [Fluent::Log::LEVEL_WARN, :warn],
|
31
|
+
error: [Fluent::Log::LEVEL_ERROR, :error],
|
32
|
+
fatal: [Fluent::Log::LEVEL_FATAL, :fatal])
|
33
|
+
def test_reflect_log_level(data)
|
34
|
+
level, expected = data
|
35
|
+
@fluent_log.level = level
|
36
|
+
console_logger = Fluent::Log::ConsoleAdapter.wrap(@fluent_log)
|
37
|
+
assert_equal(Console::Logger::LEVELS[expected],
|
38
|
+
console_logger.level)
|
39
|
+
end
|
40
|
+
|
41
|
+
data(debug: :debug,
|
42
|
+
info: :info,
|
43
|
+
warn: :warn,
|
44
|
+
error: :error,
|
45
|
+
fatal: :fatal)
|
46
|
+
def test_string_subject(level)
|
47
|
+
@console_logger.send(level, "subject")
|
48
|
+
assert_equal(["#{@timestamp_str} [#{level}]: 0.0s: subject\n"],
|
49
|
+
@logdev.logs)
|
50
|
+
end
|
51
|
+
|
52
|
+
data(debug: :debug,
|
53
|
+
info: :info,
|
54
|
+
warn: :warn,
|
55
|
+
error: :error,
|
56
|
+
fatal: :fatal)
|
57
|
+
def test_args(level)
|
58
|
+
@console_logger.send(level, "subject", 1, 2, 3)
|
59
|
+
assert_equal([
|
60
|
+
"#{@timestamp_str} [#{level}]: 0.0s: subject\n" +
|
61
|
+
" | 1\n" +
|
62
|
+
" | 2\n" +
|
63
|
+
" | 3\n"
|
64
|
+
],
|
65
|
+
@logdev.logs)
|
66
|
+
end
|
67
|
+
|
68
|
+
data(debug: :debug,
|
69
|
+
info: :info,
|
70
|
+
warn: :warn,
|
71
|
+
error: :error,
|
72
|
+
fatal: :fatal)
|
73
|
+
def test_options(level)
|
74
|
+
@console_logger.send(level, "subject", kwarg1: "opt1", kwarg2: "opt2")
|
75
|
+
assert_equal([
|
76
|
+
"#{@timestamp_str} [#{level}]: 0.0s: subject\n" +
|
77
|
+
" | {\"kwarg1\":\"opt1\",\"kwarg2\":\"opt2\"}\n"
|
78
|
+
],
|
79
|
+
@logdev.logs)
|
80
|
+
end
|
81
|
+
|
82
|
+
data(debug: :debug,
|
83
|
+
info: :info,
|
84
|
+
warn: :warn,
|
85
|
+
error: :error,
|
86
|
+
fatal: :fatal)
|
87
|
+
def test_block(level)
|
88
|
+
@console_logger.send(level, "subject") { "block message" }
|
89
|
+
assert_equal([
|
90
|
+
"#{@timestamp_str} [#{level}]: 0.0s: subject\n" +
|
91
|
+
" | block message\n"
|
92
|
+
],
|
93
|
+
@logdev.logs)
|
94
|
+
end
|
95
|
+
|
96
|
+
data(debug: :debug,
|
97
|
+
info: :info,
|
98
|
+
warn: :warn,
|
99
|
+
error: :error,
|
100
|
+
fatal: :fatal)
|
101
|
+
def test_multiple_entries(level)
|
102
|
+
@console_logger.send(level, "subject1")
|
103
|
+
@console_logger.send(level, "line2")
|
104
|
+
assert_equal([
|
105
|
+
"#{@timestamp_str} [#{level}]: 0.0s: subject1\n",
|
106
|
+
"#{@timestamp_str} [#{level}]: 0.0s: line2\n"
|
107
|
+
],
|
108
|
+
@logdev.logs)
|
109
|
+
end
|
110
|
+
end
|
@@ -98,4 +98,43 @@ class AckHandlerTest < Test::Unit::TestCase
|
|
98
98
|
w.close rescue nil
|
99
99
|
end
|
100
100
|
end
|
101
|
+
|
102
|
+
# ForwardOutput uses AckHandler in multiple threads, so we need to assume this case.
|
103
|
+
# If exclusive control for this case is implemented, this test may not be necessary.
|
104
|
+
test 'raises no error when another thread closes a socket' do
|
105
|
+
ack_handler = Fluent::Plugin::ForwardOutput::AckHandler.new(timeout: 10, log: $log, read_length: 100)
|
106
|
+
|
107
|
+
node = flexmock('node', host: '127.0.0.1', port: '1000') # for log
|
108
|
+
chunk_id = 'chunk_id 111'
|
109
|
+
ack = ack_handler.create_ack(chunk_id, node)
|
110
|
+
|
111
|
+
r, w = IO.pipe
|
112
|
+
begin
|
113
|
+
w.write(chunk_id)
|
114
|
+
stub(r).recv { |_|
|
115
|
+
sleep(1) # To ensure that multiple threads select the socket before closing.
|
116
|
+
raise IOError, 'stream closed in another thread' if r.closed?
|
117
|
+
MessagePack.pack({ 'ack' => Base64.encode64('chunk_id 111') })
|
118
|
+
}
|
119
|
+
ack.enqueue(r)
|
120
|
+
|
121
|
+
threads = []
|
122
|
+
2.times do
|
123
|
+
threads << Thread.new do
|
124
|
+
ack_handler.collect_response(1) do |cid, n, s, ret|
|
125
|
+
s&.close
|
126
|
+
end
|
127
|
+
end
|
128
|
+
end
|
129
|
+
|
130
|
+
assert_true threads.map{ |t| t.join(10) }.all?
|
131
|
+
assert_false(
|
132
|
+
$log.out.logs.any? { |log| log.include?('[error]') },
|
133
|
+
$log.out.logs.select{ |log| log.include?('[error]') }.join('\n')
|
134
|
+
)
|
135
|
+
ensure
|
136
|
+
r.close rescue nil
|
137
|
+
w.close rescue nil
|
138
|
+
end
|
139
|
+
end
|
101
140
|
end
|
data/test/plugin/test_base.rb
CHANGED
@@ -146,4 +146,102 @@ class BaseTest < Test::Unit::TestCase
|
|
146
146
|
end
|
147
147
|
end
|
148
148
|
end
|
149
|
+
|
150
|
+
test '`ArgumentError` when `conf` is not `Fluent::Config::Element`' do
|
151
|
+
assert_raise ArgumentError.new('BUG: type of conf must be Fluent::Config::Element, but Hash is passed.') do
|
152
|
+
@p.configure({})
|
153
|
+
end
|
154
|
+
end
|
155
|
+
|
156
|
+
sub_test_case 'system_config.workers value after configure' do
|
157
|
+
def assert_system_config_workers_value(data)
|
158
|
+
conf = config_element()
|
159
|
+
conf.set_target_worker_ids(data[:target_worker_ids])
|
160
|
+
@p.configure(conf)
|
161
|
+
assert{ @p.system_config.workers == data[:expected] }
|
162
|
+
end
|
163
|
+
|
164
|
+
def stub_supervisor_mode
|
165
|
+
stub(Fluent::Engine).supervisor_mode { true }
|
166
|
+
stub(Fluent::Engine).worker_id { -1 }
|
167
|
+
end
|
168
|
+
|
169
|
+
sub_test_case 'with <system> workers 3 </system>' do
|
170
|
+
setup do
|
171
|
+
system_config = Fluent::SystemConfig.new
|
172
|
+
system_config.workers = 3
|
173
|
+
stub(Fluent::Engine).system_config { system_config }
|
174
|
+
end
|
175
|
+
|
176
|
+
data(
|
177
|
+
'without <worker> directive',
|
178
|
+
{
|
179
|
+
target_worker_ids: [],
|
180
|
+
expected: 3
|
181
|
+
},
|
182
|
+
keep: true
|
183
|
+
)
|
184
|
+
data(
|
185
|
+
'with <worker 0>',
|
186
|
+
{
|
187
|
+
target_worker_ids: [0],
|
188
|
+
expected: 1
|
189
|
+
},
|
190
|
+
keep: true
|
191
|
+
)
|
192
|
+
data(
|
193
|
+
'with <worker 0-1>',
|
194
|
+
{
|
195
|
+
target_worker_ids: [0, 1],
|
196
|
+
expected: 2
|
197
|
+
},
|
198
|
+
keep: true
|
199
|
+
)
|
200
|
+
data(
|
201
|
+
'with <worker 0-2>',
|
202
|
+
{
|
203
|
+
target_worker_ids: [0, 1, 2],
|
204
|
+
expected: 3
|
205
|
+
},
|
206
|
+
keep: true
|
207
|
+
)
|
208
|
+
|
209
|
+
test 'system_config.workers value after configure' do
|
210
|
+
assert_system_config_workers_value(data)
|
211
|
+
end
|
212
|
+
|
213
|
+
test 'system_config.workers value after configure with supervisor_mode' do
|
214
|
+
stub_supervisor_mode
|
215
|
+
assert_system_config_workers_value(data)
|
216
|
+
end
|
217
|
+
end
|
218
|
+
|
219
|
+
sub_test_case 'without <system> directive' do
|
220
|
+
data(
|
221
|
+
'without <worker> directive',
|
222
|
+
{
|
223
|
+
target_worker_ids: [],
|
224
|
+
expected: 1
|
225
|
+
},
|
226
|
+
keep: true
|
227
|
+
)
|
228
|
+
data(
|
229
|
+
'with <worker 0>',
|
230
|
+
{
|
231
|
+
target_worker_ids: [0],
|
232
|
+
expected: 1
|
233
|
+
},
|
234
|
+
keep: true
|
235
|
+
)
|
236
|
+
|
237
|
+
test 'system_config.workers value after configure' do
|
238
|
+
assert_system_config_workers_value(data)
|
239
|
+
end
|
240
|
+
|
241
|
+
test 'system_config.workers value after configure with supervisor_mode' do
|
242
|
+
stub_supervisor_mode
|
243
|
+
assert_system_config_workers_value(data)
|
244
|
+
end
|
245
|
+
end
|
246
|
+
end
|
149
247
|
end
|