oci_driver.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895
  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: Wez Furlong <wez@php.net> |
  14. +----------------------------------------------------------------------+
  15. */
  16. #ifdef HAVE_CONFIG_H
  17. #include "config.h"
  18. #endif
  19. #include "php.h"
  20. #include "php_ini.h"
  21. #include "ext/standard/info.h"
  22. #include "pdo/php_pdo.h"
  23. #include "pdo/php_pdo_driver.h"
  24. #include "php_pdo_oci.h"
  25. #include "php_pdo_oci_int.h"
  26. #include "Zend/zend_exceptions.h"
  27. static inline ub4 pdo_oci_sanitize_prefetch(long prefetch);
  28. static void pdo_oci_fetch_error_func(pdo_dbh_t *dbh, pdo_stmt_t *stmt, zval *info) /* {{{ */
  29. {
  30. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  31. pdo_oci_error_info *einfo;
  32. einfo = &H->einfo;
  33. if (stmt) {
  34. pdo_oci_stmt *S = (pdo_oci_stmt*)stmt->driver_data;
  35. if (S->einfo.errmsg) {
  36. einfo = &S->einfo;
  37. }
  38. }
  39. if (einfo->errcode) {
  40. add_next_index_long(info, einfo->errcode);
  41. add_next_index_string(info, einfo->errmsg);
  42. }
  43. }
  44. /* }}} */
  45. ub4 _oci_error(OCIError *err, pdo_dbh_t *dbh, pdo_stmt_t *stmt, char *what, sword status, int isinit, const char *file, int line) /* {{{ */
  46. {
  47. text errbuf[1024] = "<<Unknown>>";
  48. char tmp_buf[2048];
  49. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  50. pdo_oci_error_info *einfo;
  51. pdo_oci_stmt *S = NULL;
  52. pdo_error_type *pdo_err = &dbh->error_code;
  53. if (stmt) {
  54. S = (pdo_oci_stmt*)stmt->driver_data;
  55. einfo = &S->einfo;
  56. pdo_err = &stmt->error_code;
  57. }
  58. else {
  59. einfo = &H->einfo;
  60. }
  61. if (einfo->errmsg) {
  62. pefree(einfo->errmsg, dbh->is_persistent);
  63. }
  64. einfo->errmsg = NULL;
  65. einfo->errcode = 0;
  66. einfo->file = file;
  67. einfo->line = line;
  68. if (isinit) { /* Initialization error */
  69. strcpy(*pdo_err, "HY000");
  70. slprintf(tmp_buf, sizeof(tmp_buf), "%s (%s:%d)", what, file, line);
  71. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  72. }
  73. else {
  74. switch (status) {
  75. case OCI_SUCCESS:
  76. strcpy(*pdo_err, "00000");
  77. break;
  78. case OCI_ERROR:
  79. OCIErrorGet(err, (ub4)1, NULL, &einfo->errcode, errbuf, (ub4)sizeof(errbuf), OCI_HTYPE_ERROR);
  80. slprintf(tmp_buf, sizeof(tmp_buf), "%s: %s (%s:%d)", what, errbuf, file, line);
  81. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  82. break;
  83. case OCI_SUCCESS_WITH_INFO:
  84. OCIErrorGet(err, (ub4)1, NULL, &einfo->errcode, errbuf, (ub4)sizeof(errbuf), OCI_HTYPE_ERROR);
  85. slprintf(tmp_buf, sizeof(tmp_buf), "%s: OCI_SUCCESS_WITH_INFO: %s (%s:%d)", what, errbuf, file, line);
  86. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  87. break;
  88. case OCI_NEED_DATA:
  89. slprintf(tmp_buf, sizeof(tmp_buf), "%s: OCI_NEED_DATA (%s:%d)", what, file, line);
  90. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  91. break;
  92. case OCI_NO_DATA:
  93. slprintf(tmp_buf, sizeof(tmp_buf), "%s: OCI_NO_DATA (%s:%d)", what, file, line);
  94. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  95. break;
  96. case OCI_INVALID_HANDLE:
  97. slprintf(tmp_buf, sizeof(tmp_buf), "%s: OCI_INVALID_HANDLE (%s:%d)", what, file, line);
  98. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  99. break;
  100. case OCI_STILL_EXECUTING:
  101. slprintf(tmp_buf, sizeof(tmp_buf), "%s: OCI_STILL_EXECUTING (%s:%d)", what, file, line);
  102. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  103. break;
  104. case OCI_CONTINUE:
  105. slprintf(tmp_buf, sizeof(tmp_buf), "%s: OCI_CONTINUE (%s:%d)", what, file, line);
  106. einfo->errmsg = pestrdup(tmp_buf, dbh->is_persistent);
  107. break;
  108. }
  109. if (einfo->errcode) {
  110. switch (einfo->errcode) {
  111. case 1013: /* user requested cancel of current operation */
  112. zend_bailout();
  113. break;
  114. case 12154: /* ORA-12154: TNS:could not resolve service name */
  115. strcpy(*pdo_err, "42S02");
  116. break;
  117. case 22: /* ORA-00022: invalid session id */
  118. case 378:
  119. case 602:
  120. case 603:
  121. case 604:
  122. case 609:
  123. case 1012: /* ORA-01012: */
  124. case 1033:
  125. case 1041:
  126. case 1043:
  127. case 1089:
  128. case 1090:
  129. case 1092:
  130. case 3113: /* ORA-03133: end of file on communication channel */
  131. case 3114:
  132. case 3122:
  133. case 3135:
  134. case 12153:
  135. case 27146:
  136. case 28511:
  137. /* consider the connection closed */
  138. dbh->is_closed = 1;
  139. H->attached = 0;
  140. strcpy(*pdo_err, "01002"); /* FIXME */
  141. break;
  142. default:
  143. strcpy(*pdo_err, "HY000");
  144. }
  145. }
  146. if (stmt) {
  147. /* always propagate the error code back up to the dbh,
  148. * so that we can catch the error information when execute
  149. * is called via query. See Bug #33707 */
  150. if (H->einfo.errmsg) {
  151. pefree(H->einfo.errmsg, dbh->is_persistent);
  152. }
  153. H->einfo = *einfo;
  154. H->einfo.errmsg = einfo->errmsg ? pestrdup(einfo->errmsg, dbh->is_persistent) : NULL;
  155. strcpy(dbh->error_code, stmt->error_code);
  156. }
  157. }
  158. /* little mini hack so that we can use this code from the dbh ctor */
  159. if (!dbh->methods && status != OCI_SUCCESS_WITH_INFO) {
  160. zend_throw_exception_ex(php_pdo_get_exception(), einfo->errcode, "SQLSTATE[%s]: %s", *pdo_err, einfo->errmsg);
  161. }
  162. return einfo->errcode;
  163. }
  164. /* }}} */
  165. static void oci_handle_closer(pdo_dbh_t *dbh) /* {{{ */
  166. {
  167. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  168. if (H->svc) {
  169. /* rollback any outstanding work */
  170. OCITransRollback(H->svc, H->err, 0);
  171. }
  172. if (H->session) {
  173. OCIHandleFree(H->session, OCI_HTYPE_SESSION);
  174. H->session = NULL;
  175. }
  176. if (H->svc) {
  177. OCIHandleFree(H->svc, OCI_HTYPE_SVCCTX);
  178. H->svc = NULL;
  179. }
  180. if (H->server && H->attached) {
  181. H->last_err = OCIServerDetach(H->server, H->err, OCI_DEFAULT);
  182. if (H->last_err) {
  183. oci_drv_error("OCIServerDetach");
  184. }
  185. H->attached = 0;
  186. }
  187. if (H->server) {
  188. OCIHandleFree(H->server, OCI_HTYPE_SERVER);
  189. H->server = NULL;
  190. }
  191. if (H->err) {
  192. OCIHandleFree(H->err, OCI_HTYPE_ERROR);
  193. H->err = NULL;
  194. }
  195. if (H->charset && H->env) {
  196. OCIHandleFree(H->env, OCI_HTYPE_ENV);
  197. H->env = NULL;
  198. }
  199. if (H->einfo.errmsg) {
  200. pefree(H->einfo.errmsg, dbh->is_persistent);
  201. H->einfo.errmsg = NULL;
  202. }
  203. pefree(H, dbh->is_persistent);
  204. }
  205. /* }}} */
  206. static bool oci_handle_preparer(pdo_dbh_t *dbh, zend_string *sql, pdo_stmt_t *stmt, zval *driver_options) /* {{{ */
  207. {
  208. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  209. pdo_oci_stmt *S = ecalloc(1, sizeof(*S));
  210. ub4 prefetch;
  211. zend_string *nsql = NULL;
  212. int ret;
  213. #ifdef HAVE_OCISTMTFETCH2
  214. S->exec_type = pdo_attr_lval(driver_options, PDO_ATTR_CURSOR,
  215. PDO_CURSOR_FWDONLY) == PDO_CURSOR_SCROLL ?
  216. OCI_STMT_SCROLLABLE_READONLY : OCI_DEFAULT;
  217. #else
  218. S->exec_type = OCI_DEFAULT;
  219. #endif
  220. S->H = H;
  221. stmt->supports_placeholders = PDO_PLACEHOLDER_NAMED;
  222. ret = pdo_parse_params(stmt, sql, &nsql);
  223. if (ret == 1) {
  224. /* query was re-written */
  225. sql = nsql;
  226. } else if (ret == -1) {
  227. /* couldn't grok it */
  228. strcpy(dbh->error_code, stmt->error_code);
  229. efree(S);
  230. return false;
  231. }
  232. /* create an OCI statement handle */
  233. OCIHandleAlloc(H->env, (dvoid*)&S->stmt, OCI_HTYPE_STMT, 0, NULL);
  234. /* and our own private error handle */
  235. OCIHandleAlloc(H->env, (dvoid*)&S->err, OCI_HTYPE_ERROR, 0, NULL);
  236. if (ZSTR_LEN(sql) != 0) {
  237. H->last_err = OCIStmtPrepare(S->stmt, H->err, (text*) ZSTR_VAL(sql), (ub4) ZSTR_LEN(sql), OCI_NTV_SYNTAX, OCI_DEFAULT);
  238. if (nsql) {
  239. zend_string_release(nsql);
  240. nsql = NULL;
  241. }
  242. if (H->last_err) {
  243. H->last_err = oci_drv_error("OCIStmtPrepare");
  244. OCIHandleFree(S->stmt, OCI_HTYPE_STMT);
  245. OCIHandleFree(S->err, OCI_HTYPE_ERROR);
  246. efree(S);
  247. return false;
  248. }
  249. }
  250. prefetch = H->prefetch; /* Note 0 is allowed so in future REF CURSORs can be used & then passed with no row loss*/
  251. H->last_err = OCIAttrSet(S->stmt, OCI_HTYPE_STMT, &prefetch, 0,
  252. OCI_ATTR_PREFETCH_ROWS, H->err);
  253. if (!H->last_err) {
  254. prefetch *= PDO_OCI_PREFETCH_ROWSIZE;
  255. H->last_err = OCIAttrSet(S->stmt, OCI_HTYPE_STMT, &prefetch, 0,
  256. OCI_ATTR_PREFETCH_MEMORY, H->err);
  257. }
  258. stmt->driver_data = S;
  259. stmt->methods = &oci_stmt_methods;
  260. if (nsql) {
  261. zend_string_release(nsql);
  262. nsql = NULL;
  263. }
  264. return true;
  265. }
  266. /* }}} */
  267. static zend_long oci_handle_doer(pdo_dbh_t *dbh, const zend_string *sql) /* {{{ */
  268. {
  269. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  270. OCIStmt *stmt;
  271. ub2 stmt_type;
  272. ub4 rowcount;
  273. int ret = -1;
  274. OCIHandleAlloc(H->env, (dvoid*)&stmt, OCI_HTYPE_STMT, 0, NULL);
  275. H->last_err = OCIStmtPrepare(stmt, H->err, (text*)ZSTR_VAL(sql), (ub4) ZSTR_LEN(sql), OCI_NTV_SYNTAX, OCI_DEFAULT);
  276. if (H->last_err) {
  277. H->last_err = oci_drv_error("OCIStmtPrepare");
  278. OCIHandleFree(stmt, OCI_HTYPE_STMT);
  279. return -1;
  280. }
  281. H->last_err = OCIAttrGet(stmt, OCI_HTYPE_STMT, &stmt_type, 0, OCI_ATTR_STMT_TYPE, H->err);
  282. if (stmt_type == OCI_STMT_SELECT) {
  283. /* invalid usage; cancel it */
  284. OCIHandleFree(stmt, OCI_HTYPE_STMT);
  285. php_error_docref(NULL, E_WARNING, "issuing a SELECT query here is invalid");
  286. return -1;
  287. }
  288. /* now we are good to go */
  289. H->last_err = OCIStmtExecute(H->svc, stmt, H->err, 1, 0, NULL, NULL,
  290. (dbh->auto_commit && !dbh->in_txn) ? OCI_COMMIT_ON_SUCCESS : OCI_DEFAULT);
  291. sword last_err = H->last_err;
  292. if (last_err) {
  293. H->last_err = oci_drv_error("OCIStmtExecute");
  294. }
  295. if (!last_err || last_err == OCI_SUCCESS_WITH_INFO) {
  296. /* return the number of affected rows */
  297. H->last_err = OCIAttrGet(stmt, OCI_HTYPE_STMT, &rowcount, 0, OCI_ATTR_ROW_COUNT, H->err);
  298. ret = rowcount;
  299. }
  300. OCIHandleFree(stmt, OCI_HTYPE_STMT);
  301. return ret;
  302. }
  303. /* }}} */
  304. static zend_string* oci_handle_quoter(pdo_dbh_t *dbh, const zend_string *unquoted, enum pdo_param_type paramtype ) /* {{{ */
  305. {
  306. int qcount = 0;
  307. char const *cu, *l, *r;
  308. char *c, *quoted;
  309. size_t quotedlen;
  310. zend_string *quoted_str;
  311. if (ZSTR_LEN(unquoted) == 0) {
  312. return zend_string_init("''", 2, 0);
  313. }
  314. /* count single quotes */
  315. for (cu = ZSTR_VAL(unquoted); (cu = strchr(cu,'\'')); qcount++, cu++)
  316. ; /* empty loop */
  317. quotedlen = ZSTR_LEN(unquoted) + qcount + 2;
  318. quoted = c = emalloc(quotedlen+1);
  319. *c++ = '\'';
  320. /* foreach (chunk that ends in a quote) */
  321. for (l = ZSTR_VAL(unquoted); (r = strchr(l,'\'')); l = r+1) {
  322. strncpy(c, l, r-l+1);
  323. c += (r-l+1);
  324. *c++ = '\''; /* add second quote */
  325. }
  326. /* Copy remainder and add enclosing quote */
  327. strncpy(c, l, quotedlen-(c-quoted)-1);
  328. quoted[quotedlen-1] = '\'';
  329. quoted[quotedlen] = '\0';
  330. quoted_str = zend_string_init(quoted, quotedlen, 0);
  331. efree(quoted);
  332. return quoted_str;
  333. }
  334. /* }}} */
  335. static bool oci_handle_begin(pdo_dbh_t *dbh) /* {{{ */
  336. {
  337. /* with Oracle, there is nothing special to be done */
  338. return true;
  339. }
  340. /* }}} */
  341. static bool oci_handle_commit(pdo_dbh_t *dbh) /* {{{ */
  342. {
  343. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  344. H->last_err = OCITransCommit(H->svc, H->err, 0);
  345. if (H->last_err) {
  346. H->last_err = oci_drv_error("OCITransCommit");
  347. return false;
  348. }
  349. return true;
  350. }
  351. /* }}} */
  352. static bool oci_handle_rollback(pdo_dbh_t *dbh) /* {{{ */
  353. {
  354. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  355. H->last_err = OCITransRollback(H->svc, H->err, 0);
  356. if (H->last_err) {
  357. H->last_err = oci_drv_error("OCITransRollback");
  358. return false;
  359. }
  360. return true;
  361. }
  362. /* }}} */
  363. static bool oci_handle_set_attribute(pdo_dbh_t *dbh, zend_long attr, zval *val) /* {{{ */
  364. {
  365. zend_long lval;
  366. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  367. switch (attr) {
  368. case PDO_ATTR_AUTOCOMMIT:
  369. {
  370. bool bval;
  371. if (!pdo_get_bool_param(&bval, val)) {
  372. return false;
  373. }
  374. if (dbh->in_txn) {
  375. /* Assume they want to commit whatever is outstanding */
  376. H->last_err = OCITransCommit(H->svc, H->err, 0);
  377. if (H->last_err) {
  378. H->last_err = oci_drv_error("OCITransCommit");
  379. return false;
  380. }
  381. dbh->in_txn = false;
  382. }
  383. dbh->auto_commit = (unsigned int) bval;
  384. return true;
  385. }
  386. case PDO_ATTR_PREFETCH:
  387. {
  388. if (!pdo_get_long_param(&lval, val)) {
  389. return false;
  390. }
  391. H->prefetch = pdo_oci_sanitize_prefetch(lval);
  392. return true;
  393. }
  394. case PDO_OCI_ATTR_ACTION:
  395. {
  396. #if (OCI_MAJOR_VERSION >= 10)
  397. zend_string *action = zval_try_get_string(val);
  398. if (UNEXPECTED(!action)) {
  399. return false;
  400. }
  401. H->last_err = OCIAttrSet(H->session, OCI_HTYPE_SESSION,
  402. (dvoid *) ZSTR_VAL(action), (ub4) ZSTR_LEN(action),
  403. OCI_ATTR_ACTION, H->err);
  404. if (H->last_err) {
  405. oci_drv_error("OCIAttrSet: OCI_ATTR_ACTION");
  406. return false;
  407. }
  408. return true;
  409. #else
  410. oci_drv_error("Unsupported attribute type");
  411. return false;
  412. #endif
  413. }
  414. case PDO_OCI_ATTR_CLIENT_INFO:
  415. {
  416. #if (OCI_MAJOR_VERSION >= 10)
  417. zend_string *client_info = zval_try_get_string(val);
  418. if (UNEXPECTED(!client_info)) {
  419. return false;
  420. }
  421. H->last_err = OCIAttrSet(H->session, OCI_HTYPE_SESSION,
  422. (dvoid *) ZSTR_VAL(client_info), (ub4) ZSTR_LEN(client_info),
  423. OCI_ATTR_CLIENT_INFO, H->err);
  424. if (H->last_err) {
  425. oci_drv_error("OCIAttrSet: OCI_ATTR_CLIENT_INFO");
  426. return false;
  427. }
  428. return true;
  429. #else
  430. oci_drv_error("Unsupported attribute type");
  431. return false;
  432. #endif
  433. }
  434. case PDO_OCI_ATTR_CLIENT_IDENTIFIER:
  435. {
  436. #if (OCI_MAJOR_VERSION >= 10)
  437. zend_string *identifier = zval_try_get_string(val);
  438. if (UNEXPECTED(!identifier)) {
  439. return false;
  440. }
  441. H->last_err = OCIAttrSet(H->session, OCI_HTYPE_SESSION,
  442. (dvoid *) ZSTR_VAL(identifier), (ub4) ZSTR_LEN(identifier),
  443. OCI_ATTR_CLIENT_IDENTIFIER, H->err);
  444. if (H->last_err) {
  445. oci_drv_error("OCIAttrSet: OCI_ATTR_CLIENT_IDENTIFIER");
  446. return false;
  447. }
  448. return true;
  449. #else
  450. oci_drv_error("Unsupported attribute type");
  451. return false;
  452. #endif
  453. }
  454. case PDO_OCI_ATTR_MODULE:
  455. {
  456. #if (OCI_MAJOR_VERSION >= 10)
  457. zend_string *module = zval_try_get_string(val);
  458. if (UNEXPECTED(!module)) {
  459. return false;
  460. }
  461. H->last_err = OCIAttrSet(H->session, OCI_HTYPE_SESSION,
  462. (dvoid *) ZSTR_VAL(module), (ub4) ZSTR_LEN(module),
  463. OCI_ATTR_MODULE, H->err);
  464. if (H->last_err) {
  465. oci_drv_error("OCIAttrSet: OCI_ATTR_MODULE");
  466. return false;
  467. }
  468. return true;
  469. #else
  470. oci_drv_error("Unsupported attribute type");
  471. return false;
  472. #endif
  473. }
  474. case PDO_OCI_ATTR_CALL_TIMEOUT:
  475. {
  476. #if (OCI_MAJOR_VERSION >= 18)
  477. if (!pdo_get_long_param(&lval, val)) {
  478. return false;
  479. }
  480. ub4 timeout = (ub4) lval;
  481. H->last_err = OCIAttrSet(H->svc, OCI_HTYPE_SVCCTX,
  482. (dvoid *) &timeout, (ub4) 0,
  483. OCI_ATTR_CALL_TIMEOUT, H->err);
  484. if (H->last_err) {
  485. oci_drv_error("OCIAttrSet: OCI_ATTR_CALL_TIMEOUT");
  486. return false;
  487. }
  488. return true;
  489. #else
  490. oci_drv_error("Unsupported attribute type");
  491. return false;
  492. #endif
  493. }
  494. default:
  495. return false;
  496. }
  497. }
  498. /* }}} */
  499. static int oci_handle_get_attribute(pdo_dbh_t *dbh, zend_long attr, zval *return_value) /* {{{ */
  500. {
  501. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  502. switch (attr) {
  503. case PDO_ATTR_SERVER_VERSION:
  504. case PDO_ATTR_SERVER_INFO:
  505. {
  506. text infostr[512];
  507. char verstr[15];
  508. ub4 vernum;
  509. if (OCIServerRelease(H->svc, H->err, infostr, (ub4)sizeof(infostr), (ub1)OCI_HTYPE_SVCCTX, &vernum))
  510. {
  511. ZVAL_STRING(return_value, "<<Unknown>>");
  512. } else {
  513. if (attr == PDO_ATTR_SERVER_INFO) {
  514. ZVAL_STRING(return_value, (char *)infostr);
  515. } else {
  516. slprintf(verstr, sizeof(verstr), "%d.%d.%d.%d.%d",
  517. (int)((vernum>>24) & 0xFF), /* version number */
  518. (int)((vernum>>20) & 0x0F), /* release number*/
  519. (int)((vernum>>12) & 0xFF), /* update number */
  520. (int)((vernum>>8) & 0x0F), /* port release number */
  521. (int)((vernum>>0) & 0xFF)); /* port update number */
  522. ZVAL_STRING(return_value, verstr);
  523. }
  524. }
  525. return TRUE;
  526. }
  527. case PDO_ATTR_CLIENT_VERSION:
  528. {
  529. #if OCI_MAJOR_VERSION > 10 || (OCI_MAJOR_VERSION == 10 && OCI_MINOR_VERSION >= 2)
  530. /* Run time client version */
  531. sword major, minor, update, patch, port_update;
  532. char verstr[15];
  533. OCIClientVersion(&major, &minor, &update, &patch, &port_update);
  534. slprintf(verstr, sizeof(verstr), "%d.%d.%d.%d.%d", major, minor, update, patch, port_update);
  535. ZVAL_STRING(return_value, verstr);
  536. #elif defined(PHP_PDO_OCI_CLIENT_VERSION)
  537. /* Compile time client version */
  538. ZVAL_STRING(return_value, PHP_PDO_OCI_CLIENT_VERSION);
  539. #else
  540. return FALSE;
  541. #endif /* Check for OCIClientVersion() support */
  542. return TRUE;
  543. }
  544. case PDO_ATTR_AUTOCOMMIT:
  545. ZVAL_BOOL(return_value, dbh->auto_commit);
  546. return TRUE;
  547. case PDO_ATTR_PREFETCH:
  548. ZVAL_LONG(return_value, H->prefetch);
  549. return TRUE;
  550. case PDO_OCI_ATTR_CALL_TIMEOUT:
  551. {
  552. #if (OCI_MAJOR_VERSION >= 18)
  553. ub4 timeout;
  554. H->last_err = OCIAttrGet(H->svc, OCI_HTYPE_SVCCTX,
  555. (dvoid *) &timeout, NULL,
  556. OCI_ATTR_CALL_TIMEOUT, H->err);
  557. if (H->last_err) {
  558. oci_drv_error("OCIAttrGet: OCI_ATTR_CALL_TIMEOUT");
  559. return FALSE;
  560. }
  561. ZVAL_LONG(return_value, (zend_long) timeout);
  562. return TRUE;
  563. #else
  564. oci_drv_error("Unsupported attribute type");
  565. return FALSE;
  566. #endif
  567. }
  568. default:
  569. return FALSE;
  570. }
  571. return FALSE;
  572. }
  573. /* }}} */
  574. static zend_result pdo_oci_check_liveness(pdo_dbh_t *dbh) /* {{{ */
  575. {
  576. pdo_oci_db_handle *H = (pdo_oci_db_handle *)dbh->driver_data;
  577. sb4 error_code = 0;
  578. #if (!((OCI_MAJOR_VERSION > 10) || ((OCI_MAJOR_VERSION == 10) && (OCI_MINOR_VERSION >= 2))))
  579. char version[256];
  580. #endif
  581. /* TODO move attached check to PDO level */
  582. if (H->attached == 0) {
  583. return FAILURE;
  584. }
  585. /* TODO add persistent_timeout check at PDO level */
  586. /* Use OCIPing instead of OCIServerVersion. If OCIPing returns ORA-1010 (invalid OCI operation)
  587. * such as from Pre-10.1 servers, the error is still from the server and we would have
  588. * successfully performed a roundtrip and validated the connection. Use OCIServerVersion for
  589. * Pre-10.2 clients
  590. */
  591. #if ((OCI_MAJOR_VERSION > 10) || ((OCI_MAJOR_VERSION == 10) && (OCI_MINOR_VERSION >= 2))) /* OCIPing available 10.2 onwards */
  592. H->last_err = OCIPing (H->svc, H->err, OCI_DEFAULT);
  593. #else
  594. /* use good old OCIServerVersion() */
  595. H->last_err = OCIServerVersion (H->svc, H->err, (text *)version, sizeof(version), OCI_HTYPE_SVCCTX);
  596. #endif
  597. if (H->last_err == OCI_SUCCESS) {
  598. return SUCCESS;
  599. }
  600. OCIErrorGet (H->err, (ub4)1, NULL, &error_code, NULL, 0, OCI_HTYPE_ERROR);
  601. if (error_code == 1010) {
  602. return SUCCESS;
  603. }
  604. return FAILURE;
  605. }
  606. /* }}} */
  607. static const struct pdo_dbh_methods oci_methods = {
  608. oci_handle_closer,
  609. oci_handle_preparer,
  610. oci_handle_doer,
  611. oci_handle_quoter,
  612. oci_handle_begin,
  613. oci_handle_commit,
  614. oci_handle_rollback,
  615. oci_handle_set_attribute,
  616. NULL, /* last_id not supported */
  617. pdo_oci_fetch_error_func,
  618. oci_handle_get_attribute,
  619. pdo_oci_check_liveness, /* check_liveness */
  620. NULL, /* get_driver_methods */
  621. NULL, /* request_shutdown */
  622. NULL, /* in transaction, use PDO's internal tracking mechanism */
  623. NULL /* get_gc */
  624. };
  625. static int pdo_oci_handle_factory(pdo_dbh_t *dbh, zval *driver_options) /* {{{ */
  626. {
  627. pdo_oci_db_handle *H;
  628. int i, ret = 0;
  629. struct pdo_data_src_parser vars[] = {
  630. { "charset", NULL, 0 },
  631. { "dbname", "", 0 },
  632. { "user", NULL, 0 },
  633. { "password", NULL, 0 }
  634. };
  635. php_pdo_parse_data_source(dbh->data_source, dbh->data_source_len, vars, 4);
  636. H = pecalloc(1, sizeof(*H), dbh->is_persistent);
  637. dbh->driver_data = H;
  638. dbh->skip_param_evt =
  639. 1 << PDO_PARAM_EVT_FETCH_PRE |
  640. 1 << PDO_PARAM_EVT_FETCH_POST |
  641. 1 << PDO_PARAM_EVT_NORMALIZE;
  642. H->prefetch = PDO_OCI_PREFETCH_DEFAULT;
  643. /* allocate an environment */
  644. #ifdef HAVE_OCIENVNLSCREATE
  645. if (vars[0].optval) {
  646. H->charset = OCINlsCharSetNameToId(pdo_oci_Env, (const oratext *)vars[0].optval);
  647. if (!H->charset) {
  648. oci_init_error("OCINlsCharSetNameToId: unknown character set name");
  649. goto cleanup;
  650. } else {
  651. if (OCIEnvNlsCreate(&H->env, PDO_OCI_INIT_MODE, 0, NULL, NULL, NULL, 0, NULL, H->charset, H->charset) != OCI_SUCCESS) {
  652. oci_init_error("OCIEnvNlsCreate: Check the character set is valid and that PHP has access to Oracle libraries and NLS data");
  653. goto cleanup;
  654. }
  655. }
  656. }
  657. #endif
  658. if (H->env == NULL) {
  659. /* use the global environment */
  660. H->env = pdo_oci_Env;
  661. }
  662. /* something to hold errors */
  663. OCIHandleAlloc(H->env, (dvoid **)&H->err, OCI_HTYPE_ERROR, 0, NULL);
  664. /* handle for the server */
  665. OCIHandleAlloc(H->env, (dvoid **)&H->server, OCI_HTYPE_SERVER, 0, NULL);
  666. H->last_err = OCIServerAttach(H->server, H->err, (text*)vars[1].optval,
  667. (sb4) strlen(vars[1].optval), OCI_DEFAULT);
  668. if (H->last_err) {
  669. oci_drv_error("pdo_oci_handle_factory");
  670. goto cleanup;
  671. }
  672. H->attached = 1;
  673. /* create a service context */
  674. H->last_err = OCIHandleAlloc(H->env, (dvoid**)&H->svc, OCI_HTYPE_SVCCTX, 0, NULL);
  675. if (H->last_err) {
  676. oci_drv_error("OCIHandleAlloc: OCI_HTYPE_SVCCTX");
  677. goto cleanup;
  678. }
  679. H->last_err = OCIHandleAlloc(H->env, (dvoid**)&H->session, OCI_HTYPE_SESSION, 0, NULL);
  680. if (H->last_err) {
  681. oci_drv_error("OCIHandleAlloc: OCI_HTYPE_SESSION");
  682. goto cleanup;
  683. }
  684. /* set server handle into service handle */
  685. H->last_err = OCIAttrSet(H->svc, OCI_HTYPE_SVCCTX, H->server, 0, OCI_ATTR_SERVER, H->err);
  686. if (H->last_err) {
  687. oci_drv_error("OCIAttrSet: OCI_ATTR_SERVER");
  688. goto cleanup;
  689. }
  690. /* username */
  691. if (!dbh->username && vars[2].optval) {
  692. dbh->username = pestrdup(vars[2].optval, dbh->is_persistent);
  693. }
  694. if (dbh->username) {
  695. H->last_err = OCIAttrSet(H->session, OCI_HTYPE_SESSION,
  696. dbh->username, (ub4) strlen(dbh->username),
  697. OCI_ATTR_USERNAME, H->err);
  698. if (H->last_err) {
  699. oci_drv_error("OCIAttrSet: OCI_ATTR_USERNAME");
  700. goto cleanup;
  701. }
  702. }
  703. /* password */
  704. if (!dbh->password && vars[3].optval) {
  705. dbh->password = pestrdup(vars[3].optval, dbh->is_persistent);
  706. }
  707. if (dbh->password) {
  708. H->last_err = OCIAttrSet(H->session, OCI_HTYPE_SESSION,
  709. dbh->password, (ub4) strlen(dbh->password),
  710. OCI_ATTR_PASSWORD, H->err);
  711. if (H->last_err) {
  712. oci_drv_error("OCIAttrSet: OCI_ATTR_PASSWORD");
  713. goto cleanup;
  714. }
  715. }
  716. /* Now fire up the session */
  717. H->last_err = OCISessionBegin(H->svc, H->err, H->session, OCI_CRED_RDBMS, OCI_DEFAULT);
  718. if (H->last_err) {
  719. oci_drv_error("OCISessionBegin");
  720. /* OCISessionBegin returns OCI_SUCCESS_WITH_INFO when
  721. * user's password has expired, but is still usable.
  722. */
  723. if (H->last_err != OCI_SUCCESS_WITH_INFO) {
  724. goto cleanup;
  725. }
  726. }
  727. /* set the server handle into service handle */
  728. H->last_err = OCIAttrSet(H->svc, OCI_HTYPE_SVCCTX, H->session, 0, OCI_ATTR_SESSION, H->err);
  729. if (H->last_err) {
  730. oci_drv_error("OCIAttrSet: OCI_ATTR_SESSION");
  731. goto cleanup;
  732. }
  733. /* Get max character width */
  734. H->last_err = OCINlsNumericInfoGet(H->env, H->err, &H->max_char_width, OCI_NLS_CHARSET_MAXBYTESZ);
  735. if (H->last_err) {
  736. oci_drv_error("OCINlsNumericInfoGet: OCI_NLS_CHARSET_MAXBYTESZ");
  737. goto cleanup;
  738. }
  739. dbh->methods = &oci_methods;
  740. dbh->alloc_own_columns = 1;
  741. dbh->native_case = PDO_CASE_UPPER;
  742. ret = 1;
  743. cleanup:
  744. for (i = 0; i < sizeof(vars)/sizeof(vars[0]); i++) {
  745. if (vars[i].freeme) {
  746. efree(vars[i].optval);
  747. }
  748. }
  749. if (!ret) {
  750. oci_handle_closer(dbh);
  751. }
  752. return ret;
  753. }
  754. /* }}} */
  755. const pdo_driver_t pdo_oci_driver = {
  756. PDO_DRIVER_HEADER(oci),
  757. pdo_oci_handle_factory
  758. };
  759. static inline ub4 pdo_oci_sanitize_prefetch(long prefetch) /* {{{ */
  760. {
  761. if (prefetch < 0) {
  762. prefetch = 0;
  763. } else if (prefetch > UB4MAXVAL / PDO_OCI_PREFETCH_ROWSIZE) {
  764. prefetch = PDO_OCI_PREFETCH_DEFAULT;
  765. }
  766. return ((ub4)prefetch);
  767. }
  768. /* }}} */