1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
|
require File.expand_path('../spec_helper', __FILE__)
require File.expand_path('../fixtures/proc', __FILE__)
load_extension("proc")
describe "C-API Proc function" do
before :each do
@p = CApiProcSpecs.new
@prc = @p.rb_proc_new
end
describe "rb_proc_new" do
it "returns a new valid Proc" do
@prc.kind_of?(Proc).should == true
end
ruby_version_is "" ... "1.8.7" do
it "calls the C function wrapped by the Proc instance when sent #call" do
@prc.call(:foo_bar).should == "[:foo_bar]"
@prc.call([:foo, :bar]).should == "[:foo, :bar]"
end
it "calls the C function wrapped by the Proc instance when sent #[]" do
@prc[:foo_bar].should == "[:foo_bar]"
@prc[[:foo, :bar]].should == "[:foo, :bar]"
end
it "returns a Proc instance correctly described in #inspect with source location" do
@prc.inspect.should =~ /^#<Proc:0x[0-9a-f]*@.*:\d+>$/
end
end
ruby_version_is "1.8.7" ... "1.9" do
it "calls the C function wrapped by the Proc instance when sent #call" do
@prc.call(:foo_bar).should == "[:foo_bar]"
@prc.call([:foo, :bar]).should == "[[:foo, :bar]]"
end
it "calls the C function wrapped by the Proc instance when sent #[]" do
@prc[:foo_bar].should == "[:foo_bar]"
@prc[[:foo, :bar]].should == "[[:foo, :bar]]"
end
end
ruby_version_is "1.9" do
it "calls the C function wrapped by the Proc instance when sent #call" do
@prc.call(:foo_bar).should == ":foo_bar"
@prc.call([:foo, :bar]).should == "[:foo, :bar]"
end
it "calls the C function wrapped by the Proc instance when sent #[]" do
@prc[:foo_bar].should == ":foo_bar"
@prc[[:foo, :bar]].should == "[:foo, :bar]"
end
it "returns a Proc instance correctly described in #inspect without source location" do
@prc.inspect.should =~ /^#<Proc:([^ :@]*?)>$/
end
end
it "returns a Proc instance with #aricy == -1" do
@prc.arity.should == -1
end
it "shouldn't be equal to another one" do
@prc.should_not == @p.rb_proc_new
end
ruby_version_is "1.9" do
it "returns a Proc instance with #source_location == nil" do
@prc.source_location.should == nil
end
end
end
end
describe "C-API when calling Proc.new from a C function" do
before :each do
@p = CApiProcSpecs.new
end
# In the scenarios below: X -> Y means execution context X called to Y.
# For example: Ruby -> C means a Ruby code called a C function.
#
# X -> Y <- X -> Z means exection context X called Y which returned to X,
# then X called Z.
# For example: C -> Ruby <- C -> Ruby means a C function called into Ruby
# code which returned to C, then C called into Ruby code again.
# Ruby -> C -> rb_funcall(Proc.new)
it "returns the Proc passed by the Ruby code calling the C function" do
prc = @p.rb_Proc_new(0) { :called }
prc.call.should == :called
end
# Ruby -> C -> Ruby <- C -> rb_funcall(Proc.new)
it "returns the Proc passed to the Ruby method when the C function calls other Ruby methods before calling Proc.new" do
prc = @p.rb_Proc_new(1) { :called }
prc.call.should == :called
end
# Ruby -> C -> Ruby -> Proc.new
it "raises an ArgumentError when the C function calls a Ruby method that calls Proc.new" do
def @p.Proc_new() Proc.new end
lambda { @p.rb_Proc_new(2) { :called } }.should raise_error(ArgumentError)
end
# Ruby -> C -> Ruby -> C -> rb_funcall(Proc.new)
it "raises an ArgumentError when the C function calls a Ruby method and that method calls a C function that calls Proc.new" do
def @p.redispatch() rb_Proc_new(0) end
lambda { @p.rb_Proc_new(3) { :called } }.should raise_error(ArgumentError)
end
# Ruby -> C -> Ruby -> C (with new block) -> rb_funcall(Proc.new)
it "returns the most recent Proc passed when the Ruby method called the C function" do
prc = @p.rb_Proc_new(4) { :called }
prc.call.should == :calling_with_block
end
# Ruby -> C -> Ruby -> C (with new block) <- Ruby <- C -> # rb_funcall(Proc.new)
it "returns the Proc passed from the original Ruby call to the C function" do
prc = @p.rb_Proc_new(5) { :called }
prc.call.should == :called
end
# Ruby -> C -> Ruby -> block_given?
it "returns false from block_given? in a Ruby method called by the C function" do
@p.rb_Proc_new(6).should be_false
end
end
|