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