486 |
aurelien |
1 |
<?php
|
|
|
2 |
// Encodage : UTF-8
|
|
|
3 |
// +-------------------------------------------------------------------------------------------------------------------+
|
|
|
4 |
/**
|
|
|
5 |
* Traitement des mails en attente de modération de l'annuaire
|
|
|
6 |
*
|
|
|
7 |
* Description : classe permettant de gérer l'envoi des mails en attente de modération dans l'annuaire
|
|
|
8 |
* Utilisation : php script.php mail
|
|
|
9 |
*
|
|
|
10 |
//Auteur original :
|
|
|
11 |
* @author Aurélien PERONNET <jpm@tela-botanica.org>
|
|
|
12 |
* @copyright Tela-Botanica 1999-2014
|
|
|
13 |
* @licence GPL v3 & CeCILL v2
|
|
|
14 |
* @version $Id$
|
|
|
15 |
*/
|
|
|
16 |
|
|
|
17 |
class Mail extends Script {
|
490 |
aurelien |
18 |
//TODO: cette classe est en doublon avec du code de l'annuaire
|
|
|
19 |
// une fois passé à la dernière version du framework, il faudrait factoriser ces fonctions
|
|
|
20 |
// dans une lib commune accessible aux scripts et au code standard
|
486 |
aurelien |
21 |
const STATUT_A_TRAITER = 'a_traiter';
|
|
|
22 |
const STATUT_EN_TRAITEMENT = 'en_traitement';
|
503 |
aurelien |
23 |
const STATUT_EN_ECHEC = 'en_echec';
|
486 |
aurelien |
24 |
|
490 |
aurelien |
25 |
// Définit le délai au bout du quel on remet des mails en traitement à traiter
|
|
|
26 |
// au format (avec la syntaxe utilisée avec INTERVAL en SQL)
|
|
|
27 |
// http://dev.mysql.com/doc/refman/5.0/fr/date-and-time-functions.html
|
|
|
28 |
const DELAI_MAX_TRAITEMENT = '10 HOUR';
|
|
|
29 |
|
486 |
aurelien |
30 |
private $modele = null;
|
|
|
31 |
|
|
|
32 |
public function executer() {
|
|
|
33 |
$this->bdd = new Bdd();
|
510 |
mathias |
34 |
// élargissement du timeout car le traitement est long;
|
|
|
35 |
// évite les erreurs 2006 "MySQL has gone away"
|
|
|
36 |
$this->bdd->executer("SET wait_timeout=300");
|
486 |
aurelien |
37 |
|
|
|
38 |
$cmd = $this->getParametre('a');
|
|
|
39 |
$this->mode_verbeux = $this->getParametre('v');
|
|
|
40 |
|
503 |
aurelien |
41 |
$retour = array();
|
|
|
42 |
|
486 |
aurelien |
43 |
switch($cmd) {
|
490 |
aurelien |
44 |
case "tous":
|
|
|
45 |
$retour = $this->traiterMailsEnAttente();
|
486 |
aurelien |
46 |
break;
|
|
|
47 |
// TODO: case supplémentaire pour traiter un mail par son id ?
|
|
|
48 |
// TODO: option "force" pour traiter les mails quelques soient leur statut ?
|
|
|
49 |
default:
|
490 |
aurelien |
50 |
}
|
|
|
51 |
|
|
|
52 |
if($this->mode_verbeux) {
|
503 |
aurelien |
53 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
54 |
echo 'Identifiants des mails traites : '.implode(',', $retour)."--";
|
|
|
55 |
}
|
486 |
aurelien |
56 |
}
|
|
|
57 |
|
|
|
58 |
private function traiterMailsEnAttente() {
|
|
|
59 |
// Gaston Lagaffe
|
503 |
aurelien |
60 |
$mails_en_retard = $this->traiterMailsEnRetard();
|
486 |
aurelien |
61 |
$mails_a_traiter = $this->obtenirMailsEnAttente();
|
|
|
62 |
|
|
|
63 |
$retour = array();
|
|
|
64 |
if(count($mails_a_traiter) > 0 && $this->mettreMailsEnCoursDeTraitement()) {
|
|
|
65 |
foreach($mails_a_traiter as $donnees_brutes_mail) {
|
|
|
66 |
$mail_a_moderer = $this->decoderDonneeTemporaire($donnees_brutes_mail);
|
|
|
67 |
$id_mail = $donnees_brutes_mail['adt_id'];
|
490 |
aurelien |
68 |
|
503 |
aurelien |
69 |
$resultat_envoi = true;
|
|
|
70 |
$envois_echoues = $this->envoyerMail($mail_a_moderer['expediteur'],
|
|
|
71 |
$mail_a_moderer['destinataires'],
|
|
|
72 |
$mail_a_moderer['sujet'],
|
486 |
aurelien |
73 |
$mail_a_moderer['message']);
|
|
|
74 |
|
490 |
aurelien |
75 |
if(empty($envois_echoues)) {
|
486 |
aurelien |
76 |
$this->supprimerMailTraite($id_mail);
|
490 |
aurelien |
77 |
} else {
|
|
|
78 |
// TODO: supprimer les destinataires qui ont fonctionné, et mettre à jour
|
|
|
79 |
// le mail dans les données temporaire avec les destinataires qui restent
|
|
|
80 |
// pour pouvoir finir de l'envoyer
|
|
|
81 |
$this->avertirModerateurEchecEnvoi($envois_echoues, $mail_a_moderer);
|
|
|
82 |
$resultat_envoi = false;
|
486 |
aurelien |
83 |
}
|
490 |
aurelien |
84 |
|
|
|
85 |
// TODO: logger également erreur d'envoi ?
|
486 |
aurelien |
86 |
$retour[$id_mail] = $resultat_envoi;
|
|
|
87 |
}
|
|
|
88 |
}
|
|
|
89 |
return $retour;
|
|
|
90 |
}
|
|
|
91 |
|
490 |
aurelien |
92 |
private function avertirModerateurEchecEnvoi($envois_echoues, $mail_a_moderer) {
|
|
|
93 |
|
503 |
aurelien |
94 |
$corps_mail_echoue = "L'envoi d'un mail modéré à échoué pour les destinataires suivants (".count($envois_echoues)." au total) : <br />";
|
|
|
95 |
$corps_mail_echoue .= implode(", ", $envois_echoues);
|
|
|
96 |
$corps_mail_echoue .= "<br /><br /><br />";
|
|
|
97 |
$corps_mail_echoue .= "--- <i> Message original ---</i><br />";
|
|
|
98 |
$corps_mail_echoue .= "Expéditeur : ".$mail_a_moderer['expediteur']."<br />";
|
|
|
99 |
$corps_mail_echoue .= "Sujet : ".$mail_a_moderer['sujet']."<br />";
|
490 |
aurelien |
100 |
$corps_mail_echoue .= "Message original : ".$mail_a_moderer['message']."<br />";
|
|
|
101 |
|
503 |
aurelien |
102 |
$sujet = "L'envoi d'un mail modéré a échoué pour un ou plusieurs destinataires";
|
|
|
103 |
|
|
|
104 |
// TODO: Que faire si l'envoi de mail d'avertissement échoue également ?
|
|
|
105 |
$envoi_avertissement = $this->envoyerMail(Config::get('adresse_mail_annuaire'),
|
|
|
106 |
Config::get('mail_moderateur'),
|
|
|
107 |
$sujet,
|
490 |
aurelien |
108 |
$corps_mail_echoue);
|
503 |
aurelien |
109 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
110 |
echo 'Envoi du mail au moderateur pour signaler un echec '."--";
|
|
|
111 |
return $envoi_avertissement;
|
|
|
112 |
}
|
|
|
113 |
|
|
|
114 |
private function avertirModerateurEchecTraitement($mails_en_echec) {
|
490 |
aurelien |
115 |
|
503 |
aurelien |
116 |
$ids_mails_en_echec = array();
|
|
|
117 |
foreach($mails_en_echec as $mail_echec) {
|
|
|
118 |
$ids_mails_en_echec[] = $mail_echec['adt_id'];
|
|
|
119 |
}
|
|
|
120 |
|
|
|
121 |
$corps_mail_mal_traite = "Échec de traitement pour : ".implode(',', $ids_mails_en_echec)." depuis plus de ".self::DELAI_MAX_TRAITEMENT." <br />";
|
|
|
122 |
$sujet = "Un ou plusieurs mails sont en échec de traitement";
|
|
|
123 |
|
|
|
124 |
$envoi_avertissement = $this->envoyerMail(Config::get('adresse_mail_annuaire'),
|
|
|
125 |
Config::get('mail_moderateur'),
|
|
|
126 |
$sujet,
|
|
|
127 |
$corps_mail_mal_traite);
|
|
|
128 |
|
|
|
129 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
130 |
echo 'Envoi du mail au moderateur pour signaler un traitement en echec depuis trop longtemps '."--";
|
490 |
aurelien |
131 |
return $envoi_avertissement;
|
|
|
132 |
}
|
|
|
133 |
|
486 |
aurelien |
134 |
private function obtenirMailsEnAttente() {
|
503 |
aurelien |
135 |
$requete = "SELECT * FROM annu_donnees_temp WHERE adt_statut = '".self::STATUT_A_TRAITER."' ";
|
486 |
aurelien |
136 |
$retour = $this->bdd->recupererTous($requete);
|
503 |
aurelien |
137 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
138 |
echo 'Il y a '.count($retour).' mails en attente '."--";
|
486 |
aurelien |
139 |
return $retour;
|
|
|
140 |
}
|
|
|
141 |
|
|
|
142 |
private function mettreMailsEnCoursDeTraitement() {
|
493 |
aurelien |
143 |
$requete = "UPDATE annu_donnees_temp SET adt_statut = '".self::STATUT_EN_TRAITEMENT."', adt_date_debut_traitement = NOW() ".
|
503 |
aurelien |
144 |
"WHERE adt_statut = '".self::STATUT_A_TRAITER."' ";
|
|
|
145 |
$maj = $this->bdd->executer($requete);
|
|
|
146 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
147 |
echo $maj.' mails ont été mis en traitement '."--";
|
486 |
aurelien |
148 |
return ($maj !== false);
|
|
|
149 |
}
|
|
|
150 |
|
491 |
aurelien |
151 |
private function mettreAJourMailMalTraite($id_mail_mal_traite, $mail_mal_traite, $envois_echoues) {
|
|
|
152 |
// TODO: utiliser cette fonction lors de l'echec de plusieurs destinataires et renvoyer le lien
|
|
|
153 |
// de confirmation
|
|
|
154 |
$mail_mal_traite['destinataires'] = $envois_echoues;
|
|
|
155 |
$mail_mal_traite = $this->encoderDonneeTemporaire($mail_mal_traite);
|
|
|
156 |
|
|
|
157 |
$requete = "UPDATE annu_donnees_temp ".
|
|
|
158 |
"SET adt_donnees = '".$mail_mal_traite."' ".
|
|
|
159 |
"WHERE adt_id = '".$mail_a_moderer['adt_id']."'";
|
|
|
160 |
|
503 |
aurelien |
161 |
$maj = $this->bdd->executer($requete);
|
491 |
aurelien |
162 |
return $maj;
|
|
|
163 |
}
|
|
|
164 |
|
486 |
aurelien |
165 |
private function supprimerMailTraite($id) {
|
493 |
aurelien |
166 |
$requete = "DELETE FROM annu_donnees_temp WHERE adt_statut = '".self::STATUT_EN_TRAITEMENT."' ".
|
503 |
aurelien |
167 |
"AND adt_id = '".$id."'";
|
|
|
168 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
169 |
echo'Suppression du mail '.$id.' qui a ete traite '."--";
|
|
|
170 |
$supp = $this->bdd->executer($requete);
|
486 |
aurelien |
171 |
return $supp;
|
|
|
172 |
}
|
|
|
173 |
|
503 |
aurelien |
174 |
private function supprimerMailsEnCoursDeTraitement() {
|
|
|
175 |
$requete = "DELETE FROM annu_donnees_temp WHERE adt_statut = '".self::STATUT_EN_TRAITEMENT."' ";
|
|
|
176 |
$supp = $this->bdd->executer($requete);
|
|
|
177 |
return $supp;
|
486 |
aurelien |
178 |
}
|
|
|
179 |
|
503 |
aurelien |
180 |
private function traiterMailsEnRetard() {
|
490 |
aurelien |
181 |
// Les mails a traiter depuis plus de 10 heures sont considérés comme échoués et donc remis à traiter
|
|
|
182 |
// (en cas de plantage du script ou du serveur de mail pendant leur traitement)
|
507 |
aurelien |
183 |
$requete = "UPDATE annu_donnees_temp SET adt_statut = '".self::STATUT_EN_ECHEC."' ".
|
493 |
aurelien |
184 |
"WHERE adt_statut = '".self::STATUT_EN_TRAITEMENT."' ".
|
|
|
185 |
"AND adt_date_debut_traitement < (DATE_SUB(now(), INTERVAL ".self::DELAI_MAX_TRAITEMENT.")) ";
|
486 |
aurelien |
186 |
|
503 |
aurelien |
187 |
$maj = $this->bdd->executer($requete);
|
|
|
188 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
189 |
echo 'Gestion des mails en retard '."--";
|
507 |
aurelien |
190 |
if($maj !== false && $maj != 0) {
|
|
|
191 |
$requete = "SELECT * FROM annu_donnees_temp WHERE adt_statut = '".self::STATUT_EN_ECHEC."' AND adt_date_debut_traitement IS NOT NULL";
|
503 |
aurelien |
192 |
$mails_en_echec = $this->bdd->recupererTous($requete);
|
507 |
aurelien |
193 |
|
503 |
aurelien |
194 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
195 |
echo 'Avertissement, des mails sont en retard : '.count($mails_en_echec)."--";
|
|
|
196 |
$this->avertirModerateurEchecTraitement($mails_en_echec);
|
507 |
aurelien |
197 |
|
|
|
198 |
// Réinitialisation de la date pour éviter que l'avertissement soit réenvoyé plusieurs fois
|
|
|
199 |
$requete = "UPDATE annu_donnees_temp SET adt_date_debut_traitement = NULL ".
|
|
|
200 |
"WHERE adt_statut = '".self::STATUT_EN_ECHEC."' ";
|
|
|
201 |
$maj = $this->bdd->executer($requete);
|
503 |
aurelien |
202 |
}
|
|
|
203 |
|
|
|
204 |
return $maj;
|
486 |
aurelien |
205 |
}
|
|
|
206 |
|
503 |
aurelien |
207 |
private function encoderDonneeTemporaire($donnee) {
|
|
|
208 |
return base64_encode(serialize($donnee));
|
491 |
aurelien |
209 |
}
|
|
|
210 |
|
503 |
aurelien |
211 |
private function decoderDonneeTemporaire($donnee_encodee) {
|
|
|
212 |
return unserialize(base64_decode($donnee_encodee['adt_donnees']));
|
486 |
aurelien |
213 |
}
|
|
|
214 |
|
503 |
aurelien |
215 |
/** Envoie un mail avec l'adresse de l'utilisateur donné en paramètre, à l'adresse donnée en paramètre.
|
|
|
216 |
* ATTENTION : le sujet et le contenu envoyer à cette méthode doivent avoir le même encodage que l'application.
|
|
|
217 |
*
|
|
|
218 |
* @param string $expediteur l'expediteur du message
|
|
|
219 |
* @param mixed $destinataires un string ou un tableau de mails qui contiennent les destinataire
|
|
|
220 |
* @param string $sujet sujet du mail
|
|
|
221 |
* @return boolean true ou false suivant le succès ou non de l'envoi
|
|
|
222 |
*/
|
|
|
223 |
public function envoyerMail($expediteur, $destinataires, $sujet, $message_html, $message_texte = '', $adresse_reponse = null) {
|
|
|
224 |
if (!is_array($destinataires)) {
|
|
|
225 |
$destinataires = array($destinataires);
|
|
|
226 |
}
|
|
|
227 |
if ($message_texte == '') {
|
|
|
228 |
$message_texte = $this->filtrerChaine($message_html);
|
|
|
229 |
}
|
|
|
230 |
|
|
|
231 |
$encodage = Config::get('appli_encodage');
|
|
|
232 |
$limite = "_----------=_parties_".md5(uniqid(rand()));
|
|
|
233 |
$eol = "\n";
|
|
|
234 |
|
|
|
235 |
$entetes = '';
|
|
|
236 |
// Définition d'un mail en texte simple et html
|
|
|
237 |
// multipart/alternative signifie même contenu de la forme la plus simple à la plus complexe
|
|
|
238 |
$entetes .= "X-Sender: <http://www.tela-botanica.org>".$eol.
|
|
|
239 |
"X-Mailer: PHP-ANNUAIRE-HTML".$eol.
|
|
|
240 |
"X-auth-smtp-user: annuaire@tela-botanica.org ".$eol.
|
|
|
241 |
"X-abuse-contact: annuaire@tela-botanica.org ".$eol.
|
|
|
242 |
'Date: '.date('r').$eol.
|
|
|
243 |
'From: '.$expediteur.$eol.
|
|
|
244 |
'MIME-Version: 1.0'.$eol;
|
|
|
245 |
if ($adresse_reponse !== null) {
|
|
|
246 |
$entetes .= 'Reply-To: '.$adresse_reponse.$eol;
|
|
|
247 |
}
|
|
|
248 |
$entetes .= "Content-Type: multipart/alternative; boundary=\"$limite\";".$eol.$eol;
|
|
|
249 |
|
|
|
250 |
// message en texte simple
|
|
|
251 |
$contenu = "--$limite".$eol.
|
|
|
252 |
"Content-Type: text/plain; charset=\"$encodage\";".$eol.
|
|
|
253 |
"Content-Transfer-Encoding: 8bit;".$eol.$eol.
|
|
|
254 |
$message_texte.$eol.$eol.
|
|
|
255 |
// le message en html est préféré s'il est lisible
|
|
|
256 |
"--$limite".$eol.
|
|
|
257 |
"Content-Type: text/html; charset=\"$encodage\";".$eol.
|
|
|
258 |
"Content-Transfer-Encoding: 8bit;".$eol.$eol.
|
|
|
259 |
$message_html.$eol.$eol.
|
|
|
260 |
"--$limite--".$eol.$eol;
|
|
|
261 |
|
490 |
aurelien |
262 |
$sujetEncode = mb_encode_mimeheader($sujet, mb_internal_encoding(), "B", "\n");
|
503 |
aurelien |
263 |
$resultats_envois_echoues = array();
|
|
|
264 |
$ok = true;
|
|
|
265 |
foreach ($destinataires as $destinataire) {
|
|
|
266 |
$ok = mail($destinataire, $sujetEncode, $contenu, $entetes);
|
|
|
267 |
if (!$ok) {
|
|
|
268 |
// echo pour que bash capte la sortie et stocke dans le log
|
|
|
269 |
echo'Echec envoi a '.$destinataire."\n";
|
|
|
270 |
$resultats_envois_echoues[] = $destinataire;
|
|
|
271 |
}
|
|
|
272 |
}
|
|
|
273 |
return $resultats_envois_echoues;
|
486 |
aurelien |
274 |
}
|
490 |
aurelien |
275 |
|
503 |
aurelien |
276 |
/** Transforme automatiquement le message html en message txt.
|
|
|
277 |
*
|
|
|
278 |
* Réalise un strip_tags et avant ça un remplacement des liens sur mesure pour les mettre au format email txt.
|
|
|
279 |
*/
|
|
|
280 |
private function filtrerChaine($messageHtml) {
|
|
|
281 |
$messageTxt = strip_tags($messageHtml);
|
|
|
282 |
if ($messageHtml != $messageTxt) {
|
|
|
283 |
$html = $this->ajouterHrefDansBalise($messageHtml);
|
|
|
284 |
$messageAvecEntites = strip_tags($html);
|
|
|
285 |
// TODO : en précisant l'encodage de l'appli dans html_entity_decode un double encodage UTF-8 se produit...
|
|
|
286 |
$messageTxt = html_entity_decode($messageAvecEntites, ENT_QUOTES);
|
|
|
287 |
}
|
|
|
288 |
return $messageTxt;
|
490 |
aurelien |
289 |
}
|
|
|
290 |
|
503 |
aurelien |
291 |
/**
|
|
|
292 |
* Extrait la valeur de l'attribut href des balises HTML de liens (a) et ajoute le lien entre
|
|
|
293 |
* chevrons (<>) dans le contenu de la balise "a".
|
|
|
294 |
*/
|
|
|
295 |
private function ajouterHrefDansBalise($html) {
|
|
|
296 |
$dom = new DOMDocument;
|
|
|
297 |
$dom->loadHTML($html);
|
|
|
298 |
foreach ($dom->getElementsByTagName('a') as $node) {
|
|
|
299 |
if ($node->hasAttribute( 'href' )) {
|
|
|
300 |
$href = $node->getAttribute('href');
|
|
|
301 |
$node->nodeValue = $node->nodeValue." < $href >";
|
|
|
302 |
}
|
|
|
303 |
}
|
|
|
304 |
$html = $dom->saveHtml();
|
|
|
305 |
return $html;
|
490 |
aurelien |
306 |
}
|
486 |
aurelien |
307 |
}
|
|
|
308 |
?>
|