61 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			61 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| --- Tserial v1.5, a simple table serializer which turns tables into Lua script
 | |
| -- @author Taehl (SelfMadeSpirit@gmail.com)
 | |
| Tserial = {}
 | |
| TSerial = Tserial	-- for backwards-compatibility
 | |
| 
 | |
| --- Serializes a table into a string, in form of Lua script.
 | |
| -- @param t table to be serialized (may not contain any circular reference)
 | |
| -- @param drop if true, unserializable types will be silently dropped instead of raising errors
 | |
| -- if drop is a function, it will be called to serialize unsupported types
 | |
| -- if drop is a table, it will be used as a serialization table (where {[value] = serial})
 | |
| -- @param indent if true, output "human readable" mode with newlines and indentation (for debug)
 | |
| -- @return string recreating given table
 | |
| function Tserial.pack(t, drop, indent)
 | |
| 	assert(type(t) == "table", "Can only Tserial.pack tables.")
 | |
| 	local s, empty, indent = "{"..(indent and "\n" or ""), true, indent and math.max(type(indent)=="number" and indent or 0,0)
 | |
| 	local function proc(k,v, omitKey)	-- encode a key/value pair
 | |
| 		empty = nil	-- helps ensure empty tables return as "{}"
 | |
| 		local tk, tv, skip = type(k), type(v)
 | |
| 		if type(drop)=="table" and drop[k] then k = "["..drop[k].."]"
 | |
| 		elseif tk == "boolean" then k = k and "[true]" or "[false]"
 | |
| 		elseif tk == "string" then local f = string.format("%q",k) if f ~= '"'..k..'"' then k = '['..f..']' end
 | |
| 		elseif tk == "number" then k = "["..k.."]"
 | |
| 		elseif tk == "table" then k = "["..Tserial.pack(k, drop, indent and indent+1).."]"
 | |
| 		elseif type(drop) == "function" then k = "["..string.format("%q",drop(k)).."]"
 | |
| 		elseif drop then skip = true
 | |
| 		else error("Attempted to Tserial.pack a table with an invalid key: "..tostring(k))
 | |
| 		end
 | |
| 		if type(drop)=="table" and drop[v] then v = drop[v]
 | |
| 		elseif tv == "boolean" then v = v and "true" or "false"
 | |
| 		elseif tv == "string" then v = string.format("%q", v)
 | |
| 		elseif tv == "number" then	-- no change needed
 | |
| 		elseif tv == "table" then v = Tserial.pack(v, drop, indent and indent+1)
 | |
| 		elseif type(drop) == "function" then v = string.format("%q",drop(v))
 | |
| 		elseif drop then skip = true
 | |
| 		else error("Attempted to Tserial.pack a table with an invalid value: "..tostring(v))
 | |
| 		end
 | |
| 		if not skip then return string.rep("\t",indent or 0)..(omitKey and "" or k.."=")..v..","..(indent and "\n" or "") end
 | |
| 		return ""
 | |
| 	end
 | |
| 	local l=-1 repeat l=l+1 until rawget(t,l+1)==nil	-- #t "can" lie!
 | |
| 	for i=1,l do s = s..proc(i, t[i], true) end	-- use ordered values when possible for better string
 | |
| 	for k, v in pairs(t) do if not (type(k)=="number" and k<=l) then s = s..proc(k, v) end end
 | |
| 	if not empty then s = string.sub(s,1,string.len(s)-1) end
 | |
| 	if indent then s = string.sub(s,1,string.len(s)-1).."\n" end
 | |
| 	return s..string.rep("\t",(indent or 1)-1).."}"
 | |
| end
 | |
| 
 | |
| --- Loads a table into memory from a string (like those output by Tserial.pack)
 | |
| -- @param s a string of Lua defining a table, such as "{2,4,8,ex='ample'}"
 | |
| -- @param safe if true, all extraneous parts of the string will be removed, leaving only a table (prevents running anomalous code when unpacking untrusted strings). Will also cause malformed tables to quietly return nil and an error message, instead of throwing an error (so your program can't be crashed with a bad string)
 | |
| -- @return a table recreated from the given string.
 | |
| function Tserial.unpack(s, safe)
 | |
| 	if safe then s = string.match(s, "(%b{})") end
 | |
| 	assert(type(s) == "string", "Can only Tserial.unpack strings.")
 | |
| 	local f, result = loadstring("Tserial.table="..s)
 | |
| 	if not safe then assert(f,result) elseif not f then return nil, result end
 | |
| 	result = f()
 | |
| 	local t = Tserial.table
 | |
| 	Tserial.table = nil
 | |
| 	return t, result
 | |
| end |