Subversion Repositories Applications.projet

Rev

Rev 431 | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
431 mathias 1
<?php
2
// $Id: ezmlm-msgdisplay.php,v 1.9 2008-05-23 10:18:37 alexandre_tb Exp $
3
//
4
// ezmlm-msgdisplay.php - ezmlm-php v2.0
5
// --------------------------------------------------------------
6
// Will parse a template (if specified) and display a message.
7
// Includes a default template.
8
// --------------------------------------------------------------
9
 
10
require_once("ezmlm.php");
11
require_once("Mail/mimeDecode.php") ;
12
 
13
class ezmlm_msgdisplay extends ezmlm_php {
14
	// our template
15
	var $msgtmpl;
16
    var $message_rendu ;
17
    var $_auth ;
18
	// display: parses a message (using ezmlm_parser) and displays it
19
	// using a template
20
    var $msgfile;
21
 
22
    function display($msgfile) {
23
        if (!is_file($msgfile)) {
24
            if (is_file($this->listdir . "/" . $msgfile)) { $msgfile = $this->listdir . "/" . $msgfile; }
25
		else if (is_file($this->listdir . "/archive/" . $msgfile)) { $msgfile = $this->listdir . "/archive/" . $msgfile; }
26
		else { return FALSE; }
27
	    }
28
	$this->msgfile = $msgfile ;
29
        $message = file_get_contents($msgfile) ;
30
	// En cours de codage
31
	// La fonction display retourne tout simplement le source du mail
32
	// Il n'y a plus d'analyse a ce niveau
33
 
34
	return $message;
35
        $mimeDecode = new Mail_mimeDecode($message) ;
36
        $mailDecode = $mimeDecode->decode(array('decode_bodies' => 'true', 'include_bodies' => 'true')) ;
37
 
38
        // $msg->msgfile contient le chemin du fichier du mail en partant de la racine
39
        // Le point d'exclamation est le delimiteur de l'expression reguliere
40
		$relfile = preg_replace('!' . $this->listdir . '!', '', $msgfile);
41
 
42
		$a1 = preg_replace('!/archive/(.*)/.*$!', '\1', $relfile);  // $a1 contient le nom du repertoire
43
		$a2 = preg_replace('!/archive/.*/(.*)$!', '\1', $relfile);  // $a2 contient le nom du fichier
44
		if (isset($mailDecode->headers['date'])) $msgtime = strtotime(preg_replace ('/CEST/', '', $mailDecode->headers['date']));
45
        $threadidx = date("Ym", $msgtime);
46
        if ($a2 <= 10) $numero_precedent = '0'.($a2 - 1) ; else $numero_precedent = ($a2 - 1) ;
47
        if ($a2 < 9) $numero_suivant = '0'.($a2 + 1) ; else $numero_suivant =  ($a2 + 1);
48
        // On teste si le message suivant existe
49
        $decoupe = explode ('/', $msgfile) ;
50
 
51
        // Les nom de fichiers sont du format :
52
        // archive/0/01
53
        // archive/0/02 ... 0/99 archive/1/01 ...
54
 
55
        $nom_fichier = $decoupe[count($decoupe)-1] ;
56
        $nom_repertoire = $decoupe[count($decoupe)-2] ;
57
        $repertoire_suivant = $nom_repertoire ; $repertoire_precedent = $nom_repertoire ;
58
        if ($nom_fichier > 8) {
59
            $fichier_suivant = $nom_fichier + 1 ;
60
            if ($nom_fichier == 99) {
61
                $fichier_suivant = '01' ;
62
                $repertoire_suivant = $nom_repertoire + 1 ;
63
            }
64
        } else {
65
            $fichier_suivant = '0'.($nom_fichier + 1) ;
66
        }
67
        if ($nom_fichier > 10) {
68
            $fichier_precedent = $nom_fichier - 1 ;
69
        } else {
70
            if ($nom_fichier == '01') {
71
                $fichier_precedent = '99' ;
72
                $repertoire_precedent = $nom_repertoire - 1 ;
73
            } else {
74
                $fichier_precedent = '0'.($nom_fichier - 1) ;
75
            }
76
        }
77
        print $this->parse_entete_mail($mailDecode) ;
78
		$this->parse_template($mailDecode, $a2, $a1);
79
        print $this->message_rendu;
80
        //print '</div>' ;
81
	}
82
 
83
    /**
84
     * Renvoie les infos des messages suivants
85
     *
86
     *
87
    */
88
    function getInfoSuivant() {
89
		$relfile = preg_replace('!' . $this->listdir . '!', '', $this->msgfile);
90
	 	$nom_repertoire = preg_replace('!/archive/(.*)/.*$!', '\1', $relfile);
91
		$nom_fichier = preg_replace('!/archive/.*/(.*)$!', '\1', $relfile);
92
 
93
		$repertoire_suivant = $nom_repertoire;
94
 
95
		// On recupere le numero du dernier message
96
		if (file_exists($this->listdir.'/archnum')) {
97
			$numero_dernier_message = file_get_contents($this->listdir.'/archnum');
98
		}
99
 
100
		// a partir du nom du fichier
101
		// et du nom du repertoire, on reconstitue
102
		// le numero du message stocke dans le fichier d index
103
		// le message 12 du repertoire 2 a le numero 212
104
 
105
		if ($nom_repertoire == '0') {
106
			$numero_message = $nom_fichier;
107
		} else {
108
			$numero_message = $nom_repertoire.$nom_fichier ;
109
		}
110
 
111
		// On ouvre le fichier d index
112
		$fichier_index = fopen ($this->listdir.'/archive/'.$nom_repertoire.'/index', 'r');
113
 
114
 
115
		$compteur_ligne = 1;
116
		if (preg_match ('/0([1-9][0-9]*)/', $nom_fichier, $match)) {
117
			$nom_fichier = $match[1];
118
			$prefixe = '0' ;
119
		} else {
120
			$prefixe = '' ;
121
		}
122
		$prefixe = $this->prefixe_nom_message($nom_fichier);
123
		//echo $numero_message;
124
		// on cherche la ligne avec le numero du message
125
		while (!feof($fichier_index)) {
126
 
127
				$temp = fgets($fichier_index,4096);
128
				list($num, $hash, $sujet) = split (':', $temp) ;
129
 
130
				if ($num == $numero_message) {
131
 
132
					$ligne_message_precedent = $compteur_ligne -2;
133
					$temp = fgets($fichier_index, 4096);
134
					$temp = fgets($fichier_index, 4096);
135
					list ($fichier_suivant,$hash, $sujet) = split(':', $temp);
136
 
137
					// Au cas ou est au dernier message du fichier d index
138
					// il faut ouvrir le suivant
139
					if (feof($fichier_index)) {
140
						$repertoire_suivant++;
141
						if (file_exists($this->listdir.'/archive/'.$repertoire_suivant.'/index')) {
142
							$fichier_index_suivant = fopen($this->listdir.'/archive/'.$repertoire_suivant.'/index', 'r');
143
							// on recupere le numero du premier message
144
							list($fichier_suivant, $hash, $sujet) = split (':', fgets($fichier_index_suivant), 4096);
145
							fclose ($fichier_index_suivant);
146
						}
147
					}
148
 
149
					// Si le numero est > 100, il faut decouper et ne retenir
150
					// que les dizaines et unites
151
					if ($fichier_suivant >= 100) {
152
						$decimal = (string) $fichier_suivant;
153
						$numero = substr($decimal, -2) ;
154
						$fichier_suivant = $numero ;
155
					} else {
156
						if ($fichier_suivant <= 9)$fichier_suivant = '0'.$fichier_suivant;
157
					}
158
 
159
					break;
160
				}
161
 
162
				// On avance d une ligne, la 2e ligne contient date hash auteur
163
				$temp2 = fgets($fichier_index, 4096);
164
				$compteur_ligne += 2;
165
		}
166
 
167
		// On utilise $ligne_message_precedent pour recupere le num du message precedent
168
		// Si $ligne_precedent est negatif soit c le premier message de la liste
169
		// soit il faut ouvrir le repertoire precedent
170
 
171
		if ($ligne_message_precedent > 0) {
172
			$compteur = 1;
173
			rewind($fichier_index);
174
			while (!feof($fichier_index)) {
175
				$temp = fgets($fichier_index, 4096);
176
				if ($ligne_message_precedent == $compteur) {
177
					list ($fichier_precedent, $hash, $sujet) = split (':', $temp) ;
178
				}
179
				$compteur++;
180
			}
181
			// Le nom du repertoire precedent est le meme que le repertoire courant
182
			$repertoire_precedent = $nom_repertoire ;
183
		// Si $ligne_message_precedent est negatif, alors il faut ouvrir
184
		// le fichier index du repertoire precedent
185
		// si le nom du repertoire est 0, alors il n y a pas de repertoire precedent
186
		// et donc pas de message precedent
187
		} else {
188
 
189
			if ($nom_repertoire != '0') {
190
				$repertoire_precedent = $nom_repertoire -1 ;
191
				// on ouvre le fichier d index et on extraie le numero
192
				// du dernier message
193
 
194
				$fichier_index_precedent = fopen ($this->listdir.'/archive/'.$repertoire_precedent.'/index', 'r') ;
195
				while (!feof($fichier_index_precedent)) {
196
					$temp = fgets($fichier_index_precedent,4096);
197
					$ligne = split (':', $temp) ;
198
					if ($ligne[0] != '') $fichier_precedent = $ligne[0];
199
					$temp = fgets($fichier_index_precedent,4096);
200
				}
201
 
202
				fclose ($fichier_index_precedent);
203
			// on se situe dans le repertoire 0 donc pas de message precedent
204
			} else {
205
				$fichier_precedent = null;
206
				$repertoire_precedent = null;
207
			}
208
		}
209
		if ($fichier_precedent > 100) {
210
			$decimal = (string) $fichier_precedent;
211
			$numero = substr($decimal, -2) ;
212
			$fichier_precedent = $numero ;
213
		} else {
214
			if ($fichier_precedent < 10 )$fichier_precedent = '0'.$fichier_precedent;
215
		}
216
		fclose ($fichier_index);
217
		//if ($fichier_precedent != null && $fichier_precedent < 10) $fichier_precedent = '0'.$fichier_precedent;
218
 
219
		return array ('fichier_suivant' => $fichier_suivant,
220
			      'repertoire_suivant' => $repertoire_suivant,
221
			      'fichier_precedent' => $fichier_precedent,
222
			      'repertoire_precedent' => $repertoire_precedent);
223
	}
224
 
225
    /**
226
    *   analyse l'entete d'un mail pour en extraire les ent�tes
227
    *   to, from, subject, date
228
    *   met � jour la variable $this->msgtmpl
229
    *
230
    */
231
 
232
    function parse_entete_mail (&$mailDecode) {
233
        $startpos = strpos(strtolower($this->msgtmpl_entete), '<ezmlm-headers>');
234
        $endpos = strpos(strtolower($this->msgtmpl_entete), '</ezmlm-headers>');
235
		$headers = substr($this->msgtmpl_entete,$startpos + 15,($endpos - $startpos - 15));
236
        $headers_replace = '' ;
237
		for ($i = 0; $i < count($this->showheaders); $i++) {
238
		    $val = $this->showheaders[$i];
239
		    $headers_replace .= $headers;
240
		    $hnpos = strpos(strtolower($headers_replace), '<ezmlm-header-name>');
241
		    $headers_replace = substr_replace($headers_replace, $this->header_en_francais[$val], $hnpos, 19);
242
		    $hvpos = strpos(strtolower($headers_replace), '<ezmlm-header-value');
243
            $headers_replace = $this->decode_iso ($headers_replace) ;
244
            switch ($val) {
245
            	case 'date':
246
            	$headers_replace = substr_replace($headers_replace, $this->date_francaise($mailDecode->headers[strtolower($val)]), $hvpos, 20);
247
            	break;
248
            	case 'from':
249
            	if ($mailDecode->headers[strtolower($val)] == '') $from = $mailDecode->headers['return-path'] ;
250
            		else $from = $mailDecode->headers['from'];
251
            	$headers_replace = substr_replace($headers_replace, $this->protect_email($this->decode_iso($from)), $hvpos, 20);
252
            	//$headers_replace = htmlspecialchars($headers_replace);
253
            	break;
254
            	default:
255
            	$headers_replace = substr_replace($headers_replace, $this->protect_email($this->decode_iso($mailDecode->headers[strtolower($val)])), $hvpos, 20);
256
            }
257
		}
258
        return substr_replace($this->msgtmpl_entete, $headers_replace, $startpos, (($endpos + 16) - $startpos));
259
    }
260
 
261
 
262
	function parse_template(&$mailDecode, $numero_mail, $numero_mois, $num_part = '') {
263
        static $profondeur = array();
264
        if ($num_part != '') array_push ($profondeur, $num_part) ;
265
        $corps = '' ;
266
 
267
		if ($mailDecode->ctype_primary == 'multipart') {
268
            include_once PROJET_CHEMIN_CLASSES.'type_fichier_mime.class.php' ;
269
			for ($i = 0; $i < count($mailDecode->parts); $i++) {
270
                switch ($mailDecode->parts[$i]->ctype_secondary) {
271
                    case 'plain' :
272
                    if ($mailDecode->parts[$i]->headers['content-transfer-encoding'] == '8bit') {
273
                    	$corps .= $this->_cte_8bit($mailDecode->parts[$i]->body);
274
                    } else if ($mailDecode->parts[$i]->headers['content-transfer-encoding'] == 'quoted-printable') {
275
                    	if ($mailDecode->parts[$i]->ctype_parameters['charset'] == 'UTF-8') {
276
                    		$corps .= utf8_decode($mailDecode->parts[$i]->body);
442 mathias 277
                    	} else {
278
                    		// Si un multipart/related, qu'on ne sait pas decoder, contient une partie plain
279
                    		// qui n'est pas en UTF-8, faut bien la recuperer... cela dit, comprend pas comment
280
                    		// ça marche dans les autres cas, hors UTF-8
281
                    		$corps .= $mailDecode->parts[$i]->body;
431 mathias 282
                    	}
283
                    }
284
                    break;
442 mathias 285
                    case 'related':
286
                    	// patch pourri : comme "multipart/related" n'est pas gere, on ignore la partie
287
                    	// (se produit apparemment lorsqu'une signature avec image est envoyee, par Thunderbird
288
                    	// sous Windows en tout cas)
289
                    	break;
431 mathias 290
                    case 'html' : $corps .= trim(strip_tags ($mailDecode->parts[$i]->body, '<br><p><a><style>'));
291
                    break ;
292
                    case 'mixed' :
293
                    case 'rfc822' :
294
                    case 'alternative' :
295
                    case 'appledouble' :
296
                        $this->parse_template($mailDecode->parts[$i], $numero_mail, $numero_mois, $i) ;
297
                    break ;
298
                    case 'applefile' : continue ;
299
                    break ;
300
                    default :
301
 
302
                    if ($mailDecode->parts[$i]->ctype_secondary == 'octet-stream') {
303
                        $nom_piece_jointe = $mailDecode->parts[$i]->ctype_parameters['name'] ;
304
                        $tab = explode ('.', $nom_piece_jointe) ;
305
                        $extension = $tab[count ($tab) - 1] ;
306
                        $mimeType = type_fichier_mime::factory($extension);
307
                        $mimeType->setCheminIcone(PROJET_CHEMIN_ICONES) ;
308
                    } else {
309
                        $nom_piece_jointe = isset ($mailDecode->parts[$i]->d_parameters['filename']) ?
310
                                            $mailDecode->parts[$i]->d_parameters['filename'] : $mailDecode->parts[$i]->ctype_parameters['name'] ;
311
                        $mimeType = new type_fichier_mime( $mailDecode->parts[$i]->ctype_primary.'/'.
312
                                            $mailDecode->parts[$i]->ctype_secondary, PROJET_CHEMIN_ICONES) ;
313
                    }
314
                    $lien = PROJET_CHEMIN_APPLI.'fichier_attache.php?nom_liste='.$this->listname.
315
                                    '&actionargs[]='.$numero_mois.
316
                                    '&actionargs[]='.$numero_mail;
317
                    if (count ($profondeur) > 0) {
318
                        array_shift($profondeur) ;
319
                        for ($j= 0; $j < count ($profondeur); $j++) $lien .= '&actionargs[]='.$profondeur[$j];
320
                    }
321
                    $lien .= '&actionargs[]='.$i ;
322
                    $corps .= '<a href="'.$lien.'">';
323
 
324
					$tableau_type_image = array ('jpg', 'jpeg', 'pjpeg');
325
 
326
                    if (in_array ($mailDecode->parts[$i]->ctype_secondary, $tableau_type_image)) {
327
                    	$corps .= '<img src="'.$lien.'&amp;min=1" alt="'.$nom_piece_jointe.'" />&nbsp;' ;
328
                    	$texte_lien = '';
329
                    } else {
330
                    	$corps .= '<img src="'.$mimeType->getCheminIcone().'" alt="'.$nom_piece_jointe.'" />&nbsp;' ;
331
                    	$texte_lien = $nom_piece_jointe;
332
                    }
333
                    $corps .= $texte_lien;
334
                    $corps .= '</a><br />' ;
335
                    break ;
336
                }
337
            }
338
            $this->message_rendu .= preg_replace('/<ezmlm-body>/i', $this->cleanup_body($corps,TRUE), $this->msgtmpl);
339
 
340
		} else if ($mailDecode->ctype_primary == 'message') {
341
 
342
            $this->message_rendu .= "\n".'<div class="message">'.$this->parse_entete_mail($mailDecode->parts[0]);
343
            $corps .= $this->parse_template($mailDecode->parts[0], $numero_mail, $numero_mois, 0) ;
344
            $this->message_rendu .= preg_replace('/<ezmlm-body>/i', $this->cleanup_body($corps,true), $this->msgtmpl).'</div>';
345
 
346
        } else if ($mailDecode->ctype_primary == 'application' || $mailDecode->ctype_primary == 'image'){
347
            if ($mailDecode->ctype_secondary == 'applefile') return ;
348
            $mimeType = new type_fichier_mime( $mailDecode->ctype_primary.'/'.$mailDecode->ctype_secondary,PROJET_CHEMIN_ICONES) ;
349
 
350
            if ($mimeType->getIdType() != 12) {
351
                $corps .= '<a href="'.PROJET_CHEMIN_APPLI.'fichier_attache.php?nom_liste='.$this->listname.'&actionargs[]='.
352
                                    $numero_mois.'&actionargs[]='.
353
                                    $numero_mail.'&actionargs[]='.$i.'">'.
354
                                    '<img src="'.$mimeType->getCheminIcone().'" alt="'.$mailDecode->ctype_parameters['name'].'" />&nbsp;' ;
355
                $corps .= $mailDecode->ctype_parameters['name'].'</a><br />' ;
356
 
357
                $this->message_rendu .= preg_replace('/<ezmlm-body>/i', $this->cleanup_body($corps,true), $this->msgtmpl);
358
            }
359
        } else {
360
			if (preg_match('/html/i', $mailDecode->ctype_secondary)) {
361
                $this->message_rendu .= preg_replace('/<ezmlm-body>/i', $this->cleanup_body($mailDecode->body,TRUE), $this->msgtmpl);
362
            } else {
363
                if (isset ($mailDecode->ctype_parameters['charset']) && $mailDecode->ctype_parameters['charset'] == 'UTF-8') {
364
                    $this->message_rendu .= preg_replace('/<ezmlm-body>/i', '<pre>' . utf8_decode($this->cleanup_body($mailDecode->body,TRUE)) . '</pre>', $this->msgtmpl);
365
                } else {
366
                    $this->message_rendu .= preg_replace('/<ezmlm-body>/i', '<pre>' . $this->cleanup_body($mailDecode->body,TRUE) . '</pre>', $this->msgtmpl);
367
                }
368
            }
369
		}
370
		array_pop ($profondeur);
371
	}
372
 
373
	function ezmlm_msgdisplay() {
374
		$this->ezmlm_php();
375
		if (($this->msgtemplate != "") and (is_file($this->msgtemplate))) {
376
			$fd = fopen($this->msgtemplate, "r");
377
			while (!feof($fd)) { $this->msgtmpl .= fgets($fd,4096); }
378
			fclose($fd);
379
		} else {
380
			$this->msgtmpl = '<pre>
381
<ezmlm-body>
382
</pre>
383
        ';
384
		}
385
        $this->msgtmpl_entete = '<dl><ezmlm-headers>
386
<dt><ezmlm-header-name> :</dt>
387
<dd><ezmlm-header-value></dd>
388
</ezmlm-headers>
389
</dl>' ;
390
	}
391
 
392
		// _cte_8bit: decode a content transfer encoding of 8bit
393
	// NOTE: this function is a little bit special. Since the end result will be displayed in
394
	// a web browser _cte_8bit decodes ASCII characters > 127 (the US-ASCII table) into the
395
	// html ordinal equivilant, it also ensures that the messages content-type is changed
396
	// to include text/html if it changes anything...
397
	function _cte_8bit($data,$simple = FALSE) {
398
		if ($simple) { return $data; }
399
		$changed = FALSE;
400
		$out = '';
401
		$chars = preg_split('//',$data);
402
		while (list($key,$val) = each($chars)) {
403
			if (ord($val) > 127) { $out .= '&#' . ord($val) . ';'; $changed = TRUE; }
404
			else { $out .= $val; }
405
		}
406
		if ($changed) { $this->headers['content-type'][1] = 'text/html'; }
407
		return $out;
408
	}
409
 
410
}