-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy pathlibcjson.lua
148 lines (148 loc) · 5.06 KB
/
libcjson.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
local require = require
local ffi = require "ffi"
local ffi_new = ffi.new
local ffi_typeof = ffi.typeof
local ffi_cdef = ffi.cdef
local ffi_load = ffi.load
local ffi_str = ffi.string
local ffi_gc = ffi.gc
local C = ffi.C
local next = next
local floor = math.floor
local inf = math.huge
local max = math.max
local pcall = pcall
local type = type
local next = next
local error = error
local pairs = pairs
local ipairs = ipairs
local tostring = tostring
local getmetatable = getmetatable
local setmetatable = setmetatable
local null = {}
if ngx and ngx.null then null = ngx.null end
ffi_cdef[[
typedef struct cJSON {
struct cJSON *next, *prev;
struct cJSON *child;
int type;
char *valuestring;
int valueint;
double valuedouble;
char *string;
} cJSON;
cJSON *cJSON_Parse(const char *value);
char *cJSON_Print(cJSON *item);
char *cJSON_PrintUnformatted(cJSON *item);
void cJSON_Delete(cJSON *c);
int cJSON_GetArraySize(cJSON *array);
cJSON *cJSON_CreateNull(void);
cJSON *cJSON_CreateTrue(void);
cJSON *cJSON_CreateFalse(void);
cJSON *cJSON_CreateBool(int b);
cJSON *cJSON_CreateNumber(double num);
cJSON *cJSON_CreateString(const char *string);
cJSON *cJSON_CreateArray(void);
cJSON *cJSON_CreateObject(void);
void cJSON_AddItemToArray(cJSON *array, cJSON *item);
void cJSON_AddItemToObject(cJSON *object,const char *string,cJSON *item);
void cJSON_Minify(char *json);
void free(void *ptr);
]]
local ok, newtab = pcall(require, "table.new")
if not ok then newtab = function() return {} end end
local cjson = ffi_load "cjson"
local json = newtab(0, 6)
local char_t = ffi_typeof("char[?]")
local mt_arr = { __index = { __jsontype = "array" }}
local mt_obj = { __index = { __jsontype = "object" }}
local function is_array(t)
local m, c = 0, 0
for k, _ in pairs(t) do
if type(k) ~= "number" or k < 0 or floor(k) ~= k then return false end
m = max(m, k)
c = c + 1
end
return c == m
end
function json.decval(j)
local t = j.type
if t == 1 then return false end
if t == 2 then return true end
if t == 4 then return null end
if t == 8 then return j.valuedouble end
if t == 16 then return ffi_str(j.valuestring) end
if t == 32 then return setmetatable(json.parse(j.child, newtab(cjson.cJSON_GetArraySize(j), 0)) or {}, mt_arr) end
if t == 64 then return setmetatable(json.parse(j.child, newtab(0, cjson.cJSON_GetArraySize(j))) or {}, mt_obj) end
return nil
end
function json.parse(j, r)
if j == nil then return nil end
local c = j
repeat
r[c.string ~= nil and ffi_str(c.string) or #r + 1] = json.decval(c)
c = c.next
until c == nil
return r
end
function json.decode(value)
if type(value) ~= "string" then return value end
local j = ffi_gc(cjson.cJSON_Parse(value), cjson.cJSON_Delete)
if j == nil then return nil end
local t = j.type
if t == 5 then return setmetatable(json.parse(j.child, newtab(cjson.cJSON_GetArraySize(j), 0)) or {}, mt_arr) end
if t == 6 then return setmetatable(json.parse(j.child, newtab(0, cjson.cJSON_GetArraySize(j))) or {}, mt_obj) end
return json.decval(j)
end
function json.encval(value)
local t = type(value)
if t == "string" then
return cjson.cJSON_CreateString(value)
elseif t == "number" then
if value ~= value then
return error "nan is not allowed in JSON"
elseif value == inf or value == -inf then
return error "inf is not allowed in JSON"
else
return cjson.cJSON_CreateNumber(value)
end
elseif t == "boolean" then
return value and cjson.cJSON_CreateTrue() or cjson.cJSON_CreateFalse()
elseif t == "table" then
if next(value) == nil then return (getmetatable(value) ~= mt_obj and is_array(value)) and cjson.cJSON_CreateArray() or cjson.cJSON_CreateObject() end
if getmetatable(value) ~= mt_obj and is_array(value) then
local j = cjson.cJSON_CreateArray()
for _, v in ipairs(value) do
cjson.cJSON_AddItemToArray(j, json.encval(v))
end
return j
end
local j = cjson.cJSON_CreateObject()
for k, v in pairs(value) do
cjson.cJSON_AddItemToObject(j, type(k) ~= "string" and tostring(k) or k, json.encval(v))
end
return j
else
return cjson.cJSON_CreateNull()
end
end
function json.encode(value, formatted)
local j = ffi_gc(json.encval(value), cjson.cJSON_Delete)
if j == nil then return nil end
return formatted ~= false and ffi_str(ffi_gc(cjson.cJSON_Print(j), C.free)) or ffi_str(ffi_gc(cjson.cJSON_PrintUnformatted(j), C.free))
end
function json.minify(value)
local t = type(value) ~= "string" and json.encode(value) or value
local m = ffi_new(char_t, #t, t)
cjson.cJSON_Minify(m)
return ffi_str(m)
end
return {
decode = json.decode,
encode = json.encode,
minify = json.minify,
array = mt_arr,
object = mt_obj,
null = null
}