1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00
ruby--ruby/spec/ruby/core/thread/raise_spec.rb
2022-06-26 14:50:14 +02:00

232 lines
5.7 KiB
Ruby

require_relative '../../spec_helper'
require_relative 'fixtures/classes'
require_relative '../../shared/kernel/raise'
describe "Thread#raise" do
it "ignores dead threads and returns nil" do
t = Thread.new { :dead }
Thread.pass while t.alive?
t.raise("Kill the thread").should == nil
t.join
end
end
describe "Thread#raise on a sleeping thread" do
before :each do
ScratchPad.clear
@thr = ThreadSpecs.sleeping_thread
Thread.pass while @thr.status and @thr.status != "sleep"
end
after :each do
@thr.kill
@thr.join
end
it "raises a RuntimeError if no exception class is given" do
@thr.raise
Thread.pass while @thr.status
ScratchPad.recorded.should be_kind_of(RuntimeError)
end
it "raises the given exception" do
@thr.raise Exception
Thread.pass while @thr.status
ScratchPad.recorded.should be_kind_of(Exception)
end
it "raises the given exception with the given message" do
@thr.raise Exception, "get to work"
Thread.pass while @thr.status
ScratchPad.recorded.should be_kind_of(Exception)
ScratchPad.recorded.message.should == "get to work"
end
it "raises the given exception and the backtrace is the one of the interrupted thread" do
@thr.raise Exception
Thread.pass while @thr.status
ScratchPad.recorded.should be_kind_of(Exception)
ScratchPad.recorded.backtrace[0].should include("sleep")
end
it "is captured and raised by Thread#value" do
t = Thread.new do
Thread.current.report_on_exception = false
sleep
end
ThreadSpecs.spin_until_sleeping(t)
t.raise
-> { t.value }.should raise_error(RuntimeError)
end
it "raises a RuntimeError when called with no arguments inside rescue" do
t = Thread.new do
Thread.current.report_on_exception = false
begin
1/0
rescue ZeroDivisionError
sleep
end
end
begin
raise RangeError
rescue
ThreadSpecs.spin_until_sleeping(t)
t.raise
end
-> { t.value }.should raise_error(RuntimeError)
end
it "re-raises a previously rescued exception without overwriting the backtrace" do
t = Thread.new do
-> { # To make sure there is at least one entry in the call stack
begin
sleep
rescue => e
e
end
}.call
end
ThreadSpecs.spin_until_sleeping(t)
begin
initial_raise_line = __LINE__; raise 'raised'
rescue => raised
raise_again_line = __LINE__; t.raise raised
raised_again = t.value
raised_again.backtrace.first.should include("#{__FILE__}:#{initial_raise_line}:")
raised_again.backtrace.first.should_not include("#{__FILE__}:#{raise_again_line}:")
end
end
it "calls #exception in both the caller and in the target thread" do
cls = Class.new(Exception) do
attr_accessor :log
def initialize(*args)
@log = [] # This is shared because the super #exception uses a shallow clone
super
end
def exception(*args)
@log << [self, Thread.current, args]
super
end
end
exc = cls.new
@thr.raise exc, "Thread#raise #exception spec"
@thr.join
ScratchPad.recorded.should.is_a?(cls)
exc.log.should == [
[exc, Thread.current, ["Thread#raise #exception spec"]],
[ScratchPad.recorded, @thr, []]
]
end
end
describe "Thread#raise on a running thread" do
before :each do
ScratchPad.clear
ThreadSpecs.clear_state
@thr = ThreadSpecs.running_thread
Thread.pass until ThreadSpecs.state == :running
end
after :each do
@thr.kill
@thr.join
end
it "raises a RuntimeError if no exception class is given" do
@thr.raise
Thread.pass while @thr.status
ScratchPad.recorded.should be_kind_of(RuntimeError)
end
it "raises the given exception" do
@thr.raise Exception
Thread.pass while @thr.status
ScratchPad.recorded.should be_kind_of(Exception)
end
it "raises the given exception with the given message" do
@thr.raise Exception, "get to work"
Thread.pass while @thr.status
ScratchPad.recorded.should be_kind_of(Exception)
ScratchPad.recorded.message.should == "get to work"
end
it "can go unhandled" do
q = Queue.new
t = Thread.new do
Thread.current.report_on_exception = false
q << true
loop { Thread.pass }
end
q.pop # wait for `report_on_exception = false`.
t.raise
-> { t.value }.should raise_error(RuntimeError)
end
it "raises the given argument even when there is an active exception" do
raised = false
t = Thread.new do
Thread.current.report_on_exception = false
begin
1/0
rescue ZeroDivisionError
raised = true
loop { Thread.pass }
end
end
begin
raise "Create an active exception for the current thread too"
rescue
Thread.pass until raised
t.raise RangeError
-> { t.value }.should raise_error(RangeError)
end
end
it "raises a RuntimeError when called with no arguments inside rescue" do
raised = false
t = Thread.new do
Thread.current.report_on_exception = false
begin
1/0
rescue ZeroDivisionError
raised = true
loop { Thread.pass }
end
end
begin
raise RangeError
rescue
Thread.pass until raised
t.raise
end
-> { t.value }.should raise_error(RuntimeError)
end
end
describe "Thread#raise on same thread" do
it_behaves_like :kernel_raise, :raise, Thread.current
it "raises a RuntimeError when called with no arguments inside rescue" do
t = Thread.new do
Thread.current.report_on_exception = false
begin
1/0
rescue ZeroDivisionError
Thread.current.raise
end
end
-> { t.value }.should raise_error(RuntimeError, '')
end
end