pictcode / lib / Cake / Model / Datasource / Database / Mysql.php @ d932d503
履歴 | 表示 | アノテート | ダウンロード (23.533 KB)
1 |
<?php
|
---|---|
2 |
/**
|
3 |
* MySQL layer for DBO
|
4 |
*
|
5 |
* CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
|
6 |
* Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
7 |
*
|
8 |
* Licensed under The MIT License
|
9 |
* For full copyright and license information, please see the LICENSE.txt
|
10 |
* Redistributions of files must retain the above copyright notice.
|
11 |
*
|
12 |
* @copyright Copyright (c) Cake Software Foundation, Inc. (http://cakefoundation.org)
|
13 |
* @link http://cakephp.org CakePHP(tm) Project
|
14 |
* @package Cake.Model.Datasource.Database
|
15 |
* @since CakePHP(tm) v 0.10.5.1790
|
16 |
* @license http://www.opensource.org/licenses/mit-license.php MIT License
|
17 |
*/
|
18 |
|
19 |
App::uses('DboSource', 'Model/Datasource'); |
20 |
|
21 |
/**
|
22 |
* MySQL DBO driver object
|
23 |
*
|
24 |
* Provides connection and SQL generation for MySQL RDMS
|
25 |
*
|
26 |
* @package Cake.Model.Datasource.Database
|
27 |
*/
|
28 |
class Mysql extends DboSource { |
29 |
|
30 |
/**
|
31 |
* Datasource description
|
32 |
*
|
33 |
* @var string
|
34 |
*/
|
35 |
public $description = "MySQL DBO Driver"; |
36 |
|
37 |
/**
|
38 |
* Base configuration settings for MySQL driver
|
39 |
*
|
40 |
* @var array
|
41 |
*/
|
42 |
protected $_baseConfig = array( |
43 |
'persistent' => true, |
44 |
'host' => 'localhost', |
45 |
'login' => 'root', |
46 |
'password' => '', |
47 |
'database' => 'cake', |
48 |
'port' => '3306', |
49 |
'flags' => array() |
50 |
); |
51 |
|
52 |
/**
|
53 |
* Reference to the PDO object connection
|
54 |
*
|
55 |
* @var PDO
|
56 |
*/
|
57 |
protected $_connection = null; |
58 |
|
59 |
/**
|
60 |
* Start quote
|
61 |
*
|
62 |
* @var string
|
63 |
*/
|
64 |
public $startQuote = "`"; |
65 |
|
66 |
/**
|
67 |
* End quote
|
68 |
*
|
69 |
* @var string
|
70 |
*/
|
71 |
public $endQuote = "`"; |
72 |
|
73 |
/**
|
74 |
* use alias for update and delete. Set to true if version >= 4.1
|
75 |
*
|
76 |
* @var bool
|
77 |
*/
|
78 |
protected $_useAlias = true; |
79 |
|
80 |
/**
|
81 |
* List of engine specific additional field parameters used on table creating
|
82 |
*
|
83 |
* @var array
|
84 |
*/
|
85 |
public $fieldParameters = array( |
86 |
'charset' => array('value' => 'CHARACTER SET', 'quote' => false, 'join' => ' ', 'column' => false, 'position' => 'beforeDefault'), |
87 |
'collate' => array('value' => 'COLLATE', 'quote' => false, 'join' => ' ', 'column' => 'Collation', 'position' => 'beforeDefault'), |
88 |
'comment' => array('value' => 'COMMENT', 'quote' => true, 'join' => ' ', 'column' => 'Comment', 'position' => 'afterDefault'), |
89 |
'unsigned' => array( |
90 |
'value' => 'UNSIGNED', 'quote' => false, 'join' => ' ', 'column' => false, 'position' => 'beforeDefault', |
91 |
'noVal' => true, |
92 |
'options' => array(true), |
93 |
'types' => array('integer', 'float', 'decimal', 'biginteger') |
94 |
) |
95 |
); |
96 |
|
97 |
/**
|
98 |
* List of table engine specific parameters used on table creating
|
99 |
*
|
100 |
* @var array
|
101 |
*/
|
102 |
public $tableParameters = array( |
103 |
'charset' => array('value' => 'DEFAULT CHARSET', 'quote' => false, 'join' => '=', 'column' => 'charset'), |
104 |
'collate' => array('value' => 'COLLATE', 'quote' => false, 'join' => '=', 'column' => 'Collation'), |
105 |
'engine' => array('value' => 'ENGINE', 'quote' => false, 'join' => '=', 'column' => 'Engine'), |
106 |
'comment' => array('value' => 'COMMENT', 'quote' => true, 'join' => '=', 'column' => 'Comment'), |
107 |
); |
108 |
|
109 |
/**
|
110 |
* MySQL column definition
|
111 |
*
|
112 |
* @var array
|
113 |
*/
|
114 |
public $columns = array( |
115 |
'primary_key' => array('name' => 'NOT NULL AUTO_INCREMENT'), |
116 |
'string' => array('name' => 'varchar', 'limit' => '255'), |
117 |
'text' => array('name' => 'text'), |
118 |
'biginteger' => array('name' => 'bigint', 'limit' => '20'), |
119 |
'integer' => array('name' => 'int', 'limit' => '11', 'formatter' => 'intval'), |
120 |
'float' => array('name' => 'float', 'formatter' => 'floatval'), |
121 |
'decimal' => array('name' => 'decimal', 'formatter' => 'floatval'), |
122 |
'datetime' => array('name' => 'datetime', 'format' => 'Y-m-d H:i:s', 'formatter' => 'date'), |
123 |
'timestamp' => array('name' => 'timestamp', 'format' => 'Y-m-d H:i:s', 'formatter' => 'date'), |
124 |
'time' => array('name' => 'time', 'format' => 'H:i:s', 'formatter' => 'date'), |
125 |
'date' => array('name' => 'date', 'format' => 'Y-m-d', 'formatter' => 'date'), |
126 |
'binary' => array('name' => 'blob'), |
127 |
'boolean' => array('name' => 'tinyint', 'limit' => '1') |
128 |
); |
129 |
|
130 |
/**
|
131 |
* Mapping of collation names to character set names
|
132 |
*
|
133 |
* @var array
|
134 |
*/
|
135 |
protected $_charsets = array(); |
136 |
|
137 |
/**
|
138 |
* Connects to the database using options in the given configuration array.
|
139 |
*
|
140 |
* MySQL supports a few additional options that other drivers do not:
|
141 |
*
|
142 |
* - `unix_socket` Set to the path of the MySQL sock file. Can be used in place
|
143 |
* of host + port.
|
144 |
* - `ssl_key` SSL key file for connecting via SSL. Must be combined with `ssl_cert`.
|
145 |
* - `ssl_cert` The SSL certificate to use when connecting via SSL. Must be
|
146 |
* combined with `ssl_key`.
|
147 |
* - `ssl_ca` The certificate authority for SSL connections.
|
148 |
*
|
149 |
* @return bool True if the database could be connected, else false
|
150 |
* @throws MissingConnectionException
|
151 |
*/
|
152 |
public function connect() { |
153 |
$config = $this->config; |
154 |
$this->connected = false; |
155 |
|
156 |
$flags = $config['flags'] + array( |
157 |
PDO::ATTR_PERSISTENT => $config['persistent'], |
158 |
PDO::MYSQL_ATTR_USE_BUFFERED_QUERY => true, |
159 |
PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION |
160 |
); |
161 |
|
162 |
if (!empty($config['encoding'])) { |
163 |
$flags[PDO::MYSQL_ATTR_INIT_COMMAND] = 'SET NAMES ' . $config['encoding']; |
164 |
} |
165 |
if (!empty($config['ssl_key']) && !empty($config['ssl_cert'])) { |
166 |
$flags[PDO::MYSQL_ATTR_SSL_KEY] = $config['ssl_key']; |
167 |
$flags[PDO::MYSQL_ATTR_SSL_CERT] = $config['ssl_cert']; |
168 |
} |
169 |
if (!empty($config['ssl_ca'])) { |
170 |
$flags[PDO::MYSQL_ATTR_SSL_CA] = $config['ssl_ca']; |
171 |
} |
172 |
if (empty($config['unix_socket'])) { |
173 |
$dsn = "mysql:host={$config['host']};port={$config['port']};dbname={$config['database']}"; |
174 |
} else {
|
175 |
$dsn = "mysql:unix_socket={$config['unix_socket']};dbname={$config['database']}"; |
176 |
} |
177 |
|
178 |
try {
|
179 |
$this->_connection = new PDO( |
180 |
$dsn,
|
181 |
$config['login'], |
182 |
$config['password'], |
183 |
$flags
|
184 |
); |
185 |
$this->connected = true; |
186 |
if (!empty($config['settings'])) { |
187 |
foreach ($config['settings'] as $key => $value) { |
188 |
$this->_execute("SET $key=$value"); |
189 |
} |
190 |
} |
191 |
} catch (PDOException $e) { |
192 |
throw new MissingConnectionException(array( |
193 |
'class' => get_class($this), |
194 |
'message' => $e->getMessage() |
195 |
)); |
196 |
} |
197 |
|
198 |
$this->_charsets = array(); |
199 |
$this->_useAlias = (bool)version_compare($this->getVersion(), "4.1", ">="); |
200 |
|
201 |
return $this->connected; |
202 |
} |
203 |
|
204 |
/**
|
205 |
* Check whether the MySQL extension is installed/loaded
|
206 |
*
|
207 |
* @return bool
|
208 |
*/
|
209 |
public function enabled() { |
210 |
return in_array('mysql', PDO::getAvailableDrivers()); |
211 |
} |
212 |
|
213 |
/**
|
214 |
* Returns an array of sources (tables) in the database.
|
215 |
*
|
216 |
* @param mixed $data List of tables.
|
217 |
* @return array Array of table names in the database
|
218 |
*/
|
219 |
public function listSources($data = null) { |
220 |
$cache = parent::listSources(); |
221 |
if ($cache) { |
222 |
return $cache; |
223 |
} |
224 |
$result = $this->_execute('SHOW TABLES FROM ' . $this->name($this->config['database'])); |
225 |
|
226 |
if (!$result) { |
227 |
$result->closeCursor();
|
228 |
return array(); |
229 |
} |
230 |
$tables = array(); |
231 |
|
232 |
while ($line = $result->fetch(PDO::FETCH_NUM)) { |
233 |
$tables[] = $line[0]; |
234 |
} |
235 |
|
236 |
$result->closeCursor();
|
237 |
parent::listSources($tables); |
238 |
return $tables; |
239 |
} |
240 |
|
241 |
/**
|
242 |
* Builds a map of the columns contained in a result
|
243 |
*
|
244 |
* @param PDOStatement $results The results to format.
|
245 |
* @return void
|
246 |
*/
|
247 |
public function resultSet($results) { |
248 |
$this->map = array(); |
249 |
$numFields = $results->columnCount(); |
250 |
$index = 0; |
251 |
|
252 |
while ($numFields-- > 0) { |
253 |
$column = $results->getColumnMeta($index); |
254 |
if ($column['len'] === 1 && (empty($column['native_type']) || $column['native_type'] === 'TINY')) { |
255 |
$type = 'boolean'; |
256 |
} else {
|
257 |
$type = empty($column['native_type']) ? 'string' : $column['native_type']; |
258 |
} |
259 |
if (!empty($column['table']) && strpos($column['name'], $this->virtualFieldSeparator) === false) { |
260 |
$this->map[$index++] = array($column['table'], $column['name'], $type); |
261 |
} else {
|
262 |
$this->map[$index++] = array(0, $column['name'], $type); |
263 |
} |
264 |
} |
265 |
} |
266 |
|
267 |
/**
|
268 |
* Fetches the next row from the current result set
|
269 |
*
|
270 |
* @return mixed array with results fetched and mapped to column names or false if there is no results left to fetch
|
271 |
*/
|
272 |
public function fetchResult() { |
273 |
if ($row = $this->_result->fetch(PDO::FETCH_NUM)) { |
274 |
$resultRow = array(); |
275 |
foreach ($this->map as $col => $meta) { |
276 |
list($table, $column, $type) = $meta; |
277 |
$resultRow[$table][$column] = $row[$col]; |
278 |
if ($type === 'boolean' && $row[$col] !== null) { |
279 |
$resultRow[$table][$column] = $this->boolean($resultRow[$table][$column]); |
280 |
} |
281 |
} |
282 |
return $resultRow; |
283 |
} |
284 |
$this->_result->closeCursor();
|
285 |
return false; |
286 |
} |
287 |
|
288 |
/**
|
289 |
* Gets the database encoding
|
290 |
*
|
291 |
* @return string The database encoding
|
292 |
*/
|
293 |
public function getEncoding() { |
294 |
return $this->_execute('SHOW VARIABLES LIKE ?', array('character_set_client'))->fetchObject()->Value; |
295 |
} |
296 |
|
297 |
/**
|
298 |
* Query charset by collation
|
299 |
*
|
300 |
* @param string $name Collation name
|
301 |
* @return string Character set name
|
302 |
*/
|
303 |
public function getCharsetName($name) { |
304 |
if ((bool)version_compare($this->getVersion(), "5", "<")) { |
305 |
return false; |
306 |
} |
307 |
if (isset($this->_charsets[$name])) { |
308 |
return $this->_charsets[$name]; |
309 |
} |
310 |
$r = $this->_execute( |
311 |
'SELECT CHARACTER_SET_NAME FROM INFORMATION_SCHEMA.COLLATIONS WHERE COLLATION_NAME = ?',
|
312 |
array($name) |
313 |
); |
314 |
$cols = $r->fetch(PDO::FETCH_ASSOC); |
315 |
|
316 |
if (isset($cols['CHARACTER_SET_NAME'])) { |
317 |
$this->_charsets[$name] = $cols['CHARACTER_SET_NAME']; |
318 |
} else {
|
319 |
$this->_charsets[$name] = false; |
320 |
} |
321 |
return $this->_charsets[$name]; |
322 |
} |
323 |
|
324 |
/**
|
325 |
* Returns an array of the fields in given table name.
|
326 |
*
|
327 |
* @param Model|string $model Name of database table to inspect or model instance
|
328 |
* @return array Fields in table. Keys are name and type
|
329 |
* @throws CakeException
|
330 |
*/
|
331 |
public function describe($model) { |
332 |
$key = $this->fullTableName($model, false); |
333 |
$cache = parent::describe($key); |
334 |
if ($cache) { |
335 |
return $cache; |
336 |
} |
337 |
$table = $this->fullTableName($model); |
338 |
|
339 |
$fields = false; |
340 |
$cols = $this->_execute('SHOW FULL COLUMNS FROM ' . $table); |
341 |
if (!$cols) { |
342 |
throw new CakeException(__d('cake_dev', 'Could not describe table for %s', $table)); |
343 |
} |
344 |
|
345 |
while ($column = $cols->fetch(PDO::FETCH_OBJ)) { |
346 |
$fields[$column->Field] = array( |
347 |
'type' => $this->column($column->Type), |
348 |
'null' => ($column->Null === 'YES' ? true : false), |
349 |
'default' => $column->Default, |
350 |
'length' => $this->length($column->Type) |
351 |
); |
352 |
if (in_array($fields[$column->Field]['type'], $this->fieldParameters['unsigned']['types'], true)) { |
353 |
$fields[$column->Field]['unsigned'] = $this->_unsigned($column->Type); |
354 |
} |
355 |
if (in_array($fields[$column->Field]['type'], array('timestamp', 'datetime')) && strtoupper($column->Default) === 'CURRENT_TIMESTAMP') { |
356 |
$fields[$column->Field]['default'] = null; |
357 |
} |
358 |
if (!empty($column->Key) && isset($this->index[$column->Key])) { |
359 |
$fields[$column->Field]['key'] = $this->index[$column->Key]; |
360 |
} |
361 |
foreach ($this->fieldParameters as $name => $value) { |
362 |
if (!empty($column->{$value['column']})) { |
363 |
$fields[$column->Field][$name] = $column->{$value['column']}; |
364 |
} |
365 |
} |
366 |
if (isset($fields[$column->Field]['collate'])) { |
367 |
$charset = $this->getCharsetName($fields[$column->Field]['collate']); |
368 |
if ($charset) { |
369 |
$fields[$column->Field]['charset'] = $charset; |
370 |
} |
371 |
} |
372 |
} |
373 |
$this->_cacheDescription($key, $fields); |
374 |
$cols->closeCursor();
|
375 |
return $fields; |
376 |
} |
377 |
|
378 |
/**
|
379 |
* Generates and executes an SQL UPDATE statement for given model, fields, and values.
|
380 |
*
|
381 |
* @param Model $model The model to update.
|
382 |
* @param array $fields The fields to update.
|
383 |
* @param array $values The values to set.
|
384 |
* @param mixed $conditions The conditions to use.
|
385 |
* @return array
|
386 |
*/
|
387 |
public function update(Model $model, $fields = array(), $values = null, $conditions = null) { |
388 |
if (!$this->_useAlias) { |
389 |
return parent::update($model, $fields, $values, $conditions); |
390 |
} |
391 |
|
392 |
if (!$values) { |
393 |
$combined = $fields; |
394 |
} else {
|
395 |
$combined = array_combine($fields, $values); |
396 |
} |
397 |
|
398 |
$alias = $joins = false; |
399 |
$fields = $this->_prepareUpdateFields($model, $combined, empty($conditions), !empty($conditions)); |
400 |
$fields = implode(', ', $fields); |
401 |
$table = $this->fullTableName($model); |
402 |
|
403 |
if (!empty($conditions)) { |
404 |
$alias = $this->name($model->alias); |
405 |
if ($model->name === $model->alias) { |
406 |
$joins = implode(' ', $this->_getJoins($model)); |
407 |
} |
408 |
} |
409 |
$conditions = $this->conditions($this->defaultConditions($model, $conditions, $alias), true, true, $model); |
410 |
|
411 |
if ($conditions === false) { |
412 |
return false; |
413 |
} |
414 |
|
415 |
if (!$this->execute($this->renderStatement('update', compact('table', 'alias', 'joins', 'fields', 'conditions')))) { |
416 |
$model->onError();
|
417 |
return false; |
418 |
} |
419 |
return true; |
420 |
} |
421 |
|
422 |
/**
|
423 |
* Generates and executes an SQL DELETE statement for given id/conditions on given model.
|
424 |
*
|
425 |
* @param Model $model The model to delete from.
|
426 |
* @param mixed $conditions The conditions to use.
|
427 |
* @return bool Success
|
428 |
*/
|
429 |
public function delete(Model $model, $conditions = null) { |
430 |
if (!$this->_useAlias) { |
431 |
return parent::delete($model, $conditions); |
432 |
} |
433 |
$alias = $this->name($model->alias); |
434 |
$table = $this->fullTableName($model); |
435 |
$joins = implode(' ', $this->_getJoins($model)); |
436 |
|
437 |
if (empty($conditions)) { |
438 |
$alias = $joins = false; |
439 |
} |
440 |
$complexConditions = false; |
441 |
foreach ((array)$conditions as $key => $value) { |
442 |
if (strpos($key, $model->alias) === false) { |
443 |
$complexConditions = true; |
444 |
break;
|
445 |
} |
446 |
} |
447 |
if (!$complexConditions) { |
448 |
$joins = false; |
449 |
} |
450 |
|
451 |
$conditions = $this->conditions($this->defaultConditions($model, $conditions, $alias), true, true, $model); |
452 |
if ($conditions === false) { |
453 |
return false; |
454 |
} |
455 |
if ($this->execute($this->renderStatement('delete', compact('alias', 'table', 'joins', 'conditions'))) === false) { |
456 |
$model->onError();
|
457 |
return false; |
458 |
} |
459 |
return true; |
460 |
} |
461 |
|
462 |
/**
|
463 |
* Sets the database encoding
|
464 |
*
|
465 |
* @param string $enc Database encoding
|
466 |
* @return bool
|
467 |
*/
|
468 |
public function setEncoding($enc) { |
469 |
return $this->_execute('SET NAMES ' . $enc) !== false; |
470 |
} |
471 |
|
472 |
/**
|
473 |
* Returns an array of the indexes in given datasource name.
|
474 |
*
|
475 |
* @param string $model Name of model to inspect
|
476 |
* @return array Fields in table. Keys are column and unique
|
477 |
*/
|
478 |
public function index($model) { |
479 |
$index = array(); |
480 |
$table = $this->fullTableName($model); |
481 |
$old = version_compare($this->getVersion(), '4.1', '<='); |
482 |
if ($table) { |
483 |
$indexes = $this->_execute('SHOW INDEX FROM ' . $table); |
484 |
// @codingStandardsIgnoreStart
|
485 |
// MySQL columns don't match the cakephp conventions.
|
486 |
while ($idx = $indexes->fetch(PDO::FETCH_OBJ)) { |
487 |
if ($old) { |
488 |
$idx = (object)current((array)$idx); |
489 |
} |
490 |
if (!isset($index[$idx->Key_name]['column'])) { |
491 |
$col = array(); |
492 |
$index[$idx->Key_name]['column'] = $idx->Column_name; |
493 |
|
494 |
if ($idx->Index_type === 'FULLTEXT') { |
495 |
$index[$idx->Key_name]['type'] = strtolower($idx->Index_type); |
496 |
} else {
|
497 |
$index[$idx->Key_name]['unique'] = (int)($idx->Non_unique == 0); |
498 |
} |
499 |
} else {
|
500 |
if (!empty($index[$idx->Key_name]['column']) && !is_array($index[$idx->Key_name]['column'])) { |
501 |
$col[] = $index[$idx->Key_name]['column']; |
502 |
} |
503 |
$col[] = $idx->Column_name; |
504 |
$index[$idx->Key_name]['column'] = $col; |
505 |
} |
506 |
if (!empty($idx->Sub_part)) { |
507 |
if (!isset($index[$idx->Key_name]['length'])) { |
508 |
$index[$idx->Key_name]['length'] = array(); |
509 |
} |
510 |
$index[$idx->Key_name]['length'][$idx->Column_name] = $idx->Sub_part; |
511 |
} |
512 |
} |
513 |
// @codingStandardsIgnoreEnd
|
514 |
$indexes->closeCursor();
|
515 |
} |
516 |
return $index; |
517 |
} |
518 |
|
519 |
/**
|
520 |
* Generate a MySQL Alter Table syntax for the given Schema comparison
|
521 |
*
|
522 |
* @param array $compare Result of a CakeSchema::compare()
|
523 |
* @param string $table The table name.
|
524 |
* @return array Array of alter statements to make.
|
525 |
*/
|
526 |
public function alterSchema($compare, $table = null) { |
527 |
if (!is_array($compare)) { |
528 |
return false; |
529 |
} |
530 |
$out = ''; |
531 |
$colList = array(); |
532 |
foreach ($compare as $curTable => $types) { |
533 |
$indexes = $tableParameters = $colList = array(); |
534 |
if (!$table || $table === $curTable) { |
535 |
$out .= 'ALTER TABLE ' . $this->fullTableName($curTable) . " \n"; |
536 |
foreach ($types as $type => $column) { |
537 |
if (isset($column['indexes'])) { |
538 |
$indexes[$type] = $column['indexes']; |
539 |
unset($column['indexes']); |
540 |
} |
541 |
if (isset($column['tableParameters'])) { |
542 |
$tableParameters[$type] = $column['tableParameters']; |
543 |
unset($column['tableParameters']); |
544 |
} |
545 |
switch ($type) { |
546 |
case 'add': |
547 |
foreach ($column as $field => $col) { |
548 |
$col['name'] = $field; |
549 |
$alter = 'ADD ' . $this->buildColumn($col); |
550 |
if (isset($col['after'])) { |
551 |
$alter .= ' AFTER ' . $this->name($col['after']); |
552 |
} |
553 |
$colList[] = $alter; |
554 |
} |
555 |
break;
|
556 |
case 'drop': |
557 |
foreach ($column as $field => $col) { |
558 |
$col['name'] = $field; |
559 |
$colList[] = 'DROP ' . $this->name($field); |
560 |
} |
561 |
break;
|
562 |
case 'change': |
563 |
foreach ($column as $field => $col) { |
564 |
if (!isset($col['name'])) { |
565 |
$col['name'] = $field; |
566 |
} |
567 |
$alter = 'CHANGE ' . $this->name($field) . ' ' . $this->buildColumn($col); |
568 |
if (isset($col['after'])) { |
569 |
$alter .= ' AFTER ' . $this->name($col['after']); |
570 |
} |
571 |
$colList[] = $alter; |
572 |
} |
573 |
break;
|
574 |
} |
575 |
} |
576 |
$colList = array_merge($colList, $this->_alterIndexes($curTable, $indexes)); |
577 |
$colList = array_merge($colList, $this->_alterTableParameters($curTable, $tableParameters)); |
578 |
$out .= "\t" . implode(",\n\t", $colList) . ";\n\n"; |
579 |
} |
580 |
} |
581 |
return $out; |
582 |
} |
583 |
|
584 |
/**
|
585 |
* Generate a "drop table" statement for the given table
|
586 |
*
|
587 |
* @param type $table Name of the table to drop
|
588 |
* @return string Drop table SQL statement
|
589 |
*/
|
590 |
protected function _dropTable($table) { |
591 |
return 'DROP TABLE IF EXISTS ' . $this->fullTableName($table) . ";"; |
592 |
} |
593 |
|
594 |
/**
|
595 |
* Generate MySQL table parameter alteration statements for a table.
|
596 |
*
|
597 |
* @param string $table Table to alter parameters for.
|
598 |
* @param array $parameters Parameters to add & drop.
|
599 |
* @return array Array of table property alteration statements.
|
600 |
*/
|
601 |
protected function _alterTableParameters($table, $parameters) { |
602 |
if (isset($parameters['change'])) { |
603 |
return $this->buildTableParameters($parameters['change']); |
604 |
} |
605 |
return array(); |
606 |
} |
607 |
|
608 |
/**
|
609 |
* Format indexes for create table
|
610 |
*
|
611 |
* @param array $indexes An array of indexes to generate SQL from
|
612 |
* @param string $table Optional table name, not used
|
613 |
* @return array An array of SQL statements for indexes
|
614 |
* @see DboSource::buildIndex()
|
615 |
*/
|
616 |
public function buildIndex($indexes, $table = null) { |
617 |
$join = array(); |
618 |
foreach ($indexes as $name => $value) { |
619 |
$out = ''; |
620 |
if ($name === 'PRIMARY') { |
621 |
$out .= 'PRIMARY '; |
622 |
$name = null; |
623 |
} else {
|
624 |
if (!empty($value['unique'])) { |
625 |
$out .= 'UNIQUE '; |
626 |
} |
627 |
$name = $this->startQuote . $name . $this->endQuote; |
628 |
} |
629 |
if (isset($value['type']) && strtolower($value['type']) === 'fulltext') { |
630 |
$out .= 'FULLTEXT '; |
631 |
} |
632 |
$out .= 'KEY ' . $name . ' ('; |
633 |
|
634 |
if (is_array($value['column'])) { |
635 |
if (isset($value['length'])) { |
636 |
$vals = array(); |
637 |
foreach ($value['column'] as $column) { |
638 |
$name = $this->name($column); |
639 |
if (isset($value['length'])) { |
640 |
$name .= $this->_buildIndexSubPart($value['length'], $column); |
641 |
} |
642 |
$vals[] = $name; |
643 |
} |
644 |
$out .= implode(', ', $vals); |
645 |
} else {
|
646 |
$out .= implode(', ', array_map(array(&$this, 'name'), $value['column'])); |
647 |
} |
648 |
} else {
|
649 |
$out .= $this->name($value['column']); |
650 |
if (isset($value['length'])) { |
651 |
$out .= $this->_buildIndexSubPart($value['length'], $value['column']); |
652 |
} |
653 |
} |
654 |
$out .= ')'; |
655 |
$join[] = $out; |
656 |
} |
657 |
return $join; |
658 |
} |
659 |
|
660 |
/**
|
661 |
* Generate MySQL index alteration statements for a table.
|
662 |
*
|
663 |
* @param string $table Table to alter indexes for
|
664 |
* @param array $indexes Indexes to add and drop
|
665 |
* @return array Index alteration statements
|
666 |
*/
|
667 |
protected function _alterIndexes($table, $indexes) { |
668 |
$alter = array(); |
669 |
if (isset($indexes['drop'])) { |
670 |
foreach ($indexes['drop'] as $name => $value) { |
671 |
$out = 'DROP '; |
672 |
if ($name === 'PRIMARY') { |
673 |
$out .= 'PRIMARY KEY'; |
674 |
} else {
|
675 |
$out .= 'KEY ' . $this->startQuote . $name . $this->endQuote; |
676 |
} |
677 |
$alter[] = $out; |
678 |
} |
679 |
} |
680 |
if (isset($indexes['add'])) { |
681 |
$add = $this->buildIndex($indexes['add']); |
682 |
foreach ($add as $index) { |
683 |
$alter[] = 'ADD ' . $index; |
684 |
} |
685 |
} |
686 |
return $alter; |
687 |
} |
688 |
|
689 |
/**
|
690 |
* Format length for text indexes
|
691 |
*
|
692 |
* @param array $lengths An array of lengths for a single index
|
693 |
* @param string $column The column for which to generate the index length
|
694 |
* @return string Formatted length part of an index field
|
695 |
*/
|
696 |
protected function _buildIndexSubPart($lengths, $column) { |
697 |
if ($lengths === null) { |
698 |
return ''; |
699 |
} |
700 |
if (!isset($lengths[$column])) { |
701 |
return ''; |
702 |
} |
703 |
return '(' . $lengths[$column] . ')'; |
704 |
} |
705 |
|
706 |
/**
|
707 |
* Returns a detailed array of sources (tables) in the database.
|
708 |
*
|
709 |
* @param string $name Table name to get parameters
|
710 |
* @return array Array of table names in the database
|
711 |
*/
|
712 |
public function listDetailedSources($name = null) { |
713 |
$condition = ''; |
714 |
if (is_string($name)) { |
715 |
$condition = ' WHERE name = ' . $this->value($name); |
716 |
} |
717 |
$result = $this->_connection->query('SHOW TABLE STATUS ' . $condition, PDO::FETCH_ASSOC); |
718 |
|
719 |
if (!$result) { |
720 |
$result->closeCursor();
|
721 |
return array(); |
722 |
} |
723 |
$tables = array(); |
724 |
foreach ($result as $row) { |
725 |
$tables[$row['Name']] = (array)$row; |
726 |
unset($tables[$row['Name']]['queryString']); |
727 |
if (!empty($row['Collation'])) { |
728 |
$charset = $this->getCharsetName($row['Collation']); |
729 |
if ($charset) { |
730 |
$tables[$row['Name']]['charset'] = $charset; |
731 |
} |
732 |
} |
733 |
} |
734 |
$result->closeCursor();
|
735 |
if (is_string($name) && isset($tables[$name])) { |
736 |
return $tables[$name]; |
737 |
} |
738 |
return $tables; |
739 |
} |
740 |
|
741 |
/**
|
742 |
* Converts database-layer column types to basic types
|
743 |
*
|
744 |
* @param string $real Real database-layer column type (i.e. "varchar(255)")
|
745 |
* @return string Abstract column type (i.e. "string")
|
746 |
*/
|
747 |
public function column($real) { |
748 |
if (is_array($real)) { |
749 |
$col = $real['name']; |
750 |
if (isset($real['limit'])) { |
751 |
$col .= '(' . $real['limit'] . ')'; |
752 |
} |
753 |
return $col; |
754 |
} |
755 |
|
756 |
$col = str_replace(')', '', $real); |
757 |
$limit = $this->length($real); |
758 |
if (strpos($col, '(') !== false) { |
759 |
list($col, $vals) = explode('(', $col); |
760 |
} |
761 |
|
762 |
if (in_array($col, array('date', 'time', 'datetime', 'timestamp'))) { |
763 |
return $col; |
764 |
} |
765 |
if (($col === 'tinyint' && $limit === 1) || $col === 'boolean') { |
766 |
return 'boolean'; |
767 |
} |
768 |
if (strpos($col, 'bigint') !== false || $col === 'bigint') { |
769 |
return 'biginteger'; |
770 |
} |
771 |
if (strpos($col, 'int') !== false) { |
772 |
return 'integer'; |
773 |
} |
774 |
if (strpos($col, 'char') !== false || $col === 'tinytext') { |
775 |
return 'string'; |
776 |
} |
777 |
if (strpos($col, 'text') !== false) { |
778 |
return 'text'; |
779 |
} |
780 |
if (strpos($col, 'blob') !== false || $col === 'binary') { |
781 |
return 'binary'; |
782 |
} |
783 |
if (strpos($col, 'float') !== false || strpos($col, 'double') !== false) { |
784 |
return 'float'; |
785 |
} |
786 |
if (strpos($col, 'decimal') !== false || strpos($col, 'numeric') !== false) { |
787 |
return 'decimal'; |
788 |
} |
789 |
if (strpos($col, 'enum') !== false) { |
790 |
return "enum($vals)"; |
791 |
} |
792 |
if (strpos($col, 'set') !== false) { |
793 |
return "set($vals)"; |
794 |
} |
795 |
return 'text'; |
796 |
} |
797 |
|
798 |
/**
|
799 |
* {@inheritDoc}
|
800 |
*/
|
801 |
public function value($data, $column = null, $null = true) { |
802 |
$value = parent::value($data, $column, $null); |
803 |
if (is_numeric($value) && substr($column, 0, 3) === 'set') { |
804 |
return $this->_connection->quote($value); |
805 |
} |
806 |
return $value; |
807 |
} |
808 |
|
809 |
/**
|
810 |
* Gets the schema name
|
811 |
*
|
812 |
* @return string The schema name
|
813 |
*/
|
814 |
public function getSchemaName() { |
815 |
return $this->config['database']; |
816 |
} |
817 |
|
818 |
/**
|
819 |
* Check if the server support nested transactions
|
820 |
*
|
821 |
* @return bool
|
822 |
*/
|
823 |
public function nestedTransactionSupported() { |
824 |
return $this->useNestedTransactions && version_compare($this->getVersion(), '4.1', '>='); |
825 |
} |
826 |
|
827 |
/**
|
828 |
* Check if column type is unsigned
|
829 |
*
|
830 |
* @param string $real Real database-layer column type (i.e. "varchar(255)")
|
831 |
* @return bool True if column is unsigned, false otherwise
|
832 |
*/
|
833 |
protected function _unsigned($real) { |
834 |
return strpos(strtolower($real), 'unsigned') !== false; |
835 |
} |
836 |
|
837 |
} |