diff options
Diffstat (limited to 'tests/common.lua')
| -rw-r--r-- | tests/common.lua | 244 |
1 files changed, 0 insertions, 244 deletions
diff --git a/tests/common.lua b/tests/common.lua deleted file mode 100644 index 7472a10..0000000 --- a/tests/common.lua +++ /dev/null | |||
| @@ -1,244 +0,0 @@ | |||
| 1 | local json = require "cjson" | ||
| 2 | |||
| 3 | -- Misc routines to assist with CJSON testing | ||
| 4 | -- | ||
| 5 | -- Mark Pulford <mark@kyne.com.au> | ||
| 6 | |||
| 7 | -- Determine with a Lua table can be treated as an array. | ||
| 8 | -- Explicitly returns "not an array" for very sparse arrays. | ||
| 9 | -- Returns: | ||
| 10 | -- -1 Not an array | ||
| 11 | -- 0 Empty table | ||
| 12 | -- >0 Highest index in the array | ||
| 13 | function is_array(table) | ||
| 14 | local max = 0 | ||
| 15 | local count = 0 | ||
| 16 | for k, v in pairs(table) do | ||
| 17 | if type(k) == "number" then | ||
| 18 | if k > max then max = k end | ||
| 19 | count = count + 1 | ||
| 20 | else | ||
| 21 | return -1 | ||
| 22 | end | ||
| 23 | end | ||
| 24 | if max > count * 2 then | ||
| 25 | return -1 | ||
| 26 | end | ||
| 27 | |||
| 28 | return max | ||
| 29 | end | ||
| 30 | |||
| 31 | function serialise_table(value, indent, depth) | ||
| 32 | local spacing, spacing2, indent2 | ||
| 33 | if indent then | ||
| 34 | spacing = "\n" .. indent | ||
| 35 | spacing2 = spacing .. " " | ||
| 36 | indent2 = indent .. " " | ||
| 37 | else | ||
| 38 | spacing, spacing2, indent2 = " ", " ", false | ||
| 39 | end | ||
| 40 | depth = depth + 1 | ||
| 41 | if depth > 50 then | ||
| 42 | return "Cannot serialise any further: too many nested tables" | ||
| 43 | end | ||
| 44 | |||
| 45 | local max = is_array(value) | ||
| 46 | |||
| 47 | local comma = false | ||
| 48 | local fragment = { "{" .. spacing2 } | ||
| 49 | if max > 0 then | ||
| 50 | -- Serialise array | ||
| 51 | for i = 1, max do | ||
| 52 | if comma then | ||
| 53 | table.insert(fragment, "," .. spacing2) | ||
| 54 | end | ||
| 55 | table.insert(fragment, serialise_value(value[i], indent2, depth)) | ||
| 56 | comma = true | ||
| 57 | end | ||
| 58 | elseif max < 0 then | ||
| 59 | -- Serialise table | ||
| 60 | for k, v in pairs(value) do | ||
| 61 | if comma then | ||
| 62 | table.insert(fragment, "," .. spacing2) | ||
| 63 | end | ||
| 64 | table.insert(fragment, string.format( | ||
| 65 | "[%s] = %s", serialise_value(k, indent2, depth), | ||
| 66 | serialise_value(v, indent2, depth)) | ||
| 67 | ) | ||
| 68 | comma = true | ||
| 69 | end | ||
| 70 | end | ||
| 71 | table.insert(fragment, spacing .. "}") | ||
| 72 | |||
| 73 | return table.concat(fragment) | ||
| 74 | end | ||
| 75 | |||
| 76 | function serialise_value(value, indent, depth) | ||
| 77 | if indent == nil then indent = "" end | ||
| 78 | if depth == nil then depth = 0 end | ||
| 79 | |||
| 80 | if value == json.null then | ||
| 81 | return "json.null" | ||
| 82 | elseif type(value) == "string" then | ||
| 83 | return string.format("%q", value) | ||
| 84 | elseif type(value) == "nil" or type(value) == "number" or | ||
| 85 | type(value) == "boolean" then | ||
| 86 | return tostring(value) | ||
| 87 | elseif type(value) == "table" then | ||
| 88 | return serialise_table(value, indent, depth) | ||
| 89 | else | ||
| 90 | return "\"<" .. type(value) .. ">\"" | ||
| 91 | end | ||
| 92 | end | ||
| 93 | |||
| 94 | function dump_value(value) | ||
| 95 | print(serialise_value(value)) | ||
| 96 | end | ||
| 97 | |||
| 98 | function file_load(filename) | ||
| 99 | local file | ||
| 100 | if filename == nil then | ||
| 101 | file = io.stdin | ||
| 102 | else | ||
| 103 | local err | ||
| 104 | file, err = io.open(filename) | ||
| 105 | if file == nil then | ||
| 106 | error(string.format("Unable to read '%s': %s", filename, err)) | ||
| 107 | end | ||
| 108 | end | ||
| 109 | local data = file:read("*a") | ||
| 110 | |||
| 111 | if filename ~= nil then | ||
| 112 | file:close() | ||
| 113 | end | ||
| 114 | |||
| 115 | if data == nil then | ||
| 116 | error("Failed to read " .. filename) | ||
| 117 | end | ||
| 118 | |||
| 119 | return data | ||
| 120 | end | ||
| 121 | |||
| 122 | function file_save(filename, data) | ||
| 123 | local file | ||
| 124 | if filename == nil then | ||
| 125 | file = io.stdout | ||
| 126 | else | ||
| 127 | local err | ||
| 128 | file, err = io.open(filename, "w") | ||
| 129 | if file == nil then | ||
| 130 | error(string.format("Unable to write '%s': %s", filename, err)) | ||
| 131 | end | ||
| 132 | end | ||
| 133 | file:write(data) | ||
| 134 | if filename ~= nil then | ||
| 135 | file:close() | ||
| 136 | end | ||
| 137 | end | ||
| 138 | |||
| 139 | function compare_values(val1, val2) | ||
| 140 | local type1 = type(val1) | ||
| 141 | local type2 = type(val2) | ||
| 142 | if type1 ~= type2 then | ||
| 143 | return false | ||
| 144 | end | ||
| 145 | |||
| 146 | -- Check for NaN | ||
| 147 | if type1 == "number" and val1 ~= val1 and val2 ~= val2 then | ||
| 148 | return true | ||
| 149 | end | ||
| 150 | |||
| 151 | if type1 ~= "table" then | ||
| 152 | return val1 == val2 | ||
| 153 | end | ||
| 154 | |||
| 155 | -- check_keys stores all the keys that must be checked in val2 | ||
| 156 | local check_keys = {} | ||
| 157 | for k, _ in pairs(val1) do | ||
| 158 | check_keys[k] = true | ||
| 159 | end | ||
| 160 | |||
| 161 | for k, v in pairs(val2) do | ||
| 162 | if not check_keys[k] then | ||
| 163 | return false | ||
| 164 | end | ||
| 165 | |||
| 166 | if not compare_values(val1[k], val2[k]) then | ||
| 167 | return false | ||
| 168 | end | ||
| 169 | |||
| 170 | check_keys[k] = nil | ||
| 171 | end | ||
| 172 | for k, _ in pairs(check_keys) do | ||
| 173 | -- Not the same if any keys from val1 were not found in val2 | ||
| 174 | return false | ||
| 175 | end | ||
| 176 | return true | ||
| 177 | end | ||
| 178 | |||
| 179 | local test_count_pass = 0 | ||
| 180 | local test_count_total = 0 | ||
| 181 | |||
| 182 | function run_test_summary() | ||
| 183 | return test_count_pass, test_count_total | ||
| 184 | end | ||
| 185 | |||
| 186 | function run_test(testname, func, input, should_work, output) | ||
| 187 | local function status_line(name, status, value) | ||
| 188 | local statusmap = { [true] = ":success", [false] = ":error" } | ||
| 189 | if status ~= nil then | ||
| 190 | name = name .. statusmap[status] | ||
| 191 | end | ||
| 192 | print(string.format("[%s] %s", name, serialise_value(value, false))) | ||
| 193 | end | ||
| 194 | |||
| 195 | local result = { pcall(func, unpack(input)) } | ||
| 196 | local success = table.remove(result, 1) | ||
| 197 | |||
| 198 | local correct = false | ||
| 199 | if success == should_work and compare_values(result, output) then | ||
| 200 | correct = true | ||
| 201 | test_count_pass = test_count_pass + 1 | ||
| 202 | end | ||
| 203 | test_count_total = test_count_total + 1 | ||
| 204 | |||
| 205 | local teststatus = { [true] = "PASS", [false] = "FAIL" } | ||
| 206 | print(string.format("==> Test[%d] / %s: %s", | ||
| 207 | test_count_total, testname, teststatus[correct])) | ||
| 208 | |||
| 209 | status_line("Input", nil, input) | ||
| 210 | if not correct then | ||
| 211 | status_line("Expected", should_work, output) | ||
| 212 | end | ||
| 213 | status_line("Received", success, result) | ||
| 214 | print() | ||
| 215 | |||
| 216 | return correct, result | ||
| 217 | end | ||
| 218 | |||
| 219 | function run_test_group(testgroup, tests) | ||
| 220 | local function run_config(configname, func) | ||
| 221 | local success, msg = pcall(func) | ||
| 222 | if msg then | ||
| 223 | print(string.format("==> Config %s: %s", configname, msg)) | ||
| 224 | end | ||
| 225 | print() | ||
| 226 | end | ||
| 227 | |||
| 228 | local function test_id(group, id) | ||
| 229 | return string.format("%s[%d]", group, id) | ||
| 230 | end | ||
| 231 | |||
| 232 | for k, v in ipairs(tests) do | ||
| 233 | if type(v) == "function" then | ||
| 234 | -- Useful for changing configuration during a batch | ||
| 235 | run_config(test_id(testgroup, k), v) | ||
| 236 | elseif type(v) == "table" then | ||
| 237 | run_test(test_id(testgroup, k), unpack(v)) | ||
| 238 | else | ||
| 239 | error("Testgroup can only contain functions and tables") | ||
| 240 | end | ||
| 241 | end | ||
| 242 | end | ||
| 243 | |||
| 244 | -- vi:ai et sw=4 ts=4: | ||
