2012-11-09 23:18:50 +11:00
|
|
|
<?php defined('SYSPATH') OR die('No direct script access.');
|
2010-08-21 14:43:03 +10:00
|
|
|
/**
|
2012-11-09 23:18:50 +11:00
|
|
|
* Database query wrapper. See [Parameterized Statements](database/query/parameterized) for usage and examples.
|
2010-08-21 14:43:03 +10:00
|
|
|
*
|
|
|
|
* @package Kohana/Database
|
|
|
|
* @category Query
|
|
|
|
* @author Kohana Team
|
|
|
|
* @copyright (c) 2008-2009 Kohana Team
|
|
|
|
* @license http://kohanaphp.com/license
|
|
|
|
*/
|
|
|
|
class Kohana_Database_Query {
|
|
|
|
|
|
|
|
// Query type
|
|
|
|
protected $_type;
|
|
|
|
|
2012-11-09 23:18:50 +11:00
|
|
|
// Execute the query during a cache hit
|
|
|
|
protected $_force_execute = FALSE;
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
// Cache lifetime
|
2011-05-13 16:00:25 +10:00
|
|
|
protected $_lifetime = NULL;
|
2010-08-21 14:43:03 +10:00
|
|
|
|
|
|
|
// SQL statement
|
|
|
|
protected $_sql;
|
|
|
|
|
|
|
|
// Quoted query parameters
|
|
|
|
protected $_parameters = array();
|
|
|
|
|
|
|
|
// Return results as associative arrays or objects
|
|
|
|
protected $_as_object = FALSE;
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
// Parameters for __construct when using object results
|
|
|
|
protected $_object_params = array();
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
/**
|
|
|
|
* Creates a new SQL query of the specified type.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param integer $type query type: Database::SELECT, Database::INSERT, etc
|
|
|
|
* @param string $sql query string
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function __construct($type, $sql)
|
|
|
|
{
|
|
|
|
$this->_type = $type;
|
|
|
|
$this->_sql = $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the SQL query string.
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2012-11-09 23:18:50 +11:00
|
|
|
public function __toString()
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// Return the SQL string
|
|
|
|
return $this->compile(Database::instance());
|
|
|
|
}
|
|
|
|
catch (Exception $e)
|
|
|
|
{
|
2011-05-13 16:00:25 +10:00
|
|
|
return Kohana_Exception::text($e);
|
2010-08-21 14:43:03 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the type of the query.
|
|
|
|
*
|
|
|
|
* @return integer
|
|
|
|
*/
|
|
|
|
public function type()
|
|
|
|
{
|
|
|
|
return $this->_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enables the query to be cached for a specified amount of time.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param integer $lifetime number of seconds to cache, 0 deletes it from the cache
|
|
|
|
* @param boolean whether or not to execute the query during a cache hit
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return $this
|
2011-05-13 16:00:25 +10:00
|
|
|
* @uses Kohana::$cache_life
|
2010-08-21 14:43:03 +10:00
|
|
|
*/
|
2012-11-09 23:18:50 +11:00
|
|
|
public function cached($lifetime = NULL, $force = FALSE)
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
2011-05-13 16:00:25 +10:00
|
|
|
if ($lifetime === NULL)
|
|
|
|
{
|
|
|
|
// Use the global setting
|
|
|
|
$lifetime = Kohana::$cache_life;
|
|
|
|
}
|
|
|
|
|
2012-11-09 23:18:50 +11:00
|
|
|
$this->_force_execute = $force;
|
2010-08-21 14:43:03 +10:00
|
|
|
$this->_lifetime = $lifetime;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns results as associative arrays
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function as_assoc()
|
|
|
|
{
|
|
|
|
$this->_as_object = FALSE;
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
$this->_object_params = array();
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns results as objects
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $class classname or TRUE for stdClass
|
|
|
|
* @param array $params
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return $this
|
|
|
|
*/
|
2011-05-13 16:00:25 +10:00
|
|
|
public function as_object($class = TRUE, array $params = NULL)
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
|
|
|
$this->_as_object = $class;
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
if ($params)
|
|
|
|
{
|
|
|
|
// Add object parameters
|
|
|
|
$this->_object_params = $params;
|
|
|
|
}
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the value of a parameter in the query.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $param parameter key to replace
|
|
|
|
* @param mixed $value value to use
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function param($param, $value)
|
|
|
|
{
|
|
|
|
// Add or overload a new parameter
|
|
|
|
$this->_parameters[$param] = $value;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Bind a variable to a parameter in the query.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param string $param parameter key to replace
|
|
|
|
* @param mixed $var variable to use
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function bind($param, & $var)
|
|
|
|
{
|
|
|
|
// Bind a value to a variable
|
|
|
|
$this->_parameters[$param] =& $var;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add multiple parameters to the query.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param array $params list of parameters
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function parameters(array $params)
|
|
|
|
{
|
|
|
|
// Merge the new parameters in
|
|
|
|
$this->_parameters = $params + $this->_parameters;
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compile the SQL query and return it. Replaces any parameters with their
|
|
|
|
* given values.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param mixed $db Database instance or name of instance
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return string
|
|
|
|
*/
|
2012-11-09 23:18:50 +11:00
|
|
|
public function compile($db = NULL)
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
2012-11-09 23:18:50 +11:00
|
|
|
if ( ! is_object($db))
|
|
|
|
{
|
|
|
|
// Get the database instance
|
|
|
|
$db = Database::instance($db);
|
|
|
|
}
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
// Import the SQL locally
|
|
|
|
$sql = $this->_sql;
|
|
|
|
|
|
|
|
if ( ! empty($this->_parameters))
|
|
|
|
{
|
|
|
|
// Quote all of the values
|
|
|
|
$values = array_map(array($db, 'quote'), $this->_parameters);
|
|
|
|
|
|
|
|
// Replace the values in the SQL
|
|
|
|
$sql = strtr($sql, $values);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sql;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute the current query on the given database.
|
|
|
|
*
|
2012-11-09 23:18:50 +11:00
|
|
|
* @param mixed $db Database instance or name of instance
|
|
|
|
* @param string result object classname, TRUE for stdClass or FALSE for array
|
|
|
|
* @param array result object constructor arguments
|
2010-08-21 14:43:03 +10:00
|
|
|
* @return object Database_Result for SELECT queries
|
|
|
|
* @return mixed the insert id for INSERT queries
|
|
|
|
* @return integer number of affected rows for all other queries
|
|
|
|
*/
|
2012-11-09 23:18:50 +11:00
|
|
|
public function execute($db = NULL, $as_object = NULL, $object_params = NULL)
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
|
|
|
if ( ! is_object($db))
|
|
|
|
{
|
|
|
|
// Get the database instance
|
|
|
|
$db = Database::instance($db);
|
|
|
|
}
|
|
|
|
|
2012-11-09 23:18:50 +11:00
|
|
|
if ($as_object === NULL)
|
|
|
|
{
|
|
|
|
$as_object = $this->_as_object;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($object_params === NULL)
|
|
|
|
{
|
|
|
|
$object_params = $this->_object_params;
|
|
|
|
}
|
|
|
|
|
2010-08-21 14:43:03 +10:00
|
|
|
// Compile the SQL query
|
|
|
|
$sql = $this->compile($db);
|
|
|
|
|
2011-05-13 16:00:25 +10:00
|
|
|
if ($this->_lifetime !== NULL AND $this->_type === Database::SELECT)
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
|
|
|
// Set the cache key based on the database instance name and SQL
|
|
|
|
$cache_key = 'Database::query("'.$db.'", "'.$sql.'")';
|
|
|
|
|
2012-11-09 23:18:50 +11:00
|
|
|
// Read the cache first to delete a possible hit with lifetime <= 0
|
|
|
|
if (($result = Kohana::cache($cache_key, NULL, $this->_lifetime)) !== NULL
|
|
|
|
AND ! $this->_force_execute)
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
|
|
|
// Return a cached result
|
2012-11-09 23:18:50 +11:00
|
|
|
return new Database_Result_Cached($result, $sql, $as_object, $object_params);
|
2010-08-21 14:43:03 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Execute the query
|
2012-11-09 23:18:50 +11:00
|
|
|
$result = $db->query($this->_type, $sql, $as_object, $object_params);
|
2010-08-21 14:43:03 +10:00
|
|
|
|
2012-11-09 23:18:50 +11:00
|
|
|
if (isset($cache_key) AND $this->_lifetime > 0)
|
2010-08-21 14:43:03 +10:00
|
|
|
{
|
|
|
|
// Cache the result array
|
|
|
|
Kohana::cache($cache_key, $result->as_array(), $this->_lifetime);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End Database_Query
|