* Copyright 2005-2007, Cake Software Foundation, Inc. * 1785 E. Sahara Avenue, Suite 490-204 * Las Vegas, Nevada 89104 * * Licensed under The MIT License * Redistributions of files must retain the above copyright notice. * * @filesource * @copyright Copyright 2005-2007, Cake Software Foundation, Inc. * @link http://www.cakefoundation.org/projects/info/cakephp CakePHP(tm) Project * @package cake * @subpackage cake.cake.libs.model.datasources * @since CakePHP(tm) v 0.10.5.1790 * @version $Revision$ * @modifiedby $LastChangedBy$ * @lastmodified $Date$ * @license http://www.opensource.org/licenses/mit-license.php The MIT License */ /** * DataSource base class * * Long description for file * * @package cake * @subpackage cake.cake.libs.model.datasources */ class DataSource extends Object { /** * Are we connected to the DataSource? * * @var boolean * @access public */ var $connected = false; /** * Print debug info? * * @var boolean * @access public */ var $debug = false; /** * Print full query debug info? * * @var boolean * @access public */ var $fullDebug = false; /** * Error description of last query * * @var unknown_type * @access public */ var $error = null; /** * String to hold how many rows were affected by the last SQL operation. * * @var string * @access public */ var $affected = null; /** * Number of rows in current resultset * * @var int * @access public */ var $numRows = null; /** * Time the last query took * * @var int * @access public */ var $took = null; /** * Enter description here... * * @var boolean */ var $cacheSources = true; /** * Enter description here... * * @var array * @access private */ var $_result = null; /** * Queries count. * * @var int * @access private */ var $_queriesCnt = 0; /** * Total duration of all queries. * * @var unknown_type * @access private */ var $_queriesTime = null; /** * Log of queries executed by this DataSource * * @var unknown_type * @access private */ var $_queriesLog = array(); /** * Maximum number of items in query log, to prevent query log taking over * too much memory on large amounts of queries -- I we've had problems at * >6000 queries on one system. * * @var int Maximum number of queries in the queries log. * @access private */ var $_queriesLogMax = 200; /** * Caches serialzed results of executed queries * * @var array Maximum number of queries in the queries log. * @access private */ var $_queryCache = array(); /** * The default configuration of a specific DataSource * * @var array * @access public */ var $_baseConfig = array(); /** * Holds references to descriptions loaded by the DataSource * * @var array * @access private */ var $__descriptions = array(); /** * Holds a list of sources (tables) contained in the DataSource * * @var array * @access protected */ var $_sources = null; /** * A reference to the physical connection of this DataSource * * @var array * @access public */ var $connection = null; /** * The DataSource configuration * * @var array * @access public */ var $config = array(); /** * The DataSource configuration key name * * @var string * @access public */ var $configKeyName = null; /** * Whether or not this DataSource is in the middle of a transaction * * @var boolean * @access protected */ var $_transactionStarted = false; /** * Constructor. */ function __construct() { parent::__construct(); if (func_num_args() > 0) { $this->setConfig(func_get_arg(0)); } } /** * Datsrouce Query abstraction * * @return resource Result resource identifier */ function query() { $args = func_get_args(); $fields = null; $order = null; $limit = null; $page = null; $recursive = null; if (count($args) == 1) { return $this->fetchAll($args[0]); } elseif (count($args) > 1 && (strpos(low($args[0]), 'findby') === 0 || strpos(low($args[0]), 'findallby') === 0)) { $params = $args[1]; if (strpos(strtolower($args[0]), 'findby') === 0) { $all = false; $field = Inflector::underscore(preg_replace('/findBy/i', '', $args[0])); } else { $all = true; $field = Inflector::underscore(preg_replace('/findAllBy/i', '', $args[0])); } $or = (strpos($field, '_or_') !== false); if ($or) { $field = explode('_or_', $field); } else { $field = explode('_and_', $field); } $off = count($field) - 1; if (isset($params[1 + $off])) { $fields = $params[1 + $off]; } if (isset($params[2 + $off])) { $order = $params[2 + $off]; } if (!array_key_exists(0, $params)) { return false; } $c = 0; $query = array(); foreach ($field as $f) { if (!is_array($params[$c]) && !empty($params[$c]) && $params[$c] !== true && $params[$c] !== false) { $query[$args[2]->name . '.' . $f] = '= ' . $params[$c]; } else { $query[$args[2]->name . '.' . $f] = $params[$c]; } $c++; } if ($or) { $query = array('OR' => $query); } if ($all) { if (isset($params[3 + $off])) { $limit = $params[3 + $off]; } if (isset($params[4 + $off])) { $page = $params[4 + $off]; } if (isset($params[5 + $off])) { $recursive = $params[5 + $off]; } return $args[2]->findAll($query, $fields, $order, $limit, $page, $recursive); } else { if (isset($params[3 + $off])) { $recursive = $params[3 + $off]; } return $args[2]->find($query, $fields, $order, $recursive); } } else { if (isset($args[1]) && $args[1] === true) { return $this->fetchAll($args[0], true); } return $this->fetchAll($args[0], false); } } /** * Returns an array of all result rows for a given SQL query. * Returns false if no rows matched. * * @param string $sql SQL statement * @param boolean $cache Enables returning/storing cached query results * @return array Array of resultset rows, or false if no rows matched */ function fetchAll($sql, $cache = true, $modelName = null) { if ($cache && isset($this->_queryCache[$sql])) { if (preg_match('/^\s*select/i', $sql)) { return $this->_queryCache[$sql]; } } if ($this->execute($sql)) { $out = array(); while ($item = $this->fetchRow()) { $out[] = $item; } if ($cache) { if (strpos(trim(strtolower($sql)), 'select') !== false) { $this->_queryCache[$sql] = $out; } } return $out; } else { return false; } } /** * Caches/returns cached results for child instances * * @return array */ function listSources($data = null) { if ($this->cacheSources === false) { return null; } if ($this->_sources != null) { return $this->_sources; } if (Configure::read() > 0) { $expires = "+30 seconds"; } else { $expires = "+999 days"; } if ($data != null) { $data = serialize($data); } $filename = ConnectionManager::getSourceName($this) . '_' . preg_replace("/[^A-Za-z0-9_-]/", "_", $this->config['database']) . '_list'; $new = cache('models' . DS . $filename, $data, $expires); if ($new != null) { $new = unserialize($new); $this->_sources = $new; } return $new; } /** * Convenience method for DboSource::listSources(). Returns source names in lowercase. * * @return array */ function sources() { $return = array_map('strtolower', $this->listSources()); return $return; } /** * Returns a Model description (metadata) or null if none found. * * @param Model $model * @return mixed */ function describe($model) { if ($this->cacheSources === false) { return null; } if (isset($this->__descriptions[$model->tablePrefix.$model->table])) { return $this->__descriptions[$model->tablePrefix.$model->table]; } $cache = $this->__cacheDescription($model->tablePrefix.$model->table); if ($cache !== null) { $this->__descriptions[$model->tablePrefix.$model->table] =& $cache; return $cache; } return null; } /** * Converts column types to basic types * * @param string $real Real column type (i.e. "varchar(255)") * @return string Abstract column type (i.e. "string") */ function column($real) { return false; } /** * To-be-overridden in subclasses. * * @param unknown_type $model * @param unknown_type $fields * @param unknown_type $values * @return unknown */ function create(&$model, $fields = null, $values = null) { return false; } /** * To-be-overridden in subclasses. * * @param unknown_type $model * @param unknown_type $queryData * @return unknown */ function read(&$model, $queryData = array()) { return false; } /** * To-be-overridden in subclasses. * * @param unknown_type $model * @param unknown_type $fields * @param unknown_type $values * @return unknown */ function update(&$model, $fields = null, $values = null) { return false; } /** * To-be-overridden in subclasses. * * @param unknown_type $model * @param unknown_type $id */ function delete(&$model, $id = null) { if ($id == null) { $id = $model->id; } } /** * Returns the ID generated from the previous INSERT operation. * * @param unknown_type $source * @return in */ function lastInsertId($source = null) { return false; } /** * Returns the ID generated from the previous INSERT operation. * * @param unknown_type $source * @return in */ function lastNumRows($source = null) { return false; } /** * Returns the ID generated from the previous INSERT operation. * * @param unknown_type $source * @return in */ function lastAffected($source = null) { return false; } /** * Returns true if the DataSource supports the given interface (method) * * @param string $interface The name of the interface (method) * @return boolean True on success */ function isInterfaceSupported($interface) { $methods = get_class_methods(get_class($this)); $methods = strtolower(implode('|', $methods)); $methods = explode('|', $methods); $return = in_array(strtolower($interface), $methods); return $return; } /** * Sets the configuration for the DataSource * * @param array $config The configuration array * @return void */ function setConfig($config) { if (is_array($this->_baseConfig)) { $this->config = $this->_baseConfig; foreach ($config as $key => $val) { $this->config[$key] = $val; } } } /** * Cache the DataSource description * * @param string $object The name of the object (model) to cache * @param mixed $data The description of the model, usually a string or array * @return void */ function __cacheDescription($object, $data = null) { if ($this->cacheSources === false) { return null; } if (Configure::read() > 0) { $expires = "+15 seconds"; } else { $expires = "+999 days"; } if ($data !== null) { $this->__descriptions[$object] =& $data; $cache = serialize($data); } else { $cache = null; } $new = cache('models' . DS . ConnectionManager::getSourceName($this) . '_' . $object, $cache, $expires); if ($new != null) { $new = unserialize($new); } return $new; } /** * Enter description here... * * @param unknown_type $query * @param unknown_type $data * @param unknown_type $association * @param unknown_type $assocData * @param Model $model * @param Model $linkModel * @param array $stack * @return unknown */ function insertQueryData($query, $data, $association, $assocData, &$model, &$linkModel, $stack) { $keys = array('{$__cakeID__$}', '{$__cakeForeignKey__$}'); foreach ($keys as $key) { $val = null; if (strpos($query, $key) !== false) { switch($key) { case '{$__cakeID__$}': if (isset($data[$model->name]) || isset($data[$association])) { if (isset($data[$model->name][$model->primaryKey])) { $val = $data[$model->name][$model->primaryKey]; } elseif (isset($data[$association][$model->primaryKey])) { $val = $data[$association][$model->primaryKey]; } } else { $found = false; foreach (array_reverse($stack) as $assoc) { if (isset($data[$assoc]) && isset($data[$assoc][$model->primaryKey])) { $val = $data[$assoc][$model->primaryKey]; $found = true; break; } } if (!$found) { $val = ''; } } break; case '{$__cakeForeignKey__$}': foreach ($model->__associations as $id => $name) { foreach ($model->$name as $assocName => $assoc) { if ($assocName === $association) { if (isset($assoc['foreignKey'])) { $foreignKey = $assoc['foreignKey']; if (isset($data[$model->name][$foreignKey])) { $val = $data[$model->name][$foreignKey]; } elseif (isset($data[$association][$foreignKey])) { $val = $data[$association][$foreignKey]; } else { $found = false; foreach (array_reverse($stack) as $assoc) { if (isset($data[$assoc]) && isset($data[$assoc][$foreignKey])) { $val = $data[$assoc][$foreignKey]; $found = true; break; } } if (!$found) { $val = ''; } } } break 3; } } } break; } if (empty($val) && $val !== '0') { return false; } $query = r($key, $this->value($val, $model->getColumnType($model->primaryKey)), $query); } } return $query; } /** * To-be-overridden in subclasses. * * @param unknown_type $model * @param unknown_type $key * @return unknown */ function resolveKey($model, $key) { return $model->name . $key; } /** * Closes the current datasource. * */ function __destruct() { if ($this->connected) { $this->close(); } } } ?>