_transactionLevel===0) { $this->_debugTimer(); parent::beginTransaction(); $this->_debugStat(self::DEBUG_TRANSACTION, 'BEGIN'); } $this->_transactionLevel++; return $this; } /** * Commit DB transaction * * @return Varien_Db_Adapter_Pdo_Mysql */ public function commit() { if ($this->_transactionLevel===1) { $this->_debugTimer(); parent::commit(); $this->_debugStat(self::DEBUG_TRANSACTION, 'COMMIT'); } $this->_transactionLevel--; return $this; } /** * Rollback DB transaction * * @return Varien_Db_Adapter_Pdo_Mysql */ public function rollback() { if ($this->_transactionLevel===1) { $this->_debugTimer(); parent::rollback(); $this->_debugStat(self::DEBUG_TRANSACTION, 'ROLLBACK'); } $this->_transactionLevel--; return $this; } /** * Get adapter transaction level state. Return 0 if all transactions are complete * * @return int */ public function getTransactionLevel() { return $this->_transactionLevel; } /** * Convert date to DB format * * @param mixed $date * @return string */ public function convertDate($date) { if ($date instanceof Zend_Date) { return $date->toString(self::ISO_DATE_FORMAT); } return strftime('%Y-%m-%d', strtotime($date)); } /** * Convert date and time to DB format * * @param mixed $date * @return string */ public function convertDateTime($datetime) { if ($datetime instanceof Zend_Date) { return $datetime->toString(self::ISO_DATETIME_FORMAT); } return strftime('%Y-%m-%d %H:%M:%S', strtotime($datetime)); } /** * Creates a PDO object and connects to the database. * */ protected function _connect() { if ($this->_connection) { return; } if (!extension_loaded('pdo_mysql')) { throw new Zend_Db_Adapter_Exception('pdo_mysql extension is not installed'); } if (strpos($this->_config['host'], '/')!==false) { $this->_config['unix_socket'] = $this->_config['host']; unset($this->_config['host']); } else if (strpos($this->_config['host'], ':')!==false) { list($this->_config['host'], $this->_config['port']) = explode(':', $this->_config['host']); } $this->_debugTimer(); parent::_connect(); $this->_debugStat(self::DEBUG_CONNECT, ''); /** @link http://bugs.mysql.com/bug.php?id=18551 */ $this->_connection->query("SET SQL_MODE=''"); if (!$this->_connectionFlagsSet) { $this->_connection->setAttribute(PDO::ATTR_EMULATE_PREPARES, true); $this->_connection->setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, true); $this->_connectionFlagsSet = true; } } /** * Run RAW Query * * @param string $sql * @return Zend_Db_Statement_Interface */ public function raw_query($sql) { $lostConnectionMessage = 'SQLSTATE[HY000]: General error: 2013 Lost connection to MySQL server during query'; $tries = 0; do { $retry = false; try { $result = $this->getConnection()->query($sql); } catch (PDOException $e) { if ($tries < 10 && $e->getMessage() == $lostConnectionMessage) { $retry = true; $tries++; } else { throw $e; } } } while ($retry); return $result; } /** * Run RAW query and Fetch First row * * @param string $sql * @param string|int $field * @return mixed */ public function raw_fetchRow($sql, $field=null) { if (!$result = $this->raw_query($sql)) { return false; } if (!$row = $result->fetch(PDO::FETCH_ASSOC)) { return false; } if (empty($field)) { return $row; } else { return isset($row[$field]) ? $row[$field] : false; } } /** * Special handling for PDO query(). * All bind parameter names must begin with ':' * * @param string|Zend_Db_Select $sql The SQL statement with placeholders. * @param array $bind An array of data to bind to the placeholders. * @return Zend_Db_Pdo_Statement * @throws Zend_Db_Adapter_Exception To re-throw PDOException. */ public function query($sql, $bind = array()) { $this->_debugTimer(); try { $sql = (string)$sql; if (strpos($sql, ':') !== false || strpos($sql, '?') !== false) { $this->_bindParams = $bind; $sql = preg_replace_callback('#(([\'"])((\\2)|((.*?[^\\\\])\\2)))#', array($this, 'proccessBindCallback'), $sql ); Varien_Exception::processPcreError(); $bind = $this->_bindParams; } $result = parent::query($sql, $bind); } catch (Exception $e) { $this->_debugStat(self::DEBUG_QUERY, $sql, $bind); $this->_debugException($e); } $this->_debugStat(self::DEBUG_QUERY, $sql, $bind, $result); return $result; } /** * Callback function for prepare Query Bind RegExp * * @param array $matches * @return string */ public function proccessBindCallback($matches) { if (isset($matches[6]) && ( strpos($matches[6], "'") !== false || strpos($matches[6], ":") !== false || strpos($matches[6], "?") !== false)) { $bindName = ':_mage_bind_var_' . ( ++ $this->_bindIncrement ); $this->_bindParams[$bindName] = $this->_unQuote($matches[6]); return ' ' . $bindName; } return $matches[0]; } /** * Unquote raw string (use for auto-bind) * * @param string $string * @return string */ protected function _unQuote($string) { $translate = array( "\\000" => "\000", "\\n" => "\n", "\\r" => "\r", "\\\\" => "\\", "\'" => "'", "\\\"" => "\"", "\\032" => "\032" ); return strtr($string, $translate); } /** * Run Multi Query * * @param string $sql * @return array */ public function multi_query($sql) { ##$result = $this->raw_query($sql); #$this->beginTransaction(); try { $stmts = $this->_splitMultiQuery($sql); $result = array(); foreach ($stmts as $stmt) { $result[] = $this->raw_query($stmt); } #$this->commit(); } catch (Exception $e) { #$this->rollback(); throw $e; } $this->resetDdlCache(); return $result; } /** * Split multi statement query * * @param $sql string * @return array */ protected function _splitMultiQuery($sql) { $parts = preg_split('#(;|\'|"|\\\\|//|--|\n|/\*|\*/)#', $sql, null, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE ); $q = false; $c = false; $stmts = array(); $s = ''; foreach ($parts as $i=>$part) { // strings if (($part==="'" || $part==='"') && ($i===0 || $parts[$i-1]!=='\\')) { if ($q===false) { $q = $part; } else if ($q===$part) { $q = false; } } // single line comments if (($part==='//' || $part==='--') && ($i===0 || $parts[$i-1]==="\n")) { $c = $part; } else if ($part==="\n" && ($c==='//' || $c==='--')) { $c = false; } // multi line comments if ($part==='/*' && $c===false) { $c = '/*'; } else if ($part==='*/' && $c==='/*') { $c = false; } // statements if ($part===';' && $q===false && $c===false) { if (trim($s)!=='') { $stmts[] = trim($s); $s = ''; } } else { $s .= $part; } } if (trim($s)!=='') { $stmts[] = trim($s); } return $stmts; } /** * Delete foreign key if it exist * * @param string $tableName * @param string $foreignKey * @param string $shemaName * @return mixed */ public function dropForeignKey($tableName, $foreignKey, $schemaName = null) { $foreignKeys = $this->getForeignKeys($tableName, $schemaName); if (isset($foreignKeys[strtoupper($foreignKey)])) { $sql = sprintf('ALTER TABLE %s DROP FOREIGN KEY %s', $this->quoteIdentifier($this->_getTableName($tableName, $schemaName)), $this->quoteIdentifier($foreignKeys[strtoupper($foreignKey)]['FK_NAME'])); $this->resetDdlCache($tableName, $schemaName); return $this->raw_query($sql); } return true; } /** * Delete index from a table if it exist * * @param string $tableName * @param string $keyName * @param string $shemaName * @return bool */ public function dropKey($tableName, $keyName, $shemaName = null) { $indexList = $this->getIndexList($tableName, $shemaName); $keyName = strtoupper($keyName); if (!isset($indexList[$keyName])) { return true; } if ($keyName == 'PRIMARY') { $cond = 'DROP PRIMARY KEY'; } else { $cond = sprintf('DROP KEY %s', $this->quoteIdentifier($indexList[$keyName]['KEY_NAME'])); } $sql = sprintf('ALTER TABLE %s %s', $this->quoteIdentifier($this->_getTableName($tableName, $shemaName)), $cond); $this->resetDdlCache($tableName, $shemaName); return $this->raw_query($sql); } /** * Prepare table before add constraint foreign key * * @param string $tableName * @param string $columnName * @param string $refTableName * @param string $refColumnName * @param string $onDelete * @return Varien_Db_Adapter_Pdo_Mysql */ public function purgeOrphanRecords($tableName, $columnName, $refTableName, $refColumnName, $onDelete = 'cascade') { if (strtoupper($onDelete) == 'CASCADE' || strtoupper($onDelete) == 'RESTRICT') { $sql = "DELETE `p`.* FROM `{$tableName}` AS `p`" . " LEFT JOIN `{$refTableName}` AS `r`" . " ON `p`.`{$columnName}` = `r`.`{$refColumnName}`" . " WHERE `r`.`{$refColumnName}` IS NULL"; $this->raw_query($sql); } else if (strtoupper($onDelete) == 'SET NULL') { $sql = "UPDATE `{$tableName}` AS `p`" . " LEFT JOIN `{$refTableName}` AS `r`" . " ON `p`.`{$columnName}` = `r`.`{$refColumnName}`" . " SET `p`.`{$columnName}`=NULL" . " WHERE `r`.`{$refColumnName}` IS NULL"; $this->raw_query($sql); } return $this; } /** * Add foreign key to table. If FK with same name exist - it will be deleted * * @param string $fkName foreign key name * @param string $tableName main table name * @param string $keyName main table field name * @param string $refTableName refered table name * @param string $refKeyName refered table field name * @param string $onUpdate on update statement * @param string $onDelete on delete statement * @param bool $purge * @return mixed */ public function addConstraint($fkName, $tableName, $columnName, $refTableName, $refColumnName, $onDelete = 'cascade', $onUpdate = 'cascade', $purge = false) { if (substr($fkName, 0, 3) != 'FK_') { $fkName = 'FK_' . $fkName; } $this->dropForeignKey($tableName, $fkName); if ($purge) { $this->purgeOrphanRecords($tableName, $columnName, $refTableName, $refColumnName, $onDelete); } $sql = 'ALTER TABLE `'.$tableName.'` ADD CONSTRAINT `'.$fkName.'`' . ' FOREIGN KEY (`'.$columnName.'`) REFERENCES `'.$refTableName.'` (`'.$refColumnName.'`)'; if (!is_null($onDelete)) { $sql .= ' ON DELETE ' . strtoupper($onDelete); } if (!is_null($onUpdate)) { $sql .= ' ON UPDATE ' . strtoupper($onUpdate); } $this->resetDdlCache($tableName); return $this->raw_query($sql); } /** * Check table column exist * * @param string $tableName * @param string $columnName * @param string $schemaName * @return bool */ public function tableColumnExists($tableName, $columnName, $schemaName = null) { $describe = $this->describeTable($tableName, $schemaName); foreach ($describe as $column) { if ($column['COLUMN_NAME'] == $columnName) { return true; } } return false; } /** * Add new column to table * * @param string $tableName * @param string $columnName * @param string $definition * @return bool */ public function addColumn($tableName, $columnName, $definition) { if ($this->tableColumnExists($tableName, $columnName)) { return true; } $sql = sprintf('ALTER TABLE %s ADD COLUMN %s %s', $this->quoteIdentifier($tableName), $this->quoteIdentifier($columnName), $definition ); $result = $this->raw_query($sql); $this->resetDdlCache($tableName); return $result; } /** * Delete table column * * @param string $tableName * @param string $columnName * @param string $shemaName * @return bool */ public function dropColumn($tableName, $columnName, $shemaName = null) { if (!$this->tableColumnExists($tableName, $columnName, $shemaName)) { return true; } $alterDrop = array(); $foreignKeys = $this->getForeignKeys($tableName, $shemaName); foreach ($foreignKeys as $fkProp) { if ($fkProp['COLUMN_NAME'] == $columnName) { $alterDrop[] = sprintf('DROP FOREIGN KEY %s', $this->quoteIdentifier($fkProp['FK_NAME'])); } } $alterDrop[] = sprintf('DROP COLUMN %s', $this->quoteIdentifier($columnName)); $sql = sprintf('ALTER TABLE %s %s', $this->quoteIdentifier($this->_getTableName($tableName, $shemaName)), join(', ', $alterDrop)); $this->resetDdlCache($tableName, $shemaName); return $this->raw_query($sql); } /** * Change column * * @param string $tableName * @param string $oldColumnName * @param string $newColumnName * @param string $definition * @param bool $showStatus * * @return mixed */ public function changeColumn($tableName, $oldColumnName, $newColumnName, $definition, $showStatus = false) { if (!$this->tableColumnExists($tableName, $oldColumnName)) { throw new Exception(sprintf('Column "%s" does not exists on table "%s"', $oldColumnName, $tableName)); } $sql = sprintf('ALTER TABLE %s CHANGE COLUMN %s %s %s', $this->quoteIdentifier($tableName), $this->quoteIdentifier($oldColumnName), $this->quoteIdentifier($newColumnName), $definition); $result = $this->raw_query($sql); if ($showStatus) { $this->showTableStatus($tableName); } $this->resetDdlCache($tableName); return $result; } /** * Modify column defination or position * * @param string $tableName * @param string $columnName * @param string $definition * @param bool $showStatus * * @return mixed */ public function modifyColumn($tableName, $columnName, $definition, $showStatus = false) { if (!$this->tableColumnExists($tableName, $columnName)) { throw new Exception(sprintf('Column "%s" does not exists on table "%s"', $columnName, $tableName)); } $sql = sprintf('ALTER TABLE %s MODIFY COLUMN %s %s', $this->quoteIdentifier($tableName), $this->quoteIdentifier($columnName), $definition); $result = $this->raw_query($sql); if ($showStatus) { $this->showTableStatus($tableName); } $this->resetDdlCache($tableName); return $result; } /** * Show table status * * @param string $tableName * @return array */ public function showTableStatus($tableName) { $sql = $this->quoteInto('SHOW TABLE STATUS LIKE ?', $tableName); return $this->raw_fetchRow($sql); } /** * Retrieve table index key list * * @deprecated use getIndexList( * @param string $tableName * @param string $schemaName * @return array */ public function getKeyList($tableName, $schemaName = null) { $keyList = array(); $indexList = $this->getIndexList($tableName, $schemaName); foreach ($indexList as $indexProp) { $keyList[$indexProp['KEY_NAME']] = $indexProp['COLUMNS_LIST']; } return $keyList; } /** * Retrieve Create Table SQL * * @param string $tableName * @param string $schemaName * @return string */ public function getCreateTable($tableName, $schemaName = null) { $cacheKey = $this->_getTableName($tableName, $schemaName); $ddl = $this->loadDdlCache($cacheKey, self::DDL_CREATE); if ($ddl === false) { $sql = sprintf('SHOW CREATE TABLE %s', $this->quoteIdentifier($tableName)); $ddl = $this->raw_fetchRow($sql, 'Create Table'); $this->saveDdlCache($cacheKey, self::DDL_CREATE, $ddl); } return $ddl; } /** * Retrieve the foreign keys descriptions for a table. * * The return value is an associative array keyed by the UPPERCASE foreign key, * as returned by the RDBMS. * * The value of each array element is an associative array * with the following keys: * * FK_NAME => string; original foreign key name * SCHEMA_NAME => string; name of database or schema * TABLE_NAME => string; * COLUMN_NAME => string; column name * REF_SCHEMA_NAME => string; name of reference database or schema * REF_TABLE_NAME => string; reference table name * REF_COLUMN_NAME => string; reference column name * ON_DELETE => string; action type on delete row * ON_UPDATE => string; action type on update row * * @param string $tableName * @param string $schemaName * @return array */ public function getForeignKeys($tableName, $schemaName = null) { $cacheKey = $this->_getTableName($tableName, $schemaName); $ddl = $this->loadDdlCache($cacheKey, self::DDL_FOREIGN_KEY); if ($ddl === false) { $ddl = array(); $createSql = $this->getCreateTable($tableName, $schemaName); // collect CONSTRAINT $regExp = '#,\s+CONSTRAINT `([^`]*)` FOREIGN KEY \(`([^`]*)`\) ' . 'REFERENCES (`[^`]*\.)?`([^`]*)` \(`([^`]*)`\)' . '( ON DELETE (RESTRICT|CASCADE|SET NULL|NO ACTION))?' . '( ON UPDATE (RESTRICT|CASCADE|SET NULL|NO ACTION))?#'; $matches = array(); preg_match_all($regExp, $createSql, $matches, PREG_SET_ORDER); foreach ($matches as $match) { $ddl[strtoupper($match[1])] = array( 'FK_NAME' => $match[1], 'SCHEMA_NAME' => $schemaName, 'TABLE_NAME' => $tableName, 'COLUMN_NAME' => $match[2], 'REF_SHEMA_NAME' => isset($match[3]) ? $match[3] : $schemaName, 'REF_TABLE_NAME' => $match[4], 'REF_COLUMN_NAME' => $match[5], 'ON_DELETE' => isset($match[6]) ? $match[7] : '', 'ON_UPDATE' => isset($match[8]) ? $match[9] : '' ); } $this->saveDdlCache($cacheKey, self::DDL_FOREIGN_KEY, $ddl); } return $ddl; } /** * Retrieve table index information * * The return value is an associative array keyed by the UPPERCASE index key, * as returned by the RDBMS. * * The value of each array element is an associative array * with the following keys: * * SCHEMA_NAME => string; name of database or schema * TABLE_NAME => string; name of the table * KEY_NAME => string; the original index name * COLUMNS_LIST => array; array of index column names * INDEX_TYPE => string; create index type * INDEX_METHOD => string; index method using * type => string; see INDEX_TYPE * fields => array; see COLUMNS_LIST * * @param string $tableName * @param string $schemaName * @return array */ public function getIndexList($tableName, $schemaName = null) { $cacheKey = $this->_getTableName($tableName, $schemaName); $ddl = $this->loadDdlCache($cacheKey, self::DDL_INDEX); if ($ddl === false) { $ddl = array(); $sql = sprintf('SHOW INDEX FROM %s', $this->quoteIdentifier($this->_getTableName($tableName, $schemaName))); foreach ($this->fetchAll($sql) as $row) { $fieldKeyName = 'Key_name'; $fieldNonUnique = 'Non_unique'; $fieldColumn = 'Column_name'; $fieldIndexType = 'Index_type'; if ($row[$fieldKeyName] == 'PRIMARY') { $indexType = 'primary'; } else if ($row[$fieldNonUnique] == 0) { $indexType = 'unique'; } else if ($row[$fieldIndexType] == 'FULLTEXT') { $indexType = 'fulltext'; } else { $indexType = 'index'; } $upperKeyName = strtoupper($row[$fieldKeyName]); if (isset($ddl[$upperKeyName])) { $ddl[$upperKeyName]['fields'][] = $row[$fieldColumn]; // for compatible $ddl[$upperKeyName]['COLUMNS_LIST'][] = $row[$fieldColumn]; } else { $ddl[$upperKeyName] = array( 'SCHEMA_NAME' => $schemaName, 'TABLE_NAME' => $tableName, 'KEY_NAME' => $row[$fieldKeyName], 'COLUMNS_LIST' => array($row[$fieldColumn]), 'INDEX_TYPE' => strtoupper($indexType), 'INDEX_METHOD' => $row[$fieldIndexType], 'type' => $indexType, // for compatible 'fields' => array($row[$fieldColumn]) // for compatible ); } } $this->saveDdlCache($cacheKey, self::DDL_INDEX, $ddl); } return $ddl; } /** * Add Index Key * * @param string $tableName * @param string $indexName * @param string|array $fields * @param string $indexType * @return */ public function addKey($tableName, $indexName, $fields, $indexType = 'index') { $columns = $this->describeTable($tableName); $keyList = $this->getKeyList($tableName); $sql = 'ALTER TABLE '.$this->quoteIdentifier($tableName); if (isset($keyList[$indexName])) { $sql .= ' DROP INDEX ' . $this->quoteIdentifier($indexName) . ','; } if (!is_array($fields)) { $fields = array($fields); } $fieldSql = array(); foreach ($fields as $field) { if (!isset($columns[$field])) { $msg = sprintf('There is no field "%s" that you are trying to create an index on "%s"', $field, $tableName); throw new Exception($msg); } $fieldSql[] = $this->quoteIdentifier($field); } $fieldSql = join(',', $fieldSql); switch (strtolower($indexType)) { case 'primary': $condition = 'PRIMARY KEY'; break; case 'unique': $condition = 'UNIQUE ' . $this->quoteIdentifier($indexName); break; case 'fulltext': $condition = 'FULLTEXT ' . $this->quoteIdentifier($indexName); break; default: $condition = 'INDEX ' . $this->quoteIdentifier($indexName); break; } $sql .= ' ADD ' . $condition . ' (' . $fieldSql . ')'; $cycle = true; while ($cycle === true) { try { $result = $this->raw_query($sql); $cycle = false; } catch (PDOException $e) { if (in_array(strtolower($indexType), array('primary', 'unique'))) { $match = array(); if (preg_match('#SQLSTATE\[23000\]: [^:]+: 1062[^\']+\'([\d-\.]+)\'#', $e->getMessage(), $match)) { $ids = explode('-', $match[1]); $this->_removeDuplicateEntry($tableName, $fields, $ids); continue; } } throw $e; } catch (Exception $e) { throw $e; } } $this->resetDdlCache($tableName); return $result; } /** * Remove duplicate entry for create key * * @param string $table * @param array $fields * @param array $ids * @return Varien_Db_Adapter_Pdo_Mysql */ protected function _removeDuplicateEntry($table, $fields, $ids) { $where = array(); $i = 0; foreach ($fields as $field) { $where[] = $this->quoteInto($field . '=?', $ids[$i]); $i ++; } if (!$where) { return $this; } $whereCond = join(' AND ', $where); $sql = sprintf('SELECT COUNT(*) as `cnt` FROM `%s` WHERE %s', $table, $whereCond); if ($cnt = $this->raw_fetchRow($sql, 'cnt')) { $sql = sprintf('DELETE FROM `%s` WHERE %s LIMIT %d', $table, $whereCond, $cnt - 1 ); $this->raw_query($sql); } return $this; } /** * Creates and returns a new Zend_Db_Select object for this adapter. * * @return Varien_Db_Select */ public function select() { return new Varien_Db_Select($this); } /** * Start debug timer * * @return Varien_Db_Adapter_Pdo_Mysql */ protected function _debugTimer() { if ($this->_debug) { $this->_debugTimer = microtime(true); } return $this; } /** * Logging debug information * * @param int $type * @param string $sql * @param array $bind * @param Zend_Db_Statement_Pdo $result * @return Varien_Db_Adapter_Pdo_Mysql */ protected function _debugStat($type, $sql, $bind = array(), $result = null) { if (!$this->_debug) { return $this; } $code = '## ' . getmypid() . ' ## '; $nl = "\n"; $time = sprintf('%.4f', microtime(true) - $this->_debugTimer); if (!$this->_logAllQueries && $time < $this->_logQueryTime) { return $this; } switch ($type) { case self::DEBUG_CONNECT: $code .= 'CONNECT' . $nl; break; case self::DEBUG_TRANSACTION: $code .= 'TRANSACTION ' . $sql . $nl; break; case self::DEBUG_QUERY: $code .= 'QUERY' . $nl; $code .= 'SQL: ' . $sql . $nl; if ($bind) { $code .= 'BIND: ' . print_r($bind, true) . $nl; } if ($result instanceof Zend_Db_Statement_Pdo) { $code .= 'AFF: ' . $result->rowCount() . $nl; } break; } $code .= 'TIME: ' . $time . $nl; if ($this->_logCallStack) { $code .= 'TRACE: ' . Varien_Debug::backtrace(true, false) . $nl; } $code .= $nl; $this->_debugWriteToFile($code); return $this; } /** * Write exception and thow * * @param Exception $e * @throws Exception */ protected function _debugException(Exception $e) { if (!$this->_debug) { throw $e; } $nl = "\n"; $code = 'EXCEPTION ' . $nl . $e . $nl . $nl; $this->_debugWriteToFile($code); throw $e; } /** * Debug write to file process * * @param string $str */ protected function _debugWriteToFile($str) { if (!$this->_debugIoAdapter) { $this->_debugIoAdapter = new Varien_Io_File(); $dir = $this->_debugIoAdapter->dirname($this->_debugFile); $this->_debugIoAdapter->checkAndCreateFolder($dir); $this->_debugIoAdapter->open(array('path' => $dir)); $this->_debugFile = basename($this->_debugFile); } $this->_debugIoAdapter->streamOpen($this->_debugFile, 'a'); $this->_debugIoAdapter->streamLock(); $this->_debugIoAdapter->streamWrite($str); $this->_debugIoAdapter->streamUnlock(); $this->_debugIoAdapter->streamClose(); } /** * Quotes a value and places into a piece of text at a placeholder. * * Method revrited for handle empty arrays in value param * * @param string $text The text with a placeholder. * @param mixed $value The value to quote. * @param string $type OPTIONAL SQL datatype * @param integer $count OPTIONAL count of placeholders to replace * @return string An SQL-safe quoted value placed into the orignal text. */ public function quoteInto($text, $value, $type = null, $count = null) { if (is_array($value) && empty($value)) { $value = new Zend_Db_Expr('NULL'); } return parent::quoteInto($text, $value, $type, $count); } /** * Retrieve ddl cache name * * @param string $tableName * @param string $schemaName */ protected function _getTableName($tableName, $schemaName = null) { return ($schemaName ? $schemaName . '.' : '') . $tableName; } /** * Retrieve Id for cache * * @param string $tableKey * @param int $ddlType * @return string */ protected function _getCacheId($tableKey, $ddlType) { return sprintf('%s_%s_%s', self::DDL_CACHE_PREFIX, $tableKey, $ddlType); } /** * Load DDL data from cache * Return false if cache does not exists * * @param string $tableCacheKey the table cache key * @param int $ddlType the DDL constant * @return string|array|int|false */ public function loadDdlCache($tableCacheKey, $ddlType) { if (!$this->_isDdlCacheAllowed) { return false; } if (isset($this->_ddlCache[$ddlType][$tableCacheKey])) { return $this->_ddlCache[$ddlType][$tableCacheKey]; } if ($this->_cacheAdapter instanceof Zend_Cache_Core) { $cacheId = $this->_getCacheId($tableCacheKey, $ddlType); $data = $this->_cacheAdapter->load($cacheId); if ($data !== false) { $data = unserialize($data); $this->_ddlCache[$ddlType][$tableCacheKey] = $data; } return $data; } return false; } /** * Save DDL data into cache * * @param string $tableCacheKey * @param int $ddlType * @return Varien_Db_Adapter_Pdo_Mysql */ public function saveDdlCache($tableCacheKey, $ddlType, $data) { if (!$this->_isDdlCacheAllowed) { return $this; } $this->_ddlCache[$ddlType][$tableCacheKey] = $data; if ($this->_cacheAdapter instanceof Zend_Cache_Core) { $cacheId = $this->_getCacheId($tableCacheKey, $ddlType); $data = serialize($data); $this->_cacheAdapter->save($data, $cacheId, array(self::DDL_CACHE_TAG)); } return $this; } /** * Reset cached DDL data from cache * if table name is null - reset all cached DDL data * * @param string $tableName * @param string $schemaName OPTIONAL * @return Varien_Db_Adapter_Pdo_Mysql */ public function resetDdlCache($tableName = null, $schemaName = null) { if (!$this->_isDdlCacheAllowed) { return $this; } if (is_null($tableName)) { $this->_ddlCache = array(); if ($this->_cacheAdapter instanceof Zend_Cache_Core) { $this->_cacheAdapter->clean(Zend_Cache::CLEANING_MODE_MATCHING_TAG, array(self::DDL_CACHE_TAG)); } } else { $cacheKey = $this->_getTableName($tableName, $schemaName); $ddlTypes = array(self::DDL_DESCRIBE, self::DDL_CREATE, self::DDL_INDEX, self::DDL_FOREIGN_KEY); foreach ($ddlTypes as $ddlType) { unset($this->_ddlCache[$ddlType][$cacheKey]); } if ($this->_cacheAdapter instanceof Zend_Cache_Core) { foreach ($ddlTypes as $ddlType) { $cacheId = $this->_getCacheId($cacheKey, $ddlType); $this->_cacheAdapter->remove($cacheId); } } } return $this; } /** * Disallow DDL caching * @return Varien_Db_Adapter_Pdo_Mysql */ public function disallowDdlCache() { $this->_isDdlCacheAllowed = false; return $this; } /** * Allow DDL caching * @return Varien_Db_Adapter_Pdo_Mysql */ public function allowDdlCache() { $this->_isDdlCacheAllowed = true; return $this; } /** * Returns the column descriptions for a table. * * The return value is an associative array keyed by the column name, * as returned by the RDBMS. * * The value of each array element is an associative array * with the following keys: * * SCHEMA_NAME => string; name of database or schema * TABLE_NAME => string; * COLUMN_NAME => string; column name * COLUMN_POSITION => number; ordinal position of column in table * DATA_TYPE => string; SQL datatype name of column * DEFAULT => string; default expression of column, null if none * NULLABLE => boolean; true if column can have nulls * LENGTH => number; length of CHAR/VARCHAR * SCALE => number; scale of NUMERIC/DECIMAL * PRECISION => number; precision of NUMERIC/DECIMAL * UNSIGNED => boolean; unsigned property of an integer type * PRIMARY => boolean; true if column is part of the primary key * PRIMARY_POSITION => integer; position of column in primary key * IDENTITY => integer; true if column is auto-generated with unique values * * @param string $tableName * @param string $schemaName OPTIONAL * @return array */ public function describeTable($tableName, $schemaName = null) { $cacheKey = $this->_getTableName($tableName, $schemaName); $ddl = $this->loadDdlCache($cacheKey, self::DDL_DESCRIBE); if ($ddl === false) { $ddl = parent::describeTable($tableName, $schemaName); $this->saveDdlCache($cacheKey, self::DDL_DESCRIBE, $ddl); } return $ddl; } /** * Truncate table * * @param string $tableName * @param string $schemaName * @return Varien_Db_Adapter_Pdo_Mysql */ public function truncate($tableName, $schemaName = null) { $tableName = $this->_getTableName($tableName, $schemaName); $sql = sprintf('TRUNCATE %s', $this->quoteIdentifier($tableName)); $this->raw_query($sql); return $this; } /** * Change table storage engine * * @param string $tableName * @param string $engine * @param string $schemaName * @return mixed */ public function changeTableEngine($tableName, $engine, $schemaName = null) { $sql = sprintf('ALTER TABLE %s ENGINE=%s', $this->quoteIdentifier($this->_getTableName($tableName, $schemaName)), $engine); return $this->raw_query($sql); } /** * Inserts a table row with specified data. * * @param mixed $table The table to insert data into. * @param array $data Column-value pairs or array of column-value pairs. * @param array $fields update fields pairs or values * @return int The number of affected rows. */ public function insertOnDuplicate($table, array $data, array $fields = array()) { // extract and quote col names from the array keys $row = reset($data); // get first element from data array $bind = array(); // SQL bind array $values = array(); if (is_array($row)) { // Array of column-value pairs $cols = array_keys($row); foreach ($data as $row) { $line = array(); if (array_diff($cols, array_keys($row))) { throw new Varien_Exception('Invalid data for insert'); } foreach ($row as $val) { if ($val instanceof Zend_Db_Expr) { $line[] = $val->__toString(); } else { $line[] = '?'; $bind[] = $val; } } $values[] = sprintf('(%s)', join(',', $line)); } unset($row); } else { // Column-value pairs $cols = array_keys($data); $line = array(); foreach ($data as $val) { if ($val instanceof Zend_Db_Expr) { $line[] = $val->__toString(); } else { $line[] = '?'; $bind[] = $val; } } $values[] = sprintf('(%s)', join(',', $line)); } $updateFields = array(); if (empty($fields)) { $fields = $cols; } // quote column names $cols = array_map(array($this, 'quoteIdentifier'), $cols); // prepare ON DUPLICATE KEY conditions foreach ($fields as $k => $v) { $field = $value = null; if (!is_numeric($k)) { $field = $this->quoteIdentifier($k); if ($v instanceof Zend_Db_Expr) { $value = $v->__toString(); } else if (is_string($v)) { $value = 'VALUES('.$this->quoteIdentifier($v).')'; } else if (is_numeric($v)) { $value = $this->quoteInto('?', $v); } } else if (is_string($v)) { $field = $this->quoteIdentifier($v); $value = 'VALUES('.$field.')'; } if ($field && $value) { $updateFields[] = "{$field}={$value}"; } } // build the statement $sql = "INSERT INTO " . $this->quoteIdentifier($table, true) . ' (' . implode(', ', $cols) . ') ' . 'VALUES ' . implode(', ', $values); if ($updateFields) { $sql .= " ON DUPLICATE KEY UPDATE " . join(', ', $updateFields); } // execute the statement and return the number of affected rows $stmt = $this->query($sql, array_values($bind)); $result = $stmt->rowCount(); return $result; } /** * Inserts a table multiply rows with specified data. * * @param mixed $table The table to insert data into. * @param array $data Column-value pairs or array of Column-value pairs. * @return int The number of affected rows. */ public function insertMultiple($table, array $data) { $row = reset($data); // support insert syntaxes if (!is_array($row)) { return $this->insert($table, $data); } // validate data array $cols = array_keys($row); $insertArray = array(); foreach ($data as $row) { $line = array(); if (array_diff($cols, array_keys($row))) { throw new Varien_Exception('Invalid data for insert'); } foreach ($cols as $field) { $line[] = $row[$field]; } $insertArray[] = $line; } unset($row); return $this->insertArray($table, $cols, $insertArray); } /** * Insert array to table based on columns definition * * @param string $table * @param array $columns * @param array $data * @return int */ public function insertArray($table, array $columns, array $data) { $vals = array(); $bind = array(); $columnsCount = count($columns); foreach ($data as $row) { if ($columnsCount != count($row)) { throw new Varien_Exception('Invalid data for insert'); } $line = array(); if ($columnsCount == 1) { if ($row instanceof Zend_Db_Expr) { $line = $row->__toString(); } else { $line = '?'; $bind[] = $row; } $vals[] = sprintf('(%s)', $line); } else { foreach ($row as $value) { if ($value instanceof Zend_Db_Expr) { $line[] = $value->__toString(); } else { $line[] = '?'; $bind[] = $value; } } $vals[] = sprintf('(%s)', join(',', $line)); } } // build the statement $columns = array_map(array($this, 'quoteIdentifier'), $columns); $sql = sprintf("INSERT INTO %s (%s) VALUES%s", $this->quoteIdentifier($table, true), implode(',', $columns), implode(', ', $vals)); // execute the statement and return the number of affected rows $stmt = $this->query($sql, $bind); $result = $stmt->rowCount(); return $result; } /** * Set cache adapter * * @param Zend_Cache_Backend_Interface $adapter * @return Varien_Db_Adapter_Pdo_Mysql */ public function setCacheAdapter($adapter) { $this->_cacheAdapter = $adapter; return $this; } /** * Return DDL Table object * * @param string $tableName the table name * @return Varien_Db_Ddl_Table */ public function newTable($tableName = null) { $table = new Varien_Db_Ddl_Table(); if (!is_null($tableName)) { $table->setName($tableName); } return $table; } /** * Create table * * @param Varien_Db_Ddl_Table $table * @throws Zend_Db_Exception * @return Zend_Db_Pdo_Statement */ public function createTable(Varien_Db_Ddl_Table $table) { $sqlFragment = array_merge( $this->_getColumnsDefinition($table), $this->_getIndexesDefinition($table), $this->_getForeignKeysDefinition($table) ); $tableOptions = $this->_getOptionsDefination($table); $sql = sprintf("CREATE TABLE %s (\n%s\n) %s", $this->quoteIdentifier($table->getName()), implode(",\n", $sqlFragment), implode(" ", $tableOptions)); return $this->query($sql); } /** * Retrieve columns and primary keys definition array for create table * * @param Varien_Db_Ddl_Table $table * @return array */ protected function _getColumnsDefinition(Varien_Db_Ddl_Table $table) { $definition = array(); $primary = array(); $columns = $table->getColumns(); if (empty($columns)) { throw new Zend_Db_Exception('Table columns are not defined'); } $dataTypes = array( Varien_Db_Ddl_Table::TYPE_BOOLEAN => 'bool', Varien_Db_Ddl_Table::TYPE_TINYINT => 'tinyint', Varien_Db_Ddl_Table::TYPE_SMALLINT => 'smallint', Varien_Db_Ddl_Table::TYPE_INTEGER => 'int', Varien_Db_Ddl_Table::TYPE_BIGINT => 'bigint', Varien_Db_Ddl_Table::TYPE_DOUBLE => 'double', Varien_Db_Ddl_Table::TYPE_FLOAT => 'float', Varien_Db_Ddl_Table::TYPE_REAL => 'real', Varien_Db_Ddl_Table::TYPE_DECIMAL => 'decimal', Varien_Db_Ddl_Table::TYPE_NUMERIC => 'decimal', Varien_Db_Ddl_Table::TYPE_DATE => 'date', Varien_Db_Ddl_Table::TYPE_TIME => 'time', Varien_Db_Ddl_Table::TYPE_TIMESTAMP => 'timestamp', Varien_Db_Ddl_Table::TYPE_CHAR => 'char', Varien_Db_Ddl_Table::TYPE_VARCHAR => 'varchar', Varien_Db_Ddl_Table::TYPE_LONGVARCHAR => 'text', Varien_Db_Ddl_Table::TYPE_CLOB => 'longtext', Varien_Db_Ddl_Table::TYPE_BINARY => 'blob', Varien_Db_Ddl_Table::TYPE_VARBINARY => 'mediumblob', Varien_Db_Ddl_Table::TYPE_LONGVARBINARY => 'longblob', Varien_Db_Ddl_Table::TYPE_BLOB => 'longblob', ); foreach ($columns as $columnData) { $cType = $dataTypes[$columnData['COLUMN_TYPE']]; $cUnsigned = $columnData['UNSIGNED'] ? ' unsigned' : ''; $cIsNull = $columnData['NULLABLE'] === false ? ' NOT NULL' : ''; $cDefault = ''; if (!is_null($columnData['DEFAULT'])) { $cDefault = $this->quoteInto(' default ?', $columnData['DEFAULT']); } // column size switch ($columnData['COLUMN_TYPE']) { case Varien_Db_Ddl_Table::TYPE_TINYINT: case Varien_Db_Ddl_Table::TYPE_SMALLINT: case Varien_Db_Ddl_Table::TYPE_INTEGER: case Varien_Db_Ddl_Table::TYPE_BIGINT: if (is_numeric($columnData['LENGTH'])) { $cType .= sprintf('(%d)', $columnData['LENGTH']); } break; case Varien_Db_Ddl_Table::TYPE_DECIMAL: case Varien_Db_Ddl_Table::TYPE_NUMERIC: $cType .= sprintf('(%d,%d)', $columnData['SCALE'], $columnData['PRECISION']); break; case Varien_Db_Ddl_Table::TYPE_CHAR: case Varien_Db_Ddl_Table::TYPE_VARCHAR: $cType .= sprintf('(%d)', $columnData['LENGTH']); break; } if ($columnData['PRIMARY']) { $primary[$columnData['COLUMN_NAME']] = $columnData['PRIMARY_POSITION']; } $definition[] = sprintf(' %s %s%s%s%s', $this->quoteIdentifier($columnData['COLUMN_NAME']), $cType, $cUnsigned, $cIsNull, $cDefault ); } // PRIMARY KEY if (!empty($primary)) { asort($primary, SORT_NUMERIC); $primary = array_map(array($this, 'quoteIdentifier'), array_keys($primary)); $definition[] = sprintf(' PRIMARY KEY (%s)', join(', ', $primary)); } return $definition; } /** * Retrieve table indexes definition array for create table * * @param Varien_Db_Ddl_Table $table * @return array */ protected function _getIndexesDefinition(Varien_Db_Ddl_Table $table) { $definition = array(); $indexes = $table->getIndexes(); if (!empty($indexes)) { foreach ($indexes as $indexData) { if ($indexData['UNIQUE']) { $indexType = 'UNIQUE'; } else if (!empty($indexData['TYPE'])) { $indexType = $indexData['TYPE']; } else { $indexType = 'KEY'; } $columns = array(); foreach ($indexData['COLUMNS'] as $columnData) { $column = $this->quoteIdentifier($columnData['NAME']); if (!empty($columnData['SIZE'])) { $column .= sprintf('(%d)', $columnData['SIZE']); } $columns[] = $column; } $definition[] = sprintf(' %s %s (%s)', $indexType, $this->quoteIdentifier($indexData['INDEX_NAME']), join(', ', $columns) ); } } return $definition; } /** * Retrieve table foreign keys definition array for create table * * @param Varien_Db_Ddl_Table $table * @return array */ protected function _getForeignKeysDefinition(Varien_Db_Ddl_Table $table) { $definition = array(); $relations = $table->getForeignKeys(); if (!empty($relations)) { foreach ($relations as $fkData) { switch ($fkData['ON_DELETE']) { case Varien_Db_Ddl_Table::ACTION_CASCADE: case Varien_Db_Ddl_Table::ACTION_RESTRICT: case Varien_Db_Ddl_Table::ACTION_SET_NULL: $onDelete = $fkData['ON_DELETE']; break; default: $onDelete = Varien_Db_Ddl_Table::ACTION_NO_ACTION; } switch ($fkData['ON_UPDATE']) { case Varien_Db_Ddl_Table::ACTION_CASCADE: case Varien_Db_Ddl_Table::ACTION_RESTRICT: case Varien_Db_Ddl_Table::ACTION_SET_NULL: $onUpdate = $fkData['ON_UPDATE']; break; default: $onUpdate = Varien_Db_Ddl_Table::ACTION_NO_ACTION; } $definition[] = sprintf(' CONSTRAINT %s FOREIGN KEY (%s) REFERENCES %s (%s) ON DELETE %s ON UPDATE %s', $this->quoteIdentifier($fkData['FK_NAME']), $this->quoteIdentifier($fkData['COLUMN_NAME']), $this->quoteIdentifier($fkData['REF_TABLE_NAME']), $this->quoteIdentifier($fkData['REF_COLUMN_NAME']), $onDelete, $onUpdate ); } } return $definition; } /** * Retrieve table options definition array for create table * * @param Varien_Db_Ddl_Table $table * @return array */ protected function _getOptionsDefination(Varien_Db_Ddl_Table $table) { $definition = array(); $tableProps = array( 'type' => 'ENGINE=%s', 'checksum' => 'CHECKSUM=%d', 'auto_increment' => 'AUTO_INCREMENT=%d', 'avg_row_length' => 'AVG_ROW_LENGTH=%d', 'comment' => 'COMMENT=\'%s\'', 'max_rows' => 'MAX_ROWS=%d', 'min_rows' => 'MIN_ROWS=%d', 'delay_key_write' => 'DELAY_KEY_WRITE=%d', 'row_format' => 'row_format=%s', 'charset' => 'charset=%s', 'collate' => 'COLLATE=%s' ); foreach ($tableProps as $key => $mask) { $v = $table->getOption($key); if (!is_null($v)) { $definition[] = sprintf($mask, $v); } } return $definition; } }