summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNobuyoshi Nakada <nobu@ruby-lang.org>2021-02-12 12:21:49 +0900
committerNobuyoshi Nakada <nobu@ruby-lang.org>2021-02-12 12:21:49 +0900
commit1f0e0dfb228fd14b3f6687539ba274ba6a2d1643 (patch)
tree0d06f1dcea4a1d63004bf51f07408034dbf58f5e
parente8b210542b7d290d6c3feeff85221ed7303d2aab (diff)
Thread::Queue.new should accept an Enumerable [Feature #17327]
Enumerable implements #to_a but not #to_array.
-rw-r--r--spec/ruby/core/queue/initialize_spec.rb14
-rw-r--r--test/ruby/test_thread_queue.rb22
-rw-r--r--thread_sync.c2
3 files changed, 27 insertions, 11 deletions
diff --git a/spec/ruby/core/queue/initialize_spec.rb b/spec/ruby/core/queue/initialize_spec.rb
index 5cecfe51a5..83c7e595fe 100644
--- a/spec/ruby/core/queue/initialize_spec.rb
+++ b/spec/ruby/core/queue/initialize_spec.rb
@@ -18,9 +18,9 @@ describe "Queue#initialize" do
q.should.empty?
end
- it "uses #to_ary on the provided Enumerable" do
+ it "uses #to_a on the provided Enumerable" do
enumerable = MockObject.new('mock-enumerable')
- enumerable.should_receive(:to_ary).and_return([1, 2, 3])
+ enumerable.should_receive(:to_a).and_return([1, 2, 3])
q = Queue.new(enumerable)
q.size.should == 3
q.should_not.empty?
@@ -30,15 +30,9 @@ describe "Queue#initialize" do
q.should.empty?
end
- it "raises if the provided Enumerable does not respond to #to_ary" do
+ it "raises if the provided Enumerable does not respond to #to_a" do
enumerable = MockObject.new('mock-enumerable')
- -> { Queue.new(enumerable) }.should raise_error(TypeError, "no implicit conversion of MockObject into Array")
- end
-
- it "raises if the provided Enumerable #to_ary does not return an Array" do
- enumerable = MockObject.new('mock-enumerable')
- enumerable.should_receive(:to_ary).and_return(14)
- -> { Queue.new(enumerable) }.should raise_error(TypeError, "can't convert MockObject to Array (MockObject#to_ary gives Integer)")
+ -> { Queue.new(enumerable) }.should raise_error(TypeError, "can't convert MockObject into Array")
end
end
end
diff --git a/test/ruby/test_thread_queue.rb b/test/ruby/test_thread_queue.rb
index b0fa878814..6185abff9f 100644
--- a/test/ruby/test_thread_queue.rb
+++ b/test/ruby/test_thread_queue.rb
@@ -54,6 +54,28 @@ class TestThreadQueue < Test::Unit::TestCase
assert_equal 0, to_workers.size
end
+ def test_queue_initialize
+ e = Class.new do
+ include Enumerable
+ def initialize(list) @list = list end
+ def each(&block) @list.each(&block) end
+ end
+
+ all_assertions_foreach(nil,
+ [Array, "Array"],
+ [e, "Enumerable"],
+ [Struct.new(:to_a), "Array-like"],
+ ) do |a, type|
+ q = Queue.new(a.new([1,2,3]))
+ assert_equal(3, q.size, type)
+ assert_not_predicate(q, :empty?, type)
+ assert_equal(1, q.pop, type)
+ assert_equal(2, q.pop, type)
+ assert_equal(3, q.pop, type)
+ assert_predicate(q, :empty?, type)
+ end
+ end
+
def test_sized_queue_initialize
q = SizedQueue.new(1)
assert_equal 1, q.max
diff --git a/thread_sync.c b/thread_sync.c
index 131ace2fda..94e6340b16 100644
--- a/thread_sync.c
+++ b/thread_sync.c
@@ -858,7 +858,7 @@ rb_queue_initialize(int argc, VALUE *argv, VALUE self)
list_head_init(queue_waitq(q));
rb_scan_args(argc, argv, "01", &initial);
if (argc == 1) {
- rb_ary_concat(q->que, rb_convert_type(initial, T_ARRAY, "Array", "to_ary"));
+ rb_ary_concat(q->que, rb_to_array(initial));
}
return self;
}