encode_test.go 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. package yaml_test
  2. import (
  3. "fmt"
  4. . "gopkg.in/check.v1"
  5. "gopkg.in/yaml.v2"
  6. "math"
  7. "strconv"
  8. "strings"
  9. "time"
  10. )
  11. var marshalIntTest = 123
  12. var marshalTests = []struct {
  13. value interface{}
  14. data string
  15. }{
  16. {
  17. &struct{}{},
  18. "{}\n",
  19. }, {
  20. map[string]string{"v": "hi"},
  21. "v: hi\n",
  22. }, {
  23. map[string]interface{}{"v": "hi"},
  24. "v: hi\n",
  25. }, {
  26. map[string]string{"v": "true"},
  27. "v: \"true\"\n",
  28. }, {
  29. map[string]string{"v": "false"},
  30. "v: \"false\"\n",
  31. }, {
  32. map[string]interface{}{"v": true},
  33. "v: true\n",
  34. }, {
  35. map[string]interface{}{"v": false},
  36. "v: false\n",
  37. }, {
  38. map[string]interface{}{"v": 10},
  39. "v: 10\n",
  40. }, {
  41. map[string]interface{}{"v": -10},
  42. "v: -10\n",
  43. }, {
  44. map[string]uint{"v": 42},
  45. "v: 42\n",
  46. }, {
  47. map[string]interface{}{"v": int64(4294967296)},
  48. "v: 4294967296\n",
  49. }, {
  50. map[string]int64{"v": int64(4294967296)},
  51. "v: 4294967296\n",
  52. }, {
  53. map[string]uint64{"v": 4294967296},
  54. "v: 4294967296\n",
  55. }, {
  56. map[string]interface{}{"v": "10"},
  57. "v: \"10\"\n",
  58. }, {
  59. map[string]interface{}{"v": 0.1},
  60. "v: 0.1\n",
  61. }, {
  62. map[string]interface{}{"v": float64(0.1)},
  63. "v: 0.1\n",
  64. }, {
  65. map[string]interface{}{"v": -0.1},
  66. "v: -0.1\n",
  67. }, {
  68. map[string]interface{}{"v": math.Inf(+1)},
  69. "v: .inf\n",
  70. }, {
  71. map[string]interface{}{"v": math.Inf(-1)},
  72. "v: -.inf\n",
  73. }, {
  74. map[string]interface{}{"v": math.NaN()},
  75. "v: .nan\n",
  76. }, {
  77. map[string]interface{}{"v": nil},
  78. "v: null\n",
  79. }, {
  80. map[string]interface{}{"v": ""},
  81. "v: \"\"\n",
  82. }, {
  83. map[string][]string{"v": []string{"A", "B"}},
  84. "v:\n- A\n- B\n",
  85. }, {
  86. map[string][]string{"v": []string{"A", "B\nC"}},
  87. "v:\n- A\n- |-\n B\n C\n",
  88. }, {
  89. map[string][]interface{}{"v": []interface{}{"A", 1, map[string][]int{"B": []int{2, 3}}}},
  90. "v:\n- A\n- 1\n- B:\n - 2\n - 3\n",
  91. }, {
  92. map[string]interface{}{"a": map[interface{}]interface{}{"b": "c"}},
  93. "a:\n b: c\n",
  94. }, {
  95. map[string]interface{}{"a": "-"},
  96. "a: '-'\n",
  97. },
  98. // Simple values.
  99. {
  100. &marshalIntTest,
  101. "123\n",
  102. },
  103. // Structures
  104. {
  105. &struct{ Hello string }{"world"},
  106. "hello: world\n",
  107. }, {
  108. &struct {
  109. A struct {
  110. B string
  111. }
  112. }{struct{ B string }{"c"}},
  113. "a:\n b: c\n",
  114. }, {
  115. &struct {
  116. A *struct {
  117. B string
  118. }
  119. }{&struct{ B string }{"c"}},
  120. "a:\n b: c\n",
  121. }, {
  122. &struct {
  123. A *struct {
  124. B string
  125. }
  126. }{},
  127. "a: null\n",
  128. }, {
  129. &struct{ A int }{1},
  130. "a: 1\n",
  131. }, {
  132. &struct{ A []int }{[]int{1, 2}},
  133. "a:\n- 1\n- 2\n",
  134. }, {
  135. &struct {
  136. B int "a"
  137. }{1},
  138. "a: 1\n",
  139. }, {
  140. &struct{ A bool }{true},
  141. "a: true\n",
  142. },
  143. // Conditional flag
  144. {
  145. &struct {
  146. A int "a,omitempty"
  147. B int "b,omitempty"
  148. }{1, 0},
  149. "a: 1\n",
  150. }, {
  151. &struct {
  152. A int "a,omitempty"
  153. B int "b,omitempty"
  154. }{0, 0},
  155. "{}\n",
  156. }, {
  157. &struct {
  158. A *struct{ X int } "a,omitempty"
  159. B int "b,omitempty"
  160. }{nil, 0},
  161. "{}\n",
  162. },
  163. // Flow flag
  164. {
  165. &struct {
  166. A []int "a,flow"
  167. }{[]int{1, 2}},
  168. "a: [1, 2]\n",
  169. }, {
  170. &struct {
  171. A map[string]string "a,flow"
  172. }{map[string]string{"b": "c", "d": "e"}},
  173. "a: {b: c, d: e}\n",
  174. }, {
  175. &struct {
  176. A struct {
  177. B, D string
  178. } "a,flow"
  179. }{struct{ B, D string }{"c", "e"}},
  180. "a: {b: c, d: e}\n",
  181. },
  182. // Unexported field
  183. {
  184. &struct {
  185. u int
  186. A int
  187. }{0, 1},
  188. "a: 1\n",
  189. },
  190. // Ignored field
  191. {
  192. &struct {
  193. A int
  194. B int "-"
  195. }{1, 2},
  196. "a: 1\n",
  197. },
  198. // Struct inlining
  199. {
  200. &struct {
  201. A int
  202. C inlineB `yaml:",inline"`
  203. }{1, inlineB{2, inlineC{3}}},
  204. "a: 1\nb: 2\nc: 3\n",
  205. },
  206. // Duration
  207. {
  208. map[string]time.Duration{"a": 3 * time.Second},
  209. "a: 3s\n",
  210. },
  211. // Issue #24: bug in map merging logic.
  212. {
  213. map[string]string{"a": "<foo>"},
  214. "a: <foo>\n",
  215. },
  216. // Issue #34: marshal unsupported base 60 floats quoted for compatibility
  217. // with old YAML 1.1 parsers.
  218. {
  219. map[string]string{"a": "1:1"},
  220. "a: \"1:1\"\n",
  221. },
  222. // Binary data.
  223. {
  224. map[string]string{"a": "\x00"},
  225. "a: \"\\0\"\n",
  226. }, {
  227. map[string]string{"a": "\x80\x81\x82"},
  228. "a: !!binary gIGC\n",
  229. }, {
  230. map[string]string{"a": strings.Repeat("\x90", 54)},
  231. "a: !!binary |\n " + strings.Repeat("kJCQ", 17) + "kJ\n CQ\n",
  232. },
  233. // Ordered maps.
  234. {
  235. &yaml.MapSlice{{"b", 2}, {"a", 1}, {"d", 4}, {"c", 3}, {"sub", yaml.MapSlice{{"e", 5}}}},
  236. "b: 2\na: 1\nd: 4\nc: 3\nsub:\n e: 5\n",
  237. },
  238. // Encode unicode as utf-8 rather than in escaped form.
  239. {
  240. map[string]string{"a": "你好"},
  241. "a: 你好\n",
  242. },
  243. }
  244. func (s *S) TestMarshal(c *C) {
  245. for _, item := range marshalTests {
  246. data, err := yaml.Marshal(item.value)
  247. c.Assert(err, IsNil)
  248. c.Assert(string(data), Equals, item.data)
  249. }
  250. }
  251. var marshalErrorTests = []struct {
  252. value interface{}
  253. error string
  254. panic string
  255. }{{
  256. value: &struct {
  257. B int
  258. inlineB ",inline"
  259. }{1, inlineB{2, inlineC{3}}},
  260. panic: `Duplicated key 'b' in struct struct \{ B int; .*`,
  261. }}
  262. func (s *S) TestMarshalErrors(c *C) {
  263. for _, item := range marshalErrorTests {
  264. if item.panic != "" {
  265. c.Assert(func() { yaml.Marshal(item.value) }, PanicMatches, item.panic)
  266. } else {
  267. _, err := yaml.Marshal(item.value)
  268. c.Assert(err, ErrorMatches, item.error)
  269. }
  270. }
  271. }
  272. func (s *S) TestMarshalTypeCache(c *C) {
  273. var data []byte
  274. var err error
  275. func() {
  276. type T struct{ A int }
  277. data, err = yaml.Marshal(&T{})
  278. c.Assert(err, IsNil)
  279. }()
  280. func() {
  281. type T struct{ B int }
  282. data, err = yaml.Marshal(&T{})
  283. c.Assert(err, IsNil)
  284. }()
  285. c.Assert(string(data), Equals, "b: 0\n")
  286. }
  287. var marshalerTests = []struct {
  288. data string
  289. value interface{}
  290. }{
  291. {"_:\n hi: there\n", map[interface{}]interface{}{"hi": "there"}},
  292. {"_:\n- 1\n- A\n", []interface{}{1, "A"}},
  293. {"_: 10\n", 10},
  294. {"_: null\n", nil},
  295. {"_: BAR!\n", "BAR!"},
  296. }
  297. type marshalerType struct {
  298. value interface{}
  299. }
  300. func (o marshalerType) MarshalYAML() (interface{}, error) {
  301. return o.value, nil
  302. }
  303. type marshalerValue struct {
  304. Field marshalerType "_"
  305. }
  306. func (s *S) TestMarshaler(c *C) {
  307. for _, item := range marshalerTests {
  308. obj := &marshalerValue{}
  309. obj.Field.value = item.value
  310. data, err := yaml.Marshal(obj)
  311. c.Assert(err, IsNil)
  312. c.Assert(string(data), Equals, string(item.data))
  313. }
  314. }
  315. func (s *S) TestMarshalerWholeDocument(c *C) {
  316. obj := &marshalerType{}
  317. obj.value = map[string]string{"hello": "world!"}
  318. data, err := yaml.Marshal(obj)
  319. c.Assert(err, IsNil)
  320. c.Assert(string(data), Equals, "hello: world!\n")
  321. }
  322. type failingMarshaler struct{}
  323. func (ft *failingMarshaler) MarshalYAML() (interface{}, error) {
  324. return nil, failingErr
  325. }
  326. func (s *S) TestMarshalerError(c *C) {
  327. _, err := yaml.Marshal(&failingMarshaler{})
  328. c.Assert(err, Equals, failingErr)
  329. }
  330. func (s *S) TestSortedOutput(c *C) {
  331. order := []interface{}{
  332. false,
  333. true,
  334. 1,
  335. uint(1),
  336. 1.0,
  337. 1.1,
  338. 1.2,
  339. 2,
  340. uint(2),
  341. 2.0,
  342. 2.1,
  343. "",
  344. ".1",
  345. ".2",
  346. ".a",
  347. "1",
  348. "2",
  349. "a!10",
  350. "a/2",
  351. "a/10",
  352. "a~10",
  353. "ab/1",
  354. "b/1",
  355. "b/01",
  356. "b/2",
  357. "b/02",
  358. "b/3",
  359. "b/03",
  360. "b1",
  361. "b01",
  362. "b3",
  363. "c2.10",
  364. "c10.2",
  365. "d1",
  366. "d12",
  367. "d12a",
  368. }
  369. m := make(map[interface{}]int)
  370. for _, k := range order {
  371. m[k] = 1
  372. }
  373. data, err := yaml.Marshal(m)
  374. c.Assert(err, IsNil)
  375. out := "\n" + string(data)
  376. last := 0
  377. for i, k := range order {
  378. repr := fmt.Sprint(k)
  379. if s, ok := k.(string); ok {
  380. if _, err = strconv.ParseFloat(repr, 32); s == "" || err == nil {
  381. repr = `"` + repr + `"`
  382. }
  383. }
  384. index := strings.Index(out, "\n"+repr+":")
  385. if index == -1 {
  386. c.Fatalf("%#v is not in the output: %#v", k, out)
  387. }
  388. if index < last {
  389. c.Fatalf("%#v was generated before %#v: %q", k, order[i-1], out)
  390. }
  391. last = index
  392. }
  393. }