2008-05-18 06:22:34 +02:00
|
|
|
require 'abstract_unit'
|
2007-02-09 09:04:31 +01:00
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
uses_mocha 'fcgi dispatcher tests' do
|
2007-02-09 09:04:31 +01:00
|
|
|
|
|
|
|
require 'fcgi_handler'
|
2007-01-22 14:43:50 +01:00
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
module ActionController; module Routing; module Routes; end end end
|
2007-01-22 14:43:50 +01:00
|
|
|
|
|
|
|
class RailsFCGIHandlerTest < Test::Unit::TestCase
|
|
|
|
def setup
|
|
|
|
@log = StringIO.new
|
|
|
|
@handler = RailsFCGIHandler.new(@log)
|
|
|
|
end
|
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_process_restart
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.stubs(:each_cgi).yields(cgi)
|
|
|
|
|
|
|
|
@handler.expects(:process_request).once
|
|
|
|
@handler.expects(:dispatcher_error).never
|
|
|
|
|
|
|
|
@handler.expects(:when_ready).returns(:restart)
|
|
|
|
@handler.expects(:close_connection).with(cgi)
|
|
|
|
@handler.expects(:reload!).never
|
2007-02-09 09:04:31 +01:00
|
|
|
@handler.expects(:restart!)
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
@handler.process!
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_process_exit
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.stubs(:each_cgi).yields(cgi)
|
|
|
|
|
|
|
|
@handler.expects(:process_request).once
|
|
|
|
@handler.expects(:dispatcher_error).never
|
|
|
|
|
|
|
|
@handler.expects(:when_ready).returns(:exit)
|
|
|
|
@handler.expects(:close_connection).with(cgi)
|
|
|
|
@handler.expects(:reload!).never
|
|
|
|
@handler.expects(:restart!).never
|
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
@handler.process!
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_process_with_system_exit_exception
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.stubs(:each_cgi).yields(cgi)
|
|
|
|
|
|
|
|
@handler.expects(:process_request).once.raises(SystemExit)
|
|
|
|
@handler.stubs(:dispatcher_log)
|
|
|
|
@handler.expects(:dispatcher_log).with(:info, regexp_matches(/^stopping/))
|
|
|
|
@handler.expects(:dispatcher_error).never
|
|
|
|
|
|
|
|
@handler.expects(:when_ready).never
|
|
|
|
@handler.expects(:close_connection).never
|
|
|
|
@handler.expects(:reload!).never
|
|
|
|
@handler.expects(:restart!).never
|
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
@handler.process!
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2008-05-18 06:22:34 +02:00
|
|
|
def test_restart_handler_outside_request
|
2007-02-09 09:04:31 +01:00
|
|
|
@handler.expects(:dispatcher_log).with(:info, "asked to restart ASAP")
|
2008-05-18 06:22:34 +02:00
|
|
|
@handler.expects(:restart!).once
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
@handler.send(:restart_handler, nil)
|
2008-05-18 06:22:34 +02:00
|
|
|
assert_equal nil, @handler.when_ready
|
2007-02-09 09:04:31 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_install_signal_handler_should_log_on_bad_signal
|
|
|
|
@handler.stubs(:trap).raises(ArgumentError)
|
|
|
|
|
|
|
|
@handler.expects(:dispatcher_log).with(:warn, "Ignoring unsupported signal CHEESECAKE.")
|
|
|
|
@handler.send(:install_signal_handler, "CHEESECAKE", nil)
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_reload
|
|
|
|
@handler.expects(:restore!)
|
|
|
|
@handler.expects(:dispatcher_log).with(:info, "reloaded")
|
|
|
|
|
|
|
|
@handler.send(:reload!)
|
|
|
|
assert_nil @handler.when_ready
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_reload_runs_gc_when_gc_request_period_set
|
|
|
|
@handler.expects(:run_gc!)
|
|
|
|
@handler.expects(:restore!)
|
|
|
|
@handler.expects(:dispatcher_log).with(:info, "reloaded")
|
|
|
|
@handler.gc_request_period = 10
|
|
|
|
@handler.send(:reload!)
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_reload_doesnt_run_gc_if_gc_request_period_isnt_set
|
|
|
|
@handler.expects(:run_gc!).never
|
|
|
|
@handler.expects(:restore!)
|
|
|
|
@handler.expects(:dispatcher_log).with(:info, "reloaded")
|
|
|
|
@handler.send(:reload!)
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_restart!
|
|
|
|
@handler.expects(:dispatcher_log).with(:info, "restarted")
|
2007-12-21 08:48:59 +01:00
|
|
|
@handler.expects(:exec).returns('restarted')
|
|
|
|
assert_equal 'restarted', @handler.send(:restart!)
|
2007-02-09 09:04:31 +01:00
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-02-09 09:04:31 +01:00
|
|
|
def test_restore!
|
|
|
|
$".expects(:replace)
|
|
|
|
Dispatcher.expects(:reset_application!)
|
|
|
|
ActionController::Routing::Routes.expects(:reload)
|
|
|
|
@handler.send(:restore!)
|
|
|
|
end
|
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
def test_uninterrupted_processing
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).yields(cgi)
|
|
|
|
@handler.expects(:process_request).with(cgi)
|
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
@handler.process!
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
assert_nil @handler.when_ready
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
class RailsFCGIHandlerSignalsTest < Test::Unit::TestCase
|
2008-05-18 06:22:34 +02:00
|
|
|
class ::RailsFCGIHandler
|
|
|
|
attr_accessor :signal
|
|
|
|
alias_method :old_gc_countdown, :gc_countdown
|
|
|
|
def gc_countdown
|
|
|
|
signal ? Process.kill(signal, $$) : old_gc_countdown
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class ::Dispatcher
|
|
|
|
class << self
|
|
|
|
attr_accessor :signal
|
|
|
|
alias_method :old_dispatch, :dispatch
|
|
|
|
def dispatch(cgi)
|
|
|
|
signal ? Process.kill(signal, $$) : old_dispatch
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
def setup
|
|
|
|
@log = StringIO.new
|
|
|
|
@handler = RailsFCGIHandler.new(@log)
|
|
|
|
end
|
2007-01-22 14:43:50 +01:00
|
|
|
|
|
|
|
def test_interrupted_via_HUP_when_not_in_request
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
2008-05-18 06:22:34 +02:00
|
|
|
@handler.expects(:signal).times(2).returns('HUP')
|
2007-12-21 08:48:59 +01:00
|
|
|
|
|
|
|
@handler.expects(:reload!).once
|
|
|
|
@handler.expects(:close_connection).never
|
|
|
|
@handler.expects(:exit).never
|
|
|
|
|
|
|
|
@handler.process!
|
2007-01-22 14:43:50 +01:00
|
|
|
assert_equal :reload, @handler.when_ready
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_interrupted_via_HUP_when_in_request
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
2008-05-18 06:22:34 +02:00
|
|
|
Dispatcher.expects(:signal).times(2).returns('HUP')
|
2007-12-21 08:48:59 +01:00
|
|
|
|
|
|
|
@handler.expects(:reload!).once
|
|
|
|
@handler.expects(:close_connection).never
|
|
|
|
@handler.expects(:exit).never
|
|
|
|
|
|
|
|
@handler.process!
|
2007-01-22 14:43:50 +01:00
|
|
|
assert_equal :reload, @handler.when_ready
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_interrupted_via_USR1_when_not_in_request
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
2008-05-18 06:22:34 +02:00
|
|
|
@handler.expects(:signal).times(2).returns('USR1')
|
2007-12-21 08:48:59 +01:00
|
|
|
@handler.expects(:exit_handler).never
|
|
|
|
|
|
|
|
@handler.expects(:reload!).never
|
|
|
|
@handler.expects(:close_connection).with(cgi).once
|
|
|
|
@handler.expects(:exit).never
|
|
|
|
|
|
|
|
@handler.process!
|
|
|
|
assert_nil @handler.when_ready
|
2007-01-22 14:43:50 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def test_interrupted_via_USR1_when_in_request
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
2008-05-18 06:22:34 +02:00
|
|
|
Dispatcher.expects(:signal).times(2).returns('USR1')
|
2007-12-21 08:48:59 +01:00
|
|
|
|
|
|
|
@handler.expects(:reload!).never
|
|
|
|
@handler.expects(:close_connection).with(cgi).once
|
|
|
|
@handler.expects(:exit).never
|
|
|
|
|
|
|
|
@handler.process!
|
2007-01-22 14:43:50 +01:00
|
|
|
assert_equal :exit, @handler.when_ready
|
|
|
|
end
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2008-05-18 06:22:34 +02:00
|
|
|
def test_restart_via_USR2_when_in_request
|
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
|
|
|
@handler.expects(:signal).times(2).returns('USR2')
|
|
|
|
@handler.expects(:exit_handler).never
|
|
|
|
|
|
|
|
@handler.expects(:reload!).never
|
|
|
|
@handler.expects(:close_connection).with(cgi).once
|
|
|
|
@handler.expects(:exit).never
|
|
|
|
@handler.expects(:restart!).once
|
|
|
|
|
|
|
|
@handler.process!
|
|
|
|
assert_equal :restart, @handler.when_ready
|
|
|
|
end
|
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
def test_interrupted_via_TERM
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
2008-05-18 06:22:34 +02:00
|
|
|
Dispatcher.expects(:signal).times(2).returns('TERM')
|
2007-12-21 08:48:59 +01:00
|
|
|
|
|
|
|
@handler.expects(:reload!).never
|
|
|
|
@handler.expects(:close_connection).never
|
|
|
|
|
|
|
|
@handler.process!
|
2007-01-22 14:43:50 +01:00
|
|
|
assert_nil @handler.when_ready
|
|
|
|
end
|
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
def test_runtime_exception_in_fcgi
|
|
|
|
error = RuntimeError.new('foo')
|
|
|
|
FCGI.expects(:each_cgi).times(2).raises(error)
|
|
|
|
@handler.expects(:dispatcher_error).with(error, regexp_matches(/^retrying/))
|
|
|
|
@handler.expects(:dispatcher_error).with(error, regexp_matches(/^stopping/))
|
|
|
|
@handler.process!
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_runtime_error_in_dispatcher
|
|
|
|
cgi = mock
|
|
|
|
error = RuntimeError.new('foo')
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
|
|
|
Dispatcher.expects(:dispatch).once.with(cgi).raises(error)
|
|
|
|
@handler.expects(:dispatcher_error).with(error, regexp_matches(/^unhandled/))
|
|
|
|
@handler.process!
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_signal_exception_in_fcgi
|
|
|
|
error = SignalException.new('USR2')
|
|
|
|
FCGI.expects(:each_cgi).once.raises(error)
|
|
|
|
@handler.expects(:dispatcher_error).with(error, regexp_matches(/^stopping/))
|
|
|
|
@handler.process!
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_signal_exception_in_dispatcher
|
|
|
|
cgi = mock
|
|
|
|
error = SignalException.new('USR2')
|
|
|
|
FCGI.expects(:each_cgi).once.yields(cgi)
|
|
|
|
Dispatcher.expects(:dispatch).once.with(cgi).raises(error)
|
|
|
|
@handler.expects(:dispatcher_error).with(error, regexp_matches(/^stopping/))
|
|
|
|
@handler.process!
|
2007-01-22 14:43:50 +01:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
class RailsFCGIHandlerPeriodicGCTest < Test::Unit::TestCase
|
|
|
|
def setup
|
|
|
|
@log = StringIO.new
|
|
|
|
end
|
|
|
|
|
|
|
|
def teardown
|
|
|
|
GC.enable
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_normal_gc
|
|
|
|
@handler = RailsFCGIHandler.new(@log)
|
|
|
|
assert_nil @handler.gc_request_period
|
|
|
|
|
|
|
|
# When GC is enabled, GC.disable disables and returns false.
|
|
|
|
assert_equal false, GC.disable
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_periodic_gc
|
|
|
|
@handler = RailsFCGIHandler.new(@log, 10)
|
|
|
|
assert_equal 10, @handler.gc_request_period
|
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
cgi = mock
|
|
|
|
FCGI.expects(:each_cgi).times(10).yields(cgi)
|
|
|
|
Dispatcher.expects(:dispatch).times(10).with(cgi)
|
2007-01-22 14:43:50 +01:00
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
@handler.expects(:run_gc!).never
|
|
|
|
9.times { @handler.process! }
|
|
|
|
@handler.expects(:run_gc!).once
|
2007-01-22 14:43:50 +01:00
|
|
|
@handler.process!
|
|
|
|
|
|
|
|
assert_nil @handler.when_ready
|
|
|
|
end
|
|
|
|
end
|
2007-02-09 09:04:31 +01:00
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
end # uses_mocha
|