mirror of https://github.com/tidwall/tile38.git
182 lines
4.0 KiB
Go
182 lines
4.0 KiB
Go
package json
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
|
|
"github.com/yuin/gopher-lua"
|
|
)
|
|
|
|
// Preload adds json to the given Lua state's package.preload table. After it
|
|
// has been preloaded, it can be loaded using require:
|
|
//
|
|
// local json = require("json")
|
|
func Preload(L *lua.LState) {
|
|
L.PreloadModule("json", Loader)
|
|
}
|
|
|
|
// Loader is the module loader function.
|
|
func Loader(L *lua.LState) int {
|
|
t := L.NewTable()
|
|
L.SetFuncs(t, api)
|
|
L.Push(t)
|
|
return 1
|
|
}
|
|
|
|
var api = map[string]lua.LGFunction{
|
|
"decode": apiDecode,
|
|
"encode": apiEncode,
|
|
}
|
|
|
|
func apiDecode(L *lua.LState) int {
|
|
str := L.CheckString(1)
|
|
|
|
value, err := Decode(L, []byte(str))
|
|
if err != nil {
|
|
L.Push(lua.LNil)
|
|
L.Push(lua.LString(err.Error()))
|
|
return 2
|
|
}
|
|
L.Push(value)
|
|
return 1
|
|
}
|
|
|
|
func apiEncode(L *lua.LState) int {
|
|
value := L.CheckAny(1)
|
|
|
|
data, err := Encode(value)
|
|
if err != nil {
|
|
L.Push(lua.LNil)
|
|
L.Push(lua.LString(err.Error()))
|
|
return 2
|
|
}
|
|
L.Push(lua.LString(string(data)))
|
|
return 1
|
|
}
|
|
|
|
var (
|
|
errNested = errors.New("cannot encode recursively nested tables to JSON")
|
|
errSparseArray = errors.New("cannot encode sparse array")
|
|
errInvalidKeys = errors.New("cannot encode mixed or invalid key types")
|
|
)
|
|
|
|
type invalidTypeError lua.LValueType
|
|
|
|
func (i invalidTypeError) Error() string {
|
|
return `cannot encode ` + lua.LValueType(i).String() + ` to JSON`
|
|
}
|
|
|
|
// Encode returns the JSON encoding of value.
|
|
func Encode(value lua.LValue) ([]byte, error) {
|
|
return json.Marshal(jsonValue{
|
|
LValue: value,
|
|
visited: make(map[*lua.LTable]bool),
|
|
})
|
|
}
|
|
|
|
type jsonValue struct {
|
|
lua.LValue
|
|
visited map[*lua.LTable]bool
|
|
}
|
|
|
|
func (j jsonValue) MarshalJSON() (data []byte, err error) {
|
|
switch converted := j.LValue.(type) {
|
|
case lua.LBool:
|
|
data, err = json.Marshal(bool(converted))
|
|
case lua.LNumber:
|
|
data, err = json.Marshal(float64(converted))
|
|
case *lua.LNilType:
|
|
data = []byte(`null`)
|
|
case lua.LString:
|
|
data, err = json.Marshal(string(converted))
|
|
case *lua.LTable:
|
|
if j.visited[converted] {
|
|
return nil, errNested
|
|
}
|
|
j.visited[converted] = true
|
|
|
|
key, value := converted.Next(lua.LNil)
|
|
|
|
switch key.Type() {
|
|
case lua.LTNil: // empty table
|
|
data = []byte(`[]`)
|
|
case lua.LTNumber:
|
|
arr := make([]jsonValue, 0, converted.Len())
|
|
expectedKey := lua.LNumber(1)
|
|
for key != lua.LNil {
|
|
if key.Type() != lua.LTNumber {
|
|
err = errInvalidKeys
|
|
return
|
|
}
|
|
if expectedKey != key {
|
|
err = errSparseArray
|
|
return
|
|
}
|
|
arr = append(arr, jsonValue{value, j.visited})
|
|
expectedKey++
|
|
key, value = converted.Next(key)
|
|
}
|
|
data, err = json.Marshal(arr)
|
|
case lua.LTString:
|
|
obj := make(map[string]jsonValue)
|
|
for key != lua.LNil {
|
|
if key.Type() != lua.LTString {
|
|
err = errInvalidKeys
|
|
return
|
|
}
|
|
obj[key.String()] = jsonValue{value, j.visited}
|
|
key, value = converted.Next(key)
|
|
}
|
|
data, err = json.Marshal(obj)
|
|
default:
|
|
err = errInvalidKeys
|
|
}
|
|
default:
|
|
err = invalidTypeError(j.LValue.Type())
|
|
}
|
|
return
|
|
}
|
|
|
|
// Decode converts the JSON encoded data to Lua values.
|
|
func Decode(L *lua.LState, data []byte) (lua.LValue, error) {
|
|
var value interface{}
|
|
err := json.Unmarshal(data, &value)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return DecodeValue(L, value), nil
|
|
}
|
|
|
|
// DecodeValue converts the value to a Lua value.
|
|
//
|
|
// This function only converts values that the encoding/json package decodes to.
|
|
// All other values will return lua.LNil.
|
|
func DecodeValue(L *lua.LState, value interface{}) lua.LValue {
|
|
switch converted := value.(type) {
|
|
case bool:
|
|
return lua.LBool(converted)
|
|
case float64:
|
|
return lua.LNumber(converted)
|
|
case string:
|
|
return lua.LString(converted)
|
|
case json.Number:
|
|
return lua.LString(converted)
|
|
case []interface{}:
|
|
arr := L.CreateTable(len(converted), 0)
|
|
for _, item := range converted {
|
|
arr.Append(DecodeValue(L, item))
|
|
}
|
|
return arr
|
|
case map[string]interface{}:
|
|
tbl := L.CreateTable(0, len(converted))
|
|
for key, item := range converted {
|
|
tbl.RawSetH(lua.LString(key), DecodeValue(L, item))
|
|
}
|
|
return tbl
|
|
case nil:
|
|
return lua.LNil
|
|
}
|
|
|
|
return lua.LNil
|
|
}
|