* $connection->select()->from('user')->where('sAMAccountName = ?', 'icinga'); * * * @copyright Copyright (c) 2013 Icinga-Web Team * @author Icinga-Web Team * @package Icinga\Protocol\Ldap * @license http://www.gnu.org/copyleft/gpl.html GNU General Public License * @package Icinga\Protocol\Ldap */ class Query { /** * @var Connection */ protected $connection; /** * @var array */ protected $filters = array(); /** * @var array */ protected $fields = array(); /** * @var */ protected $limit_count; /** * @var */ protected $limit_offset; /** * @var array */ protected $sort_columns = array(); /** * @var */ protected $count; /** * Constructor * * @param \Icinga\Protocol\Ldap\Connection $connection LDAP Connection object * @return \Icinga\Protocol\Ldap\Query */ public function __construct(Connection $connection) { $this->connection = $connection; } /** * Count result set, ignoring limits * * @return int */ public function count() { if ($this->count === null) { $this->count = $this->connection->count($this); } return $this->count; } /** * Count result set, ignoring limits * * @param null $count * @param null $offset * @throws Exception * @return int */ public function limit($count = null, $offset = null) { if (!preg_match('~^\d+~', $count . $offset)) { throw new Exception( sprintf( 'Got invalid limit: %s, %s', $count, $offset ) ); } $this->limit_count = (int)$count; $this->limit_offset = (int)$offset; return $this; } /** * Whether a limit has been set * * @return boolean */ public function hasLimit() { return $this->limit_count !== null; } /** * Whether an offset (limit) has been set * * @return boolean */ public function hasOffset() { return $this->limit_offset > 0; } /** * Retrieve result limit * * @return int */ public function getLimit() { return $this->limit_count; } /** * Retrieve result offset * * @return int */ public function getOffset() { return $this->limit_offset; } /** * Fetch result as tree * * @return Node */ public function fetchTree() { $result = $this->fetchAll(); $sorted = array(); foreach ($result as $key => & $item) { $new_key = LdapUtils::implodeDN( array_reverse( LdapUtils::explodeDN( preg_replace( '/,' . preg_quote($this->connection->getDN(), '/') . '$/', '', $key ) ) ) ); $sorted[$new_key] = $key; } unset($groups); ksort($sorted); $tree = Root::forConnection($this->connection); foreach ($sorted as $sort_key => & $key) { $tree->createChildByDN($key, $result[$key]); } return $tree; } /** * Fetch result as an array of objects * * @return array */ public function fetchAll() { return $this->connection->fetchAll($this); } /** * Fetch first result row * * @return object */ public function fetchRow() { return $this->connection->fetchRow($this); } /** * Fetch first column value from first result row * * @return mixed */ public function fetchOne() { return $this->connection->fetchOne($this); } /** * Fetch a key/value list, first column is key, second is value * * @return array */ public function fetchPairs() { // STILL TODO!! return $this->connection->fetchPairs($this); } /** * Where to select (which fields) from * * This creates an objectClass filter * * @param $objectClass * @param array $fields * @return Query */ public function from($objectClass, $fields = array()) { $this->filters['objectClass'] = $objectClass; $this->fields = $fields; return $this; } /** * Add a new filter to the query * * @param string Column to search in * @param string Filter text (asterisks are allowed) * @return Query */ public function where($key, $val) { $this->filters[$key] = $val; return $this; } /** * Sort by given column * * TODO: Sort direction is not implemented yet * * @param string Order column * @param string Order direction * @return Query */ public function order($column, $direction = 'ASC') { $this->sort_columns[] = array($column, $direction); return $this; } /** * Retrieve a list of the desired fields * * @return array */ public function listFields() { return $this->fields; } /** * Retrieve a list containing current sort columns * * @return array */ public function getSortColumns() { return $this->sort_columns; } /** * Return a pagination adapter for the current query * * @param null $limit * @param null $page * @return \Zend_Paginator */ public function paginate($limit = null, $page = null) { if ($page === null || $limit === null) { $request = \Zend_Controller_Front::getInstance()->getRequest(); if ($page === null) { $page = $request->getParam('page', 0); } if ($limit === null) { $limit = $request->getParam('limit', 20); } } $paginator = new \Zend_Paginator( // TODO: Adapter doesn't fit yet: new QueryAdapter($this) ); $paginator->setItemCountPerPage($limit); $paginator->setCurrentPageNumber($page); return $paginator; } /** * Returns the LDAP filter that will be applied * * @string */ public function __toString() { return $this->render(); } /** * Returns the LDAP filter that will be applied * * @string */ protected function render() { $parts = array(); if ($this->filters['objectClass'] === null) { throw new Exception('Object class is mandatory'); } foreach ($this->filters as $key => $value) { $parts[] = sprintf( '%s=%s', LdapUtils::quoteForSearch($key), LdapUtils::quoteForSearch($value, true) ); } return '(&(' . implode(')(', $parts) . '))'; } /** * Destructor */ public function __destruct() { // To be on the safe side: unset($this->connection); } }