blob: 163d0c2ce73148216140ce431234da59c149bf34 [file] [log] [blame]
#
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied. See the License for the
# specific language governing permissions and limitations
# under the License.
#
require 'spec_helper'
shared_examples_for "a socket" do
it "should open a socket" do
expect(@socket.open).to eq(@handle)
end
it "should be open whenever it has a handle" do
expect(@socket).not_to be_open
@socket.open
expect(@socket).to be_open
@socket.handle = nil
expect(@socket).not_to be_open
@socket.handle = @handle
@socket.close
expect(@socket).not_to be_open
end
it "should write data to the handle" do
@socket.open
expect(@handle).to receive(:write).with("foobar")
@socket.write("foobar")
expect(@handle).to receive(:write).with("fail").and_raise(StandardError)
expect { @socket.write("fail") }.to raise_error(Thrift::TransportException) { |e| expect(e.type).to eq(Thrift::TransportException::NOT_OPEN) }
end
it "should raise an error when it cannot read from the handle" do
@socket.open
expect(@handle).to receive(:readpartial).with(17).and_raise(StandardError)
expect { @socket.read(17) }.to raise_error(Thrift::TransportException) { |e| expect(e.type).to eq(Thrift::TransportException::NOT_OPEN) }
end
it "should return the data read when reading from the handle works" do
@socket.open
expect(@handle).to receive(:readpartial).with(17).and_return("test data")
expect(@socket.read(17)).to eq("test data")
end
it "should declare itself as closed when it has an error" do
@socket.open
expect(@handle).to receive(:write).with("fail").and_raise(StandardError)
expect(@socket).to be_open
expect { @socket.write("fail") }.to raise_error(Thrift::TransportException) { |e| expect(e.type).to eq(Thrift::TransportException::NOT_OPEN) }
expect(@socket).not_to be_open
end
it "should raise an error when the stream is closed" do
@socket.open
allow(@handle).to receive(:closed?).and_return(true)
expect(@socket).not_to be_open
expect { @socket.write("fail") }.to raise_error(Thrift::TransportException, "closed stream") { |e| expect(e.type).to eq(Thrift::TransportException::NOT_OPEN) }
expect { @socket.read(10) }.to raise_error(Thrift::TransportException, "closed stream") { |e| expect(e.type).to eq(Thrift::TransportException::NOT_OPEN) }
end
it "should support the timeout accessor for read" do
@socket.timeout = 3
@socket.open
expect(@handle).to receive(:read_nonblock).with(17).and_raise(IO::EAGAINWaitReadable)
expect(IO).to receive(:select) do |rd, wr, err, timeout|
expect(rd).to eq([@handle])
expect(wr).to be_nil
expect(err).to be_nil
expect(timeout).to be > 0
expect(timeout).to be <= 3
[[@handle], [], []]
end
expect(@handle).to receive(:read_nonblock).with(17).and_return("test data")
expect(@socket.read(17)).to eq("test data")
end
it "should support the timeout accessor for write" do
@socket.timeout = 3
@socket.open
write_calls = 0
expect(@handle).to receive(:write_nonblock).exactly(3).times do |chunk|
write_calls += 1
case write_calls
when 1
expect(chunk).to eq("test data")
raise IO::EAGAINWaitWritable
when 2
expect(chunk).to eq("test data")
4
when 3
expect(chunk).to eq(" data")
5
end
end
expect(IO).to receive(:select) do |rd, wr, err, timeout|
expect(rd).to be_nil
expect(wr).to eq([@handle])
expect(err).to be_nil
expect(timeout).to be > 0
expect(timeout).to be <= 3
[[], [@handle], []]
end
expect(@socket.write("test data")).to eq(9)
end
it "should raise an error when read times out" do
@socket.timeout = 0.5
@socket.open
expect(@handle).to receive(:read_nonblock).with(17).and_raise(IO::EAGAINWaitReadable)
expect(IO).to receive(:select).once { sleep(0.6); nil }
expect { @socket.read(17) }.to raise_error(Thrift::TransportException) { |e| expect(e.type).to eq(Thrift::TransportException::TIMED_OUT) }
end
it "should raise an error when write times out" do
@socket.timeout = 0.5
@socket.open
expect(@handle).to receive(:write_nonblock).with("test data").and_raise(IO::EAGAINWaitWritable)
expect(IO).to receive(:select).once { sleep(0.6); nil }
expect { @socket.write("test data") }.to raise_error(Thrift::TransportException) { |e| expect(e.type).to eq(Thrift::TransportException::TIMED_OUT) }
end
it "should read buffered SSL data without waiting on the raw socket again" do
@socket.timeout = 1
@socket.open
expect(@handle).to receive(:read_nonblock).with(4).ordered.and_raise(IO::EAGAINWaitReadable)
expect(IO).to receive(:select).once.ordered do |rd, wr, err, timeout|
expect(rd).to eq([@handle])
expect(wr).to be_nil
expect(err).to be_nil
expect(timeout).to be > 0
expect(timeout).to be <= 1
[[@handle], [], []]
end
expect(@handle).to receive(:read_nonblock).with(4).ordered.and_return("ABCD")
expect(@handle).to receive(:read_nonblock).with(5).ordered.and_return("12345")
expect(@socket.read(4)).to eq("ABCD")
expect(@socket.read(5)).to eq("12345")
end
it "should read without timeout using the blocking path" do
@socket.timeout = nil
@socket.open
expect(IO).not_to receive(:select)
expect(@handle).not_to receive(:read_nonblock)
expect(@handle).to receive(:readpartial).with(4).ordered.and_return("ABCD")
expect(@handle).to receive(:readpartial).with(5).ordered.and_return("12345")
expect(@socket.read(4)).to eq("ABCD")
expect(@socket.read(5)).to eq("12345")
end
end