ShellBanner
System:Linux MiraNet 3.0.0-14-generic-pae #23-Ubuntu SMP Mon Nov 21 22:07:10 UTC 2011 i686
Software:Apache. PHP/5.3.6-13ubuntu3.10
ID:uid=65534(nobody) gid=65534(nogroup) groups=65534(nogroup)
Safe Mode:OFF
Open_Basedir:OFF
Freespace:24.21 GB of 70.42 GB (34.39%)
MySQL: ON MSSQL: OFF Oracle: OFF PostgreSQL: OFF Curl: OFF Sockets: ON Fetch: OFF Wget: ON Perl: ON
Disabled Functions: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,

/ http/ mail/ program/ lib/ Roundcube/ - drwxr-xr-x

Directory:
Viewing file:     rcube_result_thread.php (19.57 KB)      -rw-r--r--
Select action/file-type:
(+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
<?php

/*
 +-----------------------------------------------------------------------+
 | This file is part of the Roundcube Webmail client                     |
 | Copyright (C) 2005-2011, The Roundcube Dev Team                       |
 | Copyright (C) 2011, 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:                                                              |
 |   THREAD response handler                                             |
 +-----------------------------------------------------------------------+
 | Author: Thomas Bruederli <roundcube@gmail.com>                        |
 | Author: Aleksander Machniak <alec@alec.pl>                            |
 +-----------------------------------------------------------------------+
*/

/**
 * Class for accessing IMAP's THREAD result
 *
 * @package    Framework
 * @subpackage Storage
 */
class rcube_result_thread
{
    protected 
$raw_data;
    protected 
$mailbox;
    protected 
$meta = array();
    protected 
$order 'ASC';

    const 
SEPARATOR_ELEMENT ' ';
    const 
SEPARATOR_ITEM    '~';
    const 
SEPARATOR_LEVEL   ':';


    
/**
     * Object constructor.
     */
    
public function __construct($mailbox null$data null)
    {
        
$this->mailbox $mailbox;
        
$this->init($data);
    }


    
/**
     * Initializes object with IMAP command response
     *
     * @param string $data IMAP response string
     */
    
public function init($data null)
    {
        
$this->meta = array();

        
$data explode('*', (string)$data);

        
// ...skip unilateral untagged server responses
        
for ($i=0$len=count($data); $i<$len$i++) {
            if (
preg_match('/^ THREAD/i'$data[$i])) {
                
// valid response, initialize raw_data for is_error()
                
$this->raw_data '';
                
$data[$i] = substr($data[$i], 7);
                break;
            }

            unset(
$data[$i]);
        }

        if (empty(
$data)) {
            return;
        }

        
$data array_shift($data);
        
$data trim($data);
        
$data preg_replace('/[\r\n]/'''$data);
        
$data preg_replace('/\s+/'' '$data);

        
$this->raw_data $this->parse_thread($data);
    }


    
/**
     * Checks the result from IMAP command
     *
     * @return bool True if the result is an error, False otherwise
     */
    
public function is_error()
    {
        return 
$this->raw_data === null true false;
    }


    
/**
     * Checks if the result is empty
     *
     * @return bool True if the result is empty, False otherwise
     */
    
public function is_empty()
    {
        return empty(
$this->raw_data) ? true false;
    }


    
/**
     * Returns number of elements (threads) in the result
     *
     * @return int Number of elements
     */
    
public function count()
    {
        if (
$this->meta['count'] !== null)
            return 
$this->meta['count'];

        if (empty(
$this->raw_data)) {
            
$this->meta['count'] = 0;
        }
        else {
            
$this->meta['count'] = substr_count($this->raw_dataself::SEPARATOR_ELEMENT);
        }

        if (!
$this->meta['count'])
            
$this->meta['messages'] = 0;

        return 
$this->meta['count'];
    }


    
/**
     * Returns number of all messages in the result
     *
     * @return int Number of elements
     */
    
public function count_messages()
    {
        if (
$this->meta['messages'] !== null)
            return 
$this->meta['messages'];

        if (empty(
$this->raw_data)) {
            
$this->meta['messages'] = 0;
        }
        else {
            
$this->meta['messages'] = 1
                
substr_count($this->raw_dataself::SEPARATOR_ELEMENT)
                + 
substr_count($this->raw_dataself::SEPARATOR_ITEM);
        }

        if (
$this->meta['messages'] == || $this->meta['messages'] == 1)
            
$this->meta['count'] = $this->meta['messages'];

        return 
$this->meta['messages'];
    }


    
/**
     * Returns maximum message identifier in the result
     *
     * @return int Maximum message identifier
     */
    
public function max()
    {
        if (!isset(
$this->meta['max'])) {
            
$this->meta['max'] = (int) @max($this->get());
        }
        return 
$this->meta['max'];
    }


    
/**
     * Returns minimum message identifier in the result
     *
     * @return int Minimum message identifier
     */
    
public function min()
    {
        if (!isset(
$this->meta['min'])) {
            
$this->meta['min'] = (int) @min($this->get());
        }
        return 
$this->meta['min'];
    }


    
/**
     * Slices data set.
     *
     * @param $offset Offset (as for PHP's array_slice())
     * @param $length Number of elements (as for PHP's array_slice())
     */
    
public function slice($offset$length)
    {
        
$data explode(self::SEPARATOR_ELEMENT$this->raw_data);
        
$data array_slice($data$offset$length);

        
$this->meta          = array();
        
$this->meta['count'] = count($data);
        
$this->raw_data      implode(self::SEPARATOR_ELEMENT$data);
    }


    
/**
     * Filters data set. Removes threads not listed in $roots list.
     *
     * @param array $roots List of IDs of thread roots.
     */
    
public function filter($roots)
    {
        
$datalen strlen($this->raw_data);
        
$roots   array_flip($roots);
        
$result  '';
        
$start   0;

        
$this->meta          = array();
        
$this->meta['count'] = 0;

        while ((
$pos = @strpos($this->raw_dataself::SEPARATOR_ELEMENT$start))
            || (
$start $datalen && ($pos $datalen))
        ) {
            
$len   $pos $start;
            
$elem  substr($this->raw_data$start$len);
            
$start $pos 1;

            
// extract root message ID
            
if ($npos strpos($elemself::SEPARATOR_ITEM)) {
                
$root = (int) substr($elem0$npos);
            }
            else {
                
$root $elem;
            }

            if (isset(
$roots[$root])) {
                
$this->meta['count']++;
                
$result .= self::SEPARATOR_ELEMENT $elem;
            }
        }

        
$this->raw_data ltrim($resultself::SEPARATOR_ELEMENT);
    }


    
/**
     * Reverts order of elements in the result
     */
    
public function revert()
    {
        
$this->order $this->order == 'ASC' 'DESC' 'ASC';

        if (empty(
$this->raw_data)) {
            return;
        }

        
$this->meta['pos'] = array();
        
$datalen strlen($this->raw_data);
        
$result  '';
        
$start   0;

        while ((
$pos = @strpos($this->raw_dataself::SEPARATOR_ELEMENT$start))
            || (
$start $datalen && ($pos $datalen))
        ) {
            
$len   $pos $start;
            
$elem  substr($this->raw_data$start$len);
            
$start $pos 1;

            
$result $elem self::SEPARATOR_ELEMENT $result;
        }

        
$this->raw_data rtrim($resultself::SEPARATOR_ELEMENT);
    }


    
/**
     * Check if the given message ID exists in the object
     *
     * @param int $msgid Message ID
     * @param bool $get_index When enabled element's index will be returned.
     *                        Elements are indexed starting with 0
     *
     * @return boolean True on success, False if message ID doesn't exist
     */
    
public function exists($msgid$get_index false)
    {
        
$msgid = (int) $msgid;
        
$begin implode('|', array(
            
'^',
            
preg_quote(self::SEPARATOR_ELEMENT'/'),
            
preg_quote(self::SEPARATOR_LEVEL'/'),
        ));
        
$end implode('|', array(
            
'$',
            
preg_quote(self::SEPARATOR_ELEMENT'/'),
            
preg_quote(self::SEPARATOR_ITEM'/'),
        ));

        if (
preg_match("/($begin)$msgid($end)/"$this->raw_data$m,
            
$get_index PREG_OFFSET_CAPTURE null)
        ) {
            if (
$get_index) {
                
$idx 0;
                if (
$m[0][1]) {
                    
$idx substr_count($this->raw_dataself::SEPARATOR_ELEMENT0$m[0][1]+1)
                        + 
substr_count($this->raw_dataself::SEPARATOR_ITEM0$m[0][1]+1);
                }
                
// cache position of this element, so we can use it in get_element()
                
$this->meta['pos'][$idx] = (int)$m[0][1];

                return 
$idx;
            }
            return 
true;
        }

        return 
false;
    }


    
/**
     * Return IDs of all messages in the result. Threaded data will be flattened.
     *
     * @return array List of message identifiers
     */
    
public function get()
    {
        if (empty(
$this->raw_data)) {
            return array();
        }

        
$regexp '/(' preg_quote(self::SEPARATOR_ELEMENT'/')
            . 
'|' preg_quote(self::SEPARATOR_ITEM'/') . '[0-9]+' preg_quote(self::SEPARATOR_LEVEL'/')
            .
')/';

        return 
preg_split($regexp$this->raw_data);
    }


    
/**
     * Return all messages in the result.
     *
     * @return array List of message identifiers
     */
    
public function get_compressed()
    {
        if (empty(
$this->raw_data)) {
            return 
'';
        }

        return 
rcube_imap_generic::compressMessageSet($this->get());
    }


    
/**
     * Return result element at specified index (all messages, not roots)
     *
     * @param int|string  $index  Element's index or "FIRST" or "LAST"
     *
     * @return int Element value
     */
    
public function get_element($index)
    {
        
$count $this->count();

        if (!
$count) {
            return 
null;
        }

        
// first element
        
if ($index === || $index === '0' || $index === 'FIRST') {
            
preg_match('/^([0-9]+)/'$this->raw_data$m);
            
$result = (int) $m[1];
            return 
$result;
        }

        
// last element
        
if ($index === 'LAST' || $index == $count-1) {
            
preg_match('/([0-9]+)$/'$this->raw_data$m);
            
$result = (int) $m[1];
            return 
$result;
        }

        
// do we know the position of the element or the neighbour of it?
        
if (!empty($this->meta['pos'])) {
            
$element preg_quote(self::SEPARATOR_ELEMENT'/');
            
$item    preg_quote(self::SEPARATOR_ITEM'/') . '[0-9]+' preg_quote(self::SEPARATOR_LEVEL'/') .'?';
            
$regexp  '(' $element '|' $item ')';

            if (isset(
$this->meta['pos'][$index])) {
                if (
preg_match('/([0-9]+)/'$this->raw_data$mnull$this->meta['pos'][$index]))
                    
$result $m[1];
            }
            else if (isset(
$this->meta['pos'][$index-1])) {
                
// get chunk of data after previous element
                
$data substr($this->raw_data$this->meta['pos'][$index-1]+150);
                
$data preg_replace('/^[0-9]+/'''$data); // remove UID at $index position
                
$data preg_replace("/^$regexp/"''$data); // remove separator
                
if (preg_match('/^([0-9]+)/'$data$m))
                    
$result $m[1];
            }
            else if (isset(
$this->meta['pos'][$index+1])) {
                
// get chunk of data before next element
                
$pos  max(0$this->meta['pos'][$index+1] - 50);
                
$len  min(50$this->meta['pos'][$index+1]);
                
$data substr($this->raw_data$pos$len);
                
$data preg_replace("/$regexp\$/"''$data); // remove separator

                
if (preg_match('/([0-9]+)$/'$data$m))
                    
$result $m[1];
            }

            if (isset(
$result)) {
                return (int) 
$result;
            }
        }

        
// Finally use less effective method
        
$data $this->get();

        return 
$data[$index];
    }


    
/**
     * Returns response parameters e.g. MAILBOX, ORDER
     *
     * @param string $param  Parameter name
     *
     * @return array|string Response parameters or parameter value
     */
    
public function get_parameters($param=null)
    {
        
$params $this->params;
        
$params['MAILBOX'] = $this->mailbox;
        
$params['ORDER']   = $this->order;

        if (
$param !== null) {
            return 
$params[$param];
        }

        return 
$params;
    }


    
/**
     * THREAD=REFS sorting implementation (based on provided index)
     *
     * @param rcube_result_index $index  Sorted message identifiers
     */
    
public function sort($index)
    {
        
$this->sort_order $index->get_parameters('ORDER');

        if (empty(
$this->raw_data)) {
            return;
        }

        
// when sorting search result it's good to make the index smaller
        
if ($index->count() != $this->count_messages()) {
            
$index->filter($this->get());
        }

        
$result  array_fill_keys($index->get(), null);
        
$datalen strlen($this->raw_data);
        
$start   0;

        
// Here we're parsing raw_data twice, we want only one big array
        // in memory at a time

        // Assign roots
        
while (($pos = @strpos($this->raw_dataself::SEPARATOR_ELEMENT$start))
            || (
$start $datalen && ($pos $datalen))
        ) {
            
$len   $pos $start;
            
$elem  substr($this->raw_data$start$len);
            
$start $pos 1;

            
$items explode(self::SEPARATOR_ITEM$elem);
            
$root  = (int) array_shift($items);

            if (
$root) {
                
$result[$root] = $root;
                foreach (
$items as $item) {
                    list(
$lv$id) = explode(self::SEPARATOR_LEVEL$item);
                    
$result[$id] = $root;
                }
            }
        }

        
// get only unique roots
        
$result array_filter($result); // make sure there are no nulls
        
$result array_unique($result);

        
// Re-sort raw data
        
$result array_fill_keys($resultnull);
        
$start 0;

        while ((
$pos = @strpos($this->raw_dataself::SEPARATOR_ELEMENT$start))
            || (
$start $datalen && ($pos $datalen))
        ) {
            
$len   $pos $start;
            
$elem  substr($this->raw_data$start$len);
            
$start $pos 1;

            
$npos strpos($elemself::SEPARATOR_ITEM);
            
$root = (int) ($npos substr($elem0$npos) : $elem);

            
$result[$root] = $elem;
        }

        
$this->raw_data implode(self::SEPARATOR_ELEMENT$result);
    }


    
/**
     * Returns data as tree
     *
     * @return array Data tree
     */
    
public function get_tree()
    {
        
$datalen strlen($this->raw_data);
        
$result  = array();
        
$start   0;

        while ((
$pos = @strpos($this->raw_dataself::SEPARATOR_ELEMENT$start))
            || (
$start $datalen && ($pos $datalen))
        ) {
            
$len   $pos $start;
            
$elem  substr($this->raw_data$start$len);
            
$items explode(self::SEPARATOR_ITEM$elem);
            
$result[array_shift($items)] = $this->build_thread($items);
            
$start $pos 1;
        }

        return 
$result;
    }


    
/**
     * Returns thread depth and children data
     *
     * @return array Thread data
     */
    
public function get_thread_data()
    {
        
$data     $this->get_tree();
        
$depth    = array();
        
$children = array();

        
$this->build_thread_data($data$depth$children);

        return array(
$depth$children);
    }


    
/**
     * Creates 'depth' and 'children' arrays from stored thread 'tree' data.
     */
    
protected function build_thread_data($data, &$depth, &$children$level 0)
    {
        foreach ((array)
$data as $key => $val) {
            
$empty          = empty($val) || !is_array($val);
            
$children[$key] = !$empty;
            
$depth[$key]    = $level;
            if (!
$empty) {
                
$this->build_thread_data($val$depth$children$level 1);
            }
        }
    }


    
/**
     * Converts part of the raw thread into an array
     */
    
protected function build_thread($items$level 1, &$pos 0)
    {
        
$result = array();

        for (
$len=count($items); $pos $len$pos++) {
            list(
$lv$id) = explode(self::SEPARATOR_LEVEL$items[$pos]);
            if (
$level == $lv) {
                
$pos++;
                
$result[$id] = $this->build_thread($items$level+1$pos);
            }
            else {
                
$pos--;
                break;
            }
        }

        return 
$result;
    }


    
/**
     * IMAP THREAD response parser
     */
    
protected function parse_thread($str$begin 0$end 0$depth 0)
    {
        
// Don't be tempted to change $str to pass by reference to speed this up - it will slow it down by about
        // 7 times instead :-) See comments on http://uk2.php.net/references and this article:
        // http://derickrethans.nl/files/phparch-php-variables-article.pdf
        
$node '';
        if (!
$end) {
            
$end strlen($str);
        }

        
// Let's try to store data in max. compacted stracture as a string,
        // arrays handling is much more expensive
        // For the following structure: THREAD (2)(3 6 (4 23)(44 7 96))
        // -- 2
        // -- 3
        //     \-- 6
        //         |-- 4
        //         |    \-- 23
        //         |
        //         \-- 44
        //               \-- 7
        //                    \-- 96
        //
        // The output will be: 2,3^1:6^2:4^3:23^2:44^3:7^4:96

        
if ($str[$begin] != '(') {
            
// find next bracket
            
$stop      $begin strcspn($str'()'$begin$end $begin);
            
$messages  explode(' 'trim(substr($str$begin$stop $begin)));

            if (empty(
$messages)) {
                return 
$node;
            }

            foreach (
$messages as $msg) {
                if (
$msg) {
                    
$node .= ($depth self::SEPARATOR_ITEM.$depth.self::SEPARATOR_LEVEL '').$msg;
                    
$this->meta['messages']++;
                    
$depth++;
                }
            }

            if (
$stop $end) {
                
$node .= $this->parse_thread($str$stop$end$depth);
            }
        }
        else {
            
$off $begin;
            while (
$off $end) {
                
$start $off;
                
$off++;
                
$n 1;
                while (
$n 0) {
                    
$p strpos($str')'$off);
                    if (
$p === false) {
                        
// error, wrong structure, mismatched brackets in IMAP THREAD response
                        // @TODO: write error to the log or maybe set $this->raw_data = null;
                        
return $node;
                    }
                    
$p1 strpos($str'('$off);
                    if (
$p1 !== false && $p1 $p) {
                        
$off $p1 1;
                        
$n++;
                    }
                    else {
                        
$off $p 1;
                        
$n--;
                    }
                }

                
$thread $this->parse_thread($str$start 1$off 1$depth);
                if (
$thread) {
                    if (!
$depth) {
                        if (
$node) {
                            
$node .= self::SEPARATOR_ELEMENT;
                        }
                    }
                    
$node .= $thread;
                }
            }
        }

        return 
$node;
    }
}
Command:
Quick Commands:
Upload:
[OK] Max size: 100MB
PHP Filesystem: <@ Ú
Search File:
regexp
Create File:
Overwrite [OK]
View File:
Mass Defacement:
[+] Main Directory: [+] Defacement Url:
LmfaoX Shell - Private Build [BETA] - v0.1 -; Generated: 0.208 seconds