singe/thirdparty/copas/tests/tcptimeout.lua

169 lines
4.4 KiB
Lua

-- Tests Copas socket timeouts
--
-- Run the test file, it should exit successfully without hanging.
-- make sure we are pointing to the local copas first
package.path = string.format("../src/?.lua;%s", package.path)
local platform = "unix"
if package.config:sub(1,1) == "\\" then
platform = "windows"
elseif io.popen("uname", "r"):read("*a"):find("Darwin") then
platform = "mac"
end
print("Testing platform: " .. platform)
local copas = require("copas")
local socket = require("socket")
-- hack; no way to kill copas.loop from thread
local function error(err)
print(debug.traceback(err, 2))
os.exit(-1)
end
local function assert(truthy, err)
if not truthy then
print(debug.traceback(err, 2))
os.exit(-1)
end
end
-- tcp echo server for testing against, returns `ip, port` to connect to
-- send `quit\n` to cause server to disconnect client
-- stops listen server after first connection
local function singleuseechoserver()
local server = socket.bind("127.0.0.1", 0) -- "localhost" fails because of IPv6 error
local ip, port = server:getsockname()
local function echoHandler(skt)
-- remove server after first connection
copas.removeserver(server)
skt = copas.wrap(skt)
while true do
local data = skt:receive()
if not data or data == "quit" then
break
end
skt:send(data..'\n')
end
end
copas.addserver(server, echoHandler)
return ip, port
end
local tests = {}
function tests.just_exit()
copas.loop()
end
function tests.connect_and_exit()
local ip, port = singleuseechoserver()
copas.addthread(function()
local client = socket.connect(ip, port)
client = copas.wrap(client)
client:close()
end)
copas.loop()
end
if platform == "mac" then
-- this test fails on a Mac, looks like the 'listen(0)' isn't being honoured
print("\nSkipping test on Mac!\n")
else
function tests.connect_timeout_copas()
local server = socket.tcp()
server:bind("localhost", 0)
server:listen(0) -- zero backlog, single connection will block further connections
-- note: not servicing connections
local ip, port = server:getsockname()
copas.addthread(function()
-- fill server's implicit connection backlog
socket.connect(ip,port)
local client = socket.tcp()
client = copas.wrap(client)
client:settimeout(0.01)
local status, err = client:connect(ip, port)
assert(status == nil, "connect somehow succeeded")
assert(err == "timeout", "connect failed with non-timeout error: "..tostring(err))
client:close()
end)
copas.loop()
end
function tests.connect_timeout_socket()
local server = socket.tcp()
server:bind("localhost", 0)
server:listen(0) -- zero backlog, single connection will block further connections
-- note: not servicing connections
local ip, port = server:getsockname()
copas.addthread(function()
copas.useSocketTimeoutErrors(true)
-- fill server's implicit connection backlog
socket.connect(ip,port)
local client = socket.tcp()
client = copas.wrap(client)
client:settimeout(0.01)
local status, err = client:connect(ip, port)
assert(status == nil, "connect somehow succeeded")
-- we test for a different error message becasue we expect socket errors, not copas ones
assert(err == "Operation already in progress", "connect failed with non-timeout error: "..tostring(err))
client:close()
end)
copas.loop()
end
end
function tests.receive_timeout()
local ip, port = singleuseechoserver()
copas.addthread(function()
local client = socket.tcp()
client = copas.wrap(client)
client:settimeout(0.01)
local status, err = client:connect(ip, port)
assert(status, "failed to connect: "..tostring(err))
client:send("foo\n")
local data, err = client:receive()
assert(data, "failed to recieve: "..tostring(err))
assert(data == "foo", "recieved wrong echo: "..tostring(data))
local data, err = client:receive()
assert(data == nil, "somehow recieved echo without sending")
assert(err == "timeout", "failed with non-timeout error: "..tostring(err))
client:close()
end)
copas.loop()
end
-- test "framework"
for name, test in pairs(tests) do
print("testing: "..tostring(name))
local status, err = pcall(test)
if not status then
error(err)
end
end
print("[✓] all tests completed successuly")