blob: 34140c6e086efaf72a78db232bd09454e41d63d5 [file] [log] [blame]
Andrey Andreevc5536aa2012-11-01 17:33:58 +02001<?php
Derek Allard2067d1a2008-11-13 22:59:24 +00002/**
Derek Jonesf4a4bd82011-10-20 12:18:42 -05003 * CodeIgniter
Derek Allard2067d1a2008-11-13 22:59:24 +00004 *
Phil Sturgeon07c1ac82012-03-09 17:03:37 +00005 * An open source application development framework for PHP 5.2.4 or newer
Derek Allard2067d1a2008-11-13 22:59:24 +00006 *
Derek Jonesf4a4bd82011-10-20 12:18:42 -05007 * NOTICE OF LICENSE
Andrey Andreev24276a32012-01-08 02:44:38 +02008 *
Derek Jonesf4a4bd82011-10-20 12:18:42 -05009 * Licensed under the Open Software License version 3.0
Andrey Andreev24276a32012-01-08 02:44:38 +020010 *
Derek Jonesf4a4bd82011-10-20 12:18:42 -050011 * This source file is subject to the Open Software License (OSL 3.0) that is
12 * bundled with this package in the files license.txt / license.rst. It is
13 * also available through the world wide web at this URL:
14 * http://opensource.org/licenses/OSL-3.0
15 * If you did not receive a copy of the license and are unable to obtain it
16 * through the world wide web, please send an email to
17 * licensing@ellislab.com so we can send you a copy immediately.
18 *
Derek Allard2067d1a2008-11-13 22:59:24 +000019 * @package CodeIgniter
Derek Jonesf4a4bd82011-10-20 12:18:42 -050020 * @author EllisLab Dev Team
Greg Aker0defe5d2012-01-01 18:46:41 -060021 * @copyright Copyright (c) 2008 - 2012, EllisLab, Inc. (http://ellislab.com/)
Derek Jonesf4a4bd82011-10-20 12:18:42 -050022 * @license http://opensource.org/licenses/OSL-3.0 Open Software License (OSL 3.0)
Derek Allard2067d1a2008-11-13 22:59:24 +000023 * @link http://codeigniter.com
24 * @since Version 1.0
25 * @filesource
26 */
Andrey Andreevc5536aa2012-11-01 17:33:58 +020027defined('BASEPATH') OR exit('No direct script access allowed');
Derek Allard2067d1a2008-11-13 22:59:24 +000028
Derek Allard2067d1a2008-11-13 22:59:24 +000029/**
Andrey Andreevd947eba2012-04-09 14:58:28 +030030 * Database Forge Class
Derek Allard2067d1a2008-11-13 22:59:24 +000031 *
32 * @category Database
Derek Jonesf4a4bd82011-10-20 12:18:42 -050033 * @author EllisLab Dev Team
Derek Allard2067d1a2008-11-13 22:59:24 +000034 * @link http://codeigniter.com/user_guide/database/
35 */
Timothy Warren833d5042012-03-19 16:12:03 -040036abstract class CI_DB_forge {
Derek Allard2067d1a2008-11-13 22:59:24 +000037
Andrey Andreevae85eb42012-11-02 01:42:31 +020038 /**
Andrey Andreeva287a342012-11-05 23:19:59 +020039 * Database object
40 *
41 * @var object
42 */
Andrey Andreeveaa60c72012-11-06 01:11:22 +020043 protected $db;
Andrey Andreeva287a342012-11-05 23:19:59 +020044
45 /**
Andrey Andreevae85eb42012-11-02 01:42:31 +020046 * Fields data
47 *
48 * @var array
49 */
Andrey Andreev24276a32012-01-08 02:44:38 +020050 public $fields = array();
Andrey Andreevae85eb42012-11-02 01:42:31 +020051
52 /**
53 * Keys data
54 *
55 * @var array
56 */
Andrey Andreev24276a32012-01-08 02:44:38 +020057 public $keys = array();
Andrey Andreevae85eb42012-11-02 01:42:31 +020058
59 /**
60 * Primary Keys data
61 *
62 * @var array
63 */
Andrey Andreev24276a32012-01-08 02:44:38 +020064 public $primary_keys = array();
Andrey Andreevae85eb42012-11-02 01:42:31 +020065
66 /**
67 * Database character set
68 *
69 * @var string
70 */
Andrey Andreev5fd3ae82012-10-24 14:55:35 +030071 public $db_char_set = '';
Derek Allard2067d1a2008-11-13 22:59:24 +000072
Andrey Andreevae85eb42012-11-02 01:42:31 +020073 // --------------------------------------------------------------------
74
75 /**
76 * CREATE DATABASE statement
77 *
78 * @var string
79 */
Andrey Andreevd947eba2012-04-09 14:58:28 +030080 protected $_create_database = 'CREATE DATABASE %s';
Andrey Andreevae85eb42012-11-02 01:42:31 +020081
82 /**
83 * DROP DATABASE statement
84 *
85 * @var string
86 */
Andrey Andreevd947eba2012-04-09 14:58:28 +030087 protected $_drop_database = 'DROP DATABASE %s';
Andrey Andreevae85eb42012-11-02 01:42:31 +020088
89 /**
Andrey Andreeva287a342012-11-05 23:19:59 +020090 * CREATE TABLE statement
Andrey Andreevae85eb42012-11-02 01:42:31 +020091 *
92 * @var string
93 */
Andrey Andreeva287a342012-11-05 23:19:59 +020094 protected $_create_table = "%s %s (%s\n)";
95
96 /**
97 * CREATE TABLE IF statement
98 *
99 * @var string
100 */
101 protected $_create_table_if = 'CREATE TABLE IF NOT EXISTS';
102
103 /**
104 * CREATE TABLE keys flag
105 *
106 * Whether table keys are created from within the
107 * CREATE TABLE statement.
108 *
109 * @var bool
110 */
111 protected $_create_table_keys = FALSE;
112
113 /**
114 * DROP TABLE IF EXISTS statement
115 *
116 * @var string
117 */
118 protected $_drop_table_if = 'DROP TABLE IF EXISTS';
Andrey Andreevae85eb42012-11-02 01:42:31 +0200119
120 /**
121 * RENAME TABLE statement
122 *
123 * @var string
124 */
Andrey Andreeva287a342012-11-05 23:19:59 +0200125 protected $_rename_table = 'ALTER TABLE %s RENAME TO %s;';
126
127 /**
128 * UNSIGNED support
129 *
130 * @var bool|array
131 */
132 protected $_unsigned = TRUE;
133
134 /**
135 * NULL value representatin in CREATE/ALTER TABLE statements
136 *
137 * @var string
138 */
139 protected $_null = '';
140
141 /**
142 * DEFAULT value representation in CREATE/ALTER TABLE statements
143 *
144 * @var string
145 */
146 protected $_default = ' DEFAULT ';
Andrey Andreevd947eba2012-04-09 14:58:28 +0300147
Andrey Andreevae85eb42012-11-02 01:42:31 +0200148 // --------------------------------------------------------------------
149
Andrey Andreev5fd3ae82012-10-24 14:55:35 +0300150 /**
Andrey Andreeva287a342012-11-05 23:19:59 +0200151 * Class constructor
Andrey Andreev5fd3ae82012-10-24 14:55:35 +0300152 *
Andrey Andreeveaa60c72012-11-06 01:11:22 +0200153 * @param object &$db Database object
Andrey Andreev5fd3ae82012-10-24 14:55:35 +0300154 * @return void
155 */
Andrey Andreeveaa60c72012-11-06 01:11:22 +0200156 public function __construct(&$db)
Derek Allard2067d1a2008-11-13 22:59:24 +0000157 {
Andrey Andreeveaa60c72012-11-06 01:11:22 +0200158 $this->db =& $db;
Andrey Andreev24276a32012-01-08 02:44:38 +0200159 log_message('debug', 'Database Forge Class Initialized');
Derek Allard2067d1a2008-11-13 22:59:24 +0000160 }
161
162 // --------------------------------------------------------------------
163
164 /**
165 * Create database
166 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200167 * @param string $db_name
Derek Allard2067d1a2008-11-13 22:59:24 +0000168 * @return bool
169 */
Andrey Andreev24276a32012-01-08 02:44:38 +0200170 public function create_database($db_name)
Derek Allard2067d1a2008-11-13 22:59:24 +0000171 {
Andrey Andreevd947eba2012-04-09 14:58:28 +0300172 if ($this->_create_database === FALSE)
173 {
174 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
175 }
176 elseif ( ! $this->db->query(sprintf($this->_create_database, $db_name, $this->db->char_set, $this->db->dbcollat)))
177 {
178 return ($this->db->db_debug) ? $this->db->display_error('db_unable_to_drop') : FALSE;
179 }
180
Andrey Andreev5d281762012-06-11 22:05:40 +0300181 if ( ! empty($this->db->data_cache['db_names']))
182 {
183 $this->db->data_cache['db_names'][] = $db_name;
184 }
185
Andrey Andreevd947eba2012-04-09 14:58:28 +0300186 return TRUE;
Derek Allard2067d1a2008-11-13 22:59:24 +0000187 }
188
189 // --------------------------------------------------------------------
190
191 /**
192 * Drop database
193 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200194 * @param string $db_name
Derek Allard2067d1a2008-11-13 22:59:24 +0000195 * @return bool
196 */
Andrey Andreev24276a32012-01-08 02:44:38 +0200197 public function drop_database($db_name)
Derek Allard2067d1a2008-11-13 22:59:24 +0000198 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100199 if ($db_name === '')
Andrey Andreevd947eba2012-04-09 14:58:28 +0300200 {
201 show_error('A table name is required for that operation.');
202 return FALSE;
203 }
204 elseif ($this->_drop_database === FALSE)
205 {
206 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
207 }
208 elseif ( ! $this->db->query(sprintf($this->_drop_database, $db_name)))
209 {
210 return ($this->db->db_debug) ? $this->db->display_error('db_unable_to_drop') : FALSE;
211 }
212
Andrey Andreev5d281762012-06-11 22:05:40 +0300213 if ( ! empty($this->db->data_cache['db_names']))
214 {
215 $key = array_search(strtolower($db_name), array_map('strtolower', $this->db->data_cache['db_names']), TRUE);
216 if ($key !== FALSE)
217 {
218 unset($this->db->data_cache['db_names'][$key]);
219 }
220 }
221
Andrey Andreevd947eba2012-04-09 14:58:28 +0300222 return TRUE;
Derek Allard2067d1a2008-11-13 22:59:24 +0000223 }
224
225 // --------------------------------------------------------------------
226
227 /**
228 * Add Key
229 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200230 * @param string $key
231 * @param bool $primary
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000232 * @return object
Derek Allard2067d1a2008-11-13 22:59:24 +0000233 */
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000234 public function add_key($key = '', $primary = FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000235 {
Andrey Andreeva287a342012-11-05 23:19:59 +0200236 if (empty($key))
237 {
238 show_error('Key information is required for that operation.');
239 }
240
Andrey Andreevd743cdb2012-11-06 00:00:01 +0200241 if ($primary === TRUE && is_array($key))
Derek Allard2067d1a2008-11-13 22:59:24 +0000242 {
Pascal Krietec3a4a8d2011-02-14 13:40:08 -0500243 foreach ($key as $one)
Derek Allard2067d1a2008-11-13 22:59:24 +0000244 {
245 $this->add_key($one, $primary);
246 }
Barry Mienydd671972010-10-04 16:33:58 +0200247
Andrey Andreeva287a342012-11-05 23:19:59 +0200248 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000249 }
Barry Mienydd671972010-10-04 16:33:58 +0200250
Derek Allard2067d1a2008-11-13 22:59:24 +0000251 if ($primary === TRUE)
252 {
253 $this->primary_keys[] = $key;
254 }
255 else
256 {
257 $this->keys[] = $key;
258 }
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000259
260 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000261 }
262
263 // --------------------------------------------------------------------
264
265 /**
266 * Add Field
267 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200268 * @param array $field
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000269 * @return object
Derek Allard2067d1a2008-11-13 22:59:24 +0000270 */
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000271 public function add_field($field = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000272 {
Andrey Andreeva287a342012-11-05 23:19:59 +0200273 if (empty($field))
Derek Allard2067d1a2008-11-13 22:59:24 +0000274 {
275 show_error('Field information is required.');
276 }
Barry Mienydd671972010-10-04 16:33:58 +0200277
Derek Allard2067d1a2008-11-13 22:59:24 +0000278 if (is_string($field))
279 {
Andrey Andreev24276a32012-01-08 02:44:38 +0200280 if ($field === 'id')
Derek Allard2067d1a2008-11-13 22:59:24 +0000281 {
282 $this->add_field(array(
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000283 'id' => array(
284 'type' => 'INT',
285 'constraint' => 9,
286 'auto_increment' => TRUE
287 )
288 ));
Derek Allard2067d1a2008-11-13 22:59:24 +0000289 $this->add_key('id', TRUE);
290 }
291 else
292 {
293 if (strpos($field, ' ') === FALSE)
294 {
295 show_error('Field information is required for that operation.');
296 }
Barry Mienydd671972010-10-04 16:33:58 +0200297
Derek Allard2067d1a2008-11-13 22:59:24 +0000298 $this->fields[] = $field;
299 }
300 }
Barry Mienydd671972010-10-04 16:33:58 +0200301
Derek Allard2067d1a2008-11-13 22:59:24 +0000302 if (is_array($field))
303 {
304 $this->fields = array_merge($this->fields, $field);
305 }
Andrey Andreev24276a32012-01-08 02:44:38 +0200306
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000307 return $this;
Derek Allard2067d1a2008-11-13 22:59:24 +0000308 }
309
310 // --------------------------------------------------------------------
311
312 /**
313 * Create Table
314 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200315 * @param string $table Table name
316 * @param bool $if_not_exists Whether to add IF NOT EXISTS condition
Derek Allard2067d1a2008-11-13 22:59:24 +0000317 * @return bool
318 */
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000319 public function create_table($table = '', $if_not_exists = FALSE)
Barry Mienydd671972010-10-04 16:33:58 +0200320 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100321 if ($table === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000322 {
323 show_error('A table name is required for that operation.');
324 }
Andrey Andreeva287a342012-11-05 23:19:59 +0200325 else
326 {
327 $table = $this->db->dbprefix.$table;
328 }
Barry Mienydd671972010-10-04 16:33:58 +0200329
Andrey Andreev24276a32012-01-08 02:44:38 +0200330 if (count($this->fields) === 0)
Barry Mienydd671972010-10-04 16:33:58 +0200331 {
Derek Allard2067d1a2008-11-13 22:59:24 +0000332 show_error('Field information is required.');
333 }
334
Andrey Andreeva287a342012-11-05 23:19:59 +0200335 $sql = $this->_create_table($table, $if_not_exists);
Andrey Andreev5d281762012-06-11 22:05:40 +0300336
337 if (is_bool($sql))
338 {
Andrey Andreeva287a342012-11-05 23:19:59 +0200339 $this->_reset();
340 if ($sql === FALSE)
341 {
342 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
343 }
Andrey Andreev5d281762012-06-11 22:05:40 +0300344 }
345
Andrey Andreeva287a342012-11-05 23:19:59 +0200346 if (($result = $this->db->query($sql)) !== FALSE)
Andrey Andreev5d281762012-06-11 22:05:40 +0300347 {
Andrey Andreeva287a342012-11-05 23:19:59 +0200348 empty($this->db->data_cache['table_names']) OR $this->db->data_cache['table_names'][] = $table;
349
350 // Most databases don't support creating indexes from within the CREATE TABLE statement
351 if ( ! empty($this->keys))
352 {
353 for ($i = 0, $sqls = $this->_process_indexes($table), $c = count($sqls); $i < $c; $i++)
354 {
355 $this->db->query($sqls[$i]);
356 }
357 }
Andrey Andreev5d281762012-06-11 22:05:40 +0300358 }
359
Andrey Andreeva287a342012-11-05 23:19:59 +0200360 $this->_reset();
Andrey Andreev5d281762012-06-11 22:05:40 +0300361 return $result;
Derek Allard2067d1a2008-11-13 22:59:24 +0000362 }
363
364 // --------------------------------------------------------------------
365
366 /**
Andrey Andreeva287a342012-11-05 23:19:59 +0200367 * Create Table
368 *
369 * @param string $table Table name
370 * @param bool $if_not_exists Whether to add 'IF NOT EXISTS' condition
371 * @return mixed
372 */
373 protected function _create_table($table, $if_not_exists)
374 {
375 if ($if_not_exists === TRUE && $this->_create_table_if === FALSE)
376 {
377 if ($this->db->table_exists($table))
378 {
379 return TRUE;
380 }
381 else
382 {
383 $if_not_exists = FALSE;
384 }
385 }
386
387 $sql = ($if_not_exists)
388 ? sprintf($this->_create_table_if, $this->db->escape_identifiers($table))
389 : 'CREATE TABLE';
390
391 $columns = $this->_process_fields(TRUE);
392 for ($i = 0, $c = count($columns); $i < $c; $i++)
393 {
394 $columns[$i] = ($columns[$i]['_literal'] !== FALSE)
395 ? "\n\t".$columns[$i]['_literal']
396 : "\n\t".$this->_process_column($columns[$i]);
397 }
398
399 $columns = implode(',', $columns)
400 .$this->_process_primary_keys($table);
401
402 // Are indexes created from within the CREATE TABLE statement? (e.g. in MySQL)
403 if ($this->_create_table_keys === TRUE)
404 {
405 $columns .= $this->_process_indexes();
406 }
407
408 // _create_table will usually have the following format: "%s %s (%s\n)"
409 $sql = sprintf($this->_create_table.';',
410 $sql,
411 $this->db->escape_identifiers($table),
412 $columns
413 );
414
415 return $sql;
416 }
417
418 // --------------------------------------------------------------------
419
420 /**
Derek Allard2067d1a2008-11-13 22:59:24 +0000421 * Drop Table
422 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200423 * @param string $table_name Table name
424 * @param bool $if_exists Whether to add an IF EXISTS condition
Derek Allard2067d1a2008-11-13 22:59:24 +0000425 * @return bool
426 */
Andrey Andreeva287a342012-11-05 23:19:59 +0200427 public function drop_table($table_name, $if_exists = FALSE)
Derek Allard2067d1a2008-11-13 22:59:24 +0000428 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100429 if ($table_name === '')
Andrey Andreevd947eba2012-04-09 14:58:28 +0300430 {
431 return ($this->db->db_debug) ? $this->db->display_error('db_table_name_required') : FALSE;
432 }
Andrey Andreeva287a342012-11-05 23:19:59 +0200433
434 $query = $this->_drop_table($this->db->dbprefix.$table_name, $if_exists);
435 if ($query === FALSE)
Andrey Andreevd947eba2012-04-09 14:58:28 +0300436 {
437 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
438 }
Andrey Andreeva287a342012-11-05 23:19:59 +0200439 elseif ($query === TRUE)
440 {
441 return TRUE;
442 }
Andrey Andreevd947eba2012-04-09 14:58:28 +0300443
Andrey Andreeva287a342012-11-05 23:19:59 +0200444 $query = $this->db->query($query);
Andrey Andreev5d281762012-06-11 22:05:40 +0300445
446 // Update table list cache
Andrey Andreeva287a342012-11-05 23:19:59 +0200447 if ($query && ! empty($this->db->data_cache['table_names']))
Andrey Andreev5d281762012-06-11 22:05:40 +0300448 {
449 $key = array_search(strtolower($this->db->dbprefix.$table_name), array_map('strtolower', $this->db->data_cache['table_names']), TRUE);
450 if ($key !== FALSE)
451 {
452 unset($this->db->data_cache['table_names'][$key]);
453 }
454 }
455
Andrey Andreeva287a342012-11-05 23:19:59 +0200456 return $query;
457 }
458
459 // --------------------------------------------------------------------
460
461 /**
462 * Drop Table
463 *
464 * Generates a platform-specific DROP TABLE string
465 *
466 * @param string $table Table name
467 * @param bool $if_exists Whether to add an IF EXISTS condition
468 * @return string
469 */
470 protected function _drop_table($table, $if_exists)
471 {
472 $sql = 'DROP TABLE';
473
474 if ($if_exists)
475 {
476 if ($this->_drop_table_if === FALSE)
477 {
478 if ( ! $this->db->table_exists($table))
479 {
480 return TRUE;
481 }
482 }
483 else
484 {
485 $sql = sprintf($this->_drop_table_if, $this->db->escape_identifiers($table));
486 }
487 }
488
489 return $sql.' '.$this->db->escape_identifiers($table);
Derek Allard2067d1a2008-11-13 22:59:24 +0000490 }
491
492 // --------------------------------------------------------------------
493
494 /**
495 * Rename Table
496 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200497 * @param string $table_name Old table name
498 * @param string $new_table_name New table name
Derek Allard2067d1a2008-11-13 22:59:24 +0000499 * @return bool
500 */
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000501 public function rename_table($table_name, $new_table_name)
Derek Allard2067d1a2008-11-13 22:59:24 +0000502 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100503 if ($table_name === '' OR $new_table_name === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000504 {
505 show_error('A table name is required for that operation.');
Andrey Andreevd947eba2012-04-09 14:58:28 +0300506 return FALSE;
507 }
508 elseif ($this->_rename_table === FALSE)
509 {
510 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
Derek Allard2067d1a2008-11-13 22:59:24 +0000511 }
Barry Mienydd671972010-10-04 16:33:58 +0200512
Andrey Andreev5d281762012-06-11 22:05:40 +0300513 $result = $this->db->query(sprintf($this->_rename_table,
Andrey Andreevd947eba2012-04-09 14:58:28 +0300514 $this->db->escape_identifiers($this->db->dbprefix.$table_name),
515 $this->db->escape_identifiers($this->db->dbprefix.$new_table_name))
516 );
Andrey Andreev5d281762012-06-11 22:05:40 +0300517
518 if ($result && ! empty($this->db->data_cache['table_names']))
519 {
520 $key = array_search(strtolower($this->db->dbprefix.$table_name), array_map('strtolower', $this->db->data_cache['table_names']), TRUE);
521 if ($key !== FALSE)
522 {
523 $this->db->data_cache['table_names'][$key] = $this->db->dbprefix.$new_table_name;
524 }
525 }
526
527 return $result;
Derek Allard2067d1a2008-11-13 22:59:24 +0000528 }
529
530 // --------------------------------------------------------------------
531
532 /**
533 * Column Add
534 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200535 * @param string $table Table name
536 * @param array $field Column definition
Derek Allard2067d1a2008-11-13 22:59:24 +0000537 * @return bool
538 */
Andrey Andreeva287a342012-11-05 23:19:59 +0200539 public function add_column($table = '', $field = array())
Derek Allard2067d1a2008-11-13 22:59:24 +0000540 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100541 if ($table === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000542 {
543 show_error('A table name is required for that operation.');
544 }
545
Andrey Andreeva287a342012-11-05 23:19:59 +0200546 // Work-around for literal column definitions
547 if ( ! is_array($field))
548 {
549 $field = array($field);
550 }
551
Andrey Andreev24276a32012-01-08 02:44:38 +0200552 foreach (array_keys($field) as $k)
Barry Mienydd671972010-10-04 16:33:58 +0200553 {
554 $this->add_field(array($k => $field[$k]));
Andrey Andreeva287a342012-11-05 23:19:59 +0200555 }
Robin Sowell8a54ef22009-03-04 14:49:53 +0000556
Andrey Andreeva287a342012-11-05 23:19:59 +0200557 $sqls = $this->_alter_table('ADD', $this->db->dbprefix.$table, $this->_process_fields());
558 $this->_reset();
559 if ($sqls === FALSE)
560 {
561 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
562 }
Barry Mienydd671972010-10-04 16:33:58 +0200563
Andrey Andreeva287a342012-11-05 23:19:59 +0200564 for ($i = 0, $c = count($sqls); $i < $c; $i++)
565 {
Andrey Andreev137a7422012-11-05 23:46:44 +0200566 if ($this->db->query($sqls[$i]) === FALSE)
Robin Sowell8a54ef22009-03-04 14:49:53 +0000567 {
568 return FALSE;
569 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000570 }
Barry Mienydd671972010-10-04 16:33:58 +0200571
Robin Sowell8a54ef22009-03-04 14:49:53 +0000572 return TRUE;
Derek Allard2067d1a2008-11-13 22:59:24 +0000573 }
574
575 // --------------------------------------------------------------------
576
577 /**
578 * Column Drop
579 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200580 * @param string $table Table name
581 * @param string $column_name Column name
Derek Allard2067d1a2008-11-13 22:59:24 +0000582 * @return bool
583 */
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000584 public function drop_column($table = '', $column_name = '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000585 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100586 if ($table === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000587 {
588 show_error('A table name is required for that operation.');
589 }
590
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100591 if ($column_name === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000592 {
593 show_error('A column name is required for that operation.');
594 }
595
Andrey Andreeva287a342012-11-05 23:19:59 +0200596 $sql = $this->_alter_table('DROP', $this->db->dbprefix.$table, $column_name);
597 if ($sql === FALSE)
598 {
599 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
600 }
601
602 return $this->db->query($sql);
Derek Allard2067d1a2008-11-13 22:59:24 +0000603 }
604
605 // --------------------------------------------------------------------
606
607 /**
608 * Column Modify
609 *
Andrey Andreeva287a342012-11-05 23:19:59 +0200610 * @param string $table Table name
611 * @param string $field Column definition
Derek Allard2067d1a2008-11-13 22:59:24 +0000612 * @return bool
613 */
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000614 public function modify_column($table = '', $field = array())
Derek Allard2067d1a2008-11-13 22:59:24 +0000615 {
Alex Bilbie48a2baf2012-06-02 11:09:54 +0100616 if ($table === '')
Derek Allard2067d1a2008-11-13 22:59:24 +0000617 {
618 show_error('A table name is required for that operation.');
619 }
620
Andrey Andreeva287a342012-11-05 23:19:59 +0200621 // Work-around for literal column definitions
622 if ( ! is_array($field))
623 {
624 $field = array($field);
625 }
626
Andrey Andreev24276a32012-01-08 02:44:38 +0200627 foreach (array_keys($field) as $k)
Robin Sowell8a54ef22009-03-04 14:49:53 +0000628 {
629 $this->add_field(array($k => $field[$k]));
Andrey Andreeva287a342012-11-05 23:19:59 +0200630 }
Barry Mienydd671972010-10-04 16:33:58 +0200631
Andrey Andreeva287a342012-11-05 23:19:59 +0200632 if (count($this->fields) === 0)
633 {
634 show_error('Field information is required.');
635 }
Barry Mienydd671972010-10-04 16:33:58 +0200636
Andrey Andreeva287a342012-11-05 23:19:59 +0200637 $sqls = $this->_alter_table('CHANGE', $this->db->dbprefix.$table, $this->fields);
638 $this->_reset();
639 if ($sqls === FALSE)
640 {
641 return ($this->db->db_debug) ? $this->db->display_error('db_unsuported_feature') : FALSE;
642 }
643
644 for ($i = 0, $c = count($sqls); $i < $c; $i++)
645 {
Andrey Andreev137a7422012-11-05 23:46:44 +0200646 if ($this->db->query($sqls[$i]) === FALSE)
Robin Sowell8a54ef22009-03-04 14:49:53 +0000647 {
648 return FALSE;
649 }
Derek Allard2067d1a2008-11-13 22:59:24 +0000650 }
Barry Mienydd671972010-10-04 16:33:58 +0200651
Robin Sowell8a54ef22009-03-04 14:49:53 +0000652 return TRUE;
Derek Allard2067d1a2008-11-13 22:59:24 +0000653 }
654
655 // --------------------------------------------------------------------
656
657 /**
Andrey Andreeva287a342012-11-05 23:19:59 +0200658 * ALTER TABLE
659 *
660 * @param string $alter_type ALTER type
661 * @param string $table Table name
662 * @param mixed $field Column definition
663 * @return string|string[]
664 */
665 protected function _alter_table($alter_type, $table, $field)
666 {
667 $sql = 'ALTER TABLE '.$this->db->escape_identifiers($table).' ';
668
669 // DROP has everything it needs now.
670 if ($alter_type === 'DROP')
671 {
672 return $sql.'DROP COLUMN '.$this->db->escape_identifiers($field);
673 }
674
675 $sqls = array();
676 for ($i = 0, $c = count($field), $sql .= $alter_type.' COLUMN '; $i < $c; $i++)
677 {
678 $sqls[] = $sql
679 .($field[$i]['_literal'] !== FALSE ? $field[$i]['_literal'] : $this->_process_column($field[$i]));
680 }
681
682 return $sqls;
683 }
684
685 // --------------------------------------------------------------------
686
687 /**
688 * Process fields
689 *
690 * @param bool $create_table
691 * @return array
692 */
693 protected function _process_fields($create_table = FALSE)
694 {
695 $fields = array();
696
697 foreach ($this->fields as $key => $attributes)
698 {
699 if (is_int($key) && ! is_array($attributes))
700 {
701 $fields[] = array('_literal' => $attributes);
702 continue;
703 }
704
705 $attributes = array_change_key_case($attributes, CASE_UPPER);
706
707 if ($create_table === TRUE && empty($attributes['TYPE']))
708 {
709 continue;
710 }
711
712 if (isset($attributes['TYPE']))
713 {
714 $this->_attr_type($attributes);
715 $this->_attr_unsigned($attributes, $field);
716 }
717
718 $field = array(
719 'name' => $key,
720 'new_name' => isset($attributes['NAME']) ? $attributes['NAME'] : NULL,
721 'type' => isset($attributes['TYPE']) ? $attributes['TYPE'] : NULL,
722 'length' => '',
723 'unsigned' => '',
724 'null' => '',
725 'unique' => '',
726 'default' => '',
727 'auto_increment' => '',
728 '_literal' => FALSE
729 );
730
731 $this->_attr_default($attributes, $field);
732
733 if (isset($attributes['NULL']))
734 {
735 if ($attributes['NULL'] === TRUE)
736 {
737 $field['null'] = empty($this->_null) ? '' : ' '.$this->_null;
738 }
739 elseif ($create_table === TRUE)
740 {
741 $field['null'] = ' NOT NULL';
742 }
743 }
744
745 $this->_attr_auto_increment($attributes, $field);
746 $this->_attr_unique($attributes, $field);
747
748 if (isset($attributes['TYPE']) && ! empty($attributes['CONSTRAINT']))
749 {
750 switch (strtoupper($attributes['TYPE']))
751 {
752 case 'ENUM':
753 case 'SET':
754 $attributes['CONSTRAINT'] = $this->db->escape($attributes['CONSTRAINT']);
755 default:
756 $field['length'] = is_array($attributes['CONSTRAINT'])
757 ? '('.implode(',', $attributes['CONSTRAINT']).')'
758 : '('.$attributes['CONSTRAINT'].')';
759 break;
760 }
761 }
762
763 $fields[] = $field;
764 }
765
766 return $fields;
767 }
768
769 // --------------------------------------------------------------------
770
771 /**
772 * Process column
773 *
774 * @param array $field
775 * @return string
776 */
777 protected function _process_column($field)
778 {
779 return $this->db->escape_identifiers($field['name'])
780 .' '.$field['type'].$field['length']
781 .$field['unsigned']
782 .$field['default']
783 .$field['null']
784 .$field['auto_increment']
785 .$field['unique'];
786 }
787
788 // --------------------------------------------------------------------
789
790 /**
791 * Field attribute TYPE
792 *
793 * Performs a data type mapping between different databases.
794 *
795 * @param array &$attributes
796 * @return void
797 */
798 protected function _attr_type(&$attributes)
799 {
800 // Usually overriden by drivers
801 }
802
803 // --------------------------------------------------------------------
804
805 /**
806 * Field attribute UNSIGNED
807 *
808 * Depending on the _unsigned property value:
809 *
810 * - TRUE will always set $field['unsigned'] to 'UNSIGNED'
811 * - FALSE will always set $field['unsigned'] to ''
812 * - array(TYPE) will set $field['unsigned'] to 'UNSIGNED',
813 * if $attributes['TYPE'] is found in the array
814 * - array(TYPE => UTYPE) will change $field['type'],
815 * from TYPE to UTYPE in case of a match
816 *
817 * @param array &$attributes
818 * @param array &$field
819 * @return void
820 */
821 protected function _attr_unsigned(&$attributes, &$field)
822 {
823 if (empty($attributes['UNSIGNED']) OR $attributes['UNSIGNED'] !== TRUE)
824 {
825 return;
826 }
827
828 // Reset the attribute in order to avoid issues if we do type conversion
829 $attributes['UNSIGNED'] = FALSE;
830
831 if (is_array($this->_unsigned))
832 {
833 foreach (array_keys($this->_unsigned) as $key)
834 {
835 if (is_int($key) && strcasecmp($attributes['TYPE'], $this->_unsigned[$key]) === 0)
836 {
837 $field['unsigned'] = ' UNSIGNED';
838 return;
839 }
840 elseif (is_string($key) && strcasecmp($attributes['TYPE'], $key) === 0)
841 {
842 $field['type'] = $key;
843 return;
844 }
845 }
846
847 return;
848 }
849
850 $field['unsigned'] = ($this->_unsigned === TRUE) ? ' UNSIGNED' : '';
851 }
852
853 // --------------------------------------------------------------------
854
855 /**
856 * Field attribute DEFAULT
857 *
858 * @param array &$attributes
859 * @param array &$field
860 * @return void
861 */
862 protected function _attr_default(&$attributes, &$field)
863 {
864 if ($this->_default === FALSE)
865 {
866 return;
867 }
868
869 if (array_key_exists('DEFAULT', $attributes))
870 {
871 if ($attributes['DEFAULT'] === NULL)
872 {
873 $field['default'] = empty($this->_null) ? '' : $this->_default.$this->_null;
874
875 // Override the NULL attribute if that's our default
876 $attributes['NULL'] = NULL;
877 $field['null'] = empty($this->_null) ? '' : ' '.$this->_null;
878 }
879 else
880 {
881 $field['default'] = $this->_default.$this->db->escape($attributes['DEFAULT']);
882 }
883 }
884 }
885
886 // --------------------------------------------------------------------
887
888 /**
889 * Field attribute UNIQUE
890 *
891 * @param array &$attributes
892 * @param array &$field
893 * @return void
894 */
895 protected function _attr_unique(&$attributes, &$field)
896 {
897 if ( ! empty($attributes['UNIQUE']) && $attributes['UNIQUE'] === TRUE)
898 {
899 $field['unique'] = ' UNIQUE';
900 }
901 }
902
903 // --------------------------------------------------------------------
904
905 /**
906 * Field attribute AUTO_INCREMENT
907 *
908 * @param array &$attributes
909 * @param array &$field
910 * @return void
911 */
912 protected function _attr_auto_increment(&$attributes, &$field)
913 {
914 if ( ! empty($attributes['AUTO_INCREMENT']) && $attributes['AUTO_INCREMENT'] === TRUE && stripos($field['type'], 'int') !== FALSE)
915 {
916 $field['auto_increment'] = ' AUTO_INCREMENT';
917 }
918 }
919
920 // --------------------------------------------------------------------
921
922 /**
923 * Process primary keys
924 *
925 * @param string $table Table name
926 * @return string
927 */
928 protected function _process_primary_keys($table)
929 {
930 $sql = '';
931
932 for ($i = 0, $c = count($this->primary_keys); $i < $c; $i++)
933 {
934 if ( ! isset($this->fields[$this->primary_keys[$i]]))
935 {
936 unset($this->primary_keys[$i]);
937 }
938 }
939
940 if (count($this->primary_keys) > 0)
941 {
942 $sql .= ",\n\tCONSTRAINT ".$this->db->escape_identifiers('pk_'.$table)
943 .' PRIMARY KEY('.implode(', ', $this->db->escape_identifiers($this->primary_keys)).')';
944 }
945
946 return $sql;
947 }
948
949 // --------------------------------------------------------------------
950
951 /**
952 * Process indexes
953 *
954 * @param string $table
955 * @return string
956 */
957 protected function _process_indexes($table = NULL)
958 {
959 $table = $this->db->escape_identifiers($table);
960 $sqls = array();
961
962 for ($i = 0, $c = count($this->keys); $i < $c; $i++)
963 {
964 if ( ! isset($this->fields[$this->keys[$i]]))
965 {
966 unset($this->keys[$i]);
967 continue;
968 }
969
970 is_array($this->keys[$i]) OR $this->keys[$i] = array($this->keys[$i]);
971
972 $sqls[] = 'CREATE INDEX '.$this->db->escape_identifiers(implode('_', $this->keys[$i]))
973 .' ON '.$this->db->escape_identifiers($table)
974 .' ('.implode(', ', $this->db->escape_identifiers($this->keys[$i])).');';
975 }
976
977 return $sqls;
978 }
979
980 // --------------------------------------------------------------------
981
982 /**
Derek Allard2067d1a2008-11-13 22:59:24 +0000983 * Reset
984 *
985 * Resets table creation vars
986 *
Derek Allard2067d1a2008-11-13 22:59:24 +0000987 * @return void
988 */
Phil Sturgeona7de97e2011-12-31 18:41:08 +0000989 protected function _reset()
Derek Allard2067d1a2008-11-13 22:59:24 +0000990 {
Andrey Andreev24276a32012-01-08 02:44:38 +0200991 $this->fields = $this->keys = $this->primary_keys = array();
Derek Allard2067d1a2008-11-13 22:59:24 +0000992 }
993
994}
995
996/* End of file DB_forge.php */
Timothy Warren215890b2012-03-20 09:38:16 -0400997/* Location: ./system/database/DB_forge.php */