*/ class Varien_Data_Collection_Db extends Varien_Data_Collection { /** * DB connection * * @var Zend_Db_Adapter_Abstract */ protected $_conn; /** * Select oblect * * @var Zend_Db_Select */ protected $_select; /** * Cache configuration array * * @var array */ protected $_cacheConf = null; /** * Identifier fild name for collection items * * Can be used by collections with items without defined * * @var string */ protected $_idFieldName; /** * List of binded variables for select * * @var array */ protected $_bindParams = array(); /** * All collection data array * Used for getData method * * @var array */ protected $_data = null; /** * Fields map for corellation names & real selected fields * * @var array */ protected $_map = null; /** * Database's statement for fetch item one by one * * @var Zend_Db_Statement_Pdo */ protected $_fetchStmt = null; public function __construct($conn=null) { parent::__construct(); if (!is_null($conn)) { $this->setConnection($conn); } } /** * Add variable to bind list * * @param string $name * @param mixed $value * @return Varien_Data_Collection_Db */ public function addBindParam($name, $value) { $this->_bindParams[$name] = $value; return $this; } /** * Initialize collection cache * * @param $object * @param string $idPrefix * @param array $tags * @return Varien_Data_Collection_Db */ public function initCache($object, $idPrefix, $tags) { $this->_cacheConf = array( 'object' => $object, 'prefix' => $idPrefix, 'tags' => $tags ); return $this; } /** * Specify collection objects id field name * * @param string $fieldName * @return Varien_Data_Collection_Db */ protected function _setIdFieldName($fieldName) { $this->_idFieldName = $fieldName; return $this; } /** * Id field name getter * * @return string */ public function getIdFieldName() { return $this->_idFieldName; } /** * Get collection item identifier * * @param Varien_Object $item * @return mixed */ protected function _getItemId(Varien_Object $item) { if ($field = $this->getIdFieldName()) { return $item->getData($field); } return parent::_getItemId($item); } /** * Set database connection adapter * * @param Zend_Db_Adapter_Abstract $conn * @return Varien_Data_Collection_Db */ public function setConnection($conn) { if (!$conn instanceof Zend_Db_Adapter_Abstract) { throw new Zend_Exception('dbModel read resource does not implement Zend_Db_Adapter_Abstract'); } $this->_conn = $conn; $this->_select = $this->_conn->select(); return $this; } /** * Get Zend_Db_Select instance * * @return Varien_Db_Select */ public function getSelect() { return $this->_select; } /** * Retrieve connection object * * @return Zend_Db_Adapter_Abstract */ public function getConnection() { return $this->_conn; } /** * Get collection size * * @return int */ public function getSize() { if (is_null($this->_totalRecords)) { $sql = $this->getSelectCountSql(); $this->_totalRecords = $this->getConnection()->fetchOne($sql, $this->_bindParams); } return intval($this->_totalRecords); } /** * Get SQL for get record count * * @return Varien_Db_Select */ public function getSelectCountSql() { $this->_renderFilters(); $countSelect = clone $this->getSelect(); $countSelect->reset(Zend_Db_Select::ORDER); $countSelect->reset(Zend_Db_Select::LIMIT_COUNT); $countSelect->reset(Zend_Db_Select::LIMIT_OFFSET); $countSelect->reset(Zend_Db_Select::COLUMNS); $countSelect->columns('COUNT(*)'); return $countSelect; } /** * Get sql select string or object * * @param bool $stringMode * @return string || Zend_Db_Select */ function getSelectSql($stringMode = false) { if ($stringMode) { return $this->_select->__toString(); } return $this->_select; } /** * Add select order * * @param string $field * @param string $direction * @return Varien_Data_Collection_Db */ public function setOrder($field, $direction = self::SORT_ORDER_DESC) { return $this->_setOrder($field, $direction); } /** * self::setOrder() alias * * @param string $field * @param string $direction * @return Varien_Data_Collection_Db */ public function addOrder($field, $direction = self::SORT_ORDER_DESC) { return $this->_setOrder($field, $direction); } /** * Add select order to the beginning * * @param string $field * @param string $direction * @return Varien_Data_Collection_Db */ public function unshiftOrder($field, $direction = self::SORT_ORDER_DESC) { return $this->_setOrder($field, $direction, true); } /** * Add ORDERBY to the end or to the beginning * * @param string $field * @param string $direction * @param bool $unshift * @return Varien_Data_Collection_Db */ private function _setOrder($field, $direction, $unshift = false) { $field = (string)$this->_getMappedField($field); $direction = (strtoupper($direction) == self::SORT_ORDER_ASC) ? self::SORT_ORDER_ASC : self::SORT_ORDER_DESC; // emulate associative unshift if ($unshift) { $orders = array($field => new Zend_Db_Expr($field . ' ' . $direction)); foreach ($this->_orders as $key => $expression) { if (!isset($orders[$key])) { $orders[$key] = $expression; } } $this->_orders = $orders; } else { $this->_orders[$field] = new Zend_Db_Expr($field . ' ' . $direction); } return $this; } /** * Render sql select conditions * * @return Varien_Data_Collection_Db */ protected function _renderFilters() { if ($this->_isFiltersRendered) { return $this; } $this->_renderFiltersBefore(); foreach ($this->_filters as $filter) { switch ($filter['type']) { case 'or' : $condition = $this->_conn->quoteInto($filter['field'].'=?', $filter['value']); $this->_select->orWhere($condition); break; case 'string' : $this->_select->where($filter['value']); break; case 'public': $field = $this->_getMappedField($filter['field']); $condition = $filter['value']; $this->_select->where( $this->_getConditionSql($field, $condition), null, Varien_Db_Select::TYPE_CONDITION ); break; default: $condition = $this->_conn->quoteInto($filter['field'].'=?', $filter['value']); $this->_select->where($condition); } } $this->_isFiltersRendered = true; return $this; } /** * Hook for operations before rendering filters */ protected function _renderFiltersBefore() { } /** * Add field filter to collection * * @see self::_getConditionSql for $condition * @param string $field * @param null|string|array $condition * @return Mage_Eav_Model_Entity_Collection_Abstract */ public function addFieldToFilter($field, $condition=null) { $mappedField = $this->_getMappedField($field); $quotedField = $mappedField; if ($mappedField === $field) { $quotedField = $this->getConnection()->quoteIdentifier($field); } $this->_select->where($this->_getConditionSql($quotedField, $condition), null, Varien_Db_Select::TYPE_CONDITION); return $this; } /** * Try to get mapped field name for filter to collection * * @param string * @return string */ protected function _getMappedField($field) { $mappedFiled = $field; $mapper = $this->_getMapper(); if (isset($mapper['fields'][$field])) { $mappedFiled = $mapper['fields'][$field]; } return $mappedFiled; } protected function _getMapper() { if (isset($this->_map)) { return $this->_map; } else { return false; } } /** * Build SQL statement for condition * * If $condition integer or string - exact value will be filtered * * If $condition is array is - one of the following structures is expected: * - array("from"=>$fromValue, "to"=>$toValue) * - array("like"=>$likeValue) * - array("neq"=>$notEqualValue) * - array("in"=>array($inValues)) * - array("nin"=>array($notInValues)) * * If non matched - sequential array is expected and OR conditions * will be built using above mentioned structure * * @param string|array $fieldName Field name must be already escaped with Varien_Db_Adapter_Interface::quoteIdentifier() * @param integer|string|array $condition * @return string */ protected function _getConditionSql($fieldName, $condition) { if (is_array($fieldName)) { $orSql = array(); foreach ($fieldName as $key=>$name) { if (isset($condition[$key])) { $orSql[] = '('.$this->_getConditionSql($name, $condition[$key]).')'; } else { //if nothing passed as condition adding empty condition to avoid sql error $orSql[] = $this->getConnection()->quoteInto("$name = ?", ''); } } $sql = '('. join(' or ', $orSql) .')'; return $sql; } $sql = ''; $fieldName = $this->_getConditionFieldName($fieldName); if (is_array($condition)) { if (isset($condition['from']) || isset($condition['to'])) { if (isset($condition['from'])) { if (empty($condition['date'])) { if ( empty($condition['datetime'])) { $from = $condition['from']; } else { $from = $this->getConnection()->convertDateTime($condition['from']); } } else { $from = $this->getConnection()->convertDate($condition['from']); } $sql.= $this->getConnection()->quoteInto("$fieldName >= ?", $from); } if (isset($condition['to'])) { $sql.= empty($sql) ? '' : ' and '; if (empty($condition['date'])) { if ( empty($condition['datetime'])) { $to = $condition['to']; } else { $to = $this->getConnection()->convertDateTime($condition['to']); } } else { $to = $this->getConnection()->convertDate($condition['to']); } $sql.= $this->getConnection()->quoteInto("$fieldName <= ?", $to); } } elseif (isset($condition['eq'])) { $sql = $this->getConnection()->quoteInto("$fieldName = ?", $condition['eq']); } elseif (isset($condition['neq'])) { $sql = $this->getConnection()->quoteInto("$fieldName != ?", $condition['neq']); } elseif (isset($condition['like'])) { $sql = $this->getConnection()->quoteInto("$fieldName like ?", $condition['like']); } elseif (isset($condition['nlike'])) { $sql = $this->getConnection()->quoteInto("$fieldName not like ?", $condition['nlike']); } elseif (isset($condition['in'])) { $sql = $this->getConnection()->quoteInto("$fieldName in (?)", $condition['in']); } elseif (isset($condition['nin'])) { $sql = $this->getConnection()->quoteInto("$fieldName not in (?)", $condition['nin']); } elseif (isset($condition['is'])) { $sql = $this->getConnection()->quoteInto("$fieldName is ?", $condition['is']); } elseif (isset($condition['notnull'])) { $sql = "$fieldName is NOT NULL"; } elseif (isset($condition['null'])) { $sql = "$fieldName is NULL"; } elseif (isset($condition['moreq'])) { $sql = $this->getConnection()->quoteInto("$fieldName >= ?", $condition['moreq']); } elseif (isset($condition['gt'])) { $sql = $this->getConnection()->quoteInto("$fieldName > ?", $condition['gt']); } elseif (isset($condition['lt'])) { $sql = $this->getConnection()->quoteInto("$fieldName < ?", $condition['lt']); } elseif (isset($condition['gteq'])) { $sql = $this->getConnection()->quoteInto("$fieldName >= ?", $condition['gteq']); } elseif (isset($condition['lteq'])) { $sql = $this->getConnection()->quoteInto("$fieldName <= ?", $condition['lteq']); } elseif (isset($condition['finset'])) { $sql = $this->getConnection()->quoteInto("find_in_set(?,$fieldName)", $condition['finset']); } else { $orSql = array(); foreach ($condition as $orCondition) { $orSql[] = "(".$this->_getConditionSql($fieldName, $orCondition).")"; } $sql = "(".join(" or ", $orSql).")"; } } else { $sql = $this->getConnection()->quoteInto("$fieldName = ?", (string)$condition); } return $sql; } protected function _getConditionFieldName($fieldName) { return $fieldName; } /** * Render sql select orders * * @return Varien_Data_Collection_Db */ protected function _renderOrders() { $ordersInSelect = $this->_select->getPart(Zend_Db_Select::ORDER); foreach ($this->_orders as $orderExpr) { if (!in_array($orderExpr, $ordersInSelect)) { $this->_select->order($orderExpr); } } return $this; } /** * Render sql select limit * * @return Varien_Data_Collection_Db */ protected function _renderLimit() { if($this->_pageSize){ $this->_select->limitPage($this->getCurPage(), $this->_pageSize); } return $this; } /** * Set select distinct * * @param bool $flag */ public function distinct($flag) { $this->_select->distinct($flag); return $this; } /** * Before load action * * @return Varien_Data_Collection_Db */ protected function _beforeLoad() { return $this; } /** * Load data * * @return Varien_Data_Collection_Db */ public function load($printQuery = false, $logQuery = false) { if ($this->isLoaded()) { return $this; } $this->_beforeLoad(); $this->_renderFilters() ->_renderOrders() ->_renderLimit(); $this->printLogQuery($printQuery, $logQuery); $data = $this->getData(); $this->resetData(); if (is_array($data)) { foreach ($data as $row) { $item = $this->getNewEmptyItem(); if ($this->getIdFieldName()) { $item->setIdFieldName($this->getIdFieldName()); } $item->addData($row); $this->addItem($item); } } $this->_setIsLoaded(); $this->_afterLoad(); return $this; } /** * Returns a collection item that corresponds to the fetched row * and moves the internal data pointer ahead * * return Varien_Object|bool */ public function fetchItem() { if (null === $this->_fetchStmt) { $this->_fetchStmt = $this->getConnection() ->query($this->getSelect()); } $data = $this->_fetchStmt->fetch(); if (!empty($data) && is_array($data)) { $item = $this->getNewEmptyItem(); if ($this->getIdFieldName()) { $item->setIdFieldName($this->getIdFieldName()); } $item->setData($data); return $item; } return false; } /** * Convert items array to hash for select options * unsing fetchItem method * * The difference between _toOptionHash() and this one is that this * method fetch items one by one and does not load all collection items at once * return items hash * array($value => $label) * * @see fetchItem() * * @param string $valueField * @param string $labelField * @return array */ protected function _toOptionHashOptimized($valueField='id', $labelField='name') { $result = array(); while ($item = $this->fetchItem()) { $result[$item->getData($valueField)] = $item->getData($labelField); } return $result; } /** * Get all data array for collection * * @return array */ public function getData() { if ($this->_data === null) { $this->_renderFilters() ->_renderOrders() ->_renderLimit(); $this->_data = $this->_fetchAll($this->_select); $this->_afterLoadData(); } return $this->_data; } /** * Proces loaded collection data * * @return Varien_Data_Collection_Db */ protected function _afterLoadData() { return $this; } /** * Reset loaded for collection data array * * @return Varien_Data_Collection_Db */ public function resetData() { $this->_data = null; return $this; } protected function _afterLoad() { return $this; } public function loadData($printQuery = false, $logQuery = false) { return $this->load($printQuery, $logQuery); } /** * Print and/or log query * * @param boolean $printQuery * @param boolean $logQuery * @return Varien_Data_Collection_Db */ public function printLogQuery($printQuery = false, $logQuery = false, $sql = null) { if ($printQuery) { echo is_null($sql) ? $this->getSelect()->__toString() : $sql; } if ($logQuery){ Mage::log(is_null($sql) ? $this->getSelect()->__toString() : $sql); } return $this; } /** * Reset collection * * @return Varien_Data_Collection_Db */ protected function _reset() { $this->getSelect()->reset(); $this->_initSelect(); $this->_setIsLoaded(false); $this->_items = array(); $this->_data = null; return $this; } /** * Fetch collection data * * @param Zend_Db_Select $select * @return array */ protected function _fetchAll($select) { if ($this->_canUseCache()) { $data = $this->_loadCache($select); if ($data) { $data = unserialize($data); } else { $data = $this->getConnection()->fetchAll($select, $this->_bindParams); $this->_saveCache($data, $select); } } else { $data = $this->getConnection()->fetchAll($select, $this->_bindParams); } return $data; } /** * Load cached data for select * * @param Zend_Db_Select $select * @return string | false */ protected function _loadCache($select) { $data = false; $object = $this->_getCacheInstance(); if ($object) { $data = $object->load($this->_getSelectCacheId($select)); } return $data; } /** * Save collection data to cache * * @param array $data * @param Zend_Db_Select $select * @return unknown_type */ protected function _saveCache($data, $select) { $object = $this->_getCacheInstance(); $object->save(serialize($data), $this->_getSelectCacheId($select), $this->_getCacheTags()); return $this; } /** * Check if cache can be used for collection data * * @return bool */ protected function _canUseCache() { return $this->_getCacheInstance(); } /** * Get cache identifier base on select * * @param Zend_Db_Select $select * @return string */ protected function _getSelectCacheId($select) { $id = md5($select->__toString()); if (isset($this->_cacheConf['prefix'])) { $id = $this->_cacheConf['prefix'].'_'.$id; } return $id; } /** * Retrieve cache instance * * @return Zend_Cache_Core */ protected function _getCacheInstance() { if (isset($this->_cacheConf['object'])) { return $this->_cacheConf['object']; } return false; } /** * Get cache tags list * * @return array */ protected function _getCacheTags() { if (isset($this->_cacheConf['tags'])) { return $this->_cacheConf['tags']; } return array(); } /** * Add filter to Map * * @param string $filter * @param string $alias * @param string $group default 'fields' * * @return Varien_Data_Collection_Db */ public function addFilterToMap($filter, $alias, $group = 'fields') { if (is_null($this->_map)) { $this->_map = array($group => array()); } else if(is_null($this->_map[$group])) { $this->_map[$group] = array(); } $this->_map[$group][$filter] = $alias; return $this; } }