2014-06-06 08:43:13 +02:00
|
|
|
<?php
|
2014-07-15 13:39:22 +02:00
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
2014-06-06 08:43:13 +02:00
|
|
|
|
|
|
|
namespace Icinga\Data\Db;
|
|
|
|
|
|
|
|
use Icinga\Data\SimpleQuery;
|
|
|
|
use Icinga\Application\Benchmark;
|
2014-06-17 14:34:02 +02:00
|
|
|
use Icinga\Data\Filter\FilterChain;
|
|
|
|
use Icinga\Data\Filter\FilterExpression;
|
2014-06-06 08:43:13 +02:00
|
|
|
use Icinga\Data\Filter\FilterOr;
|
|
|
|
use Icinga\Data\Filter\FilterAnd;
|
|
|
|
use Icinga\Data\Filter\FilterNot;
|
2014-08-27 16:03:15 +02:00
|
|
|
use Icinga\Exception\IcingaException;
|
2014-06-06 08:43:13 +02:00
|
|
|
use Zend_Db_Select;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Database query class
|
|
|
|
*/
|
|
|
|
class DbQuery extends SimpleQuery
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var Zend_Db_Adapter_Abstract
|
|
|
|
*/
|
|
|
|
protected $db;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Select query
|
|
|
|
*
|
|
|
|
* @var Zend_Db_Select
|
|
|
|
*/
|
|
|
|
protected $select;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether to use a subquery for counting
|
|
|
|
*
|
|
|
|
* When the query is distinct or has a HAVING or GROUP BY clause this must be set to true
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
protected $useSubqueryCount = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the count maximum
|
|
|
|
*
|
|
|
|
* If the count maximum is set, count queries will not count more than that many rows. You should set this
|
|
|
|
* property only for really heavy queries.
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
protected $maxCount;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Count query result
|
|
|
|
*
|
|
|
|
* Count queries are only executed once
|
|
|
|
*
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
protected $count;
|
|
|
|
|
|
|
|
protected function init()
|
|
|
|
{
|
|
|
|
$this->db = $this->ds->getDbAdapter();
|
|
|
|
parent::init();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function where($condition, $value = null)
|
|
|
|
{
|
|
|
|
// $this->count = $this->select = null;
|
|
|
|
return parent::where($condition, $value);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function dbSelect()
|
|
|
|
{
|
|
|
|
if ($this->select === null) {
|
2014-06-17 14:34:02 +02:00
|
|
|
$this->select = $this->db->select()->from($this->target, array());
|
2014-06-06 08:43:13 +02:00
|
|
|
}
|
|
|
|
return clone $this->select;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the select query
|
|
|
|
*
|
|
|
|
* Applies order and limit if any
|
|
|
|
*
|
|
|
|
* @return Zend_Db_Select
|
|
|
|
*/
|
|
|
|
public function getSelectQuery()
|
|
|
|
{
|
|
|
|
$select = $this->dbSelect();
|
2014-08-29 13:41:41 +02:00
|
|
|
|
|
|
|
// Add order fields to select for postgres distinct queries (#6351)
|
|
|
|
if ($this->hasOrder()
|
|
|
|
&& $this->getDatasource()->getDbType() === 'pgsql'
|
|
|
|
&& $select->getPart(Zend_Db_Select::DISTINCT) === true) {
|
|
|
|
foreach ($this->getOrder() as $fieldAndDirection) {
|
|
|
|
list($alias, $field) = explode('.', $fieldAndDirection[0]);
|
|
|
|
$this->columns[$field] = $fieldAndDirection[0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-06 08:43:13 +02:00
|
|
|
$select->columns($this->columns);
|
|
|
|
$this->applyFilterSql($select);
|
|
|
|
|
|
|
|
if ($this->hasLimit() || $this->hasOffset()) {
|
|
|
|
$select->limit($this->getLimit(), $this->getOffset());
|
|
|
|
}
|
|
|
|
if ($this->hasOrder()) {
|
|
|
|
foreach ($this->getOrder() as $fieldAndDirection) {
|
|
|
|
$select->order(
|
|
|
|
$fieldAndDirection[0] . ' ' . $fieldAndDirection[1]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2014-08-29 13:41:41 +02:00
|
|
|
|
2014-06-06 08:43:13 +02:00
|
|
|
return $select;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function applyFilterSql($query)
|
|
|
|
{
|
|
|
|
$where = $this->renderFilter($this->filter);
|
|
|
|
if ($where !== '') {
|
|
|
|
$query->where($where);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function renderFilter($filter, $level = 0)
|
|
|
|
{
|
|
|
|
$str = '';
|
2014-06-17 14:34:02 +02:00
|
|
|
if ($filter instanceof FilterChain) {
|
2014-06-06 08:43:13 +02:00
|
|
|
if ($filter instanceof FilterAnd) {
|
|
|
|
$op = ' AND ';
|
|
|
|
} elseif ($filter instanceof FilterOr) {
|
|
|
|
$op = ' OR ';
|
|
|
|
} elseif ($filter instanceof FilterNot) {
|
|
|
|
$op = ' AND ';
|
|
|
|
$str .= ' NOT ';
|
|
|
|
} else {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
'Cannot render filter: %s',
|
|
|
|
$filter
|
|
|
|
);
|
2014-06-06 08:43:13 +02:00
|
|
|
}
|
|
|
|
$parts = array();
|
|
|
|
if (! $filter->isEmpty()) {
|
|
|
|
foreach ($filter->filters() as $f) {
|
|
|
|
$parts[] = $this->renderFilter($f, $level + 1);
|
|
|
|
}
|
|
|
|
if ($level > 0) {
|
|
|
|
$str .= ' (' . implode($op, $parts) . ') ';
|
|
|
|
} else {
|
|
|
|
$str .= implode($op, $parts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2014-06-17 14:34:02 +02:00
|
|
|
$str .= $this->whereToSql($filter->getColumn(), $filter->getSign(), $filter->getExpression());
|
2014-06-06 08:43:13 +02:00
|
|
|
}
|
2014-06-20 19:14:34 +02:00
|
|
|
|
2014-06-06 08:43:13 +02:00
|
|
|
return $str;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function escapeForSql($value)
|
|
|
|
{
|
|
|
|
// bindParam? bindValue?
|
|
|
|
if (is_array($value)) {
|
|
|
|
$ret = array();
|
|
|
|
foreach ($value as $val) {
|
|
|
|
$ret[] = $this->escapeForSql($val);
|
|
|
|
}
|
|
|
|
return implode(', ', $ret);
|
|
|
|
} else {
|
|
|
|
//if (preg_match('/^\d+$/', $value)) {
|
|
|
|
// return $value;
|
|
|
|
//} else {
|
|
|
|
return $this->db->quote($value);
|
|
|
|
//}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function escapeWildcards($value)
|
|
|
|
{
|
|
|
|
return preg_replace('/\*/', '%', $value);
|
|
|
|
}
|
|
|
|
|
2014-06-21 03:09:40 +02:00
|
|
|
protected function valueToTimestamp($value)
|
|
|
|
{
|
|
|
|
// We consider integers as valid timestamps. Does not work for URL params
|
|
|
|
if (ctype_digit($value)) {
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
$value = strtotime($value);
|
|
|
|
if (! $value) {
|
|
|
|
/*
|
|
|
|
NOTE: It's too late to throw exceptions, we might finish in __toString
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(sprintf(
|
2014-06-21 03:09:40 +02:00
|
|
|
'"%s" is not a valid time expression',
|
|
|
|
$value
|
|
|
|
));
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
2014-06-20 19:14:34 +02:00
|
|
|
protected function timestampForSql($value)
|
|
|
|
{
|
|
|
|
// TODO: do this db-aware
|
|
|
|
return $this->escapeForSql(date('Y-m-d H:i:s', $value));
|
|
|
|
}
|
|
|
|
|
2014-07-18 16:30:14 +02:00
|
|
|
/**
|
2014-07-18 16:48:52 +02:00
|
|
|
* Check for timestamp fields
|
2014-07-18 16:30:14 +02:00
|
|
|
*
|
|
|
|
* TODO: This is not here to do automagic timestamp stuff. One may
|
|
|
|
* override this function for custom voodoo, IdoQuery right now
|
|
|
|
* does. IMO we need to split whereToSql functionality, however
|
|
|
|
* I'd prefer to wait with this unless we understood how other
|
|
|
|
* backends will work. We probably should also rename this
|
|
|
|
* function to isTimestampColumn().
|
|
|
|
*
|
|
|
|
* @param string $field Field Field name to checked
|
|
|
|
* @return bool Whether this field expects timestamps
|
|
|
|
*/
|
|
|
|
public function isTimestamp($field)
|
|
|
|
{
|
2014-09-19 14:29:29 +02:00
|
|
|
return false;
|
2014-07-18 16:30:14 +02:00
|
|
|
}
|
|
|
|
|
2014-06-17 14:34:02 +02:00
|
|
|
public function whereToSql($col, $sign, $expression)
|
2014-06-06 08:43:13 +02:00
|
|
|
{
|
2014-06-17 14:34:02 +02:00
|
|
|
if ($this->isTimestamp($col)) {
|
2014-06-21 03:09:40 +02:00
|
|
|
$expression = $this->valueToTimestamp($expression);
|
2014-06-17 14:34:02 +02:00
|
|
|
}
|
|
|
|
if (is_array($expression) && $sign === '=') {
|
2014-06-06 08:43:13 +02:00
|
|
|
// TODO: Should we support this? Doesn't work for blub*
|
|
|
|
return $col . ' IN (' . $this->escapeForSql($expression) . ')';
|
|
|
|
} elseif (strpos($expression, '*') === false) {
|
2014-06-17 14:34:02 +02:00
|
|
|
return $col . ' ' . $sign . ' ' . $this->escapeForSql($expression);
|
2014-06-06 08:43:13 +02:00
|
|
|
} else {
|
|
|
|
return $col . ' LIKE ' . $this->escapeForSql($this->escapeWildcards($expression));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the count query
|
|
|
|
*
|
|
|
|
* @return Zend_Db_Select
|
|
|
|
*/
|
|
|
|
public function getCountQuery()
|
|
|
|
{
|
|
|
|
// TODO: there may be situations where we should clone the "select"
|
|
|
|
$count = $this->dbSelect();
|
|
|
|
|
|
|
|
$this->applyFilterSql($count);
|
|
|
|
if ($this->useSubqueryCount) {
|
|
|
|
$columns = array('cnt' => 'COUNT(*)');
|
|
|
|
return $this->db->select()->from($count, $columns);
|
|
|
|
}
|
|
|
|
if ($this->maxCount !== null) {
|
|
|
|
return $this->db->select()->from($count->limit($this->maxCount));
|
|
|
|
}
|
|
|
|
|
|
|
|
$count->columns(array('cnt' => 'COUNT(*)'));
|
|
|
|
return $count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Count all rows of the result set
|
|
|
|
*
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
public function count()
|
|
|
|
{
|
|
|
|
if ($this->count === null) {
|
|
|
|
Benchmark::measure('DB is counting');
|
|
|
|
$this->count = $this->db->fetchOne($this->getCountQuery());
|
|
|
|
Benchmark::measure('DB finished count');
|
|
|
|
}
|
|
|
|
return $this->count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the select and count query as a textual representation
|
|
|
|
*
|
|
|
|
* @return string A string containing the select and count query, using unix style newlines as linebreaks
|
|
|
|
*/
|
|
|
|
public function dump()
|
|
|
|
{
|
|
|
|
return "QUERY\n=====\n"
|
|
|
|
. $this->getSelectQuery()
|
|
|
|
. "\n\nCOUNT\n=====\n"
|
|
|
|
. $this->getCountQuery()
|
|
|
|
. "\n\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function __toString()
|
|
|
|
{
|
|
|
|
return (string) $this->getSelectQuery();
|
|
|
|
}
|
|
|
|
}
|