summaryrefslogtreecommitdiffstats
path: root/vendor/github.com/d5/tengo/stdlib/json.go
blob: f913dc4840564aee4a49a0be67dd1c1d4506b70d (plain) (blame)
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
package stdlib

import (
	"bytes"
	gojson "encoding/json"

	"github.com/d5/tengo/objects"
	"github.com/d5/tengo/stdlib/json"
)

var jsonModule = map[string]objects.Object{
	"decode":      &objects.UserFunction{Name: "decode", Value: jsonDecode},
	"encode":      &objects.UserFunction{Name: "encode", Value: jsonEncode},
	"indent":      &objects.UserFunction{Name: "encode", Value: jsonIndent},
	"html_escape": &objects.UserFunction{Name: "html_escape", Value: jsonHTMLEscape},
}

func jsonDecode(args ...objects.Object) (ret objects.Object, err error) {
	if len(args) != 1 {
		return nil, objects.ErrWrongNumArguments
	}

	switch o := args[0].(type) {
	case *objects.Bytes:
		v, err := json.Decode(o.Value)
		if err != nil {
			return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
		}
		return v, nil
	case *objects.String:
		v, err := json.Decode([]byte(o.Value))
		if err != nil {
			return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
		}
		return v, nil
	default:
		return nil, objects.ErrInvalidArgumentType{
			Name:     "first",
			Expected: "bytes/string",
			Found:    args[0].TypeName(),
		}
	}
}

func jsonEncode(args ...objects.Object) (ret objects.Object, err error) {
	if len(args) != 1 {
		return nil, objects.ErrWrongNumArguments
	}

	b, err := json.Encode(args[0])
	if err != nil {
		return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
	}

	return &objects.Bytes{Value: b}, nil
}

func jsonIndent(args ...objects.Object) (ret objects.Object, err error) {
	if len(args) != 3 {
		return nil, objects.ErrWrongNumArguments
	}

	prefix, ok := objects.ToString(args[1])
	if !ok {
		return nil, objects.ErrInvalidArgumentType{
			Name:     "prefix",
			Expected: "string(compatible)",
			Found:    args[1].TypeName(),
		}
	}

	indent, ok := objects.ToString(args[2])
	if !ok {
		return nil, objects.ErrInvalidArgumentType{
			Name:     "indent",
			Expected: "string(compatible)",
			Found:    args[2].TypeName(),
		}
	}

	switch o := args[0].(type) {
	case *objects.Bytes:
		var dst bytes.Buffer
		err := gojson.Indent(&dst, o.Value, prefix, indent)
		if err != nil {
			return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
		}
		return &objects.Bytes{Value: dst.Bytes()}, nil
	case *objects.String:
		var dst bytes.Buffer
		err := gojson.Indent(&dst, []byte(o.Value), prefix, indent)
		if err != nil {
			return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
		}
		return &objects.Bytes{Value: dst.Bytes()}, nil
	default:
		return nil, objects.ErrInvalidArgumentType{
			Name:     "first",
			Expected: "bytes/string",
			Found:    args[0].TypeName(),
		}
	}
}

func jsonHTMLEscape(args ...objects.Object) (ret objects.Object, err error) {
	if len(args) != 1 {
		return nil, objects.ErrWrongNumArguments
	}

	switch o := args[0].(type) {
	case *objects.Bytes:
		var dst bytes.Buffer
		gojson.HTMLEscape(&dst, o.Value)
		return &objects.Bytes{Value: dst.Bytes()}, nil
	case *objects.String:
		var dst bytes.Buffer
		gojson.HTMLEscape(&dst, []byte(o.Value))
		return &objects.Bytes{Value: dst.Bytes()}, nil
	default:
		return nil, objects.ErrInvalidArgumentType{
			Name:     "first",
			Expected: "bytes/string",
			Found:    args[0].TypeName(),
		}
	}
}