blob: b5cf26536535e0046d8bb876be57a2cd569f2abe [file] [log] [blame]
Andrey Andreevc5536aa2012-11-01 17:33:58 +02001<?php
Derek Allard2067d1a2008-11-13 22:59:24 +00002/**
3 * CodeIgniter
4 *
Andrey Andreevfe9309d2015-01-09 17:48:58 +02005 * An open source application development framework for PHP
Derek Allard2067d1a2008-11-13 22:59:24 +00006 *
Andrey Andreevbdb96ca2014-10-28 00:13:31 +02007 * This content is released under the MIT License (MIT)
Andrey Andreev24abcb92012-01-05 20:40:15 +02008 *
Andrey Andreevfe9309d2015-01-09 17:48:58 +02009 * Copyright (c) 2014 - 2015, British Columbia Institute of Technology
Andrey Andreev24abcb92012-01-05 20:40:15 +020010 *
Andrey Andreevbdb96ca2014-10-28 00:13:31 +020011 * Permission is hereby granted, free of charge, to any person obtaining a copy
12 * of this software and associated documentation files (the "Software"), to deal
13 * in the Software without restriction, including without limitation the rights
14 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
15 * copies of the Software, and to permit persons to whom the Software is
16 * furnished to do so, subject to the following conditions:
Derek Jonesf4a4bd82011-10-20 12:18:42 -050017 *
Andrey Andreevbdb96ca2014-10-28 00:13:31 +020018 * The above copyright notice and this permission notice shall be included in
19 * all copies or substantial portions of the Software.
20 *
21 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
24 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
25 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
26 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
27 * THE SOFTWARE.
28 *
29 * @package CodeIgniter
30 * @author EllisLab Dev Team
darwinel871754a2014-02-11 17:34:57 +010031 * @copyright Copyright (c) 2008 - 2014, EllisLab, Inc. (http://ellislab.com/)
Andrey Andreevfe9309d2015-01-09 17:48:58 +020032 * @copyright Copyright (c) 2014 - 2015, British Columbia Institute of Technology (http://bcit.ca/)
Andrey Andreevbdb96ca2014-10-28 00:13:31 +020033 * @license http://opensource.org/licenses/MIT MIT License
34 * @link http://codeigniter.com
35 * @since Version 1.4.1
Derek Allard2067d1a2008-11-13 22:59:24 +000036 * @filesource
37 */
Andrey Andreevc5536aa2012-11-01 17:33:58 +020038defined('BASEPATH') OR exit('No direct script access allowed');
Derek Allard2067d1a2008-11-13 22:59:24 +000039
Derek Allard2067d1a2008-11-13 22:59:24 +000040/**
41 * oci8 Database Adapter Class
42 *
43 * Note: _DB is an extender class that the app controller
Jamie Rumbelow7efad202012-02-19 12:37:00 +000044 * creates dynamically based on whether the query builder
Derek Allard2067d1a2008-11-13 22:59:24 +000045 * class is being used or not.
46 *
Barry Mienydd671972010-10-04 16:33:58 +020047 * @package CodeIgniter
Derek Jones4b9c6292011-07-01 17:40:48 -050048 * @subpackage Drivers
Derek Allard2067d1a2008-11-13 22:59:24 +000049 * @category Database
Derek Jonesf4a4bd82011-10-20 12:18:42 -050050 * @author EllisLab Dev Team
Derek Allard2067d1a2008-11-13 22:59:24 +000051 * @link http://codeigniter.com/user_guide/database/
52 */
53
54/**
55 * oci8 Database Adapter Class
56 *
57 * This is a modification of the DB_driver class to
58 * permit access to oracle databases
59 *
Derek Jones4b9c6292011-07-01 17:40:48 -050060 * @author Kelly McArdle
Derek Allard2067d1a2008-11-13 22:59:24 +000061 */
Derek Allard2067d1a2008-11-13 22:59:24 +000062class CI_DB_oci8_driver extends CI_DB {
63
Andrey Andreeva24e52e2012-11-02 03:54:12 +020064 /**
65 * Database driver
66 *
67 * @var string
68 */
Andrey Andreev24abcb92012-01-05 20:40:15 +020069 public $dbdriver = 'oci8';
Barry Mienydd671972010-10-04 16:33:58 +020070
Andrey Andreeva24e52e2012-11-02 03:54:12 +020071 /**
72 * Statement ID
73 *
74 * @var resource
75 */
76 public $stmt_id;
Barry Mienydd671972010-10-04 16:33:58 +020077
Derek Allard2067d1a2008-11-13 22:59:24 +000078 /**
Andrey Andreeva24e52e2012-11-02 03:54:12 +020079 * Cursor ID
80 *
81 * @var resource
Derek Allard2067d1a2008-11-13 22:59:24 +000082 */
Andrey Andreev24abcb92012-01-05 20:40:15 +020083 public $curs_id;
Derek Allard2067d1a2008-11-13 22:59:24 +000084
Andrey Andreeva24e52e2012-11-02 03:54:12 +020085 /**
86 * Commit mode flag
87 *
88 * @var int
89 */
90 public $commit_mode = OCI_COMMIT_ON_SUCCESS;
Derek Allard2067d1a2008-11-13 22:59:24 +000091
Andrey Andreev5fd3ae82012-10-24 14:55:35 +030092 /**
Andrey Andreeva24e52e2012-11-02 03:54:12 +020093 * Limit used flag
94 *
95 * If we use LIMIT, we'll add a field that will
96 * throw off num_fields later.
97 *
98 * @var bool
99 */
100 public $limit_used;
101
102 // --------------------------------------------------------------------
103
104 /**
105 * List of reserved identifiers
106 *
107 * Identifiers that must NOT be escaped.
108 *
109 * @var string[]
110 */
111 protected $_reserved_identifiers = array('*', 'rownum');
112
113 /**
114 * ORDER BY random keyword
115 *
Andrey Andreev98e46cf2012-11-13 03:01:42 +0200116 * @var array
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200117 */
Andrey Andreev98e46cf2012-11-13 03:01:42 +0200118 protected $_random_keyword = array('ASC', 'ASC'); // not currently supported
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200119
120 /**
121 * COUNT string
122 *
123 * @used-by CI_DB_driver::count_all()
124 * @used-by CI_DB_query_builder::count_all_results()
125 *
126 * @var string
127 */
128 protected $_count_string = 'SELECT COUNT(1) AS ';
129
130 // --------------------------------------------------------------------
131
132 /**
133 * Class constructor
Andrey Andreev5fd3ae82012-10-24 14:55:35 +0300134 *
135 * @param array $params
136 * @return void
137 */
Andrey Andreevdad61c22012-02-13 01:08:06 +0200138 public function __construct($params)
139 {
140 parent::__construct($params);
141
142 $valid_dsns = array(
Andrey Andreev1a61ba42014-03-24 16:53:16 +0200143 'tns' => '/^\(DESCRIPTION=(\(.+\)){2,}\)$/', // TNS
144 // Easy Connect string (Oracle 10g+)
145 'ec' => '/^(\/\/)?[a-z0-9.:_-]+(:[1-9][0-9]{0,4})?(\/[a-z0-9$_]+)?(:[^\/])?(\/[a-z0-9$_]+)?$/i',
146 'in' => '/^[a-z0-9$_]+$/i' // Instance name (defined in tnsnames.ora)
147 );
Andrey Andreevdad61c22012-02-13 01:08:06 +0200148
149 /* Space characters don't have any effect when actually
150 * connecting, but can be a hassle while validating the DSN.
151 */
152 $this->dsn = str_replace(array("\n", "\r", "\t", ' '), '', $this->dsn);
153
154 if ($this->dsn !== '')
155 {
156 foreach ($valid_dsns as $regexp)
157 {
158 if (preg_match($regexp, $this->dsn))
159 {
160 return;
161 }
162 }
163 }
164
165 // Legacy support for TNS in the hostname configuration field
166 $this->hostname = str_replace(array("\n", "\r", "\t", ' '), '', $this->hostname);
167 if (preg_match($valid_dsns['tns'], $this->hostname))
168 {
169 $this->dsn = $this->hostname;
170 return;
171 }
172 elseif ($this->hostname !== '' && strpos($this->hostname, '/') === FALSE && strpos($this->hostname, ':') === FALSE
173 && (( ! empty($this->port) && ctype_digit($this->port)) OR $this->database !== ''))
174 {
175 /* If the hostname field isn't empty, doesn't contain
176 * ':' and/or '/' and if port and/or database aren't
177 * empty, then the hostname field is most likely indeed
178 * just a hostname. Therefore we'll try and build an
179 * Easy Connect string from these 3 settings, assuming
180 * that the database field is a service name.
181 */
182 $this->dsn = $this->hostname
Andrey Andreev1a61ba42014-03-24 16:53:16 +0200183 .(( ! empty($this->port) && ctype_digit($this->port)) ? ':'.$this->port : '')
184 .($this->database !== '' ? '/'.ltrim($this->database, '/') : '');
Andrey Andreevdad61c22012-02-13 01:08:06 +0200185
186 if (preg_match($valid_dsns['ec'], $this->dsn))
187 {
188 return;
189 }
190 }
191
192 /* At this point, we can only try and validate the hostname and
193 * database fields separately as DSNs.
194 */
195 if (preg_match($valid_dsns['ec'], $this->hostname) OR preg_match($valid_dsns['in'], $this->hostname))
196 {
197 $this->dsn = $this->hostname;
198 return;
199 }
200
201 $this->database = str_replace(array("\n", "\r", "\t", ' '), '', $this->database);
202 foreach ($valid_dsns as $regexp)
203 {
204 if (preg_match($regexp, $this->database))
205 {
206 return;
207 }
208 }
209
210 /* Well - OK, an empty string should work as well.
211 * PHP will try to use environment variables to
212 * determine which Oracle instance to connect to.
213 */
214 $this->dsn = '';
215 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000216
Andrey Andreevd5809992012-06-28 14:06:54 +0300217 // --------------------------------------------------------------------
218
Derek Allard2067d1a2008-11-13 22:59:24 +0000219 /**
220 * Non-persistent database connection
221 *
Andrey Andreev2e171022014-02-25 15:21:41 +0200222 * @param bool $persistent
Andrey Andreevaa786c92012-01-16 12:14:45 +0200223 * @return resource
Derek Allard2067d1a2008-11-13 22:59:24 +0000224 */
Andrey Andreev2e171022014-02-25 15:21:41 +0200225 public function db_connect($persistent = FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000226 {
Andrey Andreev2e171022014-02-25 15:21:41 +0200227 $func = ($persistent === TRUE) ? 'oci_pconnect' : 'oci_connect';
Andrey Andreevd5809992012-06-28 14:06:54 +0300228 return empty($this->char_set)
Andrey Andreev2e171022014-02-25 15:21:41 +0200229 ? $func($this->username, $this->password, $this->dsn)
230 : $func($this->username, $this->password, $this->dsn, $this->char_set);
Derek Allard2067d1a2008-11-13 22:59:24 +0000231 }
232
233 // --------------------------------------------------------------------
234
235 /**
Andrey Andreev08856b82012-03-03 03:19:28 +0200236 * Database version number
Derek Allard2067d1a2008-11-13 22:59:24 +0000237 *
Andrey Andreev08856b82012-03-03 03:19:28 +0200238 * @return string
Derek Allard2067d1a2008-11-13 22:59:24 +0000239 */
Andrey Andreev08856b82012-03-03 03:19:28 +0200240 public function version()
Derek Allard2067d1a2008-11-13 22:59:24 +0000241 {
Andrey Andreev2b730372012-11-05 17:01:11 +0200242 if (isset($this->data_cache['version']))
243 {
244 return $this->data_cache['version'];
245 }
Andrey Andreev2b730372012-11-05 17:01:11 +0200246
247 if ( ! $this->conn_id OR ($version = oci_server_version($this->conn_id)) === FALSE)
248 {
249 return FALSE;
250 }
251
252 return $this->data_cache['version'] = $version;
Derek Allard2067d1a2008-11-13 22:59:24 +0000253 }
254
255 // --------------------------------------------------------------------
256
257 /**
258 * Execute the query
259 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200260 * @param string $sql an SQL query
Andrey Andreevaa786c92012-01-16 12:14:45 +0200261 * @return resource
Derek Allard2067d1a2008-11-13 22:59:24 +0000262 */
Andrey Andreevbc95e472011-10-20 09:44:48 +0300263 protected function _execute($sql)
Derek Allard2067d1a2008-11-13 22:59:24 +0000264 {
Andrey Andreev24abcb92012-01-05 20:40:15 +0200265 /* Oracle must parse the query before it is run. All of the actions with
266 * the query are based on the statement id returned by oci_parse().
267 */
Derek Allard2067d1a2008-11-13 22:59:24 +0000268 $this->stmt_id = FALSE;
269 $this->_set_stmt_id($sql);
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300270 oci_set_prefetch($this->stmt_id, 1000);
Andrey Andreev2bbbd1a2014-05-09 10:24:14 +0300271 return oci_execute($this->stmt_id, $this->commit_mode);
Derek Allard2067d1a2008-11-13 22:59:24 +0000272 }
273
Andrey Andreevd5809992012-06-28 14:06:54 +0300274 // --------------------------------------------------------------------
275
Derek Allard2067d1a2008-11-13 22:59:24 +0000276 /**
277 * Generate a statement ID
278 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200279 * @param string $sql an SQL query
Andrey Andreevaa786c92012-01-16 12:14:45 +0200280 * @return void
Derek Allard2067d1a2008-11-13 22:59:24 +0000281 */
Andrey Andreevda123732012-03-20 22:27:40 +0200282 protected function _set_stmt_id($sql)
Derek Allard2067d1a2008-11-13 22:59:24 +0000283 {
284 if ( ! is_resource($this->stmt_id))
285 {
Timothy Warrend2ff0bc2012-03-19 16:52:10 -0400286 $this->stmt_id = oci_parse($this->conn_id, $sql);
Derek Allard2067d1a2008-11-13 22:59:24 +0000287 }
288 }
289
290 // --------------------------------------------------------------------
291
292 /**
Andrey Andreevda123732012-03-20 22:27:40 +0200293 * Get cursor. Returns a cursor from the database
Derek Allard2067d1a2008-11-13 22:59:24 +0000294 *
Andrey Andreevd5809992012-06-28 14:06:54 +0300295 * @return resource
Derek Allard2067d1a2008-11-13 22:59:24 +0000296 */
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300297 public function get_cursor()
Derek Allard2067d1a2008-11-13 22:59:24 +0000298 {
Andrey Andreevaa786c92012-01-16 12:14:45 +0200299 return $this->curs_id = oci_new_cursor($this->conn_id);
Derek Allard2067d1a2008-11-13 22:59:24 +0000300 }
301
302 // --------------------------------------------------------------------
303
304 /**
Derek Jones4b9c6292011-07-01 17:40:48 -0500305 * Stored Procedure. Executes a stored procedure
Derek Allard2067d1a2008-11-13 22:59:24 +0000306 *
Andrey Andreevaa786c92012-01-16 12:14:45 +0200307 * @param string package name in which the stored procedure is in
308 * @param string stored procedure name to execute
309 * @param array parameters
310 * @return mixed
Derek Allard2067d1a2008-11-13 22:59:24 +0000311 *
312 * params array keys
313 *
Derek Jones4b9c6292011-07-01 17:40:48 -0500314 * KEY OPTIONAL NOTES
Andrey Andreevaa786c92012-01-16 12:14:45 +0200315 * name no the name of the parameter should be in :<param_name> format
316 * value no the value of the parameter. If this is an OUT or IN OUT parameter,
317 * this should be a reference to a variable
318 * type yes the type of the parameter
319 * length yes the max size of the parameter
Derek Allard2067d1a2008-11-13 22:59:24 +0000320 */
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300321 public function stored_procedure($package, $procedure, $params)
Derek Allard2067d1a2008-11-13 22:59:24 +0000322 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100323 if ($package === '' OR $procedure === '' OR ! is_array($params))
Derek Allard2067d1a2008-11-13 22:59:24 +0000324 {
Andrey Andreeve18de502013-07-17 19:59:20 +0300325 log_message('error', 'Invalid query: '.$package.'.'.$procedure);
326 return ($this->db_debug) ? $this->display_error('db_invalid_query') : FALSE;
Derek Allard2067d1a2008-11-13 22:59:24 +0000327 }
Barry Mienydd671972010-10-04 16:33:58 +0200328
Derek Allard2067d1a2008-11-13 22:59:24 +0000329 // build the query string
Andrey Andreeve35d7782012-01-19 15:56:20 +0200330 $sql = 'BEGIN '.$package.'.'.$procedure.'(';
Derek Allard2067d1a2008-11-13 22:59:24 +0000331
332 $have_cursor = FALSE;
Pascal Krietec3a4a8d2011-02-14 13:40:08 -0500333 foreach ($params as $param)
Derek Allard2067d1a2008-11-13 22:59:24 +0000334 {
Andrey Andreevaa786c92012-01-16 12:14:45 +0200335 $sql .= $param['name'].',';
Barry Mienydd671972010-10-04 16:33:58 +0200336
Andrey Andreev85facfa2012-01-26 14:12:14 +0200337 if (isset($param['type']) && $param['type'] === OCI_B_CURSOR)
Derek Allard2067d1a2008-11-13 22:59:24 +0000338 {
339 $have_cursor = TRUE;
340 }
341 }
Andrey Andreev3ca060a2013-11-27 16:30:31 +0200342 $sql = trim($sql, ',').'); END;';
Barry Mienydd671972010-10-04 16:33:58 +0200343
Derek Allard2067d1a2008-11-13 22:59:24 +0000344 $this->stmt_id = FALSE;
345 $this->_set_stmt_id($sql);
346 $this->_bind_params($params);
Andrey Andreevaa786c92012-01-16 12:14:45 +0200347 return $this->query($sql, FALSE, $have_cursor);
Derek Allard2067d1a2008-11-13 22:59:24 +0000348 }
Barry Mienydd671972010-10-04 16:33:58 +0200349
Derek Allard2067d1a2008-11-13 22:59:24 +0000350 // --------------------------------------------------------------------
351
352 /**
353 * Bind parameters
354 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200355 * @param array $params
Andrey Andreevaa786c92012-01-16 12:14:45 +0200356 * @return void
Derek Allard2067d1a2008-11-13 22:59:24 +0000357 */
Andrey Andreev1b5a8562012-03-13 13:13:43 +0200358 protected function _bind_params($params)
Derek Allard2067d1a2008-11-13 22:59:24 +0000359 {
360 if ( ! is_array($params) OR ! is_resource($this->stmt_id))
361 {
362 return;
363 }
Barry Mienydd671972010-10-04 16:33:58 +0200364
Derek Allard2067d1a2008-11-13 22:59:24 +0000365 foreach ($params as $param)
366 {
Barry Mienydd671972010-10-04 16:33:58 +0200367 foreach (array('name', 'value', 'type', 'length') as $val)
Derek Allard2067d1a2008-11-13 22:59:24 +0000368 {
369 if ( ! isset($param[$val]))
370 {
371 $param[$val] = '';
372 }
373 }
374
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300375 oci_bind_by_name($this->stmt_id, $param['name'], $param['value'], $param['length'], $param['type']);
Derek Allard2067d1a2008-11-13 22:59:24 +0000376 }
377 }
378
379 // --------------------------------------------------------------------
380
381 /**
382 * Begin Transaction
383 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200384 * @param bool $test_mode
Barry Mienydd671972010-10-04 16:33:58 +0200385 * @return bool
386 */
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300387 public function trans_begin($test_mode = FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000388 {
389 if ( ! $this->trans_enabled)
390 {
391 return TRUE;
392 }
Barry Mienydd671972010-10-04 16:33:58 +0200393
Derek Allard2067d1a2008-11-13 22:59:24 +0000394 // When transactions are nested we only begin/commit/rollback the outermost ones
395 if ($this->_trans_depth > 0)
396 {
397 return TRUE;
398 }
Barry Mienydd671972010-10-04 16:33:58 +0200399
Derek Allard2067d1a2008-11-13 22:59:24 +0000400 // Reset the transaction failure flag.
401 // If the $test_mode flag is set to TRUE transactions will be rolled back
402 // even if the queries produce a successful result.
Andrey Andreev1a61ba42014-03-24 16:53:16 +0200403 $this->_trans_failure = ($test_mode === TRUE);
Barry Mienydd671972010-10-04 16:33:58 +0200404
Andrey Andreev1a61ba42014-03-24 16:53:16 +0200405 $this->commit_mode = is_php('5.3.2') ? OCI_NO_AUTO_COMMIT : OCI_DEFAULT;
Derek Allard2067d1a2008-11-13 22:59:24 +0000406 return TRUE;
407 }
408
409 // --------------------------------------------------------------------
410
411 /**
412 * Commit Transaction
413 *
Barry Mienydd671972010-10-04 16:33:58 +0200414 * @return bool
415 */
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300416 public function trans_commit()
Derek Allard2067d1a2008-11-13 22:59:24 +0000417 {
418 if ( ! $this->trans_enabled)
419 {
420 return TRUE;
421 }
422
423 // When transactions are nested we only begin/commit/rollback the outermost ones
424 if ($this->_trans_depth > 0)
425 {
426 return TRUE;
427 }
428
Andrey Andreev99013ed2012-03-05 16:17:32 +0200429 $this->commit_mode = OCI_COMMIT_ON_SUCCESS;
Andrey Andreev24abcb92012-01-05 20:40:15 +0200430 return oci_commit($this->conn_id);
Derek Allard2067d1a2008-11-13 22:59:24 +0000431 }
432
433 // --------------------------------------------------------------------
434
435 /**
436 * Rollback Transaction
437 *
Barry Mienydd671972010-10-04 16:33:58 +0200438 * @return bool
439 */
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300440 public function trans_rollback()
Derek Allard2067d1a2008-11-13 22:59:24 +0000441 {
Derek Allard2067d1a2008-11-13 22:59:24 +0000442 // When transactions are nested we only begin/commit/rollback the outermost ones
Andrey Andreeve35d7782012-01-19 15:56:20 +0200443 if ( ! $this->trans_enabled OR $this->_trans_depth > 0)
Derek Allard2067d1a2008-11-13 22:59:24 +0000444 {
445 return TRUE;
446 }
447
Andrey Andreev99013ed2012-03-05 16:17:32 +0200448 $this->commit_mode = OCI_COMMIT_ON_SUCCESS;
Andrey Andreev24abcb92012-01-05 20:40:15 +0200449 return oci_rollback($this->conn_id);
Derek Allard2067d1a2008-11-13 22:59:24 +0000450 }
451
452 // --------------------------------------------------------------------
453
454 /**
Derek Allard2067d1a2008-11-13 22:59:24 +0000455 * Affected Rows
456 *
Andrey Andreevaa786c92012-01-16 12:14:45 +0200457 * @return int
Derek Allard2067d1a2008-11-13 22:59:24 +0000458 */
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300459 public function affected_rows()
Derek Allard2067d1a2008-11-13 22:59:24 +0000460 {
Andrey Andreev2bbbd1a2014-05-09 10:24:14 +0300461 return oci_num_rows($this->stmt_id);
Derek Allard2067d1a2008-11-13 22:59:24 +0000462 }
463
464 // --------------------------------------------------------------------
465
466 /**
467 * Insert ID
468 *
Andrey Andreevaa786c92012-01-16 12:14:45 +0200469 * @return int
Derek Allard2067d1a2008-11-13 22:59:24 +0000470 */
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300471 public function insert_id()
Derek Allard2067d1a2008-11-13 22:59:24 +0000472 {
473 // not supported in oracle
Derek Allardfac8fbc2010-02-05 16:14:49 +0000474 return $this->display_error('db_unsupported_function');
Derek Allard2067d1a2008-11-13 22:59:24 +0000475 }
476
477 // --------------------------------------------------------------------
478
479 /**
Derek Allard2067d1a2008-11-13 22:59:24 +0000480 * Show table query
481 *
482 * Generates a platform-specific query string so that the table names can be fetched
483 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200484 * @param bool $prefix_limit
Andrey Andreev5c3a2022011-10-07 21:04:58 +0300485 * @return string
Derek Allard2067d1a2008-11-13 22:59:24 +0000486 */
Andrey Andreevbc95e472011-10-20 09:44:48 +0300487 protected function _list_tables($prefix_limit = FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000488 {
Andrey Andreevd3f13672012-06-24 22:13:21 +0300489 $sql = 'SELECT "TABLE_NAME" FROM "ALL_TABLES"';
Derek Allard2067d1a2008-11-13 22:59:24 +0000490
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100491 if ($prefix_limit !== FALSE && $this->dbprefix !== '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000492 {
Andrey Andreevd3f13672012-06-24 22:13:21 +0300493 return $sql.' WHERE "TABLE_NAME" LIKE \''.$this->escape_like_str($this->dbprefix)."%' "
494 .sprintf($this->_like_escape_str, $this->_like_escape_chr);
Derek Allard2067d1a2008-11-13 22:59:24 +0000495 }
Barry Mienydd671972010-10-04 16:33:58 +0200496
Derek Allard2067d1a2008-11-13 22:59:24 +0000497 return $sql;
498 }
499
500 // --------------------------------------------------------------------
501
502 /**
503 * Show column query
504 *
505 * Generates a platform-specific query string so that the column names can be fetched
506 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200507 * @param string $table
Andrey Andreevaa786c92012-01-16 12:14:45 +0200508 * @return string
Derek Allard2067d1a2008-11-13 22:59:24 +0000509 */
Andrey Andreevbc95e472011-10-20 09:44:48 +0300510 protected function _list_columns($table = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000511 {
Andrey Andreeve1580572012-11-16 16:17:54 +0200512 if (strpos($table, '.') !== FALSE)
513 {
514 sscanf($table, '%[^.].%s', $owner, $table);
515 }
516 else
517 {
518 $owner = $this->username;
519 }
520
521 return 'SELECT COLUMN_NAME FROM ALL_TAB_COLUMNS
522 WHERE UPPER(OWNER) = '.$this->escape(strtoupper($owner)).'
523 AND UPPER(TABLE_NAME) = '.$this->escape(strtoupper($table));
Derek Allard2067d1a2008-11-13 22:59:24 +0000524 }
525
526 // --------------------------------------------------------------------
527
528 /**
Andrey Andreeve1580572012-11-16 16:17:54 +0200529 * Returns an object with field data
Derek Allard2067d1a2008-11-13 22:59:24 +0000530 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200531 * @param string $table
Andrey Andreeve1580572012-11-16 16:17:54 +0200532 * @return array
Derek Allard2067d1a2008-11-13 22:59:24 +0000533 */
Andrey Andreev5350f052015-01-12 12:33:37 +0200534 public function field_data($table)
Derek Allard2067d1a2008-11-13 22:59:24 +0000535 {
Andrey Andreev5350f052015-01-12 12:33:37 +0200536 if (strpos($table, '.') !== FALSE)
Andrey Andreeve1580572012-11-16 16:17:54 +0200537 {
538 sscanf($table, '%[^.].%s', $owner, $table);
539 }
540 else
541 {
542 $owner = $this->username;
543 }
544
545 $sql = 'SELECT COLUMN_NAME, DATA_TYPE, CHAR_LENGTH, DATA_PRECISION, DATA_LENGTH, DATA_DEFAULT, NULLABLE
546 FROM ALL_TAB_COLUMNS
547 WHERE UPPER(OWNER) = '.$this->escape(strtoupper($owner)).'
548 AND UPPER(TABLE_NAME) = '.$this->escape(strtoupper($table));
549
550 if (($query = $this->query($sql)) === FALSE)
551 {
552 return FALSE;
553 }
554 $query = $query->result_object();
555
556 $retval = array();
557 for ($i = 0, $c = count($query); $i < $c; $i++)
558 {
559 $retval[$i] = new stdClass();
560 $retval[$i]->name = $query[$i]->COLUMN_NAME;
561 $retval[$i]->type = $query[$i]->DATA_TYPE;
562
563 $length = ($query[$i]->CHAR_LENGTH > 0)
564 ? $query[$i]->CHAR_LENGTH : $query[$i]->DATA_PRECISION;
565 if ($length === NULL)
566 {
567 $length = $query[$i]->DATA_LENGTH;
568 }
569 $retval[$i]->max_length = $length;
570
571 $default = $query[$i]->DATA_DEFAULT;
572 if ($default === NULL && $query[$i]->NULLABLE === 'N')
573 {
574 $default = '';
575 }
Leandro Mangini Antunes4c08ea92015-05-13 11:15:01 -0300576 $retval[$i]->default = $default;
Andrey Andreeve1580572012-11-16 16:17:54 +0200577 }
578
579 return $retval;
Derek Allard2067d1a2008-11-13 22:59:24 +0000580 }
581
582 // --------------------------------------------------------------------
583
584 /**
Andrey Andreev4be5de12012-03-02 15:45:41 +0200585 * Error
Derek Allard2067d1a2008-11-13 22:59:24 +0000586 *
Andrey Andreev4be5de12012-03-02 15:45:41 +0200587 * Returns an array containing code and message of the last
588 * database error that has occured.
Derek Allard2067d1a2008-11-13 22:59:24 +0000589 *
Andrey Andreev601f8b22012-03-01 20:11:15 +0200590 * @return array
Derek Allard2067d1a2008-11-13 22:59:24 +0000591 */
Andrey Andreev4be5de12012-03-02 15:45:41 +0200592 public function error()
Derek Allard2067d1a2008-11-13 22:59:24 +0000593 {
Andrey Andreev4be5de12012-03-02 15:45:41 +0200594 /* oci_error() returns an array that already contains the
595 * 'code' and 'message' keys, so we can just return it.
596 */
Andrey Andreev601f8b22012-03-01 20:11:15 +0200597 if (is_resource($this->curs_id))
598 {
599 return oci_error($this->curs_id);
600 }
601 elseif (is_resource($this->stmt_id))
602 {
603 return oci_error($this->stmt_id);
604 }
605 elseif (is_resource($this->conn_id))
606 {
607 return oci_error($this->conn_id);
608 }
609
610 return oci_error();
Derek Allard2067d1a2008-11-13 22:59:24 +0000611 }
Barry Mienydd671972010-10-04 16:33:58 +0200612
Derek Allard2067d1a2008-11-13 22:59:24 +0000613 // --------------------------------------------------------------------
614
615 /**
Andrey Andreev083e3c82012-11-06 12:48:32 +0200616 * Insert batch statement
Andrey Andreev99c6dd42011-09-23 03:07:01 +0300617 *
618 * Generates a platform-specific insert string from the supplied data
619 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200620 * @param string $table Table name
621 * @param array $keys INSERT keys
622 * @param array $values INSERT values
Andrey Andreevda123732012-03-20 22:27:40 +0200623 * @return string
Andrey Andreev99c6dd42011-09-23 03:07:01 +0300624 */
Andrey Andreevbc95e472011-10-20 09:44:48 +0300625 protected function _insert_batch($table, $keys, $values)
Andrey Andreev99c6dd42011-09-23 03:07:01 +0300626 {
627 $keys = implode(', ', $keys);
628 $sql = "INSERT ALL\n";
629
630 for ($i = 0, $c = count($values); $i < $c; $i++)
Andrey Andreevb83c4082011-09-23 03:32:45 +0300631 {
Andrey Andreev97f36972012-04-05 12:44:36 +0300632 $sql .= ' INTO '.$table.' ('.$keys.') VALUES '.$values[$i]."\n";
Andrey Andreevb83c4082011-09-23 03:32:45 +0300633 }
Andrey Andreev99c6dd42011-09-23 03:07:01 +0300634
Andrey Andreevaa786c92012-01-16 12:14:45 +0200635 return $sql.'SELECT * FROM dual';
Derek Allard2067d1a2008-11-13 22:59:24 +0000636 }
637
638 // --------------------------------------------------------------------
639
640 /**
641 * Truncate statement
642 *
643 * Generates a platform-specific truncate string from the supplied data
Derek Allard2067d1a2008-11-13 22:59:24 +0000644 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200645 * If the database does not support the TRUNCATE statement,
Andrey Andreeva6fe36e2012-04-05 16:00:32 +0300646 * then this method maps to 'DELETE FROM table'
Derek Allard2067d1a2008-11-13 22:59:24 +0000647 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200648 * @param string $table
Derek Allard2067d1a2008-11-13 22:59:24 +0000649 * @return string
Barry Mienydd671972010-10-04 16:33:58 +0200650 */
Andrey Andreevbc95e472011-10-20 09:44:48 +0300651 protected function _truncate($table)
Derek Allard2067d1a2008-11-13 22:59:24 +0000652 {
Andrey Andreevaa786c92012-01-16 12:14:45 +0200653 return 'TRUNCATE TABLE '.$table;
Derek Allard2067d1a2008-11-13 22:59:24 +0000654 }
Barry Mienydd671972010-10-04 16:33:58 +0200655
Derek Allard2067d1a2008-11-13 22:59:24 +0000656 // --------------------------------------------------------------------
657
658 /**
659 * Delete statement
660 *
661 * Generates a platform-specific delete string from the supplied data
662 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200663 * @param string $table
Derek Allard2067d1a2008-11-13 22:59:24 +0000664 * @return string
Barry Mienydd671972010-10-04 16:33:58 +0200665 */
Andrey Andreevb0478652012-07-18 15:34:46 +0300666 protected function _delete($table)
Derek Allard2067d1a2008-11-13 22:59:24 +0000667 {
Andrey Andreevb0478652012-07-18 15:34:46 +0300668 if ($this->qb_limit)
669 {
Andrey Andreevc9b924c2012-07-19 13:06:02 +0300670 $this->where('rownum <= ',$this->qb_limit, FALSE);
Andrey Andreevb0478652012-07-18 15:34:46 +0300671 $this->qb_limit = FALSE;
672 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000673
Andrey Andreevb0478652012-07-18 15:34:46 +0300674 return parent::_delete($table);
Derek Allard2067d1a2008-11-13 22:59:24 +0000675 }
676
677 // --------------------------------------------------------------------
678
679 /**
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200680 * LIMIT
Derek Allard2067d1a2008-11-13 22:59:24 +0000681 *
682 * Generates a platform-specific LIMIT clause
683 *
Andrey Andreeva24e52e2012-11-02 03:54:12 +0200684 * @param string $sql SQL Query
Andrey Andreevaa786c92012-01-16 12:14:45 +0200685 * @return string
Derek Allard2067d1a2008-11-13 22:59:24 +0000686 */
Andrey Andreevc9b924c2012-07-19 13:06:02 +0300687 protected function _limit($sql)
Derek Allard2067d1a2008-11-13 22:59:24 +0000688 {
Derek Allard2067d1a2008-11-13 22:59:24 +0000689 $this->limit_used = TRUE;
Andrey Andreevc9b924c2012-07-19 13:06:02 +0300690 return 'SELECT * FROM (SELECT inner_query.*, rownum rnum FROM ('.$sql.') inner_query WHERE rownum < '.($this->qb_offset + $this->qb_limit + 1).')'
Andrey Andreev1a61ba42014-03-24 16:53:16 +0200691 .($this->qb_offset ? ' WHERE rnum >= '.($this->qb_offset + 1) : '');
Barry Mienydd671972010-10-04 16:33:58 +0200692 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000693
694 // --------------------------------------------------------------------
695
696 /**
697 * Close DB Connection
698 *
Andrey Andreevaa786c92012-01-16 12:14:45 +0200699 * @return void
Derek Allard2067d1a2008-11-13 22:59:24 +0000700 */
Andrey Andreev79922c02012-05-23 12:27:17 +0300701 protected function _close()
Derek Allard2067d1a2008-11-13 22:59:24 +0000702 {
Andrey Andreev2bbbd1a2014-05-09 10:24:14 +0300703 oci_close($this->conn_id);
Derek Allard2067d1a2008-11-13 22:59:24 +0000704 }
705
Derek Allard2067d1a2008-11-13 22:59:24 +0000706}