dblib_driver.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428
  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/info.h"
  26. #include "pdo/php_pdo.h"
  27. #include "pdo/php_pdo_driver.h"
  28. #include "php_pdo_dblib.h"
  29. #include "php_pdo_dblib_int.h"
  30. #include "zend_exceptions.h"
  31. /* Cache of the server supported datatypes, initialized in handle_factory */
  32. zval* pdo_dblib_datatypes;
  33. static int dblib_fetch_error(pdo_dbh_t *dbh, pdo_stmt_t *stmt, zval *info TSRMLS_DC)
  34. {
  35. pdo_dblib_db_handle *H = (pdo_dblib_db_handle *)dbh->driver_data;
  36. pdo_dblib_err *einfo = &H->err;
  37. pdo_dblib_stmt *S = NULL;
  38. char *message;
  39. char *msg;
  40. if (stmt) {
  41. S = (pdo_dblib_stmt*)stmt->driver_data;
  42. einfo = &S->err;
  43. }
  44. if (einfo->dberr == SYBESMSG && einfo->lastmsg) {
  45. msg = einfo->lastmsg;
  46. } else if (einfo->dberr == SYBESMSG && DBLIB_G(err).lastmsg) {
  47. msg = DBLIB_G(err).lastmsg;
  48. DBLIB_G(err).lastmsg = NULL;
  49. } else {
  50. msg = einfo->dberrstr;
  51. }
  52. spprintf(&message, 0, "%s [%d] (severity %d) [%s]",
  53. msg, einfo->dberr, einfo->severity, stmt ? stmt->active_query_string : "");
  54. add_next_index_long(info, einfo->dberr);
  55. add_next_index_string(info, message, 0);
  56. add_next_index_long(info, einfo->oserr);
  57. add_next_index_long(info, einfo->severity);
  58. if (einfo->oserrstr) {
  59. add_next_index_string(info, einfo->oserrstr, 1);
  60. }
  61. return 1;
  62. }
  63. static int dblib_handle_closer(pdo_dbh_t *dbh TSRMLS_DC)
  64. {
  65. pdo_dblib_db_handle *H = (pdo_dblib_db_handle *)dbh->driver_data;
  66. if (H) {
  67. if (H->link) {
  68. dbclose(H->link);
  69. H->link = NULL;
  70. }
  71. if (H->login) {
  72. dbfreelogin(H->login);
  73. H->login = NULL;
  74. }
  75. pefree(H, dbh->is_persistent);
  76. dbh->driver_data = NULL;
  77. }
  78. return 0;
  79. }
  80. static int dblib_handle_preparer(pdo_dbh_t *dbh, const char *sql, long sql_len, pdo_stmt_t *stmt, zval *driver_options TSRMLS_DC)
  81. {
  82. pdo_dblib_db_handle *H = (pdo_dblib_db_handle *)dbh->driver_data;
  83. pdo_dblib_stmt *S = ecalloc(1, sizeof(*S));
  84. S->H = H;
  85. stmt->driver_data = S;
  86. stmt->methods = &dblib_stmt_methods;
  87. stmt->supports_placeholders = PDO_PLACEHOLDER_NONE;
  88. S->err.sqlstate = stmt->error_code;
  89. return 1;
  90. }
  91. static long dblib_handle_doer(pdo_dbh_t *dbh, const char *sql, long sql_len TSRMLS_DC)
  92. {
  93. pdo_dblib_db_handle *H = (pdo_dblib_db_handle *)dbh->driver_data;
  94. RETCODE ret, resret;
  95. dbsetuserdata(H->link, (BYTE*)&H->err);
  96. if (FAIL == dbcmd(H->link, sql)) {
  97. return -1;
  98. }
  99. if (FAIL == dbsqlexec(H->link)) {
  100. return -1;
  101. }
  102. resret = dbresults(H->link);
  103. if (resret == FAIL) {
  104. return -1;
  105. }
  106. ret = dbnextrow(H->link);
  107. if (ret == FAIL) {
  108. return -1;
  109. }
  110. if (dbnumcols(H->link) <= 0) {
  111. return DBCOUNT(H->link);
  112. }
  113. /* throw away any rows it might have returned */
  114. dbcanquery(H->link);
  115. return DBCOUNT(H->link);
  116. }
  117. static int dblib_handle_quoter(pdo_dbh_t *dbh, const char *unquoted, int unquotedlen, char **quoted, int *quotedlen, enum pdo_param_type paramtype TSRMLS_DC)
  118. {
  119. pdo_dblib_db_handle *H = (pdo_dblib_db_handle *)dbh->driver_data;
  120. char *q;
  121. int l = 1;
  122. *quoted = q = safe_emalloc(2, unquotedlen, 3);
  123. *q++ = '\'';
  124. while (unquotedlen--) {
  125. if (*unquoted == '\'') {
  126. *q++ = '\'';
  127. *q++ = '\'';
  128. l += 2;
  129. } else {
  130. *q++ = *unquoted;
  131. ++l;
  132. }
  133. unquoted++;
  134. }
  135. *q++ = '\'';
  136. *q++ = '\0';
  137. *quotedlen = l+1;
  138. return 1;
  139. }
  140. static int pdo_dblib_transaction_cmd(const char *cmd, pdo_dbh_t *dbh TSRMLS_DC)
  141. {
  142. pdo_dblib_db_handle *H = (pdo_dblib_db_handle *)dbh->driver_data;
  143. RETCODE ret;
  144. if (FAIL == dbcmd(H->link, cmd)) {
  145. return 0;
  146. }
  147. if (FAIL == dbsqlexec(H->link)) {
  148. return 0;
  149. }
  150. return 1;
  151. }
  152. static int dblib_handle_begin(pdo_dbh_t *dbh TSRMLS_DC)
  153. {
  154. return pdo_dblib_transaction_cmd("BEGIN TRANSACTION", dbh TSRMLS_CC);
  155. }
  156. static int dblib_handle_commit(pdo_dbh_t *dbh TSRMLS_DC)
  157. {
  158. return pdo_dblib_transaction_cmd("COMMIT TRANSACTION", dbh TSRMLS_CC);
  159. }
  160. static int dblib_handle_rollback(pdo_dbh_t *dbh TSRMLS_DC)
  161. {
  162. return pdo_dblib_transaction_cmd("ROLLBACK TRANSACTION", dbh TSRMLS_CC);
  163. }
  164. char *dblib_handle_last_id(pdo_dbh_t *dbh, const char *name, unsigned int *len TSRMLS_DC)
  165. {
  166. pdo_dblib_db_handle *H = (pdo_dblib_db_handle *)dbh->driver_data;
  167. RETCODE ret;
  168. char *id = NULL;
  169. /*
  170. * Would use scope_identity() but it's not implemented on Sybase
  171. */
  172. if (FAIL == dbcmd(H->link, "SELECT @@IDENTITY")) {
  173. return NULL;
  174. }
  175. if (FAIL == dbsqlexec(H->link)) {
  176. return NULL;
  177. }
  178. ret = dbresults(H->link);
  179. if (ret == FAIL || ret == NO_MORE_RESULTS) {
  180. dbcancel(H->link);
  181. return NULL;
  182. }
  183. ret = dbnextrow(H->link);
  184. if (ret == FAIL || ret == NO_MORE_ROWS) {
  185. dbcancel(H->link);
  186. return NULL;
  187. }
  188. if (dbdatlen(H->link, 1) == 0) {
  189. dbcancel(H->link);
  190. return NULL;
  191. }
  192. id = emalloc(32);
  193. *len = dbconvert(NULL, (dbcoltype(H->link, 1)) , (dbdata(H->link, 1)) , (dbdatlen(H->link, 1)), SQLCHAR, id, (DBINT)-1);
  194. dbcancel(H->link);
  195. return id;
  196. }
  197. static struct pdo_dbh_methods dblib_methods = {
  198. dblib_handle_closer,
  199. dblib_handle_preparer,
  200. dblib_handle_doer,
  201. dblib_handle_quoter,
  202. dblib_handle_begin, /* begin */
  203. dblib_handle_commit, /* commit */
  204. dblib_handle_rollback, /* rollback */
  205. NULL, /*set attr */
  206. dblib_handle_last_id, /* last insert id */
  207. dblib_fetch_error, /* fetch error */
  208. NULL, /* get attr */
  209. NULL, /* check liveness */
  210. NULL, /* get driver methods */
  211. NULL, /* request shutdown */
  212. NULL /* in transaction */
  213. };
  214. static int pdo_dblib_handle_factory(pdo_dbh_t *dbh, zval *driver_options TSRMLS_DC)
  215. {
  216. pdo_dblib_db_handle *H;
  217. int i, nvars, nvers, ret = 0;
  218. int *val;
  219. const pdo_dblib_keyval tdsver[] = {
  220. {"4.2",DBVERSION_42}
  221. ,{"4.6",DBVERSION_46}
  222. ,{"5.0",DBVERSION_70} /* FIXME: This does not work with Sybase, but environ will */
  223. ,{"6.0",DBVERSION_70}
  224. ,{"7.0",DBVERSION_70}
  225. #ifdef DBVERSION_71
  226. ,{"7.1",DBVERSION_71}
  227. #endif
  228. #ifdef DBVERSION_72
  229. ,{"7.2",DBVERSION_72}
  230. ,{"8.0",DBVERSION_72}
  231. #endif
  232. #ifdef DBVERSION_73
  233. ,{"7.3",DBVERSION_73}
  234. #endif
  235. #ifdef DBVERSION_74
  236. ,{"7.4",DBVERSION_74}
  237. #endif
  238. ,{"10.0",DBVERSION_100}
  239. ,{"auto",0} /* Only works with FreeTDS. Other drivers will bork */
  240. };
  241. struct pdo_data_src_parser vars[] = {
  242. { "charset", NULL, 0 }
  243. ,{ "appname", "PHP " PDO_DBLIB_FLAVOUR, 0 }
  244. ,{ "host", "127.0.0.1", 0 }
  245. ,{ "dbname", NULL, 0 }
  246. ,{ "secure", NULL, 0 } /* DBSETLSECURE */
  247. ,{ "version", NULL, 0 } /* DBSETLVERSION */
  248. };
  249. nvars = sizeof(vars)/sizeof(vars[0]);
  250. nvers = sizeof(tdsver)/sizeof(tdsver[0]);
  251. php_pdo_parse_data_source(dbh->data_source, dbh->data_source_len, vars, nvars);
  252. H = pecalloc(1, sizeof(*H), dbh->is_persistent);
  253. H->login = dblogin();
  254. H->err.sqlstate = dbh->error_code;
  255. if (!H->login) {
  256. goto cleanup;
  257. }
  258. DBERRHANDLE(H->login, (EHANDLEFUNC) error_handler);
  259. DBMSGHANDLE(H->login, (MHANDLEFUNC) msg_handler);
  260. if(vars[5].optval) {
  261. for(i=0;i<nvers;i++) {
  262. if(strcmp(vars[5].optval,tdsver[i].key) == 0) {
  263. if(FAIL==dbsetlversion(H->login, tdsver[i].value)) {
  264. pdo_raise_impl_error(dbh, NULL, "HY000", "PDO_DBLIB: Failed to set version specified in connection string." TSRMLS_CC);
  265. goto cleanup;
  266. }
  267. break;
  268. }
  269. }
  270. if (i==nvers) {
  271. printf("Invalid version '%s'\n", vars[5].optval);
  272. pdo_raise_impl_error(dbh, NULL, "HY000", "PDO_DBLIB: Invalid version specified in connection string." TSRMLS_CC);
  273. goto cleanup; /* unknown version specified */
  274. }
  275. }
  276. if (dbh->username) {
  277. if(FAIL == DBSETLUSER(H->login, dbh->username)) {
  278. goto cleanup;
  279. }
  280. }
  281. if (dbh->password) {
  282. if(FAIL == DBSETLPWD(H->login, dbh->password)) {
  283. goto cleanup;
  284. }
  285. }
  286. #if !PHP_DBLIB_IS_MSSQL
  287. if (vars[0].optval) {
  288. DBSETLCHARSET(H->login, vars[0].optval);
  289. }
  290. #endif
  291. DBSETLAPP(H->login, vars[1].optval);
  292. /* DBSETLDBNAME is only available in FreeTDS 0.92 or above */
  293. #ifdef DBSETLDBNAME
  294. if (vars[3].optval) {
  295. if(FAIL == DBSETLDBNAME(H->login, vars[3].optval)) goto cleanup;
  296. }
  297. #endif
  298. H->link = dbopen(H->login, vars[2].optval);
  299. if (!H->link) {
  300. goto cleanup;
  301. }
  302. /*
  303. * FreeTDS < 0.92 does not support the DBSETLDBNAME option
  304. * Send use database here after login (Will not work with SQL Azure)
  305. */
  306. #ifndef DBSETLDBNAME
  307. if (vars[3].optval) {
  308. if(FAIL == dbuse(H->link, vars[3].optval)) goto cleanup;
  309. }
  310. #endif
  311. #if PHP_DBLIB_IS_MSSQL
  312. /* dblib do not return more than this length from text/image */
  313. DBSETOPT(H->link, DBTEXTLIMIT, "2147483647");
  314. #endif
  315. /* limit text/image from network */
  316. DBSETOPT(H->link, DBTEXTSIZE, "2147483647");
  317. /* allow double quoted indentifiers */
  318. DBSETOPT(H->link, DBQUOTEDIDENT, "1");
  319. ret = 1;
  320. dbh->max_escaped_char_length = 2;
  321. dbh->alloc_own_columns = 1;
  322. cleanup:
  323. for (i = 0; i < nvars; i++) {
  324. if (vars[i].freeme) {
  325. efree(vars[i].optval);
  326. }
  327. }
  328. dbh->methods = &dblib_methods;
  329. dbh->driver_data = H;
  330. if (!ret) {
  331. zend_throw_exception_ex(php_pdo_get_exception(), DBLIB_G(err).dberr TSRMLS_CC,
  332. "SQLSTATE[%s] %s (severity %d)",
  333. DBLIB_G(err).sqlstate,
  334. DBLIB_G(err).dberrstr,
  335. DBLIB_G(err).severity);
  336. }
  337. return ret;
  338. }
  339. pdo_driver_t pdo_dblib_driver = {
  340. #if PDO_DBLIB_IS_MSSQL
  341. PDO_DRIVER_HEADER(mssql),
  342. #elif defined(PHP_WIN32)
  343. #define PDO_DBLIB_IS_SYBASE
  344. PDO_DRIVER_HEADER(sybase),
  345. #else
  346. PDO_DRIVER_HEADER(dblib),
  347. #endif
  348. pdo_dblib_handle_factory
  349. };