require File.dirname(__FILE__) + '/helper' class SenderTest < Test::Unit::TestCase def setup reset_config end def build_sender(opts = {}) config = Airbrake::Configuration.new opts.each {|opt, value| config.send(:"#{opt}=", value) } Airbrake::Sender.new(config) end def send_exception(args = {}) notice = args.delete(:notice) || build_notice_data sender = args.delete(:sender) || build_sender(args) sender.send_to_airbrake(notice) end def stub_http(options = {}) response = stub(:body => options[:body] || 'body') http = stub(:post => response, :read_timeout= => nil, :open_timeout= => nil, :ca_file= => nil, :verify_mode= => nil, :use_ssl= => nil) Net::HTTP.stubs(:new => http) http end should "post to Airbrake when using an HTTP proxy" do response = stub(:body => 'body') http = stub(:post => response, :read_timeout= => nil, :open_timeout= => nil, :use_ssl= => nil) proxy = stub(:new => http) Net::HTTP.stubs(:Proxy => proxy) url = "http://airbrake.io:80#{Airbrake::Sender::NOTICES_URI}" uri = URI.parse(url) proxy_host = 'some.host' proxy_port = 88 proxy_user = 'login' proxy_pass = 'passwd' send_exception(:proxy_host => proxy_host, :proxy_port => proxy_port, :proxy_user => proxy_user, :proxy_pass => proxy_pass) assert_received(http, :post) do |expect| expect.with(uri.path, anything, Airbrake::HEADERS) end assert_received(Net::HTTP, :Proxy) do |expect| expect.with(proxy_host, proxy_port, proxy_user, proxy_pass) end end should "return the created group's id on successful posting" do http = stub_http(:body => '3799307') assert_equal "3799307", send_exception(:secure => false) end should "return nil on failed posting" do http = stub_http http.stubs(:post).raises(Errno::ECONNREFUSED) assert_equal nil, send_exception(:secure => false) end should "not fail when posting and a timeout exception occurs" do http = stub_http http.stubs(:post).raises(TimeoutError) assert_nothing_thrown do send_exception(:secure => false) end end should "not fail when posting and a connection refused exception occurs" do http = stub_http http.stubs(:post).raises(Errno::ECONNREFUSED) assert_nothing_thrown do send_exception(:secure => false) end end should "not fail when posting any http exception occurs" do http = stub_http Airbrake::Sender::HTTP_ERRORS.each do |error| http.stubs(:post).raises(error) assert_nothing_thrown do send_exception(:secure => false) end end end should "post to the right url for non-ssl" do http = stub_http url = "http://airbrake.io:80#{Airbrake::Sender::NOTICES_URI}" uri = URI.parse(url) send_exception(:secure => false) assert_received(http, :post) {|expect| expect.with(uri.path, anything, Airbrake::HEADERS) } end should "post to the right path for ssl" do http = stub_http send_exception(:secure => true) assert_received(http, :post) {|expect| expect.with(Airbrake::Sender::NOTICES_URI, anything, Airbrake::HEADERS) } end should "verify the SSL peer when the use_ssl option is set to true" do url = "https://airbrake.io#{Airbrake::Sender::NOTICES_URI}" uri = URI.parse(url) real_http = Net::HTTP.new(uri.host, uri.port) real_http.stubs(:post => nil) proxy = stub(:new => real_http) Net::HTTP.stubs(:Proxy => proxy) File.stubs(:exist?).with(OpenSSL::X509::DEFAULT_CERT_FILE).returns(false) send_exception(:secure => true) assert(real_http.use_ssl?) assert_equal(OpenSSL::SSL::VERIFY_PEER, real_http.verify_mode) assert_equal(Airbrake::Sender.local_cert_path, real_http.ca_file) end should "verify the SSL peer when the use_ssl option is set to true and the default cert exists" do url = "https://airbrake.io#{Airbrake::Sender::NOTICES_URI}" uri = URI.parse(url) real_http = Net::HTTP.new(uri.host, uri.port) real_http.stubs(:post => nil) proxy = stub(:new => real_http) Net::HTTP.stubs(:Proxy => proxy) File.stubs(:exist?).with(OpenSSL::X509::DEFAULT_CERT_FILE).returns(true) send_exception(:secure => true) assert(real_http.use_ssl?) assert_equal(OpenSSL::SSL::VERIFY_PEER, real_http.verify_mode) assert_equal(OpenSSL::X509::DEFAULT_CERT_FILE, real_http.ca_file) end should "default the open timeout to 2 seconds" do http = stub_http send_exception assert_received(http, :open_timeout=) {|expect| expect.with(2) } end should "default the read timeout to 5 seconds" do http = stub_http send_exception assert_received(http, :read_timeout=) {|expect| expect.with(5) } end should "allow override of the open timeout" do http = stub_http send_exception(:http_open_timeout => 4) assert_received(http, :open_timeout=) {|expect| expect.with(4) } end should "allow override of the read timeout" do http = stub_http send_exception(:http_read_timeout => 10) assert_received(http, :read_timeout=) {|expect| expect.with(10) } end should "connect to the right port for ssl" do stub_http send_exception(:secure => true) assert_received(Net::HTTP, :new) {|expect| expect.with("airbrake.io", 443) } end should "connect to the right port for non-ssl" do stub_http send_exception(:secure => false) assert_received(Net::HTTP, :new) {|expect| expect.with("airbrake.io", 80) } end should "use ssl if secure" do stub_http send_exception(:secure => true, :host => 'example.org') assert_received(Net::HTTP, :new) {|expect| expect.with('example.org', 443) } end should "not use ssl if not secure" do stub_http send_exception(:secure => false, :host => 'example.org') assert_received(Net::HTTP, :new) {|expect| expect.with('example.org', 80) } end end