source: branches/2.3/expressoMail1_2/inc/class.ScriptS.inc.php @ 5120

Revision 5120, 21.8 KB checked in by rafaelraymundo, 13 years ago (diff)

Ticket #2295 - Filtro bloqueia recebimento de todos emails

  • Property svn:eol-style set to native
  • Property svn:executable set to *
Line 
1<?php
2
3//Inclui o arquivo contendo a classe SieveS;
4include('class.SieveS.inc.php');
5
6class ScriptS {
7
8    //Declaração de Variáveis;
9    var $SieveS;   // Var para criação do objeto;
10    var $reply;  // Var para resposta;
11    var $scriptfile;  // Nome do script;
12    var $username;   // Nome do usuario;
13    var $rules;  // Regras do sieve;
14    var $errstr;   // Erros retornados;
15    var $size;  // Tamanho;
16    var $so;  // Verifica se a regra foi criada por outro tipo serviço de filtros;
17    var $continuebit;
18    var $sizebit;
19    var $allofbit;
20    var $keepbit;
21    var $regexbit;
22    var $newrules = array();
23    var $newout;
24    var $teste;
25    var $EmailVoip;
26    var $EmailExpresso;
27
28    function ScriptS() {
29
30        //Cria o objeto;
31        $this->SieveS = new SieveS();
32
33        //$this->scriptfile = $GLOBALS['HTTP_SESSION_VARS']['phpgw_info']['expressomail']['user']['account_lid'];
34        $this->scriptfile = $_SESSION['phpgw_info']['expressomail']['user']['account_lid'];
35        $this->username = $this->scriptfile;
36
37        $this->reply    = "";
38        $this->rules    = "";
39        $this->errstr   = "";
40        $this->size     = "";
41
42        $this->continuebit  = 1;
43        $this->sizebit      = 2;
44        $this->allofbit     = 4;
45        $this->keepbit      = 8;
46        $this->regexbit     = 128;
47
48        $this->EmailVoip = trim($_SESSION['phpgw_info']['user']['preferences']['expressoMail']['voip_email_redirect']);
49        $this->EmailExpresso = trim($_SESSION['phpgw_info']['expressomail']['user']['email']);
50    }
51
52    function init_a() {
53
54        //Abre a conexão
55        $this->SieveS->start();
56
57        $this->reply = $this->SieveS->getscript();
58
59        if (!$this->reply) {
60
61            $aux = $this->SieveS->putscript($this->scriptfile, $this->createScript());
62            if (!$aux) {
63                // Caso de erro, grava dentro da variável errstr;
64                $this->errstr = "Error: file not created";
65                return $this->errstr;
66            }
67            // Mata a variavel;
68            unset($aux);
69            // Ativa o script;
70            $aux = $this->SieveS->activatescript($this->scriptfile);
71
72            if (!$aux) {
73                // Caso de erro, grava dentro da variavel errstr;
74                $this->errstr = "Error: error to activate file";
75                return $this->errstr;
76            } else {
77                $this->reply = $this->SieveS->getscript();
78                $this->rules = $this->readScript($this->reply);
79            }
80        } else {
81            $this->rules = $this->readScript($this->reply);
82        }
83
84        //Fecha a conexao
85        $this->SieveS->close();
86
87        if ($this->rules)
88            return $this->rules;
89    }
90
91    function rec_rules($params) {
92
93        $newr1 = array();
94        $newr2 = array();
95        $newr3 = array();
96
97        $var_decode = rawurldecode($params['arfilter']);
98        $var_decode = preg_replace('/\n\./', '.', $var_decode);
99
100        $narray = explode("_end_", $var_decode);
101
102        foreach ($narray as $key => $tmp) {
103            if ($tmp != "") {
104                $newr1[] = $tmp;
105            }
106        }
107        unset($key);
108        unset($tmp);
109        foreach ($newr1 as $key => $tmp) {
110            $tmp2 = explode("_begin_##", $tmp);
111            foreach ($tmp2 as $tmp3) {
112                if ($tmp3 != "") {
113                    $newr2[] = trim($tmp3);
114                }
115            }
116        }
117
118        unset($tmp);
119        unset($tmp2);
120        unset($tmp3);
121
122        foreach ($newr2 as $tmp) {
123            if (trim($tmp) != "") {
124                $tmp2 = explode("##", $tmp);
125                foreach ($tmp2 as $tmp3) {
126                    $tmp4 .= trim($tmp3) . "&&";
127                }
128                $newr3[] = substr($tmp4, 0, (strlen($tmp4) - 4));
129                unset($tmp2);
130                unset($tmp3);
131                unset($tmp4);
132            }
133        }
134
135        $tmp = $newr3[count($newr3) - 1];
136
137        if (substr($tmp, 0, 9) == "#vacation") {
138            $this->newout = array_pop($newr3);
139            foreach ($newr3 as $key => $tmp) {
140                $this->newrules[] = $tmp;
141            }
142        } else {
143            foreach ($newr3 as $tmp) {
144                $this->newrules[] = $tmp;
145            }
146        }
147
148        unset($tmp);
149        $tmp = explode("&&", $this->newout);
150        $tmp1 = explode(",", $tmp[2]);
151        foreach ($tmp1 as $key => $tmp2) {
152            $tmp3 .= stripslashes(trim($tmp2)) . ", ";
153        }
154        $tmp3 = substr($tmp3, 0, (strlen($tmp3) - 2));
155
156        unset($tmp);
157        unset($tmp1);
158        unset($tmp2);
159        unset($key);
160        $tmp = explode("&&", $this->newout);
161        foreach ($tmp as $key => $tmp1) {
162            if ($key == 2) {
163                $tmp2 .= trim($tmp3) . "&&";
164            } else {
165                $tmp2 .= trim($tmp1) . "&&";
166            }
167        }
168        unset($this->newout);
169        $this->newout = substr($tmp2, 0, (strlen($tmp2) - 2));
170
171        //Abre a conexao
172        $this->SieveS->start();
173        $this->errstr = "";
174
175        // Escreve a nova regra;
176        $this->reply = $this->SieveS->getscript();
177        /*
178          if($this->reply){
179          $this->errstr = $this->SieveS->deletescript($this->scriptfile);
180          }
181         */
182
183        $error_log_file = "/home/expressolivre/sieve_error.log";
184        //Escreve a(s) nova(s) regra(s);
185        $newrule = $this->write_rule();
186        if (strlen($newrule) > 0)
187            $this->errstr = $this->SieveS->putscript($this->scriptfile, $newrule);
188        else {
189            if ($_SESSION['phpgw_info']['server']['expressomail']['expressoMail_enable_log_messages'] == "True")
190                error_log(date("D M j G:i:s T Y") . ": SieveError, Invalid rule for "
191                        . $_SESSION['phpgw_info']['expressomail']['user']['userid'] . "=>"
192                        . $this->teste . "\nRule:"
193                        . $var_decode . "\n", 3, $error_log_file);
194            return "Invalid rule\n" . $this->teste;
195        }
196
197        //Ativa o script;
198        $this->errstr = $this->SieveS->activatescript($this->scriptfile);
199
200        //Fecha a conexao
201        $this->SieveS->close();
202
203        if ($this->errstr) {
204            return "Ok";
205        } else {
206            if ($_SESSION['phpgw_info']['server']['expressomail']['expressoMail_enable_log_messages'] == "True")
207                error_log(date("D M j G:i:s T Y")
208                        . ": SieveError, Problem for "
209                        . $_SESSION['phpgw_info']['expressomail']['user']['userid'] . "=>"
210                        . " "
211                        . $this->SieveS->errstr . "\n", 3, $error_log_file);
212            return "Problemas na criação do arquivo!\n" . $this->teste;
213        }
214    }
215
216    function convert_specialchar($input) {
217        $special_char = false;
218        for ($i = 0; $i < strlen($input); $i++) {
219            if (preg_match('/[^a-z0-9.@~_+=&\'*^\ \-\[\]]/i', $input[$i])) {
220                $special_char = true;
221                $input = preg_replace('/\\' . $input[$i] . '/', '=' . bin2hex($input[$i]), $input);
222            }
223        }
224
225        if ($special_char) {
226            /*             *
227             * When using arrays with pattern and replacement,
228             * the keys are processed in the order they appear in the array.
229             * See preg_replace in php.net/preg_replace
230             * */
231            $patterns[0] = '/=c3/i';
232            $patterns[1] = '/ /';
233            $replacements[1] = '';
234            $replacements[0] = '_';
235            $input = preg_replace($patterns, $replacements, $input);
236        }
237
238        return($input);
239    }
240
241    // build the rule
242    function write_rule() {
243
244        // Variaveis;
245        $rule = array();
246        $vacation = array();
247        $newruletext = "";
248        $activerules = 0;
249        $regexused = 0;
250        $rejectused = 0;
251        $notify = 0;
252        $flaggedused = 0;
253        $newscriptbody = "";
254        $continue = 1;
255        $tmpSubject = "";
256
257        $a = 0;
258
259
260        // Recebe os valores das regras;
261        foreach ($this->newrules as $tmp)
262        {
263            $tmp1 = explode("&&", $tmp);
264            $rule['priority']   = $tmp1[1];
265            $rule['status']     = $tmp1[2];
266            $rule['from']       = $this->convert_specialchar($tmp1[3]);
267            $rule['to']         = $this->convert_specialchar($tmp1[4]);
268            $tmpSubject         = $tmp1[5];
269            $rule['subject']    = $rule['subject'] = " [\"" . $this->convert_specialchar($tmp1[5]) . "\", \"" . base64_encode($tmp1[5]) . "\"]";
270            $rule['action']     = $tmp1[6];
271            $rule['action_arg'] = utf8_encode(preg_replace("/\\r\\n/", "\r\n", $tmp1[7]));
272            $rule['flg']        = $tmp1[8];
273            $rule['field']      = $tmp1[9];
274            $rule['field_val']  = $tmp1[10];
275            $rule['size']       = trim($tmp1[11]);
276            $rule['continue']   = ($tmp1[8] & $this->continuebit);
277            $rule['gthan']      = ($tmp1[8] & $this->sizebit);
278            $rule['allof']      = ($tmp1[8] & $this->allofbit);
279            $rule['keep']       = ($tmp1[8] & $this->keepbit);
280            $rule['regexp']     = ($tmp1[8] & $this->regexbit);
281            $rule['unconditional'] = 0;
282
283            if( $a < 2 )
284            {
285                error_log( print_r($tmp, true), 3, "/tmp/log" );
286
287                error_log( print_r($rule, true), 3, "/tmp/log" );
288
289                $a++;
290            }
291
292
293            if (!$rule['from'] && !$rule['to'] && !$rule['subject'] && !$rule['field'] && empty($rule['size']) && $rule['action']) {
294                $rule['unconditional'] = 1;
295
296                if ($rule['unconditional'] && ($rule['size'] == "0" || $rule['size'] == 0 ))
297                    $rule['unconditional'] = 0;
298            }
299           
300            unset($tmp1);
301
302            // Monta as regras;
303            if ($rule['status'] == 'ENABLED')
304            {
305                $activerules = 1;
306
307                // Condições para montagem das regras;
308                $anyall = "anyof";
309                if ($rule['allof'])
310                {
311                    $anyall = "allof";
312                }
313               
314                if ($rule['regexp']) {
315                    $regexused = 1;
316                }
317                $started = 0;
318
319                if (!$rule['unconditional']) {
320                    if (!$continue)
321                        $newruletext .= "els";
322                    $newruletext .= "if " . $anyall . " (";
323                    if ($rule['from']) {
324                        if (preg_match("/^\s*!/", $rule['from'])) {
325                            $newruletext .= 'not ';
326                            $rule['from'] = preg_replace("/^\s*!/", "", $rule['from']);
327                        }
328                        $match = ':contains';
329                        if (preg_match("/\*|\?/", $rule['from']))
330                            $match = ':matches';
331                        if ($rule['regexp'])
332                            $match = ':regex';
333                        $newruletext .= "header " . $match . " [\"From\"]";
334                        $newruletext .= " \"" . $rule['from'] . "\"";
335                        $started = 1;
336                    }
337                    if ($rule['to']) {
338                        if ($started)
339                            $newruletext .= ", ";
340                        if (preg_match("/^\s*!/", $rule['to'])) {
341                            $newruletext .= 'not ';
342                            $rule['to'] = preg_replace("/^\s*!/", "", $rule['to']);
343                        }
344                        $match = ':contains';
345                        if (preg_match("/\*|\?/", $rule['to']))
346                            $match = ':matches';
347                        if ($rule['regexp'])
348                            $match = ':regex';
349                        $newruletext .= "address " . $match . " [\"To\",\"TO\",\"Cc\",\"CC\"]";
350                        $newruletext .= " \"" . $rule['to'] . "\"";
351                        $started = 1;
352                    }
353                    if ($rule['subject']) {
354                        if ($started)
355                            $newruletext .= ", ";
356                        if (preg_match("/^\s*!/", $rule['subject'])) {
357                            $newruletext .= 'not ';
358                            $rule['subject'] = preg_replace("/^\s*!/", "", $rule['subject']);
359                        }
360                        $match = ':contains';
361                        if (preg_match("/\*|\?/", $rule['subject']))
362                            $match = ':matches';
363                        if ($rule['regexp'])
364                            $match = ':regex';
365                        $newruletext .= "header " . $match . " \"subject\"";
366                        $newruletext .= "" . $rule['subject'] . "";
367                        $started = 1;
368                    }
369                    if ($rule['field'] && $rule['field_val']) {
370                        if ($started)
371                            $newruletext .= ", ";
372                        if (preg_match("/^\s*!/", $rule['field_val'])) {
373                            $newruletext .= 'not ';
374                            $rule['field_val'] = preg_replace("/^\s*!/", "", $rule['field_val']);
375                        }
376                        $match = ':contains';
377                        if (preg_match("/\*|\?/", $rule['field_val']))
378                            $match = ':matches';
379                        if ($rule['regexp'])
380                            $match = ':regex';
381                        $newruletext .= "header " . $match . " \"" . $rule['field'] . "\"";
382                        $newruletext .= " \"" . $rule['field_val'] . "\"";
383                        $started = 1;
384                    }
385
386                    if ($rule['size'] != '')
387                    {
388                        if ( $rule['size'] == 0 && $rule['gthan'] )
389                        {
390                            $xthan = " :over ";
391                           
392                            if ($started)
393                                $newruletext .= ", ";
394
395                            $newruletext .= "size " . $xthan . "0K";
396                            $started = 1;
397                        }
398
399                        if ( $rule['size'] > 0 )
400                        {
401                            $xthan = " :under ";
402
403                            if ($rule['gthan'])
404                                $xthan = " :over ";
405                            if ($started)
406                                $newruletext .= ", ";
407
408                            $newruletext .= "size " . $xthan . $rule['size'] . "K";
409                            $started = 1;
410                        }
411                    }
412                }
413
414                // Don't write half rule!
415                if (strlen($newruletext) == 0)
416                    return false;
417                // Actions
418                if (!$rule['unconditional'])
419                    $newruletext .= ") {\n\t";
420
421                if (preg_match("/folder/i", $rule['action'])) {
422                    $newruletext .= "fileinto \"" . $rule['action_arg'] . "\";";
423                }
424
425                if (preg_match("/reject/i", $rule['action'])) {
426                    $newruletext .= "reject text: \n" . $rule['action_arg'] . "\n.\n;";
427                    $rejectused = 1;
428                }
429                if (preg_match("/flagged/i", $rule['action'])) {
430                    $newruletext .= "addflag \"\\\\Flagged\";";
431                    $flaggedused = 1;
432                }
433                if (preg_match("/address/i", $rule['action'])) {
434                    $newruletext .= "redirect \"" . $rule['action_arg'] . "\";";
435                }
436
437                if (preg_match("/notify/i", $rule['action'])) {
438                    $newruletext .= "notify :method \"mailto\" :options [\"" . $this->EmailVoip . "\"]:" .
439                            "message \"<expressovoip><from>" . $this->EmailExpresso . "</from>" .
440                            "<br/><Subject>" . utf8_encode($tmpSubject) . "</Subject></expressovoip>\";";
441                    $notify = 1;
442                }
443
444                if (preg_match("/discard/i", $rule['action'])) {
445                    $newruletext .= "discard;";
446                }
447
448                if ($rule['keep'] )
449                {
450                    $newruletext .= "\n\tfileinto \"INBOX\";";
451                }
452
453                if (!$rule['unconditional'])
454                {
455                    $newruletext .= "\n}";
456                }
457
458                $continue = 0;
459                if ($rule['continue'])
460                    $continue = 1;
461                if ($rule['unconditional'])
462                    $continue = 1;
463
464                $newscriptbody .= $newruletext . "\n\n";
465                unset($newruletext);
466            }
467        }// Fim do Foreach;
468        $this->teste = $newscriptbody;
469        // Para a regras fora do escritorio;
470        unset($tmp);
471        if ($this->newout != "") {
472            $aux = explode("&&", $this->newout);
473            $vacation['days'] = $aux[1];
474            $vacation['addresses'] = $aux[2];
475            $vacation['text'] = preg_replace("/\\\\n/", "\r\n", $aux[3]);
476            $vacation['status'] = $aux[4];
477        }
478
479        // Monta a regra para fora do escritorio;
480        if ($vacation['status'] == 'on') {
481            $newscriptbody .= "vacation :days " . $vacation['days'] . " :addresses [";
482            $newscriptbody .= $vacation['addresses'];
483            $newscriptbody .= "] text:\n" . utf8_encode($vacation['text']) . "\n.\n;\n\n";
484        }
485
486        // Cria o cabeçalho do arquivo;
487        $newscripthead = "";
488        $newscripthead .= "#Mail filter rules for " . $this->username . "\n";
489        $newscripthead .= '#Generated by ' . $this->username . ' using Expressomail ';
490        $newscripthead .= "\n";
491
492        // Continuação do cabeçalho do arquivo;                 
493        if ($activerules) {
494            $newscripthead .= "require [\"fileinto\"";
495
496            if ($notify) {
497                $newscripthead .= ",\"notify\"";
498            }
499            if ($regexused) {
500                $newscripthead .= ",\"regex\"";
501            }
502            if ($rejectused) {
503                $newscripthead .= ",\"reject\"";
504            }
505            if ($flaggedused) {
506                $newscripthead .= ",\"imapflags\"";
507            }
508            if ($this->newout && $vacation['status'] == 'on') {
509                $newscripthead .= ",\"vacation\"";
510            }
511            $newscripthead .= "];\n\n";
512        } else {
513            if ($vacation && $vacation['status'] == 'on') {
514                $newscripthead .= "require [\"vacation\"];\n\n";
515            }
516        }
517
518        // Cria o rodapé do arquivo;
519        $newscriptfoot = "";
520        $newscriptfoot .= "##PSEUDO script start\n";
521        // Lê as regras;
522        foreach ($this->newrules as $tmp) {
523            $newscriptfoot .= preg_replace("/[\\n\\r]/", " ", $tmp) . "\n";
524        }
525        // Lê as regras fora do escritório;
526        if ($this->newout != "") {
527            $newscriptfoot .= preg_replace("/[\\n\\r]/", "%0A", $this->newout) . "\n";
528        }
529        $newscriptfoot .= "#mode&&basic\n";
530
531        $newscript = $newscripthead . $newscriptbody . $newscriptfoot;
532        // Destroi as variaveis;
533        unset($rule);
534        unset($vacation);
535        unset($activerules);
536        unset($regexused);
537        unset($rejectused);
538        unset($flaggedused);
539        unset($newscripthead);
540        unset($newscriptbody);
541        unset($newscriptfoot);
542        unset($continue);
543        unset($this->newrules);
544        unset($this->newout);
545
546        // Retorna o script construido;
547        return $newscript;
548    }
549
550// Fim da Função
551    // Cria o script sieve, caso nao possua;
552    function createScript() {
553
554        // Cria o cabeçalho do arquivo;
555        $newScriptHead = "";
556        $newScriptHead .= "#Mail filter rules for " . $this->username . "\n";
557        $newScriptHead .= '#Generated by ' . $this->username . ' using ExpressoMail ';
558        $newScriptHead .= "\n";
559
560        //Cria o rodapé do arquivo;
561        $newScriptFoot = "";
562        $newScriptFoot .= "##PSEUDO Script Start\n";
563        $newScriptFoot .= "#mode&&basic\n";
564
565        //Para passar para o arquivo;
566        $newScript = $newScriptHead . $newScriptFoot;
567
568        return $newScript;
569    }
570
571    //Lê o conteúdo do script;
572    function readScript($scriptName) {
573
574        // Verifica se a conexão foi bem sucedida;
575        if (!$scriptName) {
576            $this->errstr = "Não foi possível conectar com o Servidor";
577            return "false 2";
578        }
579
580        // Recebe o conteúdo do array;
581        $lines = array();
582        $lines = preg_split("/\n/", $scriptName);
583
584        // Pega o tamanho da regra na primeira do script;
585        $size_rule = array_shift($lines);
586
587        // Recebe o tamanho do script, pela primeira linha;
588        $this->size = trim($size_rule);
589
590        // Verifica a composição do script;
591        $line = array_shift($lines);
592        if (!preg_match("/^# ?Mail(.*)rules for/", $line)) {
593            $this->errstr = "Formato nao reconhecido";
594            return false;
595        }
596
597        // Variaveis para a regra e o campo ferias;
598        $regexps = array('^ *##PSEUDO', '^ *#rule', '^ *#vacation', '^ *#mode');
599        $retorno['rule'] = array();
600        $retorno['vacation'] = array();
601        $retorno['mode'] = array();
602
603        $line = array_shift($lines);
604        while (isset($line)) {
605            foreach ($regexps as $regp) {
606                if (preg_match("/$regp/i", $line)) {
607                    // Recebe todas as regras criadas no servidor;
608                    if (preg_match("/^ *#rule&&/i", $line)) {
609                        $retorno['rule'][] = $line . "\n";
610                    }
611                    if (preg_match("/^ *#vacation&&/i", $line)) {
612                        $retorno['vacation'][] = $line . "\n";
613                    }
614                    if (preg_match("/^ *#mode&&(.*)/i", $line)) {
615                        $retorno['mode'][] = $line . "\n";
616                    }
617                }
618            }
619            // Pega a proxima linha do sript;
620            $line = array_shift($lines);
621        }
622        return $retorno;
623    }
624
625}
626
627//Fim da Classe
628?>
Note: See TracBrowser for help on using the repository browser.