jsone_decode_tests.erl 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. %% Copyright (c) 2013, Takeru Ohta <phjgt308@gmail.com>
  2. -module(jsone_decode_tests).
  3. -include_lib("eunit/include/eunit.hrl").
  4. decode_test_() ->
  5. [
  6. %% シンボル系
  7. {"'false'がデコード可能",
  8. fun () ->
  9. ?assertEqual({false, <<"">>}, jsone_decode:decode(<<"false">>))
  10. end},
  11. {"'true'がデコード可能",
  12. fun () ->
  13. ?assertEqual({true, <<"">>}, jsone_decode:decode(<<"true">>))
  14. end},
  15. {"'null'がデコード可能",
  16. fun () ->
  17. ?assertEqual({null, <<"">>}, jsone_decode:decode(<<"null">>))
  18. end},
  19. {"正の整数がデコード可能",
  20. fun () ->
  21. ?assertEqual({1, <<"">>}, jsone_decode:decode(<<"1">>))
  22. end},
  23. %% 数値系: 整数
  24. {"0がデコード可能",
  25. fun () ->
  26. ?assertEqual({0, <<"">>}, jsone_decode:decode(<<"0">>))
  27. end},
  28. {"負の整数がデコード可能",
  29. fun () ->
  30. ?assertEqual({-1, <<"">>}, jsone_decode:decode(<<"-1">>))
  31. end},
  32. {"整数の値の大きさに制限はなし",
  33. fun () ->
  34. ?assertEqual({111111111111111111111111111111111111111111111111111111111111111111111111111111, <<"">>},
  35. jsone_decode:decode(<<"111111111111111111111111111111111111111111111111111111111111111111111111111111">>))
  36. end},
  37. {"先頭に余計な0がつく場合は、先頭文字とそれ以降が別々のトークンと判断される",
  38. fun () ->
  39. ?assertEqual({0, <<"0">>}, jsone_decode:decode(<<"00">>)),
  40. ?assertEqual({0, <<"1">>}, jsone_decode:decode(<<"01">>)),
  41. ?assertEqual({0, <<"1">>}, jsone_decode:decode(<<"-01">>))
  42. end},
  43. {"正の整数の前の'+'記号は許可されない",
  44. fun () ->
  45. ?assertError(badarg, jsone_decode:decode(<<"+1">>))
  46. end},
  47. %% 数値系: 小数
  48. {"小数がデコード可能",
  49. fun () ->
  50. ?assertEqual({1.23, <<"">>}, jsone_decode:decode(<<"1.23">>))
  51. end},
  52. {"指数形式の小数がデコード可能",
  53. fun () ->
  54. ?assertEqual({12.345, <<"">>}, jsone_decode:decode(<<"12345e-3">>)),
  55. ?assertEqual({12.345, <<"">>}, jsone_decode:decode(<<"12345E-3">>)), % 'e'は大文字でも可
  56. ?assertEqual({12.345, <<"">>}, jsone_decode:decode(<<"12345.0e-3">>)),
  57. ?assertEqual({12.345, <<"">>}, jsone_decode:decode(<<"0.12345E2">>)),
  58. ?assertEqual({12.345, <<"">>}, jsone_decode:decode(<<"0.12345e+2">>)), % 指数部では'+'をつけても良い
  59. ?assertEqual({12.345, <<"">>}, jsone_decode:decode(<<"0.12345E+2">>)), % 指数部では'+'をつけても良い
  60. ?assertEqual({-12.345, <<"">>}, jsone_decode:decode(<<"-0.012345e3">>))
  61. end},
  62. {"不正な形式の小数",
  63. fun () ->
  64. ?assertError(badarg, jsone_decode:decode(<<".123">>)), % 整数部が省略されている
  65. ?assertError(badarg, jsone_decode:decode(<<"0.">>)), % '.'の後ろに小数部が続かない
  66. ?assertError(badarg, jsone_decode:decode(<<"0.e+3">>)), % '.'の後ろに指数部が来る
  67. ?assertError(badarg, jsone_decode:decode(<<"0.1e">>)), % 指数部が欠けている
  68. ?assertError(badarg, jsone_decode:decode(<<"0.1e-">>)), % 指数部が欠けている
  69. ?assertError(badarg, jsone_decode:decode(<<"0.1ee-1">>)), % 'e'が複数ある
  70. ?assertError(badarg, jsone_decode:decode(<<"0.1e--1">>)), % 符号が複数ある
  71. ?assertEqual({0.1, <<".2">>}, jsone_decode:decode(<<"0.1.2">>)) % '.'が複数ある => 別々のトークンと判断される
  72. end},
  73. %% 文字列系
  74. {"文字列がデコード可能",
  75. fun () ->
  76. ?assertEqual({<<"abc">>, <<"">>}, jsone_decode:decode(<<"\"abc\"">>))
  77. end},
  78. {"各種エスケープ文字がデコード可能",
  79. fun () ->
  80. Input = list_to_binary([$", [[$\\, C] || C <- [$", $/, $\\, $b, $f, $n, $r, $t]], $"]),
  81. Expected = <<"\"\/\\\b\f\n\r\t">>,
  82. ?assertEqual({Expected, <<"">>}, jsone_decode:decode(Input))
  83. end},
  84. {"エスケープされたUTF-16文字列がデコード可能",
  85. fun () ->
  86. %% 日本語
  87. Input1 = <<"\"\\u3042\\u3044\\u3046\\u3048\\u304A\"">>,
  88. Expected1 = <<"あいうえお">>, % このファイルの文字エンコーディングがUTF-8であることが前提
  89. ?assertEqual({Expected1, <<"">>}, jsone_decode:decode(Input1)),
  90. %% ascii
  91. Input2 = <<"\"\\u0061\\u0062\\u0063\"">>,
  92. Expected2 = <<"abc">>,
  93. ?assertEqual({Expected2, <<"">>}, jsone_decode:decode(Input2)),
  94. %% 日本語以外のマルチバイト文字
  95. Input3 = <<"\"\\u06DD\\u06DE\\u10AE\\u10AF\"">>,
  96. Expected3 = <<"۝۞ႮႯ">>,
  97. ?assertEqual({Expected3, <<"">>}, jsone_decode:decode(Input3))
  98. end},
  99. {"サロゲートペアを含む文字列がデコード可能",
  100. fun () ->
  101. Input = <<"\"\\ud848\\udc49\\ud848\\udc9a\\ud848\\udcfc\"">>,
  102. Expected = <<"𢁉𢂚𢃼">>,
  103. ?assertEqual({Expected, <<"">>}, jsone_decode:decode(Input))
  104. end},
  105. {"不正なエスケープ文字",
  106. fun () ->
  107. ?assertError(badarg, jsone_decode:decode(<<"\"\\z\"">>)), % '\z'は未定義のエスケープ文字
  108. ?assertError(badarg, jsone_decode:decode(<<"\"\\uab\"">>)), % '\u'の後ろに続く数値が足りない
  109. ?assertError(badarg, jsone_decode:decode(<<"\"\\ud848\"">>)), % 上位サロゲートが単独で出現
  110. ?assertError(badarg, jsone_decode:decode(<<"\"\\udc49\"">>)), % 下位サロゲーが単独で出現
  111. ?assertError(badarg, jsone_decode:decode(<<"\"\\ud848\\u0061\"">>)) % 上位サロゲートの後ろに下位サロゲートが続かない
  112. end},
  113. %% 配列系
  114. {"配列がデコード可能",
  115. fun () ->
  116. Input = <<"[1, 2, \"abc\", null]">>,
  117. Expected = [1, 2, <<"abc">>, null],
  118. ?assertEqual({Expected, <<"">>}, jsone_decode:decode(Input))
  119. end},
  120. {"空配列がデコード可能",
  121. fun () ->
  122. ?assertEqual({[], <<"">>}, jsone_decode:decode(<<"[]">>)),
  123. ?assertEqual({[], <<"">>}, jsone_decode:decode(<<"[ \t\r\n]">>))
  124. end},
  125. {"配列の末尾のカンマは許容されない",
  126. fun () ->
  127. Input = <<"[1, 2, \"abc\", null, ]">>,
  128. ?assertError(badarg, jsone_decode:decode(Input))
  129. end},
  130. {"区切り文字のカンマが抜けているとエラーとなる",
  131. fun () ->
  132. Input = <<"[1 2, \"abc\", null]">>, % 1と2の間にカンマがない
  133. ?assertError(badarg, jsone_decode:decode(Input))
  134. end},
  135. {"配列が閉じていないとエラー",
  136. fun () ->
  137. Input = <<"[1, 2, \"abc\", null">>,
  138. ?assertError(badarg, jsone_decode:decode(Input))
  139. end},
  140. %% オブジェクト系
  141. {"オブジェクトがデコード可能",
  142. fun () ->
  143. Input = <<"{\"1\":2, \"key\":\"value\"}">>,
  144. Expected = {[{<<"key">>, <<"value">>}, {<<"1">>, 2}]},
  145. ?assertEqual({Expected, <<"">>}, jsone_decode:decode(Input))
  146. end},
  147. {"空オブジェクトがデコード可能",
  148. fun () ->
  149. ?assertEqual({{[]}, <<"">>}, jsone_decode:decode(<<"{}">>)),
  150. ?assertEqual({{[]}, <<"">>}, jsone_decode:decode(<<"{ \t\r\n}">>))
  151. end},
  152. {"オブジェクトの末尾のカンマは許容されない",
  153. fun () ->
  154. Input = <<"{\"1\":2, \"key\":\"value\", }">>,
  155. io:format("~p\n", [catch jsone_decode:decode(Input)]),
  156. ?assertError(badarg, jsone_decode:decode(Input))
  157. end},
  158. {"区切り文字のカンマが抜けているとエラーとなる",
  159. fun () ->
  160. Input = <<"{\"1\":2 \"key\":\"value\"}">>,
  161. ?assertError(badarg, jsone_decode:decode(Input))
  162. end},
  163. {"メンバのキーがない場合はエラー",
  164. fun () ->
  165. Input = <<"{:2, \"key\":\"value\"}">>,
  166. ?assertError(badarg, jsone_decode:decode(Input))
  167. end},
  168. {"メンバのキーが文字列以外の場合はエラー",
  169. fun () ->
  170. Input = <<"{1:2, \"key\":\"value\"}">>,
  171. ?assertError(badarg, jsone_decode:decode(Input))
  172. end},
  173. {"メンバの値がない場合はエラー",
  174. fun () ->
  175. Input = <<"{\"1\", \"key\":\"value\"}">>,
  176. ?assertError(badarg, jsone_decode:decode(Input))
  177. end},
  178. {"オブジェクトが閉じていないとエラー",
  179. fun () ->
  180. Input = <<"{\"1\":2 \"key\":\"value\"">>,
  181. ?assertError(badarg, jsone_decode:decode(Input))
  182. end},
  183. %% その他
  184. {"複雑なデータがデコード可能",
  185. fun () ->
  186. Input = <<" [true, {\"1\" : 2, \"array\":[[[[1]]], {\"ab\":\"cd\"}, false]}, null] ">>,
  187. Expected = [true, {[{<<"array">>, [[[[1]]], {[{<<"ab">>, <<"cd">>}]}, false]}, {<<"1">>, 2}]}, null],
  188. ?assertEqual({Expected, <<" ">>}, jsone_decode:decode(Input))
  189. end}
  190. ].