require 'test/unit' require 'timeout' require 'tempfile' require_relative 'envutil' class TestSignal < Test::Unit::TestCase def test_signal begin x = 0 oldtrap = Signal.trap(:INT) {|sig| x = 2 } Process.kill :INT, Process.pid 10.times do break if 2 == x sleep 0.1 end assert_equal 2, x Signal.trap(:INT) { raise "Interrupt" } ex = assert_raise(RuntimeError) { Process.kill :INT, Process.pid sleep 0.1 } assert_kind_of Exception, ex assert_match(/Interrupt/, ex.message) ensure Signal.trap :INT, oldtrap if oldtrap end end if Process.respond_to?(:kill) def test_signal_process_group bug4362 = '[ruby-dev:43169]' assert_nothing_raised(bug4362) do pid = Process.spawn(EnvUtil.rubybin, '-e', 'sleep 10', :pgroup => true) Process.kill(:"-TERM", pid) Process.waitpid(pid) assert_equal(true, $?.signaled?) assert_equal(Signal.list["TERM"], $?.termsig) end end if Process.respond_to?(:kill) and Process.respond_to?(:pgroup) # for mswin32 def test_exit_action if Signal.list[sig = "USR1"] term = :TERM else sig = "INT" term = :KILL end IO.popen([EnvUtil.rubybin, '-e', <<-"End"], 'r+') do |io| Signal.trap(:#{sig}, "EXIT") STDOUT.syswrite("a") Thread.start { sleep(2) } STDIN.sysread(4096) End pid = io.pid io.sysread(1) sleep 0.1 assert_nothing_raised("[ruby-dev:26128]") { Process.kill(term, pid) begin Timeout.timeout(3) { Process.waitpid pid } rescue Timeout::Error if term Process.kill(term, pid) term = (:KILL if term != :KILL) retry end raise end } end end if Process.respond_to?(:kill) def test_invalid_signal_name assert_raise(ArgumentError) { Process.kill(:XXXXXXXXXX, $$) } end if Process.respond_to?(:kill) def test_signal_exception assert_raise(ArgumentError) { SignalException.new } assert_raise(ArgumentError) { SignalException.new(-1) } assert_raise(ArgumentError) { SignalException.new(:XXXXXXXXXX) } Signal.list.each do |signm, signo| next if signm == "EXIT" assert_equal(SignalException.new(signm).signo, signo) assert_equal(SignalException.new(signm.to_sym).signo, signo) assert_equal(SignalException.new(signo).signo, signo) end end def test_interrupt assert_raise(Interrupt) { raise Interrupt.new } end def test_signal2 begin x = false oldtrap = Signal.trap(:INT) {|sig| x = true } GC.start assert_raise(ArgumentError) { Process.kill } Timeout.timeout(10) do x = false Process.kill(SignalException.new(:INT).signo, $$) sleep(0.01) until x x = false Process.kill("INT", $$) sleep(0.01) until x x = false Process.kill("SIGINT", $$) sleep(0.01) until x x = false o = Object.new def o.to_str; "SIGINT"; end Process.kill(o, $$) sleep(0.01) until x end assert_raise(ArgumentError) { Process.kill(Object.new, $$) } ensure Signal.trap(:INT, oldtrap) if oldtrap end end if Process.respond_to?(:kill) def test_trap begin oldtrap = Signal.trap(:INT) {|sig| } assert_raise(ArgumentError) { Signal.trap } assert_raise(SecurityError) do s = proc {}.taint Signal.trap(:INT, s) end # FIXME! Signal.trap(:INT, nil) Signal.trap(:INT, "") Signal.trap(:INT, "SIG_IGN") Signal.trap(:INT, "IGNORE") Signal.trap(:INT, "SIG_DFL") Signal.trap(:INT, "SYSTEM_DEFAULT") Signal.trap(:INT, "EXIT") Signal.trap(:INT, "xxxxxx") Signal.trap(:INT, "xxxx") Signal.trap(SignalException.new(:INT).signo, "SIG_DFL") assert_raise(ArgumentError) { Signal.trap(-1, "xxxx") } o = Object.new def o.to_str; "SIGINT"; end Signal.trap(o, "SIG_DFL") assert_raise(ArgumentError) { Signal.trap("XXXXXXXXXX", "SIG_DFL") } ensure Signal.trap(:INT, oldtrap) if oldtrap end end if Process.respond_to?(:kill) def test_kill_immediately_before_termination Signal.list[sig = "USR1"] or sig = "INT" assert_in_out_err(["-e", <<-"end;"], "", %w"foo") Signal.trap(:#{sig}) { STDOUT.syswrite("foo") } Process.kill :#{sig}, $$ end; end if Process.respond_to?(:kill) def test_signal_requiring t = Tempfile.new(%w"require_ensure_test .rb") t.puts "sleep" t.close error = IO.popen([EnvUtil.rubybin, "-e", < File::NULL]) do |child| trap(:INT, "DEFAULT") th = Thread.new do begin require ARGV[0] ensure err = $! ? [$!, $!.backtrace] : $! Marshal.dump(err, STDOUT) STDOUT.flush end end Thread.pass while th.running? Process.kill(:INT, $$) th.join EOS Marshal.load(child) end t.close! assert_nil(error) end if Process.respond_to?(:kill) def test_reserved_signal assert_raise(ArgumentError) { Signal.trap(:SEGV) {} } assert_raise(ArgumentError) { Signal.trap(:BUS) {} } assert_raise(ArgumentError) { Signal.trap(:ILL) {} } assert_raise(ArgumentError) { Signal.trap(:FPE) {} } assert_raise(ArgumentError) { Signal.trap(:VTALRM) {} } end def test_signame 10.times do IO.popen([EnvUtil.rubybin, "-e", < File::NULL]) do |child| Signal.trap("INT") do |signo| signame = Signal.signame(signo) Marshal.dump(signame, STDOUT) STDOUT.flush exit 0 end Process.kill("INT", $$) sleep 1 # wait signal deliver EOS signame = Marshal.load(child) assert_equal(signame, "INT") end end end if Process.respond_to?(:kill) def test_trap_puts assert_in_out_err([], <<-INPUT, ["a"*10000], []) Signal.trap(:INT) { # for enable internal io mutex STDOUT.sync = false # larger than internal io buffer print "a"*10000 } Process.kill :INT, $$ sleep 0.1 INPUT end if Process.respond_to?(:kill) def test_hup_me # [Bug #7951] [ruby-core:52864] # This is MRI specific spec. ruby has no guarantee # that signal will be deliverd synchronously. # This ugly workaround was introduced to don't break # compatibility against silly example codes. assert_raise(SignalException) { Process.kill('HUP', Process.pid) } bug8137 = '[ruby-dev:47182] [Bug #8137]' assert_nothing_raised(bug8137) { Timeout.timeout(1) { Process.kill(0, Process.pid) } } end if Process.respond_to?(:kill) and Signal.list.key?('HUP') end