jsone_decode.erl 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. %%% @doc JSON decoding module
  2. %%% @private
  3. %%% @end
  4. %%%
  5. %%% Copyright (c) 2013-2016, Takeru Ohta <phjgt308@gmail.com>
  6. %%%
  7. %%% The MIT License
  8. %%%
  9. %%% Permission is hereby granted, free of charge, to any person obtaining a copy
  10. %%% of this software and associated documentation files (the "Software"), to deal
  11. %%% in the Software without restriction, including without limitation the rights
  12. %%% to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  13. %%% copies of the Software, and to permit persons to whom the Software is
  14. %%% furnished to do so, subject to the following conditions:
  15. %%%
  16. %%% The above copyright notice and this permission notice shall be included in
  17. %%% all copies or substantial portions of the Software.
  18. %%%
  19. %%% THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  20. %%% IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  21. %%% FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  22. %%% AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  23. %%% LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  24. %%% OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  25. %%% THE SOFTWARE.
  26. %%%
  27. %%%---------------------------------------------------------------------------------------
  28. -module(jsone_decode).
  29. -ifdef(ENABLE_HIPE).
  30. -compile([native, {hipe, [o3]}]).
  31. -endif.
  32. %%--------------------------------------------------------------------------------
  33. %% Exported API
  34. %%--------------------------------------------------------------------------------
  35. -export([decode/1, decode/2]).
  36. %%--------------------------------------------------------------------------------
  37. %% Macros & Records & Types
  38. %%--------------------------------------------------------------------------------
  39. -define(ERROR(Function, Args), {error, {badarg, [{?MODULE, Function, Args, [{line, ?LINE}]}]}}).
  40. -ifdef('NO_MAP_TYPE').
  41. -define(DEFAULT_OBJECT_FORMAT, tuple).
  42. -define(LIST_TO_MAP(X), error({this_erts_does_not_support_maps, X})).
  43. -else.
  44. -define(DEFAULT_OBJECT_FORMAT, map).
  45. -define(LIST_TO_MAP(X), maps:from_list(X)).
  46. -endif.
  47. -type next() :: {array_next, [jsone:json_value()]}
  48. | {object_value, jsone:json_object_members()}
  49. | {object_next, jsone:json_string(), jsone:json_object_members()}.
  50. -type whitespace_next() :: value
  51. | array
  52. | object
  53. | {array_next, [jsone:json_value()]}
  54. | {object_key, jsone:json_object_members()}
  55. | {object_value, jsone:json_string(), jsone:json_object_members()}
  56. | {object_next, jsone:json_object_members()}.
  57. -type decode_result() :: {ok, jsone:json_value(), Rest::binary()} | {error, {Reason::term(), [jsone:stack_item()]}}.
  58. -record(decode_opt_v2,
  59. {
  60. object_format=?DEFAULT_OBJECT_FORMAT :: tuple | proplist | map,
  61. allow_ctrl_chars=false :: boolean(),
  62. reject_invalid_utf8=false :: boolean(),
  63. keys=binary :: 'binary' | 'atom' | 'existing_atom' | 'attempt_atom',
  64. undefined_as_null=false :: boolean(),
  65. duplicate_map_keys=first :: first | last
  66. }).
  67. -define(OPT, #decode_opt_v2).
  68. -type opt() :: #decode_opt_v2{}.
  69. %%--------------------------------------------------------------------------------
  70. %% Exported Functions
  71. %%--------------------------------------------------------------------------------
  72. -spec decode(binary()) -> decode_result().
  73. decode(Json) ->
  74. decode(Json, []).
  75. -spec decode(binary(), [jsone:decode_option()]) -> decode_result().
  76. decode(<<Json/binary>>, Options) ->
  77. Opt = parse_options(Options),
  78. whitespace(Json, value, [], <<"">>, Opt).
  79. %%--------------------------------------------------------------------------------
  80. %% Internal Functions
  81. %%--------------------------------------------------------------------------------
  82. -spec next(binary(), jsone:json_value(), [next()], binary(), opt()) -> decode_result().
  83. next(<<Bin/binary>>, Value, [], _Buf, _Opt) ->
  84. {ok, Value, Bin};
  85. next(<<Bin/binary>>, Value, [Next | Nexts], Buf, Opt) ->
  86. case Next of
  87. {array_next, Values} -> whitespace(Bin, {array_next, [Value | Values]}, Nexts, Buf, Opt);
  88. {object_value, Members} -> whitespace(Bin, {object_value, Value, Members}, Nexts, Buf, Opt);
  89. {object_next, Key, Members} -> whitespace(Bin, {object_next, [{Key, Value} | Members]}, Nexts, Buf, Opt)
  90. end.
  91. -spec whitespace(binary(), whitespace_next(), [next()], binary(), opt()) -> decode_result().
  92. whitespace(<<$ , Bin/binary>>, Next, Nexts, Buf, Opt) -> whitespace(Bin, Next, Nexts, Buf, Opt);
  93. whitespace(<<$\t, Bin/binary>>, Next, Nexts, Buf, Opt) -> whitespace(Bin, Next, Nexts, Buf, Opt);
  94. whitespace(<<$\r, Bin/binary>>, Next, Nexts, Buf, Opt) -> whitespace(Bin, Next, Nexts, Buf, Opt);
  95. whitespace(<<$\n, Bin/binary>>, Next, Nexts, Buf, Opt) -> whitespace(Bin, Next, Nexts, Buf, Opt);
  96. whitespace(<<Bin/binary>>, Next, Nexts, Buf, Opt) ->
  97. case Next of
  98. value -> value(Bin, Nexts, Buf, Opt);
  99. array -> array(Bin, Nexts, Buf, Opt);
  100. object -> object(Bin, Nexts, Buf, Opt);
  101. {object_key, Members} -> object_key(Bin, Members, Nexts, Buf, Opt);
  102. {array_next, Values} -> array_next(Bin, Values, Nexts, Buf, Opt);
  103. {object_value, Key, Members} -> object_value(Bin, Key, Members, Nexts, Buf, Opt);
  104. {object_next, Members} -> object_next(Bin, Members, Nexts, Buf, Opt)
  105. end.
  106. -spec value(binary(), [next()], binary(), opt()) -> decode_result().
  107. value(<<"false", Bin/binary>>, Nexts, Buf, Opt) -> next(Bin, false, Nexts, Buf, Opt);
  108. value(<<"true", Bin/binary>>, Nexts, Buf, Opt) -> next(Bin, true, Nexts, Buf, Opt);
  109. value(<<"null", Bin/binary>>, Nexts, Buf,
  110. Opt = ?OPT{undefined_as_null = true}) -> next(Bin, undefined, Nexts, Buf, Opt);
  111. value(<<"null", Bin/binary>>, Nexts, Buf, Opt) -> next(Bin, null, Nexts, Buf, Opt);
  112. value(<<$[, Bin/binary>>, Nexts, Buf, Opt) -> whitespace(Bin, array, Nexts, Buf, Opt);
  113. value(<<${, Bin/binary>>, Nexts, Buf, Opt) -> whitespace(Bin, object, Nexts, Buf, Opt);
  114. value(<<$", Bin/binary>>, Nexts, Buf, Opt) -> string(Bin, byte_size(Buf), Nexts, Buf, Opt);
  115. value(<<Bin/binary>>, Nexts, Buf, Opt) -> number(Bin, Nexts, Buf, Opt).
  116. -spec array(binary(), [next()], binary(), opt()) -> decode_result().
  117. array(<<$], Bin/binary>>, Nexts, Buf, Opt) -> next(Bin, [], Nexts, Buf, Opt);
  118. array(<<Bin/binary>>, Nexts, Buf, Opt) -> value(Bin, [{array_next, []} | Nexts], Buf, Opt).
  119. -spec array_next(binary(), [jsone:json_value()], [next()], binary(), opt()) -> decode_result().
  120. array_next(<<$], Bin/binary>>, Values, Nexts, Buf, Opt) -> next(Bin, lists:reverse(Values), Nexts, Buf, Opt);
  121. array_next(<<$,, Bin/binary>>, Values, Nexts, Buf, Opt) -> whitespace(Bin, value, [{array_next, Values} | Nexts], Buf, Opt);
  122. array_next(Bin, Values, Nexts, Buf, Opt) -> ?ERROR(array_next, [Bin, Values, Nexts, Buf, Opt]).
  123. -spec object(binary(), [next()], binary(), opt()) -> decode_result().
  124. object(<<$}, Bin/binary>>, Nexts, Buf, Opt) -> next(Bin, make_object([], Opt), Nexts, Buf, Opt);
  125. object(<<Bin/binary>>, Nexts, Buf, Opt) -> object_key(Bin, [], Nexts, Buf, Opt).
  126. -spec object_key(binary(), jsone:json_object_members(), [next()], binary(), opt()) -> decode_result().
  127. object_key(<<$", Bin/binary>>, Members, Nexts, Buf, Opt) -> string(Bin, byte_size(Buf), [{object_value, Members} | Nexts], Buf, Opt);
  128. object_key(<<Bin/binary>>, Members, Nexts, Buf, Opt) -> ?ERROR(object_key, [Bin, Members, Nexts, Buf, Opt]).
  129. -spec object_value(binary(), jsone:json_string(), jsone:json_object_members(), [next()], binary(), opt()) -> decode_result().
  130. object_value(<<$:, Bin/binary>>, Key, Members, Nexts, Buf, Opt) ->
  131. whitespace(Bin, value, [{object_next, object_key(Key, Opt), Members} | Nexts], Buf, Opt);
  132. object_value(Bin, Key, Members, Nexts, Buf, Opt) -> ?ERROR(object_value, [Bin, Key, Members, Nexts, Buf, Opt]).
  133. -compile({inline, [object_key/2]}).
  134. object_key(Key, ?OPT{keys = binary}) -> Key;
  135. object_key(Key, ?OPT{keys = atom}) -> binary_to_atom(Key, utf8);
  136. object_key(Key, ?OPT{keys = existing_atom}) -> binary_to_existing_atom(Key, utf8);
  137. object_key(Key, ?OPT{keys = attempt_atom}) ->
  138. try binary_to_existing_atom(Key, utf8)
  139. catch error:badarg -> Key
  140. end.
  141. -spec object_next(binary(), jsone:json_object_members(), [next()], binary(), opt()) -> decode_result().
  142. object_next(<<$}, Bin/binary>>, Members, Nexts, Buf, Opt) -> next(Bin, make_object(Members, Opt), Nexts, Buf, Opt);
  143. object_next(<<$,, Bin/binary>>, Members, Nexts, Buf, Opt) -> whitespace(Bin, {object_key, Members}, Nexts, Buf, Opt);
  144. object_next(Bin, Members, Nexts, Buf, Opt) -> ?ERROR(object_next, [Bin, Members, Nexts, Buf, Opt]).
  145. -spec string(binary(), non_neg_integer(), [next()], binary(), opt()) -> decode_result().
  146. string(<<Bin/binary>>, Start, Nexts, Buf, Opt) ->
  147. string(Bin, Bin, Start, Nexts, Buf, Opt).
  148. -spec string(binary(), binary(), non_neg_integer(), [next()], binary(), opt()) -> decode_result().
  149. string(<<$", Bin/binary>>, Base, Start, Nexts, Buf, Opt) ->
  150. Prefix = binary:part(Base, 0, byte_size(Base) - byte_size(Bin) - 1),
  151. case Start =:= byte_size(Buf) of
  152. true -> next(Bin, Prefix, Nexts, Buf, Opt);
  153. false ->
  154. Buf2 = <<Buf/binary, Prefix/binary>>,
  155. next(Bin, binary:part(Buf2, Start, byte_size(Buf2) - Start), Nexts, Buf2, Opt)
  156. end;
  157. string(<<$\\, B/binary>>, Base, Start, Nexts, Buf, Opt) ->
  158. Prefix = binary:part(Base, 0, byte_size(Base) - byte_size(B) - 1),
  159. case B of
  160. <<$", Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $">>, Opt);
  161. <<$/, Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $/>>, Opt);
  162. <<$\\,Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $\\>>, Opt);
  163. <<$b, Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $\b>>, Opt);
  164. <<$f, Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $\f>>, Opt);
  165. <<$n, Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $\n>>, Opt);
  166. <<$r, Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $\r>>, Opt);
  167. <<$t, Bin/binary>> -> string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary, $\t>>, Opt);
  168. <<$u, Bin/binary>> -> unicode_string(Bin, Start, Nexts, <<Buf/binary, Prefix/binary>>, Opt);
  169. _ -> ?ERROR(string, [<<$\\, B/binary>>, Base, Start, Nexts, Buf, Opt])
  170. end;
  171. string(<<_, Bin/binary>>, Base, Start, Nexts, Buf, Opt) when Opt?OPT.allow_ctrl_chars, not Opt?OPT.reject_invalid_utf8 ->
  172. string(Bin, Base, Start, Nexts, Buf, Opt);
  173. string(<<C, Bin/binary>>, Base, Start, Nexts, Buf, Opt) when 16#20 =< C, not Opt?OPT.reject_invalid_utf8 ->
  174. string(Bin, Base, Start, Nexts, Buf, Opt);
  175. string(<<_/utf8, Bin/binary>>, Base, Start, Nexts, Buf, Opt) when Opt?OPT.allow_ctrl_chars ->
  176. string(Bin, Base, Start, Nexts, Buf, Opt);
  177. string(<<C/utf8, Bin/binary>>, Base, Start, Nexts, Buf, Opt) when 16#20 =< C ->
  178. string(Bin, Base, Start, Nexts, Buf, Opt);
  179. string(Bin, Base, Start, Nexts, Buf, Opt) ->
  180. ?ERROR(string, [Bin, Base, Start, Nexts, Buf, Opt]).
  181. -spec unicode_string(binary(), non_neg_integer(), [next()], binary(), opt()) -> decode_result().
  182. unicode_string(<<N:4/binary, Bin/binary>>, Start, Nexts, Buf, Opt) ->
  183. try binary_to_integer(N, 16) of
  184. High when 16#D800 =< High, High =< 16#DBFF ->
  185. %% surrogate pair
  186. case Bin of
  187. <<$\\, $u, N2:4/binary, Bin2/binary>> ->
  188. try binary_to_integer(N2, 16) of
  189. Low when 16#DC00 =< Low, Low =< 16#DFFF ->
  190. <<Unicode/utf16>> = <<High:16, Low:16>>,
  191. string(Bin2, Start, Nexts, <<Buf/binary, Unicode/utf8>>, Opt);
  192. _ -> ?ERROR(unicode_string, [<<N/binary, Bin/binary>>, Start, Nexts, Buf, Opt])
  193. catch error:badarg -> ?ERROR(unicode_string, [<<N/binary, Bin/binary>>, Start, Nexts, Buf, Opt])
  194. end;
  195. _ -> ?ERROR(unicode_string, [<<N/binary, Bin/binary>>, Start, Nexts, Buf, Opt])
  196. end;
  197. Unicode when 16#DC00 =< Unicode, Unicode =< 16#DFFF; % second part of surrogate pair (without first part)
  198. 0 > Unicode ->
  199. ?ERROR(unicode_string, [<<N/binary, Bin/binary>>, Start, Nexts, Buf, Opt]);
  200. Unicode ->
  201. string(Bin, Start, Nexts, <<Buf/binary, Unicode/utf8>>, Opt)
  202. catch error:badarg -> ?ERROR(unicode_string, [<<N/binary, Bin/binary>>, Start, Nexts, Buf, Opt])
  203. end;
  204. unicode_string(Bin, Start, Nexts, Buf, Opt) ->
  205. ?ERROR(unicode_string, [Bin, Start, Nexts, Buf, Opt]).
  206. -spec number(binary(), [next()], binary(), opt()) -> decode_result().
  207. number(<<$-, Bin/binary>>, Nexts, Buf, Opt) -> number_integer_part(Bin, -1, Nexts, Buf, Opt);
  208. number(<<Bin/binary>>, Nexts, Buf, Opt) -> number_integer_part(Bin, 1, Nexts, Buf, Opt).
  209. -spec number_integer_part(binary(), 1|-1, [next()], binary(), opt()) -> decode_result().
  210. number_integer_part(<<$0, Bin/binary>>, Sign, Nexts, Buf, Opt) ->
  211. number_fraction_part(Bin, Sign, 0, Nexts, Buf, Opt);
  212. number_integer_part(<<C, Bin/binary>>, Sign, Nexts, Buf, Opt) when $1 =< C, C =< $9 ->
  213. number_integer_part_rest(Bin, C - $0, Sign, Nexts, Buf, Opt);
  214. number_integer_part(Bin, Sign, Nexts, Buf, Opt) ->
  215. ?ERROR(number_integer_part, [Bin, Sign, Nexts, Buf, Opt]).
  216. -spec number_integer_part_rest(binary(), non_neg_integer(), 1|-1, [next()], binary(), opt()) -> decode_result().
  217. number_integer_part_rest(<<C, Bin/binary>>, N, Sign, Nexts, Buf, Opt) when $0 =< C, C =< $9 ->
  218. number_integer_part_rest(Bin, N * 10 + C - $0, Sign, Nexts, Buf, Opt);
  219. number_integer_part_rest(<<Bin/binary>>, N, Sign, Nexts, Buf, Opt) ->
  220. number_fraction_part(Bin, Sign, N, Nexts, Buf, Opt).
  221. -spec number_fraction_part(binary(), 1|-1, non_neg_integer(), [next()], binary(), opt()) -> decode_result().
  222. number_fraction_part(<<$., Bin/binary>>, Sign, Int, Nexts, Buf, Opt) ->
  223. number_fraction_part_rest(Bin, Sign, Int, 0, Nexts, Buf, Opt);
  224. number_fraction_part(<<Bin/binary>>, Sign, Int, Nexts, Buf, Opt) ->
  225. number_exponation_part(Bin, Sign * Int, 0, Nexts, Buf, Opt).
  226. -spec number_fraction_part_rest(binary(), 1|-1, non_neg_integer(), non_neg_integer(), [next()], binary(), opt()) -> decode_result().
  227. number_fraction_part_rest(<<C, Bin/binary>>, Sign, N, DecimalOffset, Nexts, Buf, Opt) when $0 =< C, C =< $9 ->
  228. number_fraction_part_rest(Bin, Sign, N * 10 + C - $0, DecimalOffset + 1, Nexts, Buf, Opt);
  229. number_fraction_part_rest(<<Bin/binary>>, Sign, N, DecimalOffset, Nexts, Buf, Opt) when DecimalOffset > 0 ->
  230. number_exponation_part(Bin, Sign * N, DecimalOffset, Nexts, Buf, Opt);
  231. number_fraction_part_rest(Bin, Sign, N, DecimalOffset, Nexts, Buf, Opt) ->
  232. ?ERROR(number_fraction_part_rest, [Bin, Sign, N, DecimalOffset, Nexts, Buf, Opt]).
  233. -spec number_exponation_part(binary(), integer(), non_neg_integer(), [next()], binary(), opt()) -> decode_result().
  234. number_exponation_part(<<$e, $+, Bin/binary>>, N, DecimalOffset, Nexts, Buf, Opt) ->
  235. number_exponation_part(Bin, N, DecimalOffset, 1, 0, true, Nexts, Buf, Opt);
  236. number_exponation_part(<<$E, $+, Bin/binary>>, N, DecimalOffset, Nexts, Buf, Opt) ->
  237. number_exponation_part(Bin, N, DecimalOffset, 1, 0, true, Nexts, Buf, Opt);
  238. number_exponation_part(<<$e, $-, Bin/binary>>, N, DecimalOffset, Nexts, Buf, Opt) ->
  239. number_exponation_part(Bin, N, DecimalOffset, -1, 0, true, Nexts, Buf, Opt);
  240. number_exponation_part(<<$E, $-, Bin/binary>>, N, DecimalOffset, Nexts, Buf, Opt) ->
  241. number_exponation_part(Bin, N, DecimalOffset, -1, 0, true, Nexts, Buf, Opt);
  242. number_exponation_part(<<$e, Bin/binary>>, N, DecimalOffset, Nexts, Buf, Opt) ->
  243. number_exponation_part(Bin, N, DecimalOffset, 1, 0, true, Nexts, Buf, Opt);
  244. number_exponation_part(<<$E, Bin/binary>>, N, DecimalOffset, Nexts, Buf, Opt) ->
  245. number_exponation_part(Bin, N, DecimalOffset, 1, 0, true, Nexts, Buf, Opt);
  246. number_exponation_part(<<Bin/binary>>, N, DecimalOffset, Nexts, Buf, Opt) ->
  247. case DecimalOffset of
  248. 0 -> next(Bin, N, Nexts, Buf, Opt);
  249. _ -> next(Bin, N / math:pow(10, DecimalOffset), Nexts, Buf, Opt)
  250. end.
  251. -spec number_exponation_part(binary(), integer(), non_neg_integer(), 1|-1, non_neg_integer(), boolean(), [next()], binary(), opt()) -> decode_result().
  252. number_exponation_part(<<C, Bin/binary>>, N, DecimalOffset, ExpSign, Exp, _, Nexts, Buf, Opt) when $0 =< C, C =< $9 ->
  253. number_exponation_part(Bin, N, DecimalOffset, ExpSign, Exp * 10 + C - $0, false, Nexts, Buf, Opt);
  254. number_exponation_part(<<Bin/binary>>, N, DecimalOffset, ExpSign, Exp, false, Nexts, Buf, Opt) ->
  255. Pos = ExpSign * Exp - DecimalOffset,
  256. try N * math:pow(10, Pos)
  257. of Res -> next(Bin, Res, Nexts, Buf, Opt)
  258. catch error:badarith ->
  259. ?ERROR(number_exponation_part, [Bin, N, DecimalOffset, ExpSign, Exp, false, Nexts, Buf, Opt])
  260. end;
  261. number_exponation_part(Bin, N, DecimalOffset, ExpSign, Exp, IsFirst, Nexts, Buf, Opt) ->
  262. ?ERROR(number_exponation_part, [Bin, N, DecimalOffset, ExpSign, Exp, IsFirst, Nexts, Buf, Opt]).
  263. -spec make_object(jsone:json_object_members(), opt()) -> jsone:json_object().
  264. make_object(Members, ?OPT{object_format = tuple}) -> {lists:reverse(Members)};
  265. make_object(Members, ?OPT{object_format = map, duplicate_map_keys = last}) ->
  266. ?LIST_TO_MAP(lists:reverse(Members));
  267. make_object(Members, ?OPT{object_format = map}) -> ?LIST_TO_MAP(Members);
  268. make_object([], _) -> [{}];
  269. make_object(Members, _) -> lists:reverse(Members).
  270. -spec parse_options([jsone:decode_option()]) -> opt().
  271. parse_options(Options) ->
  272. parse_option(Options, ?OPT{}).
  273. -spec parse_option([jsone:decode_option()], opt()) -> opt().
  274. parse_option([], Opt) -> Opt;
  275. parse_option([{object_format,F}|T], Opt) when F =:= tuple; F =:= proplist; F =:= map ->
  276. parse_option(T, Opt?OPT{object_format=F});
  277. parse_option([{allow_ctrl_chars,B}|T], Opt) when is_boolean(B) ->
  278. parse_option(T, Opt?OPT{allow_ctrl_chars=B});
  279. parse_option([reject_invalid_utf8|T], Opt) ->
  280. parse_option(T, Opt?OPT{reject_invalid_utf8=true});
  281. parse_option([{keys, K}|T], Opt)
  282. when K =:= binary; K =:= atom; K =:= existing_atom; K =:= attempt_atom ->
  283. parse_option(T, Opt?OPT{keys = K});
  284. parse_option([undefined_as_null|T], Opt) ->
  285. parse_option(T, Opt?OPT{undefined_as_null = true});
  286. parse_option([{duplicate_map_keys, V} | T], Opt)
  287. when V =:= first; V =:= last ->
  288. parse_option(T, Opt?OPT{duplicate_map_keys=V});
  289. parse_option(List, Opt) ->
  290. error(badarg, [List, Opt]).