mirror of
https://github.com/OpenMW/openmw.git
synced 2025-10-24 14:26:42 +00:00
218 lines
5.9 KiB
Lua
218 lines
5.9 KiB
Lua
local module = {}
|
|
|
|
---
|
|
-- Matcher verifying that distance between given value and expected is not greater than maxDistance.
|
|
-- @function elementsAreArray
|
|
-- @param expected#vector.
|
|
-- @usage
|
|
-- expectThat(util.vector2(0, 0), closeToVector(util.vector2(0, 1), 1))
|
|
function module.closeToVector(expected, maxDistance)
|
|
return function(actual)
|
|
local distance = (expected - actual):length()
|
|
if distance <= maxDistance then
|
|
return ''
|
|
end
|
|
return string.format('%s is too far from expected %s: %s > %s', actual, expected, distance, maxDistance)
|
|
end
|
|
end
|
|
|
|
---
|
|
-- Matcher verifying that given value is an array each element of which matches elements of expected.
|
|
-- @function elementsAreArray
|
|
-- @param expected#array of values or matcher functions.
|
|
-- @usage
|
|
-- local t = {42, 13}
|
|
-- local matcher = function(actual)
|
|
-- if actual ~= 42 then
|
|
-- return string.format('%s is not 42', actual)
|
|
-- end
|
|
-- return ''
|
|
-- end
|
|
-- expectThat({42, 13}, elementsAreArray({matcher, 13}))
|
|
function module.elementsAreArray(expected)
|
|
local expected_matchers = {}
|
|
for i, v in ipairs(expected) do
|
|
if type(v) == 'function' then
|
|
expected_matchers[i] = v
|
|
else
|
|
expected_matchers[i] = function (other)
|
|
if expected[i].__eq(expected[i], other) then
|
|
return ''
|
|
end
|
|
return string.format('%s element %s does no match expected: %s', i, other, expected[i])
|
|
end
|
|
end
|
|
end
|
|
return function(actual)
|
|
if #actual < #expected_matchers then
|
|
return string.format('number of elements is less than expected: %s < %s', #actual, #expected_matchers)
|
|
end
|
|
local message = ''
|
|
for i, v in ipairs(actual) do
|
|
if i > #expected_matchers then
|
|
message = string.format('%s\n%s element is out of expected range: %s', message, i, #expected_matchers)
|
|
break
|
|
end
|
|
local match_message = expected_matchers[i](v)
|
|
if match_message ~= '' then
|
|
message = string.format('%s\n%s', message, match_message)
|
|
end
|
|
end
|
|
return message
|
|
end
|
|
end
|
|
|
|
---
|
|
-- Matcher verifying that given number is not a nan.
|
|
-- @function isNotNan
|
|
-- @usage
|
|
-- expectThat(value, isNotNan())
|
|
function module.isNotNan()
|
|
return function(actual)
|
|
if actual ~= actual then
|
|
return 'actual value is nan, expected to be not nan'
|
|
end
|
|
return ''
|
|
end
|
|
end
|
|
|
|
---
|
|
-- Matcher accepting any value.
|
|
-- @function isAny
|
|
-- @usage
|
|
-- expectThat(value, isAny())
|
|
function module.isAny()
|
|
return function(actual)
|
|
return ''
|
|
end
|
|
end
|
|
|
|
local function serializeArray(a)
|
|
local result = nil
|
|
for _, v in ipairs(a) do
|
|
if result == nil then
|
|
result = string.format('{%s', serialize(v))
|
|
else
|
|
result = string.format('%s, %s', result, serialize(v))
|
|
end
|
|
end
|
|
if result == nil then
|
|
return '{}'
|
|
end
|
|
return string.format('%s}', result)
|
|
end
|
|
|
|
local function serializeTable(t)
|
|
local result = nil
|
|
for k, v in pairs(t) do
|
|
if result == nil then
|
|
result = string.format('{%q = %s', k, serialize(v))
|
|
else
|
|
result = string.format('%s, %q = %s', result, k, serialize(v))
|
|
end
|
|
end
|
|
if result == nil then
|
|
return '{}'
|
|
end
|
|
return string.format('%s}', result)
|
|
end
|
|
|
|
local function isArray(t)
|
|
local i = 1
|
|
for _ in pairs(t) do
|
|
if t[i] == nil then
|
|
return false
|
|
end
|
|
i = i + 1
|
|
end
|
|
return true
|
|
end
|
|
|
|
function serialize(v)
|
|
local t = type(v)
|
|
if t == 'string' then
|
|
return string.format('%q', v)
|
|
elseif t == 'table' then
|
|
if isArray(v) then
|
|
return serializeArray(v)
|
|
end
|
|
return serializeTable(v)
|
|
end
|
|
return string.format('%s', v)
|
|
end
|
|
|
|
local function compareScalars(v1, v2)
|
|
if v1 == v2 then
|
|
return ''
|
|
end
|
|
if type(v1) == 'string' then
|
|
return string.format('%q ~= %q', v1, v2)
|
|
end
|
|
return string.format('%s ~= %s', v1, v2)
|
|
end
|
|
|
|
local function collectKeys(t)
|
|
local result = {}
|
|
for key in pairs(t) do
|
|
table.insert(result, key)
|
|
end
|
|
table.sort(result)
|
|
return result
|
|
end
|
|
|
|
local function compareTables(t1, t2)
|
|
local keys1 = collectKeys(t1)
|
|
local keys2 = collectKeys(t2)
|
|
if #keys1 ~= #keys2 then
|
|
return string.format('table size mismatch: %d ~= %d', #keys1, #keys2)
|
|
end
|
|
for i = 1, #keys1 do
|
|
local key1 = keys1[i]
|
|
local key2 = keys2[i]
|
|
if key1 ~= key2 then
|
|
return string.format('table keys mismatch: %q ~= %q', key1, key2)
|
|
end
|
|
local d = compare(t1[key1], t2[key2])
|
|
if d ~= '' then
|
|
return string.format('table values mismatch at key %s: %s', serialize(key1), d)
|
|
end
|
|
end
|
|
return ''
|
|
end
|
|
|
|
function compare(v1, v2)
|
|
local type1 = type(v1)
|
|
local type2 = type(v2)
|
|
if type2 == 'function' then
|
|
return v2(v1)
|
|
end
|
|
if type1 ~= type2 then
|
|
return string.format('types mismatch: %s ~= %s', type1, type2)
|
|
end
|
|
if type1 == 'nil' then
|
|
return ''
|
|
elseif type1 == 'table' then
|
|
return compareTables(v1, v2)
|
|
elseif type1 == 'nil' or type1 == 'boolean' or type1 == 'number' or type1 == 'string' then
|
|
return compareScalars(v1, v2)
|
|
end
|
|
error('unsupported type: %s', type1)
|
|
end
|
|
|
|
---
|
|
-- Matcher verifying that given value is equal to expected. Accepts nil, boolean, number, string and table or matcher
|
|
-- function.
|
|
-- @function equalTo
|
|
-- @usage
|
|
-- expectThat({a = {42, 'foo', {b = true}}}, equalTo({a = {42, 'foo', {b = true}}}))
|
|
function module.equalTo(expected)
|
|
return function(actual)
|
|
local diff = compare(actual, expected)
|
|
if diff == '' then
|
|
return ''
|
|
end
|
|
return string.format('%s; actual: %s; expected: %s', diff, serialize(actual, ''), serialize(expected, ''))
|
|
end
|
|
end
|
|
|
|
return module
|