require 'spec_helper' describe LogStashLogger do # The type of socket we're testing def socket_type @socket_type ||= (ENV['SOCKET_TYPE'] || 'UDP').to_s.downcase.to_sym end before(:all) { puts "Testing with #{socket_type.to_s.upcase} socket type" } let(:host) { '0.0.0.0' } let(:hostname) { Socket.gethostname } let(:port) { 5228 } # The logstash logger let(:logger) { LogStashLogger.new(host, port, socket_type) } # The log device that the logger writes to let(:logdev) { logger.instance_variable_get(:@logdev) } let! :listener do case socket_type when :tcp TCPServer.new(port) when :udp UDPSocket.new.tap {|socket| socket.bind(host, port)} end end # The TCP socket written to by the TCP logstash listener server let(:tcp_client) { listener.accept } # The logstash event to log let(:logstash_event) do LogStash::Event.new.tap do |event| event['message'] = 'test' event['severity'] = 'INFO' end end # The raw input received by the logstash listener let :listener_input do case socket_type when :tcp then tcp_client.readline when :udp then listener.recvfrom(8192)[0] end end # The logstash event received by the listener let(:listener_event) { LogStash::Event.new(JSON.parse listener_input) } #before(:each) do # Sync socket writes so we can receive them in the listener immediately #@socket = logdev.instance_variable_get(:@dev).send(:connect) #@socket.sync = true #end after(:each) do listener.close end # The socket that the logger is writing to #let(:socket) { @socket } it 'uses a LogStashLogger::Socket as the log device' do expect(logdev).to be_a Logger::LogDevice expect(logdev.instance_variable_get(:@dev)).to be_a LogStashLogger::Socket end it 'takes a string message as input and writes a logstash event' do message = 'test' expect(logdev).to receive(:write).and_call_original do |event| expect(event).to be_a LogStash::Event expect(event.source).to eql(hostname) expect(event['message']).to eql(message) expect(event['severity']).to eql('INFO') end logger.info(message) expect(listener_event['message']).to eq(message) expect(listener_event['source']).to eq(hostname) end it 'takes a logstash-formatted json string as input and writes out a logstash event' do expect(logdev).to receive(:write).and_call_original do |event| expect(event).to be_a LogStash::Event expect(event['message']).to eql(logstash_event['message']) expect(event.source).to eql(hostname) end logger.info(logstash_event.to_json) expect(listener_event['message']).to eq(logstash_event['message']) expect(listener_event['source']).to eq(hostname) end it 'takes a LogStash::Event as input and writes it out intact' do expect(logdev).to receive(:write).and_call_original do |event| expect(event).to be_a LogStash::Event expect(event['message']).to eql(logstash_event['message']) expect(event['severity']).to eql(logstash_event['severity']) expect(event.timestamp).to eql(logstash_event.timestamp) expect(event.source).to eql(hostname) end logger.warn(logstash_event) expect(listener_event['message']).to eq(logstash_event['message']) expect(listener_event['severity']).to eq(logstash_event['severity']) expect(listener_event['source']).to eq(hostname) end it 'takes a data hash as input and writes out a logstash event' do data = { 'message' => 'test', 'severity' => 'INFO', 'foo' => 'bar' } expect(logdev).to receive(:write).and_call_original do |event| expect(event).to be_a LogStash::Event expect(event['message']).to eql('test') expect(event['severity']).to eql('INFO') expect(event['foo']).to eql('bar') expect(event.source).to eql(hostname) end logger.info(data.dup) expect(listener_event['message']).to eq(data["message"]) expect(listener_event['severity']).to eq(data['severity']) expect(listener_event['foo']).to eq(data['foo']) expect(listener_event['source']).to eq(hostname) expect(listener_event['@timestamp']).to_not be_nil end end