go-yaml/decode_test.go

1588 lines
32 KiB
Go
Raw Normal View History

2019-10-18 00:00:58 +09:00
package yaml_test
import (
"bytes"
"fmt"
2019-11-14 09:12:39 +09:00
"log"
2019-10-18 00:00:58 +09:00
"math"
"reflect"
2020-01-14 22:50:50 +09:00
"strconv"
2019-10-18 00:00:58 +09:00
"strings"
"testing"
2019-11-01 14:09:33 +09:00
"time"
2019-10-18 00:00:58 +09:00
"github.com/goccy/go-yaml"
)
type Child struct {
B int
C int `yaml:"-"`
}
2019-10-18 00:00:58 +09:00
func TestDecoder(t *testing.T) {
tests := []struct {
source string
value interface{}
}{
{
"null\n",
(*struct{})(nil),
},
{
"v: hi\n",
map[string]string{"v": "hi"},
},
{
"v: \"true\"\n",
map[string]string{"v": "true"},
},
{
"v: \"false\"\n",
map[string]string{"v": "false"},
},
{
"v: true\n",
map[string]interface{}{"v": true},
},
{
"v: true\n",
map[string]string{"v": "true"},
},
2019-10-30 17:26:26 +09:00
{
"v: 10\n",
map[string]string{"v": "10"},
},
{
"v: -10\n",
map[string]string{"v": "-10"},
},
{
"v: 1.234\n",
map[string]string{"v": "1.234"},
},
2019-10-18 00:00:58 +09:00
{
"v: false\n",
map[string]bool{"v": false},
},
{
"v: 10\n",
map[string]int{"v": 10},
},
{
"v: 10",
map[string]interface{}{"v": 10},
},
{
"v: 0b10",
map[string]interface{}{"v": 2},
},
{
"v: -0b101010",
map[string]interface{}{"v": -42},
},
{
"v: -0b1000000000000000000000000000000000000000000000000000000000000000",
map[string]interface{}{"v": -9223372036854775808},
},
{
"v: 0xA",
map[string]interface{}{"v": 10},
},
{
"v: .1",
map[string]interface{}{"v": 0.1},
},
{
"v: -.1",
map[string]interface{}{"v": -0.1},
},
2019-10-18 00:00:58 +09:00
{
"v: -10\n",
map[string]int{"v": -10},
},
{
"v: 4294967296\n",
map[string]int{"v": 4294967296},
},
{
"v: 0.1\n",
map[string]interface{}{"v": 0.1},
},
{
"v: 0.99\n",
map[string]float32{"v": 0.99},
},
{
"v: -0.1\n",
map[string]float64{"v": -0.1},
},
{
"v: 6.8523e+5",
map[string]interface{}{"v": 6.8523e+5},
},
{
"v: 685.230_15e+03",
map[string]interface{}{"v": 685.23015e+03},
},
{
"v: 685_230.15",
map[string]interface{}{"v": 685230.15},
},
{
"v: 685_230.15",
map[string]float64{"v": 685230.15},
},
{
"v: 685230",
map[string]interface{}{"v": 685230},
},
{
"v: +685_230",
map[string]interface{}{"v": 685230},
},
{
"v: 02472256",
map[string]interface{}{"v": 685230},
},
{
"v: 0x_0A_74_AE",
map[string]interface{}{"v": 685230},
},
{
"v: 0b1010_0111_0100_1010_1110",
map[string]interface{}{"v": 685230},
},
{
"v: +685_230",
map[string]int{"v": 685230},
},
2019-10-29 19:46:55 +09:00
2019-10-31 17:23:56 +09:00
// Bools from spec
{
"v: True",
map[string]interface{}{"v": true},
},
{
"v: TRUE",
map[string]interface{}{"v": true},
},
{
"v: False",
map[string]interface{}{"v": false},
},
{
"v: FALSE",
map[string]interface{}{"v": false},
},
{
"v: y",
map[string]interface{}{"v": "y"}, // y or yes or Yes is string
},
{
"v: NO",
map[string]interface{}{"v": "NO"}, // no or No or NO is string
},
{
"v: on",
map[string]interface{}{"v": "on"}, // on is string
},
2019-10-29 19:46:55 +09:00
// Some cross type conversions
{
"v: 42",
map[string]uint{"v": 42},
}, {
"v: 4294967296",
map[string]uint64{"v": 4294967296},
},
2019-10-29 20:15:27 +09:00
// int
{
"v: 2147483647",
map[string]int{"v": math.MaxInt32},
},
{
"v: -2147483648",
map[string]int{"v": math.MinInt32},
},
// int64
{
"v: 9223372036854775807",
map[string]int64{"v": math.MaxInt64},
},
{
"v: 0b111111111111111111111111111111111111111111111111111111111111111",
map[string]int64{"v": math.MaxInt64},
},
{
"v: -9223372036854775808",
map[string]int64{"v": math.MinInt64},
},
{
"v: -0b111111111111111111111111111111111111111111111111111111111111111",
map[string]int64{"v": -math.MaxInt64},
},
// uint
{
"v: 0",
map[string]uint{"v": 0},
},
{
"v: 4294967295",
map[string]uint{"v": math.MaxUint32},
},
// uint64
{
"v: 0",
map[string]uint{"v": 0},
},
{
"v: 18446744073709551615",
map[string]uint64{"v": math.MaxUint64},
},
{
"v: 0b1111111111111111111111111111111111111111111111111111111111111111",
map[string]uint64{"v": math.MaxUint64},
},
{
"v: 9223372036854775807",
map[string]uint64{"v": math.MaxInt64},
},
// float32
{
"v: 3.40282346638528859811704183484516925440e+38",
map[string]float32{"v": math.MaxFloat32},
},
{
"v: 1.401298464324817070923729583289916131280e-45",
map[string]float32{"v": math.SmallestNonzeroFloat32},
},
{
"v: 18446744073709551615",
map[string]float32{"v": float32(math.MaxUint64)},
},
{
"v: 18446744073709551616",
map[string]float32{"v": float32(math.MaxUint64 + 1)},
},
// float64
{
"v: 1.797693134862315708145274237317043567981e+308",
map[string]float64{"v": math.MaxFloat64},
},
{
"v: 4.940656458412465441765687928682213723651e-324",
map[string]float64{"v": math.SmallestNonzeroFloat64},
},
{
"v: 18446744073709551615",
map[string]float64{"v": float64(math.MaxUint64)},
},
{
"v: 18446744073709551616",
map[string]float64{"v": float64(math.MaxUint64 + 1)},
},
2019-11-01 14:09:33 +09:00
// Timestamps
{
// Date only.
"v: 2015-01-01\n",
map[string]time.Time{"v": time.Date(2015, 1, 1, 0, 0, 0, 0, time.UTC)},
},
{
// RFC3339
"v: 2015-02-24T18:19:39.12Z\n",
map[string]time.Time{"v": time.Date(2015, 2, 24, 18, 19, 39, .12e9, time.UTC)},
},
{
// RFC3339 with short dates.
"v: 2015-2-3T3:4:5Z",
map[string]time.Time{"v": time.Date(2015, 2, 3, 3, 4, 5, 0, time.UTC)},
},
{
// ISO8601 lower case t
"v: 2015-02-24t18:19:39Z\n",
map[string]time.Time{"v": time.Date(2015, 2, 24, 18, 19, 39, 0, time.UTC)},
},
{
// space separate, no time zone
"v: 2015-02-24 18:19:39\n",
map[string]time.Time{"v": time.Date(2015, 2, 24, 18, 19, 39, 0, time.UTC)},
},
2019-10-29 20:15:27 +09:00
// Quoted values.
{
"'1': '\"2\"'",
map[interface{}]interface{}{"1": "\"2\""},
},
2019-10-29 20:36:34 +09:00
{
"a: -b_c",
map[string]interface{}{"a": "-b_c"},
},
{
"a: +b_c",
map[string]interface{}{"a": "+b_c"},
},
{
"a: 50cent_of_dollar",
map[string]interface{}{"a": "50cent_of_dollar"},
},
2019-10-30 16:57:59 +09:00
// Nulls
{
"v:",
map[string]interface{}{"v": nil},
},
{
"v: ~",
map[string]interface{}{"v": nil},
},
{
"~: null key",
map[interface{}]string{nil: "null key"},
},
{
"v:",
map[string]*bool{"v": nil},
},
2019-10-31 16:52:18 +09:00
{
"v: null",
map[string]*string{"v": nil},
},
{
"v: null",
map[string]string{"v": ""},
},
{
"v: null",
map[string]interface{}{"v": nil},
},
2019-10-31 17:23:56 +09:00
{
"v: Null",
map[string]interface{}{"v": nil},
},
{
"v: NULL",
map[string]interface{}{"v": nil},
},
2019-10-31 16:52:18 +09:00
{
"v: ~",
map[string]*string{"v": nil},
},
{
"v: ~",
map[string]string{"v": ""},
},
2019-10-31 17:38:53 +09:00
2019-10-18 00:00:58 +09:00
{
"v: .inf\n",
map[string]interface{}{"v": math.Inf(0)},
},
2019-10-31 17:23:56 +09:00
{
"v: .Inf\n",
map[string]interface{}{"v": math.Inf(0)},
},
{
"v: .INF\n",
map[string]interface{}{"v": math.Inf(0)},
},
2019-10-18 00:00:58 +09:00
{
"v: -.inf\n",
map[string]interface{}{"v": math.Inf(-1)},
},
2019-10-31 17:23:56 +09:00
{
"v: -.Inf\n",
map[string]interface{}{"v": math.Inf(-1)},
},
{
"v: -.INF\n",
map[string]interface{}{"v": math.Inf(-1)},
},
2019-10-18 00:00:58 +09:00
{
"v: .nan\n",
map[string]interface{}{"v": math.NaN()},
},
2019-10-31 17:23:56 +09:00
{
"v: .NaN\n",
map[string]interface{}{"v": math.NaN()},
},
{
"v: .NAN\n",
map[string]interface{}{"v": math.NaN()},
},
2019-10-31 17:38:53 +09:00
// Explicit tags.
{
"v: !!float '1.1'",
map[string]interface{}{"v": 1.1},
},
{
"v: !!float 0",
map[string]interface{}{"v": float64(0)},
},
{
"v: !!float -1",
map[string]interface{}{"v": float64(-1)},
},
{
"v: !!null ''",
map[string]interface{}{"v": nil},
},
2019-11-01 14:22:06 +09:00
{
"v: !!timestamp \"2015-01-01\"",
map[string]time.Time{"v": time.Date(2015, 1, 1, 0, 0, 0, 0, time.UTC)},
},
{
"v: !!timestamp 2015-01-01",
map[string]time.Time{"v": time.Date(2015, 1, 1, 0, 0, 0, 0, time.UTC)},
},
2019-10-31 17:38:53 +09:00
2019-10-31 17:56:04 +09:00
// Flow sequence
{
"v: [A,B]",
map[string]interface{}{"v": []interface{}{"A", "B"}},
},
{
"v: [A,B,C,]",
map[string][]string{"v": []string{"A", "B", "C"}},
},
{
"v: [A,1,C]",
map[string][]string{"v": []string{"A", "1", "C"}},
},
{
"v: [A,1,C]",
map[string]interface{}{"v": []interface{}{"A", 1, "C"}},
},
2019-10-31 17:58:26 +09:00
// Block sequence
{
"v:\n - A\n - B",
map[string]interface{}{"v": []interface{}{"A", "B"}},
},
{
"v:\n - A\n - B\n - C",
map[string][]string{"v": []string{"A", "B", "C"}},
},
{
"v:\n - A\n - 1\n - C",
map[string][]string{"v": []string{"A", "1", "C"}},
},
{
"v:\n - A\n - 1\n - C",
map[string]interface{}{"v": []interface{}{"A", 1, "C"}},
},
2019-10-31 18:02:06 +09:00
// Map inside interface with no type hints.
{
"a: {b: c}",
map[interface{}]interface{}{"a": map[interface{}]interface{}{"b": "c"}},
},
2019-10-18 00:00:58 +09:00
{
"v: \"\"\n",
map[string]string{"v": ""},
},
{
"v:\n- A\n- B\n",
2019-10-21 01:28:48 +09:00
map[string][]string{"v": {"A", "B"}},
2019-10-18 00:00:58 +09:00
},
{
"a: '-'\n",
map[string]string{"a": "-"},
},
{
"123\n",
123,
},
{
"hello: world\n",
map[string]string{"hello": "world"},
},
2019-11-09 17:20:39 +09:00
{
"hello: world\r\n",
map[string]string{"hello": "world"},
},
{
"hello: world\rGo: Gopher",
map[string]string{"hello": "world", "Go": "Gopher"},
},
2019-11-01 12:44:32 +09:00
// Structs and type conversions.
{
"hello: world",
struct{ Hello string }{"world"},
},
{
"a: {b: c}",
struct{ A struct{ B string } }{struct{ B string }{"c"}},
},
{
"a: {b: c}",
struct{ A map[string]string }{map[string]string{"b": "c"}},
},
{
"a:",
struct{ A map[string]string }{},
},
{
"a: 1",
struct{ A int }{1},
},
{
"a: 1",
struct{ A float64 }{1},
},
{
"a: 1.0",
struct{ A int }{1},
},
{
"a: 1.0",
struct{ A uint }{1},
},
{
"a: [1, 2]",
struct{ A []int }{[]int{1, 2}},
},
{
"a: [1, 2]",
struct{ A [2]int }{[2]int{1, 2}},
},
{
"a: 1",
struct{ B int }{0},
},
{
"a: 1",
struct {
2019-11-13 12:14:49 +09:00
B int `yaml:"a"`
2019-11-01 12:44:32 +09:00
}{1},
},
2019-10-18 00:00:58 +09:00
{
"v:\n- A\n- 1\n- B:\n - 2\n - 3\n",
map[string]interface{}{
"v": []interface{}{
"A",
1,
map[string][]int{
2019-10-21 01:28:48 +09:00
"B": {2, 3},
2019-10-18 00:00:58 +09:00
},
},
},
},
{
"a:\n b: c\n",
map[string]interface{}{
"a": map[string]string{
"b": "c",
},
},
},
{
"a: {x: 1}\n",
map[string]map[string]int{
2019-10-21 01:28:48 +09:00
"a": {
2019-10-18 00:00:58 +09:00
"x": 1,
},
},
},
{
"t2: 2018-01-09T10:40:47Z\nt4: 2098-01-09T10:40:47Z\n",
map[string]string{
"t2": "2018-01-09T10:40:47Z",
"t4": "2098-01-09T10:40:47Z",
},
},
{
"a: [1, 2]\n",
map[string][]int{
2019-10-21 01:28:48 +09:00
"a": {1, 2},
2019-10-18 00:00:58 +09:00
},
},
{
"a: {b: c, d: e}\n",
map[string]interface{}{
"a": map[string]string{
"b": "c",
"d": "e",
},
},
},
{
"a: 3s\n",
map[string]string{
"a": "3s",
},
},
{
"a: <foo>\n",
map[string]string{"a": "<foo>"},
},
{
"a: \"1:1\"\n",
map[string]string{"a": "1:1"},
},
{
"a: 1.2.3.4\n",
map[string]string{"a": "1.2.3.4"},
},
{
"a: 'b: c'\n",
map[string]string{"a": "b: c"},
},
{
"a: 'Hello #comment'\n",
map[string]string{"a": "Hello #comment"},
},
{
"a: 100.5\n",
map[string]interface{}{
"a": 100.5,
},
},
{
"a: \"\\0\"\n",
map[string]string{"a": "\\0"},
},
{
"b: 2\na: 1\nd: 4\nc: 3\nsub:\n e: 5\n",
map[string]interface{}{
"b": 2,
"a": 1,
"d": 4,
"c": 3,
"sub": map[string]int{
"e": 5,
},
},
},
{
" a : b \n",
map[string]string{"a": "b"},
},
{
"a: b # comment\nb: c\n",
map[string]string{
"a": "b",
"b": "c",
},
},
{
"---\na: b\n",
map[string]string{"a": "b"},
},
{
"a: b\n...\n",
map[string]string{"a": "b"},
},
{
"%YAML 1.2\n---\n",
(*struct{})(nil),
},
{
"a: !!binary gIGC\n",
map[string]string{"a": "\x80\x81\x82"},
},
{
"a: !!binary |\n " + strings.Repeat("kJCQ", 17) + "kJ\n CQ\n",
map[string]string{"a": strings.Repeat("\x90", 54)},
},
{
"v:\n- A\n- |-\n B\n C\n",
map[string][]string{
2019-10-21 01:28:48 +09:00
"v": {
2019-10-18 00:00:58 +09:00
"A", "B\nC",
},
},
},
{
"v:\n- A\n- >-\n B\n C\n",
map[string][]string{
2019-10-21 01:28:48 +09:00
"v": {
2019-10-18 00:00:58 +09:00
"A", "B C",
},
},
},
2019-10-18 12:55:38 +09:00
{
"a: b\nc: d\n",
2019-10-18 13:05:36 +09:00
struct {
2019-10-18 12:55:38 +09:00
A string
C string `yaml:"c"`
}{
"b", "d",
},
},
{
"a: 1\nb: 2\n",
2019-10-18 13:05:36 +09:00
struct {
2019-10-18 12:55:38 +09:00
A int
B int `yaml:"-"`
}{
1, 0,
},
},
{
"a: 1\nb: 2\n",
struct {
A int
Child `yaml:",inline"`
}{
1,
Child{
B: 2,
C: 0,
},
},
},
2019-11-01 12:44:32 +09:00
// Anchors and aliases.
2019-10-18 14:27:49 +09:00
{
"a: &x 1\nb: &y 2\nc: *x\nd: *y\n",
struct{ A, B, C, D int }{1, 2, 1, 2},
},
{
"a: &a {c: 1}\nb: *a\n",
struct {
A, B struct {
C int
}
}{struct{ C int }{1}, struct{ C int }{1}},
},
{
"a: &a [1, 2]\nb: *a\n",
struct{ B []int }{[]int{1, 2}},
},
2019-11-01 12:44:32 +09:00
2019-10-26 18:39:15 +09:00
{
"tags:\n- hello-world\na: foo",
struct {
Tags []string
2019-10-28 22:12:23 +09:00
A string
2019-10-26 18:39:15 +09:00
}{Tags: []string{"hello-world"}, A: "foo"},
},
2019-10-29 16:22:55 +09:00
{
"",
(*struct{})(nil),
},
{
"{}", struct{}{},
},
2019-11-06 19:28:47 +09:00
{
"v: /a/{b}",
map[string]string{"v": "/a/{b}"},
},
{
"v: 1[]{},!%?&*",
map[string]string{"v": "1[]{},!%?&*"},
},
2019-11-07 18:08:12 +09:00
{
"v: user's item",
map[string]string{"v": "user's item"},
},
2019-11-07 13:00:00 +09:00
{
"v: [1,[2,[3,[4,5],6],7],8]",
map[string]interface{}{
"v": []interface{}{
1,
[]interface{}{
2,
[]interface{}{
3,
[]int{4, 5},
6,
},
7,
},
8,
},
},
},
{
"v: {a: {b: {c: {d: e},f: g},h: i},j: k}",
map[string]interface{}{
"v": map[string]interface{}{
"a": map[string]interface{}{
"b": map[string]interface{}{
"c": map[string]string{
"d": "e",
},
"f": "g",
},
"h": "i",
},
"j": "k",
},
},
},
2019-11-07 17:18:17 +09:00
// Multi bytes
{
2019-11-07 18:01:45 +09:00
"v: あいうえお\nv2: かきくけこ",
map[string]string{"v": "あいうえお", "v2": "かきくけこ"},
2019-11-07 17:18:17 +09:00
},
2019-10-18 00:00:58 +09:00
}
for _, test := range tests {
buf := bytes.NewBufferString(test.source)
dec := yaml.NewDecoder(buf)
typ := reflect.ValueOf(test.value).Type()
value := reflect.New(typ)
if err := dec.Decode(value.Interface()); err != nil {
t.Fatalf("%s: %+v", test.source, err)
2019-10-18 00:00:58 +09:00
}
2019-11-01 12:44:32 +09:00
actual := fmt.Sprintf("%+v", value.Elem().Interface())
expect := fmt.Sprintf("%+v", test.value)
2019-10-18 00:00:58 +09:00
if actual != expect {
2019-11-01 12:44:32 +09:00
t.Fatalf("failed to test [%s], actual=[%s], expect=[%s]", test.source, actual, expect)
2019-10-18 00:00:58 +09:00
}
}
}
func TestDecoder_TypeConversionError(t *testing.T) {
t.Run("type conversion for struct", func(t *testing.T) {
type T struct {
A int
B uint
C float32
D bool
}
type U struct {
*T `yaml:",inline"`
}
t.Run("string to int", func(t *testing.T) {
var v T
err := yaml.Unmarshal([]byte(`a: str`), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal string into Go struct field T.A of type int"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
})
t.Run("string to bool", func(t *testing.T) {
var v T
err := yaml.Unmarshal([]byte(`d: str`), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal string into Go struct field T.D of type bool"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
})
t.Run("string to int at inline", func(t *testing.T) {
var v U
err := yaml.Unmarshal([]byte(`a: str`), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal string into Go struct field U.T.A of type int"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
})
})
t.Run("type conversion for array", func(t *testing.T) {
t.Run("string to int", func(t *testing.T) {
var v map[string][]int
err := yaml.Unmarshal([]byte(`v: [A,1,C]`), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal string into Go value of type int"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
if len(v) == 0 || len(v["v"]) == 0 {
t.Fatal("failed to decode value")
}
if v["v"][0] != 1 {
t.Fatal("failed to decode value")
}
})
t.Run("string to int", func(t *testing.T) {
var v map[string][]int
err := yaml.Unmarshal([]byte("v:\n - A\n - 1\n - C"), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal string into Go value of type int"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
if len(v) == 0 || len(v["v"]) == 0 {
t.Fatal("failed to decode value")
}
if v["v"][0] != 1 {
t.Fatal("failed to decode value")
}
})
})
t.Run("overflow error", func(t *testing.T) {
t.Run("negative number to uint", func(t *testing.T) {
var v map[string]uint
err := yaml.Unmarshal([]byte("v: -42"), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal -42 into Go value of type uint ( overflow )"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
if v["v"] != 0 {
t.Fatal("failed to decode value")
}
})
t.Run("negative number to uint64", func(t *testing.T) {
var v map[string]uint64
err := yaml.Unmarshal([]byte("v: -4294967296"), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal -4294967296 into Go value of type uint64 ( overflow )"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
if v["v"] != 0 {
t.Fatal("failed to decode value")
}
})
t.Run("larger number for int32", func(t *testing.T) {
var v map[string]int32
err := yaml.Unmarshal([]byte("v: 4294967297"), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal 4294967297 into Go value of type int32 ( overflow )"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
if v["v"] != 0 {
t.Fatal("failed to decode value")
}
})
t.Run("larger number for int8", func(t *testing.T) {
var v map[string]int8
err := yaml.Unmarshal([]byte("v: 128"), &v)
if err == nil {
t.Fatal("expected to error")
}
msg := "cannot unmarshal 128 into Go value of type int8 ( overflow )"
if err.Error() != msg {
t.Fatalf("unexpected error message: %s. expect: %s", err.Error(), msg)
}
if v["v"] != 0 {
t.Fatal("failed to decode value")
}
})
})
}
2019-10-30 17:26:26 +09:00
func TestDecoder_AnchorReferenceDirs(t *testing.T) {
buf := bytes.NewBufferString("a: *a\n")
dec := yaml.NewDecoder(buf, yaml.ReferenceDirs("testdata"))
var v struct {
A struct {
B int
C string
}
}
if err := dec.Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
2019-10-30 17:26:26 +09:00
if v.A.B != 1 {
t.Fatal("failed to decode by reference dirs")
}
if v.A.C != "hello" {
t.Fatal("failed to decode by reference dirs")
}
}
func TestDecoder_AnchorReferenceDirsRecursive(t *testing.T) {
buf := bytes.NewBufferString("a: *a\n")
dec := yaml.NewDecoder(
buf,
yaml.RecursiveDir(true),
yaml.ReferenceDirs("testdata"),
)
var v struct {
A struct {
B int
C string
}
}
if err := dec.Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
if v.A.B != 1 {
t.Fatal("failed to decode by reference dirs")
}
if v.A.C != "hello" {
t.Fatal("failed to decode by reference dirs")
}
}
func TestDecoder_AnchorFiles(t *testing.T) {
buf := bytes.NewBufferString("a: *a\n")
dec := yaml.NewDecoder(buf, yaml.ReferenceFiles("testdata/anchor.yml"))
var v struct {
A struct {
B int
C string
}
}
if err := dec.Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
if v.A.B != 1 {
t.Fatal("failed to decode by reference dirs")
}
if v.A.C != "hello" {
t.Fatal("failed to decode by reference dirs")
}
}
2019-10-21 14:54:26 +09:00
func TestDecodeWithMergeKey(t *testing.T) {
yml := `
a: &a
b: 1
c: hello
items:
- <<: *a
- <<: *a
c: world
`
type Item struct {
B int
C string
}
type T struct {
Items []*Item
}
buf := bytes.NewBufferString(yml)
dec := yaml.NewDecoder(buf)
var v T
if err := dec.Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
if len(v.Items) != 2 {
t.Fatal("failed to decode with merge key")
}
if v.Items[0].B != 1 || v.Items[0].C != "hello" {
t.Fatal("failed to decode with merge key")
}
if v.Items[1].B != 1 || v.Items[1].C != "world" {
t.Fatal("failed to decode with merge key")
}
2020-03-02 16:18:13 +09:00
t.Run("decode with interface{}", func(t *testing.T) {
buf := bytes.NewBufferString(yml)
dec := yaml.NewDecoder(buf)
var v interface{}
if err := dec.Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
items := v.(map[string]interface{})["items"].([]interface{})
if len(items) != 2 {
t.Fatal("failed to decode with merge key")
}
b0 := items[0].(map[string]interface{})["b"]
if _, ok := b0.(uint64); !ok {
t.Fatal("failed to decode with merge key")
}
if b0.(uint64) != 1 {
t.Fatal("failed to decode with merge key")
}
c0 := items[0].(map[string]interface{})["c"]
if _, ok := c0.(string); !ok {
t.Fatal("failed to decode with merge key")
}
if c0.(string) != "hello" {
t.Fatal("failed to decode with merge key")
}
b1 := items[1].(map[string]interface{})["b"]
if _, ok := b1.(uint64); !ok {
t.Fatal("failed to decode with merge key")
}
if b1.(uint64) != 1 {
t.Fatal("failed to decode with merge key")
}
c1 := items[1].(map[string]interface{})["c"]
if _, ok := c1.(string); !ok {
t.Fatal("failed to decode with merge key")
}
if c1.(string) != "world" {
t.Fatal("failed to decode with merge key")
}
})
t.Run("decode with map", func(t *testing.T) {
var v struct {
Items []map[string]interface{}
}
buf := bytes.NewBufferString(yml)
dec := yaml.NewDecoder(buf)
if err := dec.Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
if len(v.Items) != 2 {
t.Fatal("failed to decode with merge key")
}
b0 := v.Items[0]["b"]
if _, ok := b0.(uint64); !ok {
t.Fatal("failed to decode with merge key")
}
if b0.(uint64) != 1 {
t.Fatal("failed to decode with merge key")
}
c0 := v.Items[0]["c"]
if _, ok := c0.(string); !ok {
t.Fatal("failed to decode with merge key")
}
if c0.(string) != "hello" {
t.Fatal("failed to decode with merge key")
}
b1 := v.Items[1]["b"]
if _, ok := b1.(uint64); !ok {
t.Fatal("failed to decode with merge key")
}
if b1.(uint64) != 1 {
t.Fatal("failed to decode with merge key")
}
c1 := v.Items[1]["c"]
if _, ok := c1.(string); !ok {
t.Fatal("failed to decode with merge key")
}
if c1.(string) != "world" {
t.Fatal("failed to decode with merge key")
}
})
2019-10-21 14:54:26 +09:00
}
2019-10-24 11:07:40 +09:00
2019-10-28 22:12:23 +09:00
func TestDecoder_Inline(t *testing.T) {
type Base struct {
A int
B string
}
yml := `---
a: 1
b: hello
c: true
`
var v struct {
*Base `yaml:",inline"`
C bool
}
if err := yaml.NewDecoder(strings.NewReader(yml)).Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
if v.A != 1 {
t.Fatal("failed to decode with inline key")
}
if v.B != "hello" {
t.Fatal("failed to decode with inline key")
}
if !v.C {
t.Fatal("failed to decode with inline key")
}
}
func TestDecoder_InlineAndConflictKey(t *testing.T) {
type Base struct {
A int
B string
}
yml := `---
a: 1
b: hello
c: true
`
var v struct {
*Base `yaml:",inline"`
A int
C bool
}
if err := yaml.NewDecoder(strings.NewReader(yml)).Decode(&v); err != nil {
t.Fatalf("%+v", err)
}
if v.A != 1 {
t.Fatal("failed to decode with inline key")
}
if v.B != "hello" {
t.Fatal("failed to decode with inline key")
}
if !v.C {
t.Fatal("failed to decode with inline key")
}
if v.Base.A != 0 {
t.Fatal("failed to decode with inline key")
}
}
2019-10-24 11:07:40 +09:00
func TestDecoder_InvalidCases(t *testing.T) {
const src = `---
a:
- b
c: d
`
var v struct {
A []string
}
err := yaml.NewDecoder(strings.NewReader(src)).Decode(&v)
if err == nil {
t.Fatalf("expected error")
}
if err.Error() != yaml.FormatError(err, false, true) {
t.Logf("err.Error() = %s", err.Error())
t.Logf("yaml.FormatError(err, false, true) = %s", yaml.FormatError(err, false, true))
t.Fatal(`err.Error() should match yaml.FormatError(err, false, true)`)
}
2019-10-24 11:07:40 +09:00
//TODO: properly check if errors are colored/have source
t.Logf("%s", err)
t.Logf("%s", yaml.FormatError(err, true, false))
t.Logf("%s", yaml.FormatError(err, false, true))
t.Logf("%s", yaml.FormatError(err, true, true))
}
2019-11-13 10:31:00 +09:00
func TestDecoder_JSONTags(t *testing.T) {
var v struct {
2019-11-14 09:12:39 +09:00
A string `json:"a_json"` // no YAML tag
2019-11-13 10:31:00 +09:00
B string `json:"b_json" yaml:"b_yaml"` // both tags
}
const src = `---
a_json: a_json_value
b_json: b_json_value
b_yaml: b_yaml_value
`
if err := yaml.NewDecoder(strings.NewReader(src)).Decode(&v); err != nil {
t.Fatalf(`parsing should succeed: %s`, err)
}
if v.A != "a_json_value" {
t.Fatalf("v.A should be `a_json_value`, got `%s`", v.A)
}
if v.B != "b_yaml_value" {
t.Fatalf("v.B should be `b_yaml_value`, got `%s`", v.B)
}
}
2019-11-14 09:12:39 +09:00
func TestDecoder_DisallowUnknownField(t *testing.T) {
2019-12-13 15:00:04 +09:00
t.Run("different level keys with same name", func(t *testing.T) {
var v struct {
C Child `yaml:"c"`
}
yml := `---
b: 1
c:
b: 1
`
err := yaml.NewDecoder(strings.NewReader(yml), yaml.DisallowUnknownField()).Decode(&v)
if err == nil {
t.Fatalf("error expected")
}
})
t.Run("inline", func(t *testing.T) {
var v struct {
Child `yaml:",inline"`
A string `yaml:"a"`
}
yml := `---
a: a
b: 1
`
if err := yaml.NewDecoder(strings.NewReader(yml), yaml.DisallowUnknownField()).Decode(&v); err != nil {
t.Fatalf(`parsing should succeed: %s`, err)
}
if v.A != "a" {
t.Fatalf("v.A should be `a`, got `%s`", v.A)
}
if v.B != 1 {
t.Fatalf("v.B should be 1, got %d", v.B)
}
if v.C != 0 {
t.Fatalf("v.C should be 0, got %d", v.C)
}
})
2019-12-18 13:26:59 +09:00
t.Run("list", func(t *testing.T) {
type C struct {
Child `yaml:",inline"`
}
var v struct {
Children []C `yaml:"children"`
}
yml := `---
children:
- b: 1
- b: 2
`
if err := yaml.NewDecoder(strings.NewReader(yml), yaml.DisallowUnknownField()).Decode(&v); err != nil {
t.Fatalf(`parsing should succeed: %s`, err)
}
if len(v.Children) != 2 {
t.Fatalf(`len(v.Children) should be 2, got %d`, len(v.Children))
}
if v.Children[0].B != 1 {
t.Fatalf(`v.Children[0].B should be 1, got %d`, v.Children[0].B)
}
if v.Children[1].B != 2 {
t.Fatalf(`v.Children[1].B should be 2, got %d`, v.Children[1].B)
}
})
}
func TestDecoder_DefaultValues(t *testing.T) {
v := struct {
A string `yaml:"a"`
B string `yaml:"b"`
c string // private
}{
B: "defaultBValue",
c: "defaultCValue",
}
const src = `---
a: a_value
`
if err := yaml.NewDecoder(strings.NewReader(src)).Decode(&v); err != nil {
t.Fatalf(`parsing should succeed: %s`, err)
}
if v.A != "a_value" {
t.Fatalf("v.A should be `a_value`, got `%s`", v.A)
}
if v.B != "defaultBValue" {
t.Fatalf("v.B should be `defaultValue`, got `%s`", v.B)
}
if v.c != "defaultCValue" {
t.Fatalf("v.c should be `defaultCValue`, got `%s`", v.c)
}
}
2019-11-14 09:12:39 +09:00
func Example_YAMLTags() {
yml := `---
foo: 1
bar: c
A: 2
B: d
`
var v struct {
A int `yaml:"foo" json:"A"`
B string `yaml:"bar" json:"B"`
}
if err := yaml.Unmarshal([]byte(yml), &v); err != nil {
log.Fatal(err)
}
fmt.Println(v.A)
fmt.Println(v.B)
// OUTPUT:
// 1
// c
}
func Example_JSONTags() {
yml := `---
foo: 1
bar: c
`
var v struct {
A int `json:"foo"`
B string `json:"bar"`
}
if err := yaml.Unmarshal([]byte(yml), &v); err != nil {
log.Fatal(err)
}
fmt.Println(v.A)
fmt.Println(v.B)
// OUTPUT:
// 1
// c
}
2019-12-10 18:41:23 +09:00
func Example_DisallowUnknownField() {
var v struct {
A string `yaml:"simple"`
2019-12-11 13:07:34 +09:00
C string `yaml:"complicated"`
2019-12-10 18:41:23 +09:00
}
const src = `---
simple: string
2019-12-11 13:07:34 +09:00
complecated: string
2019-12-10 18:41:23 +09:00
`
err := yaml.NewDecoder(strings.NewReader(src), yaml.DisallowUnknownField()).Decode(&v)
fmt.Printf("%v\n", err)
// OUTPUT:
2019-12-11 13:07:34 +09:00
// [3:1] unknown field "complecated"
2019-12-10 18:41:23 +09:00
// 1 | ---
// 2 | simple: string
2019-12-11 13:07:34 +09:00
// > 3 | complecated: string
2019-12-10 18:41:23 +09:00
// ^
}
2020-01-14 22:50:50 +09:00
type unmarshalableStringValue string
func (v *unmarshalableStringValue) UnmarshalYAML(raw []byte) error {
*v = unmarshalableStringValue(string(raw))
return nil
}
type unmarshalableStringContainer struct {
V unmarshalableStringValue `yaml:"value" json:"value"`
}
func TestUnmarshalableString(t *testing.T) {
t.Run("empty string", func(t *testing.T) {
t.Parallel()
var container unmarshalableStringContainer
if err := yaml.Unmarshal([]byte(`value: ""`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != "" {
t.Fatalf("expected empty string, but %q is set", container.V)
}
})
t.Run("filled string", func(t *testing.T) {
t.Parallel()
var container unmarshalableStringContainer
if err := yaml.Unmarshal([]byte(`value: "aaa"`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != "aaa" {
t.Fatalf("expected \"aaa\", but %q is set", container.V)
}
})
t.Run("single-quoted string", func(t *testing.T) {
t.Parallel()
var container unmarshalableStringContainer
if err := yaml.Unmarshal([]byte(`value: 'aaa'`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != "aaa" {
t.Fatalf("expected \"aaa\", but %q is set", container.V)
}
})
}
type unmarshalablePtrStringContainer struct {
V *string `yaml:"value" json:"value"`
}
func TestUnmarshalablePtrString(t *testing.T) {
t.Run("empty string", func(t *testing.T) {
t.Parallel()
var container unmarshalablePtrStringContainer
if err := yaml.Unmarshal([]byte(`value: ""`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if *container.V != "" {
t.Fatalf("expected empty string, but %q is set", *container.V)
}
})
t.Run("null", func(t *testing.T) {
t.Parallel()
var container unmarshalablePtrStringContainer
if err := yaml.Unmarshal([]byte(`value: null`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != (*string)(nil) {
t.Fatalf("expected nil, but %q is set", *container.V)
}
})
}
2020-01-14 22:50:50 +09:00
type unmarshalableIntValue int
func (v *unmarshalableIntValue) UnmarshalYAML(raw []byte) error {
i, err := strconv.Atoi(string(raw))
if err != nil {
return err
}
*v = unmarshalableIntValue(i)
return nil
}
type unmarshalableIntContainer struct {
V unmarshalableIntValue `yaml:"value" json:"value"`
}
func TestUnmarshalableInt(t *testing.T) {
t.Run("empty int", func(t *testing.T) {
t.Parallel()
var container unmarshalableIntContainer
if err := yaml.Unmarshal([]byte(``), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != 0 {
t.Fatalf("expected empty int, but %d is set", container.V)
}
})
t.Run("filled int", func(t *testing.T) {
t.Parallel()
var container unmarshalableIntContainer
if err := yaml.Unmarshal([]byte(`value: 9`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != 9 {
t.Fatalf("expected 9, but %d is set", container.V)
}
})
t.Run("filled number", func(t *testing.T) {
t.Parallel()
var container unmarshalableIntContainer
if err := yaml.Unmarshal([]byte(`value: 9`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != 9 {
t.Fatalf("expected 9, but %d is set", container.V)
}
})
}
type unmarshalablePtrIntContainer struct {
V *int `yaml:"value" json:"value"`
}
func TestUnmarshalablePtrInt(t *testing.T) {
t.Run("empty int", func(t *testing.T) {
t.Parallel()
var container unmarshalablePtrIntContainer
if err := yaml.Unmarshal([]byte(`value: 0`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if *container.V != 0 {
t.Fatalf("expected 0, but %q is set", *container.V)
}
})
t.Run("null", func(t *testing.T) {
t.Parallel()
var container unmarshalablePtrIntContainer
if err := yaml.Unmarshal([]byte(`value: null`), &container); err != nil {
t.Fatalf("failed to unmarshal %v", err)
}
if container.V != (*int)(nil) {
t.Fatalf("expected nil, but %q is set", *container.V)
}
})
}