dblib_stmt.c 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 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: Wez Furlong <wez@php.net> |
  16. | Frank M. Kromann <frank@kromann.info> |
  17. +----------------------------------------------------------------------+
  18. */
  19. /* $Id$ */
  20. #ifdef HAVE_CONFIG_H
  21. # include "config.h"
  22. #endif
  23. #include "php.h"
  24. #include "php_ini.h"
  25. #include "ext/standard/php_string.h"
  26. #include "ext/standard/info.h"
  27. #include "pdo/php_pdo.h"
  28. #include "pdo/php_pdo_driver.h"
  29. #include "php_pdo_dblib.h"
  30. #include "php_pdo_dblib_int.h"
  31. #include "zend_exceptions.h"
  32. /* {{{ pdo_dblib_get_field_name
  33. *
  34. * Return the data type name for a given TDS number
  35. *
  36. */
  37. static char *pdo_dblib_get_field_name(int type)
  38. {
  39. /*
  40. * I don't return dbprtype(type) because it does not fully describe the type
  41. * (example: varchar is reported as char by dbprtype)
  42. *
  43. * FIX ME: Cache datatypes from server systypes table in pdo_dblib_handle_factory()
  44. * to make this future proof.
  45. */
  46. switch (type) {
  47. case 31: return "nvarchar";
  48. case 34: return "image";
  49. case 35: return "text";
  50. case 36: return "uniqueidentifier";
  51. case 37: return "varbinary"; /* & timestamp - Sybase AS12 */
  52. case 38: return "bigint"; /* & bigintn - Sybase AS12 */
  53. case 39: return "varchar"; /* & sysname & nvarchar - Sybase AS12 */
  54. case 40: return "date";
  55. case 41: return "time";
  56. case 42: return "datetime2";
  57. case 43: return "datetimeoffset";
  58. case 45: return "binary"; /* Sybase AS12 */
  59. case 47: return "char"; /* & nchar & uniqueidentifierstr Sybase AS12 */
  60. case 48: return "tinyint";
  61. case 50: return "bit"; /* Sybase AS12 */
  62. case 52: return "smallint";
  63. case 55: return "decimal"; /* Sybase AS12 */
  64. case 56: return "int";
  65. case 58: return "smalldatetime";
  66. case 59: return "real";
  67. case 60: return "money";
  68. case 61: return "datetime";
  69. case 62: return "float";
  70. case 63: return "numeric"; /* or uint, ubigint, usmallint Sybase AS12 */
  71. case 98: return "sql_variant";
  72. case 99: return "ntext";
  73. case 104: return "bit";
  74. case 106: return "decimal"; /* decimal n on sybase */
  75. case 108: return "numeric"; /* numeric n on sybase */
  76. case 122: return "smallmoney";
  77. case 127: return "bigint";
  78. case 165: return "varbinary";
  79. case 167: return "varchar";
  80. case 173: return "binary";
  81. case 175: return "char";
  82. case 189: return "timestamp";
  83. case 231: return "nvarchar";
  84. case 239: return "nchar";
  85. case 240: return "geometry";
  86. case 241: return "xml";
  87. default: return "unknown";
  88. }
  89. }
  90. /* }}} */
  91. static int pdo_dblib_stmt_cursor_closer(pdo_stmt_t *stmt TSRMLS_DC)
  92. {
  93. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  94. pdo_dblib_db_handle *H = S->H;
  95. /* Cancel any pending results */
  96. dbcancel(H->link);
  97. return 1;
  98. }
  99. static int pdo_dblib_stmt_dtor(pdo_stmt_t *stmt TSRMLS_DC)
  100. {
  101. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  102. efree(S);
  103. return 1;
  104. }
  105. static int pdo_dblib_stmt_next_rowset(pdo_stmt_t *stmt TSRMLS_DC)
  106. {
  107. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  108. pdo_dblib_db_handle *H = S->H;
  109. RETCODE ret;
  110. ret = dbresults(H->link);
  111. if (FAIL == ret) {
  112. pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "PDO_DBLIB: dbresults() returned FAIL" TSRMLS_CC);
  113. return 0;
  114. }
  115. if(NO_MORE_RESULTS == ret) {
  116. return 0;
  117. }
  118. stmt->row_count = DBCOUNT(H->link);
  119. stmt->column_count = dbnumcols(H->link);
  120. return 1;
  121. }
  122. static int pdo_dblib_stmt_execute(pdo_stmt_t *stmt TSRMLS_DC)
  123. {
  124. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  125. pdo_dblib_db_handle *H = S->H;
  126. RETCODE ret;
  127. dbsetuserdata(H->link, (BYTE*) &S->err);
  128. pdo_dblib_stmt_cursor_closer(stmt TSRMLS_CC);
  129. if (FAIL == dbcmd(H->link, stmt->active_query_string)) {
  130. return 0;
  131. }
  132. if (FAIL == dbsqlexec(H->link)) {
  133. return 0;
  134. }
  135. ret = pdo_dblib_stmt_next_rowset(stmt TSRMLS_CC);
  136. stmt->row_count = DBCOUNT(H->link);
  137. stmt->column_count = dbnumcols(H->link);
  138. return 1;
  139. }
  140. static int pdo_dblib_stmt_fetch(pdo_stmt_t *stmt,
  141. enum pdo_fetch_orientation ori, long offset TSRMLS_DC)
  142. {
  143. RETCODE ret;
  144. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  145. pdo_dblib_db_handle *H = S->H;
  146. ret = dbnextrow(H->link);
  147. if (FAIL == ret) {
  148. pdo_raise_impl_error(stmt->dbh, stmt, "HY000", "PDO_DBLIB: dbnextrow() returned FAIL" TSRMLS_CC);
  149. return 0;
  150. }
  151. if(NO_MORE_ROWS == ret) {
  152. return 0;
  153. }
  154. return 1;
  155. }
  156. static int pdo_dblib_stmt_describe(pdo_stmt_t *stmt, int colno TSRMLS_DC)
  157. {
  158. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  159. pdo_dblib_db_handle *H = S->H;
  160. struct pdo_column_data *col;
  161. char *fname;
  162. if(colno >= stmt->column_count || colno < 0) {
  163. return FAILURE;
  164. }
  165. col = &stmt->columns[colno];
  166. fname = (char*)dbcolname(H->link, colno+1);
  167. if (fname && *fname) {
  168. col->name = estrdup(fname);
  169. col->namelen = strlen(col->name);
  170. } else {
  171. col->namelen = spprintf(&col->name, 0, "computed%d", colno);
  172. }
  173. col->maxlen = dbcollen(H->link, colno+1);
  174. col->param_type = PDO_PARAM_STR;
  175. return 1;
  176. }
  177. static int pdo_dblib_stmt_get_col(pdo_stmt_t *stmt, int colno, char **ptr,
  178. unsigned long *len, int *caller_frees TSRMLS_DC)
  179. {
  180. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  181. pdo_dblib_db_handle *H = S->H;
  182. int coltype;
  183. unsigned int tmp_len;
  184. char *tmp_ptr = NULL;
  185. coltype = dbcoltype(H->link, colno+1);
  186. *len = dbdatlen(H->link, colno+1);
  187. *ptr = dbdata(H->link, colno+1);
  188. if (*len == 0 && *ptr == NULL) {
  189. return 1;
  190. }
  191. switch (coltype) {
  192. case SQLVARBINARY:
  193. case SQLBINARY:
  194. case SQLIMAGE:
  195. case SQLTEXT:
  196. /* FIXME: Above types should be returned as a stream as they can be VERY large */
  197. case SQLCHAR:
  198. case SQLVARCHAR:
  199. tmp_ptr = emalloc(*len + 1);
  200. memcpy(tmp_ptr, *ptr, *len);
  201. tmp_ptr[*len] = '\0';
  202. *ptr = tmp_ptr;
  203. break;
  204. case SQLMONEY:
  205. case SQLMONEY4:
  206. case SQLMONEYN: {
  207. DBFLT8 money_value;
  208. dbconvert(NULL, coltype, *ptr, *len, SQLFLT8, (LPBYTE)&money_value, 8);
  209. *len = spprintf(&tmp_ptr, 0, "%.4f", money_value);
  210. *ptr = tmp_ptr;
  211. break;
  212. }
  213. case SQLUNIQUE: {
  214. *len = 36+1;
  215. tmp_ptr = emalloc(*len + 1);
  216. /* uniqueidentifier is a 16-byte binary number, convert to 32 char hex string */
  217. *len = dbconvert(NULL, SQLUNIQUE, *ptr, *len, SQLCHAR, tmp_ptr, *len);
  218. php_strtoupper(tmp_ptr, *len);
  219. *ptr = tmp_ptr;
  220. break;
  221. }
  222. case SQLDATETIM4:
  223. case SQLDATETIME: {
  224. DBDATETIME dt;
  225. DBDATEREC di;
  226. dbconvert(H->link, coltype, (BYTE*) *ptr, -1, SQLDATETIME, (LPBYTE) &dt, -1);
  227. dbdatecrack(H->link, &di, &dt);
  228. *len = spprintf((char**) &tmp_ptr, 20, "%d-%02d-%02d %02d:%02d:%02d",
  229. #if defined(PHP_DBLIB_IS_MSSQL) || defined(MSDBLIB)
  230. di.year, di.month, di.day, di.hour, di.minute, di.second
  231. #else
  232. di.dateyear, di.datemonth+1, di.datedmonth, di.datehour, di.dateminute, di.datesecond
  233. #endif
  234. );
  235. *ptr = (char*) tmp_ptr;
  236. break;
  237. }
  238. default:
  239. if (dbwillconvert(coltype, SQLCHAR)) {
  240. tmp_len = 32 + (2 * (*len)); /* FIXME: We allocate more than we need here */
  241. tmp_ptr = emalloc(tmp_len);
  242. *len = dbconvert(NULL, coltype, *ptr, *len, SQLCHAR, tmp_ptr, -1);
  243. *ptr = tmp_ptr;
  244. } else {
  245. *len = 0; /* FIXME: Silently fails and returns null on conversion errors */
  246. *ptr = NULL;
  247. }
  248. }
  249. *caller_frees = 1;
  250. return 1;
  251. }
  252. static int pdo_dblib_stmt_param_hook(pdo_stmt_t *stmt, struct pdo_bound_param_data *param,
  253. enum pdo_param_event event_type TSRMLS_DC)
  254. {
  255. return 1;
  256. }
  257. static int pdo_dblib_stmt_get_column_meta(pdo_stmt_t *stmt, long colno, zval *return_value TSRMLS_DC)
  258. {
  259. pdo_dblib_stmt *S = (pdo_dblib_stmt*)stmt->driver_data;
  260. pdo_dblib_db_handle *H = S->H;
  261. DBTYPEINFO* dbtypeinfo;
  262. if(colno >= stmt->column_count || colno < 0) {
  263. return FAILURE;
  264. }
  265. array_init(return_value);
  266. dbtypeinfo = dbcoltypeinfo(H->link, colno+1);
  267. if(!dbtypeinfo) return FAILURE;
  268. add_assoc_long(return_value, "max_length", dbcollen(H->link, colno+1) );
  269. add_assoc_long(return_value, "precision", (int) dbtypeinfo->precision );
  270. add_assoc_long(return_value, "scale", (int) dbtypeinfo->scale );
  271. add_assoc_string(return_value, "column_source", dbcolsource(H->link, colno+1), 1);
  272. add_assoc_string(return_value, "native_type", pdo_dblib_get_field_name(dbcoltype(H->link, colno+1)), 1);
  273. add_assoc_long(return_value, "native_type_id", dbcoltype(H->link, colno+1));
  274. add_assoc_long(return_value, "native_usertype_id", dbcolutype(H->link, colno+1));
  275. return 1;
  276. }
  277. struct pdo_stmt_methods dblib_stmt_methods = {
  278. pdo_dblib_stmt_dtor,
  279. pdo_dblib_stmt_execute,
  280. pdo_dblib_stmt_fetch,
  281. pdo_dblib_stmt_describe,
  282. pdo_dblib_stmt_get_col,
  283. pdo_dblib_stmt_param_hook,
  284. NULL, /* set attr */
  285. NULL, /* get attr */
  286. pdo_dblib_stmt_get_column_meta, /* meta */
  287. pdo_dblib_stmt_next_rowset, /* nextrow */
  288. pdo_dblib_stmt_cursor_closer
  289. };