jsoniter_object_test.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. package jsoniter
  2. import (
  3. "encoding/json"
  4. "testing"
  5. "github.com/json-iterator/go/require"
  6. "bytes"
  7. )
  8. func Test_empty_object(t *testing.T) {
  9. should := require.New(t)
  10. iter := ParseString(`{}`)
  11. field := iter.ReadObject()
  12. should.Equal("", field)
  13. iter = ParseString(`{}`)
  14. iter.ReadObjectCB(func(iter *Iterator, field string) bool {
  15. should.FailNow("should not call")
  16. return true
  17. })
  18. }
  19. func Test_one_field(t *testing.T) {
  20. should := require.New(t)
  21. iter := ParseString(`{"a": "b"}`)
  22. field := iter.ReadObject()
  23. should.Equal("a", field)
  24. value := iter.ReadString()
  25. should.Equal("b", value)
  26. field = iter.ReadObject()
  27. should.Equal("", field)
  28. iter = ParseString(`{"a": "b"}`)
  29. should.True(iter.ReadObjectCB(func(iter *Iterator, field string) bool {
  30. should.Equal("a", field)
  31. return true
  32. }))
  33. }
  34. func Test_two_field(t *testing.T) {
  35. should := require.New(t)
  36. iter := ParseString(`{ "a": "b" , "c": "d" }`)
  37. field := iter.ReadObject()
  38. should.Equal("a", field)
  39. value := iter.ReadString()
  40. should.Equal("b", value)
  41. field = iter.ReadObject()
  42. should.Equal("c", field)
  43. value = iter.ReadString()
  44. should.Equal("d", value)
  45. field = iter.ReadObject()
  46. should.Equal("", field)
  47. iter = ParseString(`{"field1": "1", "field2": 2}`)
  48. for field := iter.ReadObject(); field != ""; field = iter.ReadObject() {
  49. switch field {
  50. case "field1":
  51. iter.ReadString()
  52. case "field2":
  53. iter.ReadInt64()
  54. default:
  55. iter.reportError("bind object", "unexpected field")
  56. }
  57. }
  58. }
  59. func Test_read_object_as_any(t *testing.T) {
  60. should := require.New(t)
  61. any, err := UnmarshalAnyFromString(`{"a":"b","c":"d"}`)
  62. should.Nil(err)
  63. should.Equal(`{"a":"b","c":"d"}`, any.ToString())
  64. // partial parse
  65. should.Equal("b", any.Get("a").ToString())
  66. should.Equal("d", any.Get("c").ToString())
  67. should.Equal(2, len(any.Keys()))
  68. any, err = UnmarshalAnyFromString(`{"a":"b","c":"d"}`)
  69. // full parse
  70. should.Equal(2, len(any.Keys()))
  71. }
  72. func Test_object_any_lazy_iterator(t *testing.T) {
  73. should := require.New(t)
  74. any, err := UnmarshalAnyFromString(`{"a":"b","c":"d"}`)
  75. should.Nil(err)
  76. // iterator parse
  77. vals := map[string]string{}
  78. var k string
  79. var v Any
  80. next, hasNext := any.IterateObject()
  81. should.True(hasNext)
  82. k, v, hasNext = next()
  83. should.True(hasNext)
  84. vals[k] = v.ToString()
  85. // trigger full parse
  86. should.Equal(2, len(any.Keys()))
  87. k, v, hasNext = next()
  88. should.False(hasNext)
  89. vals[k] = v.ToString()
  90. should.Equal(map[string]string{"a":"b", "c":"d"}, vals)
  91. vals = map[string]string{}
  92. for next, hasNext := any.IterateObject(); hasNext; k, v, hasNext = next() {
  93. vals[k] = v.ToString()
  94. }
  95. should.Equal(map[string]string{"a":"b", "c":"d"}, vals)
  96. }
  97. func Test_object_any_with_two_lazy_iterators(t *testing.T) {
  98. should := require.New(t)
  99. any, err := UnmarshalAnyFromString(`{"a":"b","c":"d","e":"f"}`)
  100. should.Nil(err)
  101. var k string
  102. var v Any
  103. next1, hasNext1 := any.IterateObject()
  104. next2, hasNext2 := any.IterateObject()
  105. should.True(hasNext1)
  106. k, v, hasNext1 = next1()
  107. should.True(hasNext1)
  108. should.Equal("a", k)
  109. should.Equal("b", v.ToString())
  110. should.True(hasNext2)
  111. k, v, hasNext2 = next2()
  112. should.True(hasNext2)
  113. should.Equal("a", k)
  114. should.Equal("b", v.ToString())
  115. k, v, hasNext1 = next1()
  116. should.True(hasNext1)
  117. should.Equal("c", k)
  118. should.Equal("d", v.ToString())
  119. k, v, hasNext2 = next2()
  120. should.True(hasNext2)
  121. should.Equal("c", k)
  122. should.Equal("d", v.ToString())
  123. }
  124. func Test_write_object(t *testing.T) {
  125. should := require.New(t)
  126. buf := &bytes.Buffer{}
  127. stream := NewStream(buf, 4096)
  128. stream.IndentionStep = 2
  129. stream.WriteObjectStart()
  130. stream.WriteObjectField("hello")
  131. stream.WriteInt(1)
  132. stream.WriteMore()
  133. stream.WriteObjectField("world")
  134. stream.WriteInt(2)
  135. stream.WriteObjectEnd()
  136. stream.Flush()
  137. should.Nil(stream.Error)
  138. should.Equal("{\n \"hello\":1,\n \"world\":2\n}", buf.String())
  139. }
  140. type TestObj struct {
  141. Field1 string
  142. Field2 uint64
  143. }
  144. func Benchmark_jsoniter_object(b *testing.B) {
  145. for n := 0; n < b.N; n++ {
  146. iter := ParseString(`{"field1": "1", "field2": 2}`)
  147. obj := TestObj{}
  148. for field := iter.ReadObject(); field != ""; field = iter.ReadObject() {
  149. switch field {
  150. case "field1":
  151. obj.Field1 = iter.ReadString()
  152. case "field2":
  153. obj.Field2 = iter.ReadUint64()
  154. default:
  155. iter.reportError("bind object", "unexpected field")
  156. }
  157. }
  158. }
  159. }
  160. func Benchmark_json_object(b *testing.B) {
  161. for n := 0; n < b.N; n++ {
  162. result := TestObj{}
  163. json.Unmarshal([]byte(`{"field1": "1", "field2": 2}`), &result)
  164. }
  165. }