Spaces:
Sleeping
Sleeping
-- | |
-- json.lua | |
-- | |
-- Copyright (c) 2020 rxi | |
-- | |
-- Permission is hereby granted, free of charge, to any person obtaining a copy of | |
-- this software and associated documentation files (the "Software"), to deal in | |
-- the Software without restriction, including without limitation the rights to | |
-- use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies | |
-- of the Software, and to permit persons to whom the Software is furnished to do | |
-- so, subject to the following conditions: | |
-- | |
-- The above copyright notice and this permission notice shall be included in all | |
-- copies or substantial portions of the Software. | |
-- | |
-- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
-- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
-- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |
-- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
-- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
-- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | |
-- SOFTWARE. | |
-- | |
json = (function () | |
local json = { _version = "0.1.2" } | |
------------------------------------------------------------------------------- | |
-- Encode | |
------------------------------------------------------------------------------- | |
local encode | |
local escape_char_map = { | |
[ "\\" ] = "\\", | |
[ "\"" ] = "\"", | |
[ "\b" ] = "b", | |
[ "\f" ] = "f", | |
[ "\n" ] = "n", | |
[ "\r" ] = "r", | |
[ "\t" ] = "t", | |
} | |
local escape_char_map_inv = { [ "/" ] = "/" } | |
for k, v in pairs(escape_char_map) do | |
escape_char_map_inv[v] = k | |
end | |
local function escape_char(c) | |
return "\\" .. (escape_char_map[c] or string.format("u%04x", c:byte())) | |
end | |
local function encode_nil(_) | |
return "null" | |
end | |
local function encode_table(val, stack) | |
local res = {} | |
stack = stack or {} | |
-- Circular reference? | |
if stack[val] then error("circular reference") end | |
stack[val] = true | |
if rawget(val, 1) ~= nil or next(val) == nil then | |
-- Treat as array -- check keys are valid and it is not sparse | |
local n = 0 | |
for k in pairs(val) do | |
if type(k) ~= "number" then | |
error("invalid table: mixed or invalid key types") | |
end | |
n = n + 1 | |
end | |
if n ~= #val then | |
error("invalid table: sparse array") | |
end | |
-- Encode | |
for _, v in ipairs(val) do | |
table.insert(res, encode(v, stack)) | |
end | |
stack[val] = nil | |
return "[" .. table.concat(res, ",") .. "]" | |
else | |
-- Treat as an object | |
for k, v in pairs(val) do | |
if type(k) ~= "string" then | |
error("invalid table: mixed or invalid key types") | |
end | |
table.insert(res, encode(k, stack) .. ":" .. encode(v, stack)) | |
end | |
stack[val] = nil | |
return "{" .. table.concat(res, ",") .. "}" | |
end | |
end | |
local function encode_string(val) | |
return '"' .. val:gsub('[%z\1-\31\\"]', escape_char) .. '"' | |
end | |
local function encode_number(val) | |
-- Check for NaN, -inf and inf | |
if val ~= val or val <= -math.huge or val >= math.huge then | |
error("unexpected number value '" .. tostring(val) .. "'") | |
end | |
return string.format("%.14g", val) | |
end | |
local type_func_map = { | |
[ "nil" ] = encode_nil, | |
[ "table" ] = encode_table, | |
[ "string" ] = encode_string, | |
[ "number" ] = encode_number, | |
[ "boolean" ] = tostring, | |
} | |
encode = function(val, stack) | |
local t = type(val) | |
local f = type_func_map[t] | |
if f then | |
return f(val, stack) | |
end | |
error("unexpected type '" .. t .. "'") | |
end | |
function json.encode(val) | |
return ( encode(val) ) | |
end | |
------------------------------------------------------------------------------- | |
-- Decode | |
------------------------------------------------------------------------------- | |
local parse | |
local function create_set(...) | |
local res = {} | |
for i = 1, select("#", ...) do | |
res[ select(i, ...) ] = true | |
end | |
return res | |
end | |
local space_chars = create_set(" ", "\t", "\r", "\n") | |
local delim_chars = create_set(" ", "\t", "\r", "\n", "]", "}", ",") | |
local escape_chars = create_set("\\", "/", '"', "b", "f", "n", "r", "t", "u") | |
local literals = create_set("true", "false", "null") | |
local literal_map = { | |
[ "true" ] = true, | |
[ "false" ] = false, | |
[ "null" ] = nil, | |
} | |
local function next_char(str, idx, set, negate) | |
for i = idx, #str do | |
if set[str:sub(i, i)] ~= negate then | |
return i | |
end | |
end | |
return #str + 1 | |
end | |
local function decode_error(str, idx, msg) | |
local line_count = 1 | |
local col_count = 1 | |
for i = 1, idx - 1 do | |
col_count = col_count + 1 | |
if str:sub(i, i) == "\n" then | |
line_count = line_count + 1 | |
col_count = 1 | |
end | |
end | |
error( string.format("%s at line %d col %d", msg, line_count, col_count) ) | |
end | |
local function codepoint_to_utf8(n) | |
-- http://scripts.sil.org/cms/scripts/page.php?site_id=nrsi&id=iws-appendixa | |
local f = math.floor | |
if n <= 0x7f then | |
return string.char(n) | |
elseif n <= 0x7ff then | |
return string.char(f(n / 64) + 192, n % 64 + 128) | |
elseif n <= 0xffff then | |
return string.char(f(n / 4096) + 224, f(n % 4096 / 64) + 128, n % 64 + 128) | |
elseif n <= 0x10ffff then | |
return string.char(f(n / 262144) + 240, f(n % 262144 / 4096) + 128, | |
f(n % 4096 / 64) + 128, n % 64 + 128) | |
end | |
error( string.format("invalid unicode codepoint '%x'", n) ) | |
end | |
local function parse_unicode_escape(s) | |
local n1 = tonumber( s:sub(1, 4), 16 ) | |
local n2 = tonumber( s:sub(7, 10), 16 ) | |
-- Surrogate pair? | |
if n2 then | |
return codepoint_to_utf8((n1 - 0xd800) * 0x400 + (n2 - 0xdc00) + 0x10000) | |
else | |
return codepoint_to_utf8(n1) | |
end | |
end | |
local function parse_string(str, i) | |
local res = "" | |
local j = i + 1 | |
local k = j | |
while j <= #str do | |
local x = str:byte(j) | |
if x < 32 then | |
decode_error(str, j, "control character in string") | |
elseif x == 92 then -- `\`: Escape | |
res = res .. str:sub(k, j - 1) | |
j = j + 1 | |
local c = str:sub(j, j) | |
if c == "u" then | |
local hex = str:match("^[dD][89aAbB]%x%x\\u%x%x%x%x", j + 1) | |
or str:match("^%x%x%x%x", j + 1) | |
or decode_error(str, j - 1, "invalid unicode escape in string") | |
res = res .. parse_unicode_escape(hex) | |
j = j + #hex | |
else | |
if not escape_chars[c] then | |
decode_error(str, j - 1, "invalid escape char '" .. c .. "' in string") | |
end | |
res = res .. escape_char_map_inv[c] | |
end | |
k = j + 1 | |
elseif x == 34 then -- `"`: End of string | |
res = res .. str:sub(k, j - 1) | |
return res, j + 1 | |
end | |
j = j + 1 | |
end | |
decode_error(str, i, "expected closing quote for string") | |
end | |
local function parse_number(str, i) | |
local x = next_char(str, i, delim_chars) | |
local s = str:sub(i, x - 1) | |
local n = tonumber(s) | |
if not n then | |
decode_error(str, i, "invalid number '" .. s .. "'") | |
end | |
return n, x | |
end | |
local function parse_literal(str, i) | |
local x = next_char(str, i, delim_chars) | |
local word = str:sub(i, x - 1) | |
if not literals[word] then | |
decode_error(str, i, "invalid literal '" .. word .. "'") | |
end | |
return literal_map[word], x | |
end | |
local function parse_array(str, i) | |
local res = {} | |
local n = 1 | |
i = i + 1 | |
while 1 do | |
local x | |
i = next_char(str, i, space_chars, true) | |
-- Empty / end of array? | |
if str:sub(i, i) == "]" then | |
i = i + 1 | |
break | |
end | |
-- Read token | |
x, i = parse(str, i) | |
res[n] = x | |
n = n + 1 | |
-- Next token | |
i = next_char(str, i, space_chars, true) | |
local chr = str:sub(i, i) | |
i = i + 1 | |
if chr == "]" then break end | |
if chr ~= "," then decode_error(str, i, "expected ']' or ','") end | |
end | |
return res, i | |
end | |
local function parse_object(str, i) | |
local res = {} | |
i = i + 1 | |
while 1 do | |
local key, val | |
i = next_char(str, i, space_chars, true) | |
-- Empty / end of object? | |
if str:sub(i, i) == "}" then | |
i = i + 1 | |
break | |
end | |
-- Read key | |
if str:sub(i, i) ~= '"' then | |
decode_error(str, i, "expected string for key") | |
end | |
key, i = parse(str, i) | |
-- Read ':' delimiter | |
i = next_char(str, i, space_chars, true) | |
if str:sub(i, i) ~= ":" then | |
decode_error(str, i, "expected ':' after key") | |
end | |
i = next_char(str, i + 1, space_chars, true) | |
-- Read value | |
val, i = parse(str, i) | |
-- Set | |
res[key] = val | |
-- Next token | |
i = next_char(str, i, space_chars, true) | |
local chr = str:sub(i, i) | |
i = i + 1 | |
if chr == "}" then break end | |
if chr ~= "," then decode_error(str, i, "expected '}' or ','") end | |
end | |
return res, i | |
end | |
local char_func_map = { | |
[ '"' ] = parse_string, | |
[ "0" ] = parse_number, | |
[ "1" ] = parse_number, | |
[ "2" ] = parse_number, | |
[ "3" ] = parse_number, | |
[ "4" ] = parse_number, | |
[ "5" ] = parse_number, | |
[ "6" ] = parse_number, | |
[ "7" ] = parse_number, | |
[ "8" ] = parse_number, | |
[ "9" ] = parse_number, | |
[ "-" ] = parse_number, | |
[ "t" ] = parse_literal, | |
[ "f" ] = parse_literal, | |
[ "n" ] = parse_literal, | |
[ "[" ] = parse_array, | |
[ "{" ] = parse_object, | |
} | |
parse = function(str, idx) | |
local chr = str:sub(idx, idx) | |
local f = char_func_map[chr] | |
if f then | |
return f(str, idx) | |
end | |
decode_error(str, idx, "unexpected character '" .. chr .. "'") | |
end | |
function json.decode(str) | |
if type(str) ~= "string" then | |
error("expected argument of type string, got " .. type(str)) | |
end | |
local res, idx = parse(str, next_char(str, 1, space_chars, true)) | |
idx = next_char(str, idx, space_chars, true) | |
if idx <= #str then | |
decode_error(str, idx, "trailing garbage") | |
end | |
return res | |
end | |
return json | |
end)() | |