See Release Notes
Long Term Support Release
<?php /** * Copyright 2008-2017 Horde LLC (http://www.horde.org/) * * See the enclosed file LICENSE for license information (LGPL). If you * did not receive this file, see http://www.horde.org/licenses/lgpl21. * * @category Horde * @copyright 2008-2017 Horde LLC * @license http://www.horde.org/licenses/lgpl21 LGPL 2.1 * @package Imap_Client */ /** * Abstraction of the IMAP4rev1 search criteria (see RFC 3501 [6.4.4]). * Allows translation between abstracted search criteria and a generated IMAP * search criteria string suitable for sending to a remote IMAP server. * * @author Michael Slusarz <slusarz@horde.org> * @category Horde * @copyright 2008-2017 Horde LLC * @license http://www.horde.org/licenses/lgpl21 LGPL 2.1 * @package Imap_Client */ class Horde_Imap_Client_Search_Query implements Serializable { /** * Serialized version. */ const VERSION = 3; /** * Constants for dateSearch() */ const DATE_BEFORE = 'BEFORE'; const DATE_ON = 'ON'; const DATE_SINCE = 'SINCE'; /** * Constants for intervalSearch() */ const INTERVAL_OLDER = 'OLDER'; const INTERVAL_YOUNGER = 'YOUNGER'; /** * The charset of the search strings. All text strings must be in * this charset. By default, this is 'US-ASCII' (see RFC 3501 [6.4.4]). * * @var string */ protected $_charset = null; /** * The list of search params. * * @var array */ protected $_search = array(); /** * String representation: The IMAP search string. */ public function __toString() { try { $res = $this->build(null); return $res['query']->escape(); } catch (Exception $e) { return ''; } } /** * Sets the charset of the search text. * * @param string $charset The charset to use for the search. * @param boolean $convert Convert existing text values? * * @throws Horde_Imap_Client_Exception_SearchCharset */ public function charset($charset, $convert = true) { $oldcharset = $this->_charset; $this->_charset = Horde_String::upper($charset); if (!$convert || ($oldcharset == $this->_charset)) { return; } foreach (array('and', 'or') as $item) { if (isset($this->_search[$item])) { foreach ($this->_search[$item] as &$val) { $val->charset($charset, $convert); } } } // Unset the reference to avoid corrupting $this->_search below. unset($val); foreach (array('header', 'text') as $item) { if (isset($this->_search[$item])) { foreach ($this->_search[$item] as $key => $val) { $new_val = Horde_String::convertCharset($val['text'], $oldcharset, $this->_charset); if (Horde_String::convertCharset($new_val, $this->_charset, $oldcharset) != $val['text']) { throw new Horde_Imap_Client_Exception_SearchCharset($this->_charset); } $this->_search[$item][$key]['text'] = $new_val; } } } } /** * Builds an IMAP4rev1 compliant search string. * * @todo Change default of $exts to null. * * @param Horde_Imap_Client_Base $exts The server object this query will * be run on (@since 2.24.0), a * Horde_Imap_Client_Data_Capability * object (@since 2.24.0), or the * list of extensions present * on the server (@deprecated). * If null, all extensions are * assumed to be available. * * @return array An array with these elements: * - charset: (string) The charset of the search string. If null, no * text strings appear in query. * - exts: (array) The list of IMAP extensions used to create the * string. * - query: (Horde_Imap_Client_Data_Format_List) The IMAP search * command. * * @throws Horde_Imap_Client_Data_Format_Exception * @throws Horde_Imap_Client_Exception_NoSupportExtension */ public function build($exts = array()) { /* @todo: BC */ if (is_array($exts)) { $tmp = new Horde_Imap_Client_Data_Capability_Imap(); foreach ($exts as $key => $val) { $tmp->add($key, is_array($val) ? $val : null); } $exts = $tmp; } elseif (!is_null($exts)) { if ($exts instanceof Horde_Imap_Client_Base) { $exts = $exts->capability; } elseif (!($exts instanceof Horde_Imap_Client_Data_Capability)) { throw new InvalidArgumentException('Incorrect $exts parameter'); } } $temp = array( 'cmds' => new Horde_Imap_Client_Data_Format_List(), 'exts' => $exts, 'exts_used' => array() ); $cmds = &$temp['cmds']; $charset = $charset_cname = null; $default_search = true; $exts_used = &$temp['exts_used']; $ptr = &$this->_search; $charset_get = function ($c) use (&$charset, &$charset_cname) { $charset = is_null($c) ? 'US-ASCII' : strval($c); $charset_cname = ($charset === 'US-ASCII') ? 'Horde_Imap_Client_Data_Format_Astring' : 'Horde_Imap_Client_Data_Format_Astring_Nonascii'; }; $create_return = function ($charset, $exts_used, $cmds) { return array( 'charset' => $charset, 'exts' => array_keys(array_flip($exts_used)), 'query' => $cmds ); }; /* Do IDs check first. If there is an empty ID query (without a NOT * qualifier), the rest of this query is irrelevant since we already * know the search will return no results. */ if (isset($ptr['ids'])) { if (!count($ptr['ids']['ids']) && !$ptr['ids']['ids']->special) { if (empty($ptr['ids']['not'])) { /* This is a match on an empty list of IDs. We do need to * process any OR queries that may exist, since they are * independent of this result. */ if (isset($ptr['or'])) { $this->_buildAndOr( 'OR', $ptr['or'], $charset, $exts_used, $cmds ); } return $create_return($charset, $exts_used, $cmds); } /* If reached here, this a NOT search of an empty list. We can * safely discard this from the output. */ } else { $this->_addFuzzy(!empty($ptr['ids']['fuzzy']), $temp); if (!empty($ptr['ids']['not'])) { $cmds->add('NOT'); } if (!$ptr['ids']['ids']->sequence) { $cmds->add('UID'); } $cmds->add(strval($ptr['ids']['ids'])); } } if (isset($ptr['new'])) { $this->_addFuzzy(!empty($ptr['newfuzzy']), $temp); if ($ptr['new']) { $cmds->add('NEW'); unset($ptr['flag']['UNSEEN']); } else { $cmds->add('OLD'); } unset($ptr['flag']['RECENT']); } if (!empty($ptr['flag'])) { foreach ($ptr['flag'] as $key => $val) { $this->_addFuzzy(!empty($val['fuzzy']), $temp); $tmp = ''; if (empty($val['set'])) { // This is a 'NOT' search. All system flags but \Recent // have 'UN' equivalents. if ($key == 'RECENT') { $cmds->add('NOT'); } else { $tmp = 'UN'; } } if ($val['type'] == 'keyword') { $cmds->add(array( $tmp . 'KEYWORD', $key )); } else { $cmds->add($tmp . $key); } } } if (!empty($ptr['header'])) { /* The list of 'system' headers that have a specific search * query. */ $systemheaders = array( 'BCC', 'CC', 'FROM', 'SUBJECT', 'TO' ); foreach ($ptr['header'] as $val) { $this->_addFuzzy(!empty($val['fuzzy']), $temp); if (!empty($val['not'])) { $cmds->add('NOT'); } if (in_array($val['header'], $systemheaders)) { $cmds->add($val['header']); } else { $cmds->add(array( 'HEADER', new Horde_Imap_Client_Data_Format_Astring($val['header']) )); } $charset_get($this->_charset); $cmds->add( new $charset_cname(isset($val['text']) ? $val['text'] : '') ); } } if (!empty($ptr['text'])) { foreach ($ptr['text'] as $val) { $this->_addFuzzy(!empty($val['fuzzy']), $temp); if (!empty($val['not'])) { $cmds->add('NOT'); } $charset_get($this->_charset); $cmds->add(array( $val['type'], new $charset_cname($val['text']) )); } } if (!empty($ptr['size'])) { foreach ($ptr['size'] as $key => $val) { $this->_addFuzzy(!empty($val['fuzzy']), $temp); if (!empty($val['not'])) { $cmds->add('NOT'); } $cmds->add(array( $key, new Horde_Imap_Client_Data_Format_Number( empty($val['size']) ? 0 : $val['size'] ) )); } } if (!empty($ptr['date'])) { foreach ($ptr['date'] as $val) { $this->_addFuzzy(!empty($val['fuzzy']), $temp); if (!empty($val['not'])) { $cmds->add('NOT'); } if (empty($val['header'])) { $cmds->add($val['range']); } else { $cmds->add('SENT' . $val['range']); } $cmds->add($val['date']); } } if (!empty($ptr['within'])) { if (is_null($exts) || $exts->query('WITHIN')) { $exts_used[] = 'WITHIN'; } foreach ($ptr['within'] as $key => $val) { $this->_addFuzzy(!empty($val['fuzzy']), $temp); if (!empty($val['not'])) { $cmds->add('NOT'); } if (is_null($exts) || $exts->query('WITHIN')) { $cmds->add(array( $key, new Horde_Imap_Client_Data_Format_Number($val['interval']) )); } else { // This workaround is only accurate to within 1 day, due // to limitations with the IMAP4rev1 search commands. $cmds->add(array( ($key == self::INTERVAL_OLDER) ? self::DATE_BEFORE : self::DATE_SINCE, new Horde_Imap_Client_Data_Format_Date('now -' . $val['interval'] . ' seconds') )); } } } if (!empty($ptr['modseq'])) { if (!is_null($exts) && !$exts->query('CONDSTORE')) { throw new Horde_Imap_Client_Exception_NoSupportExtension('CONDSTORE'); } $exts_used[] = 'CONDSTORE'; $this->_addFuzzy(!empty($ptr['modseq']['fuzzy']), $temp); if (!empty($ptr['modseq']['not'])) { $cmds->add('NOT'); } $cmds->add('MODSEQ'); if (isset($ptr['modseq']['name'])) { $cmds->add(array( new Horde_Imap_Client_Data_Format_String($ptr['modseq']['name']), $ptr['modseq']['type'] )); } $cmds->add(new Horde_Imap_Client_Data_Format_Number($ptr['modseq']['value'])); } if (isset($ptr['prevsearch'])) { if (!is_null($exts) && !$exts->query('SEARCHRES')) { throw new Horde_Imap_Client_Exception_NoSupportExtension('SEARCHRES'); } $exts_used[] = 'SEARCHRES'; $this->_addFuzzy(!empty($ptr['prevsearchfuzzy']), $temp); if (!$ptr['prevsearch']) { $cmds->add('NOT'); } $cmds->add('$'); } // Add AND'ed queries if (!empty($ptr['and'])) { $default_search = $this->_buildAndOr( 'AND', $ptr['and'], $charset, $exts_used, $cmds ); } // Add OR'ed queries if (!empty($ptr['or'])) { $default_search = $this->_buildAndOr( 'OR', $ptr['or'], $charset, $exts_used, $cmds ); } // Default search is 'ALL' if ($default_search && !count($cmds)) { $cmds->add('ALL'); } return $create_return($charset, $exts_used, $cmds); } /** * Builds the AND/OR query. * * @param string $type 'AND' or 'OR'. * @param array $data Query data. * @param string &$charset Search charset. * @param array &$exts_used IMAP extensions used. * @param Horde_Imap_Client_Data_Format_List &$cmds Command list. * * @return boolean True if query might return results. */ protected function _buildAndOr($type, $data, &$charset, &$exts_used, &$cmds) { $results = false; foreach ($data as $val) { $ret = $val->build(); /* Empty sub-query. */ if (!count($ret['query'])) { switch ($type) { case 'AND': /* Any empty sub-query means that the query MUST return * no results. */ $cmds = new Horde_Imap_Client_Data_Format_List(); $exts_used = array(); return false; case 'OR': /* Skip this query. */ continue 2; } } $results = true; if (!is_null($ret['charset']) && ($ret['charset'] != 'US-ASCII')) { if (!is_null($charset) && ($charset != 'US-ASCII') && ($charset != $ret['charset'])) { throw new InvalidArgumentException( 'AND/OR queries must all have the same charset.' ); } $charset = $ret['charset']; } $exts_used = array_merge($exts_used, $ret['exts']); switch ($type) { case 'AND': $cmds->add($ret['query'], true); break; case 'OR': // First OR'd query if (count($cmds)) { $new_cmds = new Horde_Imap_Client_Data_Format_List(); $new_cmds->add(array( 'OR', $ret['query'], $cmds )); $cmds = $new_cmds; } else { $cmds = $ret['query']; } break; } } return $results; } /** * Adds fuzzy modifier to search keys. * * @param boolean $add Add the fuzzy modifier? * @param array $temp Temporary build data. * * @throws Horde_Imap_Client_Exception_NoSupport_Extension */ protected function _addFuzzy($add, &$temp) { if ($add) { if (!$temp['exts']->query('SEARCH', 'FUZZY')) { throw new Horde_Imap_Client_Exception_NoSupportExtension('SEARCH=FUZZY'); } $temp['cmds']->add('FUZZY'); $temp['exts_used'][] = 'SEARCH=FUZZY'; } } /** * Search for a flag/keywords. * * @param string $name The flag or keyword name. * @param boolean $set If true, search for messages that have the flag * set. If false, search for messages that do not * have the flag set. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function flag($name, $set = true, array $opts = array()) { $name = Horde_String::upper(ltrim($name, '\\')); if (!isset($this->_search['flag'])) { $this->_search['flag'] = array(); } /* The list of defined system flags (see RFC 3501 [2.3.2]). */ $systemflags = array( 'ANSWERED', 'DELETED', 'DRAFT', 'FLAGGED', 'RECENT', 'SEEN' ); $this->_search['flag'][$name] = array_filter(array( 'fuzzy' => !empty($opts['fuzzy']), 'set' => $set, 'type' => in_array($name, $systemflags) ? 'flag' : 'keyword' )); } /** * Determines if flags are a part of the search. * * @return boolean True if search query involves flags. */ public function flagSearch() { return !empty($this->_search['flag']); } /** * Search for either new messages (messages that have the '\Recent' flag * but not the '\Seen' flag) or old messages (messages that do not have * the '\Recent' flag). If new messages are searched, this will clear * any '\Recent' or '\Unseen' flag searches. If old messages are searched, * this will clear any '\Recent' flag search. * * @param boolean $newmsgs If true, searches for new messages. Else, * search for old messages. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function newMsgs($newmsgs = true, array $opts = array()) { $this->_search['new'] = $newmsgs; if (!empty($opts['fuzzy'])) { $this->_search['newfuzzy'] = true; } } /** * Search for text in the header of a message. * * @param string $header The header field. * @param string $text The search text. * @param boolean $not If true, do a 'NOT' search of $text. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function headerText($header, $text, $not = false, array $opts = array()) { if (!isset($this->_search['header'])) { $this->_search['header'] = array(); } $this->_search['header'][] = array_filter(array( 'fuzzy' => !empty($opts['fuzzy']), 'header' => Horde_String::upper($header), 'text' => $text, 'not' => $not )); } /** * Search for text in either the entire message, or just the body. * * @param string $text The search text.< * @param string $bodyonly If true, only search in the body of the> * @param boolean $bodyonly If true, only search in the body of the* message. If false, also search in the headers. * @param boolean $not If true, do a 'NOT' search of $text. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function text($text, $bodyonly = true, $not = false, array $opts = array()) { if (!isset($this->_search['text'])) { $this->_search['text'] = array(); } $this->_search['text'][] = array_filter(array( 'fuzzy' => !empty($opts['fuzzy']), 'not' => $not, 'text' => $text, 'type' => $bodyonly ? 'BODY' : 'TEXT' )); } /** * Search for messages smaller/larger than a certain size. * * @todo: Remove $not for 3.0 * * @param integer $size The size (in bytes). * @param boolean $larger Search for messages larger than $size? * @param boolean $not If true, do a 'NOT' search of $text. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function size($size, $larger = false, $not = false, array $opts = array()) { if (!isset($this->_search['size'])) { $this->_search['size'] = array(); } $this->_search['size'][$larger ? 'LARGER' : 'SMALLER'] = array_filter(array( 'fuzzy' => !empty($opts['fuzzy']), 'not' => $not, 'size' => (float)$size )); } /** * Search for messages within a given UID range. Only one message range * can be specified per query. * * @param Horde_Imap_Client_Ids $ids The list of UIDs to search. * @param boolean $not If true, do a 'NOT' search of the * UIDs. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function ids(Horde_Imap_Client_Ids $ids, $not = false, array $opts = array()) { $this->_search['ids'] = array_filter(array( 'fuzzy' => !empty($opts['fuzzy']), 'ids' => $ids, 'not' => $not )); } /** * Search for messages within a date range. * * @param mixed $date DateTime or Horde_Date object. * @param string $range Either: * - Horde_Imap_Client_Search_Query::DATE_BEFORE * - Horde_Imap_Client_Search_Query::DATE_ON * - Horde_Imap_Client_Search_Query::DATE_SINCE * @param boolean $header If true, search using the date in the message * headers. If false, search using the internal * IMAP date (usually arrival time). * @param boolean $not If true, do a 'NOT' search of the range. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function dateSearch($date, $range, $header = true, $not = false, array $opts = array()) { if (!isset($this->_search['date'])) { $this->_search['date'] = array(); } // We should really be storing the raw DateTime object as data, // but all versions of the query object have converted at this stage. $ob = new Horde_Imap_Client_Data_Format_Date($date); $this->_search['date'][] = array_filter(array( 'date' => $ob->escape(), 'fuzzy' => !empty($opts['fuzzy']), 'header' => $header, 'range' => $range, 'not' => $not )); } /** * Search for messages within a date and time range. * * @param mixed $date DateTime or Horde_Date object. * @param string $range Either: * - Horde_Imap_Client_Search_Query::DATE_BEFORE * - Horde_Imap_Client_Search_Query::DATE_ON * - Horde_Imap_Client_Search_Query::DATE_SINCE * @param boolean $header If true, search using the date in the message * headers. If false, search using the internal * IMAP date (usually arrival time). * @param boolean $not If true, do a 'NOT' search of the range. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function dateTimeSearch($date, $range, $header = true, $not = false, array $opts = array()) { if (!isset($this->_search['date'])) { $this->_search['date'] = array(); } // We should really be storing the raw DateTime object as data, // but all versions of the query object have converted at this stage. $ob = new Horde_Imap_Client_Data_Format_DateTime($date); $this->_search['date'][] = array_filter(array( 'date' => $ob->escape(), 'fuzzy' => !empty($opts['fuzzy']), 'header' => $header, 'range' => $range, 'not' => $not )); } /** * Search for messages within a given interval. Only one interval of each * type can be specified per search query. If the IMAP server supports * the WITHIN extension (RFC 5032), it will be used. Otherwise, the * search query will be dynamically created using IMAP4rev1 search * terms. * * @param integer $interval Seconds from the present. * @param string $range Either: * - Horde_Imap_Client_Search_Query::INTERVAL_OLDER * - Horde_Imap_Client_Search_Query::INTERVAL_YOUNGER * @param boolean $not If true, do a 'NOT' search. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function intervalSearch($interval, $range, $not = false, array $opts = array()) { if (!isset($this->_search['within'])) { $this->_search['within'] = array(); } $this->_search['within'][$range] = array( 'fuzzy' => !empty($opts['fuzzy']), 'interval' => $interval, 'not' => $not ); } /** * AND queries - the contents of this query will be AND'ed (in its * entirety) with the contents of EACH of the queries passed in. All * AND'd queries must share the same charset as this query. * * @param mixed $queries A query, or an array of queries, to AND with the * current query. */ public function andSearch($queries) { if (!isset($this->_search['and'])) { $this->_search['and'] = array(); } if ($queries instanceof Horde_Imap_Client_Search_Query) { $queries = array($queries); } $this->_search['and'] = array_merge($this->_search['and'], $queries); } /** * OR a query - the contents of this query will be OR'ed (in its entirety) * with the contents of EACH of the queries passed in. All OR'd queries * must share the same charset as this query. All contents of any single * query will be AND'ed together. * * @param mixed $queries A query, or an array of queries, to OR with the * current query. */ public function orSearch($queries) { if (!isset($this->_search['or'])) { $this->_search['or'] = array(); } if ($queries instanceof Horde_Imap_Client_Search_Query) { $queries = array($queries); } $this->_search['or'] = array_merge($this->_search['or'], $queries); } /** * Search for messages modified since a specific moment. The IMAP server * must support the CONDSTORE extension (RFC 7162) for this query to be * used. * * @param integer $value The mod-sequence value. * @param string $name The entry-name string. * @param string $type Either 'shared', 'priv', or 'all'. Defaults to * 'all' * @param boolean $not If true, do a 'NOT' search. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function modseq($value, $name = null, $type = null, $not = false, array $opts = array()) { if (!is_null($type)) { $type = Horde_String::lower($type); if (!in_array($type, array('shared', 'priv', 'all'))) { $type = 'all'; } } $this->_search['modseq'] = array_filter(array( 'fuzzy' => !empty($opts['fuzzy']), 'name' => $name, 'not' => $not, 'type' => (!is_null($name) && is_null($type)) ? 'all' : $type, 'value' => $value )); } /** * Use the results from the previous SEARCH command. The IMAP server must * support the SEARCHRES extension (RFC 5182) for this query to be used. * * @param boolean $not If true, don't match the previous query. * @param array $opts Additional options: * - fuzzy: (boolean) If true, perform a fuzzy search. The IMAP server * MUST support RFC 6203. */ public function previousSearch($not = false, array $opts = array()) { $this->_search['prevsearch'] = $not; if (!empty($opts['fuzzy'])) { $this->_search['prevsearchfuzzy'] = true; } } /* Serializable methods. */ /** * Serialization. * * @return string Serialized data. */ public function serialize() { $data = array( // Serialized data ID. self::VERSION, $this->_search ); if (!is_null($this->_charset)) { $data[] = $this->_charset; } return serialize($data); } /** * Unserialization. * * @param string $data Serialized data. * * @throws Exception */ public function unserialize($data) { $data = @unserialize($data); if (!is_array($data) || !isset($data[0]) || ($data[0] != self::VERSION)) { throw new Exception('Cache version change'); } $this->_search = $data[1]; if (isset($data[2])) { $this->_charset = $data[2]; } } }