@ -1,9 +1,61 @@
-- DFHack developer test harness
--@ module = true
local json = require ' json '
local json = require ' json '
local script = require ' gui.script '
local script = require ' gui.script '
local utils = require ' utils '
local utils = require ' utils '
local args = { ... }
local help_text =
local done_command = args [ 1 ]
[ ==== [
test / main
=========
Run DFHack tests .
Usage :
test / main [ < options > ] [ < done_command > ]
If a done_command is specified , it will be run after the tests complete .
Options :
- h , --help display this help message and exit.
- d , --test_dir specifies which directory to look in for tests. defaults to
the " hack/scripts/test " folder in your DF installation .
- n , --nocache don't skip tests marked as completed in test_status.json.
- m , --modes only run tests in the given comma separated list of modes.
valid modes are ' none ' ( test can be run on any screen ) and
' title ' ( test must be run on the DF title screen ) . if not
specified , no modes are filtered .
- t , --tests only run tests that match one of the comma separated list of
patterns . if not specified , no tests are filtered .
Examples :
test / main runs all tests that haven ' t been run before
test / main - n reruns all tests
test / main - nm none reruns tests that don ' t need the game to be in a
specific mode
test / main - nt quickfort reruns quickfort tests
test / main - nd / path / to / dfhack - scripts / repo / test
runs tests in your in - development branch of the
scripts repo
Default values for the options may be set in a file named test_config.json in
your DF folder . Options with comma - separated values should be written as json
arrays . For example :
{
" test_dir " : " /home/myk/src/dfhack-scripts/test " ,
" nocache " : true ,
" modes " : [ " none " ] ,
" tests " : [ " quickfort " , " devel " ] ,
" done_command " : " devel/luacov "
}
] ==== ]
local CONFIG_FILE = ' test_config.json '
local CONFIG_FILE = ' test_config.json '
local STATUS_FILE = ' test_status.json '
local STATUS_FILE = ' test_status.json '
@ -46,21 +98,58 @@ end
function expect . ge ( a , b , comment )
function expect . ge ( a , b , comment )
return a >= b , comment , ( ' %s < %s ' ) : format ( a , b )
return a >= b , comment , ( ' %s < %s ' ) : format ( a , b )
end
end
function expect . table_eq ( a , b , comment )
local function table_eq_recurse ( a , b , keys , known_eq )
if a == b then return true end
local checked = { }
local checked = { }
for k , v in pairs ( a ) do
for k , v in pairs ( a ) do
if a [ k ] ~= b [ k ] then
if type ( a [ k ] ) == ' table ' then
return false , comment , ( ' key "%s": %s ~= %s ' ) : format ( k , a [ k ] , b [ k ] )
if known_eq [ a [ k ] ] and known_eq [ a [ k ] ] [ b [ k ] ] then goto skip end
end
table.insert ( keys , tostring ( k ) )
if type ( b [ k ] ) ~= ' table ' then
return false , keys , { tostring ( a [ k ] ) , tostring ( b [ k ] ) }
end
if not known_eq [ a [ k ] ] then known_eq [ a [ k ] ] = { } end
for eq_tab , _ in pairs ( known_eq [ a [ k ] ] ) do
known_eq [ eq_tab ] [ b [ k ] ] = true
end
known_eq [ a [ k ] ] [ b [ k ] ] = true
if not known_eq [ b [ k ] ] then known_eq [ b [ k ] ] = { } end
for eq_tab , _ in pairs ( known_eq [ b [ k ] ] ) do
known_eq [ eq_tab ] [ a [ k ] ] = true
end
known_eq [ b [ k ] ] [ a [ k ] ] = true
local matched , keys_at_diff , diff =
table_eq_recurse ( a [ k ] , b [ k ] , keys , known_eq )
if not matched then return false , keys_at_diff , diff end
keys [ # keys ] = nil
elseif a [ k ] ~= b [ k ] then
table.insert ( keys , tostring ( k ) )
return false , keys , { tostring ( a [ k ] ) , tostring ( b [ k ] ) }
end
:: skip ::
checked [ k ] = true
checked [ k ] = true
end
end
for k in pairs ( b ) do
for k in pairs ( b ) do
if not checked [ k ] then
if not checked [ k ] then
return false , comment , ( ' key "%s": %s ~= %s ' ) : format ( k , a [ k ] , b [ k ] )
table.insert ( keys , tostring ( k ) )
return false , keys , { tostring ( a [ k ] ) , tostring ( b [ k ] ) }
end
end
end
end
return true
return true
end
end
function expect . table_eq ( a , b , comment )
if ( type ( a ) ~= ' table ' and type ( a ) ~= ' userdata ' ) or
( type ( b ) ~= ' table ' and type ( b ) ~= ' userdata ' ) then
return false , comment , ' operands to table_eq must be tables or userdata '
end
local keys , known_eq = { } , { }
local matched , keys_at_diff , diff = table_eq_recurse ( a , b , keys , known_eq )
if matched then return true end
local keystr = ' [ ' .. keys_at_diff [ 1 ] .. ' ] '
for i = 2 , # keys_at_diff do keystr = keystr .. ' [ ' .. keys_at_diff [ i ] .. ' ] ' end
return false , comment ,
( ' key %s: "%s" ~= "%s" ' ) : format ( keystr , diff [ 1 ] , diff [ 2 ] )
end
function expect . error ( func , ... )
function expect . error ( func , ... )
local ok , ret = pcall ( func , ... )
local ok , ret = pcall ( func , ... )
if ok then
if ok then
@ -99,12 +188,12 @@ function expect.not_pairs_contains(table, key, comment)
return true
return true
end
end
function delay ( frames )
local function delay ( frames )
frames = frames or 1
frames = frames or 1
script.sleep ( frames , ' frames ' )
script.sleep ( frames , ' frames ' )
end
end
function clean_require ( module )
local function clean_require ( module )
-- wrapper around require() - forces a clean load of every module to ensure
-- wrapper around require() - forces a clean load of every module to ensure
-- that modules checking for dfhack.internal.IN_TEST at load time behave
-- that modules checking for dfhack.internal.IN_TEST at load time behave
-- properly
-- properly
@ -114,7 +203,7 @@ function clean_require(module)
return require ( module )
return require ( module )
end
end
function ensure_title_screen ( )
local function ensure_title_screen ( )
if df.viewscreen_titlest : is_instance ( dfhack.gui . getCurViewscreen ( ) ) then
if df.viewscreen_titlest : is_instance ( dfhack.gui . getCurViewscreen ( ) ) then
return
return
end
end
@ -139,7 +228,7 @@ local MODE_NAVIGATE_FNS = {
title = ensure_title_screen ,
title = ensure_title_screen ,
}
}
function load_test_config ( config_file )
local function load_test_config ( config_file )
local config = { }
local config = { }
if dfhack.filesystem . isfile ( config_file ) then
if dfhack.filesystem . isfile ( config_file ) then
config = json.decode_file ( config_file )
config = json.decode_file ( config_file )
@ -148,14 +237,11 @@ function load_test_config(config_file)
if not config.test_dir then
if not config.test_dir then
config.test_dir = dfhack.getHackPath ( ) .. ' scripts/test '
config.test_dir = dfhack.getHackPath ( ) .. ' scripts/test '
end
end
if not dfhack.filesystem . isdir ( config.test_dir ) then
error ( ' Invalid test folder: ' .. config.test_dir )
end
return config
return config
end
end
function build_test_env ( )
local function build_test_env ( )
local env = {
local env = {
test = utils.OrderedTable ( ) ,
test = utils.OrderedTable ( ) ,
config = {
config = {
@ -195,8 +281,9 @@ function build_test_env()
return env , private
return env , private
end
end
function get_test_files ( test_dir )
local function get_test_files ( test_dir )
local files = { }
local files = { }
print ( ' Loading tests from ' .. test_dir )
for _ , entry in ipairs ( dfhack.filesystem . listdir_recursive ( test_dir ) ) do
for _ , entry in ipairs ( dfhack.filesystem . listdir_recursive ( test_dir ) ) do
if not entry.isdir and not entry.path : match ( ' main.lua ' ) then
if not entry.isdir and not entry.path : match ( ' main.lua ' ) then
table.insert ( files , entry.path )
table.insert ( files , entry.path )
@ -206,25 +293,25 @@ function get_test_files(test_dir)
return files
return files
end
end
function load_test_status ( )
local function load_test_status ( )
if dfhack.filesystem . isfile ( STATUS_FILE ) then
if dfhack.filesystem . isfile ( STATUS_FILE ) then
return json.decode_file ( STATUS_FILE )
return json.decode_file ( STATUS_FILE )
end
end
end
end
function save_test_status ( status )
local function save_test_status ( status )
json.encode_file ( status , STATUS_FILE )
json.encode_file ( status , STATUS_FILE )
end
end
function finish_tests ( )
local function finish_tests ( done_command )
dfhack.internal . IN_TEST = false
dfhack.internal . IN_TEST = false
if done_command then
if done_command and # done_command > 0 then
dfhack.run_command ( done_command )
dfhack.run_command ( done_command )
end
end
end
end
function load_tests ( file , tests )
local function load_tests ( file , tests )
local short_filename = file : sub ( file : find ( ' test ' ) , - 1 )
local short_filename = file : sub ( ( file : find ( ' test ' ) or - 4 ) + 5 , - 1 )
print ( ' Loading file: ' .. short_filename )
print ( ' Loading file: ' .. short_filename )
local env , env_private = build_test_env ( )
local env , env_private = build_test_env ( )
local code , err = loadfile ( file , ' t ' , env )
local code , err = loadfile ( file , ' t ' , env )
@ -258,7 +345,7 @@ function load_tests(file, tests)
return true
return true
end
end
function sort_tests ( tests )
local function sort_tests ( tests )
-- to make sort stable
-- to make sort stable
local test_index = utils.invert ( tests )
local test_index = utils.invert ( tests )
table.sort ( tests , function ( a , b )
table.sort ( tests , function ( a , b )
@ -270,7 +357,7 @@ function sort_tests(tests)
end )
end )
end
end
function run_test ( test , status , counts )
local function run_test ( test , status , counts )
test.private . checks = 0
test.private . checks = 0
test.private . checks_ok = 0
test.private . checks_ok = 0
counts.tests = counts.tests + 1
counts.tests = counts.tests + 1
@ -292,55 +379,75 @@ function run_test(test, status, counts)
return passed
return passed
end
end
function main ( )
local function get_tests ( test_files , counts )
local config = load_test_config ( CONFIG_FILE )
local files = get_test_files ( config.test_dir )
local counts = {
tests = 0 ,
tests_ok = 0 ,
checks = 0 ,
checks_ok = 0 ,
file_errors = 0 ,
}
local passed = true
ensure_title_screen ( )
print ( ' Loading tests ' )
local tests = { }
local tests = { }
for _ , file in ipairs ( files) do
for _ , file in ipairs ( test_files ) do
if not load_tests ( file , tests ) then
if not load_tests ( file , tests ) then
passed = false
counts.file_errors = counts.file_errors + 1
counts.file_errors = counts.file_errors + 1
end
end
end
end
return tests
end
print ( ' Filtering tests ' )
local function filter_tests ( tests , config )
if config.tests then
if config.tests or config.modes then
for _ , filter in ipairs ( { { ' tests ' , ' name pattern ' } , { ' modes ' , ' mode ' } } ) do
if config [ filter [ 1 ] ] and # config [ filter [ 1 ] ] > 0 then
print ( string.format ( ' Filtering tests by %s: ' , filter [ 2 ] ) )
for _ , v in ipairs ( config [ filter [ 1 ] ] ) do
print ( string.format ( ' %s ' , v ) )
end
end
end
local orig_length = # tests
local orig_length = # tests
for i = # tests , 1 , - 1 do
for i = # tests , 1 , - 1 do
local remove = false
if config.modes then
remove = true
-- allow test if it matches any of the given modes
for _ , mode in pairs ( config.modes ) do
if tests [ i ] . config.mode == mode then
remove = false
break
end
end
end
if config.tests and not remove then
remove = true
-- allow test if it matches any of the given patterns
for _ , pattern in pairs ( config.tests ) do
for _ , pattern in pairs ( config.tests ) do
if not tests [ i ] . name : match ( pattern ) then
if tests [ i ] . name : match ( pattern ) then
table.remove ( tests , i )
remove = false
break
end
end
end
end
end
if remove then table.remove ( tests , i ) end
end
end
print ( ' Selected tests: ' .. # tests .. ' / ' .. orig_length )
print ( ' Selected tests: ' .. # tests .. ' / ' .. orig_length )
end
end
local status = load_test_status ( ) or { }
local status = { }
if not config.nocache then
status = load_test_status ( ) or status
for i = # tests , 1 , - 1 do
for i = # tests , 1 , - 1 do
local test = tests [ i ]
local test = tests [ i ]
if not status [ test.full_name ] then
if not status [ test.full_name ] then
status [ test.full_name ] = TestStatus.PENDING
status [ test.full_name ] = TestStatus.PENDING
elseif status [ test.full_name ] ~= TestStatus.PENDING then
elseif status [ test.full_name ] ~= TestStatus.PENDING then
print ( ' skipping test: ' .. test.name .. ' : state = ' .. status [ test.full_name ] .. ' ) ' )
print ( ( ' skipping test: %s: state = %s) ' ) : format (
test.name , status [ test.full_name ] ) )
table.remove ( tests , i )
table.remove ( tests , i )
end
end
end
end
end
sort_tests ( tests )
sort_tests ( tests )
return status
end
print ( ' Running ' .. # tests .. ' tests ' )
local function run_tests ( tests , status , counts )
print ( ( ' Running %d tests ' ) : format ( # tests ) )
for _ , test in pairs ( tests ) do
for _ , test in pairs ( tests ) do
MODE_NAVIGATE_FNS [ test.config . mode ] ( )
MODE_NAVIGATE_FNS [ test.config . mode ] ( )
local passed = run_test ( test , status , counts )
local passed = run_test ( test , status , counts )
@ -354,6 +461,55 @@ function main()
print ( ( ' %d test files failed to load ' ) : format ( counts.file_errors ) )
print ( ( ' %d test files failed to load ' ) : format ( counts.file_errors ) )
end
end
local function main ( args )
local help , nocache , test_dir , mode_filter , test_filter =
false , false , nil , { } , { }
local other_args = utils.processArgsGetopt ( args , {
{ ' h ' , ' help ' , handler = function ( ) help = true end } ,
{ ' d ' , ' test_dir ' , hasArg = true ,
handler = function ( arg ) test_dir = arg end } ,
{ ' n ' , ' nocache ' , handler = function ( ) nocache = true end } ,
{ ' m ' , ' modes ' , hasArg = true ,
handler = function ( arg ) mode_filter = arg : split ( ' , ' ) end } ,
{ ' t ' , ' tests ' , hasArg = true ,
handler = function ( arg ) test_filter = arg : split ( ' , ' ) end } ,
} )
if help then print ( help_text ) return end
local done_command = table.concat ( other_args , ' ' )
local config = load_test_config ( CONFIG_FILE )
-- override config with any params specified on the commandline
if test_dir then config.test_dir = test_dir end
if nocache then config.nocache = true end
if # mode_filter > 0 then config.modes = mode_filter end
if # test_filter > 0 then config.tests = test_filter end
if # done_command > 0 then config.done_command = done_command end
if not dfhack.filesystem . isdir ( config.test_dir ) then
qerror ( ( ' Invalid test folder: "%s" ' ) : format ( config.test_dir ) )
end
local counts = {
tests = 0 ,
tests_ok = 0 ,
checks = 0 ,
checks_ok = 0 ,
file_errors = 0 ,
}
local test_files = get_test_files ( config.test_dir )
local tests = get_tests ( test_files , counts )
local status = filter_tests ( tests , config )
script.start ( function ( )
script.start ( function ( )
dfhack.with_finalize ( finish_tests , main )
dfhack.call_with_finalizer ( 1 , true ,
finish_tests , config.done_command ,
run_tests , tests , status , counts )
end )
end )
end
if not dfhack_flags.module then
main ( { ... } )
end