Test that a disconnect after the write request but before the data is an error
This commit is contained in:
@@ -101,7 +101,7 @@ void write_not_zeroes(struct client* client, uint64_t from, int len)
|
|||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
#define DO_READ(dst, len) FATAL_IF_NEGATIVE( \
|
#define DO_READ(dst, len) ERROR_IF_NEGATIVE( \
|
||||||
readloop( \
|
readloop( \
|
||||||
client->socket, \
|
client->socket, \
|
||||||
(dst), \
|
(dst), \
|
||||||
|
@@ -114,7 +114,7 @@ class Environment
|
|||||||
raise "no addr" unless addr
|
raise "no addr" unless addr
|
||||||
raise "no port" unless port
|
raise "no port" unless port
|
||||||
@fake_pid = fork do
|
@fake_pid = fork do
|
||||||
exec fake + " " + addr.to_s + " " + port.to_s
|
exec fake + " " + addr.to_s + " " + port.to_s + " " + @nbd1.pid.to_s
|
||||||
end
|
end
|
||||||
sleep(0.5)
|
sleep(0.5)
|
||||||
end
|
end
|
||||||
|
@@ -9,7 +9,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
client = server.accept( "Timed out waiting for a connection" )
|
client = server.accept( "Timed out waiting for a connection" )
|
||||||
client.write_hello
|
client.write_hello
|
||||||
client.close
|
client.close
|
||||||
|
@@ -10,7 +10,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
client = server.accept( "Timed out waiting for a connection" )
|
client = server.accept( "Timed out waiting for a connection" )
|
||||||
client.write_hello
|
client.write_hello
|
||||||
client.read_request
|
client.read_request
|
||||||
|
@@ -4,7 +4,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
client = server.accept
|
client = server.accept
|
||||||
|
|
||||||
client.write_hello
|
client.write_hello
|
||||||
|
@@ -10,7 +10,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
client = server.accept( "Client didn't make a connection" )
|
client = server.accept( "Client didn't make a connection" )
|
||||||
|
|
||||||
# Sleep for one second past the timeout (a bit of slop in case ruby
|
# Sleep for one second past the timeout (a bit of slop in case ruby
|
||||||
|
@@ -8,7 +8,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
client1 = server.accept( server )
|
client1 = server.accept( server )
|
||||||
client1.write_hello
|
client1.write_hello
|
||||||
client1.read_request
|
client1.read_request
|
||||||
|
@@ -6,7 +6,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
client1 = server.accept
|
client1 = server.accept
|
||||||
|
|
||||||
# Launch a second thread so that we can spot the reconnection attempt
|
# Launch a second thread so that we can spot the reconnection attempt
|
||||||
|
@@ -7,7 +7,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
client = server.accept
|
client = server.accept
|
||||||
|
|
||||||
t = Thread.new do
|
t = Thread.new do
|
||||||
|
@@ -5,7 +5,8 @@
|
|||||||
require 'flexnbd/fake_dest'
|
require 'flexnbd/fake_dest'
|
||||||
include FlexNBD
|
include FlexNBD
|
||||||
|
|
||||||
server = FakeDest.new( *ARGV )
|
addr, port = *ARGV
|
||||||
|
server = FakeDest.new( addr, port )
|
||||||
server.accept.close
|
server.accept.close
|
||||||
|
|
||||||
server.close
|
server.close
|
||||||
|
29
tests/acceptance/fakes/source/close_after_write.rb
Executable file
29
tests/acceptance/fakes/source/close_after_write.rb
Executable file
@@ -0,0 +1,29 @@
|
|||||||
|
#!/usr/bin/env ruby
|
||||||
|
# encoding: utf-8
|
||||||
|
|
||||||
|
# We connect, pause the server, issue a write request, disconnect,
|
||||||
|
# then cont the server. This ensures that our disconnect happens
|
||||||
|
# while the server is trying to read the write data.
|
||||||
|
|
||||||
|
require 'flexnbd/fake_source'
|
||||||
|
include FlexNBD
|
||||||
|
|
||||||
|
addr, port, srv_pid = *ARGV
|
||||||
|
|
||||||
|
client = FakeSource.new( addr, port, "Timed out connecting" )
|
||||||
|
client.read_hello
|
||||||
|
Process.kill( "STOP", srv_pid.to_i )
|
||||||
|
client.write_write_request( 0, 8 )
|
||||||
|
client.close
|
||||||
|
Process.kill( "CONT", srv_pid.to_i )
|
||||||
|
|
||||||
|
# This sleep ensures that we don't return control to the test runner
|
||||||
|
# too soon, giving the flexnbd process time to fall over if it's going
|
||||||
|
# to.
|
||||||
|
sleep(0.25)
|
||||||
|
|
||||||
|
# ...and can we reconnect?
|
||||||
|
client2 = FakeSource.new( addr, port, "Timed out connecting" )
|
||||||
|
client2.close
|
||||||
|
|
||||||
|
exit(0)
|
@@ -29,7 +29,7 @@ module FlexNBD
|
|||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
def write_write_request( from, len, handle )
|
def write_write_request( from, len, handle="myhandle" )
|
||||||
fail "Bad handle" unless handle.length == 8
|
fail "Bad handle" unless handle.length == 8
|
||||||
|
|
||||||
@sock.write( "\x25\x60\x95\x13" )
|
@sock.write( "\x25\x60\x95\x13" )
|
||||||
|
@@ -50,6 +50,10 @@ class TestDestErrorHandling < Test::Unit::TestCase
|
|||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
|
def test_disconnect_before_write_reply_causes_error
|
||||||
|
run_fake( "source/close_after_write" )
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
def run_fake( name )
|
def run_fake( name )
|
||||||
@env.run_fake( name, @env.ip, @env.port1 )
|
@env.run_fake( name, @env.ip, @env.port1 )
|
||||||
|
Reference in New Issue
Block a user