json.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Author: Omar Kilani <omar@php.net> |
  16. | Jakub Zelenka <bukka@php.net> |
  17. +----------------------------------------------------------------------+
  18. */
  19. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #include "php_ini.h"
  24. #include "ext/standard/info.h"
  25. #include "ext/standard/html.h"
  26. #include "zend_smart_str.h"
  27. #include "php_json.h"
  28. #include "php_json_encoder.h"
  29. #include "php_json_parser.h"
  30. #include <zend_exceptions.h>
  31. static PHP_MINFO_FUNCTION(json);
  32. static PHP_FUNCTION(json_encode);
  33. static PHP_FUNCTION(json_decode);
  34. static PHP_FUNCTION(json_last_error);
  35. static PHP_FUNCTION(json_last_error_msg);
  36. PHP_JSON_API zend_class_entry *php_json_serializable_ce;
  37. PHP_JSON_API zend_class_entry *php_json_exception_ce;
  38. PHP_JSON_API ZEND_DECLARE_MODULE_GLOBALS(json)
  39. /* {{{ arginfo */
  40. ZEND_BEGIN_ARG_INFO_EX(arginfo_json_encode, 0, 0, 1)
  41. ZEND_ARG_INFO(0, value)
  42. ZEND_ARG_INFO(0, options)
  43. ZEND_ARG_INFO(0, depth)
  44. ZEND_END_ARG_INFO()
  45. ZEND_BEGIN_ARG_INFO_EX(arginfo_json_decode, 0, 0, 1)
  46. ZEND_ARG_INFO(0, json)
  47. ZEND_ARG_INFO(0, assoc)
  48. ZEND_ARG_INFO(0, depth)
  49. ZEND_ARG_INFO(0, options)
  50. ZEND_END_ARG_INFO()
  51. ZEND_BEGIN_ARG_INFO(arginfo_json_last_error, 0)
  52. ZEND_END_ARG_INFO()
  53. ZEND_BEGIN_ARG_INFO(arginfo_json_last_error_msg, 0)
  54. ZEND_END_ARG_INFO()
  55. /* }}} */
  56. /* {{{ json_functions[] */
  57. static const zend_function_entry json_functions[] = {
  58. PHP_FE(json_encode, arginfo_json_encode)
  59. PHP_FE(json_decode, arginfo_json_decode)
  60. PHP_FE(json_last_error, arginfo_json_last_error)
  61. PHP_FE(json_last_error_msg, arginfo_json_last_error_msg)
  62. PHP_FE_END
  63. };
  64. /* }}} */
  65. /* {{{ JsonSerializable methods */
  66. ZEND_BEGIN_ARG_INFO(json_serialize_arginfo, 0)
  67. /* No arguments */
  68. ZEND_END_ARG_INFO();
  69. static const zend_function_entry json_serializable_interface[] = {
  70. PHP_ABSTRACT_ME(JsonSerializable, jsonSerialize, json_serialize_arginfo)
  71. PHP_FE_END
  72. };
  73. /* }}} */
  74. /* Register constant for options and errors */
  75. #define PHP_JSON_REGISTER_CONSTANT(_name, _value) \
  76. REGISTER_LONG_CONSTANT(_name, _value, CONST_CS | CONST_PERSISTENT);
  77. /* {{{ MINIT */
  78. static PHP_MINIT_FUNCTION(json)
  79. {
  80. zend_class_entry ce;
  81. INIT_CLASS_ENTRY(ce, "JsonSerializable", json_serializable_interface);
  82. php_json_serializable_ce = zend_register_internal_interface(&ce);
  83. INIT_CLASS_ENTRY(ce, "JsonException", NULL);
  84. php_json_exception_ce = zend_register_internal_class_ex(&ce, zend_ce_exception);
  85. /* options for json_encode */
  86. PHP_JSON_REGISTER_CONSTANT("JSON_HEX_TAG", PHP_JSON_HEX_TAG);
  87. PHP_JSON_REGISTER_CONSTANT("JSON_HEX_AMP", PHP_JSON_HEX_AMP);
  88. PHP_JSON_REGISTER_CONSTANT("JSON_HEX_APOS", PHP_JSON_HEX_APOS);
  89. PHP_JSON_REGISTER_CONSTANT("JSON_HEX_QUOT", PHP_JSON_HEX_QUOT);
  90. PHP_JSON_REGISTER_CONSTANT("JSON_FORCE_OBJECT", PHP_JSON_FORCE_OBJECT);
  91. PHP_JSON_REGISTER_CONSTANT("JSON_NUMERIC_CHECK", PHP_JSON_NUMERIC_CHECK);
  92. PHP_JSON_REGISTER_CONSTANT("JSON_UNESCAPED_SLASHES", PHP_JSON_UNESCAPED_SLASHES);
  93. PHP_JSON_REGISTER_CONSTANT("JSON_PRETTY_PRINT", PHP_JSON_PRETTY_PRINT);
  94. PHP_JSON_REGISTER_CONSTANT("JSON_UNESCAPED_UNICODE", PHP_JSON_UNESCAPED_UNICODE);
  95. PHP_JSON_REGISTER_CONSTANT("JSON_PARTIAL_OUTPUT_ON_ERROR", PHP_JSON_PARTIAL_OUTPUT_ON_ERROR);
  96. PHP_JSON_REGISTER_CONSTANT("JSON_PRESERVE_ZERO_FRACTION", PHP_JSON_PRESERVE_ZERO_FRACTION);
  97. PHP_JSON_REGISTER_CONSTANT("JSON_UNESCAPED_LINE_TERMINATORS", PHP_JSON_UNESCAPED_LINE_TERMINATORS);
  98. /* options for json_decode */
  99. PHP_JSON_REGISTER_CONSTANT("JSON_OBJECT_AS_ARRAY", PHP_JSON_OBJECT_AS_ARRAY);
  100. PHP_JSON_REGISTER_CONSTANT("JSON_BIGINT_AS_STRING", PHP_JSON_BIGINT_AS_STRING);
  101. /* common options for json_decode and json_encode */
  102. PHP_JSON_REGISTER_CONSTANT("JSON_INVALID_UTF8_IGNORE", PHP_JSON_INVALID_UTF8_IGNORE);
  103. PHP_JSON_REGISTER_CONSTANT("JSON_INVALID_UTF8_SUBSTITUTE", PHP_JSON_INVALID_UTF8_SUBSTITUTE);
  104. PHP_JSON_REGISTER_CONSTANT("JSON_THROW_ON_ERROR", PHP_JSON_THROW_ON_ERROR);
  105. /* json error constants */
  106. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_NONE", PHP_JSON_ERROR_NONE);
  107. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_DEPTH", PHP_JSON_ERROR_DEPTH);
  108. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_STATE_MISMATCH", PHP_JSON_ERROR_STATE_MISMATCH);
  109. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_CTRL_CHAR", PHP_JSON_ERROR_CTRL_CHAR);
  110. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_SYNTAX", PHP_JSON_ERROR_SYNTAX);
  111. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_UTF8", PHP_JSON_ERROR_UTF8);
  112. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_RECURSION", PHP_JSON_ERROR_RECURSION);
  113. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_INF_OR_NAN", PHP_JSON_ERROR_INF_OR_NAN);
  114. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_UNSUPPORTED_TYPE", PHP_JSON_ERROR_UNSUPPORTED_TYPE);
  115. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_INVALID_PROPERTY_NAME", PHP_JSON_ERROR_INVALID_PROPERTY_NAME);
  116. PHP_JSON_REGISTER_CONSTANT("JSON_ERROR_UTF16", PHP_JSON_ERROR_UTF16);
  117. return SUCCESS;
  118. }
  119. /* }}} */
  120. /* {{{ PHP_GINIT_FUNCTION
  121. */
  122. static PHP_GINIT_FUNCTION(json)
  123. {
  124. #if defined(COMPILE_DL_JSON) && defined(ZTS)
  125. ZEND_TSRMLS_CACHE_UPDATE();
  126. #endif
  127. json_globals->encoder_depth = 0;
  128. json_globals->error_code = 0;
  129. json_globals->encode_max_depth = PHP_JSON_PARSER_DEFAULT_DEPTH;
  130. }
  131. /* }}} */
  132. /* {{{ json_module_entry
  133. */
  134. zend_module_entry json_module_entry = {
  135. STANDARD_MODULE_HEADER,
  136. "json",
  137. json_functions,
  138. PHP_MINIT(json),
  139. NULL,
  140. NULL,
  141. NULL,
  142. PHP_MINFO(json),
  143. PHP_JSON_VERSION,
  144. PHP_MODULE_GLOBALS(json),
  145. PHP_GINIT(json),
  146. NULL,
  147. NULL,
  148. STANDARD_MODULE_PROPERTIES_EX
  149. };
  150. /* }}} */
  151. #ifdef COMPILE_DL_JSON
  152. #ifdef ZTS
  153. ZEND_TSRMLS_CACHE_DEFINE()
  154. #endif
  155. ZEND_GET_MODULE(json)
  156. #endif
  157. /* {{{ PHP_MINFO_FUNCTION
  158. */
  159. static PHP_MINFO_FUNCTION(json)
  160. {
  161. php_info_print_table_start();
  162. php_info_print_table_row(2, "json support", "enabled");
  163. php_info_print_table_row(2, "json version", PHP_JSON_VERSION);
  164. php_info_print_table_end();
  165. }
  166. /* }}} */
  167. PHP_JSON_API int php_json_encode_ex(smart_str *buf, zval *val, int options, zend_long depth) /* {{{ */
  168. {
  169. php_json_encoder encoder;
  170. int return_code;
  171. php_json_encode_init(&encoder);
  172. encoder.max_depth = depth;
  173. return_code = php_json_encode_zval(buf, val, options, &encoder);
  174. JSON_G(error_code) = encoder.error_code;
  175. return return_code;
  176. }
  177. /* }}} */
  178. PHP_JSON_API int php_json_encode(smart_str *buf, zval *val, int options) /* {{{ */
  179. {
  180. return php_json_encode_ex(buf, val, options, JSON_G(encode_max_depth));
  181. }
  182. /* }}} */
  183. static const char *php_json_get_error_msg(php_json_error_code error_code) /* {{{ */
  184. {
  185. switch(error_code) {
  186. case PHP_JSON_ERROR_NONE:
  187. return "No error";
  188. case PHP_JSON_ERROR_DEPTH:
  189. return "Maximum stack depth exceeded";
  190. case PHP_JSON_ERROR_STATE_MISMATCH:
  191. return "State mismatch (invalid or malformed JSON)";
  192. case PHP_JSON_ERROR_CTRL_CHAR:
  193. return "Control character error, possibly incorrectly encoded";
  194. case PHP_JSON_ERROR_SYNTAX:
  195. return "Syntax error";
  196. case PHP_JSON_ERROR_UTF8:
  197. return "Malformed UTF-8 characters, possibly incorrectly encoded";
  198. case PHP_JSON_ERROR_RECURSION:
  199. return "Recursion detected";
  200. case PHP_JSON_ERROR_INF_OR_NAN:
  201. return "Inf and NaN cannot be JSON encoded";
  202. case PHP_JSON_ERROR_UNSUPPORTED_TYPE:
  203. return "Type is not supported";
  204. case PHP_JSON_ERROR_INVALID_PROPERTY_NAME:
  205. return "The decoded property name is invalid";
  206. case PHP_JSON_ERROR_UTF16:
  207. return "Single unpaired UTF-16 surrogate in unicode escape";
  208. default:
  209. return "Unknown error";
  210. }
  211. }
  212. /* }}} */
  213. PHP_JSON_API int php_json_decode_ex(zval *return_value, char *str, size_t str_len, zend_long options, zend_long depth) /* {{{ */
  214. {
  215. php_json_parser parser;
  216. php_json_parser_init(&parser, return_value, str, str_len, (int)options, (int)depth);
  217. if (php_json_yyparse(&parser)) {
  218. php_json_error_code error_code = php_json_parser_error_code(&parser);
  219. if (!(options & PHP_JSON_THROW_ON_ERROR)) {
  220. JSON_G(error_code) = error_code;
  221. } else {
  222. zend_throw_exception(php_json_exception_ce, php_json_get_error_msg(error_code), error_code);
  223. }
  224. RETVAL_NULL();
  225. return FAILURE;
  226. }
  227. return SUCCESS;
  228. }
  229. /* }}} */
  230. /* {{{ proto string json_encode(mixed data [, int options[, int depth]])
  231. Returns the JSON representation of a value */
  232. static PHP_FUNCTION(json_encode)
  233. {
  234. zval *parameter;
  235. php_json_encoder encoder;
  236. smart_str buf = {0};
  237. zend_long options = 0;
  238. zend_long depth = PHP_JSON_PARSER_DEFAULT_DEPTH;
  239. ZEND_PARSE_PARAMETERS_START(1, 3)
  240. Z_PARAM_ZVAL(parameter)
  241. Z_PARAM_OPTIONAL
  242. Z_PARAM_LONG(options)
  243. Z_PARAM_LONG(depth)
  244. ZEND_PARSE_PARAMETERS_END();
  245. php_json_encode_init(&encoder);
  246. encoder.max_depth = (int)depth;
  247. php_json_encode_zval(&buf, parameter, (int)options, &encoder);
  248. if (!(options & PHP_JSON_THROW_ON_ERROR) || (options & PHP_JSON_PARTIAL_OUTPUT_ON_ERROR)) {
  249. JSON_G(error_code) = encoder.error_code;
  250. if (encoder.error_code != PHP_JSON_ERROR_NONE && !(options & PHP_JSON_PARTIAL_OUTPUT_ON_ERROR)) {
  251. smart_str_free(&buf);
  252. RETURN_FALSE;
  253. }
  254. } else {
  255. if (encoder.error_code != PHP_JSON_ERROR_NONE) {
  256. smart_str_free(&buf);
  257. zend_throw_exception(php_json_exception_ce, php_json_get_error_msg(encoder.error_code), encoder.error_code);
  258. RETURN_FALSE;
  259. }
  260. }
  261. smart_str_0(&buf); /* copy? */
  262. if (buf.s) {
  263. RETURN_NEW_STR(buf.s);
  264. }
  265. RETURN_EMPTY_STRING();
  266. }
  267. /* }}} */
  268. /* {{{ proto mixed json_decode(string json [, bool assoc [, int depth]])
  269. Decodes the JSON representation into a PHP value */
  270. static PHP_FUNCTION(json_decode)
  271. {
  272. char *str;
  273. size_t str_len;
  274. zend_bool assoc = 0; /* return JS objects as PHP objects by default */
  275. zend_bool assoc_null = 1;
  276. zend_long depth = PHP_JSON_PARSER_DEFAULT_DEPTH;
  277. zend_long options = 0;
  278. ZEND_PARSE_PARAMETERS_START(1, 4)
  279. Z_PARAM_STRING(str, str_len)
  280. Z_PARAM_OPTIONAL
  281. Z_PARAM_BOOL_EX(assoc, assoc_null, 1, 0)
  282. Z_PARAM_LONG(depth)
  283. Z_PARAM_LONG(options)
  284. ZEND_PARSE_PARAMETERS_END();
  285. if (!(options & PHP_JSON_THROW_ON_ERROR)) {
  286. JSON_G(error_code) = PHP_JSON_ERROR_NONE;
  287. }
  288. if (!str_len) {
  289. if (!(options & PHP_JSON_THROW_ON_ERROR)) {
  290. JSON_G(error_code) = PHP_JSON_ERROR_SYNTAX;
  291. } else {
  292. zend_throw_exception(php_json_exception_ce, php_json_get_error_msg(PHP_JSON_ERROR_SYNTAX), PHP_JSON_ERROR_SYNTAX);
  293. }
  294. RETURN_NULL();
  295. }
  296. if (depth <= 0) {
  297. php_error_docref(NULL, E_WARNING, "Depth must be greater than zero");
  298. RETURN_NULL();
  299. }
  300. if (depth > INT_MAX) {
  301. php_error_docref(NULL, E_WARNING, "Depth must be lower than %d", INT_MAX);
  302. RETURN_NULL();
  303. }
  304. /* For BC reasons, the bool $assoc overrides the long $options bit for PHP_JSON_OBJECT_AS_ARRAY */
  305. if (!assoc_null) {
  306. if (assoc) {
  307. options |= PHP_JSON_OBJECT_AS_ARRAY;
  308. } else {
  309. options &= ~PHP_JSON_OBJECT_AS_ARRAY;
  310. }
  311. }
  312. php_json_decode_ex(return_value, str, str_len, options, depth);
  313. }
  314. /* }}} */
  315. /* {{{ proto int json_last_error()
  316. Returns the error code of the last json_encode() or json_decode() call. */
  317. static PHP_FUNCTION(json_last_error)
  318. {
  319. if (zend_parse_parameters_none() == FAILURE) {
  320. return;
  321. }
  322. RETURN_LONG(JSON_G(error_code));
  323. }
  324. /* }}} */
  325. /* {{{ proto string json_last_error_msg()
  326. Returns the error string of the last json_encode() or json_decode() call. */
  327. static PHP_FUNCTION(json_last_error_msg)
  328. {
  329. if (zend_parse_parameters_none() == FAILURE) {
  330. return;
  331. }
  332. RETURN_STRING(php_json_get_error_msg(JSON_G(error_code)));
  333. }
  334. /* }}} */
  335. /*
  336. * Local variables:
  337. * tab-width: 4
  338. * c-basic-offset: 4
  339. * End:
  340. * vim600: noet sw=4 ts=4 fdm=marker
  341. * vim<600: noet sw=4 ts=4
  342. */