Merge pull request #2711 from psychon/improve_test_runner
Improve the test runner: allow "direct" tests (instead of only steps-based tests)
This commit is contained in:
commit
a802d0c2c8
|
@ -43,77 +43,23 @@ runner.step_kill_clients = function(step)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
runner.run_steps = function(steps, options)
|
--- Print a message if verbose mode is enabled.
|
||||||
-- Setup timer/timeout to limit waiting for signal and quitting awesome.
|
-- @tparam string message The message to print.
|
||||||
local t = timer({timeout=0})
|
function runner.verbose(message)
|
||||||
local wait=20
|
|
||||||
local step=1
|
|
||||||
local step_count=0
|
|
||||||
options = options or {
|
|
||||||
kill_clients=true,
|
|
||||||
}
|
|
||||||
assert(not running, "run_steps() was called twice")
|
|
||||||
running = true
|
|
||||||
|
|
||||||
if options.kill_clients then
|
|
||||||
-- Add a final step to kill all clients and wait for them to finish.
|
|
||||||
-- Ref: https://github.com/awesomeWM/awesome/pull/1904#issuecomment-312793006
|
|
||||||
steps[#steps + 1] = runner.step_kill_clients
|
|
||||||
end
|
|
||||||
|
|
||||||
t:connect_signal("timeout", function() timer.delayed_call(function()
|
|
||||||
io.flush() -- for "tail -f".
|
|
||||||
step_count = step_count + 1
|
|
||||||
local step_as_string = step..'/'..#steps..' (@'..step_count..')'
|
|
||||||
if verbose then
|
if verbose then
|
||||||
io.stderr:write(string.format('Running step %s..\n', step_as_string))
|
io.stderr:write(message .. "\n")
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Call the current step's function.
|
--- When using run_direct(), this function indicates that the test is now done.
|
||||||
local success, result = xpcall(function()
|
-- @tparam[opt=nil] string message An error message explaining the test failure, if it failed.
|
||||||
return steps[step](step_count)
|
function runner.done(message)
|
||||||
end, debug.traceback)
|
if message then
|
||||||
|
io.stderr:write("Error: " .. message .. "\n")
|
||||||
if not success then
|
|
||||||
io.stderr:write('Error: running function for step '
|
|
||||||
..step_as_string..': '..tostring(result)..'!\n')
|
|
||||||
t:stop()
|
|
||||||
if not runner.quit_awesome_on_error then
|
|
||||||
io.stderr:write("Keeping awesome open...\n")
|
|
||||||
return -- keep awesome open on error.
|
|
||||||
end
|
|
||||||
|
|
||||||
elseif result then
|
|
||||||
-- true: test succeeded.
|
|
||||||
if step < #steps then
|
|
||||||
-- Next step.
|
|
||||||
step = step+1
|
|
||||||
step_count = 0
|
|
||||||
wait = 20
|
|
||||||
t.timeout = 0
|
|
||||||
t:again()
|
|
||||||
return
|
|
||||||
end
|
|
||||||
|
|
||||||
elseif result == false then
|
|
||||||
io.stderr:write("Step "..step_as_string.." failed (returned false).\n")
|
|
||||||
if not runner.quit_awesome_on_error then
|
if not runner.quit_awesome_on_error then
|
||||||
io.stderr:write("Keeping awesome open...\n")
|
io.stderr:write("Keeping awesome open...\n")
|
||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
else
|
|
||||||
-- No result yet, run this step again.
|
|
||||||
wait = wait-1
|
|
||||||
if wait > 0 then
|
|
||||||
t.timeout = 0.1
|
|
||||||
t:again()
|
|
||||||
else
|
|
||||||
io.stderr:write("Error: timeout waiting for signal in step "
|
|
||||||
..step_as_string..".\n")
|
|
||||||
t:stop()
|
|
||||||
end
|
|
||||||
return
|
|
||||||
end
|
end
|
||||||
|
|
||||||
local client_count = #client.get()
|
local client_count = #client.get()
|
||||||
|
@ -127,10 +73,82 @@ runner.run_steps = function(steps, options)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
if success and result then
|
if not message then
|
||||||
io.stderr:write("Test finished successfully.\n")
|
io.stderr:write("Test finished successfully.\n")
|
||||||
end
|
end
|
||||||
awesome.quit()
|
awesome.quit()
|
||||||
|
end
|
||||||
|
|
||||||
|
--- This function is called to indicate that a test does not use the run_steps()
|
||||||
|
-- facility, but instead runs something else directly.
|
||||||
|
function runner.run_direct()
|
||||||
|
assert(not running, "API abuse: Test was started twice")
|
||||||
|
running = true
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Start some step-wise tests. The given steps are called in order until all
|
||||||
|
-- succeeded. Each step is a function that can return true/false to indicate
|
||||||
|
-- success/failure, but can also return nothing if it needs to be called again
|
||||||
|
-- later.
|
||||||
|
function runner.run_steps(steps, options)
|
||||||
|
-- Setup timer/timeout to limit waiting for signal and quitting awesome.
|
||||||
|
local t = timer({timeout=0})
|
||||||
|
local wait=20
|
||||||
|
local step=1
|
||||||
|
local step_count=0
|
||||||
|
options = options or {
|
||||||
|
kill_clients=true,
|
||||||
|
}
|
||||||
|
runner.run_direct()
|
||||||
|
|
||||||
|
if options.kill_clients then
|
||||||
|
-- Add a final step to kill all clients and wait for them to finish.
|
||||||
|
-- Ref: https://github.com/awesomeWM/awesome/pull/1904#issuecomment-312793006
|
||||||
|
steps[#steps + 1] = runner.step_kill_clients
|
||||||
|
end
|
||||||
|
|
||||||
|
t:connect_signal("timeout", function() timer.delayed_call(function()
|
||||||
|
io.flush() -- for "tail -f".
|
||||||
|
step_count = step_count + 1
|
||||||
|
local step_as_string = step..'/'..#steps..' (@'..step_count..')'
|
||||||
|
runner.verbose(string.format('Running step %s..\n', step_as_string))
|
||||||
|
|
||||||
|
-- Call the current step's function.
|
||||||
|
local success, result = xpcall(function()
|
||||||
|
return steps[step](step_count)
|
||||||
|
end, debug.traceback)
|
||||||
|
|
||||||
|
if not success then
|
||||||
|
runner.done('running function for step '
|
||||||
|
..step_as_string..': '..tostring(result)..'!')
|
||||||
|
t:stop()
|
||||||
|
elseif result then
|
||||||
|
-- true: test succeeded.
|
||||||
|
if step < #steps then
|
||||||
|
-- Next step.
|
||||||
|
step = step+1
|
||||||
|
step_count = 0
|
||||||
|
wait = 20
|
||||||
|
t.timeout = 0
|
||||||
|
t:again()
|
||||||
|
else
|
||||||
|
-- All steps finished, we are done.
|
||||||
|
runner.done()
|
||||||
|
end
|
||||||
|
elseif result == false then
|
||||||
|
runner.done("Step "..step_as_string.." failed (returned false).")
|
||||||
|
else
|
||||||
|
-- No result yet, run this step again.
|
||||||
|
wait = wait-1
|
||||||
|
if wait > 0 then
|
||||||
|
t.timeout = 0.1
|
||||||
|
t:again()
|
||||||
|
else
|
||||||
|
runner.done("timeout waiting for signal in step "
|
||||||
|
..step_as_string..".")
|
||||||
|
t:stop()
|
||||||
|
end
|
||||||
|
end
|
||||||
end) end)
|
end) end)
|
||||||
t:start()
|
t:start()
|
||||||
end
|
end
|
||||||
|
|
|
@ -3,23 +3,27 @@
|
||||||
local runner = require("_runner")
|
local runner = require("_runner")
|
||||||
local spawn = require("awful.spawn")
|
local spawn = require("awful.spawn")
|
||||||
|
|
||||||
local todo = 2
|
local had_exit, had_success
|
||||||
local had_error = false
|
local had_error = false
|
||||||
|
|
||||||
local function wait_a_bit(count)
|
local function check_done()
|
||||||
if todo == 0 or count == 5 then
|
if had_exit and had_success then
|
||||||
return true
|
if had_error then
|
||||||
|
runner.done("Some error occurred, see above")
|
||||||
|
else
|
||||||
|
runner.done()
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
runner.run_steps({
|
end
|
||||||
function()
|
|
||||||
local err = spawn.with_line_callback(
|
local err = spawn.with_line_callback(
|
||||||
{ os.getenv("build_dir") .. "/test-gravity" },
|
{ os.getenv("build_dir") .. "/test-gravity" },
|
||||||
{
|
{
|
||||||
exit = function(what, code)
|
exit = function(what, code)
|
||||||
assert(what == "exit", what)
|
assert(what == "exit", what)
|
||||||
assert(code == 0, "Exit code was " .. code)
|
assert(code == 0, "Exit code was " .. code)
|
||||||
todo = todo - 1
|
had_exit = true
|
||||||
|
check_done()
|
||||||
end,
|
end,
|
||||||
stderr = function(line)
|
stderr = function(line)
|
||||||
had_error = true
|
had_error = true
|
||||||
|
@ -27,38 +31,16 @@ runner.run_steps({
|
||||||
end,
|
end,
|
||||||
stdout = function(line)
|
stdout = function(line)
|
||||||
if line == "SUCCESS" then
|
if line == "SUCCESS" then
|
||||||
todo = todo - 1
|
had_success = true
|
||||||
|
check_done()
|
||||||
elseif line:sub(1, 5) ~= "LOG: " then
|
elseif line:sub(1, 5) ~= "LOG: " then
|
||||||
had_error = true
|
had_error = true
|
||||||
print("Read on stdout: " .. line)
|
print("Read on stdout: " .. line)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
})
|
})
|
||||||
|
|
||||||
assert(type(err) ~= "string", err)
|
assert(type(err) ~= "string", err)
|
||||||
return true
|
runner.run_direct()
|
||||||
end,
|
|
||||||
-- Buy the external program some time to finish
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
wait_a_bit,
|
|
||||||
function()
|
|
||||||
if todo == 0 then
|
|
||||||
assert(not had_error, "Some error occurred, see above")
|
|
||||||
return true
|
|
||||||
end
|
|
||||||
end
|
|
||||||
})
|
|
||||||
|
|
||||||
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|
||||||
|
|
Loading…
Reference in New Issue