jsone_encode_tests.erl 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. %% Copyright (c) 2013-2014, Takeru Ohta <phjgt308@gmail.com>
  2. %% coding: latin-1
  3. -module(jsone_encode_tests).
  4. -include_lib("eunit/include/eunit.hrl").
  5. -ifdef('NO_MAP_TYPE').
  6. -define(OBJ0, {[]}).
  7. -define(OBJ1(K, V), {[{K, V}]}).
  8. -define(OBJ2(K1, V1, K2, V2), {[{K1, V1}, {K2, V2}]}).
  9. -else.
  10. -define(OBJ0, #{}).
  11. -define(OBJ1(K, V), #{K => V}).
  12. -define(OBJ2(K1, V1, K2, V2), #{K1 => V1, K2 => V2}).
  13. -endif.
  14. encode_test_() ->
  15. [
  16. %% Symbols
  17. {"false",
  18. fun () ->
  19. ?assertEqual({ok, <<"false">>}, jsone_encode:encode(false))
  20. end},
  21. {"true",
  22. fun () ->
  23. ?assertEqual({ok, <<"true">>}, jsone_encode:encode(true))
  24. end},
  25. {"null",
  26. fun () ->
  27. ?assertEqual({ok, <<"null">>}, jsone_encode:encode(null))
  28. end},
  29. %% Numbers: Inline json term
  30. {"json",
  31. fun () ->
  32. ?assertEqual(
  33. {ok, <<"{\"foo\":[1,2,3],\"bar\":\"",195,169,"ok\"}">>},
  34. jsone_encode:encode(
  35. ?OBJ2(foo, {{json, ["["|[$1, ",2",<<",3]">>]]}},
  36. <<"bar">>, {{json_utf8, [$", 233, "ok", $"]}}))),
  37. ?assertEqual(
  38. {ok, <<"{\"foo\":[1,2,3],\"bar\":\"",233,"ok\"}">>},
  39. jsone_encode:encode(
  40. ?OBJ2(foo, {{json, ["["|[$1, ",2",<<",3]">>]]}},
  41. <<"bar">>, {{json, [$", 233, "ok", $"]}}))),
  42. ?assertEqual(
  43. {ok, <<"{\"json\":\"[1,2,3]\"}">>},
  44. jsone_encode:encode([{json, <<"[1,2,3]">>}])),
  45. ?assertEqual(
  46. {ok, <<"[[1,2,3]]">>},
  47. jsone_encode:encode([{{json, <<"[1,2,3]">>}}]))
  48. end},
  49. %% Numbers: Integer
  50. {"zero",
  51. fun () ->
  52. ?assertEqual({ok, <<"0">>}, jsone_encode:encode(0))
  53. end},
  54. {"positive integer",
  55. fun () ->
  56. ?assertEqual({ok, <<"1">>}, jsone_encode:encode(1))
  57. end},
  58. {"negative integer",
  59. fun () ->
  60. ?assertEqual({ok, <<"-1">>}, jsone_encode:encode(-1))
  61. end},
  62. {"large number",
  63. fun () ->
  64. ?assertEqual({ok, <<"11111111111111111111111111111111111111111111111111111111111111111111111">>},
  65. jsone_encode:encode(11111111111111111111111111111111111111111111111111111111111111111111111))
  66. end},
  67. %% Numbers: Float",
  68. {"float",
  69. fun () ->
  70. Input = 1.234,
  71. ?assertMatch({ok, _}, jsone_encode:encode(Input)),
  72. ?assertEqual(Input, binary_to_float(element(2, jsone_encode:encode(Input))))
  73. end},
  74. {"float_format option",
  75. fun () ->
  76. Input = 1.23,
  77. ?assertEqual({ok, <<"1.22999999999999998224e+00">>}, jsone_encode:encode(Input)),
  78. ?assertEqual({ok, <<"1.2300e+00">>}, jsone_encode:encode(Input, [{float_format, [{scientific, 4}]}])),
  79. ?assertEqual({ok, <<"1.2e+00">>}, jsone_encode:encode(Input, [{float_format, [{scientific, 1}]}])),
  80. ?assertEqual({ok, <<"1.2300">>}, jsone_encode:encode(Input, [{float_format, [{decimals, 4}]}])),
  81. ?assertEqual({ok, <<"1.23">>}, jsone_encode:encode(Input, [{float_format, [{decimals, 4}, compact]}]))
  82. end},
  83. %% Strings
  84. {"simple string",
  85. fun () ->
  86. ?assertEqual({ok, <<"\"abc\"">>}, jsone_encode:encode(<<"abc">>))
  87. end},
  88. {"atom is regarded as string",
  89. fun () ->
  90. ?assertEqual({ok, <<"\"abc\"">>}, jsone_encode:encode(abc))
  91. end},
  92. {"string: contains escaped characters",
  93. fun () ->
  94. Input = <<"\"\/\\\b\f\n\r\t">>,
  95. Expected = list_to_binary([$", [[$\\, C] || C <- [$", $/, $\\, $b, $f, $n, $r, $t]], $"]),
  96. ?assertEqual({ok, Expected}, jsone_encode:encode(Input)),
  97. ?assertEqual({ok, Expected}, jsone_encode:encode(Input, [native_utf8]))
  98. end},
  99. {"string: contains control characters",
  100. fun () ->
  101. Ctrls = lists:seq(16#00, 16#1F) -- [$\b, $\f, $\n, $\r, $\t],
  102. Input = list_to_binary(Ctrls),
  103. Expected = list_to_binary([$", [io_lib:format("\\u00~2.16.0b", [C]) || C <- Ctrls], $"]),
  104. ?assertEqual({ok, Expected}, jsone_encode:encode(Input)),
  105. ?assertEqual({ok, Expected}, jsone_encode:encode(Input, [native_utf8]))
  106. end},
  107. {"string: contains multi-byte (UTF-8 encoded) characters",
  108. fun () ->
  109. %% japanese
  110. Input1 = <<"あいうえお">>, % assumed that the encoding of this file is UTF-8
  111. Expected1 = <<"\"\\u3042\\u3044\\u3046\\u3048\\u304a\"">>,
  112. ?assertEqual({ok, Expected1}, jsone_encode:encode(Input1)),
  113. Expected12 = <<$", Input1/binary, $">>,
  114. ?assertEqual({ok, Expected12}, jsone_encode:encode(Input1, [native_utf8])),
  115. %% other multi-byte characters
  116. Input2 = <<"۝۞ႮႯ">>,
  117. Expected2 = <<"\"\\u06dd\\u06de\\u10ae\\u10af\"">>,
  118. ?assertEqual({ok, Expected2}, jsone_encode:encode(Input2)),
  119. Expected22 = <<$", Input2/binary, $">>,
  120. ?assertEqual({ok, Expected22}, jsone_encode:encode(Input2, [native_utf8]))
  121. end},
  122. {"string: containts surrogate pairs",
  123. fun () ->
  124. Input = <<"𢁉𢂚𢃼">>,
  125. Expected = <<"\"\\ud848\\udc49\\ud848\\udc9a\\ud848\\udcfc\"">>,
  126. ?assertEqual({ok, Expected}, jsone_encode:encode(Input))
  127. end},
  128. %% Strings variant: Datetimes
  129. {"datetime: iso8601: utc",
  130. fun () ->
  131. ?assertEqual({ok, <<"\"2015-06-25T14:57:25Z\"">>}, jsone_encode:encode({{2015,6,25},{14,57,25}})),
  132. ?assertEqual({ok, <<"\"2015-06-25T14:57:25Z\"">>}, jsone_encode:encode({{2015,6,25},{14,57,25}}, [{datetime_format, iso8601}])),
  133. ?assertEqual({ok, <<"\"2015-06-25T14:57:25Z\"">>}, jsone_encode:encode({{2015,6,25},{14,57,25}}, [{datetime_format, {iso8601, utc}}]))
  134. end},
  135. {"datetime: iso8601: local",
  136. fun () ->
  137. {ok, Json} = jsone_encode:encode({{2015,6,25},{14,57,25}}, [{datetime_format, {iso8601, local}}]),
  138. UTC = {{1970, 1, 2}, {0,0,0}},
  139. Local = calendar:universal_time_to_local_time({{1970, 1, 2}, {0,0,0}}),
  140. case UTC =:= Local of
  141. false -> ?assertMatch(<<"\"2015-06-25T14:57:25",_:6/binary,"\"">>, Json);
  142. true -> ?assertMatch(<<"\"2015-06-25T14:57:25Z\"">>, Json)
  143. end
  144. end},
  145. {"datetime: iso8601: timezone",
  146. fun () ->
  147. ?assertEqual({ok, <<"\"2015-06-25T14:57:25Z\"">>}, jsone_encode:encode({{2015,6,25},{14,57,25}}, [{datetime_format, {iso8601, 0}}])),
  148. ?assertEqual({ok, <<"\"2015-06-25T14:57:25+00:01\"">>}, jsone_encode:encode({{2015,6,25},{14,57,25}}, [{datetime_format, {iso8601, 60}}])),
  149. ?assertEqual({ok, <<"\"2015-06-25T14:57:25-00:01\"">>}, jsone_encode:encode({{2015,6,25},{14,57,25}}, [{datetime_format, {iso8601, -60}}]))
  150. end},
  151. {"datetime as head of array",
  152. ?_assertEqual({ok, <<"[\"2015-06-25T14:57:25Z\"]">>},
  153. jsone_encode:encode([{{2015,6,25},{14,57,25}}]))},
  154. {"datetime: iso8601: with fractions of seconds",
  155. fun () ->
  156. ?assertEqual({ok, <<"\"2015-06-25T14:57:25.325Z\"">>},
  157. jsone_encode:encode({{2015,6,25},{14,57,25.3245}})),
  158. ?assertEqual({ok, <<"\"2015-06-25T14:57:05.320Z\"">>},
  159. jsone_encode:encode({{2015,6,25},{14,57,5.32}}))
  160. end},
  161. %% Arrays
  162. {"simple array",
  163. fun () ->
  164. Input = [1, 2, 3],
  165. Expected = <<"[1,2,3]">>,
  166. ?assertEqual({ok, Expected}, jsone_encode:encode(Input))
  167. end},
  168. {"empty array",
  169. fun () ->
  170. Input = [],
  171. Expected = <<"[]">>,
  172. ?assertEqual({ok, Expected}, jsone_encode:encode(Input))
  173. end},
  174. %% Objects
  175. {"simple object",
  176. fun () ->
  177. Input1 = {[{<<"key">>, <<"value">>}, {<<"1">>, 2}]},
  178. Input2 = [{<<"key">>, <<"value">>}, {<<"1">>, 2}],
  179. Expected = <<"{\"key\":\"value\",\"1\":2}">>,
  180. ?assertEqual({ok, Expected}, jsone_encode:encode(Input1)),
  181. ?assertEqual({ok, Expected}, jsone_encode:encode(Input2))
  182. end},
  183. {"empty object",
  184. fun () ->
  185. Input1 = {[]},
  186. Input2 = [{}],
  187. Expected = <<"{}">>,
  188. ?assertEqual({ok, Expected}, jsone_encode:encode(Input1)),
  189. ?assertEqual({ok, Expected}, jsone_encode:encode(Input2))
  190. end},
  191. {"simple object: map",
  192. fun () ->
  193. Input = ?OBJ2(<<"1">>, 2, <<"key">>, <<"value">>),
  194. Expected = <<"{\"1\":2,\"key\":\"value\"}">>,
  195. ?assertEqual({ok, Expected}, jsone_encode:encode(Input))
  196. end},
  197. {"empty object: map",
  198. fun () ->
  199. Input = ?OBJ0,
  200. Expected = <<"{}">>,
  201. ?assertEqual({ok, Expected}, jsone_encode:encode(Input))
  202. end},
  203. {"atom key is allowed",
  204. fun () ->
  205. Expected = <<"{\"key\":2}">>,
  206. ?assertEqual({ok, Expected}, jsone_encode:encode({[{key, 2}]}))
  207. end},
  208. {"object_key_type option",
  209. fun () ->
  210. %% key: atom
  211. ?assertEqual({ok, <<"{\"a\":2}">>}, jsone_encode:encode(?OBJ1(a, 2), [{object_key_type, string}])), % OK
  212. ?assertEqual({ok, <<"{\"a\":2}">>}, jsone_encode:encode(?OBJ1(a, 2), [{object_key_type, scalar}])), % OK
  213. ?assertEqual({ok, <<"{\"a\":2}">>}, jsone_encode:encode(?OBJ1(a, 2), [{object_key_type, value}])), % OK
  214. %% key: number
  215. ?assertMatch({error, {badarg, _}}, jsone_encode:encode(?OBJ1(1, 2), [{object_key_type, string}])), % NG
  216. ?assertEqual({ok, <<"{\"1\":2}">>}, jsone_encode:encode(?OBJ1(1, 2), [{object_key_type, scalar}])), % OK
  217. ?assertEqual({ok, <<"{\"1\":2}">>}, jsone_encode:encode(?OBJ1(1, 2), [{object_key_type, value}])), % OK
  218. %% key: datetime
  219. ?assertMatch({error, {badarg, _}}, jsone_encode:encode(?OBJ1({{2000,1,1}, {0,0,0}}, 2), [{object_key_type, string}])), % NG
  220. ?assertEqual({ok, <<"{\"2000-01-01T00:00:00Z\":2}">>}, jsone_encode:encode(?OBJ1({{2000,1,1}, {0,0,0}}, 2), [{object_key_type, scalar}])), % OK
  221. ?assertEqual({ok, <<"{\"2000-01-01T00:00:00Z\":2}">>}, jsone_encode:encode(?OBJ1({{2000,1,1}, {0,0,0}}, 2), [{object_key_type, value}])), % OK
  222. %% key: array
  223. ?assertMatch({error, {badarg, _}}, jsone_encode:encode(?OBJ1([1], 2), [{object_key_type, string}])), % NG
  224. ?assertMatch({error, {badarg, _}}, jsone_encode:encode(?OBJ1([1], 2), [{object_key_type, scalar}])), % NG
  225. ?assertEqual({ok, <<"{\"[1]\":2}">>}, jsone_encode:encode(?OBJ1([1], 2), [{object_key_type, value}])), % OK
  226. %% key: object
  227. ?assertMatch({error, {badarg, _}}, jsone_encode:encode(?OBJ1(?OBJ0, 2), [{object_key_type, string}])), % NG
  228. ?assertMatch({error, {badarg, _}}, jsone_encode:encode(?OBJ1(?OBJ0, 2), [{object_key_type, scalar}])), % NG
  229. ?assertEqual({ok, <<"{\"{}\":2}">>}, jsone_encode:encode(?OBJ1(?OBJ0, 2), [{object_key_type, value}])) % OK
  230. end},
  231. {"non binary object member key is disallowed",
  232. fun () ->
  233. ?assertMatch({error, {badarg, _}}, jsone_encode:encode({[{1, 2}]})),
  234. ?assertMatch({error, {badarg, _}}, jsone_encode:encode({[{"1", 2}]}))
  235. end},
  236. {"undefined_as_null option",
  237. fun() ->
  238. ?assertEqual({ok,<<"null">>}, jsone_encode:encode(undefined,[undefined_as_null])), % OK
  239. ?assertEqual({ok,<<"\"undefined\"">>}, jsone_encode:encode(undefined,[])) % OK
  240. end},
  241. %% Pretty Print
  242. {"space",
  243. fun () ->
  244. ?assertEqual({ok, <<"[1, 2, 3]">>}, jsone_encode:encode([1,2,3], [{space, 1}])),
  245. ?assertEqual({ok, <<"[1, 2, 3]">>}, jsone_encode:encode([1,2,3], [{space, 2}])),
  246. ?assertEqual({ok, <<"{\"a\": 1, \"b\": 2}">>}, jsone_encode:encode(?OBJ2(a, 1, b, 2), [{space, 1}])),
  247. ?assertEqual({ok, <<"{\"a\": 1, \"b\": 2}">>}, jsone_encode:encode(?OBJ2(a, 1, b, 2), [{space, 2}]))
  248. end},
  249. {"indent",
  250. fun () ->
  251. ?assertEqual({ok, <<"[\n 1,\n 2,\n 3\n]">>}, jsone_encode:encode([1,2,3], [{indent, 1}])),
  252. ?assertEqual({ok, <<"[\n 1,\n 2,\n 3\n]">>}, jsone_encode:encode([1,2,3], [{indent, 2}])),
  253. ?assertEqual({ok, <<"{\n \"a\":1,\n \"b\":2\n}">>}, jsone_encode:encode(?OBJ2(a, 1, b, 2), [{indent, 1}])),
  254. ?assertEqual({ok, <<"{\n \"a\":1,\n \"b\":2\n}">>}, jsone_encode:encode(?OBJ2(a, 1, b, 2), [{indent, 2}]))
  255. end},
  256. {"indent+space",
  257. fun () ->
  258. ?assertEqual({ok, <<"[\n 1,\n 2,\n 3\n]">>}, jsone_encode:encode([1,2,3], [{indent, 1}, {space, 1}])),
  259. ?assertEqual({ok, <<"[\n 1,\n 2,\n 3\n]">>}, jsone_encode:encode([1,2,3], [{indent, 2}, {space, 2}])),
  260. ?assertEqual({ok, <<"{\n \"a\": 1,\n \"b\": 2\n}">>}, jsone_encode:encode(?OBJ2(a, 1, b, 2), [{indent, 1}, {space, 1}])),
  261. ?assertEqual({ok, <<"{\n \"a\": 1,\n \"b\": 2\n}">>}, jsone_encode:encode(?OBJ2(a, 1, b, 2), [{indent, 2}, {space, 2}]))
  262. end},
  263. %% Others
  264. {"compound data",
  265. fun () ->
  266. Input = [true, {[{<<"1">>, 2}, {<<"array">>, [[[[1]]], {[{<<"ab">>, <<"cd">>}]}, false]}]}, null],
  267. Expected = <<"[true,{\"1\":2,\"array\":[[[[1]]],{\"ab\":\"cd\"},false]},null]">>,
  268. ?assertEqual({ok, Expected}, jsone_encode:encode(Input)),
  269. PpExpected = <<"[\n true,\n {\n \"1\": 2,\n \"array\": [\n [\n [\n [\n 1\n ]\n ]\n ],\n {\n \"ab\": \"cd\"\n },\n false\n ]\n },\n null\n]">>,
  270. ?assertEqual({ok, PpExpected}, jsone_encode:encode(Input, [{indent, 1}, {space, 1}]))
  271. end},
  272. {"invalid value",
  273. fun () ->
  274. ?assertMatch({error, {badarg, _}}, jsone_encode:encode(self()))
  275. end},
  276. {"wrong option",
  277. fun () ->
  278. ?assertError(badarg, jsone_encode:encode(1, [{no_such_option, hoge}]))
  279. end},
  280. {"canonical_form",
  281. fun () ->
  282. Obj1 = maps:from_list( [{<<"key", (integer_to_binary(I))/binary >>, I} || I <- lists:seq(1000, 0, -1)] ),
  283. Obj2 = maps:from_list( [{<<"key", (integer_to_binary(I))/binary >>, I} || I <- lists:seq(0, 1000, 1)] ),
  284. ?assertEqual(jsone_encode:encode(Obj1, [canonical_form]), jsone_encode:encode(Obj2, [canonical_form]))
  285. end}
  286. ].