require_relative '../../spec_helper'

describe "Signal.trap" do
  platform_is_not :windows do
    before :each do
      ScratchPad.clear
      @proc = -> {}
      @saved_trap = Signal.trap(:HUP, @proc)
      @hup_number = Signal.list["HUP"]
    end

    after :each do
      Signal.trap(:HUP, @saved_trap) if @saved_trap
    end

    it "returns the previous handler" do
      Signal.trap(:HUP, @saved_trap).should equal(@proc)
    end

    it "accepts a block" do
      done = false

      Signal.trap(:HUP) do |signo|
        signo.should == @hup_number
        ScratchPad.record :block_trap
        done = true
      end

      Process.kill :HUP, Process.pid
      Thread.pass until done

      ScratchPad.recorded.should == :block_trap
    end

    it "accepts a proc" do
      done = false

      handler = -> signo {
        signo.should == @hup_number
        ScratchPad.record :proc_trap
        done = true
      }

      Signal.trap(:HUP, handler)

      Process.kill :HUP, Process.pid
      Thread.pass until done

      ScratchPad.recorded.should == :proc_trap
    end

    it "accepts a method" do
      done = false

      handler_class = Class.new
      hup_number = @hup_number

      handler_class.define_method :handler_method do |signo|
        signo.should == hup_number
        ScratchPad.record :method_trap
        done = true
      end

      handler_method = handler_class.new.method(:handler_method)

      Signal.trap(:HUP, handler_method)

      Process.kill :HUP, Process.pid
      Thread.pass until done

      ScratchPad.recorded.should == :method_trap
    end

    it "accepts anything you can call" do
      done = false

      callable = Object.new
      hup_number = @hup_number

      callable.singleton_class.define_method :call do |signo|
        signo.should == hup_number
        ScratchPad.record :callable_trap
        done = true
      end

      Signal.trap(:HUP, callable)

      Process.kill :HUP, Process.pid
      Thread.pass until done

      ScratchPad.recorded.should == :callable_trap
    end

    it "raises an exception for a non-callable at the point of use" do
      not_callable = Object.new
      Signal.trap(:HUP, not_callable)
      -> {
        Process.kill :HUP, Process.pid
        loop { Thread.pass }
      }.should raise_error(NoMethodError)
    end

    it "accepts a non-callable that becomes callable when used" do
      done = false

      late_callable = Object.new
      hup_number = @hup_number

      Signal.trap(:HUP, late_callable)

      late_callable.singleton_class.define_method :call do |signo|
        signo.should == hup_number
        ScratchPad.record :late_callable_trap
        done = true
      end

      Process.kill :HUP, Process.pid
      Thread.pass until done

      ScratchPad.recorded.should == :late_callable_trap
    end

    it "is possible to create a new Thread when the handler runs" do
      done = false

      Signal.trap(:HUP) do
        thr = Thread.new { }
        thr.join
        ScratchPad.record(thr.group == Thread.main.group)

        done = true
      end

      Process.kill :HUP, Process.pid
      Thread.pass until done

      ScratchPad.recorded.should be_true
    end

    it "registers an handler doing nothing with :IGNORE" do
      Signal.trap :HUP, :IGNORE
      Signal.trap(:HUP, @saved_trap).should == "IGNORE"
    end

    it "can register a new handler after :IGNORE" do
      Signal.trap :HUP, :IGNORE

      done = false
      Signal.trap(:HUP) do
        ScratchPad.record :block_trap
        done = true
      end

      Process.kill(:HUP, Process.pid).should == 1
      Thread.pass until done
      ScratchPad.recorded.should == :block_trap
    end

    it "ignores the signal when passed nil" do
      Signal.trap :HUP, nil
      Signal.trap(:HUP, @saved_trap).should be_nil
    end

    it "accepts :DEFAULT in place of a proc" do
      Signal.trap :HUP, :DEFAULT
      Signal.trap(:HUP, @saved_trap).should == "DEFAULT"
    end

    it "accepts :SIG_DFL in place of a proc" do
      Signal.trap :HUP, :SIG_DFL
      Signal.trap(:HUP, @saved_trap).should == "DEFAULT"
    end

    it "accepts :SIG_IGN in place of a proc" do
      Signal.trap :HUP, :SIG_IGN
      Signal.trap(:HUP, @saved_trap).should == "IGNORE"
    end

    it "accepts :IGNORE in place of a proc" do
      Signal.trap :HUP, :IGNORE
      Signal.trap(:HUP, @saved_trap).should == "IGNORE"
    end

    it "accepts 'SIG_DFL' in place of a proc" do
      Signal.trap :HUP, "SIG_DFL"
      Signal.trap(:HUP, @saved_trap).should == "DEFAULT"
    end

    it "accepts 'DEFAULT' in place of a proc" do
      Signal.trap :HUP, "DEFAULT"
      Signal.trap(:HUP, @saved_trap).should == "DEFAULT"
    end

    it "accepts 'SIG_IGN' in place of a proc" do
      Signal.trap :HUP, "SIG_IGN"
      Signal.trap(:HUP, @saved_trap).should == "IGNORE"
    end

    it "accepts 'IGNORE' in place of a proc" do
      Signal.trap :HUP, "IGNORE"
      Signal.trap(:HUP, @saved_trap).should == "IGNORE"
    end

    it "accepts long names as Strings" do
      Signal.trap "SIGHUP", @proc
      Signal.trap("SIGHUP", @saved_trap).should equal(@proc)
    end

    it "accepts short names as Strings" do
      Signal.trap "HUP", @proc
      Signal.trap("HUP", @saved_trap).should equal(@proc)
    end

    it "accepts long names as Symbols" do
      Signal.trap :SIGHUP, @proc
      Signal.trap(:SIGHUP, @saved_trap).should equal(@proc)
    end

    it "accepts short names as Symbols" do
      Signal.trap :HUP, @proc
      Signal.trap(:HUP, @saved_trap).should equal(@proc)
    end

    it "calls #to_str on an object to convert to a String" do
      obj = mock("signal")
      obj.should_receive(:to_str).exactly(2).times.and_return("HUP")
      Signal.trap obj, @proc
      Signal.trap(obj, @saved_trap).should equal(@proc)
    end

    it "accepts Integer values" do
      hup = Signal.list["HUP"]
      Signal.trap hup, @proc
      Signal.trap(hup, @saved_trap).should equal(@proc)
    end

    it "does not call #to_int on an object to convert to an Integer" do
      obj = mock("signal")
      obj.should_not_receive(:to_int)
      -> { Signal.trap obj, @proc }.should raise_error(ArgumentError, /bad signal type/)
    end

    it "raises ArgumentError when passed unknown signal" do
      -> { Signal.trap(300) { } }.should raise_error(ArgumentError, "invalid signal number (300)")
      -> { Signal.trap("USR10") { } }.should raise_error(ArgumentError, "unsupported signal `SIGUSR10'")
      -> { Signal.trap("SIGUSR10") { } }.should raise_error(ArgumentError, "unsupported signal `SIGUSR10'")
    end

    it "raises ArgumentError when passed signal is not Integer, String or Symbol" do
      -> { Signal.trap(nil) { } }.should raise_error(ArgumentError, "bad signal type NilClass")
      -> { Signal.trap(100.0) { } }.should raise_error(ArgumentError, "bad signal type Float")
      -> { Signal.trap(Rational(100)) { } }.should raise_error(ArgumentError, "bad signal type Rational")
    end

    # See man 2 signal
    %w[KILL STOP].each do |signal|
      it "raises ArgumentError or Errno::EINVAL for SIG#{signal}" do
        -> {
          Signal.trap(signal, -> {})
        }.should raise_error(StandardError) { |e|
          [ArgumentError, Errno::EINVAL].should include(e.class)
          e.message.should =~ /Invalid argument|Signal already used by VM or OS/
        }
      end
    end

    it "allows to register a handler for all known signals, except reserved signals for which it raises ArgumentError" do
      out = ruby_exe(fixture(__FILE__, "trap_all.rb"), args: "2>&1")
      out.should == "OK\n"
      $?.exitstatus.should == 0
    end

    it "returns 'DEFAULT' for the initial SIGINT handler" do
      ruby_exe("print Signal.trap(:INT) { abort }").should == 'DEFAULT'
    end

    it "returns SYSTEM_DEFAULT if passed DEFAULT and no handler was ever set" do
      Signal.trap("PROF", "DEFAULT").should == "SYSTEM_DEFAULT"
    end

    it "accepts 'SYSTEM_DEFAULT' and uses the OS handler for SIGPIPE" do
      code = <<-RUBY
        p Signal.trap('PIPE', 'SYSTEM_DEFAULT')
        r, w = IO.pipe
        r.close
        loop { w.write("a"*1024) }
      RUBY
      out = ruby_exe(code, exit_status: :SIGPIPE)
      status = $?
      out.should == "nil\n"
      status.should.signaled?
    end
  end

  describe "the special EXIT signal code" do
    it "accepts the EXIT code" do
      code = "Signal.trap(:EXIT, proc { print 1 })"
      ruby_exe(code).should == "1"
    end

    it "runs the proc before at_exit handlers" do
      code = "at_exit {print 1}; Signal.trap(:EXIT, proc {print 2}); at_exit {print 3}"
      ruby_exe(code).should == "231"
    end

    it "can unset the handler" do
      code = "Signal.trap(:EXIT, proc { print 1 }); Signal.trap(:EXIT, 'DEFAULT')"
      ruby_exe(code).should == ""
    end
  end

end
