- GRAYBYTE UNDETECTABLE CODES -

403Webshell
Server IP : 184.154.167.98  /  Your IP : 18.217.228.195
Web Server : Apache
System : Linux pink.dnsnetservice.com 4.18.0-553.22.1.lve.1.el8.x86_64 #1 SMP Tue Oct 8 15:52:54 UTC 2024 x86_64
User : puertode ( 1767)
PHP Version : 7.2.34
Disable Function : NONE
MySQL : OFF  |  cURL : ON  |  WGET : ON  |  Perl : ON  |  Python : ON  |  Sudo : ON  |  Pkexec : ON
Directory :  /home/puertode/www/mesa/include/

Upload File :
current_dir [ Writeable ] document_root [ Writeable ]

 

Command :


[ Back ]     

Current File : /home/puertode/www/mesa/include/class.filter.php
<?php
/*********************************************************************
    class.filter.php

    Ticket Filter

    Peter Rotich <peter@osticket.com>
    Copyright (c)  2006-2013 osTicket
    http://www.osticket.com

    Released under the GNU General Public License WITHOUT ANY WARRANTY.
    See LICENSE.TXT for details.

    vim: expandtab sw=4 ts=4 sts=4:
**********************************************************************/

require_once INCLUDE_DIR . 'class.filter_action.php';

class Filter
extends VerySimpleModel {
    static $meta = array(
        'table' => FILTER_TABLE,
        'pk' => array('id'),
        'ordering' => array('execorder'),
        'joins' => array(
            'rules' => array(
                'reverse' => 'FilterRule.filter',
            ),
            'actions' => array(
                'reverse' => 'FilterAction.filter',
            ),
        ),
    );

    const FLAG_INACTIVE_HT = 0x0001;
    const FLAG_INACTIVE_DEPT  = 0x0002;
    const FLAG_DELETED_OBJECT  = 0x0004;

    static $match_types = array(
        /* @trans */ 'User Information' => array(
            array('name'      =>    /* @trans */ 'Name',
                'email'     =>      /* @trans */ 'Email',
            ),
            900
        ),
        /* @trans */ 'Email Meta-Data' => array(
            array('reply-to'  =>    /* @trans */ 'Reply-To Email',
                'reply-to-name' =>  /* @trans */ 'Reply-To Name',
                'addressee' =>      /* @trans */ 'Addressee (To and Cc)',
            ),
            200
        ),
    );

    function __construct($vars=array()) {
        parent::__construct($vars);
        $this->created = SqlFunction::NOW();
    }

    function getId() {
        return $this->id;
    }

    function getTarget() {
        return $this->target;
    }

    function getName() {
        return $this->name;
    }

    function getNotes() {
        return $this->notes;
    }

    function getInfo() {
        $ht = $this->ht;
        if (static::$meta['joins'])
            foreach (static::$meta['joins'] as $k => $v)
                unset($ht[$k]);
        return $ht;
    }

    function getNumRules() {
        return $this->rules->count();
    }

    function getExecOrder() {
        return $this->execorder;
    }

    function getEmailId() {
        return $this->email_id;
    }

    function isActive() {
        return ($this->isactive);
    }

    function isSystemBanlist() {
        return !strcasecmp($this->getName(),'SYSTEM BAN LIST');
    }

    function getDeptId() {
        return $this->dept_id;
    }

    function getStatusId() {
        return $this->status_id;
    }

    function getPriorityId() {
        return $this->priority_id;
    }

    function getSLAId() {
        return $this->sla_id;
    }

    function getStaffId() {
        return $this->staff_id;
    }

    function getTeamId() {
        return $this->team_id;
    }

    function getCannedResponse() {
        return $this->canned_response_id;
    }

    function getHelpTopic() {
        return $this->topic_id;
    }

    public function setFlag($flag, $val) {
        $vars = array();
        $errors = array();
        if ($val)
            $this->flags |= $flag;
        else
            $this->flags &= ~$flag;
        $vars['rules']= $this->getRules();
        $this->ht['pass'] = true;
        $this->update($this->ht, $errors);
    }

    function hasFlag($flag) {
        return 0 !== ($this->ht['flags'] & $flag);
    }

    function stopOnMatch() {
        return ($this->stop_onmatch);
    }

    function matchAllRules() {
        return ($this->match_all_rules);
    }

    function rejectOnMatch() {
        return ($this->reject_ticket);
    }

    function useReplyToEmail() {
        return ($this->use_replyto_email);
    }

    static function disableFilters($object) {
        switch (get_class($object)) {
            case 'Topic':
                $object_id = 'topic_id';
                $FAClass = 'FA_AssignTopic';
                break;
            case 'Dept':
                $object_id = 'dept_id';
                $FAClass = 'FA_RouteDepartment';
                break;
            case 'Staff':
                $object_id = 'staff_id';
                $FAClass = 'FA_AssignAgent';
                break;
            case 'Team':
                $object_id = 'team_id';
                $FAClass = 'FA_AssignTeam';
                break;
            case 'SLA':
                $object_id = 'sla_id';
                $FAClass = 'FA_AssignSLA';
                break;
            case 'TicketStatus':
                $object_id = 'status_id';
                $FAClass = 'FA_SetStatus';
                break;
            case 'Email':
                $object_id = 'from';
                $FAClass = 'FA_SendEmail';
                break;
            case 'Canned':
                $object_id = 'canned_id';
                $FAClass = 'FA_AutoCannedResponse';
            default:
                return false;
        }
        $id = $object->getId();
        $actions = FilterAction::objects()
            ->filter(array('type' => $FAClass::$type,
                           'configuration__like' => sprintf('%%"%s":%s}', $object_id, $id)));
        foreach($actions as $fa) {
            // Put a flag on the filter
            FilterAction::setFilterFlags($fa, 'Filter::FLAG_DELETED_OBJECT', true);
            $fa->save();

            // Disable the filter
            $filter = Filter::lookup($fa->filter_id);
            $filter->isactive = 0;
            $filter->save();
        }
    }

    function disableAlerts() {
        return ($this->disable_autoresponder);
    }

    function sendAlerts() {
        return (!$this->disableAlerts());
    }

    function getRules() {
        $rules = [];
        foreach ($this->rules as $r)
            $rules[] = array('w'=>$r->what,'h'=>$r->how,'v'=>$r->val);

        return $rules;
    }

    function addRule($what, $how, $val,$extra=array()) {
        if (isset($extra['notes']))
            $extra['notes'] = Format::sanitize($extra['notes']);
        $rule = array_merge($extra,array('what'=>$what, 'how'=>$how, 'val'=>$val));
        $rule = new FilterRule($rule);
        $rule->created = SqlFunction::NOW();
        $this->rules->add($rule);
        if ($rule->save())
            return true;
    }

    function removeRule($what, $how, $val) {
        return $this->rules->filter([
            'what' => $what,
            'how' => $how,
            'val' => $val,
        ])->delete();
    }

    function getRule($id) {
        return $this->getRuleById($id);
    }

    function getRuleById($id) {
        return FilterRule::lookup(array('id'=>$id, 'filter_id'=>$this->getId()));
    }

    function containsRule($what, $how, $val) {
        return $this->rules->filter([
            'what' => $what,
            'how' => $how,
            'val' => $val,
        ])->exists();
    }

    /**
     * Simple true/false if the rules defined for this filter match the
     * incoming email
     *
     * $info is an ARRAY, which has valid keys
     *   email - FROM email address of the ticket owner
     *   name - name of ticket owner
     *   subject - subject line of the ticket
     *   body - body content of the message (no attachments, please)
     *   reply-to - reply-to email address
     *   reply-to-name - name of sender to reply-to
     *   headers - array of email headers
     *   emailId - osTicket system email id
     */
    function matches($what) {

        if(!$what || !is_array($what)) return false;

        $how = array(
            # how => array(function, null or === this, null or !== this)
            'equal'     => array('strcasecmp', 0),
            'not_equal' => array('strcasecmp', null, 0),
            'contains'  => array('stripos', null, false),
            'dn_contain'=> array('stripos', false),
            'starts'    => array('stripos', 0),
            'ends'      => array('iendsWith', true),
            'match'     => array('pregMatchB', 1),
            'not_match' => array('pregMatchB', null, 1),
        );

        $match = false;
        # Respect configured filter email-id
        if ($this->getEmailId()
                && !strcasecmp($this->getTarget(), 'Email')
                && $this->getEmailId() != $what['emailId'])
            return false;

        foreach ($this->getRules() as $rule) {
            if (!isset($how[$rule['h']])) continue;
            list($func, $pos, $neg) = $how[$rule['h']];

            $result = call_user_func($func, $what[$rule['w']], $rule['v']);
            if (($pos === null && $result !== $neg) or ($result === $pos)) {
                # Match.
                $match = true;
                if (!$this->matchAllRules()) break;
            } else {
                # No match. Continue?
                if ($this->matchAllRules()) {
                    $match = false;
                    break;
                }
            }
        }

        return $match;
    }

    function getActions() {
        return $this->actions;
    }

    /**
     * If the matches() method returns TRUE, send the initial ticket to this
     * method to apply the filter actions defined
     */
    function apply(&$ticket, $vars, $postCreate=false) {
        foreach ($this->getActions() as $a) {
            //control when certain actions should be applied
            //if action is send email and postCreate == false, skip
            //if action is not send email and postCreate == true, skip
            if ((($a->type == 'email') ? !$postCreate : $postCreate))
                continue;

            $a->setFilter($this);
            $a->apply($ticket, $vars);
        }
    }

    function getVars() {
        return $this->vars;
    }

    static function getSupportedMatches() {
        foreach (static::$match_types as $k=>&$v) {
            if (is_callable($v[0]))
                $v[0] = $v[0]();
        }
        unset($v);
        uasort(static::$match_types, function($a, $b) { return $a[1] - $b[1]; });
        return array_map(function($a) { return $a[0]; }, static::$match_types);
    }

    static function addSupportedMatches($group, $callable, $order=10) {
        static::$match_types[$group] = array($callable, $order);
    }

    static function getSupportedMatchFields() {
        $keys = array();
        foreach (static::getSupportedMatches() as $group=>$matches) {
            foreach ($matches as $key=>$label) {
                $keys[] = $key;
            }
        }
        return $keys;
    }

    static function getSupportedMatchTypes() {
        return array(
            'equal'=>       __('Equal'),
            'not_equal'=>   __('Not Equal'),
            'contains'=>    __('Contains'),
            'dn_contain'=>  __('Does Not Contain'),
            'starts'=>      __('Starts With'),
            'ends'=>        __('Ends With'),
            'match'=>       __('Matches Regex'),
            'not_match'=>   __('Does Not Match Regex'),
        );
    }

    function update($vars,&$errors) {
        //validate filter actions before moving on
        if (!self::validate_actions($vars, $errors))
            return false;

        $vars['flags'] = $this->flags;

        if(!$vars['execorder'])
            $errors['execorder'] = __('Order required');
        elseif(!is_numeric($vars['execorder']))
            $errors['execorder'] = __('Must be numeric value');

        if(!$vars['name'])
            $errors['name'] = __('Name required');
        elseif(($filter=static::getByName($vars['name'])) && $filter->id!=$this->id)
            $errors['name'] = __('Name already in use');

        if(!$errors && !$this->validate_rules($vars,$errors) && !$errors['rules'])
            $errors['rules'] = __('Unable to validate rules as entered');

        $targets = self::getTargets();
        if(!$vars['target'])
            $errors['target'] = __('Target required');
        else if(!is_numeric($vars['target']) && !$targets[$vars['target']])
            $errors['target'] = __('Unknown or invalid target');

        if($errors) return false;

        $emailId = 0;
        if(is_numeric($vars['target'])) {
            $emailId = $vars['target'];
            $vars['target'] = 'Email';
        }

        //Note: this will be set when validating filters
        if ($vars['email_id'])
            $emailId = $vars['email_id'];
        $this->isactive = $vars['isactive'];
        $this->flags = $vars['flags'];
        $this->target = $vars['target'];
        $this->name = $vars['name'];
        $this->execorder = $vars['execorder'];
        $this->email_id = $emailId;
        $this->match_all_rules = $vars['match_all_rules'];
        $this->stop_onmatch = $vars['stop_onmatch'];
        $this->notes = Format::sanitize($vars['notes']);

        if (!$this->save()) {
            if (!$this->__new__) {
                $errors['err']=sprintf(__('Unable to update %s.'), __('this ticket filter'))
                   .' '.__('Internal error occurred');
            }
            else {
                $errors['err']=sprintf(__('Unable to add %s.'), __('this ticket filter'))
                   .' '.__('Internal error occurred');
            }
            return false;
        }

        // Attempt to create/update the actions. Collect the errors
        $this->save_actions($this->getId(), $vars, $errors);
        if ($errors)
            return false;

        //Success with update/create...save the rules. We can't recover from any errors at this point.
        # Don't care about errors stashed in $xerrors
        $xerrors = array();
        if (!$this->save_rules($vars,$xerrors))
            return false;

        return true;
    }

    function delete() {
        try {
            parent::delete();
            $type = array('type' => 'deleted');
            Signal::send('object.deleted', $this, $type);
            $this->rules->expunge();
            $this->actions->expunge();
        }
        catch (OrmException $e) {
            return false;
        }
        return true;
    }

    /** static functions **/
    static function getTargets() {
        return array(
                'Any' => __('Any'),
                'Web' => __('Web Forms'),
                'API' => __('API Calls'),
                'Email' => __('Emails'));
    }

    static function getByName($name) {
        return static::lookup(['name' => $name]);
    }

    function validate_rules($vars,&$errors) {
        $matches = self::getSupportedMatchFields();
        $types = array_keys(self::getSupportedMatchTypes());
        $rules = array();
        foreach ($vars['rules'] as $i=>$rule) {
            if ($rule->ht) {
                $rule = $rule->ht;
                $rule["w"] = $rule["what"];
                $rule["h"] = $rule["how"];
                $rule["v"] = $rule["val"];
            }

            if (is_array($rule)) {
                if($rule["w"] || $rule["h"]) {
                    // Check for REGEX compile errors
                    if (in_array($rule["h"], array('match','not_match'))) {
                        $wrapped = "/".$rule["v"]."/iu";
                        if (false === @preg_match($rule["v"], ' ')
                                && (false !== @preg_match($wrapped, ' ')))
                            $rule["v"] = $wrapped;
                    }

                    if(!$rule["w"] || !in_array($rule["w"],$matches))
                        $errors["rule_$i"]=__('Invalid match selection');
                    elseif(!$rule["h"] || !in_array($rule["h"],$types))
                        $errors["rule_$i"]=__('Invalid match type selection');
                    elseif(!$rule["v"])
                        $errors["rule_$i"]=__('Value required');
                    elseif($rule["w"]=='email'
                            && $rule["h"]=='equal'
                            && !Validator::is_email($rule["v"]))
                        $errors["rule_$i"]=__('Valid email required for the match type');
                    elseif (in_array($rule["h"], array('match','not_match'))
                            && (false === @preg_match($rule["v"], ' ')))
                        $errors["rule_$i"] = sprintf(__('Regex compile error: (#%s)'),
                            preg_last_error());
                    else //for everything-else...we assume it's valid.
                        $rules[]=array('what'=>$rule["w"],
                            'how'=>$rule["h"],'val'=>trim($rule["v"]));
                } elseif($rule["v"]) {
                    $errors["rule_$i"]=__('Incomplete selection');
                }
            }
        }

        if(!$rules && !$errors)
            $errors['rules']=__('You must set at least one rule.');

        return $rules;
    }

    function save_rules($vars, &$errors) {
        $rules = $this->validate_rules($vars, $errors);

        if ($errors)
            return false;

        //Clear existing rules...we're doing mass replace on each save!!
        $this->rules->expunge();
        $num = 0;
        foreach ($rules as $rule) {
            $rule = new FilterRule($rule);
            $this->rules->add($rule);
            $rule->save();
            $num++;
        }

        return $num;
    }

    function save($refetch=false) {
        if ($this->dirty)
            $this->updated = SqlFunction::NOW();
        return parent::save($refetch || $this->dirty);
    }

    static function __create($vars,&$errors) {
        $filter = new static($vars);
        if ($filter->save()) {
            $filter->save_rules($vars, $errors);
            $filter->save_actions($filter->getId(), $vars, $errors);
            return $filter;
        }
    }

    static function create($vars,&$errors) {
        $filter = new static($vars);
        if ($filter->save())
            return $filter;
    }

    static function validate_actions($vars, &$errors) {
        //allow the save if it is to set a filter flag
        if ($vars['pass'])
            return true;

        if (!is_array(@$vars['actions']))
            return;
        foreach ($vars['actions'] as $sort=>$v) {
          if (is_array($v)) {
              $info = $v['type'];
              $sort = $v['sort'] ?: $sort;
          } else
              $info = substr($v, 1);
          $action = new FilterAction(array(
              'type'=>$info,
              'sort' => (int) $sort,
          ));
          $err = array();
          $action->setConfiguration($err, $vars);
          $config = json_decode($action->ht['configuration'], true);
          if (is_numeric($action->ht['type'])) {
              foreach ($config as $key => $value) {
                  switch ($key) {
                      case 'topic_id':
                          $action->ht['type'] = 'topic';
                          $config['topic_id'] = $value;
                          break;
                      case 'dept_id':
                          $action->ht['type'] = 'dept';
                          $config['dept_id'] = $value;
                          break;
                      case 'sla_id':
                          $action->ht['type'] = __('SLA');
                          break;
                      case 'team_id':
                          $action->ht['type'] = __('Team');
                          break;
                      case 'staff_id':
                          $action->ht['type'] = __('Agent');
                          break;
                      case 'status_id':
                          $action->ht['type'] = __('Ticket Status');
                          break;
                      case 'canned_id':
                          $action->ht['type'] = __('Canned Response');
                          break;
                      default:
                          $action->ht['type'] = __('All Actions');
                          break;
                  }
              }
          }

          // do not throw an error if we are deleting an action
          if (substr($v, 0, 1) != 'D') {
              switch ($action->ht['type']) {
                case 'dept':
                  $dept = Dept::lookup($config['dept_id']);
                  if (!$dept || !$dept->isActive()) {
                    $errors['err'] = sprintf(__('Unable to save: Please choose an active %s'), 'Department');
                  }
                  break;
                case 'topic':
                  $topic = Topic::lookup($config['topic_id']);
                  if (!$topic || !$topic->isActive()) {
                    $errors['err'] = sprintf(__('Unable to save: Please choose an active %s'), 'Help Topic');
                  }
                  break;
                default:
                  foreach ($config as $key => $value) {
                    if (!$value || is_null($value)) {
                        $errors['err'] = sprintf(__('Unable to save: Please insert a value for %s'), $action->ht['type']);
                        return 1;
                    }
                  }
                  break;
              }
          }
      }
      if (count($errors) == 0) {
          $fa = FilterAction::lookup($info);
          if ($fa) {
              $filter = Filter::lookup($fa->getFilterId());
              //Clear flags that may have been set on a successful save
              $filter->setFlag(constant('Filter::FLAG_DELETED_OBJECT'), false);
              $filter->setFlag(constant('Filter::FLAG_INACTIVE_DEPT'), false);
              $filter->setFlag(constant('Filter::FLAG_INACTIVE_HT'), false);
          }
      }
      return count($errors) == 0;
    }

    function save_actions($id, $vars, &$errors) {
        if (!is_array(@$vars['actions']))
            return;
        foreach ($vars['actions'] as $sort=>$v) {
            if (is_array($v)) {
                $info = $v['type'];
                $sort = $v['sort'] ?: $sort;
                $action = 'N';
            }
            else {
                $action = $v[0];
                $info = substr($v, 1);
            }
            switch ($action) {
            case 'N': # new filter action
                $I = new FilterAction(array(
                    'type'=>$info,
                    'filter_id'=>$id,
                    'sort' => (int) $sort,
                ));
                $I->setConfiguration($errors, $vars);
                $I->save();
                break;
            case 'I': # existing filter action
                if ($I = FilterAction::lookup($info)) {
                    $I->setConfiguration($errors, $vars);
                    $I->sort = (int) $sort;
                    $I->save();
                }
                break;
            case 'D': # deleted filter action
                if ($I = FilterAction::lookup($info))
                    $I->delete();
                break;
            }
        }
    }
}
Signal::connect('object.deleted', array('Filter', 'disableFilters'));

class FilterRule
extends VerySimpleModel {
    static $meta = array(
        'table' => FILTER_RULE_TABLE,
        'pk' => array('id'),
        'joins' => array(
            'filter' => array(
                'constraint' => array('filter_id' => 'Filter.id'),
            ),
        ),
    );

    function getId() {
        return $this->id;
    }

    function isActive() {
        return ($this->isactive);
    }

    function getHashtable() {
        return $this->ht;
    }

    function getInfo() {
        return $this->getHashtable();
    }

    function getFilterId() {
        return $this->filter_id;
    }

    function getFilter() {
        return $this->filter;
    }

    function update($vars, &$errors) {
        if (!$vars['filter_id'])
            $errors['err']=__('Parent filter ID required');

        if ($errors)
            return false;

        $this->what = $vars['what'];
        $this->how = $vars['how'];
        $this->val = $vars['val'];
        $this->isactive = isset($vars['isactive']) ? (int) $vars['isactive'] : 1;

        if (isset($vars['notes']))
            $this->notes = Format::sanitize($vars['notes']);

        if ($this->save())
            return true;
    }

    function save($refetch=false) {
        if ($this->dirty)
            $this->updated = SqlFunction::NOW();

        return parent::save($refetch || $this->dirty);
    }
}

/**
 * Applies rules defined in the admin control panel > Settings tab > "Ticket Filters". Each
 * filter can have up to 25 rules (*currently). This will attempt to match
 * the incoming tickets against the defined rules, and, if the email matches,
 * the ticket will be modified as described in the filter actions.
 */
class TicketFilter {

    var $target;
    var $vars;

    /**
     * Construct a list of filters to handle a new ticket
     * taking into account the source/origin of the ticket.
     *
     * $vars is an ARRAY, which has valid keys
     *  *email - email address of user
     *   name - name of user
     *   subject - subject of the ticket
     *   emailId - id of osTicket's system email (for emailed tickets)
     *  ---------------
     *  @see Filter::matches() for a complete list of supported keys
     */
    function __construct($origin, $vars=array()) {

        //Normalize the target based on ticket's origin.
        $this->target = self::origin2target($origin);

        //Extract the vars we care about (fields we filter by!).
        $this->vars = array('body'=>$vars['message']);
        $interest = Filter::getSupportedMatchFields();
        // emailId is always significant to the filter process
        $interest[] = 'emailId';
        foreach ($vars as $k=>$v) {
            if (in_array($k, $interest))
                $this->vars[$k] = trim($v);
        }
        if (isset($vars['recipients']) && $vars['recipients']) {
            foreach ($vars['recipients'] as $r) {
                $this->vars['addressee'][] = $r['name'];
                $this->vars['addressee'][] = $r['email'];
            }
            $this->vars['addressee'] = implode(' ', $this->vars['addressee']);
        }

         //Init filters.
        $this->build();
    }

    function build() {
        //Clear any memoized filters
        $this->filters = array();
        $this->short_list = null;

        //Query DB for "possibly" matching filters.
        foreach ($this->getAllActive() as $filter)
            $this->filters[] = $filter;

        return $this->filters;
    }

    function getTarget() {
        return $this->target;
    }

    /**
     * Fetches the short list of filters that match the ticket vars received in the
     * constructor. This function is memoized so subsequent calls will
     * return immediately.
     */
    function getMatchingFilterList() {

        if (!isset($this->short_list)) {
            $this->short_list = array();
            foreach ($this->filters as $filter)
                if ($filter->matches($this->vars))
                    $this->short_list[] = $filter;
        }

        return $this->short_list;
    }
    /**
     * Determine if any filters match the received email, and if so, apply
     * actions defined in those filters to the ticket-to-be-created.
     *
     * Throws:
     * RejectedException if the email should not be acceptable. If the email
     * should be rejected, the first filter that matches and has reject
     * ticket set is returned.
     */
    function apply(&$ticket, $postCreate=false) {
        foreach ($this->getMatchingFilterList() as $filter) {
            $filter->apply($ticket, $this->vars, $postCreate);
            if ($filter->stopOnMatch()) break;
        }
    }

    function getAllActive() {
        $filters = Filter::objects()->filter([
            'isactive' => 1,
            'target__in' => array('Any', $this->getTarget()),
        ]);

        #Take into account email ID.
        if ($this->vars['emailId'])
            $filters = $filters->filter([
                'email_id__in' => array(0, $this->vars['emailId'])
            ]);

        return $filters->order_by('execorder')->all();
    }

    /**
     * Simple true/false if the headers of the email indicate that the email
     * is an automatic response.
     *
     * Thanks to http://wiki.exim.org/EximAutoReply
     * X-Auto-Response-Supress is outlined here,
     *    http://msdn.microsoft.com/en-us/library/ee219609(v=exchg.80).aspx
     */
    /* static */
    static function isAutoReply($headers) {

        if($headers && !is_array($headers))
            $headers = Mail_Parse::splitHeaders($headers);

        $auto_headers = array(
            'Auto-Submitted'    => array('AUTO-REPLIED', 'AUTO-GENERATED'),
            'Precedence'        => array('AUTO_REPLY', 'BULK', 'JUNK', 'LIST'),
            'X-Precedence'      => array('AUTO_REPLY', 'BULK', 'JUNK', 'LIST'),
            'X-Autoreply'       => 'YES',
            'X-Auto-Response-Suppress' => array('AutoReply'),
            'X-Autoresponse'    => '*',
            'X-AutoReply-From'  => '*',
            'X-Autorespond'     => '*',
            'X-Mail-Autoreply'  => '*',
            'X-Autogenerated'   => 'REPLY',
            'X-AMAZON-MAIL-RELAY-TYPE' => 'NOTIFICATION',
        );

        foreach ($auto_headers as $header=>$find) {
            $header = strtolower($header);
            if(!isset($headers[$header])) continue;

            $value = strtoupper($headers[$header]);
            # Search text must be found at the beginning of the header
            # value. This is especially import for something like the
            # subject line, where something like an autoreponse may
            # appear somewhere else in the value.

            if (is_array($find)) {
                foreach ($find as $f)
                    if (strpos($value, $f) === 0)
                        return true;
            } elseif ($find === '*') {
                return true;
            } elseif (strpos($value, $find) === 0) {
                return true;
            }
        }

        return false;
    }

    static function isBounce($headers) {
        if ($headers && !is_array($headers))
            $headers = Mail_Parse::splitHeaders($headers);

        $bounce_headers = array(
            'From'  => array('stripos',
                        array('MAILER-DAEMON', '<>', 'postmaster@'), null, false),
            'Subject'   => array('stripos',
                array('DELIVERY FAILURE', 'DELIVERY STATUS',
                    'UNDELIVERABLE:', 'Undelivered Mail Returned'), 0),
            'Return-Path'   => array('strcmp', array('<>'), 0),
            'Content-Type'  => array('stripos', array('report-type=delivery-status'), null, false),
            'X-Failed-Recipients' => array('strpos', array('@'), null, false)
        );

        foreach ($bounce_headers as $header => $find) {
            $header = strtolower($header);
            if(!isset($headers[$header])) continue;

            @list($func, $searches, $pos, $neg) = $find;

            if(!($value = $headers[$header]) || !is_array($searches))
                continue;

            foreach ($searches as $f) {
                $result = call_user_func($func, $value, $f);
                if (($pos === null && $result !== $neg) or ($result === $pos))
                    return true;
            }
        }

        return false;
    }

    /**
     * Normalize ticket source to supported filter target
     *
     */
    static function origin2target($origin) {
        $sources=array('web' => 'Web', 'email' => 'Email', 'phone' => 'Web', 'staff' => 'Web', 'api' => 'API');

        return $sources[strtolower($origin)];
    }
}

class RejectedException extends Exception {
    var $filter;
    var $vars;

    function __construct(Filter $filter, $vars) {
        parent::__construct('Ticket rejected by a filter');
        $this->filter = $filter;
        $this->vars = $vars;
    }

    function getRejectingFilter() {
        return $this->filter;
    }

    function get($what) {
        return $this->vars[$what];
    }
}

class FilterDataChanged extends Exception {
    var $data;

    function __construct($data) {
         parent::__construct('Ticket filter data changed');
         $this->data = $data;
    }

    function getData() {
        return $this->data;
    }
}

/**
 * Function: endsWith
 *
 * Returns TRUE if the haystack ends with needle and FALSE otherwise.
 * Thanks, http://stackoverflow.com/a/834355
 */
function iendsWith($haystack, $needle)
{
    $length = mb_strlen($needle);
    if ($length == 0) {
        return true;
    }

    return (strcasecmp(mb_substr($haystack, -$length), $needle) === 0);
}

function pregMatchB($subject, $pattern) {
    return preg_match($pattern, $subject);
}
?>

Youez - 2016 - github.com/yon3zu
LinuXploit