mysqli_prop.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449
  1. /*
  2. +----------------------------------------------------------------------+
  3. | Copyright (c) The PHP Group |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | https://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Author: Georg Richter <georg@php.net> |
  14. | Andrey Hristov <andrey@php.net> |
  15. +----------------------------------------------------------------------+
  16. */
  17. #ifdef HAVE_CONFIG_H
  18. #include "config.h"
  19. #endif
  20. #include <signal.h>
  21. #include "php.h"
  22. #include "php_ini.h"
  23. #include "ext/standard/info.h"
  24. #include "php_mysqli_structs.h"
  25. #include "mysqli_priv.h"
  26. #define CHECK_STATUS(value, quiet) \
  27. if (!obj->ptr || ((MYSQLI_RESOURCE *)obj->ptr)->status < value ) { \
  28. if (!quiet) { \
  29. zend_throw_error(NULL, "Property access is not allowed yet"); \
  30. } \
  31. return FAILURE; \
  32. } \
  33. #define MYSQLI_GET_MYSQL(statusval) \
  34. MYSQL *p; \
  35. if (!obj->ptr || !(MY_MYSQL *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr) { \
  36. if (!quiet) { \
  37. zend_throw_error(NULL, "%s object is already closed", ZSTR_VAL(obj->zo.ce->name)); \
  38. } \
  39. return FAILURE; \
  40. } else { \
  41. CHECK_STATUS(statusval, quiet);\
  42. p = (MYSQL *)((MY_MYSQL *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr)->mysql;\
  43. }
  44. #define MYSQLI_GET_RESULT(statusval) \
  45. MYSQL_RES *p; \
  46. if (!obj->ptr) { \
  47. if (!quiet) { \
  48. zend_throw_error(NULL, "%s object is already closed", ZSTR_VAL(obj->zo.ce->name)); \
  49. } \
  50. return FAILURE; \
  51. } else { \
  52. CHECK_STATUS(statusval, quiet);\
  53. p = (MYSQL_RES *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr; \
  54. }
  55. #define MYSQLI_GET_STMT(statusval) \
  56. MYSQL_STMT *p; \
  57. if (!obj->ptr) { \
  58. if (!quiet) { \
  59. zend_throw_error(NULL, "%s object is already closed", ZSTR_VAL(obj->zo.ce->name)); \
  60. } \
  61. return FAILURE; \
  62. } else { \
  63. CHECK_STATUS(statusval, quiet); \
  64. p = (MYSQL_STMT *)((MY_STMT *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr)->stmt; \
  65. }
  66. #define MYSQLI_MAP_PROPERTY_FUNC_LONG_OR_STR( __func, __int_func, __get_type, __ret_type, __ret_type_sprint_mod)\
  67. static int __func(mysqli_object *obj, zval *retval, bool quiet) \
  68. {\
  69. __ret_type l;\
  70. __get_type;\
  71. ZEND_ASSERT(p);\
  72. l = (__ret_type)__int_func(p);\
  73. if (l < ZEND_LONG_MAX) {\
  74. ZVAL_LONG(retval, (zend_long) l);\
  75. } else { \
  76. ZVAL_NEW_STR(retval, strpprintf(0, __ret_type_sprint_mod, l)); \
  77. } \
  78. return SUCCESS; \
  79. }
  80. #define MYSQLI_MAP_PROPERTY_FUNC_LONG( __func, __int_func, __get_type, __ret_type, __ret_type_sprint_mod)\
  81. static int __func(mysqli_object *obj, zval *retval, bool quiet) \
  82. {\
  83. __ret_type l;\
  84. __get_type;\
  85. ZEND_ASSERT(p);\
  86. l = (__ret_type)__int_func(p);\
  87. ZEND_ASSERT(l < ZEND_LONG_MAX);\
  88. ZVAL_LONG(retval, (zend_long) l);\
  89. return SUCCESS; \
  90. }
  91. #define MYSQLI_MAP_PROPERTY_FUNC_STR_OR_NULL(__func, __int_func, __get_type)\
  92. static int __func(mysqli_object *obj, zval *retval, bool quiet)\
  93. {\
  94. char *c;\
  95. __get_type;\
  96. ZEND_ASSERT(p);\
  97. c = (char *)__int_func(p);\
  98. if (c) {\
  99. ZVAL_STRING(retval, c);\
  100. } else {\
  101. ZVAL_NULL(retval);\
  102. }\
  103. return SUCCESS; \
  104. }
  105. #define MYSQLI_MAP_PROPERTY_FUNC_STR(__func, __int_func, __get_type)\
  106. static int __func(mysqli_object *obj, zval *retval, bool quiet)\
  107. {\
  108. char *c;\
  109. __get_type;\
  110. ZEND_ASSERT(p);\
  111. c = (char *)__int_func(p);\
  112. ZEND_ASSERT(c);\
  113. ZVAL_STRING(retval, c);\
  114. return SUCCESS; \
  115. }
  116. /* {{{ property link_client_version_read */
  117. static int link_client_version_read(mysqli_object *obj, zval *retval, bool quiet)
  118. {
  119. ZVAL_LONG(retval, MYSQL_VERSION_ID);
  120. return SUCCESS;
  121. }
  122. /* }}} */
  123. /* {{{ property link_client_info_read */
  124. static int link_client_info_read(mysqli_object *obj, zval *retval, bool quiet)
  125. {
  126. ZVAL_STRING(retval, MYSQL_SERVER_VERSION);
  127. return SUCCESS;
  128. }
  129. /* }}} */
  130. /* {{{ property link_connect_errno_read */
  131. static int link_connect_errno_read(mysqli_object *obj, zval *retval, bool quiet)
  132. {
  133. ZVAL_LONG(retval, (zend_long)MyG(error_no));
  134. return SUCCESS;
  135. }
  136. /* }}} */
  137. /* {{{ property link_connect_error_read */
  138. static int link_connect_error_read(mysqli_object *obj, zval *retval, bool quiet)
  139. {
  140. if (MyG(error_msg)) {
  141. ZVAL_STRING(retval, MyG(error_msg));
  142. } else {
  143. ZVAL_NULL(retval);
  144. }
  145. return SUCCESS;
  146. }
  147. /* }}} */
  148. /* {{{ property link_affected_rows_read */
  149. static int link_affected_rows_read(mysqli_object *obj, zval *retval, bool quiet)
  150. {
  151. MY_MYSQL *mysql;
  152. my_ulonglong rc;
  153. CHECK_STATUS(MYSQLI_STATUS_VALID, quiet);
  154. mysql = (MY_MYSQL *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  155. ZEND_ASSERT(mysql);
  156. rc = mysql_affected_rows(mysql->mysql);
  157. if (rc == (my_ulonglong) -1) {
  158. ZVAL_LONG(retval, -1);
  159. return SUCCESS;
  160. }
  161. if (rc < ZEND_LONG_MAX) {
  162. ZVAL_LONG(retval, (zend_long) rc);
  163. } else {
  164. ZVAL_NEW_STR(retval, strpprintf(0, MYSQLI_LLU_SPEC, rc));
  165. }
  166. return SUCCESS;
  167. }
  168. /* }}} */
  169. /* {{{ property link_error_list_read */
  170. static int link_error_list_read(mysqli_object *obj, zval *retval, bool quiet)
  171. {
  172. MY_MYSQL *mysql;
  173. CHECK_STATUS(MYSQLI_STATUS_VALID, quiet);
  174. mysql = (MY_MYSQL *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  175. if (mysql) {
  176. array_init(retval);
  177. #ifdef MYSQLI_USE_MYSQLND
  178. if (1) {
  179. MYSQLND_ERROR_LIST_ELEMENT * message;
  180. zend_llist_position pos;
  181. for (message = (MYSQLND_ERROR_LIST_ELEMENT *) zend_llist_get_first_ex(&mysql->mysql->data->error_info->error_list, &pos);
  182. message;
  183. message = (MYSQLND_ERROR_LIST_ELEMENT *) zend_llist_get_next_ex(&mysql->mysql->data->error_info->error_list, &pos))
  184. {
  185. zval single_error;
  186. array_init(&single_error);
  187. add_assoc_long_ex(&single_error, "errno", sizeof("errno") - 1, message->error_no);
  188. add_assoc_string_ex(&single_error, "sqlstate", sizeof("sqlstate") - 1, message->sqlstate);
  189. add_assoc_string_ex(&single_error, "error", sizeof("error") - 1, message->error);
  190. add_next_index_zval(retval, &single_error);
  191. }
  192. }
  193. #else
  194. if (mysql_errno(mysql->mysql)) {
  195. zval single_error;
  196. array_init(&single_error);
  197. add_assoc_long_ex(&single_error, "errno", sizeof("errno") - 1, mysql_errno(mysql->mysql));
  198. add_assoc_string_ex(&single_error, "sqlstate", sizeof("sqlstate") - 1, mysql_sqlstate(mysql->mysql));
  199. add_assoc_string_ex(&single_error, "error", sizeof("error") - 1, mysql_error(mysql->mysql));
  200. add_next_index_zval(retval, &single_error);
  201. }
  202. #endif
  203. } else {
  204. ZVAL_EMPTY_ARRAY(retval);
  205. }
  206. return SUCCESS;
  207. }
  208. /* }}} */
  209. /* link properties */
  210. MYSQLI_MAP_PROPERTY_FUNC_LONG(link_errno_read, mysql_errno, MYSQLI_GET_MYSQL(MYSQLI_STATUS_INITIALIZED), zend_ulong, ZEND_ULONG_FMT)
  211. MYSQLI_MAP_PROPERTY_FUNC_STR(link_error_read, mysql_error, MYSQLI_GET_MYSQL(MYSQLI_STATUS_INITIALIZED))
  212. MYSQLI_MAP_PROPERTY_FUNC_LONG(link_field_count_read, mysql_field_count, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  213. MYSQLI_MAP_PROPERTY_FUNC_STR(link_host_info_read, mysql_get_host_info, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID))
  214. MYSQLI_MAP_PROPERTY_FUNC_STR_OR_NULL(link_info_read, mysql_info, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID))
  215. MYSQLI_MAP_PROPERTY_FUNC_LONG_OR_STR(link_insert_id_read, mysql_insert_id, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID), my_ulonglong, MYSQLI_LLU_SPEC)
  216. MYSQLI_MAP_PROPERTY_FUNC_LONG(link_protocol_version_read, mysql_get_proto_info, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  217. MYSQLI_MAP_PROPERTY_FUNC_STR(link_server_info_read, mysql_get_server_info, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID))
  218. MYSQLI_MAP_PROPERTY_FUNC_LONG(link_server_version_read, mysql_get_server_version, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  219. MYSQLI_MAP_PROPERTY_FUNC_STR(link_sqlstate_read, mysql_sqlstate, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID))
  220. MYSQLI_MAP_PROPERTY_FUNC_LONG(link_thread_id_read, mysql_thread_id, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  221. MYSQLI_MAP_PROPERTY_FUNC_LONG(link_warning_count_read, mysql_warning_count, MYSQLI_GET_MYSQL(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  222. /* result properties */
  223. /* {{{ property result_type_read */
  224. static int result_type_read(mysqli_object *obj, zval *retval, bool quiet)
  225. {
  226. MYSQL_RES *p;
  227. CHECK_STATUS(MYSQLI_STATUS_VALID, quiet);
  228. p = (MYSQL_RES *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  229. ZEND_ASSERT(p);
  230. ZVAL_LONG(retval, mysqli_result_is_unbuffered(p) ? MYSQLI_USE_RESULT:MYSQLI_STORE_RESULT);
  231. return SUCCESS;
  232. }
  233. /* }}} */
  234. /* {{{ property result_lengths_read */
  235. static int result_lengths_read(mysqli_object *obj, zval *retval, bool quiet)
  236. {
  237. MYSQL_RES *p;
  238. #ifdef MYSQLI_USE_MYSQLND
  239. const size_t *ret;
  240. #else
  241. const unsigned long *ret;
  242. #endif
  243. uint32_t field_count;
  244. CHECK_STATUS(MYSQLI_STATUS_VALID, quiet);
  245. p = (MYSQL_RES *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  246. field_count = mysql_num_fields(p);
  247. if (!p || !field_count || !(ret = mysql_fetch_lengths(p))) {
  248. ZVAL_NULL(retval);
  249. } else {
  250. zend_ulong i;
  251. array_init(retval);
  252. for (i = 0; i < field_count; i++) {
  253. add_index_long(retval, i, ret[i]);
  254. }
  255. }
  256. return SUCCESS;
  257. }
  258. /* }}} */
  259. MYSQLI_MAP_PROPERTY_FUNC_LONG(result_current_field_read, mysql_field_tell, MYSQLI_GET_RESULT(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  260. MYSQLI_MAP_PROPERTY_FUNC_LONG(result_field_count_read, mysql_num_fields, MYSQLI_GET_RESULT(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  261. MYSQLI_MAP_PROPERTY_FUNC_LONG_OR_STR(result_num_rows_read, mysql_num_rows, MYSQLI_GET_RESULT(MYSQLI_STATUS_VALID), my_ulonglong, MYSQLI_LLU_SPEC)
  262. /* statement properties */
  263. /* {{{ property stmt_id_read */
  264. static int stmt_id_read(mysqli_object *obj, zval *retval, bool quiet)
  265. {
  266. MY_STMT *p;
  267. CHECK_STATUS(MYSQLI_STATUS_VALID, quiet);
  268. p = (MY_STMT*)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  269. ZEND_ASSERT(p);
  270. ZVAL_LONG(retval, mysqli_stmt_get_id(p->stmt));
  271. return SUCCESS;
  272. }
  273. /* }}} */
  274. /* {{{ property stmt_affected_rows_read */
  275. static int stmt_affected_rows_read(mysqli_object *obj, zval *retval, bool quiet)
  276. {
  277. MY_STMT *p;
  278. my_ulonglong rc;
  279. CHECK_STATUS(MYSQLI_STATUS_VALID, quiet);
  280. p = (MY_STMT *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  281. ZEND_ASSERT(p);
  282. rc = mysql_stmt_affected_rows(p->stmt);
  283. if (rc == (my_ulonglong) -1) {
  284. ZVAL_LONG(retval, -1);
  285. return SUCCESS;
  286. }
  287. if (rc < ZEND_LONG_MAX) {
  288. ZVAL_LONG(retval, (zend_long) rc);
  289. } else {
  290. ZVAL_NEW_STR(retval, strpprintf(0, MYSQLI_LLU_SPEC, rc));
  291. }
  292. return SUCCESS;
  293. }
  294. /* }}} */
  295. /* {{{ property stmt_error_list_read */
  296. static int stmt_error_list_read(mysqli_object *obj, zval *retval, bool quiet)
  297. {
  298. MY_STMT * stmt;
  299. CHECK_STATUS(MYSQLI_STATUS_INITIALIZED, quiet);
  300. stmt = (MY_STMT *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  301. if (stmt && stmt->stmt) {
  302. array_init(retval);
  303. #ifdef MYSQLI_USE_MYSQLND
  304. if (stmt->stmt->data && stmt->stmt->data->error_info) {
  305. MYSQLND_ERROR_LIST_ELEMENT * message;
  306. zend_llist_position pos;
  307. for (message = (MYSQLND_ERROR_LIST_ELEMENT *) zend_llist_get_first_ex(&stmt->stmt->data->error_info->error_list, &pos);
  308. message;
  309. message = (MYSQLND_ERROR_LIST_ELEMENT *) zend_llist_get_next_ex(&stmt->stmt->data->error_info->error_list, &pos))
  310. {
  311. zval single_error;
  312. array_init(&single_error);
  313. add_assoc_long_ex(&single_error, "errno", sizeof("errno") - 1, message->error_no);
  314. add_assoc_string_ex(&single_error, "sqlstate", sizeof("sqlstate") - 1, message->sqlstate);
  315. add_assoc_string_ex(&single_error, "error", sizeof("error") - 1, message->error);
  316. add_next_index_zval(retval, &single_error);
  317. }
  318. }
  319. #else
  320. if (mysql_stmt_errno(stmt->stmt)) {
  321. zval single_error;
  322. array_init(&single_error);
  323. add_assoc_long_ex(&single_error, "errno", sizeof("errno") - 1, mysql_stmt_errno(stmt->stmt));
  324. add_assoc_string_ex(&single_error, "sqlstate", sizeof("sqlstate") - 1, mysql_stmt_sqlstate(stmt->stmt));
  325. add_assoc_string_ex(&single_error, "error", sizeof("error") - 1, mysql_stmt_error(stmt->stmt));
  326. add_next_index_zval(retval, &single_error);
  327. }
  328. #endif
  329. } else {
  330. ZVAL_EMPTY_ARRAY(retval);
  331. }
  332. return SUCCESS;
  333. }
  334. /* }}} */
  335. MYSQLI_MAP_PROPERTY_FUNC_LONG_OR_STR(stmt_insert_id_read, mysql_stmt_insert_id, MYSQLI_GET_STMT(MYSQLI_STATUS_VALID), my_ulonglong, MYSQLI_LLU_SPEC)
  336. MYSQLI_MAP_PROPERTY_FUNC_LONG_OR_STR(stmt_num_rows_read, mysql_stmt_num_rows, MYSQLI_GET_STMT(MYSQLI_STATUS_VALID), my_ulonglong, MYSQLI_LLU_SPEC)
  337. MYSQLI_MAP_PROPERTY_FUNC_LONG(stmt_param_count_read, mysql_stmt_param_count, MYSQLI_GET_STMT(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  338. MYSQLI_MAP_PROPERTY_FUNC_LONG(stmt_field_count_read, mysql_stmt_field_count, MYSQLI_GET_STMT(MYSQLI_STATUS_VALID), zend_ulong, ZEND_ULONG_FMT)
  339. MYSQLI_MAP_PROPERTY_FUNC_LONG(stmt_errno_read, mysql_stmt_errno, MYSQLI_GET_STMT(MYSQLI_STATUS_INITIALIZED), zend_ulong, ZEND_ULONG_FMT)
  340. MYSQLI_MAP_PROPERTY_FUNC_STR(stmt_error_read, mysql_stmt_error, MYSQLI_GET_STMT(MYSQLI_STATUS_INITIALIZED))
  341. MYSQLI_MAP_PROPERTY_FUNC_STR(stmt_sqlstate_read, mysql_stmt_sqlstate, MYSQLI_GET_STMT(MYSQLI_STATUS_INITIALIZED))
  342. /* }}} */
  343. const mysqli_property_entry mysqli_link_property_entries[] = {
  344. {"affected_rows", sizeof("affected_rows") - 1, link_affected_rows_read, NULL},
  345. {"client_info", sizeof("client_info") - 1, link_client_info_read, NULL},
  346. {"client_version", sizeof("client_version") - 1, link_client_version_read, NULL},
  347. {"connect_errno", sizeof("connect_errno") - 1, link_connect_errno_read, NULL},
  348. {"connect_error", sizeof("connect_error") - 1, link_connect_error_read, NULL},
  349. {"errno", sizeof("errno") - 1, link_errno_read, NULL},
  350. {"error", sizeof("error") - 1, link_error_read, NULL},
  351. {"error_list", sizeof("error_list") - 1, link_error_list_read, NULL},
  352. {"field_count", sizeof("field_count") - 1, link_field_count_read, NULL},
  353. {"host_info", sizeof("host_info") - 1, link_host_info_read, NULL},
  354. {"info", sizeof("info") - 1, link_info_read, NULL},
  355. {"insert_id", sizeof("insert_id") - 1, link_insert_id_read, NULL},
  356. {"server_info", sizeof("server_info") - 1, link_server_info_read, NULL},
  357. {"server_version", sizeof("server_version") - 1, link_server_version_read, NULL},
  358. {"sqlstate", sizeof("sqlstate") - 1, link_sqlstate_read, NULL},
  359. {"protocol_version",sizeof("protocol_version") - 1, link_protocol_version_read, NULL},
  360. {"thread_id", sizeof("thread_id") - 1, link_thread_id_read, NULL},
  361. {"warning_count", sizeof("warning_count") - 1, link_warning_count_read, NULL},
  362. {NULL, 0, NULL, NULL}
  363. };
  364. const mysqli_property_entry mysqli_result_property_entries[] = {
  365. {"current_field",sizeof("current_field")-1, result_current_field_read, NULL},
  366. {"field_count", sizeof("field_count") - 1, result_field_count_read, NULL},
  367. {"lengths", sizeof("lengths") - 1, result_lengths_read, NULL},
  368. {"num_rows", sizeof("num_rows") - 1, result_num_rows_read, NULL},
  369. {"type", sizeof("type") - 1, result_type_read, NULL},
  370. {NULL, 0, NULL, NULL}
  371. };
  372. const mysqli_property_entry mysqli_stmt_property_entries[] = {
  373. {"affected_rows", sizeof("affected_rows")-1,stmt_affected_rows_read, NULL},
  374. {"insert_id", sizeof("insert_id") - 1, stmt_insert_id_read, NULL},
  375. {"num_rows", sizeof("num_rows") - 1, stmt_num_rows_read, NULL},
  376. {"param_count", sizeof("param_count") - 1, stmt_param_count_read, NULL},
  377. {"field_count", sizeof("field_count") - 1, stmt_field_count_read, NULL},
  378. {"errno", sizeof("errno") - 1, stmt_errno_read, NULL},
  379. {"error", sizeof("error") - 1, stmt_error_read, NULL},
  380. {"error_list", sizeof("error_list") - 1, stmt_error_list_read, NULL},
  381. {"sqlstate", sizeof("sqlstate") - 1, stmt_sqlstate_read, NULL},
  382. {"id", sizeof("id") - 1, stmt_id_read, NULL},
  383. {NULL, 0, NULL, NULL}
  384. };