logstash-core/spec/logstash/runner_spec.rb (423 lines of code) (raw):
# Licensed to Elasticsearch B.V. under one or more contributor
# license agreements. See the NOTICE file distributed with
# this work for additional information regarding copyright
# ownership. Elasticsearch B.V. licenses this file to you under
# the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied. See the License for the
# specific language governing permissions and limitations
# under the License.
require "spec_helper"
require "logstash/runner"
require "stud/task"
require "stud/trap"
require "stud/temporary"
require "logstash/util/java_version"
require "logstash/config/source_loader"
require "json"
require "webmock/rspec"
require_relative "../support/helpers"
require_relative "../support/matchers"
describe LogStash::Runner do
subject(:runner) { LogStash::Runner }
let(:logger) { double("logger") }
let(:agent) { double("agent") }
before :each do
clear_data_dir
WebMock.disable_net_connect!(allow_localhost: true)
allow(LogStash::Runner).to receive(:logger).and_return(logger)
allow(logger).to receive(:debug?).and_return(true)
allow(logger).to receive(:subscribe).with(any_args)
allow(logger).to receive(:debug) {}
allow(logger).to receive(:log) {}
allow(logger).to receive(:info) {}
allow(logger).to receive(:fatal) {}
allow(logger).to receive(:warn) {}
allow(LogStash::ShutdownWatcher).to receive(:logger).and_return(logger)
allow(LogStash::Logging::Logger).to receive(:configure_logging) do |level, path|
allow(logger).to receive(:level).and_return(level.to_sym)
end
allow(LogStash::Logging::Logger).to receive(:reconfigure).with(any_args)
# Make sure we don't start a real pipeline here.
# because we cannot easily close the pipeline
allow(LogStash::Agent).to receive(:new).with(any_args).and_return(agent)
allow(agent).to receive(:execute)
allow(agent).to receive(:shutdown)
end
describe "argument precedence" do
let(:config) { "input {} output {}" }
let(:cli_args) { ["-e", config, "-w", "20"] }
let(:settings_yml_hash) { { "pipeline.workers" => 2 } }
before :each do
allow(LogStash::SETTINGS).to receive(:read_yaml).and_return(settings_yml_hash)
end
it "favors the last occurence of an option" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.get("config.string")).to eq(config)
expect(settings.get("pipeline.workers")).to eq(20)
end.and_return(agent)
subject.run("bin/logstash", cli_args)
end
end
describe "argument parsing" do
subject { LogStash::Runner.new("") }
context "when -e is given" do
let(:args) { ["-e", "input {} output {}"] }
let(:agent) { double("agent") }
before do
allow(agent).to receive(:shutdown)
end
it "should execute the agent" do
expect(subject).to receive(:create_agent).and_return(agent)
expect(agent).to receive(:execute).once
subject.run(args)
end
end
end
context "--pluginpath" do
subject { LogStash::Runner.new("") }
let(:valid_directory) { Stud::Temporary.directory }
let(:invalid_directory) { "/a/path/that/doesnt/exist" }
let(:multiple_paths) { [Stud::Temporary.directory, Stud::Temporary.directory] }
it "should pass -p contents to the configure_plugin_paths method" do
args = ["-p", valid_directory]
expect(subject).to receive(:configure_plugin_paths).with([valid_directory])
expect { subject.run(args) }.to_not raise_error
end
it "should add single valid dir path to the environment" do
expect(LogStash::Environment).to receive(:add_plugin_path).with(valid_directory)
subject.configure_plugin_paths(valid_directory)
end
it "should fail with single invalid dir path" do
expect(LogStash::Environment).not_to receive(:add_plugin_path)
expect {subject.configure_plugin_paths(invalid_directory)}.to raise_error(Clamp::UsageError)
end
it "should add multiple valid dir path to the environment" do
multiple_paths.each {|path| expect(LogStash::Environment).to receive(:add_plugin_path).with(path)}
subject.configure_plugin_paths(multiple_paths)
end
end
context "--auto-reload" do
subject { LogStash::Runner.new("") }
context "when -e is given" do
let(:args) { ["-r", "-e", "input {} output {}"] }
it "should exit immediately" do
expect(subject).to receive(:signal_usage_error).and_call_original
expect(subject).to receive(:show_short_help)
expect(subject.run(args)).to eq(1)
end
end
end
describe "--config.test_and_exit" do
subject { LogStash::Runner.new("") }
let(:args) { ["-t", "-e", pipeline_string] }
context "with a good configuration" do
let(:pipeline_string) { "input { } filter { } output { }" }
it "should exit successfully" do
expect(logger).not_to receive(:fatal)
expect(subject.run(args)).to eq(0)
end
end
context "with a bad configuration" do
let(:pipeline_string) { "rlwekjhrewlqrkjh" }
it "should fail by returning a bad exit code" do
expect(logger).to receive(:fatal)
expect(subject.run(args)).to eq(1)
end
end
context "with invalid field reference literal" do
let(:pipeline_string) { "input { } output { if [[f[[[oo] == [bar] { } }" }
it "should fail by returning a bad exit code" do
expect(logger).to receive(:fatal)
expect(subject.run(args)).to eq(1)
end
end
end
describe "pipeline settings" do
let(:pipeline_string) { "input { stdin {} } output { stdout {} }" }
let(:main_pipeline_settings) { { :pipeline_id => "main" } }
let(:pipeline) { double("pipeline") }
before(:each) do
allow_any_instance_of(LogStash::Agent).to receive(:execute).and_return(true)
task = Stud::Task.new { 1 }
allow(pipeline).to receive(:run).and_return(task)
allow(pipeline).to receive(:shutdown)
end
context "when :path.data is defined by the user" do
let(:test_data_path) { "/tmp/ls-test-data" }
let(:test_queue_path) { test_data_path + "/" + "queue" }
let(:test_dlq_path) { test_data_path + "/" + "dead_letter_queue" }
it "should set data paths" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.get("path.data")).to eq(test_data_path)
expect(settings.get("path.queue")).to eq(test_queue_path)
expect(settings.get("path.dead_letter_queue")).to eq(test_dlq_path)
end
args = ["--path.data", test_data_path, "-e", pipeline_string]
subject.run("bin/logstash", args)
end
context "and path.queue is manually set" do
let(:queue_override_path) { "/tmp/queue-override_path" }
it "should set data paths" do
LogStash::SETTINGS.set("path.queue", queue_override_path)
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.get("path.data")).to eq(test_data_path)
expect(settings.get("path.queue")).to eq(queue_override_path)
end
args = ["--path.data", test_data_path, "-e", pipeline_string]
subject.run("bin/logstash", args)
end
end
context "and path.dead_letter_queue is manually set" do
let(:queue_override_path) { "/tmp/queue-override_path" }
it "should set data paths" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.get("path.data")).to eq(test_data_path)
expect(settings.get("path.dead_letter_queue")).to eq(queue_override_path)
end
LogStash::SETTINGS.set("path.dead_letter_queue", queue_override_path)
args = ["--path.data", test_data_path, "-e", pipeline_string]
subject.run("bin/logstash", args)
end
end
end
context ':api.http.host' do
context "when undefined by the user" do
let(:args) { ["-e", pipeline_string] }
it "creates an Agent whose `api.http.host` uses the default value" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.set?("api.http.host")).to be_falsey
expect(settings.get("api.http.host")).to eq("127.0.0.1")
end
subject.run("bin/logstash", args)
end
end
context "when defined by the user" do
let(:args) { ["--api.http.host", "localhost", "-e", pipeline_string]}
it "creates an Agent whose `api.http.host` uses provided value" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.set?("api.http.host")).to be(true)
expect(settings.get("api.http.host")).to eq("localhost")
end
subject.run("bin/logstash", args)
end
end
end
context ":api.http.port" do
context "when undefined by the user" do
let(:args) { ["-e", pipeline_string] }
it "creates an Agent whose `api.http.port` uses the default value" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.set?("api.http.port")).to be_falsey
expect(settings.get("api.http.port")).to eq(9600..9700)
end
subject.run("bin/logstash", args)
end
end
context "when defined by the user" do
let(:args) { ["--api.http.port", port_argument, "-e", pipeline_string] }
context "as a single-value string" do
let(:port_argument) { "10000" }
it "creates an Agent whose `api.http.port` is an appropriate single-element range" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.set?("api.http.port")).to be(true)
expect(settings.get("api.http.port")).to eq(10000..10000)
end
subject.run("bin/logstash", args)
end
end
context "as a range" do
let(:port_argument) { "10000-20000" }
it "creates an Agent whose `api.http.port` uses the appropriate inclusive-end range" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.set?("api.http.port")).to be(true)
expect(settings.get("api.http.port")).to eq(10000..20000)
end
subject.run("bin/logstash", args)
end
end
end
end
context "when :pipeline_workers is not defined by the user" do
it "should not pass the value to the pipeline" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.set?("pipeline.workers")).to be(false)
end
args = ["-e", pipeline_string]
subject.run("bin/logstash", args)
end
end
context "when :pipeline_workers flag is passed without a value" do
it "should raise an error" do
args = ["-e", pipeline_string, "-w"]
expect { subject.run("bin/logstash", args) }.to raise_error
end
end
context "when :pipeline_workers is defined by the user" do
it "should pass the value to the pipeline" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.set?("pipeline.workers")).to be(true)
expect(settings.get("pipeline.workers")).to be(2)
end
args = ["-w", "2", "-e", pipeline_string]
subject.run("bin/logstash", args)
end
end
describe "config.debug" do
after(:each) do
LogStash::SETTINGS.set("config.debug", false)
end
it "should set 'config.debug' to false by default" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.get("config.debug")).to eq(false)
end
args = ["--log.level", "debug", "-e", pipeline_string]
subject.run("bin/logstash", args)
end
it "should allow overriding config.debug" do
expect(LogStash::Agent).to receive(:new) do |settings|
expect(settings.get("config.debug")).to eq(true)
end
args = ["--log.level", "debug", "--config.debug", "-e", pipeline_string]
subject.run("bin/logstash", args)
end
end
end
describe 'jackson defaults' do
subject { LogStash::Runner.new("") }
let(:args) { ["-e", "input {} output {}"] }
it 'should be verified' do
expect(LogStash::Util::Jackson).to receive(:verify_jackson_overrides)
subject.run(args)
end
end
describe "--log.level" do
before :each do
allow_any_instance_of(subject).to receive(:show_version)
end
context "when not set" do
it "should set log level to warn" do
args = ["--version"]
subject.run("bin/logstash", args)
expect(logger.level).to eq(:info)
end
end
context "when setting to debug" do
it "should set log level to debug" do
args = ["--log.level", "debug", "--version"]
subject.run("bin/logstash", args)
expect(logger.level).to eq(:debug)
end
end
context "when setting to verbose" do
it "should set log level to info" do
args = ["--log.level", "info", "--version"]
subject.run("bin/logstash", args)
expect(logger.level).to eq(:info)
end
end
context "when setting to quiet" do
it "should set log level to error" do
args = ["--log.level", "error", "--version"]
subject.run("bin/logstash", args)
expect(logger.level).to eq(:error)
end
end
context "deprecated flags" do
let(:deprecation_logger_stub) { double("DeprecationLogger").as_null_object }
before(:each) { allow(runner).to receive(:deprecation_logger).and_return(deprecation_logger_stub) }
context "when using --quiet" do
it "should warn about the deprecated flag" do
expect(deprecation_logger_stub).to receive(:deprecated).with(/DEPRECATION WARNING/)
args = ["--quiet", "--version"]
subject.run("bin/logstash", args)
end
it "should still set the log level accordingly" do
args = ["--quiet", "--version"]
subject.run("bin/logstash", args)
expect(logger.level).to eq(:error)
end
end
context "when using --debug" do
it "should warn about the deprecated flag" do
expect(deprecation_logger_stub).to receive(:deprecated).with(/DEPRECATION WARNING/)
args = ["--debug", "--version"]
subject.run("bin/logstash", args)
end
it "should still set the log level accordingly" do
args = ["--debug", "--version"]
subject.run("bin/logstash", args)
expect(logger.level).to eq(:debug)
end
end
context "when using --verbose" do
it "should warn about the deprecated flag" do
expect(deprecation_logger_stub).to receive(:deprecated).with(/DEPRECATION WARNING/)
args = ["--verbose", "--version"]
subject.run("bin/logstash", args)
end
it "should still set the log level accordingly" do
args = ["--verbose", "--version"]
subject.run("bin/logstash", args)
expect(logger.level).to eq(:info)
end
end
end
end
describe "path.settings" do
subject { LogStash::Runner.new("") }
context "if does not exist" do
let(:args) { ["--path.settings", "/tmp/a/a/a/a", "-e", "input { generator { count => 1000 }} output {}"] }
it "should not terminate logstash" do
# The runner should just pass the code from the agent execute
allow(agent).to receive(:execute).and_return(0)
expect(subject.run(args)).to eq(0)
end
context "but if --help is passed" do
let(:args) { ["--path.settings", "/tmp/a/a/a/a", "--help"] }
it "should show help" do
expect { subject.run(args) }.to raise_error(Clamp::HelpWanted)
end
end
end
end
describe "allow_superuser" do
subject { LogStash::Runner.new("") }
let(:args) { ["-e", "input {} output {}"] }
let(:deprecation_logger_stub) { double("DeprecationLogger").as_null_object }
before(:each) { allow(runner).to receive(:deprecation_logger).and_return(deprecation_logger_stub) }
if LogStash::Environment.windows?
context "unintentionally running logstash as superuser" do
it "runs successfully" do
LogStash::SETTINGS.set("allow_superuser", false)
expect(logger).not_to receive(:fatal)
expect { subject.run(args) }.not_to raise_error
end
end
else
context "unintentionally running logstash as superuser" do
before do
expect(Process).to receive(:euid).and_return(0)
end
it "fails with bad exit" do
LogStash::SETTINGS.set("allow_superuser", false)
expect(logger).to receive(:fatal) do |msg, hash|
expect(msg).to eq("An unexpected error occurred!")
expect(hash[:error].to_s).to match("Logstash cannot be run as superuser.")
end
expect(subject.run(args)).to eq(1)
end
end
context "intentionally running logstash as superuser " do
before do
expect(Process).to receive(:euid).and_return(0)
end
it "runs successfully with warning message" do
LogStash::SETTINGS.set("allow_superuser", true)
expect(logger).not_to receive(:fatal)
expect(logger).to receive(:warn).with(/NOTICE: Allowing Logstash to run as superuser is heavily discouraged as it poses a security risk./)
expect { subject.run(args) }.not_to raise_error
end
end
context "running logstash as non-root " do
before do
expect(Process).to receive(:euid).and_return(100)
end
it "runs successfully without any messages" do
LogStash::SETTINGS.set("allow_superuser", false)
expect(logger).not_to receive(:fatal)
expect(logger).not_to receive(:warn).with(/NOTICE: Allowing Logstash to run as superuser is heavily discouraged as it poses a security risk./)
expect { subject.run(args) }.not_to raise_error
end
end
end
end
end