| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229 | 
							- <?php
 - 
 - /**
 -  +-----------------------------------------------------------------------+
 -  | This file is part of the Roundcube Webmail client                     |
 -  |                                                                       |
 -  | Copyright (C) 2013, The Roundcube Dev Team                            |
 -  | Copyright (C) 2014, Kolab Systems AG                                  |
 -  |                                                                       |
 -  | Licensed under the GNU General Public License version 3 or            |
 -  | any later version with exceptions for skins & plugins.                |
 -  | See the README file for a full license statement.                     |
 -  |                                                                       |
 -  | PURPOSE:                                                              |
 -  |   Execute (multi-threaded) searches in multiple IMAP folders          |
 -  +-----------------------------------------------------------------------+
 -  | Author: Thomas Bruederli <roundcube@gmail.com>                        |
 -  +-----------------------------------------------------------------------+
 - */
 - 
 - /**
 -  * Class to control search jobs on multiple IMAP folders.
 -  *
 -  * @package    Framework
 -  * @subpackage Storage
 -  * @author     Thomas Bruederli <roundcube@gmail.com>
 -  */
 - class rcube_imap_search
 - {
 -     public $options = array();
 - 
 -     protected $jobs      = array();
 -     protected $timelimit = 0;
 -     protected $results;
 -     protected $conn;
 - 
 -     /**
 -      * Default constructor
 -      */
 -     public function __construct($options, $conn)
 -     {
 -         $this->options = $options;
 -         $this->conn    = $conn;
 -     }
 - 
 -     /**
 -      * Invoke search request to IMAP server
 -      *
 -      * @param  array   $folders    List of IMAP folders to search in
 -      * @param  string  $str        Search criteria
 -      * @param  string  $charset    Search charset
 -      * @param  string  $sort_field Header field to sort by
 -      * @param  boolean $threading  True if threaded listing is active
 -      */
 -     public function exec($folders, $str, $charset = null, $sort_field = null, $threading=null)
 -     {
 -         $start   = floor(microtime(true));
 -         $results = new rcube_result_multifolder($folders);
 - 
 -         // start a search job for every folder to search in
 -         foreach ($folders as $folder) {
 -             // a complete result for this folder already exists
 -             $result = $this->results ? $this->results->get_set($folder) : false;
 -             if ($result && !$result->incomplete) {
 -                 $results->add($result);
 -             }
 -             else {
 -                 $search = is_array($str) && $str[$folder] ? $str[$folder] : $str;
 -                 $job = new rcube_imap_search_job($folder, $search, $charset, $sort_field, $threading);
 -                 $job->worker = $this;
 -                 $this->jobs[] = $job;
 -             }
 -         }
 - 
 -         // execute jobs and gather results
 -         foreach ($this->jobs as $job) {
 -             // only run search if within the configured time limit
 -             // TODO: try to estimate the required time based on folder size and previous search performance
 -             if (!$this->timelimit || floor(microtime(true)) - $start < $this->timelimit) {
 -                 $job->run();
 -             }
 - 
 -             // add result (may have ->incomplete flag set)
 -             $results->add($job->get_result());
 -         }
 - 
 -         return $results;
 -     }
 - 
 -     /**
 -      * Setter for timelimt property
 -      */
 -     public function set_timelimit($seconds)
 -     {
 -         $this->timelimit = $seconds;
 -     }
 - 
 -     /**
 -      * Setter for previous (potentially incomplete) search results
 -      */
 -     public function set_results($res)
 -     {
 -         $this->results = $res;
 -     }
 - 
 -     /**
 -      * Get connection to the IMAP server
 -      * (used for single-thread mode)
 -      */
 -     public function get_imap()
 -     {
 -         return $this->conn;
 -     }
 - }
 - 
 - 
 - /**
 -  * Stackable item to run the search on a specific IMAP folder
 -  */
 - class rcube_imap_search_job /* extends Stackable */
 - {
 -     private $folder;
 -     private $search;
 -     private $charset;
 -     private $sort_field;
 -     private $threading;
 -     private $result;
 - 
 -     public function __construct($folder, $str, $charset = null, $sort_field = null, $threading=false)
 -     {
 -         $this->folder     = $folder;
 -         $this->search     = $str;
 -         $this->charset    = $charset;
 -         $this->sort_field = $sort_field;
 -         $this->threading  = $threading;
 - 
 -         $this->result = new rcube_result_index($folder);
 -         $this->result->incomplete = true;
 -     }
 - 
 -     public function run()
 -     {
 -         $this->result = $this->search_index();
 -     }
 - 
 -     /**
 -      * Copy of rcube_imap::search_index()
 -      */
 -     protected function search_index()
 -     {
 -         $criteria = $this->search;
 -         $charset  = $this->charset;
 -         $imap     = $this->worker->get_imap();
 - 
 -         if (!$imap->connected()) {
 -             trigger_error("No IMAP connection for $this->folder", E_USER_WARNING);
 - 
 -             if ($this->threading) {
 -                 return new rcube_result_thread($this->folder);
 -             }
 -             else {
 -                 return new rcube_result_index($this->folder);
 -             }
 -         }
 - 
 -         if ($this->worker->options['skip_deleted'] && !preg_match('/UNDELETED/', $criteria)) {
 -             $criteria = 'UNDELETED '.$criteria;
 -         }
 - 
 -         // unset CHARSET if criteria string is ASCII, this way
 -         // SEARCH won't be re-sent after "unsupported charset" response
 -         if ($charset && $charset != 'US-ASCII' && is_ascii($criteria)) {
 -             $charset = 'US-ASCII';
 -         }
 - 
 -         if ($this->threading) {
 -             $threads = $imap->thread($this->folder, $this->threading, $criteria, true, $charset);
 - 
 -             // Error, try with US-ASCII (RFC5256: SORT/THREAD must support US-ASCII and UTF-8,
 -             // but I've seen that Courier doesn't support UTF-8)
 -             if ($threads->is_error() && $charset && $charset != 'US-ASCII') {
 -                 $threads = $imap->thread($this->folder, $this->threading,
 -                     rcube_imap::convert_criteria($criteria, $charset), true, 'US-ASCII');
 -             }
 - 
 -             return $threads;
 -         }
 - 
 -         if ($this->sort_field) {
 -             $messages = $imap->sort($this->folder, $this->sort_field, $criteria, true, $charset);
 - 
 -             // Error, try with US-ASCII (RFC5256: SORT/THREAD must support US-ASCII and UTF-8,
 -             // but I've seen Courier with disabled UTF-8 support)
 -             if ($messages->is_error() && $charset && $charset != 'US-ASCII') {
 -                 $messages = $imap->sort($this->folder, $this->sort_field,
 -                     rcube_imap::convert_criteria($criteria, $charset), true, 'US-ASCII');
 -             }
 -         }
 - 
 -         if (!$messages || $messages->is_error()) {
 -             $messages = $imap->search($this->folder,
 -                 ($charset && $charset != 'US-ASCII' ? "CHARSET $charset " : '') . $criteria, true);
 - 
 -             // Error, try with US-ASCII (some servers may support only US-ASCII)
 -             if ($messages->is_error() && $charset && $charset != 'US-ASCII') {
 -                 $messages = $imap->search($this->folder,
 -                     rcube_imap::convert_criteria($criteria, $charset), true);
 -             }
 -         }
 - 
 -         return $messages;
 -     }
 - 
 -     public function get_search_set()
 -     {
 -         return array(
 -             $this->search,
 -             $this->result,
 -             $this->charset,
 -             $this->sort_field,
 -             $this->threading,
 -         );
 -     }
 - 
 -     public function get_result()
 -     {
 -         return $this->result;
 -     }
 - }
 
 
  |